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" 1756a7dab0SRafael Espindola #include "llvm/Analysis/Verifier.h" 18c1b1729bSDaniel Dunbar #include "llvm/Bitcode/ReaderWriter.h" 19c1b1729bSDaniel Dunbar #include "llvm/CodeGen/RegAllocRegistry.h" 20c1b1729bSDaniel Dunbar #include "llvm/CodeGen/SchedulerRegistry.h" 21ffd5551bSChandler Carruth #include "llvm/IR/DataLayout.h" 220a50c496SChandler Carruth #include "llvm/IR/IRPrintingPasses.h" 23ffd5551bSChandler Carruth #include "llvm/IR/Module.h" 24eeb486ddSEvan Cheng #include "llvm/MC/SubtargetFeature.h" 253a02247dSChandler Carruth #include "llvm/PassManager.h" 26c1b1729bSDaniel Dunbar #include "llvm/Support/CommandLine.h" 27c1b1729bSDaniel Dunbar #include "llvm/Support/FormattedStream.h" 28c1b1729bSDaniel Dunbar #include "llvm/Support/PrettyStackTrace.h" 29494eb062SEvan Cheng #include "llvm/Support/TargetRegistry.h" 30c1b1729bSDaniel Dunbar #include "llvm/Support/Timer.h" 31c1b1729bSDaniel Dunbar #include "llvm/Support/raw_ostream.h" 3256a7dab0SRafael Espindola #include "llvm/Target/TargetLibraryInfo.h" 33c1b1729bSDaniel Dunbar #include "llvm/Target/TargetMachine.h" 34c1b1729bSDaniel Dunbar #include "llvm/Target/TargetOptions.h" 3556a7dab0SRafael Espindola #include "llvm/Transforms/IPO.h" 3656a7dab0SRafael Espindola #include "llvm/Transforms/IPO/PassManagerBuilder.h" 373a02247dSChandler Carruth #include "llvm/Transforms/Instrumentation.h" 3890cae773SMichael Gottesman #include "llvm/Transforms/ObjCARC.h" 3956a7dab0SRafael Espindola #include "llvm/Transforms/Scalar.h" 40c1b1729bSDaniel Dunbar using namespace clang; 41c1b1729bSDaniel Dunbar using namespace llvm; 42c1b1729bSDaniel Dunbar 43c1b1729bSDaniel Dunbar namespace { 44c1b1729bSDaniel Dunbar 45c1b1729bSDaniel Dunbar class EmitAssemblyHelper { 469c902b55SDavid Blaikie DiagnosticsEngine &Diags; 47c1b1729bSDaniel Dunbar const CodeGenOptions &CodeGenOpts; 48432add5bSNick Lewycky const clang::TargetOptions &TargetOpts; 49fec0ff84SDan Gohman const LangOptions &LangOpts; 50c1b1729bSDaniel Dunbar Module *TheModule; 51c1b1729bSDaniel Dunbar 52c1b1729bSDaniel Dunbar Timer CodeGenerationTime; 53c1b1729bSDaniel Dunbar 54195fa003SDaniel Dunbar mutable PassManager *CodeGenPasses; 55c1b1729bSDaniel Dunbar mutable PassManager *PerModulePasses; 56c1b1729bSDaniel Dunbar mutable FunctionPassManager *PerFunctionPasses; 57c1b1729bSDaniel Dunbar 58c1b1729bSDaniel Dunbar private: 59f4e22381SAlp Toker PassManager *getCodeGenPasses() const { 60c1b1729bSDaniel Dunbar if (!CodeGenPasses) { 61195fa003SDaniel Dunbar CodeGenPasses = new PassManager(); 62dd31ca10SMicah Villmow CodeGenPasses->add(new DataLayout(TheModule)); 63ed0f133bSChandler Carruth if (TM) 64ed0f133bSChandler Carruth TM->addAnalysisPasses(*CodeGenPasses); 65c1b1729bSDaniel Dunbar } 66c1b1729bSDaniel Dunbar return CodeGenPasses; 67c1b1729bSDaniel Dunbar } 68c1b1729bSDaniel Dunbar 69f4e22381SAlp Toker PassManager *getPerModulePasses() const { 70c1b1729bSDaniel Dunbar if (!PerModulePasses) { 71c1b1729bSDaniel Dunbar PerModulePasses = new PassManager(); 72dd31ca10SMicah Villmow PerModulePasses->add(new DataLayout(TheModule)); 73ed0f133bSChandler Carruth if (TM) 74ed0f133bSChandler Carruth TM->addAnalysisPasses(*PerModulePasses); 75c1b1729bSDaniel Dunbar } 76c1b1729bSDaniel Dunbar return PerModulePasses; 77c1b1729bSDaniel Dunbar } 78c1b1729bSDaniel Dunbar 79f4e22381SAlp Toker FunctionPassManager *getPerFunctionPasses() const { 80c1b1729bSDaniel Dunbar if (!PerFunctionPasses) { 81c1b1729bSDaniel Dunbar PerFunctionPasses = new FunctionPassManager(TheModule); 82dd31ca10SMicah Villmow PerFunctionPasses->add(new DataLayout(TheModule)); 83ed0f133bSChandler Carruth if (TM) 84ed0f133bSChandler Carruth TM->addAnalysisPasses(*PerFunctionPasses); 85c1b1729bSDaniel Dunbar } 86c1b1729bSDaniel Dunbar return PerFunctionPasses; 87c1b1729bSDaniel Dunbar } 88c1b1729bSDaniel Dunbar 89f4e22381SAlp Toker void CreatePasses(); 90ec57ab37SNadav Rotem 91dc06b2d3SNadav Rotem /// CreateTargetMachine - Generates the TargetMachine. 92dc06b2d3SNadav Rotem /// Returns Null if it is unable to create the target machine. 93dc06b2d3SNadav Rotem /// Some of our clang tests specify triples which are not built 94dc06b2d3SNadav Rotem /// into clang. This is okay because these tests check the generated 95dc06b2d3SNadav Rotem /// IR, and they require DataLayout which depends on the triple. 96dc06b2d3SNadav Rotem /// In this case, we allow this method to fail and not report an error. 97dc06b2d3SNadav Rotem /// When MustCreateTM is used, we print an error if we are unable to load 98dc06b2d3SNadav Rotem /// the requested target. 99dc06b2d3SNadav Rotem TargetMachine *CreateTargetMachine(bool MustCreateTM); 100c1b1729bSDaniel Dunbar 101c1b1729bSDaniel Dunbar /// AddEmitPasses - Add passes necessary to emit assembly or LLVM IR. 102c1b1729bSDaniel Dunbar /// 103c1b1729bSDaniel Dunbar /// \return True on success. 104f4e22381SAlp Toker bool AddEmitPasses(BackendAction Action, formatted_raw_ostream &OS); 105c1b1729bSDaniel Dunbar 106c1b1729bSDaniel Dunbar public: 1079c902b55SDavid Blaikie EmitAssemblyHelper(DiagnosticsEngine &_Diags, 108432add5bSNick Lewycky const CodeGenOptions &CGOpts, 109432add5bSNick Lewycky const clang::TargetOptions &TOpts, 110fec0ff84SDan Gohman const LangOptions &LOpts, 111c1b1729bSDaniel Dunbar Module *M) 112fec0ff84SDan Gohman : Diags(_Diags), CodeGenOpts(CGOpts), TargetOpts(TOpts), LangOpts(LOpts), 113c1b1729bSDaniel Dunbar TheModule(M), CodeGenerationTime("Code Generation Time"), 114c1b1729bSDaniel Dunbar CodeGenPasses(0), PerModulePasses(0), PerFunctionPasses(0) {} 115c1b1729bSDaniel Dunbar 116c1b1729bSDaniel Dunbar ~EmitAssemblyHelper() { 117c1b1729bSDaniel Dunbar delete CodeGenPasses; 118c1b1729bSDaniel Dunbar delete PerModulePasses; 119c1b1729bSDaniel Dunbar delete PerFunctionPasses; 120f4e22381SAlp Toker if (CodeGenOpts.DisableFree) 121ce2c726eSKostya Serebryany BuryPointer(TM.take()); 122c1b1729bSDaniel Dunbar } 123c1b1729bSDaniel Dunbar 124f4e22381SAlp Toker llvm::OwningPtr<TargetMachine> TM; 125f4e22381SAlp Toker 126c1b1729bSDaniel Dunbar void EmitAssembly(BackendAction Action, raw_ostream *OS); 127c1b1729bSDaniel Dunbar }; 128c1b1729bSDaniel Dunbar 129c6515b6aSAlexey Samsonov // We need this wrapper to access LangOpts and CGOpts from extension functions 130c6515b6aSAlexey Samsonov // that we add to the PassManagerBuilder. 1310e96becfSAlexey Samsonov class PassManagerBuilderWrapper : public PassManagerBuilder { 1320e96becfSAlexey Samsonov public: 1339ab73626SAlexey Samsonov PassManagerBuilderWrapper(const CodeGenOptions &CGOpts, 1349ab73626SAlexey Samsonov const LangOptions &LangOpts) 1359ab73626SAlexey Samsonov : PassManagerBuilder(), CGOpts(CGOpts), LangOpts(LangOpts) {} 1369ab73626SAlexey Samsonov const CodeGenOptions &getCGOpts() const { return CGOpts; } 1370e96becfSAlexey Samsonov const LangOptions &getLangOpts() const { return LangOpts; } 1380e96becfSAlexey Samsonov private: 1399ab73626SAlexey Samsonov const CodeGenOptions &CGOpts; 1400e96becfSAlexey Samsonov const LangOptions &LangOpts; 1410e96becfSAlexey Samsonov }; 1420e96becfSAlexey Samsonov 143c1b1729bSDaniel Dunbar } 144c1b1729bSDaniel Dunbar 1455932ce25SDan Gohman static void addObjCARCAPElimPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 1465932ce25SDan Gohman if (Builder.OptLevel > 0) 1475932ce25SDan Gohman PM.add(createObjCARCAPElimPass()); 1485932ce25SDan Gohman } 1495932ce25SDan Gohman 150fec0ff84SDan Gohman static void addObjCARCExpandPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 151fec0ff84SDan Gohman if (Builder.OptLevel > 0) 152fec0ff84SDan Gohman PM.add(createObjCARCExpandPass()); 153fec0ff84SDan Gohman } 154fec0ff84SDan Gohman 155fec0ff84SDan Gohman static void addObjCARCOptPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 156fec0ff84SDan Gohman if (Builder.OptLevel > 0) 157fec0ff84SDan Gohman PM.add(createObjCARCOptPass()); 158fec0ff84SDan Gohman } 159fec0ff84SDan Gohman 1605c29705cSDiego Novillo static void addSampleProfileLoaderPass(const PassManagerBuilder &Builder, 1615c29705cSDiego Novillo PassManagerBase &PM) { 1625c29705cSDiego Novillo const PassManagerBuilderWrapper &BuilderWrapper = 1635c29705cSDiego Novillo static_cast<const PassManagerBuilderWrapper &>(Builder); 1645c29705cSDiego Novillo const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 1655c29705cSDiego Novillo PM.add(createSampleProfileLoaderPass(CGOpts.SampleProfileFile)); 1665c29705cSDiego Novillo } 1675c29705cSDiego Novillo 168a425589fSNuno Lopes static void addBoundsCheckingPass(const PassManagerBuilder &Builder, 169a425589fSNuno Lopes PassManagerBase &PM) { 170ebc59d5eSJoey Gouly PM.add(createBoundsCheckingPass()); 171a425589fSNuno Lopes } 172a425589fSNuno Lopes 1730e96becfSAlexey Samsonov static void addAddressSanitizerPasses(const PassManagerBuilder &Builder, 1748855ff61SKostya Serebryany PassManagerBase &PM) { 1750e96becfSAlexey Samsonov const PassManagerBuilderWrapper &BuilderWrapper = 1760e96becfSAlexey Samsonov static_cast<const PassManagerBuilderWrapper&>(Builder); 1779ab73626SAlexey Samsonov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 1780e96becfSAlexey Samsonov const LangOptions &LangOpts = BuilderWrapper.getLangOpts(); 17929524a98SAlexey Samsonov PM.add(createAddressSanitizerFunctionPass( 18029524a98SAlexey Samsonov LangOpts.Sanitize.InitOrder, 181f54319c8SWill Dietz LangOpts.Sanitize.UseAfterReturn, 182f54319c8SWill Dietz LangOpts.Sanitize.UseAfterScope, 18329524a98SAlexey Samsonov CGOpts.SanitizerBlacklistFile, 18429524a98SAlexey Samsonov CGOpts.SanitizeAddressZeroBaseShadow)); 18529524a98SAlexey Samsonov PM.add(createAddressSanitizerModulePass( 18629524a98SAlexey Samsonov LangOpts.Sanitize.InitOrder, 18729524a98SAlexey Samsonov CGOpts.SanitizerBlacklistFile, 18829524a98SAlexey Samsonov CGOpts.SanitizeAddressZeroBaseShadow)); 1898855ff61SKostya Serebryany } 1908855ff61SKostya Serebryany 191aea92e53SEvgeniy Stepanov static void addMemorySanitizerPass(const PassManagerBuilder &Builder, 192aea92e53SEvgeniy Stepanov PassManagerBase &PM) { 193ad8ab3d9SEvgeniy Stepanov const PassManagerBuilderWrapper &BuilderWrapper = 194ad8ab3d9SEvgeniy Stepanov static_cast<const PassManagerBuilderWrapper&>(Builder); 195ad8ab3d9SEvgeniy Stepanov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 19629524a98SAlexey Samsonov PM.add(createMemorySanitizerPass(CGOpts.SanitizeMemoryTrackOrigins, 197c6515b6aSAlexey Samsonov CGOpts.SanitizerBlacklistFile)); 19810284671SEvgeniy Stepanov 19910284671SEvgeniy Stepanov // MemorySanitizer inserts complex instrumentation that mostly follows 20010284671SEvgeniy Stepanov // the logic of the original code, but operates on "shadow" values. 20110284671SEvgeniy Stepanov // It can benefit from re-running some general purpose optimization passes. 20210284671SEvgeniy Stepanov if (Builder.OptLevel > 0) { 20310284671SEvgeniy Stepanov PM.add(createEarlyCSEPass()); 20410284671SEvgeniy Stepanov PM.add(createReassociatePass()); 20510284671SEvgeniy Stepanov PM.add(createLICMPass()); 20610284671SEvgeniy Stepanov PM.add(createGVNPass()); 20710284671SEvgeniy Stepanov PM.add(createInstructionCombiningPass()); 20810284671SEvgeniy Stepanov PM.add(createDeadStoreEliminationPass()); 20910284671SEvgeniy Stepanov } 210aea92e53SEvgeniy Stepanov } 211aea92e53SEvgeniy Stepanov 21228a7a119SKostya Serebryany static void addThreadSanitizerPass(const PassManagerBuilder &Builder, 21328a7a119SKostya Serebryany PassManagerBase &PM) { 214c6515b6aSAlexey Samsonov const PassManagerBuilderWrapper &BuilderWrapper = 215c6515b6aSAlexey Samsonov static_cast<const PassManagerBuilderWrapper&>(Builder); 216c6515b6aSAlexey Samsonov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 217c6515b6aSAlexey Samsonov PM.add(createThreadSanitizerPass(CGOpts.SanitizerBlacklistFile)); 21828a7a119SKostya Serebryany } 21928a7a119SKostya Serebryany 220c377275aSPeter Collingbourne static void addDataFlowSanitizerPass(const PassManagerBuilder &Builder, 221c377275aSPeter Collingbourne PassManagerBase &PM) { 222276be3c5SPeter Collingbourne const PassManagerBuilderWrapper &BuilderWrapper = 223276be3c5SPeter Collingbourne static_cast<const PassManagerBuilderWrapper&>(Builder); 224276be3c5SPeter Collingbourne const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 225276be3c5SPeter Collingbourne PM.add(createDataFlowSanitizerPass(CGOpts.SanitizerBlacklistFile)); 226c377275aSPeter Collingbourne } 227c377275aSPeter Collingbourne 228f4e22381SAlp Toker void EmitAssemblyHelper::CreatePasses() { 229c1b1729bSDaniel Dunbar unsigned OptLevel = CodeGenOpts.OptimizationLevel; 230b0eea8b5SDouglas Gregor CodeGenOptions::InliningMethod Inlining = CodeGenOpts.getInlining(); 231c1b1729bSDaniel Dunbar 232c1b1729bSDaniel Dunbar // Handle disabling of LLVM optimization, where we want to preserve the 233c1b1729bSDaniel Dunbar // internal module before any optimization. 234c1b1729bSDaniel Dunbar if (CodeGenOpts.DisableLLVMOpts) { 235c1b1729bSDaniel Dunbar OptLevel = 0; 236c1b1729bSDaniel Dunbar Inlining = CodeGenOpts.NoInlining; 237c1b1729bSDaniel Dunbar } 238c1b1729bSDaniel Dunbar 2399ab73626SAlexey Samsonov PassManagerBuilderWrapper PMBuilder(CodeGenOpts, LangOpts); 240ecf0ba5bSChris Lattner PMBuilder.OptLevel = OptLevel; 241ecf0ba5bSChris Lattner PMBuilder.SizeLevel = CodeGenOpts.OptimizeSize; 242d3f3e4f0SNick Lewycky PMBuilder.BBVectorize = CodeGenOpts.VectorizeBB; 243d3f3e4f0SNick Lewycky PMBuilder.SLPVectorize = CodeGenOpts.VectorizeSLP; 244d3f3e4f0SNick Lewycky PMBuilder.LoopVectorize = CodeGenOpts.VectorizeLoop; 245d98cec5cSChris Lattner 246ecf0ba5bSChris Lattner PMBuilder.DisableUnitAtATime = !CodeGenOpts.UnitAtATime; 247ecf0ba5bSChris Lattner PMBuilder.DisableUnrollLoops = !CodeGenOpts.UnrollLoops; 248ce0697f4SHal Finkel PMBuilder.RerollLoops = CodeGenOpts.RerollLoops; 2495c12367bSChris Lattner 2505c29705cSDiego Novillo if (!CodeGenOpts.SampleProfileFile.empty()) 2515c29705cSDiego Novillo PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 2525c29705cSDiego Novillo addSampleProfileLoaderPass); 2535c29705cSDiego Novillo 254fec0ff84SDan Gohman // In ObjC ARC mode, add the main ARC optimization passes. 255fec0ff84SDan Gohman if (LangOpts.ObjCAutoRefCount) { 256fec0ff84SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 257fec0ff84SDan Gohman addObjCARCExpandPass); 2585932ce25SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_ModuleOptimizerEarly, 2595932ce25SDan Gohman addObjCARCAPElimPass); 260fec0ff84SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_ScalarOptimizerLate, 261fec0ff84SDan Gohman addObjCARCOptPass); 262fec0ff84SDan Gohman } 263fec0ff84SDan Gohman 2646b53e224SRichard Smith if (LangOpts.Sanitize.LocalBounds) { 265a425589fSNuno Lopes PMBuilder.addExtension(PassManagerBuilder::EP_ScalarOptimizerLate, 266a425589fSNuno Lopes addBoundsCheckingPass); 267a425589fSNuno Lopes PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 268a425589fSNuno Lopes addBoundsCheckingPass); 269a425589fSNuno Lopes } 270a425589fSNuno Lopes 271f54319c8SWill Dietz if (LangOpts.Sanitize.Address) { 2727e247f22SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 2730e96becfSAlexey Samsonov addAddressSanitizerPasses); 274d4768576SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 2750e96becfSAlexey Samsonov addAddressSanitizerPasses); 2768855ff61SKostya Serebryany } 2778855ff61SKostya Serebryany 278f54319c8SWill Dietz if (LangOpts.Sanitize.Memory) { 279aea92e53SEvgeniy Stepanov PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 280aea92e53SEvgeniy Stepanov addMemorySanitizerPass); 281aea92e53SEvgeniy Stepanov PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 282aea92e53SEvgeniy Stepanov addMemorySanitizerPass); 283aea92e53SEvgeniy Stepanov } 284aea92e53SEvgeniy Stepanov 285f54319c8SWill Dietz if (LangOpts.Sanitize.Thread) { 286d18cb502SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 28728a7a119SKostya Serebryany addThreadSanitizerPass); 28828a7a119SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 28928a7a119SKostya Serebryany addThreadSanitizerPass); 29028a7a119SKostya Serebryany } 29128a7a119SKostya Serebryany 292c377275aSPeter Collingbourne if (LangOpts.Sanitize.DataFlow) { 293c377275aSPeter Collingbourne PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 294c377275aSPeter Collingbourne addDataFlowSanitizerPass); 295c377275aSPeter Collingbourne PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 296c377275aSPeter Collingbourne addDataFlowSanitizerPass); 297c377275aSPeter Collingbourne } 298c377275aSPeter Collingbourne 2995c12367bSChris Lattner // Figure out TargetLibraryInfo. 30028b9e8b3SBill Wendling Triple TargetTriple(TheModule->getTargetTriple()); 301ecf0ba5bSChris Lattner PMBuilder.LibraryInfo = new TargetLibraryInfo(TargetTriple); 302fa222dfbSChris Lattner if (!CodeGenOpts.SimplifyLibCalls) 303ecf0ba5bSChris Lattner PMBuilder.LibraryInfo->disableAllFunctions(); 304c1b1729bSDaniel Dunbar 305c1b1729bSDaniel Dunbar switch (Inlining) { 306c1b1729bSDaniel Dunbar case CodeGenOptions::NoInlining: break; 307c1b1729bSDaniel Dunbar case CodeGenOptions::NormalInlining: { 308c1b1729bSDaniel Dunbar // FIXME: Derive these constants in a principled fashion. 309c1b1729bSDaniel Dunbar unsigned Threshold = 225; 3109a5f84faSBob Wilson if (CodeGenOpts.OptimizeSize == 1) // -Os 311c1b1729bSDaniel Dunbar Threshold = 75; 3129a5f84faSBob Wilson else if (CodeGenOpts.OptimizeSize == 2) // -Oz 3139a5f84faSBob Wilson Threshold = 25; 314c1b1729bSDaniel Dunbar else if (OptLevel > 2) 315c1b1729bSDaniel Dunbar Threshold = 275; 316ecf0ba5bSChris Lattner PMBuilder.Inliner = createFunctionInliningPass(Threshold); 317c1b1729bSDaniel Dunbar break; 318c1b1729bSDaniel Dunbar } 319c1b1729bSDaniel Dunbar case CodeGenOptions::OnlyAlwaysInlining: 3205c12367bSChris Lattner // Respect always_inline. 321e2c45069SChad Rosier if (OptLevel == 0) 322e2c45069SChad Rosier // Do not insert lifetime intrinsics at -O0. 323e2c45069SChad Rosier PMBuilder.Inliner = createAlwaysInlinerPass(false); 324e2c45069SChad Rosier else 325ecf0ba5bSChris Lattner PMBuilder.Inliner = createAlwaysInlinerPass(); 326c1b1729bSDaniel Dunbar break; 327c1b1729bSDaniel Dunbar } 328c1b1729bSDaniel Dunbar 3295c12367bSChris Lattner // Set up the per-function pass manager. 330f4e22381SAlp Toker FunctionPassManager *FPM = getPerFunctionPasses(); 3315c12367bSChris Lattner if (CodeGenOpts.VerifyModule) 3325c12367bSChris Lattner FPM->add(createVerifierPass()); 3335c12367bSChris Lattner PMBuilder.populateFunctionPassManager(*FPM); 3345c12367bSChris Lattner 3355c12367bSChris Lattner // Set up the per-module pass manager. 336f4e22381SAlp Toker PassManager *MPM = getPerModulePasses(); 337d98cec5cSChris Lattner 338c02bbb61SNick Lewycky if (!CodeGenOpts.DisableGCov && 339c02bbb61SNick Lewycky (CodeGenOpts.EmitGcovArcs || CodeGenOpts.EmitGcovNotes)) { 340c8bf8249SNick Lewycky // Not using 'GCOVOptions::getDefault' allows us to avoid exiting if 341c8bf8249SNick Lewycky // LLVM's -default-gcov-version flag is set to something invalid. 342c8bf8249SNick Lewycky GCOVOptions Options; 343c8bf8249SNick Lewycky Options.EmitNotes = CodeGenOpts.EmitGcovNotes; 344c8bf8249SNick Lewycky Options.EmitData = CodeGenOpts.EmitGcovArcs; 345c8bf8249SNick Lewycky memcpy(Options.Version, CodeGenOpts.CoverageVersion, 4); 346c8bf8249SNick Lewycky Options.UseCfgChecksum = CodeGenOpts.CoverageExtraChecksum; 347c8bf8249SNick Lewycky Options.NoRedZone = CodeGenOpts.DisableRedZone; 348c8bf8249SNick Lewycky Options.FunctionNamesInData = 3496f15b290SNick Lewycky !CodeGenOpts.CoverageNoFunctionNamesInData; 350c8bf8249SNick Lewycky MPM->add(createGCOVProfilerPass(Options)); 351b0eea8b5SDouglas Gregor if (CodeGenOpts.getDebugInfo() == CodeGenOptions::NoDebugInfo) 352207bce31SNick Lewycky MPM->add(createStripSymbolsPass(true)); 353207bce31SNick Lewycky } 354207bce31SNick Lewycky 3555c12367bSChris Lattner PMBuilder.populateModulePassManager(*MPM); 356c1b1729bSDaniel Dunbar } 357c1b1729bSDaniel Dunbar 358dc06b2d3SNadav Rotem TargetMachine *EmitAssemblyHelper::CreateTargetMachine(bool MustCreateTM) { 359c1b1729bSDaniel Dunbar // Create the TargetMachine for generating code. 360c1b1729bSDaniel Dunbar std::string Error; 361c1b1729bSDaniel Dunbar std::string Triple = TheModule->getTargetTriple(); 362c1b1729bSDaniel Dunbar const llvm::Target *TheTarget = TargetRegistry::lookupTarget(Triple, Error); 363c1b1729bSDaniel Dunbar if (!TheTarget) { 364dc06b2d3SNadav Rotem if (MustCreateTM) 365ecafbe60SChad Rosier Diags.Report(diag::err_fe_unable_to_create_target) << Error; 366ec57ab37SNadav Rotem return 0; 367c1b1729bSDaniel Dunbar } 368c1b1729bSDaniel Dunbar 369c1b1729bSDaniel Dunbar // FIXME: Expose these capabilities via actual APIs!!!! Aside from just 370c1b1729bSDaniel Dunbar // being gross, this is also totally broken if we ever care about 371c1b1729bSDaniel Dunbar // concurrency. 372bb7ac52eSDaniel Dunbar 373c1b1729bSDaniel Dunbar TargetMachine::setAsmVerbosityDefault(CodeGenOpts.AsmVerbose); 374c1b1729bSDaniel Dunbar 375c1b1729bSDaniel Dunbar TargetMachine::setFunctionSections(CodeGenOpts.FunctionSections); 376c1b1729bSDaniel Dunbar TargetMachine::setDataSections (CodeGenOpts.DataSections); 377c1b1729bSDaniel Dunbar 378c1b1729bSDaniel Dunbar // FIXME: Parse this earlier. 3790d3d7770SBenjamin Kramer llvm::CodeModel::Model CM; 380c1b1729bSDaniel Dunbar if (CodeGenOpts.CodeModel == "small") { 3810d3d7770SBenjamin Kramer CM = llvm::CodeModel::Small; 382c1b1729bSDaniel Dunbar } else if (CodeGenOpts.CodeModel == "kernel") { 3830d3d7770SBenjamin Kramer CM = llvm::CodeModel::Kernel; 384c1b1729bSDaniel Dunbar } else if (CodeGenOpts.CodeModel == "medium") { 3850d3d7770SBenjamin Kramer CM = llvm::CodeModel::Medium; 386c1b1729bSDaniel Dunbar } else if (CodeGenOpts.CodeModel == "large") { 3870d3d7770SBenjamin Kramer CM = llvm::CodeModel::Large; 388c1b1729bSDaniel Dunbar } else { 389c1b1729bSDaniel Dunbar assert(CodeGenOpts.CodeModel.empty() && "Invalid code model!"); 3900d3d7770SBenjamin Kramer CM = llvm::CodeModel::Default; 391c1b1729bSDaniel Dunbar } 392c1b1729bSDaniel Dunbar 39309d20eefSDavid Blaikie SmallVector<const char *, 16> BackendArgs; 394c1b1729bSDaniel Dunbar BackendArgs.push_back("clang"); // Fake program name. 395c1b1729bSDaniel Dunbar if (!CodeGenOpts.DebugPass.empty()) { 396c1b1729bSDaniel Dunbar BackendArgs.push_back("-debug-pass"); 397c1b1729bSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.DebugPass.c_str()); 398c1b1729bSDaniel Dunbar } 399c1b1729bSDaniel Dunbar if (!CodeGenOpts.LimitFloatPrecision.empty()) { 400c1b1729bSDaniel Dunbar BackendArgs.push_back("-limit-float-precision"); 401c1b1729bSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.LimitFloatPrecision.c_str()); 402c1b1729bSDaniel Dunbar } 403c1b1729bSDaniel Dunbar if (llvm::TimePassesIsEnabled) 404c1b1729bSDaniel Dunbar BackendArgs.push_back("-time-passes"); 40512100e2cSDaniel Dunbar for (unsigned i = 0, e = CodeGenOpts.BackendOptions.size(); i != e; ++i) 40612100e2cSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.BackendOptions[i].c_str()); 407ba3df1d3SChad Rosier if (CodeGenOpts.NoGlobalMerge) 408ba3df1d3SChad Rosier BackendArgs.push_back("-global-merge=false"); 409c1b1729bSDaniel Dunbar BackendArgs.push_back(0); 410c1b1729bSDaniel Dunbar llvm::cl::ParseCommandLineOptions(BackendArgs.size() - 1, 41109d20eefSDavid Blaikie BackendArgs.data()); 412c1b1729bSDaniel Dunbar 413c1b1729bSDaniel Dunbar std::string FeaturesStr; 414adc79598SEvan Cheng if (TargetOpts.Features.size()) { 415c1b1729bSDaniel Dunbar SubtargetFeatures Features; 416c1b1729bSDaniel Dunbar for (std::vector<std::string>::const_iterator 417c1b1729bSDaniel Dunbar it = TargetOpts.Features.begin(), 418c1b1729bSDaniel Dunbar ie = TargetOpts.Features.end(); it != ie; ++it) 419c1b1729bSDaniel Dunbar Features.AddFeature(*it); 420c1b1729bSDaniel Dunbar FeaturesStr = Features.getString(); 421c1b1729bSDaniel Dunbar } 4223f37dd06SEvan Cheng 4233f37dd06SEvan Cheng llvm::Reloc::Model RM = llvm::Reloc::Default; 4243f37dd06SEvan Cheng if (CodeGenOpts.RelocationModel == "static") { 4253f37dd06SEvan Cheng RM = llvm::Reloc::Static; 4263f37dd06SEvan Cheng } else if (CodeGenOpts.RelocationModel == "pic") { 4273f37dd06SEvan Cheng RM = llvm::Reloc::PIC_; 4283f37dd06SEvan Cheng } else { 4293f37dd06SEvan Cheng assert(CodeGenOpts.RelocationModel == "dynamic-no-pic" && 4303f37dd06SEvan Cheng "Invalid PIC model!"); 4313f37dd06SEvan Cheng RM = llvm::Reloc::DynamicNoPIC; 4323f37dd06SEvan Cheng } 4333f37dd06SEvan Cheng 434dd286bceSEvan Cheng CodeGenOpt::Level OptLevel = CodeGenOpt::Default; 435dd286bceSEvan Cheng switch (CodeGenOpts.OptimizationLevel) { 436dd286bceSEvan Cheng default: break; 437dd286bceSEvan Cheng case 0: OptLevel = CodeGenOpt::None; break; 438dd286bceSEvan Cheng case 3: OptLevel = CodeGenOpt::Aggressive; break; 439dd286bceSEvan Cheng } 440dd286bceSEvan Cheng 441432add5bSNick Lewycky llvm::TargetOptions Options; 442432add5bSNick Lewycky 443432add5bSNick Lewycky // Set frame pointer elimination mode. 444432add5bSNick Lewycky if (!CodeGenOpts.DisableFPElim) { 445432add5bSNick Lewycky Options.NoFramePointerElim = false; 446432add5bSNick Lewycky } else if (CodeGenOpts.OmitLeafFramePointer) { 447432add5bSNick Lewycky Options.NoFramePointerElim = false; 448432add5bSNick Lewycky } else { 449432add5bSNick Lewycky Options.NoFramePointerElim = true; 450432add5bSNick Lewycky } 451432add5bSNick Lewycky 45266aa045fSRafael Espindola if (CodeGenOpts.UseInitArray) 45366aa045fSRafael Espindola Options.UseInitArray = true; 45466aa045fSRafael Espindola 455432add5bSNick Lewycky // Set float ABI type. 456432add5bSNick Lewycky if (CodeGenOpts.FloatABI == "soft" || CodeGenOpts.FloatABI == "softfp") 457432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Soft; 458432add5bSNick Lewycky else if (CodeGenOpts.FloatABI == "hard") 459432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Hard; 460432add5bSNick Lewycky else { 461432add5bSNick Lewycky assert(CodeGenOpts.FloatABI.empty() && "Invalid float abi!"); 462432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Default; 463432add5bSNick Lewycky } 464432add5bSNick Lewycky 465aa53b936SLang Hames // Set FP fusion mode. 46665992f45SLang Hames switch (CodeGenOpts.getFPContractMode()) { 46765992f45SLang Hames case CodeGenOptions::FPC_Off: 468aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Strict; 469aa53b936SLang Hames break; 47065992f45SLang Hames case CodeGenOptions::FPC_On: 471aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Standard; 472aa53b936SLang Hames break; 47365992f45SLang Hames case CodeGenOptions::FPC_Fast: 474aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Fast; 475aa53b936SLang Hames break; 476aa53b936SLang Hames } 477aa53b936SLang Hames 478432add5bSNick Lewycky Options.LessPreciseFPMADOption = CodeGenOpts.LessPreciseFPMAD; 479432add5bSNick Lewycky Options.NoInfsFPMath = CodeGenOpts.NoInfsFPMath; 480432add5bSNick Lewycky Options.NoNaNsFPMath = CodeGenOpts.NoNaNsFPMath; 481432add5bSNick Lewycky Options.NoZerosInBSS = CodeGenOpts.NoZeroInitializedInBSS; 482432add5bSNick Lewycky Options.UnsafeFPMath = CodeGenOpts.UnsafeFPMath; 483432add5bSNick Lewycky Options.UseSoftFloat = CodeGenOpts.SoftFloat; 484f4d3f7a0SNick Lewycky Options.StackAlignmentOverride = CodeGenOpts.StackAlignment; 4851c8c436aSNick Lewycky Options.DisableTailCalls = CodeGenOpts.DisableTailCalls; 48614adb360SBob Wilson Options.TrapFuncName = CodeGenOpts.TrapFuncName; 487097d019cSChandler Carruth Options.PositionIndependentExecutable = LangOpts.PIELevel != 0; 488006208cfSEric Christopher Options.EnableSegmentedStacks = CodeGenOpts.EnableSegmentedStacks; 489432add5bSNick Lewycky 490adc79598SEvan Cheng TargetMachine *TM = TheTarget->createTargetMachine(Triple, TargetOpts.CPU, 491432add5bSNick Lewycky FeaturesStr, Options, 492432add5bSNick Lewycky RM, CM, OptLevel); 493c1b1729bSDaniel Dunbar 494c1b1729bSDaniel Dunbar if (CodeGenOpts.RelaxAll) 495c1b1729bSDaniel Dunbar TM->setMCRelaxAll(true); 49667919b2aSDaniel Dunbar if (CodeGenOpts.SaveTempLabels) 49767919b2aSDaniel Dunbar TM->setMCSaveTempLabels(true); 498e264187cSRafael Espindola if (CodeGenOpts.NoDwarf2CFIAsm) 499e264187cSRafael Espindola TM->setMCUseCFI(false); 500dd424a5bSNick Lewycky if (!CodeGenOpts.NoDwarfDirectoryAsm) 501dd424a5bSNick Lewycky TM->setMCUseDwarfDirectory(true); 502ca6b90d8SNick Lewycky if (CodeGenOpts.NoExecStack) 503ca6b90d8SNick Lewycky TM->setMCNoExecStack(true); 504c1b1729bSDaniel Dunbar 505ec57ab37SNadav Rotem return TM; 506ec57ab37SNadav Rotem } 507ec57ab37SNadav Rotem 508ec57ab37SNadav Rotem bool EmitAssemblyHelper::AddEmitPasses(BackendAction Action, 509f4e22381SAlp Toker formatted_raw_ostream &OS) { 510ec57ab37SNadav Rotem 511c1b1729bSDaniel Dunbar // Create the code generator passes. 512f4e22381SAlp Toker PassManager *PM = getCodeGenPasses(); 513c1b1729bSDaniel Dunbar 514b1cfc686SChad Rosier // Add LibraryInfo. 515aa437df9SDaniel Dunbar llvm::Triple TargetTriple(TheModule->getTargetTriple()); 516aa437df9SDaniel Dunbar TargetLibraryInfo *TLI = new TargetLibraryInfo(TargetTriple); 517b1cfc686SChad Rosier if (!CodeGenOpts.SimplifyLibCalls) 518b1cfc686SChad Rosier TLI->disableAllFunctions(); 519b1cfc686SChad Rosier PM->add(TLI); 520e9c233b1SNadav Rotem 521ed0f133bSChandler Carruth // Add Target specific analysis passes. 522ed0f133bSChandler Carruth TM->addAnalysisPasses(*PM); 523b1cfc686SChad Rosier 524c1b1729bSDaniel Dunbar // Normal mode, emit a .s or .o file by running the code generator. Note, 525c1b1729bSDaniel Dunbar // this also adds codegenerator level optimization passes. 526c1b1729bSDaniel Dunbar TargetMachine::CodeGenFileType CGFT = TargetMachine::CGFT_AssemblyFile; 527c1b1729bSDaniel Dunbar if (Action == Backend_EmitObj) 528c1b1729bSDaniel Dunbar CGFT = TargetMachine::CGFT_ObjectFile; 529c1b1729bSDaniel Dunbar else if (Action == Backend_EmitMCNull) 530c1b1729bSDaniel Dunbar CGFT = TargetMachine::CGFT_Null; 531c1b1729bSDaniel Dunbar else 532c1b1729bSDaniel Dunbar assert(Action == Backend_EmitAssembly && "Invalid action!"); 533fec0ff84SDan Gohman 534fec0ff84SDan Gohman // Add ObjC ARC final-cleanup optimizations. This is done as part of the 535fec0ff84SDan Gohman // "codegen" passes so that it isn't run multiple times when there is 536fec0ff84SDan Gohman // inlining happening. 5371170b088SDan Gohman if (LangOpts.ObjCAutoRefCount && 5381170b088SDan Gohman CodeGenOpts.OptimizationLevel > 0) 539fec0ff84SDan Gohman PM->add(createObjCARCContractPass()); 540fec0ff84SDan Gohman 541dd286bceSEvan Cheng if (TM->addPassesToEmitFile(*PM, OS, CGFT, 542c1b1729bSDaniel Dunbar /*DisableVerify=*/!CodeGenOpts.VerifyModule)) { 543c1b1729bSDaniel Dunbar Diags.Report(diag::err_fe_unable_to_interface_with_target); 544c1b1729bSDaniel Dunbar return false; 545c1b1729bSDaniel Dunbar } 546c1b1729bSDaniel Dunbar 547c1b1729bSDaniel Dunbar return true; 548c1b1729bSDaniel Dunbar } 549c1b1729bSDaniel Dunbar 550c1b1729bSDaniel Dunbar void EmitAssemblyHelper::EmitAssembly(BackendAction Action, raw_ostream *OS) { 551c1b1729bSDaniel Dunbar TimeRegion Region(llvm::TimePassesIsEnabled ? &CodeGenerationTime : 0); 552c1b1729bSDaniel Dunbar llvm::formatted_raw_ostream FormattedOS; 553c1b1729bSDaniel Dunbar 554dc06b2d3SNadav Rotem bool UsesCodeGen = (Action != Backend_EmitNothing && 555dc06b2d3SNadav Rotem Action != Backend_EmitBC && 556dc06b2d3SNadav Rotem Action != Backend_EmitLL); 557f4e22381SAlp Toker if (!TM) 558f4e22381SAlp Toker TM.reset(CreateTargetMachine(UsesCodeGen)); 559f4e22381SAlp Toker 560ecafbe60SChad Rosier if (UsesCodeGen && !TM) return; 561f4e22381SAlp Toker CreatePasses(); 562ec57ab37SNadav Rotem 563c1b1729bSDaniel Dunbar switch (Action) { 564c1b1729bSDaniel Dunbar case Backend_EmitNothing: 565c1b1729bSDaniel Dunbar break; 566c1b1729bSDaniel Dunbar 567c1b1729bSDaniel Dunbar case Backend_EmitBC: 568f4e22381SAlp Toker getPerModulePasses()->add(createBitcodeWriterPass(*OS)); 569c1b1729bSDaniel Dunbar break; 570c1b1729bSDaniel Dunbar 571c1b1729bSDaniel Dunbar case Backend_EmitLL: 572c1b1729bSDaniel Dunbar FormattedOS.setStream(*OS, formatted_raw_ostream::PRESERVE_STREAM); 573*809403ffSChandler Carruth getPerModulePasses()->add(createPrintModulePass(FormattedOS)); 574c1b1729bSDaniel Dunbar break; 575c1b1729bSDaniel Dunbar 576c1b1729bSDaniel Dunbar default: 577c1b1729bSDaniel Dunbar FormattedOS.setStream(*OS, formatted_raw_ostream::PRESERVE_STREAM); 578f4e22381SAlp Toker if (!AddEmitPasses(Action, FormattedOS)) 579c1b1729bSDaniel Dunbar return; 580c1b1729bSDaniel Dunbar } 581c1b1729bSDaniel Dunbar 58215e36e8eSAndrew Trick // Before executing passes, print the final values of the LLVM options. 58315e36e8eSAndrew Trick cl::PrintOptionValues(); 58415e36e8eSAndrew Trick 585c1b1729bSDaniel Dunbar // Run passes. For now we do all passes at once, but eventually we 586c1b1729bSDaniel Dunbar // would like to have the option of streaming code generation. 587c1b1729bSDaniel Dunbar 588c1b1729bSDaniel Dunbar if (PerFunctionPasses) { 589c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Per-function optimization"); 590c1b1729bSDaniel Dunbar 591c1b1729bSDaniel Dunbar PerFunctionPasses->doInitialization(); 592c1b1729bSDaniel Dunbar for (Module::iterator I = TheModule->begin(), 593c1b1729bSDaniel Dunbar E = TheModule->end(); I != E; ++I) 594c1b1729bSDaniel Dunbar if (!I->isDeclaration()) 595c1b1729bSDaniel Dunbar PerFunctionPasses->run(*I); 596c1b1729bSDaniel Dunbar PerFunctionPasses->doFinalization(); 597c1b1729bSDaniel Dunbar } 598c1b1729bSDaniel Dunbar 599c1b1729bSDaniel Dunbar if (PerModulePasses) { 600c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Per-module optimization passes"); 601c1b1729bSDaniel Dunbar PerModulePasses->run(*TheModule); 602c1b1729bSDaniel Dunbar } 603c1b1729bSDaniel Dunbar 604c1b1729bSDaniel Dunbar if (CodeGenPasses) { 605c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Code generation"); 606195fa003SDaniel Dunbar CodeGenPasses->run(*TheModule); 607c1b1729bSDaniel Dunbar } 608c1b1729bSDaniel Dunbar } 609c1b1729bSDaniel Dunbar 6109c902b55SDavid Blaikie void clang::EmitBackendOutput(DiagnosticsEngine &Diags, 6119c902b55SDavid Blaikie const CodeGenOptions &CGOpts, 612432add5bSNick Lewycky const clang::TargetOptions &TOpts, 613e83b9060SAlp Toker const LangOptions &LOpts, StringRef TDesc, 614e83b9060SAlp Toker Module *M, BackendAction Action, 615e83b9060SAlp Toker raw_ostream *OS) { 616fec0ff84SDan Gohman EmitAssemblyHelper AsmHelper(Diags, CGOpts, TOpts, LOpts, M); 617c1b1729bSDaniel Dunbar 618c1b1729bSDaniel Dunbar AsmHelper.EmitAssembly(Action, OS); 619e83b9060SAlp Toker 620e83b9060SAlp Toker // If an optional clang TargetInfo description string was passed in, use it to 621e83b9060SAlp Toker // verify the LLVM TargetMachine's DataLayout. 622e83b9060SAlp Toker if (AsmHelper.TM && !TDesc.empty()) { 623e83b9060SAlp Toker std::string DLDesc = 624e83b9060SAlp Toker AsmHelper.TM->getDataLayout()->getStringRepresentation(); 625e83b9060SAlp Toker if (DLDesc != TDesc) { 626e83b9060SAlp Toker unsigned DiagID = Diags.getCustomDiagID( 627e83b9060SAlp Toker DiagnosticsEngine::Error, "backend data layout '%0' does not match " 628e83b9060SAlp Toker "expected target description '%1'"); 629e83b9060SAlp Toker Diags.Report(DiagID) << DLDesc << TDesc; 630e83b9060SAlp Toker } 631e83b9060SAlp Toker } 632c1b1729bSDaniel Dunbar } 633