blob: 1ca6a548f66fd87f6d361f86459dd55caf2aa678 [file] [log] [blame]
Justin Bogner7f28d732017-09-02 23:43:04 +00001//===-- 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"
Justin Bogner9ea7fbd2017-10-12 04:35:32 +000011#include "llvm/ADT/Triple.h"
Justin Bogner7f28d732017-09-02 23:43:04 +000012#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
18using namespace llvm;
19
20void 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
Justin Bogner9ea7fbd2017-10-12 04:35:32 +000034void 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
Justin Bogner7f28d732017-09-02 23:43:04 +000070int 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}