1 | //===-- tools/bugpoint/ToolRunner.h -----------------------------*- C++ -*-===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | // |
9 | // This file exposes an abstraction around a platform C compiler, used to |
10 | // compile C and assembly code. It also exposes an "AbstractIntepreter" |
11 | // interface, which is used to execute code using one of the LLVM execution |
12 | // engines. |
13 | // |
14 | //===----------------------------------------------------------------------===// |
15 | |
16 | #ifndef LLVM_TOOLS_BUGPOINT_TOOLRUNNER_H |
17 | #define LLVM_TOOLS_BUGPOINT_TOOLRUNNER_H |
18 | |
19 | #include "llvm/Support/CommandLine.h" |
20 | #include "llvm/Support/Error.h" |
21 | #include "llvm/Support/Path.h" |
22 | #include "llvm/Support/SystemUtils.h" |
23 | #include "llvm/TargetParser/Triple.h" |
24 | #include <exception> |
25 | #include <vector> |
26 | |
27 | namespace llvm { |
28 | |
29 | extern cl::opt<bool> SaveTemps; |
30 | extern Triple TargetTriple; |
31 | |
32 | class LLC; |
33 | |
34 | //===---------------------------------------------------------------------===// |
35 | // CC abstraction |
36 | // |
37 | class CC { |
38 | std::string CCPath; // The path to the cc executable. |
39 | std::string RemoteClientPath; // The path to the rsh / ssh executable. |
40 | std::vector<std::string> ccArgs; // CC-specific arguments. |
41 | CC(StringRef ccPath, StringRef RemotePath, |
42 | const std::vector<std::string> *CCArgs) |
43 | : CCPath(std::string(ccPath)), RemoteClientPath(std::string(RemotePath)) { |
44 | if (CCArgs) |
45 | ccArgs = *CCArgs; |
46 | } |
47 | |
48 | public: |
49 | enum FileType { AsmFile, ObjectFile, CFile }; |
50 | |
51 | static CC *create(const char *Argv0, std::string &Message, |
52 | const std::string &CCBinary, |
53 | const std::vector<std::string> *Args); |
54 | |
55 | /// ExecuteProgram - Execute the program specified by "ProgramFile" (which is |
56 | /// either a .s file, or a .c file, specified by FileType), with the specified |
57 | /// arguments. Standard input is specified with InputFile, and standard |
58 | /// Output is captured to the specified OutputFile location. The SharedLibs |
59 | /// option specifies optional native shared objects that can be loaded into |
60 | /// the program for execution. |
61 | /// |
62 | Expected<int> ExecuteProgram( |
63 | const std::string &ProgramFile, const std::vector<std::string> &Args, |
64 | FileType fileType, const std::string &InputFile, |
65 | const std::string &OutputFile, |
66 | const std::vector<std::string> &CCArgs = std::vector<std::string>(), |
67 | unsigned Timeout = 0, unsigned MemoryLimit = 0); |
68 | |
69 | /// MakeSharedObject - This compiles the specified file (which is either a .c |
70 | /// file or a .s file) into a shared object. |
71 | /// |
72 | Error MakeSharedObject(const std::string &InputFile, FileType fileType, |
73 | std::string &OutputFile, |
74 | const std::vector<std::string> &ArgsForCC); |
75 | }; |
76 | |
77 | //===---------------------------------------------------------------------===// |
78 | /// AbstractInterpreter Class - Subclasses of this class are used to execute |
79 | /// LLVM bitcode in a variety of ways. This abstract interface hides this |
80 | /// complexity behind a simple interface. |
81 | /// |
82 | class AbstractInterpreter { |
83 | virtual void anchor(); |
84 | |
85 | public: |
86 | static LLC *createLLC(const char *Argv0, std::string &Message, |
87 | const std::string &CCBinary, |
88 | const std::vector<std::string> *Args = nullptr, |
89 | const std::vector<std::string> *CCArgs = nullptr, |
90 | bool UseIntegratedAssembler = false); |
91 | |
92 | static AbstractInterpreter * |
93 | createLLI(const char *Argv0, std::string &Message, |
94 | const std::vector<std::string> *Args = nullptr); |
95 | |
96 | static AbstractInterpreter * |
97 | createJIT(const char *Argv0, std::string &Message, |
98 | const std::vector<std::string> *Args = nullptr); |
99 | |
100 | static AbstractInterpreter * |
101 | createCustomCompiler(const char *Argv0, std::string &Message, |
102 | const std::string &CompileCommandLine); |
103 | |
104 | static AbstractInterpreter * |
105 | createCustomExecutor(const char *Argv0, std::string &Message, |
106 | const std::string &ExecCommandLine); |
107 | |
108 | virtual ~AbstractInterpreter() {} |
109 | |
110 | /// compileProgram - Compile the specified program from bitcode to executable |
111 | /// code. This does not produce any output, it is only used when debugging |
112 | /// the code generator. It returns false if the code generator fails. |
113 | virtual Error compileProgram(const std::string &Bitcode, unsigned Timeout = 0, |
114 | unsigned MemoryLimit = 0) { |
115 | return Error::success(); |
116 | } |
117 | |
118 | /// Compile the specified program from bitcode to code understood by the CC |
119 | /// driver (either C or asm). Returns an error if the code generator fails,, |
120 | /// otherwise, the type of code emitted. |
121 | virtual Expected<CC::FileType> OutputCode(const std::string &Bitcode, |
122 | std::string &OutFile, |
123 | unsigned Timeout = 0, |
124 | unsigned MemoryLimit = 0) { |
125 | return make_error<StringError>( |
126 | Args: "OutputCode not supported by this AbstractInterpreter!" , |
127 | Args: inconvertibleErrorCode()); |
128 | } |
129 | |
130 | /// ExecuteProgram - Run the specified bitcode file, emitting output to the |
131 | /// specified filename. This sets RetVal to the exit code of the program or |
132 | /// returns an Error if a problem was encountered that prevented execution of |
133 | /// the program. |
134 | /// |
135 | virtual Expected<int> ExecuteProgram( |
136 | const std::string &Bitcode, const std::vector<std::string> &Args, |
137 | const std::string &InputFile, const std::string &OutputFile, |
138 | const std::vector<std::string> &CCArgs = std::vector<std::string>(), |
139 | const std::vector<std::string> &SharedLibs = std::vector<std::string>(), |
140 | unsigned Timeout = 0, unsigned MemoryLimit = 0) = 0; |
141 | }; |
142 | |
143 | //===---------------------------------------------------------------------===// |
144 | // LLC Implementation of AbstractIntepreter interface |
145 | // |
146 | class LLC : public AbstractInterpreter { |
147 | std::string LLCPath; // The path to the LLC executable. |
148 | std::vector<std::string> ToolArgs; // Extra args to pass to LLC. |
149 | CC *cc; |
150 | bool UseIntegratedAssembler; |
151 | |
152 | public: |
153 | LLC(const std::string &llcPath, CC *cc, const std::vector<std::string> *Args, |
154 | bool useIntegratedAssembler) |
155 | : LLCPath(llcPath), cc(cc), |
156 | UseIntegratedAssembler(useIntegratedAssembler) { |
157 | ToolArgs.clear(); |
158 | if (Args) |
159 | ToolArgs = *Args; |
160 | } |
161 | ~LLC() override { delete cc; } |
162 | |
163 | /// compileProgram - Compile the specified program from bitcode to executable |
164 | /// code. This does not produce any output, it is only used when debugging |
165 | /// the code generator. Returns false if the code generator fails. |
166 | Error compileProgram(const std::string &Bitcode, unsigned Timeout = 0, |
167 | unsigned MemoryLimit = 0) override; |
168 | |
169 | Expected<int> ExecuteProgram( |
170 | const std::string &Bitcode, const std::vector<std::string> &Args, |
171 | const std::string &InputFile, const std::string &OutputFile, |
172 | const std::vector<std::string> &CCArgs = std::vector<std::string>(), |
173 | const std::vector<std::string> &SharedLibs = std::vector<std::string>(), |
174 | unsigned Timeout = 0, unsigned MemoryLimit = 0) override; |
175 | |
176 | Expected<CC::FileType> OutputCode(const std::string &Bitcode, |
177 | std::string &OutFile, unsigned Timeout = 0, |
178 | unsigned MemoryLimit = 0) override; |
179 | }; |
180 | |
181 | /// Find the first executable file \ExeName, either in the user's PATH or, |
182 | /// failing that, in the same directory as argv[0]. This allows us to find |
183 | /// another LLVM tool if it is built in the same directory. If no executable is |
184 | /// found, an error is returned. |
185 | ErrorOr<std::string> FindProgramByName(const std::string &ExeName, |
186 | const char *Argv0, void *MainAddr); |
187 | |
188 | } // End llvm namespace |
189 | |
190 | #endif |
191 | |