diff options
Diffstat (limited to 'tools/llvm-lto/llvm-lto.cpp')
| -rw-r--r-- | tools/llvm-lto/llvm-lto.cpp | 51 | 
1 files changed, 38 insertions, 13 deletions
diff --git a/tools/llvm-lto/llvm-lto.cpp b/tools/llvm-lto/llvm-lto.cpp index 9cd031eaf8d39..9cd6587c26223 100644 --- a/tools/llvm-lto/llvm-lto.cpp +++ b/tools/llvm-lto/llvm-lto.cpp @@ -26,9 +26,13 @@  using namespace llvm; -static cl::opt<bool> -DisableOpt("disable-opt", cl::init(false), -  cl::desc("Do not run any optimization passes")); +static cl::opt<char> +OptLevel("O", +         cl::desc("Optimization level. [-O0, -O1, -O2, or -O3] " +                  "(default = '-O2')"), +         cl::Prefix, +         cl::ZeroOrMore, +         cl::init('2'));  static cl::opt<bool>  DisableInline("disable-inlining", cl::init(false), @@ -69,14 +73,18 @@ static cl::opt<bool> ListSymbolsOnly(      "list-symbols-only", cl::init(false),      cl::desc("Instead of running LTO, list the symbols in each IR file")); +static cl::opt<bool> SetMergedModule( +    "set-merged-module", cl::init(false), +    cl::desc("Use the first input module as the merged module")); +  namespace {  struct ModuleInfo {    std::vector<bool> CanBeHidden;  };  } -void handleDiagnostics(lto_codegen_diagnostic_severity_t Severity, -                       const char *Msg, void *) { +static void handleDiagnostics(lto_codegen_diagnostic_severity_t Severity, +                              const char *Msg, void *) {    switch (Severity) {    case LTO_DS_NOTE:      errs() << "note: "; @@ -94,7 +102,7 @@ void handleDiagnostics(lto_codegen_diagnostic_severity_t Severity,    errs() << Msg << "\n";  } -std::unique_ptr<LTOModule> +static std::unique_ptr<LTOModule>  getLocalLTOModule(StringRef Path, std::unique_ptr<MemoryBuffer> &Buffer,                    const TargetOptions &Options, std::string &Error) {    ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr = @@ -114,7 +122,7 @@ getLocalLTOModule(StringRef Path, std::unique_ptr<MemoryBuffer> &Buffer,  /// functionality that's exposed by the C API to list symbols.  Moreover, this  /// provides testing coverage for modules that have been created in their own  /// contexts. -int listSymbols(StringRef Command, const TargetOptions &Options) { +static int listSymbols(StringRef Command, const TargetOptions &Options) {    for (auto &Filename : InputFilenames) {      std::string Error;      std::unique_ptr<MemoryBuffer> Buffer; @@ -142,6 +150,11 @@ int main(int argc, char **argv) {    llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.    cl::ParseCommandLineOptions(argc, argv, "llvm LTO linker\n"); +  if (OptLevel < '0' || OptLevel > '3') { +    errs() << argv[0] << ": optimization level must be between 0 and 3\n"; +    return 1; +  } +    // Initialize the configured targets.    InitializeAllTargets();    InitializeAllTargetMCs(); @@ -194,15 +207,22 @@ int main(int argc, char **argv) {        return 1;      } -    if (!CodeGen.addModule(Module.get())) +    LTOModule *LTOMod = Module.get(); + +    // We use the first input module as the destination module when +    // SetMergedModule is true. +    if (SetMergedModule && i == BaseArg) { +      // Transfer ownership to the code generator. +      CodeGen.setModule(Module.release()); +    } else if (!CodeGen.addModule(Module.get()))        return 1; -    unsigned NumSyms = Module->getSymbolCount(); +    unsigned NumSyms = LTOMod->getSymbolCount();      for (unsigned I = 0; I < NumSyms; ++I) { -      StringRef Name = Module->getSymbolName(I); +      StringRef Name = LTOMod->getSymbolName(I);        if (!DSOSymbolsSet.count(Name))          continue; -      lto_symbol_attributes Attrs = Module->getSymbolAttributes(I); +      lto_symbol_attributes Attrs = LTOMod->getSymbolAttributes(I);        unsigned Scope = Attrs & LTO_SYMBOL_SCOPE_MASK;        if (Scope != LTO_SYMBOL_SCOPE_DEFAULT_CAN_BE_HIDDEN)          KeptDSOSyms.push_back(Name); @@ -217,6 +237,11 @@ int main(int argc, char **argv) {    for (unsigned i = 0; i < KeptDSOSyms.size(); ++i)      CodeGen.addMustPreserveSymbol(KeptDSOSyms[i].c_str()); +  // Set cpu and attrs strings for the default target/subtarget. +  CodeGen.setCpu(MCPU.c_str()); + +  CodeGen.setOptLevel(OptLevel - '0'); +    std::string attrs;    for (unsigned i = 0; i < MAttrs.size(); ++i) {      if (i > 0) @@ -231,7 +256,7 @@ int main(int argc, char **argv) {      size_t len = 0;      std::string ErrorInfo;      const void *Code = -        CodeGen.compile(&len, DisableOpt, DisableInline, DisableGVNLoadPRE, +        CodeGen.compile(&len, DisableInline, DisableGVNLoadPRE,                          DisableLTOVectorization, ErrorInfo);      if (!Code) {        errs() << argv[0] @@ -251,7 +276,7 @@ int main(int argc, char **argv) {    } else {      std::string ErrorInfo;      const char *OutputName = nullptr; -    if (!CodeGen.compile_to_file(&OutputName, DisableOpt, DisableInline, +    if (!CodeGen.compile_to_file(&OutputName, DisableInline,                                   DisableGVNLoadPRE, DisableLTOVectorization,                                   ErrorInfo)) {        errs() << argv[0]  | 
