| Chris Lattner | b79757c | 2001-10-04 01:40:53 +0000 | [diff] [blame] | 1 | //===-- llc.cpp - Implement the LLVM Compiler -----------------------------===// |
| Chris Lattner | e737c7a | 2001-09-07 22:20:50 +0000 | [diff] [blame] | 2 | // |
| 3 | // This is the llc compiler driver. |
| 4 | // |
| Chris Lattner | b79757c | 2001-10-04 01:40:53 +0000 | [diff] [blame] | 5 | //===----------------------------------------------------------------------===// |
| Vikram S. Adve | cb465fc | 2001-07-21 12:42:29 +0000 | [diff] [blame] | 6 | |
| Vikram S. Adve | cb465fc | 2001-07-21 12:42:29 +0000 | [diff] [blame] | 7 | #include "llvm/Bytecode/Reader.h" |
| Chris Lattner | b26bcc5 | 2001-09-14 05:34:53 +0000 | [diff] [blame] | 8 | #include "llvm/Target/Sparc.h" |
| Vikram S. Adve | 805eb96 | 2001-09-18 13:10:45 +0000 | [diff] [blame] | 9 | #include "llvm/Target/TargetMachine.h" |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 10 | #include "llvm/Transforms/Instrumentation/TraceValues.h" |
| Chris Lattner | d7db863 | 2002-01-22 01:04:08 +0000 | [diff] [blame] | 11 | #include "llvm/Transforms/ChangeAllocations.h" |
| Chris Lattner | 65f1b89 | 2002-05-07 20:03:27 +0000 | [diff] [blame] | 12 | #include "llvm/Transforms/Scalar.h" |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 13 | #include "llvm/Transforms/Utils/Linker.h" |
| Chris Lattner | 2197d21 | 2001-10-18 20:32:07 +0000 | [diff] [blame] | 14 | #include "llvm/Assembly/PrintModulePass.h" |
| 15 | #include "llvm/Bytecode/WriteBytecodePass.h" |
| Chris Lattner | c2c9dd1 | 2001-10-18 20:06:31 +0000 | [diff] [blame] | 16 | #include "llvm/Transforms/ConstantMerge.h" |
| Chris Lattner | 46ac43c | 2001-09-07 21:26:31 +0000 | [diff] [blame] | 17 | #include "llvm/Module.h" |
| Chris Lattner | 2fbfdcf | 2002-04-07 20:49:59 +0000 | [diff] [blame] | 18 | #include "llvm/Function.h" |
| Chris Lattner | cd50d3f | 2002-01-31 00:46:45 +0000 | [diff] [blame] | 19 | #include "llvm/PassManager.h" |
| Chris Lattner | cee8f9a | 2001-11-27 00:03:19 +0000 | [diff] [blame] | 20 | #include "Support/CommandLine.h" |
| Chris Lattner | 76d1229 | 2002-04-18 19:55:25 +0000 | [diff] [blame] | 21 | #include "Support/Signals.h" |
| Chris Lattner | da784ee | 2001-09-18 17:04:18 +0000 | [diff] [blame] | 22 | #include <memory> |
| Chris Lattner | 78f7e1a | 2001-09-19 16:52:09 +0000 | [diff] [blame] | 23 | #include <fstream> |
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 24 | using std::string; |
| Anand Shukla | cf17bcc | 2002-06-25 21:57:48 +0000 | [diff] [blame^] | 25 | using std::cerr; |
| Chris Lattner | 8f367bd | 2001-07-23 02:35:57 +0000 | [diff] [blame] | 26 | |
| Chris Lattner | 4ddcd54 | 2002-02-26 21:36:53 +0000 | [diff] [blame] | 27 | static cl::String InputFilename ("", "Input filename", cl::NoFlags, "-"); |
| 28 | static cl::String OutputFilename("o", "Output filename", cl::NoFlags, ""); |
| 29 | static cl::Flag Force ("f", "Overwrite output files"); |
| 30 | static cl::Flag DumpAsm ("d", "Print bytecode before native code generation", cl::Hidden); |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 31 | static cl::String TraceLibPath ("tracelibpath", "Path to libinstr for trace code", cl::Hidden, ""); |
| Chris Lattner | 4ddcd54 | 2002-02-26 21:36:53 +0000 | [diff] [blame] | 32 | |
| 33 | enum TraceLevel { |
| Chris Lattner | 2fbfdcf | 2002-04-07 20:49:59 +0000 | [diff] [blame] | 34 | TraceOff, TraceFunctions, TraceBasicBlocks |
| Chris Lattner | 4ddcd54 | 2002-02-26 21:36:53 +0000 | [diff] [blame] | 35 | }; |
| 36 | |
| 37 | static cl::Enum<enum TraceLevel> TraceValues("trace", cl::NoFlags, |
| Chris Lattner | 2fbfdcf | 2002-04-07 20:49:59 +0000 | [diff] [blame] | 38 | "Trace values through functions or basic blocks", |
| Chris Lattner | 4ddcd54 | 2002-02-26 21:36:53 +0000 | [diff] [blame] | 39 | clEnumValN(TraceOff , "off", "Disable trace code"), |
| Chris Lattner | 2fbfdcf | 2002-04-07 20:49:59 +0000 | [diff] [blame] | 40 | clEnumValN(TraceFunctions , "function", "Trace each function"), |
| Chris Lattner | 4ddcd54 | 2002-02-26 21:36:53 +0000 | [diff] [blame] | 41 | clEnumValN(TraceBasicBlocks, "basicblock", "Trace each basic block"), 0); |
| Chris Lattner | 8f367bd | 2001-07-23 02:35:57 +0000 | [diff] [blame] | 42 | |
| Vikram S. Adve | 805eb96 | 2001-09-18 13:10:45 +0000 | [diff] [blame] | 43 | |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 44 | // GetFileNameRoot - Helper function to get the basename of a filename... |
| 45 | static inline string GetFileNameRoot(const string &InputFilename) { |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 46 | string IFN = InputFilename; |
| 47 | string outputFilename; |
| 48 | int Len = IFN.length(); |
| 49 | if (IFN[Len-3] == '.' && IFN[Len-2] == 'b' && IFN[Len-1] == 'c') { |
| 50 | outputFilename = string(IFN.begin(), IFN.end()-3); // s/.bc/.s/ |
| 51 | } else { |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 52 | outputFilename = IFN; |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 53 | } |
| 54 | return outputFilename; |
| 55 | } |
| 56 | |
| Chris Lattner | 7e70829 | 2002-06-25 16:13:24 +0000 | [diff] [blame] | 57 | static void insertTraceCodeFor(Module &M) { |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 58 | PassManager Passes; |
| 59 | |
| 60 | // Insert trace code in all functions in the module |
| 61 | switch (TraceValues) { |
| 62 | case TraceBasicBlocks: |
| 63 | Passes.add(createTraceValuesPassForBasicBlocks()); |
| 64 | break; |
| 65 | case TraceFunctions: |
| 66 | Passes.add(createTraceValuesPassForFunction()); |
| 67 | break; |
| 68 | default: |
| 69 | assert(0 && "Bad value for TraceValues!"); |
| 70 | abort(); |
| 71 | } |
| 72 | |
| 73 | // Eliminate duplication in constant pool |
| Chris Lattner | 7e70829 | 2002-06-25 16:13:24 +0000 | [diff] [blame] | 74 | Passes.add(createConstantMergePass()); |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 75 | |
| 76 | // Run passes to insert and clean up trace code... |
| 77 | Passes.run(M); |
| 78 | |
| 79 | std::string ErrorMessage; |
| 80 | |
| 81 | // Load the module that contains the runtime helper routines neccesary for |
| 82 | // pointer hashing and stuff... link this module into the program if possible |
| 83 | // |
| 84 | Module *TraceModule = ParseBytecodeFile(TraceLibPath+"libinstr.bc"); |
| 85 | |
| 86 | // Ok, the TraceLibPath didn't contain a valid module. Try to load the module |
| 87 | // from the current LLVM-GCC install directory. This is kindof a hack, but |
| 88 | // allows people to not HAVE to have built the library. |
| 89 | // |
| 90 | if (TraceModule == 0) |
| 91 | TraceModule = ParseBytecodeFile("/home/vadve/lattner/cvs/gcc_install/lib/" |
| 92 | "gcc-lib/llvm/3.1/libinstr.bc"); |
| 93 | |
| 94 | // If we still didn't get it, cancel trying to link it in... |
| 95 | if (TraceModule == 0) { |
| 96 | cerr << "Warning, could not load trace routines to link into program!\n"; |
| 97 | } else { |
| 98 | |
| 99 | // Link in the trace routines... if the link fails, don't panic, because the |
| 100 | // compile should still succeed, just the native linker will probably fail. |
| 101 | // |
| 102 | std::auto_ptr<Module> TraceRoutines(TraceModule); |
| Chris Lattner | 7e70829 | 2002-06-25 16:13:24 +0000 | [diff] [blame] | 103 | if (LinkModules(&M, TraceRoutines.get(), &ErrorMessage)) |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 104 | cerr << "Warning: Error linking in trace routines: " |
| 105 | << ErrorMessage << "\n"; |
| 106 | } |
| 107 | |
| 108 | |
| 109 | // Write out the module with tracing code just before code generation |
| 110 | if (InputFilename != "-") { |
| 111 | string TraceFilename = GetFileNameRoot(InputFilename) + ".trace.bc"; |
| 112 | |
| 113 | std::ofstream Out(TraceFilename.c_str()); |
| 114 | if (!Out.good()) { |
| 115 | cerr << "Error opening '" << TraceFilename |
| 116 | << "'!: Skipping output of trace code as bytecode\n"; |
| 117 | } else { |
| 118 | cerr << "Emitting trace code to '" << TraceFilename |
| 119 | << "' for comparison...\n"; |
| Chris Lattner | 7e70829 | 2002-06-25 16:13:24 +0000 | [diff] [blame] | 120 | WriteBytecodeToFile(&M, Out); |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 121 | } |
| 122 | } |
| 123 | |
| 124 | } |
| 125 | |
| Vikram S. Adve | 805eb96 | 2001-09-18 13:10:45 +0000 | [diff] [blame] | 126 | |
| 127 | //===---------------------------------------------------------------------===// |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 128 | // Function main() |
| 129 | // |
| 130 | // Entry point for the llc compiler. |
| 131 | //===---------------------------------------------------------------------===// |
| 132 | |
| Chris Lattner | 25c1229 | 2001-10-15 17:41:24 +0000 | [diff] [blame] | 133 | int main(int argc, char **argv) { |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 134 | cl::ParseCommandLineOptions(argc, argv, " llvm system compiler\n"); |
| 135 | |
| 136 | // Allocate a target... in the future this will be controllable on the |
| 137 | // command line. |
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 138 | std::auto_ptr<TargetMachine> target(allocateSparcTargetMachine()); |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 139 | assert(target.get() && "Could not allocate target machine!"); |
| 140 | |
| 141 | TargetMachine &Target = *target.get(); |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 142 | |
| 143 | // Load the module to be compiled... |
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 144 | std::auto_ptr<Module> M(ParseBytecodeFile(InputFilename)); |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 145 | if (M.get() == 0) { |
| 146 | cerr << "bytecode didn't read correctly.\n"; |
| 147 | return 1; |
| 148 | } |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 149 | |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 150 | if (TraceValues != TraceOff) // If tracing enabled... |
| Chris Lattner | 7e70829 | 2002-06-25 16:13:24 +0000 | [diff] [blame] | 151 | insertTraceCodeFor(*M.get()); // Hack up module before using passmanager... |
| Chris Lattner | 39fd659 | 2002-05-20 21:20:08 +0000 | [diff] [blame] | 152 | |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 153 | // Build up all of the passes that we want to do to the module... |
| Chris Lattner | f4de63f | 2002-01-21 07:31:50 +0000 | [diff] [blame] | 154 | PassManager Passes; |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 155 | |
| Vikram S. Adve | d32e70a | 2002-03-24 03:19:54 +0000 | [diff] [blame] | 156 | // Decompose multi-dimensional refs into a sequence of 1D refs |
| 157 | Passes.add(createDecomposeMultiDimRefsPass()); |
| 158 | |
| Vikram S. Adve | 712ac2b | 2001-10-18 18:20:20 +0000 | [diff] [blame] | 159 | // Replace malloc and free instructions with library calls. |
| 160 | // Do this after tracing until lli implements these lib calls. |
| 161 | // For now, it will emulate malloc and free internally. |
| Chris Lattner | 4ddcd54 | 2002-02-26 21:36:53 +0000 | [diff] [blame] | 162 | Passes.add(createLowerAllocationsPass(Target.DataLayout)); |
| Vikram S. Adve | 712ac2b | 2001-10-18 18:20:20 +0000 | [diff] [blame] | 163 | |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 164 | // If LLVM dumping after transformations is requested, add it to the pipeline |
| 165 | if (DumpAsm) |
| Chris Lattner | 02d6ef8 | 2002-04-08 22:05:01 +0000 | [diff] [blame] | 166 | Passes.add(new PrintFunctionPass("Code after xformations: \n", &cerr)); |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 167 | |
| Chris Lattner | e41576d | 2002-02-03 23:43:19 +0000 | [diff] [blame] | 168 | // Figure out where we are going to send the output... |
| 169 | std::ostream *Out = 0; |
| 170 | if (OutputFilename != "") { // Specified an output filename? |
| 171 | if (!Force && std::ifstream(OutputFilename.c_str())) { |
| 172 | // If force is not specified, make sure not to overwrite a file! |
| 173 | cerr << "Error opening '" << OutputFilename << "': File exists!\n" |
| 174 | << "Use -f command line argument to force output\n"; |
| 175 | return 1; |
| 176 | } |
| 177 | Out = new std::ofstream(OutputFilename.c_str()); |
| Chris Lattner | 76d1229 | 2002-04-18 19:55:25 +0000 | [diff] [blame] | 178 | |
| 179 | // Make sure that the Out file gets unlink'd from the disk if we get a |
| 180 | // SIGINT |
| 181 | RemoveFileOnSignal(OutputFilename); |
| Chris Lattner | e41576d | 2002-02-03 23:43:19 +0000 | [diff] [blame] | 182 | } else { |
| 183 | if (InputFilename == "-") { |
| 184 | OutputFilename = "-"; |
| 185 | Out = &std::cout; |
| 186 | } else { |
| 187 | string OutputFilename = GetFileNameRoot(InputFilename); |
| 188 | OutputFilename += ".s"; |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 189 | |
| Chris Lattner | 888912d | 2002-01-22 21:07:24 +0000 | [diff] [blame] | 190 | if (!Force && std::ifstream(OutputFilename.c_str())) { |
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 191 | // If force is not specified, make sure not to overwrite a file! |
| 192 | cerr << "Error opening '" << OutputFilename << "': File exists!\n" |
| 193 | << "Use -f command line argument to force output\n"; |
| 194 | return 1; |
| 195 | } |
| Chris Lattner | e41576d | 2002-02-03 23:43:19 +0000 | [diff] [blame] | 196 | |
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 197 | Out = new std::ofstream(OutputFilename.c_str()); |
| Chris Lattner | e41576d | 2002-02-03 23:43:19 +0000 | [diff] [blame] | 198 | if (!Out->good()) { |
| 199 | cerr << "Error opening " << OutputFilename << "!\n"; |
| 200 | delete Out; |
| 201 | return 1; |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 202 | } |
| Chris Lattner | 76d1229 | 2002-04-18 19:55:25 +0000 | [diff] [blame] | 203 | // Make sure that the Out file gets unlink'd from the disk if we get a |
| 204 | // SIGINT |
| 205 | RemoveFileOnSignal(OutputFilename); |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 206 | } |
| Chris Lattner | 3524fc2 | 2001-10-15 17:30:47 +0000 | [diff] [blame] | 207 | } |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 208 | |
| Chris Lattner | e41576d | 2002-02-03 23:43:19 +0000 | [diff] [blame] | 209 | Target.addPassesToEmitAssembly(Passes, *Out); |
| 210 | |
| Chris Lattner | f4de63f | 2002-01-21 07:31:50 +0000 | [diff] [blame] | 211 | // Run our queue of passes all at once now, efficiently. |
| Chris Lattner | 7e70829 | 2002-06-25 16:13:24 +0000 | [diff] [blame] | 212 | Passes.run(*M.get()); |
| Chris Lattner | 05e5e07 | 2001-10-18 01:31:22 +0000 | [diff] [blame] | 213 | |
| Chris Lattner | e41576d | 2002-02-03 23:43:19 +0000 | [diff] [blame] | 214 | if (Out != &std::cout) delete Out; |
| 215 | |
| Chris Lattner | d7477ee | 2001-10-18 20:33:21 +0000 | [diff] [blame] | 216 | return 0; |
| Vikram S. Adve | 2f64f9f | 2001-10-14 23:29:28 +0000 | [diff] [blame] | 217 | } |