forked from OSchip/llvm-project
Refactor ThinLTO-related code in BitcodeCompiler.cpp. NFC.
Summary: Refactor ThinLTO-related code in BitcodeCompiler.cpp. NFC. Reviewers: rdhindsa, espindola Subscribers: emaste, inglorion, arichardson, llvm-commits, eraman Differential Revision: https://reviews.llvm.org/D46549 llvm-svn: 331689
This commit is contained in:
parent
4ef47eb121
commit
d54f1c2fed
|
@ -67,38 +67,6 @@ static void checkError(Error E) {
|
|||
[&](ErrorInfoBase &EIB) { error(EIB.message()); });
|
||||
}
|
||||
|
||||
// With the ThinLTOIndexOnly option, only the thin link is performed, and will
|
||||
// generate index files for the ThinLTO backends in a distributed build system.
|
||||
// The distributed build system may expect that index files are created for all
|
||||
// input bitcode objects provided to the linker for the thin link. However,
|
||||
// index files will not normally be created for input bitcode objects that
|
||||
// either aren't selected by the linker (i.e. in a static library and not
|
||||
// needed), or because they don't have a summary. Therefore we need to create
|
||||
// empty dummy index file outputs in those cases.
|
||||
// If SkipModule is true then .thinlto.bc should contain just
|
||||
// SkipModuleByDistributedBackend flag which requests distributed backend
|
||||
// to skip the compilation of the corresponding module and produce an empty
|
||||
// object file.
|
||||
static void writeEmptyDistributedBuildOutputs(StringRef ModulePath,
|
||||
bool SkipModule) {
|
||||
std::string NewModulePath =
|
||||
lto::getThinLTOOutputFile(ModulePath, Config->ThinLTOPrefixReplace.first,
|
||||
Config->ThinLTOPrefixReplace.second);
|
||||
|
||||
std::error_code EC;
|
||||
raw_fd_ostream OS(NewModulePath + ".thinlto.bc", EC,
|
||||
sys::fs::OpenFlags::F_None);
|
||||
if (EC)
|
||||
error("failed to write " + NewModulePath + ".thinlto.bc" + ": " +
|
||||
EC.message());
|
||||
|
||||
if (SkipModule) {
|
||||
ModuleSummaryIndex Index(false);
|
||||
Index.setSkipModuleByDistributedBackend();
|
||||
WriteIndexToFile(Index, OS);
|
||||
}
|
||||
}
|
||||
|
||||
// Creates an empty file to store a list of object files for final
|
||||
// linking of distributed ThinLTO.
|
||||
static std::unique_ptr<raw_fd_ostream> openFile(StringRef File) {
|
||||
|
@ -108,11 +76,20 @@ static std::unique_ptr<raw_fd_ostream> openFile(StringRef File) {
|
|||
std::error_code EC;
|
||||
auto Ret =
|
||||
llvm::make_unique<raw_fd_ostream>(File, EC, sys::fs::OpenFlags::F_None);
|
||||
if (EC)
|
||||
error("cannot create " + File + ": " + EC.message());
|
||||
if (EC) {
|
||||
error("cannot open " + File + ": " + EC.message());
|
||||
return nullptr;
|
||||
}
|
||||
return Ret;
|
||||
}
|
||||
|
||||
static std::string getThinLTOOutputFile(StringRef ModulePath) {
|
||||
return lto::getThinLTOOutputFile(ModulePath,
|
||||
Config->ThinLTOPrefixReplace.first,
|
||||
Config->ThinLTOPrefixReplace.second) +
|
||||
".thinlto.bc";
|
||||
}
|
||||
|
||||
// Initializes IndexFile, Backend and LTOObj members.
|
||||
void BitcodeCompiler::init() {
|
||||
lto::Config Conf;
|
||||
|
@ -191,7 +168,7 @@ void BitcodeCompiler::add(BitcodeFile &F) {
|
|||
|
||||
// Create the empty files which, if indexed, will be overwritten later.
|
||||
if (Config->ThinLTOIndexOnly)
|
||||
writeEmptyDistributedBuildOutputs(Obj.getName(), false);
|
||||
openFile(getThinLTOOutputFile(Obj.getName()));
|
||||
|
||||
unsigned SymNum = 0;
|
||||
std::vector<Symbol *> Syms = F.getSymbols();
|
||||
|
@ -250,12 +227,6 @@ std::vector<InputFile *> BitcodeCompiler::compile() {
|
|||
Buff.resize(MaxTasks);
|
||||
Files.resize(MaxTasks);
|
||||
|
||||
// If LazyObjFile has not been added to link, emit empty index files
|
||||
if (Config->ThinLTOIndexOnly)
|
||||
for (LazyObjFile *F : LazyObjFiles)
|
||||
if (!F->AddedToLink && isBitcode(F->MB))
|
||||
addLazyObjFile(F);
|
||||
|
||||
// The --thinlto-cache-dir option specifies the path to a directory in which
|
||||
// to cache native object files for ThinLTO incremental builds. If a path was
|
||||
// specified, configure LTO to use it as the cache directory.
|
||||
|
@ -290,21 +261,32 @@ std::vector<InputFile *> BitcodeCompiler::compile() {
|
|||
Ret.push_back(Obj);
|
||||
}
|
||||
|
||||
// ThinLTO with index only option is required to generate only the index
|
||||
// files. After that, we exit from linker and ThinLTO backend runs in a
|
||||
// distributed environment.
|
||||
if (Config->ThinLTOIndexOnly)
|
||||
// If LazyObjFile has not been added to link, emit empty index files.
|
||||
// This is needed because this is what GNU gold plugin does and we have a
|
||||
// distributed build system that depends on that behavior.
|
||||
if (Config->ThinLTOIndexOnly) {
|
||||
for (LazyObjFile *F : LazyObjFiles) {
|
||||
if (F->AddedToLink || !isBitcode(F->MB))
|
||||
continue;
|
||||
|
||||
std::unique_ptr<raw_fd_ostream> OS =
|
||||
openFile(getThinLTOOutputFile(F->getName()));
|
||||
if (!OS)
|
||||
continue;
|
||||
|
||||
ModuleSummaryIndex M(false);
|
||||
M.setSkipModuleByDistributedBackend();
|
||||
WriteIndexToFile(M, *OS);
|
||||
}
|
||||
|
||||
// ThinLTO with index only option is required to generate only the index
|
||||
// files. After that, we exit from linker and ThinLTO backend runs in a
|
||||
// distributed environment.
|
||||
exit(0);
|
||||
}
|
||||
|
||||
for (std::unique_ptr<MemoryBuffer> &File : Files)
|
||||
if (File)
|
||||
Ret.push_back(createObjectFile(*File));
|
||||
|
||||
return Ret;
|
||||
}
|
||||
|
||||
// For lazy object files not added to link, adds empty index files
|
||||
void BitcodeCompiler::addLazyObjFile(LazyObjFile *File) {
|
||||
writeEmptyDistributedBuildOutputs(File->getBuffer().getBufferIdentifier(),
|
||||
/*SkipModule=*/true);
|
||||
}
|
||||
|
|
|
@ -48,7 +48,6 @@ public:
|
|||
|
||||
void add(BitcodeFile &F);
|
||||
std::vector<InputFile *> compile();
|
||||
void addLazyObjFile(LazyObjFile *File);
|
||||
|
||||
private:
|
||||
void init();
|
||||
|
|
|
@ -38,7 +38,7 @@
|
|||
; RUN: touch %t4.o.thinlto.bc
|
||||
; RUN: chmod 400 %t4.o.thinlto.bc
|
||||
; RUN: ld.lld -m elf_x86_64 --plugin-opt=thinlto-index-only -shared %t.o %t4.o -o %t5 2>&1 | FileCheck %s --check-prefix=ERR
|
||||
; ERR: failed to write {{.*}}4.o.thinlto.bc: {{P|p}}ermission denied
|
||||
; ERR: cannot open {{.*}}4.o.thinlto.bc: {{P|p}}ermission denied
|
||||
|
||||
; Ensure lld doesn't generates index files when thinlto-index-only is not enabled
|
||||
; RUN: rm -f %t.o.thinlto.bc
|
||||
|
|
Loading…
Reference in New Issue