1 //===-- FuzzerCLI.cpp -----------------------------------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "llvm/FuzzMutate/FuzzerCLI.h"
11 #include "llvm/ADT/Triple.h"
12 #include "llvm/Support/CommandLine.h"
13 #include "llvm/Support/Compiler.h"
14 #include "llvm/Support/Error.h"
15 #include "llvm/Support/MemoryBuffer.h"
16 #include "llvm/Support/raw_ostream.h"
17 
18 using namespace llvm;
19 
20 void llvm::parseFuzzerCLOpts(int ArgC, char *ArgV[]) {
21   std::vector<const char *> CLArgs;
22   CLArgs.push_back(ArgV[0]);
23 
24   int I = 1;
25   while (I < ArgC)
26     if (StringRef(ArgV[I++]).equals("-ignore_remaining_args=1"))
27       break;
28   while (I < ArgC)
29     CLArgs.push_back(ArgV[I++]);
30 
31   cl::ParseCommandLineOptions(CLArgs.size(), CLArgs.data());
32 }
33 
34 void llvm::handleExecNameEncodedBEOpts(StringRef ExecName) {
35   std::vector<std::string> Args{ExecName};
36 
37   auto NameAndArgs = ExecName.split("--");
38   if (NameAndArgs.second.empty())
39     return;
40 
41   SmallVector<StringRef, 4> Opts;
42   NameAndArgs.second.split(Opts, '-');
43   for (StringRef Opt : Opts) {
44     if (Opt.equals("gisel")) {
45       Args.push_back("-global-isel");
46       // For now we default GlobalISel to -O0
47       Args.push_back("-O0");
48     } else if (Opt.startswith("O")) {
49       Args.push_back("-" + Opt.str());
50     } else if (Triple::getArchTypeForLLVMName(Opt)) {
51       Args.push_back("-mtriple=" + Opt.str());
52     } else {
53       errs() << ExecName << ": Unknown option: " << Opt << ".\n";
54       exit(1);
55     }
56   }
57   errs() << NameAndArgs.first << ": Injected args:";
58   for (int I = 1, E = Args.size(); I < E; ++I)
59     errs() << " " << Args[I];
60   errs() << "\n";
61 
62   std::vector<const char *> CLArgs;
63   CLArgs.reserve(Args.size());
64   for (std::string &S : Args)
65     CLArgs.push_back(S.c_str());
66 
67   cl::ParseCommandLineOptions(CLArgs.size(), CLArgs.data());
68 }
69 
70 int llvm::runFuzzerOnInputs(int ArgC, char *ArgV[], FuzzerTestFun TestOne,
71                             FuzzerInitFun Init) {
72   errs() << "*** This tool was not linked to libFuzzer.\n"
73          << "*** No fuzzing will be performed.\n";
74   if (int RC = Init(&ArgC, &ArgV)) {
75     errs() << "Initialization failed\n";
76     return RC;
77   }
78 
79   for (int I = 1; I < ArgC; ++I) {
80     StringRef Arg(ArgV[I]);
81     if (Arg.startswith("-")) {
82       if (Arg.equals("-ignore_remaining_args=1"))
83         break;
84       continue;
85     }
86 
87     auto BufOrErr = MemoryBuffer::getFile(Arg, /*FileSize-*/ -1,
88                                           /*RequiresNullTerminator=*/false);
89     if (std::error_code EC = BufOrErr.getError()) {
90       errs() << "Error reading file: " << Arg << ": " << EC.message() << "\n";
91       return 1;
92     }
93     std::unique_ptr<MemoryBuffer> Buf = std::move(BufOrErr.get());
94     errs() << "Running: " << Arg << " (" << Buf->getBufferSize() << " bytes)\n";
95     TestOne(reinterpret_cast<const uint8_t *>(Buf->getBufferStart()),
96             Buf->getBufferSize());
97   }
98   return 0;
99 }
100