blob: a70dad37dfcff7e20c260ee1ef3d39c0f875e59b [file] [log] [blame]
Justin Bogner16edf9b2017-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 Bogner58658af2017-10-12 04:35:32 +000011#include "llvm/ADT/Triple.h"
Igor Laevsky1404e4c2017-11-16 15:23:08 +000012#include "llvm/Bitcode/BitcodeReader.h"
13#include "llvm/Bitcode/BitcodeWriter.h"
14#include "llvm/IR/LLVMContext.h"
Justin Bogner16edf9b2017-09-02 23:43:04 +000015#include "llvm/Support/CommandLine.h"
16#include "llvm/Support/Compiler.h"
17#include "llvm/Support/Error.h"
18#include "llvm/Support/MemoryBuffer.h"
Igor Laevsky1404e4c2017-11-16 15:23:08 +000019#include "llvm/Support/SourceMgr.h"
Justin Bogner16edf9b2017-09-02 23:43:04 +000020#include "llvm/Support/raw_ostream.h"
Igor Laevskya3288792018-02-05 11:05:47 +000021#include "llvm/IR/Verifier.h"
Justin Bogner16edf9b2017-09-02 23:43:04 +000022
23using namespace llvm;
24
25void llvm::parseFuzzerCLOpts(int ArgC, char *ArgV[]) {
26 std::vector<const char *> CLArgs;
27 CLArgs.push_back(ArgV[0]);
28
29 int I = 1;
30 while (I < ArgC)
31 if (StringRef(ArgV[I++]).equals("-ignore_remaining_args=1"))
32 break;
33 while (I < ArgC)
34 CLArgs.push_back(ArgV[I++]);
35
36 cl::ParseCommandLineOptions(CLArgs.size(), CLArgs.data());
37}
38
Justin Bogner58658af2017-10-12 04:35:32 +000039void llvm::handleExecNameEncodedBEOpts(StringRef ExecName) {
40 std::vector<std::string> Args{ExecName};
41
Matt Morehouse5cb6ac02017-10-13 00:18:32 +000042 auto NameAndArgs = ExecName.split("--");
Justin Bogner58658af2017-10-12 04:35:32 +000043 if (NameAndArgs.second.empty())
44 return;
45
46 SmallVector<StringRef, 4> Opts;
47 NameAndArgs.second.split(Opts, '-');
48 for (StringRef Opt : Opts) {
49 if (Opt.equals("gisel")) {
50 Args.push_back("-global-isel");
51 // For now we default GlobalISel to -O0
52 Args.push_back("-O0");
53 } else if (Opt.startswith("O")) {
54 Args.push_back("-" + Opt.str());
Justin Bogner6ec40b12017-10-17 02:39:40 +000055 } else if (Triple(Opt).getArch()) {
Justin Bogner58658af2017-10-12 04:35:32 +000056 Args.push_back("-mtriple=" + Opt.str());
57 } else {
58 errs() << ExecName << ": Unknown option: " << Opt << ".\n";
59 exit(1);
60 }
61 }
62 errs() << NameAndArgs.first << ": Injected args:";
63 for (int I = 1, E = Args.size(); I < E; ++I)
64 errs() << " " << Args[I];
65 errs() << "\n";
66
67 std::vector<const char *> CLArgs;
68 CLArgs.reserve(Args.size());
69 for (std::string &S : Args)
70 CLArgs.push_back(S.c_str());
71
72 cl::ParseCommandLineOptions(CLArgs.size(), CLArgs.data());
73}
74
Igor Laevsky089f8862017-11-10 12:19:08 +000075void llvm::handleExecNameEncodedOptimizerOpts(StringRef ExecName) {
76 // TODO: Refactor parts common with the 'handleExecNameEncodedBEOpts'
77 std::vector<std::string> Args{ExecName};
78
79 auto NameAndArgs = ExecName.split("--");
80 if (NameAndArgs.second.empty())
81 return;
82
83 SmallVector<StringRef, 4> Opts;
84 NameAndArgs.second.split(Opts, '-');
85 for (StringRef Opt : Opts) {
Igor Laevskyde603ec2018-02-19 11:57:07 +000086 if (Opt == "instcombine") {
Igor Laevsky089f8862017-11-10 12:19:08 +000087 Args.push_back("-passes=instcombine");
Igor Laevskyde603ec2018-02-19 11:57:07 +000088 } else if (Opt == "earlycse") {
Igor Laevsky20b5be12018-01-24 09:57:17 +000089 Args.push_back("-passes=early-cse");
Igor Laevskyde603ec2018-02-19 11:57:07 +000090 } else if (Opt == "simplifycfg") {
Igor Laevsky20b5be12018-01-24 09:57:17 +000091 Args.push_back("-passes=simplify-cfg");
Igor Laevskyde603ec2018-02-19 11:57:07 +000092 } else if (Opt == "gvn") {
Igor Laevsky20b5be12018-01-24 09:57:17 +000093 Args.push_back("-passes=gvn");
Igor Laevskyde603ec2018-02-19 11:57:07 +000094 } else if (Opt == "sccp") {
Igor Laevsky20b5be12018-01-24 09:57:17 +000095 Args.push_back("-passes=sccp");
Fangrui Songaf7b1832018-07-30 19:41:25 +000096
Igor Laevskyde603ec2018-02-19 11:57:07 +000097 } else if (Opt == "loop_predication") {
98 Args.push_back("-passes=loop-predication");
99 } else if (Opt == "guard_widening") {
100 Args.push_back("-passes=guard-widening");
101 } else if (Opt == "loop_rotate") {
102 Args.push_back("-passes=loop(rotate)");
103 } else if (Opt == "loop_unswitch") {
104 Args.push_back("-passes=loop(unswitch)");
105 } else if (Opt == "loop_unroll") {
106 Args.push_back("-passes=unroll");
107 } else if (Opt == "loop_vectorize") {
108 Args.push_back("-passes=loop-vectorize");
109 } else if (Opt == "licm") {
110 Args.push_back("-passes=licm");
111 } else if (Opt == "indvars") {
112 Args.push_back("-passes=indvars");
113 } else if (Opt == "strength_reduce") {
114 Args.push_back("-passes=strength-reduce");
Igor Laevsky7724d6b2018-03-20 11:32:13 +0000115 } else if (Opt == "irce") {
116 Args.push_back("-passes=irce");
Fangrui Songaf7b1832018-07-30 19:41:25 +0000117
Igor Laevsky089f8862017-11-10 12:19:08 +0000118 } else if (Triple(Opt).getArch()) {
119 Args.push_back("-mtriple=" + Opt.str());
120 } else {
121 errs() << ExecName << ": Unknown option: " << Opt << ".\n";
122 exit(1);
123 }
124 }
125
126 errs() << NameAndArgs.first << ": Injected args:";
127 for (int I = 1, E = Args.size(); I < E; ++I)
128 errs() << " " << Args[I];
129 errs() << "\n";
130
131 std::vector<const char *> CLArgs;
132 CLArgs.reserve(Args.size());
133 for (std::string &S : Args)
134 CLArgs.push_back(S.c_str());
135
136 cl::ParseCommandLineOptions(CLArgs.size(), CLArgs.data());
137}
138
Justin Bogner16edf9b2017-09-02 23:43:04 +0000139int llvm::runFuzzerOnInputs(int ArgC, char *ArgV[], FuzzerTestFun TestOne,
140 FuzzerInitFun Init) {
141 errs() << "*** This tool was not linked to libFuzzer.\n"
142 << "*** No fuzzing will be performed.\n";
143 if (int RC = Init(&ArgC, &ArgV)) {
144 errs() << "Initialization failed\n";
145 return RC;
146 }
147
148 for (int I = 1; I < ArgC; ++I) {
149 StringRef Arg(ArgV[I]);
150 if (Arg.startswith("-")) {
151 if (Arg.equals("-ignore_remaining_args=1"))
152 break;
153 continue;
154 }
155
156 auto BufOrErr = MemoryBuffer::getFile(Arg, /*FileSize-*/ -1,
157 /*RequiresNullTerminator=*/false);
158 if (std::error_code EC = BufOrErr.getError()) {
159 errs() << "Error reading file: " << Arg << ": " << EC.message() << "\n";
160 return 1;
161 }
162 std::unique_ptr<MemoryBuffer> Buf = std::move(BufOrErr.get());
163 errs() << "Running: " << Arg << " (" << Buf->getBufferSize() << " bytes)\n";
164 TestOne(reinterpret_cast<const uint8_t *>(Buf->getBufferStart()),
165 Buf->getBufferSize());
166 }
167 return 0;
168}
Igor Laevsky1404e4c2017-11-16 15:23:08 +0000169
170std::unique_ptr<Module> llvm::parseModule(
171 const uint8_t *Data, size_t Size, LLVMContext &Context) {
172
173 if (Size <= 1)
174 // We get bogus data given an empty corpus - just create a new module.
175 return llvm::make_unique<Module>("M", Context);
176
177 auto Buffer = MemoryBuffer::getMemBuffer(
178 StringRef(reinterpret_cast<const char *>(Data), Size), "Fuzzer input",
179 /*RequiresNullTerminator=*/false);
180
181 SMDiagnostic Err;
182 auto M = parseBitcodeFile(Buffer->getMemBufferRef(), Context);
183 if (Error E = M.takeError()) {
184 errs() << toString(std::move(E)) << "\n";
185 return nullptr;
186 }
187 return std::move(M.get());
188}
189
190size_t llvm::writeModule(const Module &M, uint8_t *Dest, size_t MaxSize) {
191 std::string Buf;
192 {
193 raw_string_ostream OS(Buf);
Rafael Espindola06d62072018-02-14 19:11:32 +0000194 WriteBitcodeToFile(M, OS);
Igor Laevsky1404e4c2017-11-16 15:23:08 +0000195 }
196 if (Buf.size() > MaxSize)
197 return 0;
198 memcpy(Dest, Buf.data(), Buf.size());
199 return Buf.size();
200}
Igor Laevskya3288792018-02-05 11:05:47 +0000201
202std::unique_ptr<Module> llvm::parseAndVerify(const uint8_t *Data, size_t Size,
203 LLVMContext &Context) {
204 auto M = parseModule(Data, Size, Context);
205 if (!M || verifyModule(*M, &errs()))
206 return nullptr;
Fangrui Songaf7b1832018-07-30 19:41:25 +0000207
Igor Laevskya3288792018-02-05 11:05:47 +0000208 return M;
209}