llvm-lto.cpp revision 438900938c3ac9d7fac2dd5d2c85ca4b9b2e35f7
1//===-- llvm-lto: a simple command-line program to link modules with LTO --===// 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// This program takes in a list of bitcode files, links them, performs link-time 11// optimization, and outputs an object file. 12// 13//===----------------------------------------------------------------------===// 14 15#include "llvm/CodeGen/CommandFlags.h" 16#include "llvm/LTO/LTOCodeGenerator.h" 17#include "llvm/LTO/LTOModule.h" 18#include "llvm/Support/CommandLine.h" 19#include "llvm/Support/ManagedStatic.h" 20#include "llvm/Support/PrettyStackTrace.h" 21#include "llvm/Support/Signals.h" 22#include "llvm/Support/raw_ostream.h" 23#include "llvm/Support/TargetSelect.h" 24 25using namespace llvm; 26 27static cl::opt<bool> 28DisableOpt("disable-opt", cl::init(false), 29 cl::desc("Do not run any optimization passes")); 30 31static cl::opt<bool> 32DisableInline("disable-inlining", cl::init(false), 33 cl::desc("Do not run the inliner pass")); 34 35static cl::opt<bool> 36DisableGVNLoadPRE("disable-gvn-loadpre", cl::init(false), 37 cl::desc("Do not run the GVN load PRE pass")); 38 39static cl::list<std::string> 40InputFilenames(cl::Positional, cl::OneOrMore, 41 cl::desc("<input bitcode files>")); 42 43static cl::opt<std::string> 44OutputFilename("o", cl::init(""), 45 cl::desc("Override output filename"), 46 cl::value_desc("filename")); 47 48static cl::list<std::string> 49ExportedSymbols("exported-symbol", 50 cl::desc("Symbol to export from the resulting object file"), 51 cl::ZeroOrMore); 52 53static cl::list<std::string> 54DSOSymbols("dso-symbol", 55 cl::desc("Symbol to put in the symtab in the resulting dso"), 56 cl::ZeroOrMore); 57 58int main(int argc, char **argv) { 59 // Print a stack trace if we signal out. 60 sys::PrintStackTraceOnErrorSignal(); 61 PrettyStackTraceProgram X(argc, argv); 62 63 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit. 64 cl::ParseCommandLineOptions(argc, argv, "llvm LTO linker\n"); 65 66 // Initialize the configured targets. 67 InitializeAllTargets(); 68 InitializeAllTargetMCs(); 69 InitializeAllAsmPrinters(); 70 InitializeAllAsmParsers(); 71 72 // set up the TargetOptions for the machine 73 TargetOptions Options; 74 Options.LessPreciseFPMADOption = EnableFPMAD; 75 Options.NoFramePointerElim = DisableFPElim; 76 Options.AllowFPOpFusion = FuseFPOps; 77 Options.UnsafeFPMath = EnableUnsafeFPMath; 78 Options.NoInfsFPMath = EnableNoInfsFPMath; 79 Options.NoNaNsFPMath = EnableNoNaNsFPMath; 80 Options.HonorSignDependentRoundingFPMathOption = 81 EnableHonorSignDependentRoundingFPMath; 82 Options.UseSoftFloat = GenerateSoftFloatCalls; 83 if (FloatABIForCalls != FloatABI::Default) 84 Options.FloatABIType = FloatABIForCalls; 85 Options.NoZerosInBSS = DontPlaceZerosInBSS; 86 Options.GuaranteedTailCallOpt = EnableGuaranteedTailCallOpt; 87 Options.DisableTailCalls = DisableTailCalls; 88 Options.StackAlignmentOverride = OverrideStackAlignment; 89 Options.TrapFuncName = TrapFuncName; 90 Options.PositionIndependentExecutable = EnablePIE; 91 Options.EnableSegmentedStacks = SegmentedStacks; 92 Options.UseInitArray = UseInitArray; 93 94 unsigned BaseArg = 0; 95 96 LTOCodeGenerator CodeGen; 97 98 CodeGen.setCodePICModel(LTO_CODEGEN_PIC_MODEL_DYNAMIC); 99 CodeGen.setDebugInfo(LTO_DEBUG_MODEL_DWARF); 100 CodeGen.setTargetOptions(Options); 101 102 for (unsigned i = BaseArg; i < InputFilenames.size(); ++i) { 103 std::string error; 104 OwningPtr<LTOModule> Module(LTOModule::makeLTOModule(InputFilenames[i].c_str(), 105 Options, error)); 106 if (!error.empty()) { 107 errs() << argv[0] << ": error loading file '" << InputFilenames[i] 108 << "': " << error << "\n"; 109 return 1; 110 } 111 112 113 if (!CodeGen.addModule(Module.get(), error)) { 114 errs() << argv[0] << ": error adding file '" << InputFilenames[i] 115 << "': " << error << "\n"; 116 return 1; 117 } 118 } 119 120 // Add all the exported symbols to the table of symbols to preserve. 121 for (unsigned i = 0; i < ExportedSymbols.size(); ++i) 122 CodeGen.addMustPreserveSymbol(ExportedSymbols[i].c_str()); 123 124 // Add all the dso symbols to the table of symbols to expose. 125 for (unsigned i = 0; i < DSOSymbols.size(); ++i) 126 CodeGen.addDSOSymbol(DSOSymbols[i].c_str()); 127 128 if (!OutputFilename.empty()) { 129 size_t len = 0; 130 std::string ErrorInfo; 131 const void *Code = CodeGen.compile(&len, DisableOpt, DisableInline, 132 DisableGVNLoadPRE, ErrorInfo); 133 if (Code == NULL) { 134 errs() << argv[0] 135 << ": error compiling the code: " << ErrorInfo << "\n"; 136 return 1; 137 } 138 139 raw_fd_ostream FileStream(OutputFilename.c_str(), ErrorInfo); 140 if (!ErrorInfo.empty()) { 141 errs() << argv[0] << ": error opening the file '" << OutputFilename 142 << "': " << ErrorInfo << "\n"; 143 return 1; 144 } 145 146 FileStream.write(reinterpret_cast<const char *>(Code), len); 147 } else { 148 std::string ErrorInfo; 149 const char *OutputName = NULL; 150 if (!CodeGen.compile_to_file(&OutputName, DisableOpt, DisableInline, 151 DisableGVNLoadPRE, ErrorInfo)) { 152 errs() << argv[0] 153 << ": error compiling the code: " << ErrorInfo 154 << "\n"; 155 return 1; 156 } 157 158 outs() << "Wrote native object file '" << OutputName << "'\n"; 159 } 160 161 return 0; 162} 163