Index: lld/ELF/Config.h =================================================================== --- lld/ELF/Config.h +++ lld/ELF/Config.h @@ -96,6 +96,7 @@ llvm::StringRef Sysroot; llvm::StringRef ThinLTOCacheDir; llvm::StringRef ThinLTOIndexOnlyArg; + std::pair ThinLTOObjectSuffixReplace; std::pair ThinLTOPrefixReplace; std::string Rpath; std::vector VersionDefinitions; Index: lld/ELF/Driver.cpp =================================================================== --- lld/ELF/Driver.cpp +++ lld/ELF/Driver.cpp @@ -813,6 +813,14 @@ if (Config->ThinLTOPrefixReplace.second.empty()) error("thinlto-prefix-replace expects 'old;new' format, but got " + S.substr(23)); + } else if (S.startswith("thinlto-object-suffix-replace=")) { + std::tie(Config->ThinLTOObjectSuffixReplace.first, + Config->ThinLTOObjectSuffixReplace.second) = + S.substr(30).split(';'); + if (Config->ThinLTOObjectSuffixReplace.second.empty()) + error( + "thinlto-object-suffix-replace expects 'old;new' format, but got " + + S.substr(30)); } else if (!S.startswith("/") && !S.startswith("-fresolution=") && !S.startswith("-pass-through=") && !S.startswith("thinlto")) { parseClangOption(S, Arg->getSpelling()); Index: lld/ELF/InputFiles.cpp =================================================================== --- lld/ELF/InputFiles.cpp +++ lld/ELF/InputFiles.cpp @@ -1025,9 +1025,15 @@ // this causes a collision which result in only one of the objects being // taken into consideration at LTO time (which very likely causes undefined // symbols later in the link stage). + StringRef Path = MB.getBufferIdentifier(); + std::string UpdatedPath = Path.str(); + if (Config->ThinLTOIndexOnly) { + Path.consume_back(Config->ThinLTOObjectSuffixReplace.first.str()); + UpdatedPath = Path.str() + Config->ThinLTOObjectSuffixReplace.second.str(); + } MemoryBufferRef MBRef( MB.getBuffer(), - Saver.save(ArchiveName + MB.getBufferIdentifier() + + Saver.save(ArchiveName + UpdatedPath + (ArchiveName.empty() ? "" : utostr(OffsetInArchive)))); Obj = CHECK(lto::InputFile::create(MBRef), this); Index: lld/ELF/LTO.cpp =================================================================== --- lld/ELF/LTO.cpp +++ lld/ELF/LTO.cpp @@ -266,8 +266,12 @@ if (F->AddedToLink || !isBitcode(F->MB)) continue; - std::string Path = getThinLTOOutputFile(F->getName()); - std::unique_ptr OS = openFile(Path + ".thinlto.bc"); + StringRef Path = F->getName(); + Path.consume_back(Config->ThinLTOObjectSuffixReplace.first); + std::string UpdatedPath = getThinLTOOutputFile(Path) + + Config->ThinLTOObjectSuffixReplace.second.str(); + std::unique_ptr OS = + openFile(UpdatedPath + ".thinlto.bc"); if (!OS) continue; @@ -276,7 +280,7 @@ WriteIndexToFile(M, *OS); if (Config->ThinLTOEmitImportsFiles) - openFile(Path + ".imports"); + openFile(UpdatedPath + ".imports"); } if (!Config->LTOObjPath.empty()) Index: lld/test/ELF/lto/thinlto-object-suffix-replace.ll =================================================================== --- /dev/null +++ lld/test/ELF/lto/thinlto-object-suffix-replace.ll @@ -0,0 +1,40 @@ +; Test to make sure the thinlto-object-suffix-replace option is handled +; correctly. + +; Generate bitcode file with summary, as well as a minimized bitcode without +; the debug metadata for the thin link. +; RUN: opt -thinlto-bc %s -thin-link-bitcode-file=%t1.thinlink.bc -o %t1.o + +; First perform the thin link on the normal bitcode file, and save the +; resulting index. +; RUN: ld.lld -m elf_x86_64 --plugin-opt=thinlto-index-only -shared %t1.o -o %t3 +; RUN: cp %t1.o.thinlto.bc %t1.o.thinlto.bc.orig + +; Next perform the thin link on the minimized bitcode file, and compare dump +; of the resulting index to the above dump to ensure they are identical. +; RUN: rm -f %t1.o.thinlto.bc +; Make sure it isn't inadvertently using the regular bitcode file. +; RUN: rm -f %t1.o +; RUN: ld.lld -m elf_x86_64 --plugin-opt=thinlto-index-only \ +; RUN: --plugin-opt=thinlto-object-suffix-replace=".thinlink.bc;.o" \ +; RUN: -shared %t1.thinlink.bc -o %t3 +; RUN: diff %t1.o.thinlto.bc.orig %t1.o.thinlto.bc + +; Ensure that lld generates error if object suffix replace option does not have 'old;new' format +; RUN: rm -f %t1.o.thinlto.bc +; RUN: not ld.lld -m elf_x86_64 --plugin-opt=thinlto-index-only \ +; RUN: --plugin-opt=thinlto-object-suffix-replace="abc:def" -shared %t1.thinlink.bc -o %t3 +; ERR: thinlto-object-suffix-replace expects 'old;new' format, but got abc:def + +target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128" +target triple = "x86_64-unknown-linux-gnu" + +define void @f() { +entry: + ret void +} + +!llvm.dbg.cu = !{} + +!1 = !{i32 2, !"Debug Info Version", i32 3} +!llvm.module.flags = !{!1}