1c1b1729bSDaniel Dunbar //===--- BackendUtil.cpp - LLVM Backend Utilities -------------------------===// 2c1b1729bSDaniel Dunbar // 3c1b1729bSDaniel Dunbar // The LLVM Compiler Infrastructure 4c1b1729bSDaniel Dunbar // 5c1b1729bSDaniel Dunbar // This file is distributed under the University of Illinois Open Source 6c1b1729bSDaniel Dunbar // License. See LICENSE.TXT for details. 7c1b1729bSDaniel Dunbar // 8c1b1729bSDaniel Dunbar //===----------------------------------------------------------------------===// 9c1b1729bSDaniel Dunbar 10c1b1729bSDaniel Dunbar #include "clang/CodeGen/BackendUtil.h" 11c1b1729bSDaniel Dunbar #include "clang/Basic/Diagnostic.h" 12fec0ff84SDan Gohman #include "clang/Basic/LangOptions.h" 133a02247dSChandler Carruth #include "clang/Basic/TargetOptions.h" 148509824cSChandler Carruth #include "clang/Frontend/CodeGenOptions.h" 15c1b1729bSDaniel Dunbar #include "clang/Frontend/FrontendDiagnostic.h" 16ce2c726eSKostya Serebryany #include "clang/Frontend/Utils.h" 1762849c67SSaleem Abdulrasool #include "llvm/ADT/StringSwitch.h" 18418bd1abSChandler Carruth #include "llvm/Analysis/TargetLibraryInfo.h" 19aab5ec07SChandler Carruth #include "llvm/Analysis/TargetTransformInfo.h" 201b3304deSChandler Carruth #include "llvm/Bitcode/BitcodeWriterPass.h" 21c1b1729bSDaniel Dunbar #include "llvm/CodeGen/RegAllocRegistry.h" 22c1b1729bSDaniel Dunbar #include "llvm/CodeGen/SchedulerRegistry.h" 23ffd5551bSChandler Carruth #include "llvm/IR/DataLayout.h" 240a50c496SChandler Carruth #include "llvm/IR/IRPrintingPasses.h" 25ffd5551bSChandler Carruth #include "llvm/IR/Module.h" 26ca88474aSChandler Carruth #include "llvm/IR/Verifier.h" 27eeb486ddSEvan Cheng #include "llvm/MC/SubtargetFeature.h" 283a02247dSChandler Carruth #include "llvm/PassManager.h" 29c1b1729bSDaniel Dunbar #include "llvm/Support/CommandLine.h" 30c1b1729bSDaniel Dunbar #include "llvm/Support/FormattedStream.h" 31c1b1729bSDaniel Dunbar #include "llvm/Support/PrettyStackTrace.h" 32494eb062SEvan Cheng #include "llvm/Support/TargetRegistry.h" 33c1b1729bSDaniel Dunbar #include "llvm/Support/Timer.h" 34c1b1729bSDaniel Dunbar #include "llvm/Support/raw_ostream.h" 35c1b1729bSDaniel Dunbar #include "llvm/Target/TargetMachine.h" 36c1b1729bSDaniel Dunbar #include "llvm/Target/TargetOptions.h" 373de0853bSEric Christopher #include "llvm/Target/TargetSubtargetInfo.h" 3856a7dab0SRafael Espindola #include "llvm/Transforms/IPO.h" 3956a7dab0SRafael Espindola #include "llvm/Transforms/IPO/PassManagerBuilder.h" 403a02247dSChandler Carruth #include "llvm/Transforms/Instrumentation.h" 4190cae773SMichael Gottesman #include "llvm/Transforms/ObjCARC.h" 4256a7dab0SRafael Espindola #include "llvm/Transforms/Scalar.h" 4376a4b95aSSaleem Abdulrasool #include "llvm/Transforms/Utils/SymbolRewriter.h" 44dfca6f97SAhmed Charles #include <memory> 45c1b1729bSDaniel Dunbar using namespace clang; 46c1b1729bSDaniel Dunbar using namespace llvm; 47c1b1729bSDaniel Dunbar 48c1b1729bSDaniel Dunbar namespace { 49c1b1729bSDaniel Dunbar 50c1b1729bSDaniel Dunbar class EmitAssemblyHelper { 519c902b55SDavid Blaikie DiagnosticsEngine &Diags; 52c1b1729bSDaniel Dunbar const CodeGenOptions &CodeGenOpts; 53432add5bSNick Lewycky const clang::TargetOptions &TargetOpts; 54fec0ff84SDan Gohman const LangOptions &LangOpts; 55c1b1729bSDaniel Dunbar Module *TheModule; 56c1b1729bSDaniel Dunbar 57c1b1729bSDaniel Dunbar Timer CodeGenerationTime; 58c1b1729bSDaniel Dunbar 59195fa003SDaniel Dunbar mutable PassManager *CodeGenPasses; 60c1b1729bSDaniel Dunbar mutable PassManager *PerModulePasses; 61c1b1729bSDaniel Dunbar mutable FunctionPassManager *PerFunctionPasses; 62c1b1729bSDaniel Dunbar 63c1b1729bSDaniel Dunbar private: 64d294bdb5SChandler Carruth TargetIRAnalysis getTargetIRAnalysis() const { 65aab5ec07SChandler Carruth if (TM) 66d294bdb5SChandler Carruth return TM->getTargetIRAnalysis(); 67aab5ec07SChandler Carruth 68d294bdb5SChandler Carruth return TargetIRAnalysis(); 69aab5ec07SChandler Carruth } 70aab5ec07SChandler Carruth 71f4e22381SAlp Toker PassManager *getCodeGenPasses() const { 72c1b1729bSDaniel Dunbar if (!CodeGenPasses) { 73195fa003SDaniel Dunbar CodeGenPasses = new PassManager(); 744de669f5SRafael Espindola CodeGenPasses->add(new DataLayoutPass()); 75d294bdb5SChandler Carruth CodeGenPasses->add( 76d294bdb5SChandler Carruth createTargetTransformInfoWrapperPass(getTargetIRAnalysis())); 77c1b1729bSDaniel Dunbar } 78c1b1729bSDaniel Dunbar return CodeGenPasses; 79c1b1729bSDaniel Dunbar } 80c1b1729bSDaniel Dunbar 81f4e22381SAlp Toker PassManager *getPerModulePasses() const { 82c1b1729bSDaniel Dunbar if (!PerModulePasses) { 83c1b1729bSDaniel Dunbar PerModulePasses = new PassManager(); 844de669f5SRafael Espindola PerModulePasses->add(new DataLayoutPass()); 85d294bdb5SChandler Carruth PerModulePasses->add( 86d294bdb5SChandler Carruth createTargetTransformInfoWrapperPass(getTargetIRAnalysis())); 87c1b1729bSDaniel Dunbar } 88c1b1729bSDaniel Dunbar return PerModulePasses; 89c1b1729bSDaniel Dunbar } 90c1b1729bSDaniel Dunbar 91f4e22381SAlp Toker FunctionPassManager *getPerFunctionPasses() const { 92c1b1729bSDaniel Dunbar if (!PerFunctionPasses) { 93c1b1729bSDaniel Dunbar PerFunctionPasses = new FunctionPassManager(TheModule); 944de669f5SRafael Espindola PerFunctionPasses->add(new DataLayoutPass()); 95d294bdb5SChandler Carruth PerFunctionPasses->add( 96d294bdb5SChandler Carruth createTargetTransformInfoWrapperPass(getTargetIRAnalysis())); 97c1b1729bSDaniel Dunbar } 98c1b1729bSDaniel Dunbar return PerFunctionPasses; 99c1b1729bSDaniel Dunbar } 100c1b1729bSDaniel Dunbar 101f4e22381SAlp Toker void CreatePasses(); 102ec57ab37SNadav Rotem 103dc06b2d3SNadav Rotem /// CreateTargetMachine - Generates the TargetMachine. 104dc06b2d3SNadav Rotem /// Returns Null if it is unable to create the target machine. 105dc06b2d3SNadav Rotem /// Some of our clang tests specify triples which are not built 106dc06b2d3SNadav Rotem /// into clang. This is okay because these tests check the generated 107dc06b2d3SNadav Rotem /// IR, and they require DataLayout which depends on the triple. 108dc06b2d3SNadav Rotem /// In this case, we allow this method to fail and not report an error. 109dc06b2d3SNadav Rotem /// When MustCreateTM is used, we print an error if we are unable to load 110dc06b2d3SNadav Rotem /// the requested target. 111dc06b2d3SNadav Rotem TargetMachine *CreateTargetMachine(bool MustCreateTM); 112c1b1729bSDaniel Dunbar 113c1b1729bSDaniel Dunbar /// AddEmitPasses - Add passes necessary to emit assembly or LLVM IR. 114c1b1729bSDaniel Dunbar /// 115c1b1729bSDaniel Dunbar /// \return True on success. 116f4e22381SAlp Toker bool AddEmitPasses(BackendAction Action, formatted_raw_ostream &OS); 117c1b1729bSDaniel Dunbar 118c1b1729bSDaniel Dunbar public: 1199c902b55SDavid Blaikie EmitAssemblyHelper(DiagnosticsEngine &_Diags, 120432add5bSNick Lewycky const CodeGenOptions &CGOpts, 121432add5bSNick Lewycky const clang::TargetOptions &TOpts, 122fec0ff84SDan Gohman const LangOptions &LOpts, 123c1b1729bSDaniel Dunbar Module *M) 124fec0ff84SDan Gohman : Diags(_Diags), CodeGenOpts(CGOpts), TargetOpts(TOpts), LangOpts(LOpts), 125c1b1729bSDaniel Dunbar TheModule(M), CodeGenerationTime("Code Generation Time"), 1268a13c418SCraig Topper CodeGenPasses(nullptr), PerModulePasses(nullptr), 1278a13c418SCraig Topper PerFunctionPasses(nullptr) {} 128c1b1729bSDaniel Dunbar 129c1b1729bSDaniel Dunbar ~EmitAssemblyHelper() { 130c1b1729bSDaniel Dunbar delete CodeGenPasses; 131c1b1729bSDaniel Dunbar delete PerModulePasses; 132c1b1729bSDaniel Dunbar delete PerFunctionPasses; 133f4e22381SAlp Toker if (CodeGenOpts.DisableFree) 134a97eaa1bSDavid Blaikie BuryPointer(std::move(TM)); 135c1b1729bSDaniel Dunbar } 136c1b1729bSDaniel Dunbar 137b8984329SAhmed Charles std::unique_ptr<TargetMachine> TM; 138f4e22381SAlp Toker 139c1b1729bSDaniel Dunbar void EmitAssembly(BackendAction Action, raw_ostream *OS); 140c1b1729bSDaniel Dunbar }; 141c1b1729bSDaniel Dunbar 142c6515b6aSAlexey Samsonov // We need this wrapper to access LangOpts and CGOpts from extension functions 143c6515b6aSAlexey Samsonov // that we add to the PassManagerBuilder. 1440e96becfSAlexey Samsonov class PassManagerBuilderWrapper : public PassManagerBuilder { 1450e96becfSAlexey Samsonov public: 1469ab73626SAlexey Samsonov PassManagerBuilderWrapper(const CodeGenOptions &CGOpts, 1479ab73626SAlexey Samsonov const LangOptions &LangOpts) 1489ab73626SAlexey Samsonov : PassManagerBuilder(), CGOpts(CGOpts), LangOpts(LangOpts) {} 1499ab73626SAlexey Samsonov const CodeGenOptions &getCGOpts() const { return CGOpts; } 1500e96becfSAlexey Samsonov const LangOptions &getLangOpts() const { return LangOpts; } 1510e96becfSAlexey Samsonov private: 1529ab73626SAlexey Samsonov const CodeGenOptions &CGOpts; 1530e96becfSAlexey Samsonov const LangOptions &LangOpts; 1540e96becfSAlexey Samsonov }; 1550e96becfSAlexey Samsonov 156c1b1729bSDaniel Dunbar } 157c1b1729bSDaniel Dunbar 1585932ce25SDan Gohman static void addObjCARCAPElimPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 1595932ce25SDan Gohman if (Builder.OptLevel > 0) 1605932ce25SDan Gohman PM.add(createObjCARCAPElimPass()); 1615932ce25SDan Gohman } 1625932ce25SDan Gohman 163fec0ff84SDan Gohman static void addObjCARCExpandPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 164fec0ff84SDan Gohman if (Builder.OptLevel > 0) 165fec0ff84SDan Gohman PM.add(createObjCARCExpandPass()); 166fec0ff84SDan Gohman } 167fec0ff84SDan Gohman 168fec0ff84SDan Gohman static void addObjCARCOptPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 169fec0ff84SDan Gohman if (Builder.OptLevel > 0) 170fec0ff84SDan Gohman PM.add(createObjCARCOptPass()); 171fec0ff84SDan Gohman } 172fec0ff84SDan Gohman 1735c29705cSDiego Novillo static void addSampleProfileLoaderPass(const PassManagerBuilder &Builder, 1745c29705cSDiego Novillo PassManagerBase &PM) { 1755c29705cSDiego Novillo const PassManagerBuilderWrapper &BuilderWrapper = 1765c29705cSDiego Novillo static_cast<const PassManagerBuilderWrapper &>(Builder); 1775c29705cSDiego Novillo const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 1785c29705cSDiego Novillo PM.add(createSampleProfileLoaderPass(CGOpts.SampleProfileFile)); 1795c29705cSDiego Novillo } 1805c29705cSDiego Novillo 181b56be645SDiego Novillo static void addAddDiscriminatorsPass(const PassManagerBuilder &Builder, 182b56be645SDiego Novillo PassManagerBase &PM) { 183b56be645SDiego Novillo PM.add(createAddDiscriminatorsPass()); 184b56be645SDiego Novillo } 185b56be645SDiego Novillo 186a425589fSNuno Lopes static void addBoundsCheckingPass(const PassManagerBuilder &Builder, 187a425589fSNuno Lopes PassManagerBase &PM) { 188ebc59d5eSJoey Gouly PM.add(createBoundsCheckingPass()); 189a425589fSNuno Lopes } 190a425589fSNuno Lopes 19175b4f9e1SKostya Serebryany static void addSanitizerCoveragePass(const PassManagerBuilder &Builder, 19275b4f9e1SKostya Serebryany PassManagerBase &PM) { 19375b4f9e1SKostya Serebryany const PassManagerBuilderWrapper &BuilderWrapper = 19475b4f9e1SKostya Serebryany static_cast<const PassManagerBuilderWrapper&>(Builder); 19575b4f9e1SKostya Serebryany const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 19675b4f9e1SKostya Serebryany PM.add(createSanitizerCoverageModulePass(CGOpts.SanitizeCoverage)); 19775b4f9e1SKostya Serebryany } 19875b4f9e1SKostya Serebryany 1990e96becfSAlexey Samsonov static void addAddressSanitizerPasses(const PassManagerBuilder &Builder, 2008855ff61SKostya Serebryany PassManagerBase &PM) { 201e595e1adSAlexey Samsonov PM.add(createAddressSanitizerFunctionPass()); 202c94285a1SAlexey Samsonov PM.add(createAddressSanitizerModulePass()); 2038855ff61SKostya Serebryany } 2048855ff61SKostya Serebryany 205aea92e53SEvgeniy Stepanov static void addMemorySanitizerPass(const PassManagerBuilder &Builder, 206aea92e53SEvgeniy Stepanov PassManagerBase &PM) { 207ad8ab3d9SEvgeniy Stepanov const PassManagerBuilderWrapper &BuilderWrapper = 208ad8ab3d9SEvgeniy Stepanov static_cast<const PassManagerBuilderWrapper&>(Builder); 209ad8ab3d9SEvgeniy Stepanov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 2101cf07ea6SAlexey Samsonov PM.add(createMemorySanitizerPass(CGOpts.SanitizeMemoryTrackOrigins)); 21110284671SEvgeniy Stepanov 21210284671SEvgeniy Stepanov // MemorySanitizer inserts complex instrumentation that mostly follows 21310284671SEvgeniy Stepanov // the logic of the original code, but operates on "shadow" values. 21410284671SEvgeniy Stepanov // It can benefit from re-running some general purpose optimization passes. 21510284671SEvgeniy Stepanov if (Builder.OptLevel > 0) { 21610284671SEvgeniy Stepanov PM.add(createEarlyCSEPass()); 21710284671SEvgeniy Stepanov PM.add(createReassociatePass()); 21810284671SEvgeniy Stepanov PM.add(createLICMPass()); 21910284671SEvgeniy Stepanov PM.add(createGVNPass()); 22010284671SEvgeniy Stepanov PM.add(createInstructionCombiningPass()); 22110284671SEvgeniy Stepanov PM.add(createDeadStoreEliminationPass()); 22210284671SEvgeniy Stepanov } 223aea92e53SEvgeniy Stepanov } 224aea92e53SEvgeniy Stepanov 22528a7a119SKostya Serebryany static void addThreadSanitizerPass(const PassManagerBuilder &Builder, 22628a7a119SKostya Serebryany PassManagerBase &PM) { 2271cf07ea6SAlexey Samsonov PM.add(createThreadSanitizerPass()); 22828a7a119SKostya Serebryany } 22928a7a119SKostya Serebryany 230c377275aSPeter Collingbourne static void addDataFlowSanitizerPass(const PassManagerBuilder &Builder, 231c377275aSPeter Collingbourne PassManagerBase &PM) { 232276be3c5SPeter Collingbourne const PassManagerBuilderWrapper &BuilderWrapper = 233276be3c5SPeter Collingbourne static_cast<const PassManagerBuilderWrapper&>(Builder); 234ae5804f3SAlexey Samsonov const LangOptions &LangOpts = BuilderWrapper.getLangOpts(); 235*a511cdd2SAlexey Samsonov PM.add(createDataFlowSanitizerPass(LangOpts.SanitizerBlacklistFiles)); 236c377275aSPeter Collingbourne } 237c377275aSPeter Collingbourne 23857bb7c7cSChandler Carruth static TargetLibraryInfoImpl *createTLII(llvm::Triple &TargetTriple, 239b8a71c5eSRafael Espindola const CodeGenOptions &CodeGenOpts) { 24057bb7c7cSChandler Carruth TargetLibraryInfoImpl *TLII = new TargetLibraryInfoImpl(TargetTriple); 241b8a71c5eSRafael Espindola if (!CodeGenOpts.SimplifyLibCalls) 24257bb7c7cSChandler Carruth TLII->disableAllFunctions(); 24357bb7c7cSChandler Carruth return TLII; 244b8a71c5eSRafael Espindola } 245b8a71c5eSRafael Espindola 24676a4b95aSSaleem Abdulrasool static void addSymbolRewriterPass(const CodeGenOptions &Opts, 24776a4b95aSSaleem Abdulrasool PassManager *MPM) { 24876a4b95aSSaleem Abdulrasool llvm::SymbolRewriter::RewriteDescriptorList DL; 24976a4b95aSSaleem Abdulrasool 25076a4b95aSSaleem Abdulrasool llvm::SymbolRewriter::RewriteMapParser MapParser; 25176a4b95aSSaleem Abdulrasool for (const auto &MapFile : Opts.RewriteMapFiles) 25276a4b95aSSaleem Abdulrasool MapParser.parse(MapFile, &DL); 25376a4b95aSSaleem Abdulrasool 25476a4b95aSSaleem Abdulrasool MPM->add(createRewriteSymbolsPass(DL)); 25576a4b95aSSaleem Abdulrasool } 25676a4b95aSSaleem Abdulrasool 257f4e22381SAlp Toker void EmitAssemblyHelper::CreatePasses() { 258c1b1729bSDaniel Dunbar unsigned OptLevel = CodeGenOpts.OptimizationLevel; 259b0eea8b5SDouglas Gregor CodeGenOptions::InliningMethod Inlining = CodeGenOpts.getInlining(); 260c1b1729bSDaniel Dunbar 261c1b1729bSDaniel Dunbar // Handle disabling of LLVM optimization, where we want to preserve the 262c1b1729bSDaniel Dunbar // internal module before any optimization. 263c1b1729bSDaniel Dunbar if (CodeGenOpts.DisableLLVMOpts) { 264c1b1729bSDaniel Dunbar OptLevel = 0; 265c1b1729bSDaniel Dunbar Inlining = CodeGenOpts.NoInlining; 266c1b1729bSDaniel Dunbar } 267c1b1729bSDaniel Dunbar 2689ab73626SAlexey Samsonov PassManagerBuilderWrapper PMBuilder(CodeGenOpts, LangOpts); 269ecf0ba5bSChris Lattner PMBuilder.OptLevel = OptLevel; 270ecf0ba5bSChris Lattner PMBuilder.SizeLevel = CodeGenOpts.OptimizeSize; 271d3f3e4f0SNick Lewycky PMBuilder.BBVectorize = CodeGenOpts.VectorizeBB; 272d3f3e4f0SNick Lewycky PMBuilder.SLPVectorize = CodeGenOpts.VectorizeSLP; 273d3f3e4f0SNick Lewycky PMBuilder.LoopVectorize = CodeGenOpts.VectorizeLoop; 274d98cec5cSChris Lattner 27585e349fdSDuncan P. N. Exon Smith PMBuilder.DisableTailCalls = CodeGenOpts.DisableTailCalls; 276ecf0ba5bSChris Lattner PMBuilder.DisableUnitAtATime = !CodeGenOpts.UnitAtATime; 277ecf0ba5bSChris Lattner PMBuilder.DisableUnrollLoops = !CodeGenOpts.UnrollLoops; 278f04f237eSNick Lewycky PMBuilder.MergeFunctions = CodeGenOpts.MergeFunctions; 279ce0697f4SHal Finkel PMBuilder.RerollLoops = CodeGenOpts.RerollLoops; 2805c12367bSChris Lattner 281b56be645SDiego Novillo PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 282b56be645SDiego Novillo addAddDiscriminatorsPass); 283b56be645SDiego Novillo 2845c29705cSDiego Novillo if (!CodeGenOpts.SampleProfileFile.empty()) 2855c29705cSDiego Novillo PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 2865c29705cSDiego Novillo addSampleProfileLoaderPass); 2875c29705cSDiego Novillo 288fec0ff84SDan Gohman // In ObjC ARC mode, add the main ARC optimization passes. 289fec0ff84SDan Gohman if (LangOpts.ObjCAutoRefCount) { 290fec0ff84SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 291fec0ff84SDan Gohman addObjCARCExpandPass); 2925932ce25SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_ModuleOptimizerEarly, 2935932ce25SDan Gohman addObjCARCAPElimPass); 294fec0ff84SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_ScalarOptimizerLate, 295fec0ff84SDan Gohman addObjCARCOptPass); 296fec0ff84SDan Gohman } 297fec0ff84SDan Gohman 298edf99a92SAlexey Samsonov if (LangOpts.Sanitize.has(SanitizerKind::LocalBounds)) { 299a425589fSNuno Lopes PMBuilder.addExtension(PassManagerBuilder::EP_ScalarOptimizerLate, 300a425589fSNuno Lopes addBoundsCheckingPass); 301a425589fSNuno Lopes PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 302a425589fSNuno Lopes addBoundsCheckingPass); 303a425589fSNuno Lopes } 304a425589fSNuno Lopes 30575b4f9e1SKostya Serebryany if (CodeGenOpts.SanitizeCoverage) { 30675b4f9e1SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 30775b4f9e1SKostya Serebryany addSanitizerCoveragePass); 30875b4f9e1SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 30975b4f9e1SKostya Serebryany addSanitizerCoveragePass); 31075b4f9e1SKostya Serebryany } 31175b4f9e1SKostya Serebryany 312edf99a92SAlexey Samsonov if (LangOpts.Sanitize.has(SanitizerKind::Address)) { 3137e247f22SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 3140e96becfSAlexey Samsonov addAddressSanitizerPasses); 315d4768576SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 3160e96becfSAlexey Samsonov addAddressSanitizerPasses); 3178855ff61SKostya Serebryany } 3188855ff61SKostya Serebryany 319edf99a92SAlexey Samsonov if (LangOpts.Sanitize.has(SanitizerKind::Memory)) { 320aea92e53SEvgeniy Stepanov PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 321aea92e53SEvgeniy Stepanov addMemorySanitizerPass); 322aea92e53SEvgeniy Stepanov PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 323aea92e53SEvgeniy Stepanov addMemorySanitizerPass); 324aea92e53SEvgeniy Stepanov } 325aea92e53SEvgeniy Stepanov 326edf99a92SAlexey Samsonov if (LangOpts.Sanitize.has(SanitizerKind::Thread)) { 327d18cb502SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 32828a7a119SKostya Serebryany addThreadSanitizerPass); 32928a7a119SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 33028a7a119SKostya Serebryany addThreadSanitizerPass); 33128a7a119SKostya Serebryany } 33228a7a119SKostya Serebryany 333edf99a92SAlexey Samsonov if (LangOpts.Sanitize.has(SanitizerKind::DataFlow)) { 334c377275aSPeter Collingbourne PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 335c377275aSPeter Collingbourne addDataFlowSanitizerPass); 336c377275aSPeter Collingbourne PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 337c377275aSPeter Collingbourne addDataFlowSanitizerPass); 338c377275aSPeter Collingbourne } 339c377275aSPeter Collingbourne 3405c12367bSChris Lattner // Figure out TargetLibraryInfo. 34128b9e8b3SBill Wendling Triple TargetTriple(TheModule->getTargetTriple()); 34257bb7c7cSChandler Carruth PMBuilder.LibraryInfo = createTLII(TargetTriple, CodeGenOpts); 343c1b1729bSDaniel Dunbar 344c1b1729bSDaniel Dunbar switch (Inlining) { 345c1b1729bSDaniel Dunbar case CodeGenOptions::NoInlining: break; 346c1b1729bSDaniel Dunbar case CodeGenOptions::NormalInlining: { 347b5cfa8c2SEli Bendersky PMBuilder.Inliner = 348b5cfa8c2SEli Bendersky createFunctionInliningPass(OptLevel, CodeGenOpts.OptimizeSize); 349c1b1729bSDaniel Dunbar break; 350c1b1729bSDaniel Dunbar } 351c1b1729bSDaniel Dunbar case CodeGenOptions::OnlyAlwaysInlining: 3525c12367bSChris Lattner // Respect always_inline. 353e2c45069SChad Rosier if (OptLevel == 0) 354e2c45069SChad Rosier // Do not insert lifetime intrinsics at -O0. 355e2c45069SChad Rosier PMBuilder.Inliner = createAlwaysInlinerPass(false); 356e2c45069SChad Rosier else 357ecf0ba5bSChris Lattner PMBuilder.Inliner = createAlwaysInlinerPass(); 358c1b1729bSDaniel Dunbar break; 359c1b1729bSDaniel Dunbar } 360c1b1729bSDaniel Dunbar 3615c12367bSChris Lattner // Set up the per-function pass manager. 362f4e22381SAlp Toker FunctionPassManager *FPM = getPerFunctionPasses(); 3635c12367bSChris Lattner if (CodeGenOpts.VerifyModule) 3645c12367bSChris Lattner FPM->add(createVerifierPass()); 3655c12367bSChris Lattner PMBuilder.populateFunctionPassManager(*FPM); 3665c12367bSChris Lattner 3675c12367bSChris Lattner // Set up the per-module pass manager. 368f4e22381SAlp Toker PassManager *MPM = getPerModulePasses(); 36976a4b95aSSaleem Abdulrasool if (!CodeGenOpts.RewriteMapFiles.empty()) 37076a4b95aSSaleem Abdulrasool addSymbolRewriterPass(CodeGenOpts, MPM); 37152eaffeeSDuncan P. N. Exon Smith if (CodeGenOpts.VerifyModule) 37252eaffeeSDuncan P. N. Exon Smith MPM->add(createDebugInfoVerifierPass()); 373d98cec5cSChris Lattner 374c02bbb61SNick Lewycky if (!CodeGenOpts.DisableGCov && 375c02bbb61SNick Lewycky (CodeGenOpts.EmitGcovArcs || CodeGenOpts.EmitGcovNotes)) { 376c8bf8249SNick Lewycky // Not using 'GCOVOptions::getDefault' allows us to avoid exiting if 377c8bf8249SNick Lewycky // LLVM's -default-gcov-version flag is set to something invalid. 378c8bf8249SNick Lewycky GCOVOptions Options; 379c8bf8249SNick Lewycky Options.EmitNotes = CodeGenOpts.EmitGcovNotes; 380c8bf8249SNick Lewycky Options.EmitData = CodeGenOpts.EmitGcovArcs; 381c8bf8249SNick Lewycky memcpy(Options.Version, CodeGenOpts.CoverageVersion, 4); 382c8bf8249SNick Lewycky Options.UseCfgChecksum = CodeGenOpts.CoverageExtraChecksum; 383c8bf8249SNick Lewycky Options.NoRedZone = CodeGenOpts.DisableRedZone; 384c8bf8249SNick Lewycky Options.FunctionNamesInData = 3856f15b290SNick Lewycky !CodeGenOpts.CoverageNoFunctionNamesInData; 386c8bf8249SNick Lewycky MPM->add(createGCOVProfilerPass(Options)); 387b0eea8b5SDouglas Gregor if (CodeGenOpts.getDebugInfo() == CodeGenOptions::NoDebugInfo) 388207bce31SNick Lewycky MPM->add(createStripSymbolsPass(true)); 389207bce31SNick Lewycky } 390207bce31SNick Lewycky 391970ac605SJustin Bogner if (CodeGenOpts.ProfileInstrGenerate) { 392970ac605SJustin Bogner InstrProfOptions Options; 393970ac605SJustin Bogner Options.NoRedZone = CodeGenOpts.DisableRedZone; 394970ac605SJustin Bogner MPM->add(createInstrProfilingPass(Options)); 395970ac605SJustin Bogner } 396970ac605SJustin Bogner 3975c12367bSChris Lattner PMBuilder.populateModulePassManager(*MPM); 398c1b1729bSDaniel Dunbar } 399c1b1729bSDaniel Dunbar 400dc06b2d3SNadav Rotem TargetMachine *EmitAssemblyHelper::CreateTargetMachine(bool MustCreateTM) { 401c1b1729bSDaniel Dunbar // Create the TargetMachine for generating code. 402c1b1729bSDaniel Dunbar std::string Error; 403c1b1729bSDaniel Dunbar std::string Triple = TheModule->getTargetTriple(); 404c1b1729bSDaniel Dunbar const llvm::Target *TheTarget = TargetRegistry::lookupTarget(Triple, Error); 405c1b1729bSDaniel Dunbar if (!TheTarget) { 406dc06b2d3SNadav Rotem if (MustCreateTM) 407ecafbe60SChad Rosier Diags.Report(diag::err_fe_unable_to_create_target) << Error; 4088a13c418SCraig Topper return nullptr; 409c1b1729bSDaniel Dunbar } 410c1b1729bSDaniel Dunbar 41162849c67SSaleem Abdulrasool unsigned CodeModel = 41262849c67SSaleem Abdulrasool llvm::StringSwitch<unsigned>(CodeGenOpts.CodeModel) 41362849c67SSaleem Abdulrasool .Case("small", llvm::CodeModel::Small) 41462849c67SSaleem Abdulrasool .Case("kernel", llvm::CodeModel::Kernel) 41562849c67SSaleem Abdulrasool .Case("medium", llvm::CodeModel::Medium) 41661449c6bSSaleem Abdulrasool .Case("large", llvm::CodeModel::Large) 41762849c67SSaleem Abdulrasool .Case("default", llvm::CodeModel::Default) 41862849c67SSaleem Abdulrasool .Default(~0u); 41962849c67SSaleem Abdulrasool assert(CodeModel != ~0u && "invalid code model!"); 42062849c67SSaleem Abdulrasool llvm::CodeModel::Model CM = static_cast<llvm::CodeModel::Model>(CodeModel); 421c1b1729bSDaniel Dunbar 42209d20eefSDavid Blaikie SmallVector<const char *, 16> BackendArgs; 423c1b1729bSDaniel Dunbar BackendArgs.push_back("clang"); // Fake program name. 424c1b1729bSDaniel Dunbar if (!CodeGenOpts.DebugPass.empty()) { 425c1b1729bSDaniel Dunbar BackendArgs.push_back("-debug-pass"); 426c1b1729bSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.DebugPass.c_str()); 427c1b1729bSDaniel Dunbar } 428c1b1729bSDaniel Dunbar if (!CodeGenOpts.LimitFloatPrecision.empty()) { 429c1b1729bSDaniel Dunbar BackendArgs.push_back("-limit-float-precision"); 430c1b1729bSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.LimitFloatPrecision.c_str()); 431c1b1729bSDaniel Dunbar } 432c1b1729bSDaniel Dunbar if (llvm::TimePassesIsEnabled) 433c1b1729bSDaniel Dunbar BackendArgs.push_back("-time-passes"); 43412100e2cSDaniel Dunbar for (unsigned i = 0, e = CodeGenOpts.BackendOptions.size(); i != e; ++i) 43512100e2cSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.BackendOptions[i].c_str()); 436ba3df1d3SChad Rosier if (CodeGenOpts.NoGlobalMerge) 43725363c00SBob Wilson BackendArgs.push_back("-enable-global-merge=false"); 4388a13c418SCraig Topper BackendArgs.push_back(nullptr); 439c1b1729bSDaniel Dunbar llvm::cl::ParseCommandLineOptions(BackendArgs.size() - 1, 44009d20eefSDavid Blaikie BackendArgs.data()); 441c1b1729bSDaniel Dunbar 442c1b1729bSDaniel Dunbar std::string FeaturesStr; 4436ee521c7SAlexander Kornienko if (!TargetOpts.Features.empty()) { 444c1b1729bSDaniel Dunbar SubtargetFeatures Features; 445c1b1729bSDaniel Dunbar for (std::vector<std::string>::const_iterator 446c1b1729bSDaniel Dunbar it = TargetOpts.Features.begin(), 447c1b1729bSDaniel Dunbar ie = TargetOpts.Features.end(); it != ie; ++it) 448c1b1729bSDaniel Dunbar Features.AddFeature(*it); 449c1b1729bSDaniel Dunbar FeaturesStr = Features.getString(); 450c1b1729bSDaniel Dunbar } 4513f37dd06SEvan Cheng 4523f37dd06SEvan Cheng llvm::Reloc::Model RM = llvm::Reloc::Default; 4533f37dd06SEvan Cheng if (CodeGenOpts.RelocationModel == "static") { 4543f37dd06SEvan Cheng RM = llvm::Reloc::Static; 4553f37dd06SEvan Cheng } else if (CodeGenOpts.RelocationModel == "pic") { 4563f37dd06SEvan Cheng RM = llvm::Reloc::PIC_; 4573f37dd06SEvan Cheng } else { 4583f37dd06SEvan Cheng assert(CodeGenOpts.RelocationModel == "dynamic-no-pic" && 4593f37dd06SEvan Cheng "Invalid PIC model!"); 4603f37dd06SEvan Cheng RM = llvm::Reloc::DynamicNoPIC; 4613f37dd06SEvan Cheng } 4623f37dd06SEvan Cheng 463dd286bceSEvan Cheng CodeGenOpt::Level OptLevel = CodeGenOpt::Default; 464dd286bceSEvan Cheng switch (CodeGenOpts.OptimizationLevel) { 465dd286bceSEvan Cheng default: break; 466dd286bceSEvan Cheng case 0: OptLevel = CodeGenOpt::None; break; 467dd286bceSEvan Cheng case 3: OptLevel = CodeGenOpt::Aggressive; break; 468dd286bceSEvan Cheng } 469dd286bceSEvan Cheng 470432add5bSNick Lewycky llvm::TargetOptions Options; 471432add5bSNick Lewycky 472b140a100SJonathan Roelofs Options.ThreadModel = 473b140a100SJonathan Roelofs llvm::StringSwitch<llvm::ThreadModel::Model>(CodeGenOpts.ThreadModel) 474b140a100SJonathan Roelofs .Case("posix", llvm::ThreadModel::POSIX) 475b140a100SJonathan Roelofs .Case("single", llvm::ThreadModel::Single); 476b140a100SJonathan Roelofs 47733ebd217SRafael Espindola if (CodeGenOpts.DisableIntegratedAS) 47833ebd217SRafael Espindola Options.DisableIntegratedAS = true; 47933ebd217SRafael Espindola 4807e2fd943SDavid Blaikie if (CodeGenOpts.CompressDebugSections) 4817e2fd943SDavid Blaikie Options.CompressDebugSections = true; 4827e2fd943SDavid Blaikie 483432add5bSNick Lewycky // Set frame pointer elimination mode. 484432add5bSNick Lewycky if (!CodeGenOpts.DisableFPElim) { 485432add5bSNick Lewycky Options.NoFramePointerElim = false; 486432add5bSNick Lewycky } else if (CodeGenOpts.OmitLeafFramePointer) { 487432add5bSNick Lewycky Options.NoFramePointerElim = false; 488432add5bSNick Lewycky } else { 489432add5bSNick Lewycky Options.NoFramePointerElim = true; 490432add5bSNick Lewycky } 491432add5bSNick Lewycky 49266aa045fSRafael Espindola if (CodeGenOpts.UseInitArray) 49366aa045fSRafael Espindola Options.UseInitArray = true; 49466aa045fSRafael Espindola 495432add5bSNick Lewycky // Set float ABI type. 496432add5bSNick Lewycky if (CodeGenOpts.FloatABI == "soft" || CodeGenOpts.FloatABI == "softfp") 497432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Soft; 498432add5bSNick Lewycky else if (CodeGenOpts.FloatABI == "hard") 499432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Hard; 500432add5bSNick Lewycky else { 501432add5bSNick Lewycky assert(CodeGenOpts.FloatABI.empty() && "Invalid float abi!"); 502432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Default; 503432add5bSNick Lewycky } 504432add5bSNick Lewycky 505aa53b936SLang Hames // Set FP fusion mode. 50665992f45SLang Hames switch (CodeGenOpts.getFPContractMode()) { 50765992f45SLang Hames case CodeGenOptions::FPC_Off: 508aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Strict; 509aa53b936SLang Hames break; 51065992f45SLang Hames case CodeGenOptions::FPC_On: 511aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Standard; 512aa53b936SLang Hames break; 51365992f45SLang Hames case CodeGenOptions::FPC_Fast: 514aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Fast; 515aa53b936SLang Hames break; 516aa53b936SLang Hames } 517aa53b936SLang Hames 518432add5bSNick Lewycky Options.LessPreciseFPMADOption = CodeGenOpts.LessPreciseFPMAD; 519432add5bSNick Lewycky Options.NoInfsFPMath = CodeGenOpts.NoInfsFPMath; 520432add5bSNick Lewycky Options.NoNaNsFPMath = CodeGenOpts.NoNaNsFPMath; 521432add5bSNick Lewycky Options.NoZerosInBSS = CodeGenOpts.NoZeroInitializedInBSS; 522432add5bSNick Lewycky Options.UnsafeFPMath = CodeGenOpts.UnsafeFPMath; 523432add5bSNick Lewycky Options.UseSoftFloat = CodeGenOpts.SoftFloat; 524f4d3f7a0SNick Lewycky Options.StackAlignmentOverride = CodeGenOpts.StackAlignment; 5251c8c436aSNick Lewycky Options.DisableTailCalls = CodeGenOpts.DisableTailCalls; 52614adb360SBob Wilson Options.TrapFuncName = CodeGenOpts.TrapFuncName; 527097d019cSChandler Carruth Options.PositionIndependentExecutable = LangOpts.PIELevel != 0; 528acca0085SEric Christopher Options.FunctionSections = CodeGenOpts.FunctionSections; 529acca0085SEric Christopher Options.DataSections = CodeGenOpts.DataSections; 530432add5bSNick Lewycky 5317e72a093SEric Christopher Options.MCOptions.MCRelaxAll = CodeGenOpts.RelaxAll; 5327e72a093SEric Christopher Options.MCOptions.MCSaveTempLabels = CodeGenOpts.SaveTempLabels; 533e76eee46SEric Christopher Options.MCOptions.MCUseDwarfDirectory = !CodeGenOpts.NoDwarfDirectoryAsm; 5347e72a093SEric Christopher Options.MCOptions.MCNoExecStack = CodeGenOpts.NoExecStack; 535a43604adSJoerg Sonnenberger Options.MCOptions.MCFatalWarnings = CodeGenOpts.FatalWarnings; 5364015e124SEric Christopher Options.MCOptions.AsmVerbose = CodeGenOpts.AsmVerbose; 5372b8ef99aSEric Christopher Options.MCOptions.ABIName = TargetOpts.ABI; 5387e72a093SEric Christopher 539adc79598SEvan Cheng TargetMachine *TM = TheTarget->createTargetMachine(Triple, TargetOpts.CPU, 540432add5bSNick Lewycky FeaturesStr, Options, 541432add5bSNick Lewycky RM, CM, OptLevel); 542c1b1729bSDaniel Dunbar 543ec57ab37SNadav Rotem return TM; 544ec57ab37SNadav Rotem } 545ec57ab37SNadav Rotem 546ec57ab37SNadav Rotem bool EmitAssemblyHelper::AddEmitPasses(BackendAction Action, 547f4e22381SAlp Toker formatted_raw_ostream &OS) { 548ec57ab37SNadav Rotem 549c1b1729bSDaniel Dunbar // Create the code generator passes. 550f4e22381SAlp Toker PassManager *PM = getCodeGenPasses(); 551c1b1729bSDaniel Dunbar 552b1cfc686SChad Rosier // Add LibraryInfo. 553aa437df9SDaniel Dunbar llvm::Triple TargetTriple(TheModule->getTargetTriple()); 55457bb7c7cSChandler Carruth std::unique_ptr<TargetLibraryInfoImpl> TLII( 55557bb7c7cSChandler Carruth createTLII(TargetTriple, CodeGenOpts)); 55657bb7c7cSChandler Carruth PM->add(new TargetLibraryInfoWrapperPass(*TLII)); 557e9c233b1SNadav Rotem 558c1b1729bSDaniel Dunbar // Normal mode, emit a .s or .o file by running the code generator. Note, 559c1b1729bSDaniel Dunbar // this also adds codegenerator level optimization passes. 560c1b1729bSDaniel Dunbar TargetMachine::CodeGenFileType CGFT = TargetMachine::CGFT_AssemblyFile; 561c1b1729bSDaniel Dunbar if (Action == Backend_EmitObj) 562c1b1729bSDaniel Dunbar CGFT = TargetMachine::CGFT_ObjectFile; 563c1b1729bSDaniel Dunbar else if (Action == Backend_EmitMCNull) 564c1b1729bSDaniel Dunbar CGFT = TargetMachine::CGFT_Null; 565c1b1729bSDaniel Dunbar else 566c1b1729bSDaniel Dunbar assert(Action == Backend_EmitAssembly && "Invalid action!"); 567fec0ff84SDan Gohman 568fec0ff84SDan Gohman // Add ObjC ARC final-cleanup optimizations. This is done as part of the 569fec0ff84SDan Gohman // "codegen" passes so that it isn't run multiple times when there is 570fec0ff84SDan Gohman // inlining happening. 5711170b088SDan Gohman if (LangOpts.ObjCAutoRefCount && 5721170b088SDan Gohman CodeGenOpts.OptimizationLevel > 0) 573fec0ff84SDan Gohman PM->add(createObjCARCContractPass()); 574fec0ff84SDan Gohman 575dd286bceSEvan Cheng if (TM->addPassesToEmitFile(*PM, OS, CGFT, 576c1b1729bSDaniel Dunbar /*DisableVerify=*/!CodeGenOpts.VerifyModule)) { 577c1b1729bSDaniel Dunbar Diags.Report(diag::err_fe_unable_to_interface_with_target); 578c1b1729bSDaniel Dunbar return false; 579c1b1729bSDaniel Dunbar } 580c1b1729bSDaniel Dunbar 581c1b1729bSDaniel Dunbar return true; 582c1b1729bSDaniel Dunbar } 583c1b1729bSDaniel Dunbar 584c1b1729bSDaniel Dunbar void EmitAssemblyHelper::EmitAssembly(BackendAction Action, raw_ostream *OS) { 5858a13c418SCraig Topper TimeRegion Region(llvm::TimePassesIsEnabled ? &CodeGenerationTime : nullptr); 586c1b1729bSDaniel Dunbar llvm::formatted_raw_ostream FormattedOS; 587c1b1729bSDaniel Dunbar 588dc06b2d3SNadav Rotem bool UsesCodeGen = (Action != Backend_EmitNothing && 589dc06b2d3SNadav Rotem Action != Backend_EmitBC && 590dc06b2d3SNadav Rotem Action != Backend_EmitLL); 591f4e22381SAlp Toker if (!TM) 592f4e22381SAlp Toker TM.reset(CreateTargetMachine(UsesCodeGen)); 593f4e22381SAlp Toker 594ecafbe60SChad Rosier if (UsesCodeGen && !TM) return; 595f4e22381SAlp Toker CreatePasses(); 596ec57ab37SNadav Rotem 597c1b1729bSDaniel Dunbar switch (Action) { 598c1b1729bSDaniel Dunbar case Backend_EmitNothing: 599c1b1729bSDaniel Dunbar break; 600c1b1729bSDaniel Dunbar 601c1b1729bSDaniel Dunbar case Backend_EmitBC: 602f4e22381SAlp Toker getPerModulePasses()->add(createBitcodeWriterPass(*OS)); 603c1b1729bSDaniel Dunbar break; 604c1b1729bSDaniel Dunbar 605c1b1729bSDaniel Dunbar case Backend_EmitLL: 606c1b1729bSDaniel Dunbar FormattedOS.setStream(*OS, formatted_raw_ostream::PRESERVE_STREAM); 607809403ffSChandler Carruth getPerModulePasses()->add(createPrintModulePass(FormattedOS)); 608c1b1729bSDaniel Dunbar break; 609c1b1729bSDaniel Dunbar 610c1b1729bSDaniel Dunbar default: 611c1b1729bSDaniel Dunbar FormattedOS.setStream(*OS, formatted_raw_ostream::PRESERVE_STREAM); 612f4e22381SAlp Toker if (!AddEmitPasses(Action, FormattedOS)) 613c1b1729bSDaniel Dunbar return; 614c1b1729bSDaniel Dunbar } 615c1b1729bSDaniel Dunbar 61615e36e8eSAndrew Trick // Before executing passes, print the final values of the LLVM options. 61715e36e8eSAndrew Trick cl::PrintOptionValues(); 61815e36e8eSAndrew Trick 619c1b1729bSDaniel Dunbar // Run passes. For now we do all passes at once, but eventually we 620c1b1729bSDaniel Dunbar // would like to have the option of streaming code generation. 621c1b1729bSDaniel Dunbar 622c1b1729bSDaniel Dunbar if (PerFunctionPasses) { 623c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Per-function optimization"); 624c1b1729bSDaniel Dunbar 625c1b1729bSDaniel Dunbar PerFunctionPasses->doInitialization(); 626c1b1729bSDaniel Dunbar for (Module::iterator I = TheModule->begin(), 627c1b1729bSDaniel Dunbar E = TheModule->end(); I != E; ++I) 628c1b1729bSDaniel Dunbar if (!I->isDeclaration()) 629c1b1729bSDaniel Dunbar PerFunctionPasses->run(*I); 630c1b1729bSDaniel Dunbar PerFunctionPasses->doFinalization(); 631c1b1729bSDaniel Dunbar } 632c1b1729bSDaniel Dunbar 633c1b1729bSDaniel Dunbar if (PerModulePasses) { 634c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Per-module optimization passes"); 635c1b1729bSDaniel Dunbar PerModulePasses->run(*TheModule); 636c1b1729bSDaniel Dunbar } 637c1b1729bSDaniel Dunbar 638c1b1729bSDaniel Dunbar if (CodeGenPasses) { 639c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Code generation"); 640195fa003SDaniel Dunbar CodeGenPasses->run(*TheModule); 641c1b1729bSDaniel Dunbar } 642c1b1729bSDaniel Dunbar } 643c1b1729bSDaniel Dunbar 6449c902b55SDavid Blaikie void clang::EmitBackendOutput(DiagnosticsEngine &Diags, 6459c902b55SDavid Blaikie const CodeGenOptions &CGOpts, 646432add5bSNick Lewycky const clang::TargetOptions &TOpts, 647e83b9060SAlp Toker const LangOptions &LOpts, StringRef TDesc, 648e83b9060SAlp Toker Module *M, BackendAction Action, 649e83b9060SAlp Toker raw_ostream *OS) { 650fec0ff84SDan Gohman EmitAssemblyHelper AsmHelper(Diags, CGOpts, TOpts, LOpts, M); 651c1b1729bSDaniel Dunbar 652c1b1729bSDaniel Dunbar AsmHelper.EmitAssembly(Action, OS); 653e83b9060SAlp Toker 654e83b9060SAlp Toker // If an optional clang TargetInfo description string was passed in, use it to 655e83b9060SAlp Toker // verify the LLVM TargetMachine's DataLayout. 656e83b9060SAlp Toker if (AsmHelper.TM && !TDesc.empty()) { 657611dfed9SEric Christopher std::string DLDesc = 658611dfed9SEric Christopher AsmHelper.TM->getDataLayout()->getStringRepresentation(); 659e83b9060SAlp Toker if (DLDesc != TDesc) { 660e83b9060SAlp Toker unsigned DiagID = Diags.getCustomDiagID( 661e83b9060SAlp Toker DiagnosticsEngine::Error, "backend data layout '%0' does not match " 662e83b9060SAlp Toker "expected target description '%1'"); 663e83b9060SAlp Toker Diags.Report(DiagID) << DLDesc << TDesc; 664e83b9060SAlp Toker } 665e83b9060SAlp Toker } 666c1b1729bSDaniel Dunbar } 667