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