From d6e77b7d157be995bbcc6b818c3e4a2fd8aa0ee9 Mon Sep 17 00:00:00 2001 From: Teresa Johnson Date: Mon, 19 Oct 2015 14:30:44 +0000 Subject: [PATCH] llvm-lto support for generating combined function indexes Summary: This patch adds support to llvm-lto that mirrors the support added by r249270 to the gold plugin. This enables better testing of combined index generation for ThinLTO. Added a new test, and this support will be used in the test in D13515. Reviewers: joker.eph Subscribers: llvm-commits Differential Revision: http://reviews.llvm.org/D13847 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@250699 91177308-0d34-0410-b5e6-96231b3b80d8 --- test/tools/llvm-lto/Inputs/thinlto.ll | 4 ++ test/tools/llvm-lto/thinlto.ll | 24 ++++++++++ tools/llvm-lto/CMakeLists.txt | 3 ++ tools/llvm-lto/LLVMBuild.txt | 2 +- tools/llvm-lto/llvm-lto.cpp | 64 +++++++++++++++++++++++++++ 5 files changed, 96 insertions(+), 1 deletion(-) create mode 100644 test/tools/llvm-lto/Inputs/thinlto.ll create mode 100644 test/tools/llvm-lto/thinlto.ll diff --git a/test/tools/llvm-lto/Inputs/thinlto.ll b/test/tools/llvm-lto/Inputs/thinlto.ll new file mode 100644 index 00000000000..4e0840f3691 --- /dev/null +++ b/test/tools/llvm-lto/Inputs/thinlto.ll @@ -0,0 +1,4 @@ +define void @g() { +entry: + ret void +} diff --git a/test/tools/llvm-lto/thinlto.ll b/test/tools/llvm-lto/thinlto.ll new file mode 100644 index 00000000000..29d3881b9df --- /dev/null +++ b/test/tools/llvm-lto/thinlto.ll @@ -0,0 +1,24 @@ +; Test combined function index generation for ThinLTO via llvm-lto. +; RUN: llvm-as -function-summary %s -o %t.o +; RUN: llvm-as -function-summary %p/Inputs/thinlto.ll -o %t2.o +; RUN: llvm-lto -thinlto -o %t3 %t.o %t2.o +; RUN: llvm-bcanalyzer -dump %t3.thinlto.bc | FileCheck %s --check-prefix=COMBINED +; RUN: not test -e %t3 + +; COMBINED: UseDiagnosticHandler("use-diagnostic-handler", cl::init(false), cl::desc("Use a diagnostic handler to test the handler interface")); +static cl::opt ThinLTO( + "thinlto", cl::init(false), + cl::desc("Only write combined global index for ThinLTO backends")); + static cl::list InputFilenames(cl::Positional, cl::OneOrMore, cl::desc("")); @@ -151,6 +158,61 @@ static int listSymbols(StringRef Command, const TargetOptions &Options) { return 0; } +/// Parse the function index out of an IR file and return the function +/// index object if found, or nullptr if not. +static std::unique_ptr getFunctionIndexForFile( + StringRef Path, std::string &Error, LLVMContext &Context) { + std::unique_ptr Buffer; + ErrorOr> BufferOrErr = + MemoryBuffer::getFile(Path); + if (std::error_code EC = BufferOrErr.getError()) { + Error = EC.message(); + return nullptr; + } + Buffer = std::move(BufferOrErr.get()); + ErrorOr> ObjOrErr = + object::FunctionIndexObjectFile::create(Buffer->getMemBufferRef(), + Context); + if (std::error_code EC = ObjOrErr.getError()) { + Error = EC.message(); + return nullptr; + } + return (*ObjOrErr)->takeIndex(); +} + +/// Create a combined index file from the input IR files and write it. +/// +/// This is meant to enable testing of ThinLTO combined index generation, +/// currently available via the gold plugin via -thinlto. +static int createCombinedFunctionIndex(StringRef Command) { + LLVMContext Context; + FunctionInfoIndex CombinedIndex; + uint64_t NextModuleId = 0; + for (auto &Filename : InputFilenames) { + std::string Error; + std::unique_ptr Index = + getFunctionIndexForFile(Filename, Error, Context); + if (!Index) { + errs() << Command << ": error loading file '" << Filename + << "': " << Error << "\n"; + return 1; + } + CombinedIndex.mergeFrom(std::move(Index), ++NextModuleId); + } + std::error_code EC; + assert(!OutputFilename.empty()); + raw_fd_ostream OS(OutputFilename + ".thinlto.bc", EC, + sys::fs::OpenFlags::F_None); + if (EC) { + errs() << Command << ": error opening the file '" << OutputFilename + << ".thinlto.bc': " << EC.message() << "\n"; + return 1; + } + WriteFunctionSummaryToFile(&CombinedIndex, OS); + OS.close(); + return 0; +} + int main(int argc, char **argv) { // Print a stack trace if we signal out. sys::PrintStackTraceOnErrorSignal(); @@ -176,6 +238,8 @@ int main(int argc, char **argv) { if (ListSymbolsOnly) return listSymbols(argv[0], Options); + if (ThinLTO) return createCombinedFunctionIndex(argv[0]); + unsigned BaseArg = 0; LTOCodeGenerator CodeGen; -- 2.34.1