From 5b01c014963f0b4f79aeda48eebad3cc587dba13 Mon Sep 17 00:00:00 2001 From: Tobias Edler von Koch Date: Thu, 19 Nov 2015 23:59:24 +0000 Subject: [PATCH] [LTO] Add option to emit assembly from LTOCodeGenerator This adds a new API, LTOCodeGenerator::setFileType, to choose the output file format for LTO CodeGen. A corresponding change to use this new API from llvm-lto and a test case is coming in a separate commit. Differential Revision: http://reviews.llvm.org/D14554 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@253622 91177308-0d34-0410-b5e6-96231b3b80d8 --- include/llvm/CodeGen/ParallelCG.h | 8 +++++--- include/llvm/LTO/LTOCodeGenerator.h | 26 ++++++++++++++++---------- lib/CodeGen/ParallelCG.cpp | 15 ++++++++------- lib/LTO/LTOCodeGenerator.cpp | 10 +++++++--- 4 files changed, 36 insertions(+), 23 deletions(-) diff --git a/include/llvm/CodeGen/ParallelCG.h b/include/llvm/CodeGen/ParallelCG.h index 73ac46125c9..fa7002fa21f 100644 --- a/include/llvm/CodeGen/ParallelCG.h +++ b/include/llvm/CodeGen/ParallelCG.h @@ -16,6 +16,7 @@ #include "llvm/ADT/ArrayRef.h" #include "llvm/Support/CodeGen.h" +#include "llvm/Target/TargetMachine.h" namespace llvm { @@ -24,8 +25,8 @@ class TargetOptions; class raw_pwrite_stream; /// Split M into OSs.size() partitions, and generate code for each. Writes -/// OSs.size() object files to the output streams in OSs. The resulting object -/// files if linked together are intended to be equivalent to the single object +/// OSs.size() output files to the output streams in OSs. The resulting output +/// files if linked together are intended to be equivalent to the single output /// file that would have been code generated from M. /// /// \returns M if OSs.size() == 1, otherwise returns std::unique_ptr(). @@ -34,7 +35,8 @@ splitCodeGen(std::unique_ptr M, ArrayRef OSs, StringRef CPU, StringRef Features, const TargetOptions &Options, Reloc::Model RM = Reloc::Default, CodeModel::Model CM = CodeModel::Default, - CodeGenOpt::Level OL = CodeGenOpt::Default); + CodeGenOpt::Level OL = CodeGenOpt::Default, + TargetMachine::CodeGenFileType FT = TargetMachine::CGFT_ObjectFile); } // namespace llvm diff --git a/include/llvm/LTO/LTOCodeGenerator.h b/include/llvm/LTO/LTOCodeGenerator.h index 68ef3d5f22c..0d3c79bf5e8 100644 --- a/include/llvm/LTO/LTOCodeGenerator.h +++ b/include/llvm/LTO/LTOCodeGenerator.h @@ -40,6 +40,7 @@ #include "llvm/ADT/SmallPtrSet.h" #include "llvm/ADT/StringMap.h" #include "llvm/Linker/Linker.h" +#include "llvm/Target/TargetMachine.h" #include "llvm/Target/TargetOptions.h" #include #include @@ -74,6 +75,10 @@ struct LTOCodeGenerator { void setTargetOptions(TargetOptions Options); void setDebugInfo(lto_debug_model); void setCodePICModel(Reloc::Model Model) { RelocModel = Model; } + + /// Set the file type to be emitted (assembly or object code). + /// The default is TargetMachine::CGFT_ObjectFile. + void setFileType(TargetMachine::CodeGenFileType FT) { FileType = FT; } void setCpu(const char *MCpu) { this->MCpu = MCpu; } void setAttr(const char *MAttr) { this->MAttr = MAttr; } @@ -103,21 +108,21 @@ struct LTOCodeGenerator { /// true on success. bool writeMergedModules(const char *Path); - /// Compile the merged module into a *single* object file; the path to object + /// Compile the merged module into a *single* output file; the path to output /// file is returned to the caller via argument "name". Return true on /// success. /// - /// \note It is up to the linker to remove the intermediate object file. Do + /// \note It is up to the linker to remove the intermediate output file. Do /// not try to remove the object file in LTOCodeGenerator's destructor as we - /// don't who (LTOCodeGenerator or the obj file) will last longer. + /// don't who (LTOCodeGenerator or the output file) will last longer. bool compile_to_file(const char **Name, bool DisableVerify, bool DisableInline, bool DisableGVNLoadPRE, bool DisableVectorization); /// As with compile_to_file(), this function compiles the merged module into - /// single object file. Instead of returning the object-file-path to the - /// caller (linker), it brings the object to a buffer, and return the buffer - /// to the caller. This function should delete intermediate object file once + /// single output file. Instead of returning the output file path to the + /// caller (linker), it brings the output to a buffer, and returns the buffer + /// to the caller. This function should delete the intermediate file once /// its content is brought to memory. Return NULL if the compilation was not /// successful. std::unique_ptr compile(bool DisableVerify, bool DisableInline, @@ -128,15 +133,15 @@ struct LTOCodeGenerator { bool optimize(bool DisableVerify, bool DisableInline, bool DisableGVNLoadPRE, bool DisableVectorization); - /// Compiles the merged optimized module into a single object file. It brings - /// the object to a buffer, and returns the buffer to the caller. Return NULL + /// Compiles the merged optimized module into a single output file. It brings + /// the output to a buffer, and returns the buffer to the caller. Return NULL /// if the compilation was not successful. std::unique_ptr compileOptimized(); - /// Compile the merged optimized module into out.size() object files each + /// Compile the merged optimized module into out.size() output files each /// representing a linkable partition of the module. If out contains more /// than one element, code generation is done in parallel with out.size() - /// threads. Object files will be written to members of out. Returns true on + /// threads. Output files will be written to members of out. Returns true on /// success. bool compileOptimized(ArrayRef Out); @@ -185,6 +190,7 @@ private: void *DiagContext = nullptr; bool ShouldInternalize = true; bool ShouldEmbedUselists = false; + TargetMachine::CodeGenFileType FileType = TargetMachine::CGFT_ObjectFile; }; } #endif diff --git a/lib/CodeGen/ParallelCG.cpp b/lib/CodeGen/ParallelCG.cpp index 748d3883ea1..e73ba029604 100644 --- a/lib/CodeGen/ParallelCG.cpp +++ b/lib/CodeGen/ParallelCG.cpp @@ -28,13 +28,13 @@ using namespace llvm; static void codegen(Module *M, llvm::raw_pwrite_stream &OS, const Target *TheTarget, StringRef CPU, StringRef Features, const TargetOptions &Options, Reloc::Model RM, - CodeModel::Model CM, CodeGenOpt::Level OL) { + CodeModel::Model CM, CodeGenOpt::Level OL, + TargetMachine::CodeGenFileType FileType) { std::unique_ptr TM(TheTarget->createTargetMachine( M->getTargetTriple(), CPU, Features, Options, RM, CM, OL)); legacy::PassManager CodeGenPasses; - if (TM->addPassesToEmitFile(CodeGenPasses, OS, - TargetMachine::CGFT_ObjectFile)) + if (TM->addPassesToEmitFile(CodeGenPasses, OS, FileType)) report_fatal_error("Failed to setup codegen"); CodeGenPasses.run(*M); } @@ -43,7 +43,8 @@ std::unique_ptr llvm::splitCodeGen(std::unique_ptr M, ArrayRef OSs, StringRef CPU, StringRef Features, const TargetOptions &Options, - Reloc::Model RM, CodeModel::Model CM, CodeGenOpt::Level OL) { + Reloc::Model RM, CodeModel::Model CM, CodeGenOpt::Level OL, + TargetMachine::CodeGenFileType FileType) { StringRef TripleStr = M->getTargetTriple(); std::string ErrMsg; const Target *TheTarget = TargetRegistry::lookupTarget(TripleStr, ErrMsg); @@ -52,7 +53,7 @@ llvm::splitCodeGen(std::unique_ptr M, if (OSs.size() == 1) { codegen(M.get(), *OSs[0], TheTarget, CPU, Features, Options, RM, CM, - OL); + OL, FileType); return M; } @@ -69,7 +70,7 @@ llvm::splitCodeGen(std::unique_ptr M, llvm::raw_pwrite_stream *ThreadOS = OSs[Threads.size()]; Threads.emplace_back( - [TheTarget, CPU, Features, Options, RM, CM, OL, + [TheTarget, CPU, Features, Options, RM, CM, OL, FileType, ThreadOS](const SmallVector &BC) { LLVMContext Ctx; ErrorOr> MOrErr = @@ -81,7 +82,7 @@ llvm::splitCodeGen(std::unique_ptr M, std::unique_ptr MPartInCtx = std::move(MOrErr.get()); codegen(MPartInCtx.get(), *ThreadOS, TheTarget, CPU, Features, - Options, RM, CM, OL); + Options, RM, CM, OL, FileType); }, // Pass BC using std::move to ensure that it get moved rather than // copied into the thread's context. diff --git a/lib/LTO/LTOCodeGenerator.cpp b/lib/LTO/LTOCodeGenerator.cpp index 1604769aba8..931bcf0d23f 100644 --- a/lib/LTO/LTOCodeGenerator.cpp +++ b/lib/LTO/LTOCodeGenerator.cpp @@ -206,11 +206,15 @@ bool LTOCodeGenerator::writeMergedModules(const char *Path) { } bool LTOCodeGenerator::compileOptimizedToFile(const char **Name) { - // make unique temp .o file to put generated object file + // make unique temp output file to put generated code SmallString<128> Filename; int FD; + + const char *Extension = + (FileType == TargetMachine::CGFT_AssemblyFile ? "s" : "o"); + std::error_code EC = - sys::fs::createTemporaryFile("lto-llvm", "o", FD, Filename); + sys::fs::createTemporaryFile("lto-llvm", Extension, FD, Filename); if (EC) { emitError(EC.message()); return false; @@ -514,7 +518,7 @@ bool LTOCodeGenerator::compileOptimized(ArrayRef Out) { // MergedModule. MergedModule = splitCodeGen(std::move(MergedModule), Out, MCpu, FeatureStr, Options, - RelocModel, CodeModel::Default, CGOptLevel); + RelocModel, CodeModel::Default, CGOptLevel, FileType); return true; } -- 2.34.1