| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 1 | //===- LTO.cpp ------------------------------------------------------------===// | 
|  | 2 | // | 
| Chandler Carruth | 2946cd7 | 2019-01-19 08:50:56 +0000 | [diff] [blame] | 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 | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 6 | // | 
|  | 7 | //===----------------------------------------------------------------------===// | 
|  | 8 |  | 
|  | 9 | #include "LTO.h" | 
|  | 10 | #include "Config.h" | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 11 | #include "InputFiles.h" | 
| Rui Ueyama | 3f85170 | 2017-10-02 21:00:41 +0000 | [diff] [blame] | 12 | #include "LinkerScript.h" | 
| Rafael Espindola | 4b075bb | 2017-07-26 23:39:10 +0000 | [diff] [blame] | 13 | #include "SymbolTable.h" | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 14 | #include "Symbols.h" | 
| Bob Haarman | b8a59c8 | 2017-10-25 22:28:38 +0000 | [diff] [blame] | 15 | #include "lld/Common/ErrorHandler.h" | 
| Rui Ueyama | 3f85170 | 2017-10-02 21:00:41 +0000 | [diff] [blame] | 16 | #include "lld/Common/TargetOptionsCommandFlags.h" | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 17 | #include "llvm/ADT/STLExtras.h" | 
| Rui Ueyama | 8c6a5aa | 2016-11-05 22:37:59 +0000 | [diff] [blame] | 18 | #include "llvm/ADT/SmallString.h" | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 19 | #include "llvm/ADT/StringRef.h" | 
|  | 20 | #include "llvm/ADT/Twine.h" | 
| Zachary Turner | 264b5d9 | 2017-06-07 03:48:56 +0000 | [diff] [blame] | 21 | #include "llvm/BinaryFormat/ELF.h" | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 22 | #include "llvm/Bitcode/BitcodeReader.h" | 
|  | 23 | #include "llvm/Bitcode/BitcodeWriter.h" | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 24 | #include "llvm/IR/DiagnosticPrinter.h" | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 25 | #include "llvm/LTO/Caching.h" | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 26 | #include "llvm/LTO/Config.h" | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 27 | #include "llvm/LTO/LTO.h" | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 28 | #include "llvm/Object/SymbolicFile.h" | 
|  | 29 | #include "llvm/Support/CodeGen.h" | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 30 | #include "llvm/Support/Error.h" | 
|  | 31 | #include "llvm/Support/FileSystem.h" | 
|  | 32 | #include "llvm/Support/MemoryBuffer.h" | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 33 | #include <algorithm> | 
|  | 34 | #include <cstddef> | 
|  | 35 | #include <memory> | 
|  | 36 | #include <string> | 
|  | 37 | #include <system_error> | 
|  | 38 | #include <vector> | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 39 |  | 
|  | 40 | using namespace llvm; | 
|  | 41 | using namespace llvm::object; | 
|  | 42 | using namespace llvm::ELF; | 
|  | 43 |  | 
|  | 44 | using namespace lld; | 
|  | 45 | using namespace lld::elf; | 
|  | 46 |  | 
| Rui Ueyama | 66a9f25 | 2018-05-07 17:46:28 +0000 | [diff] [blame] | 47 | // Creates an empty file to store a list of object files for final | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 48 | // linking of distributed ThinLTO. | 
| Rui Ueyama | 66a9f25 | 2018-05-07 17:46:28 +0000 | [diff] [blame] | 49 | static std::unique_ptr<raw_fd_ostream> openFile(StringRef File) { | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 50 | std::error_code EC; | 
| Rui Ueyama | 66a9f25 | 2018-05-07 17:46:28 +0000 | [diff] [blame] | 51 | auto Ret = | 
|  | 52 | llvm::make_unique<raw_fd_ostream>(File, EC, sys::fs::OpenFlags::F_None); | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 53 | if (EC) { | 
|  | 54 | error("cannot open " + File + ": " + EC.message()); | 
|  | 55 | return nullptr; | 
|  | 56 | } | 
| Rui Ueyama | 66a9f25 | 2018-05-07 17:46:28 +0000 | [diff] [blame] | 57 | return Ret; | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 58 | } | 
|  | 59 |  | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 60 | static std::string getThinLTOOutputFile(StringRef ModulePath) { | 
|  | 61 | return lto::getThinLTOOutputFile(ModulePath, | 
|  | 62 | Config->ThinLTOPrefixReplace.first, | 
| Rumeet Dhindsa | 4fb5119 | 2018-05-07 23:14:12 +0000 | [diff] [blame] | 63 | Config->ThinLTOPrefixReplace.second); | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 64 | } | 
|  | 65 |  | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 66 | static lto::Config createConfig() { | 
|  | 67 | lto::Config C; | 
| Davide Italiano | d26c4a1 | 2016-05-15 19:29:38 +0000 | [diff] [blame] | 68 |  | 
| Peter Collingbourne | 82a7f14 | 2018-08-06 20:12:12 +0000 | [diff] [blame] | 69 | // LLD supports the new relocations and address-significance tables. | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 70 | C.Options = InitTargetOptionsFromCodeGenFlags(); | 
|  | 71 | C.Options.RelaxELFRelocations = true; | 
| Peter Collingbourne | 82a7f14 | 2018-08-06 20:12:12 +0000 | [diff] [blame] | 72 | C.Options.EmitAddrsig = true; | 
| Davide Italiano | df24d5b | 2016-06-02 22:58:11 +0000 | [diff] [blame] | 73 |  | 
| Davide Italiano | 957f120 | 2017-07-24 20:15:07 +0000 | [diff] [blame] | 74 | // Always emit a section per function/datum with LTO. | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 75 | C.Options.FunctionSections = true; | 
|  | 76 | C.Options.DataSections = true; | 
| Davide Italiano | 1f4e29c | 2017-07-24 19:38:13 +0000 | [diff] [blame] | 77 |  | 
| Evgeniy Stepanov | f12ac5b | 2017-05-22 21:11:44 +0000 | [diff] [blame] | 78 | if (Config->Relocatable) | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 79 | C.RelocModel = None; | 
| Evgeniy Stepanov | f12ac5b | 2017-05-22 21:11:44 +0000 | [diff] [blame] | 80 | else if (Config->Pic) | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 81 | C.RelocModel = Reloc::PIC_; | 
| Evgeniy Stepanov | f12ac5b | 2017-05-22 21:11:44 +0000 | [diff] [blame] | 82 | else | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 83 | C.RelocModel = Reloc::Static; | 
|  | 84 |  | 
|  | 85 | C.CodeModel = GetCodeModelFromCMModel(); | 
|  | 86 | C.DisableVerify = Config->DisableVerify; | 
|  | 87 | C.DiagHandler = diagnosticHandler; | 
|  | 88 | C.OptLevel = Config->LTOO; | 
|  | 89 | C.CPU = GetCPUStr(); | 
| Fangrui Song | ccfc841 | 2018-11-01 20:02:49 +0000 | [diff] [blame] | 90 | C.MAttrs = GetMAttrs(); | 
| Davide Italiano | df24d5b | 2016-06-02 22:58:11 +0000 | [diff] [blame] | 91 |  | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 92 | // Set up a custom pipeline if we've been asked to. | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 93 | C.OptPipeline = Config->LTONewPmPasses; | 
|  | 94 | C.AAPipeline = Config->LTOAAPipeline; | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 95 |  | 
| Davide Italiano | db4b0a7 | 2017-02-13 17:49:18 +0000 | [diff] [blame] | 96 | // Set up optimization remarks if we've been asked to. | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 97 | C.RemarksFilename = Config->OptRemarksFilename; | 
|  | 98 | C.RemarksWithHotness = Config->OptRemarksWithHotness; | 
|  | 99 |  | 
|  | 100 | C.SampleProfile = Config->LTOSampleProfile; | 
|  | 101 | C.UseNewPM = Config->LTONewPassManager; | 
|  | 102 | C.DebugPassManager = Config->LTODebugPassManager; | 
| Yunlian Jiang | 496fb3e | 2018-07-16 17:55:48 +0000 | [diff] [blame] | 103 | C.DwoDir = Config->DwoDir; | 
| Davide Italiano | db4b0a7 | 2017-02-13 17:49:18 +0000 | [diff] [blame] | 104 |  | 
| Rui Ueyama | 9f49990 | 2018-12-14 21:58:49 +0000 | [diff] [blame] | 105 | if (Config->EmitLLVM) { | 
|  | 106 | C.PostInternalizeModuleHook = [](size_t Task, const Module &M) { | 
|  | 107 | if (std::unique_ptr<raw_fd_ostream> OS = openFile(Config->OutputFile)) | 
|  | 108 | WriteBitcodeToFile(M, *OS, false); | 
|  | 109 | return false; | 
|  | 110 | }; | 
|  | 111 | } | 
|  | 112 |  | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 113 | if (Config->SaveTemps) | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 114 | checkError(C.addSaveTemps(Config->OutputFile.str() + ".", | 
|  | 115 | /*UseInputModulePath*/ true)); | 
|  | 116 | return C; | 
|  | 117 | } | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 118 |  | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 119 | BitcodeCompiler::BitcodeCompiler() { | 
| Rui Ueyama | 98e4a5c | 2018-09-11 14:37:27 +0000 | [diff] [blame] | 120 | // Initialize IndexFile. | 
|  | 121 | if (!Config->ThinLTOIndexOnlyArg.empty()) | 
|  | 122 | IndexFile = openFile(Config->ThinLTOIndexOnlyArg); | 
|  | 123 |  | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 124 | // Initialize LTOObj. | 
| Davide Italiano | 7a7b35a | 2016-10-10 18:12:53 +0000 | [diff] [blame] | 125 | lto::ThinBackend Backend; | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 126 | if (Config->ThinLTOIndexOnly) { | 
| Rui Ueyama | 98e4a5c | 2018-09-11 14:37:27 +0000 | [diff] [blame] | 127 | auto OnIndexWrite = [&](StringRef S) { ThinIndices.erase(S); }; | 
| Rui Ueyama | 4454b3d | 2018-05-07 17:59:43 +0000 | [diff] [blame] | 128 | Backend = lto::createWriteIndexesThinBackend( | 
|  | 129 | Config->ThinLTOPrefixReplace.first, Config->ThinLTOPrefixReplace.second, | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 130 | Config->ThinLTOEmitImportsFiles, IndexFile.get(), OnIndexWrite); | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 131 | } else if (Config->ThinLTOJobs != -1U) { | 
|  | 132 | Backend = lto::createInProcessThinBackend(Config->ThinLTOJobs); | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 133 | } | 
|  | 134 |  | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 135 | LTOObj = llvm::make_unique<lto::LTO>(createConfig(), Backend, | 
| Rui Ueyama | 66a9f25 | 2018-05-07 17:46:28 +0000 | [diff] [blame] | 136 | Config->LTOPartitions); | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 137 |  | 
| Rui Ueyama | 397dffd | 2018-05-07 23:24:07 +0000 | [diff] [blame] | 138 | // Initialize UsedStartStop. | 
| Rui Ueyama | f52496e | 2017-11-03 21:21:47 +0000 | [diff] [blame] | 139 | for (Symbol *Sym : Symtab->getSymbols()) { | 
| Rui Ueyama | 98e4a5c | 2018-09-11 14:37:27 +0000 | [diff] [blame] | 140 | StringRef S = Sym->getName(); | 
| Rafael Espindola | 4b075bb | 2017-07-26 23:39:10 +0000 | [diff] [blame] | 141 | for (StringRef Prefix : {"__start_", "__stop_"}) | 
| Rui Ueyama | 98e4a5c | 2018-09-11 14:37:27 +0000 | [diff] [blame] | 142 | if (S.startswith(Prefix)) | 
|  | 143 | UsedStartStop.insert(S.substr(Prefix.size())); | 
| Rafael Espindola | 4b075bb | 2017-07-26 23:39:10 +0000 | [diff] [blame] | 144 | } | 
|  | 145 | } | 
| Rafael Espindola | ae605c1 | 2016-04-21 20:35:25 +0000 | [diff] [blame] | 146 |  | 
| Eugene Zelenko | 22886a2 | 2016-11-05 01:00:56 +0000 | [diff] [blame] | 147 | BitcodeCompiler::~BitcodeCompiler() = default; | 
| Rui Ueyama | 412c802 | 2016-04-22 21:16:18 +0000 | [diff] [blame] | 148 |  | 
| Rui Ueyama | f52496e | 2017-11-03 21:21:47 +0000 | [diff] [blame] | 149 | static void undefine(Symbol *S) { | 
| Rafael Espindola | bec3765 | 2017-11-17 01:37:50 +0000 | [diff] [blame] | 150 | replaceSymbol<Undefined>(S, nullptr, S->getName(), STB_GLOBAL, STV_DEFAULT, | 
|  | 151 | S->Type); | 
| Peter Collingbourne | 0ef3874 | 2016-05-12 19:46:14 +0000 | [diff] [blame] | 152 | } | 
|  | 153 |  | 
| Peter Smith | 3a52eb0 | 2017-02-01 10:26:03 +0000 | [diff] [blame] | 154 | void BitcodeCompiler::add(BitcodeFile &F) { | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 155 | lto::InputFile &Obj = *F.Obj; | 
| Rui Ueyama | d31b54b | 2018-05-08 17:50:54 +0000 | [diff] [blame] | 156 | bool IsExec = !Config->Shared && !Config->Relocatable; | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 157 |  | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 158 | if (Config->ThinLTOIndexOnly) | 
| Rui Ueyama | 98e4a5c | 2018-09-11 14:37:27 +0000 | [diff] [blame] | 159 | ThinIndices.insert(Obj.getName()); | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 160 |  | 
| Rui Ueyama | 2f9fa42 | 2018-05-08 17:50:43 +0000 | [diff] [blame] | 161 | ArrayRef<Symbol *> Syms = F.getSymbols(); | 
| Rui Ueyama | d31b54b | 2018-05-08 17:50:54 +0000 | [diff] [blame] | 162 | ArrayRef<lto::InputFile::Symbol> ObjSyms = Obj.symbols(); | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 163 | std::vector<lto::SymbolResolution> Resols(Syms.size()); | 
| Davide Italiano | 334fce9 | 2016-05-11 01:07:22 +0000 | [diff] [blame] | 164 |  | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 165 | // Provide a resolution to the LTO API for each symbol. | 
| Rui Ueyama | d31b54b | 2018-05-08 17:50:54 +0000 | [diff] [blame] | 166 | for (size_t I = 0, E = Syms.size(); I != E; ++I) { | 
|  | 167 | Symbol *Sym = Syms[I]; | 
|  | 168 | const lto::InputFile::Symbol &ObjSym = ObjSyms[I]; | 
|  | 169 | lto::SymbolResolution &R = Resols[I]; | 
| Davide Italiano | 86f2bd5 | 2016-03-29 21:46:35 +0000 | [diff] [blame] | 170 |  | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 171 | // Ideally we shouldn't check for SF_Undefined but currently IRObjectFile | 
|  | 172 | // reports two symbols for module ASM defined. Without this check, lld | 
|  | 173 | // flags an undefined in IR with a definition in ASM as prevailing. | 
|  | 174 | // Once IRObjectFile is fixed to report only one symbol this hack can | 
|  | 175 | // be removed. | 
| Rafael Espindola | dfebd36 | 2017-11-29 22:47:35 +0000 | [diff] [blame] | 176 | R.Prevailing = !ObjSym.isUndefined() && Sym->File == &F; | 
| Peter Collingbourne | 3ad1c1e | 2016-05-05 17:13:49 +0000 | [diff] [blame] | 177 |  | 
| George Rimar | 3a1af22 | 2017-08-22 08:36:54 +0000 | [diff] [blame] | 178 | // We ask LTO to preserve following global symbols: | 
|  | 179 | // 1) All symbols when doing relocatable link, so that them can be used | 
|  | 180 | //    for doing final link. | 
|  | 181 | // 2) Symbols that are used in regular objects. | 
|  | 182 | // 3) C named sections if we have corresponding __start_/__stop_ symbol. | 
|  | 183 | // 4) Symbols that are defined in bitcode files and used for dynamic linking. | 
|  | 184 | R.VisibleToRegularObj = Config->Relocatable || Sym->IsUsedInRegularObj || | 
| Rafael Espindola | affe720 | 2017-07-25 22:51:05 +0000 | [diff] [blame] | 185 | (R.Prevailing && Sym->includeInDynsym()) || | 
| Rafael Espindola | 4b075bb | 2017-07-26 23:39:10 +0000 | [diff] [blame] | 186 | UsedStartStop.count(ObjSym.getSectionName()); | 
| Dmitry Mikulin | c84e0ee | 2018-02-07 00:49:51 +0000 | [diff] [blame] | 187 | const auto *DR = dyn_cast<Defined>(Sym); | 
| Rafael Espindola | c6df38c | 2018-01-16 16:49:05 +0000 | [diff] [blame] | 188 | R.FinalDefinitionInLinkageUnit = | 
| Rui Ueyama | d31b54b | 2018-05-08 17:50:54 +0000 | [diff] [blame] | 189 | (IsExec || Sym->Visibility != STV_DEFAULT) && DR && | 
| Dmitry Mikulin | c84e0ee | 2018-02-07 00:49:51 +0000 | [diff] [blame] | 190 | // Skip absolute symbols from ELF objects, otherwise PC-rel relocations | 
|  | 191 | // will be generated by for them, triggering linker errors. | 
|  | 192 | // Symbol section is always null for bitcode symbols, hence the check | 
| Dmitry Mikulin | 8ddd922 | 2018-02-08 04:25:52 +0000 | [diff] [blame] | 193 | // for isElf(). Skip linker script defined symbols as well: they have | 
|  | 194 | // no File defined. | 
|  | 195 | !(DR->Section == nullptr && (!Sym->File || Sym->File->isElf())); | 
| Rafael Espindola | c6df38c | 2018-01-16 16:49:05 +0000 | [diff] [blame] | 196 |  | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 197 | if (R.Prevailing) | 
| Peter Smith | 3a52eb0 | 2017-02-01 10:26:03 +0000 | [diff] [blame] | 198 | undefine(Sym); | 
| George Rimar | d28c26b | 2017-09-25 09:31:43 +0000 | [diff] [blame] | 199 |  | 
| George Rimar | c4ccfb5 | 2018-01-30 09:04:27 +0000 | [diff] [blame] | 200 | // We tell LTO to not apply interprocedural optimization for wrapped | 
|  | 201 | // (with --wrap) symbols because otherwise LTO would inline them while | 
|  | 202 | // their values are still not final. | 
|  | 203 | R.LinkerRedefined = !Sym->CanInline; | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 204 | } | 
| Davide Italiano | 3bfa081 | 2016-11-26 05:37:04 +0000 | [diff] [blame] | 205 | checkError(LTOObj->add(std::move(F.Obj), Resols)); | 
| Davide Italiano | bc17663 | 2016-04-15 22:38:10 +0000 | [diff] [blame] | 206 | } | 
|  | 207 |  | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 208 | static void createEmptyIndex(StringRef ModulePath) { | 
|  | 209 | std::string Path = replaceThinLTOSuffix(getThinLTOOutputFile(ModulePath)); | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 210 | std::unique_ptr<raw_fd_ostream> OS = openFile(Path + ".thinlto.bc"); | 
|  | 211 | if (!OS) | 
|  | 212 | return; | 
|  | 213 |  | 
| Teresa Johnson | 566b402 | 2018-06-06 22:22:13 +0000 | [diff] [blame] | 214 | ModuleSummaryIndex M(/*HaveGVs*/ false); | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 215 | M.setSkipModuleByDistributedBackend(); | 
|  | 216 | WriteIndexToFile(M, *OS); | 
|  | 217 |  | 
|  | 218 | if (Config->ThinLTOEmitImportsFiles) | 
|  | 219 | openFile(Path + ".imports"); | 
|  | 220 | } | 
|  | 221 |  | 
| Rui Ueyama | 2599248 | 2016-03-22 20:52:10 +0000 | [diff] [blame] | 222 | // Merge all the bitcode files we have seen, codegen the result | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 223 | // and return the resulting ObjectFile(s). | 
| Rui Ueyama | 38dbd3e | 2016-09-14 00:05:51 +0000 | [diff] [blame] | 224 | std::vector<InputFile *> BitcodeCompiler::compile() { | 
| Davide Italiano | 3bfa081 | 2016-11-26 05:37:04 +0000 | [diff] [blame] | 225 | unsigned MaxTasks = LTOObj->getMaxTasks(); | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 226 | Buf.resize(MaxTasks); | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 227 | Files.resize(MaxTasks); | 
| Davide Italiano | 828ac541 | 2016-03-28 15:44:21 +0000 | [diff] [blame] | 228 |  | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 229 | // The --thinlto-cache-dir option specifies the path to a directory in which | 
|  | 230 | // to cache native object files for ThinLTO incremental builds. If a path was | 
|  | 231 | // specified, configure LTO to use it as the cache directory. | 
|  | 232 | lto::NativeObjectCache Cache; | 
|  | 233 | if (!Config->ThinLTOCacheDir.empty()) | 
| Peter Collingbourne | 128423f | 2017-03-17 00:34:07 +0000 | [diff] [blame] | 234 | Cache = check( | 
|  | 235 | lto::localCache(Config->ThinLTOCacheDir, | 
| Teresa Johnson | 2c2ed3c | 2018-02-20 20:21:59 +0000 | [diff] [blame] | 236 | [&](size_t Task, std::unique_ptr<MemoryBuffer> MB) { | 
|  | 237 | Files[Task] = std::move(MB); | 
|  | 238 | })); | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 239 |  | 
|  | 240 | checkError(LTOObj->run( | 
|  | 241 | [&](size_t Task) { | 
|  | 242 | return llvm::make_unique<lto::NativeObjectStream>( | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 243 | llvm::make_unique<raw_svector_ostream>(Buf[Task])); | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 244 | }, | 
|  | 245 | Cache)); | 
| Rafael Espindola | abf6c65 | 2016-04-17 23:20:08 +0000 | [diff] [blame] | 246 |  | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 247 | // Emit empty index files for non-indexed files | 
| Rui Ueyama | 98e4a5c | 2018-09-11 14:37:27 +0000 | [diff] [blame] | 248 | for (StringRef S : ThinIndices) { | 
|  | 249 | std::string Path = getThinLTOOutputFile(S); | 
|  | 250 | openFile(Path + ".thinlto.bc"); | 
|  | 251 | if (Config->ThinLTOEmitImportsFiles) | 
|  | 252 | openFile(Path + ".imports"); | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 253 | } | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 254 |  | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 255 | // If LazyObjFile has not been added to link, emit empty index files. | 
|  | 256 | // This is needed because this is what GNU gold plugin does and we have a | 
|  | 257 | // distributed build system that depends on that behavior. | 
|  | 258 | if (Config->ThinLTOIndexOnly) { | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 259 | for (LazyObjFile *F : LazyObjFiles) | 
|  | 260 | if (!F->AddedToLink && isBitcode(F->MB)) | 
|  | 261 | createEmptyIndex(F->getName()); | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 262 |  | 
| Rumeet Dhindsa | b5b7d6e | 2018-05-08 22:37:57 +0000 | [diff] [blame] | 263 | if (!Config->LTOObjPath.empty()) | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 264 | saveBuffer(Buf[0], Config->LTOObjPath); | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 265 |  | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 266 | // ThinLTO with index only option is required to generate only the index | 
|  | 267 | // files. After that, we exit from linker and ThinLTO backend runs in a | 
|  | 268 | // distributed environment. | 
| Rui Ueyama | 554adb2 | 2018-05-07 22:11:34 +0000 | [diff] [blame] | 269 | if (IndexFile) | 
|  | 270 | IndexFile->close(); | 
|  | 271 | return {}; | 
| Rui Ueyama | d54f1c2 | 2018-05-07 22:11:24 +0000 | [diff] [blame] | 272 | } | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 273 |  | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 274 | if (!Config->ThinLTOCacheDir.empty()) | 
|  | 275 | pruneCache(Config->ThinLTOCacheDir, Config->ThinLTOCachePolicy); | 
|  | 276 |  | 
|  | 277 | std::vector<InputFile *> Ret; | 
|  | 278 | for (unsigned I = 0; I != MaxTasks; ++I) { | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 279 | if (Buf[I].empty()) | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 280 | continue; | 
|  | 281 | if (Config->SaveTemps) { | 
|  | 282 | if (I == 0) | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 283 | saveBuffer(Buf[I], Config->OutputFile + ".lto.o"); | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 284 | else | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 285 | saveBuffer(Buf[I], Config->OutputFile + Twine(I) + ".lto.o"); | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 286 | } | 
| Rui Ueyama | f06d494 | 2018-05-17 18:27:12 +0000 | [diff] [blame] | 287 | InputFile *Obj = createObjectFile(MemoryBufferRef(Buf[I], "lto.tmp")); | 
| Rumeet Dhindsa | 1888326 | 2018-05-08 20:12:07 +0000 | [diff] [blame] | 288 | Ret.push_back(Obj); | 
|  | 289 | } | 
| Rumeet Dhindsa | d366e36 | 2018-05-02 21:40:07 +0000 | [diff] [blame] | 290 |  | 
| Peter Collingbourne | e02775f | 2017-03-01 23:00:10 +0000 | [diff] [blame] | 291 | for (std::unique_ptr<MemoryBuffer> &File : Files) | 
|  | 292 | if (File) | 
|  | 293 | Ret.push_back(createObjectFile(*File)); | 
| Davide Italiano | 786d8e3 | 2016-09-29 00:40:08 +0000 | [diff] [blame] | 294 | return Ret; | 
| Rui Ueyama | 961f2ff | 2016-03-23 21:19:27 +0000 | [diff] [blame] | 295 | } |