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" 171b3304deSChandler Carruth #include "llvm/Bitcode/BitcodeWriterPass.h" 18c1b1729bSDaniel Dunbar #include "llvm/CodeGen/RegAllocRegistry.h" 19c1b1729bSDaniel Dunbar #include "llvm/CodeGen/SchedulerRegistry.h" 20ffd5551bSChandler Carruth #include "llvm/IR/DataLayout.h" 210a50c496SChandler Carruth #include "llvm/IR/IRPrintingPasses.h" 22ffd5551bSChandler Carruth #include "llvm/IR/Module.h" 23ca88474aSChandler Carruth #include "llvm/IR/Verifier.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" 40dfca6f97SAhmed Charles #include <memory> 41c1b1729bSDaniel Dunbar using namespace clang; 42c1b1729bSDaniel Dunbar using namespace llvm; 43c1b1729bSDaniel Dunbar 44c1b1729bSDaniel Dunbar namespace { 45c1b1729bSDaniel Dunbar 46c1b1729bSDaniel Dunbar class EmitAssemblyHelper { 479c902b55SDavid Blaikie DiagnosticsEngine &Diags; 48c1b1729bSDaniel Dunbar const CodeGenOptions &CodeGenOpts; 49432add5bSNick Lewycky const clang::TargetOptions &TargetOpts; 50fec0ff84SDan Gohman const LangOptions &LangOpts; 51c1b1729bSDaniel Dunbar Module *TheModule; 52c1b1729bSDaniel Dunbar 53c1b1729bSDaniel Dunbar Timer CodeGenerationTime; 54c1b1729bSDaniel Dunbar 55195fa003SDaniel Dunbar mutable PassManager *CodeGenPasses; 56c1b1729bSDaniel Dunbar mutable PassManager *PerModulePasses; 57c1b1729bSDaniel Dunbar mutable FunctionPassManager *PerFunctionPasses; 58c1b1729bSDaniel Dunbar 59c1b1729bSDaniel Dunbar private: 60f4e22381SAlp Toker PassManager *getCodeGenPasses() const { 61c1b1729bSDaniel Dunbar if (!CodeGenPasses) { 62195fa003SDaniel Dunbar CodeGenPasses = new PassManager(); 63303f8b06SRafael Espindola CodeGenPasses->add(new DataLayoutPass(TheModule)); 64ed0f133bSChandler Carruth if (TM) 65ed0f133bSChandler Carruth TM->addAnalysisPasses(*CodeGenPasses); 66c1b1729bSDaniel Dunbar } 67c1b1729bSDaniel Dunbar return CodeGenPasses; 68c1b1729bSDaniel Dunbar } 69c1b1729bSDaniel Dunbar 70f4e22381SAlp Toker PassManager *getPerModulePasses() const { 71c1b1729bSDaniel Dunbar if (!PerModulePasses) { 72c1b1729bSDaniel Dunbar PerModulePasses = new PassManager(); 73303f8b06SRafael Espindola PerModulePasses->add(new DataLayoutPass(TheModule)); 74ed0f133bSChandler Carruth if (TM) 75ed0f133bSChandler Carruth TM->addAnalysisPasses(*PerModulePasses); 76c1b1729bSDaniel Dunbar } 77c1b1729bSDaniel Dunbar return PerModulePasses; 78c1b1729bSDaniel Dunbar } 79c1b1729bSDaniel Dunbar 80f4e22381SAlp Toker FunctionPassManager *getPerFunctionPasses() const { 81c1b1729bSDaniel Dunbar if (!PerFunctionPasses) { 82c1b1729bSDaniel Dunbar PerFunctionPasses = new FunctionPassManager(TheModule); 83303f8b06SRafael Espindola PerFunctionPasses->add(new DataLayoutPass(TheModule)); 84ed0f133bSChandler Carruth if (TM) 85ed0f133bSChandler Carruth TM->addAnalysisPasses(*PerFunctionPasses); 86c1b1729bSDaniel Dunbar } 87c1b1729bSDaniel Dunbar return PerFunctionPasses; 88c1b1729bSDaniel Dunbar } 89c1b1729bSDaniel Dunbar 90f4e22381SAlp Toker void CreatePasses(); 91ec57ab37SNadav Rotem 92dc06b2d3SNadav Rotem /// CreateTargetMachine - Generates the TargetMachine. 93dc06b2d3SNadav Rotem /// Returns Null if it is unable to create the target machine. 94dc06b2d3SNadav Rotem /// Some of our clang tests specify triples which are not built 95dc06b2d3SNadav Rotem /// into clang. This is okay because these tests check the generated 96dc06b2d3SNadav Rotem /// IR, and they require DataLayout which depends on the triple. 97dc06b2d3SNadav Rotem /// In this case, we allow this method to fail and not report an error. 98dc06b2d3SNadav Rotem /// When MustCreateTM is used, we print an error if we are unable to load 99dc06b2d3SNadav Rotem /// the requested target. 100dc06b2d3SNadav Rotem TargetMachine *CreateTargetMachine(bool MustCreateTM); 101c1b1729bSDaniel Dunbar 102c1b1729bSDaniel Dunbar /// AddEmitPasses - Add passes necessary to emit assembly or LLVM IR. 103c1b1729bSDaniel Dunbar /// 104c1b1729bSDaniel Dunbar /// \return True on success. 105f4e22381SAlp Toker bool AddEmitPasses(BackendAction Action, formatted_raw_ostream &OS); 106c1b1729bSDaniel Dunbar 107c1b1729bSDaniel Dunbar public: 1089c902b55SDavid Blaikie EmitAssemblyHelper(DiagnosticsEngine &_Diags, 109432add5bSNick Lewycky const CodeGenOptions &CGOpts, 110432add5bSNick Lewycky const clang::TargetOptions &TOpts, 111fec0ff84SDan Gohman const LangOptions &LOpts, 112c1b1729bSDaniel Dunbar Module *M) 113fec0ff84SDan Gohman : Diags(_Diags), CodeGenOpts(CGOpts), TargetOpts(TOpts), LangOpts(LOpts), 114c1b1729bSDaniel Dunbar TheModule(M), CodeGenerationTime("Code Generation Time"), 115c1b1729bSDaniel Dunbar CodeGenPasses(0), PerModulePasses(0), PerFunctionPasses(0) {} 116c1b1729bSDaniel Dunbar 117c1b1729bSDaniel Dunbar ~EmitAssemblyHelper() { 118c1b1729bSDaniel Dunbar delete CodeGenPasses; 119c1b1729bSDaniel Dunbar delete PerModulePasses; 120c1b1729bSDaniel Dunbar delete PerFunctionPasses; 121f4e22381SAlp Toker if (CodeGenOpts.DisableFree) 1229a16beb8SAhmed Charles BuryPointer(TM.release()); 123c1b1729bSDaniel Dunbar } 124c1b1729bSDaniel Dunbar 125b8984329SAhmed Charles std::unique_ptr<TargetMachine> TM; 126f4e22381SAlp Toker 127c1b1729bSDaniel Dunbar void EmitAssembly(BackendAction Action, raw_ostream *OS); 128c1b1729bSDaniel Dunbar }; 129c1b1729bSDaniel Dunbar 130c6515b6aSAlexey Samsonov // We need this wrapper to access LangOpts and CGOpts from extension functions 131c6515b6aSAlexey Samsonov // that we add to the PassManagerBuilder. 1320e96becfSAlexey Samsonov class PassManagerBuilderWrapper : public PassManagerBuilder { 1330e96becfSAlexey Samsonov public: 1349ab73626SAlexey Samsonov PassManagerBuilderWrapper(const CodeGenOptions &CGOpts, 1359ab73626SAlexey Samsonov const LangOptions &LangOpts) 1369ab73626SAlexey Samsonov : PassManagerBuilder(), CGOpts(CGOpts), LangOpts(LangOpts) {} 1379ab73626SAlexey Samsonov const CodeGenOptions &getCGOpts() const { return CGOpts; } 1380e96becfSAlexey Samsonov const LangOptions &getLangOpts() const { return LangOpts; } 1390e96becfSAlexey Samsonov private: 1409ab73626SAlexey Samsonov const CodeGenOptions &CGOpts; 1410e96becfSAlexey Samsonov const LangOptions &LangOpts; 1420e96becfSAlexey Samsonov }; 1430e96becfSAlexey Samsonov 144c1b1729bSDaniel Dunbar } 145c1b1729bSDaniel Dunbar 1465932ce25SDan Gohman static void addObjCARCAPElimPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 1475932ce25SDan Gohman if (Builder.OptLevel > 0) 1485932ce25SDan Gohman PM.add(createObjCARCAPElimPass()); 1495932ce25SDan Gohman } 1505932ce25SDan Gohman 151fec0ff84SDan Gohman static void addObjCARCExpandPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 152fec0ff84SDan Gohman if (Builder.OptLevel > 0) 153fec0ff84SDan Gohman PM.add(createObjCARCExpandPass()); 154fec0ff84SDan Gohman } 155fec0ff84SDan Gohman 156fec0ff84SDan Gohman static void addObjCARCOptPass(const PassManagerBuilder &Builder, PassManagerBase &PM) { 157fec0ff84SDan Gohman if (Builder.OptLevel > 0) 158fec0ff84SDan Gohman PM.add(createObjCARCOptPass()); 159fec0ff84SDan Gohman } 160fec0ff84SDan Gohman 1615c29705cSDiego Novillo static void addSampleProfileLoaderPass(const PassManagerBuilder &Builder, 1625c29705cSDiego Novillo PassManagerBase &PM) { 1635c29705cSDiego Novillo const PassManagerBuilderWrapper &BuilderWrapper = 1645c29705cSDiego Novillo static_cast<const PassManagerBuilderWrapper &>(Builder); 1655c29705cSDiego Novillo const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 1665c29705cSDiego Novillo PM.add(createSampleProfileLoaderPass(CGOpts.SampleProfileFile)); 1675c29705cSDiego Novillo } 1685c29705cSDiego Novillo 169b56be645SDiego Novillo static void addAddDiscriminatorsPass(const PassManagerBuilder &Builder, 170b56be645SDiego Novillo PassManagerBase &PM) { 171b56be645SDiego Novillo PM.add(createAddDiscriminatorsPass()); 172b56be645SDiego Novillo } 173b56be645SDiego Novillo 174a425589fSNuno Lopes static void addBoundsCheckingPass(const PassManagerBuilder &Builder, 175a425589fSNuno Lopes PassManagerBase &PM) { 176ebc59d5eSJoey Gouly PM.add(createBoundsCheckingPass()); 177a425589fSNuno Lopes } 178a425589fSNuno Lopes 1790e96becfSAlexey Samsonov static void addAddressSanitizerPasses(const PassManagerBuilder &Builder, 1808855ff61SKostya Serebryany PassManagerBase &PM) { 1810e96becfSAlexey Samsonov const PassManagerBuilderWrapper &BuilderWrapper = 1820e96becfSAlexey Samsonov static_cast<const PassManagerBuilderWrapper&>(Builder); 1839ab73626SAlexey Samsonov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 1840e96becfSAlexey Samsonov const LangOptions &LangOpts = BuilderWrapper.getLangOpts(); 18529524a98SAlexey Samsonov PM.add(createAddressSanitizerFunctionPass( 18629524a98SAlexey Samsonov LangOpts.Sanitize.InitOrder, 187f54319c8SWill Dietz LangOpts.Sanitize.UseAfterReturn, 188f54319c8SWill Dietz LangOpts.Sanitize.UseAfterScope, 189d04b861eSEvgeniy Stepanov CGOpts.SanitizerBlacklistFile)); 19029524a98SAlexey Samsonov PM.add(createAddressSanitizerModulePass( 19129524a98SAlexey Samsonov LangOpts.Sanitize.InitOrder, 192d04b861eSEvgeniy Stepanov CGOpts.SanitizerBlacklistFile)); 1938855ff61SKostya Serebryany } 1948855ff61SKostya Serebryany 195aea92e53SEvgeniy Stepanov static void addMemorySanitizerPass(const PassManagerBuilder &Builder, 196aea92e53SEvgeniy Stepanov PassManagerBase &PM) { 197ad8ab3d9SEvgeniy Stepanov const PassManagerBuilderWrapper &BuilderWrapper = 198ad8ab3d9SEvgeniy Stepanov static_cast<const PassManagerBuilderWrapper&>(Builder); 199ad8ab3d9SEvgeniy Stepanov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 20029524a98SAlexey Samsonov PM.add(createMemorySanitizerPass(CGOpts.SanitizeMemoryTrackOrigins, 201c6515b6aSAlexey Samsonov CGOpts.SanitizerBlacklistFile)); 20210284671SEvgeniy Stepanov 20310284671SEvgeniy Stepanov // MemorySanitizer inserts complex instrumentation that mostly follows 20410284671SEvgeniy Stepanov // the logic of the original code, but operates on "shadow" values. 20510284671SEvgeniy Stepanov // It can benefit from re-running some general purpose optimization passes. 20610284671SEvgeniy Stepanov if (Builder.OptLevel > 0) { 20710284671SEvgeniy Stepanov PM.add(createEarlyCSEPass()); 20810284671SEvgeniy Stepanov PM.add(createReassociatePass()); 20910284671SEvgeniy Stepanov PM.add(createLICMPass()); 21010284671SEvgeniy Stepanov PM.add(createGVNPass()); 21110284671SEvgeniy Stepanov PM.add(createInstructionCombiningPass()); 21210284671SEvgeniy Stepanov PM.add(createDeadStoreEliminationPass()); 21310284671SEvgeniy Stepanov } 214aea92e53SEvgeniy Stepanov } 215aea92e53SEvgeniy Stepanov 21628a7a119SKostya Serebryany static void addThreadSanitizerPass(const PassManagerBuilder &Builder, 21728a7a119SKostya Serebryany PassManagerBase &PM) { 218c6515b6aSAlexey Samsonov const PassManagerBuilderWrapper &BuilderWrapper = 219c6515b6aSAlexey Samsonov static_cast<const PassManagerBuilderWrapper&>(Builder); 220c6515b6aSAlexey Samsonov const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 221c6515b6aSAlexey Samsonov PM.add(createThreadSanitizerPass(CGOpts.SanitizerBlacklistFile)); 22228a7a119SKostya Serebryany } 22328a7a119SKostya Serebryany 224c377275aSPeter Collingbourne static void addDataFlowSanitizerPass(const PassManagerBuilder &Builder, 225c377275aSPeter Collingbourne PassManagerBase &PM) { 226276be3c5SPeter Collingbourne const PassManagerBuilderWrapper &BuilderWrapper = 227276be3c5SPeter Collingbourne static_cast<const PassManagerBuilderWrapper&>(Builder); 228276be3c5SPeter Collingbourne const CodeGenOptions &CGOpts = BuilderWrapper.getCGOpts(); 229276be3c5SPeter Collingbourne PM.add(createDataFlowSanitizerPass(CGOpts.SanitizerBlacklistFile)); 230c377275aSPeter Collingbourne } 231c377275aSPeter Collingbourne 232f4e22381SAlp Toker void EmitAssemblyHelper::CreatePasses() { 233c1b1729bSDaniel Dunbar unsigned OptLevel = CodeGenOpts.OptimizationLevel; 234b0eea8b5SDouglas Gregor CodeGenOptions::InliningMethod Inlining = CodeGenOpts.getInlining(); 235c1b1729bSDaniel Dunbar 236c1b1729bSDaniel Dunbar // Handle disabling of LLVM optimization, where we want to preserve the 237c1b1729bSDaniel Dunbar // internal module before any optimization. 238c1b1729bSDaniel Dunbar if (CodeGenOpts.DisableLLVMOpts) { 239c1b1729bSDaniel Dunbar OptLevel = 0; 240c1b1729bSDaniel Dunbar Inlining = CodeGenOpts.NoInlining; 241c1b1729bSDaniel Dunbar } 242c1b1729bSDaniel Dunbar 2439ab73626SAlexey Samsonov PassManagerBuilderWrapper PMBuilder(CodeGenOpts, LangOpts); 244ecf0ba5bSChris Lattner PMBuilder.OptLevel = OptLevel; 245ecf0ba5bSChris Lattner PMBuilder.SizeLevel = CodeGenOpts.OptimizeSize; 246d3f3e4f0SNick Lewycky PMBuilder.BBVectorize = CodeGenOpts.VectorizeBB; 247d3f3e4f0SNick Lewycky PMBuilder.SLPVectorize = CodeGenOpts.VectorizeSLP; 248d3f3e4f0SNick Lewycky PMBuilder.LoopVectorize = CodeGenOpts.VectorizeLoop; 249d98cec5cSChris Lattner 250ecf0ba5bSChris Lattner PMBuilder.DisableUnitAtATime = !CodeGenOpts.UnitAtATime; 251ecf0ba5bSChris Lattner PMBuilder.DisableUnrollLoops = !CodeGenOpts.UnrollLoops; 252ce0697f4SHal Finkel PMBuilder.RerollLoops = CodeGenOpts.RerollLoops; 2535c12367bSChris Lattner 254b56be645SDiego Novillo PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 255b56be645SDiego Novillo addAddDiscriminatorsPass); 256b56be645SDiego Novillo 2575c29705cSDiego Novillo if (!CodeGenOpts.SampleProfileFile.empty()) 2585c29705cSDiego Novillo PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 2595c29705cSDiego Novillo addSampleProfileLoaderPass); 2605c29705cSDiego Novillo 261fec0ff84SDan Gohman // In ObjC ARC mode, add the main ARC optimization passes. 262fec0ff84SDan Gohman if (LangOpts.ObjCAutoRefCount) { 263fec0ff84SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_EarlyAsPossible, 264fec0ff84SDan Gohman addObjCARCExpandPass); 2655932ce25SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_ModuleOptimizerEarly, 2665932ce25SDan Gohman addObjCARCAPElimPass); 267fec0ff84SDan Gohman PMBuilder.addExtension(PassManagerBuilder::EP_ScalarOptimizerLate, 268fec0ff84SDan Gohman addObjCARCOptPass); 269fec0ff84SDan Gohman } 270fec0ff84SDan Gohman 2716b53e224SRichard Smith if (LangOpts.Sanitize.LocalBounds) { 272a425589fSNuno Lopes PMBuilder.addExtension(PassManagerBuilder::EP_ScalarOptimizerLate, 273a425589fSNuno Lopes addBoundsCheckingPass); 274a425589fSNuno Lopes PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 275a425589fSNuno Lopes addBoundsCheckingPass); 276a425589fSNuno Lopes } 277a425589fSNuno Lopes 278f54319c8SWill Dietz if (LangOpts.Sanitize.Address) { 2797e247f22SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 2800e96becfSAlexey Samsonov addAddressSanitizerPasses); 281d4768576SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 2820e96becfSAlexey Samsonov addAddressSanitizerPasses); 2838855ff61SKostya Serebryany } 2848855ff61SKostya Serebryany 285f54319c8SWill Dietz if (LangOpts.Sanitize.Memory) { 286aea92e53SEvgeniy Stepanov PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 287aea92e53SEvgeniy Stepanov addMemorySanitizerPass); 288aea92e53SEvgeniy Stepanov PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 289aea92e53SEvgeniy Stepanov addMemorySanitizerPass); 290aea92e53SEvgeniy Stepanov } 291aea92e53SEvgeniy Stepanov 292f54319c8SWill Dietz if (LangOpts.Sanitize.Thread) { 293d18cb502SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 29428a7a119SKostya Serebryany addThreadSanitizerPass); 29528a7a119SKostya Serebryany PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 29628a7a119SKostya Serebryany addThreadSanitizerPass); 29728a7a119SKostya Serebryany } 29828a7a119SKostya Serebryany 299c377275aSPeter Collingbourne if (LangOpts.Sanitize.DataFlow) { 300c377275aSPeter Collingbourne PMBuilder.addExtension(PassManagerBuilder::EP_OptimizerLast, 301c377275aSPeter Collingbourne addDataFlowSanitizerPass); 302c377275aSPeter Collingbourne PMBuilder.addExtension(PassManagerBuilder::EP_EnabledOnOptLevel0, 303c377275aSPeter Collingbourne addDataFlowSanitizerPass); 304c377275aSPeter Collingbourne } 305c377275aSPeter Collingbourne 3065c12367bSChris Lattner // Figure out TargetLibraryInfo. 30728b9e8b3SBill Wendling Triple TargetTriple(TheModule->getTargetTriple()); 308ecf0ba5bSChris Lattner PMBuilder.LibraryInfo = new TargetLibraryInfo(TargetTriple); 309fa222dfbSChris Lattner if (!CodeGenOpts.SimplifyLibCalls) 310ecf0ba5bSChris Lattner PMBuilder.LibraryInfo->disableAllFunctions(); 311c1b1729bSDaniel Dunbar 312c1b1729bSDaniel Dunbar switch (Inlining) { 313c1b1729bSDaniel Dunbar case CodeGenOptions::NoInlining: break; 314c1b1729bSDaniel Dunbar case CodeGenOptions::NormalInlining: { 315b5cfa8c2SEli Bendersky PMBuilder.Inliner = 316b5cfa8c2SEli Bendersky createFunctionInliningPass(OptLevel, CodeGenOpts.OptimizeSize); 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(); 337*52eaffeeSDuncan P. N. Exon Smith if (CodeGenOpts.VerifyModule) 338*52eaffeeSDuncan P. N. Exon Smith MPM->add(createDebugInfoVerifierPass()); 339d98cec5cSChris Lattner 340c02bbb61SNick Lewycky if (!CodeGenOpts.DisableGCov && 341c02bbb61SNick Lewycky (CodeGenOpts.EmitGcovArcs || CodeGenOpts.EmitGcovNotes)) { 342c8bf8249SNick Lewycky // Not using 'GCOVOptions::getDefault' allows us to avoid exiting if 343c8bf8249SNick Lewycky // LLVM's -default-gcov-version flag is set to something invalid. 344c8bf8249SNick Lewycky GCOVOptions Options; 345c8bf8249SNick Lewycky Options.EmitNotes = CodeGenOpts.EmitGcovNotes; 346c8bf8249SNick Lewycky Options.EmitData = CodeGenOpts.EmitGcovArcs; 347c8bf8249SNick Lewycky memcpy(Options.Version, CodeGenOpts.CoverageVersion, 4); 348c8bf8249SNick Lewycky Options.UseCfgChecksum = CodeGenOpts.CoverageExtraChecksum; 349c8bf8249SNick Lewycky Options.NoRedZone = CodeGenOpts.DisableRedZone; 350c8bf8249SNick Lewycky Options.FunctionNamesInData = 3516f15b290SNick Lewycky !CodeGenOpts.CoverageNoFunctionNamesInData; 352c8bf8249SNick Lewycky MPM->add(createGCOVProfilerPass(Options)); 353b0eea8b5SDouglas Gregor if (CodeGenOpts.getDebugInfo() == CodeGenOptions::NoDebugInfo) 354207bce31SNick Lewycky MPM->add(createStripSymbolsPass(true)); 355207bce31SNick Lewycky } 356207bce31SNick Lewycky 3575c12367bSChris Lattner PMBuilder.populateModulePassManager(*MPM); 358c1b1729bSDaniel Dunbar } 359c1b1729bSDaniel Dunbar 360dc06b2d3SNadav Rotem TargetMachine *EmitAssemblyHelper::CreateTargetMachine(bool MustCreateTM) { 361c1b1729bSDaniel Dunbar // Create the TargetMachine for generating code. 362c1b1729bSDaniel Dunbar std::string Error; 363c1b1729bSDaniel Dunbar std::string Triple = TheModule->getTargetTriple(); 364c1b1729bSDaniel Dunbar const llvm::Target *TheTarget = TargetRegistry::lookupTarget(Triple, Error); 365c1b1729bSDaniel Dunbar if (!TheTarget) { 366dc06b2d3SNadav Rotem if (MustCreateTM) 367ecafbe60SChad Rosier Diags.Report(diag::err_fe_unable_to_create_target) << Error; 368ec57ab37SNadav Rotem return 0; 369c1b1729bSDaniel Dunbar } 370c1b1729bSDaniel Dunbar 371c1b1729bSDaniel Dunbar // FIXME: Expose these capabilities via actual APIs!!!! Aside from just 372c1b1729bSDaniel Dunbar // being gross, this is also totally broken if we ever care about 373c1b1729bSDaniel Dunbar // concurrency. 374bb7ac52eSDaniel Dunbar 375c1b1729bSDaniel Dunbar TargetMachine::setAsmVerbosityDefault(CodeGenOpts.AsmVerbose); 376c1b1729bSDaniel Dunbar 377c1b1729bSDaniel Dunbar TargetMachine::setFunctionSections(CodeGenOpts.FunctionSections); 378c1b1729bSDaniel Dunbar TargetMachine::setDataSections (CodeGenOpts.DataSections); 379c1b1729bSDaniel Dunbar 380c1b1729bSDaniel Dunbar // FIXME: Parse this earlier. 3810d3d7770SBenjamin Kramer llvm::CodeModel::Model CM; 382c1b1729bSDaniel Dunbar if (CodeGenOpts.CodeModel == "small") { 3830d3d7770SBenjamin Kramer CM = llvm::CodeModel::Small; 384c1b1729bSDaniel Dunbar } else if (CodeGenOpts.CodeModel == "kernel") { 3850d3d7770SBenjamin Kramer CM = llvm::CodeModel::Kernel; 386c1b1729bSDaniel Dunbar } else if (CodeGenOpts.CodeModel == "medium") { 3870d3d7770SBenjamin Kramer CM = llvm::CodeModel::Medium; 388c1b1729bSDaniel Dunbar } else if (CodeGenOpts.CodeModel == "large") { 3890d3d7770SBenjamin Kramer CM = llvm::CodeModel::Large; 390c1b1729bSDaniel Dunbar } else { 391c1b1729bSDaniel Dunbar assert(CodeGenOpts.CodeModel.empty() && "Invalid code model!"); 3920d3d7770SBenjamin Kramer CM = llvm::CodeModel::Default; 393c1b1729bSDaniel Dunbar } 394c1b1729bSDaniel Dunbar 39509d20eefSDavid Blaikie SmallVector<const char *, 16> BackendArgs; 396c1b1729bSDaniel Dunbar BackendArgs.push_back("clang"); // Fake program name. 397c1b1729bSDaniel Dunbar if (!CodeGenOpts.DebugPass.empty()) { 398c1b1729bSDaniel Dunbar BackendArgs.push_back("-debug-pass"); 399c1b1729bSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.DebugPass.c_str()); 400c1b1729bSDaniel Dunbar } 401c1b1729bSDaniel Dunbar if (!CodeGenOpts.LimitFloatPrecision.empty()) { 402c1b1729bSDaniel Dunbar BackendArgs.push_back("-limit-float-precision"); 403c1b1729bSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.LimitFloatPrecision.c_str()); 404c1b1729bSDaniel Dunbar } 405c1b1729bSDaniel Dunbar if (llvm::TimePassesIsEnabled) 406c1b1729bSDaniel Dunbar BackendArgs.push_back("-time-passes"); 40712100e2cSDaniel Dunbar for (unsigned i = 0, e = CodeGenOpts.BackendOptions.size(); i != e; ++i) 40812100e2cSDaniel Dunbar BackendArgs.push_back(CodeGenOpts.BackendOptions[i].c_str()); 409ba3df1d3SChad Rosier if (CodeGenOpts.NoGlobalMerge) 410ba3df1d3SChad Rosier BackendArgs.push_back("-global-merge=false"); 411c1b1729bSDaniel Dunbar BackendArgs.push_back(0); 412c1b1729bSDaniel Dunbar llvm::cl::ParseCommandLineOptions(BackendArgs.size() - 1, 41309d20eefSDavid Blaikie BackendArgs.data()); 414c1b1729bSDaniel Dunbar 415c1b1729bSDaniel Dunbar std::string FeaturesStr; 416adc79598SEvan Cheng if (TargetOpts.Features.size()) { 417c1b1729bSDaniel Dunbar SubtargetFeatures Features; 418c1b1729bSDaniel Dunbar for (std::vector<std::string>::const_iterator 419c1b1729bSDaniel Dunbar it = TargetOpts.Features.begin(), 420c1b1729bSDaniel Dunbar ie = TargetOpts.Features.end(); it != ie; ++it) 421c1b1729bSDaniel Dunbar Features.AddFeature(*it); 422c1b1729bSDaniel Dunbar FeaturesStr = Features.getString(); 423c1b1729bSDaniel Dunbar } 4243f37dd06SEvan Cheng 4253f37dd06SEvan Cheng llvm::Reloc::Model RM = llvm::Reloc::Default; 4263f37dd06SEvan Cheng if (CodeGenOpts.RelocationModel == "static") { 4273f37dd06SEvan Cheng RM = llvm::Reloc::Static; 4283f37dd06SEvan Cheng } else if (CodeGenOpts.RelocationModel == "pic") { 4293f37dd06SEvan Cheng RM = llvm::Reloc::PIC_; 4303f37dd06SEvan Cheng } else { 4313f37dd06SEvan Cheng assert(CodeGenOpts.RelocationModel == "dynamic-no-pic" && 4323f37dd06SEvan Cheng "Invalid PIC model!"); 4333f37dd06SEvan Cheng RM = llvm::Reloc::DynamicNoPIC; 4343f37dd06SEvan Cheng } 4353f37dd06SEvan Cheng 436dd286bceSEvan Cheng CodeGenOpt::Level OptLevel = CodeGenOpt::Default; 437dd286bceSEvan Cheng switch (CodeGenOpts.OptimizationLevel) { 438dd286bceSEvan Cheng default: break; 439dd286bceSEvan Cheng case 0: OptLevel = CodeGenOpt::None; break; 440dd286bceSEvan Cheng case 3: OptLevel = CodeGenOpt::Aggressive; break; 441dd286bceSEvan Cheng } 442dd286bceSEvan Cheng 443432add5bSNick Lewycky llvm::TargetOptions Options; 444432add5bSNick Lewycky 44533ebd217SRafael Espindola if (CodeGenOpts.DisableIntegratedAS) 44633ebd217SRafael Espindola Options.DisableIntegratedAS = true; 44733ebd217SRafael Espindola 4487e2fd943SDavid Blaikie if (CodeGenOpts.CompressDebugSections) 4497e2fd943SDavid Blaikie Options.CompressDebugSections = true; 4507e2fd943SDavid Blaikie 451432add5bSNick Lewycky // Set frame pointer elimination mode. 452432add5bSNick Lewycky if (!CodeGenOpts.DisableFPElim) { 453432add5bSNick Lewycky Options.NoFramePointerElim = false; 454432add5bSNick Lewycky } else if (CodeGenOpts.OmitLeafFramePointer) { 455432add5bSNick Lewycky Options.NoFramePointerElim = false; 456432add5bSNick Lewycky } else { 457432add5bSNick Lewycky Options.NoFramePointerElim = true; 458432add5bSNick Lewycky } 459432add5bSNick Lewycky 46066aa045fSRafael Espindola if (CodeGenOpts.UseInitArray) 46166aa045fSRafael Espindola Options.UseInitArray = true; 46266aa045fSRafael Espindola 463432add5bSNick Lewycky // Set float ABI type. 464432add5bSNick Lewycky if (CodeGenOpts.FloatABI == "soft" || CodeGenOpts.FloatABI == "softfp") 465432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Soft; 466432add5bSNick Lewycky else if (CodeGenOpts.FloatABI == "hard") 467432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Hard; 468432add5bSNick Lewycky else { 469432add5bSNick Lewycky assert(CodeGenOpts.FloatABI.empty() && "Invalid float abi!"); 470432add5bSNick Lewycky Options.FloatABIType = llvm::FloatABI::Default; 471432add5bSNick Lewycky } 472432add5bSNick Lewycky 473aa53b936SLang Hames // Set FP fusion mode. 47465992f45SLang Hames switch (CodeGenOpts.getFPContractMode()) { 47565992f45SLang Hames case CodeGenOptions::FPC_Off: 476aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Strict; 477aa53b936SLang Hames break; 47865992f45SLang Hames case CodeGenOptions::FPC_On: 479aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Standard; 480aa53b936SLang Hames break; 48165992f45SLang Hames case CodeGenOptions::FPC_Fast: 482aa53b936SLang Hames Options.AllowFPOpFusion = llvm::FPOpFusion::Fast; 483aa53b936SLang Hames break; 484aa53b936SLang Hames } 485aa53b936SLang Hames 486432add5bSNick Lewycky Options.LessPreciseFPMADOption = CodeGenOpts.LessPreciseFPMAD; 487432add5bSNick Lewycky Options.NoInfsFPMath = CodeGenOpts.NoInfsFPMath; 488432add5bSNick Lewycky Options.NoNaNsFPMath = CodeGenOpts.NoNaNsFPMath; 489432add5bSNick Lewycky Options.NoZerosInBSS = CodeGenOpts.NoZeroInitializedInBSS; 490432add5bSNick Lewycky Options.UnsafeFPMath = CodeGenOpts.UnsafeFPMath; 491432add5bSNick Lewycky Options.UseSoftFloat = CodeGenOpts.SoftFloat; 492f4d3f7a0SNick Lewycky Options.StackAlignmentOverride = CodeGenOpts.StackAlignment; 4931c8c436aSNick Lewycky Options.DisableTailCalls = CodeGenOpts.DisableTailCalls; 49414adb360SBob Wilson Options.TrapFuncName = CodeGenOpts.TrapFuncName; 495097d019cSChandler Carruth Options.PositionIndependentExecutable = LangOpts.PIELevel != 0; 496432add5bSNick Lewycky 497adc79598SEvan Cheng TargetMachine *TM = TheTarget->createTargetMachine(Triple, TargetOpts.CPU, 498432add5bSNick Lewycky FeaturesStr, Options, 499432add5bSNick Lewycky RM, CM, OptLevel); 500c1b1729bSDaniel Dunbar 501c1b1729bSDaniel Dunbar if (CodeGenOpts.RelaxAll) 502c1b1729bSDaniel Dunbar TM->setMCRelaxAll(true); 50367919b2aSDaniel Dunbar if (CodeGenOpts.SaveTempLabels) 50467919b2aSDaniel Dunbar TM->setMCSaveTempLabels(true); 505e264187cSRafael Espindola if (CodeGenOpts.NoDwarf2CFIAsm) 506e264187cSRafael Espindola TM->setMCUseCFI(false); 507dd424a5bSNick Lewycky if (!CodeGenOpts.NoDwarfDirectoryAsm) 508dd424a5bSNick Lewycky TM->setMCUseDwarfDirectory(true); 509ca6b90d8SNick Lewycky if (CodeGenOpts.NoExecStack) 510ca6b90d8SNick Lewycky TM->setMCNoExecStack(true); 511c1b1729bSDaniel Dunbar 512ec57ab37SNadav Rotem return TM; 513ec57ab37SNadav Rotem } 514ec57ab37SNadav Rotem 515ec57ab37SNadav Rotem bool EmitAssemblyHelper::AddEmitPasses(BackendAction Action, 516f4e22381SAlp Toker formatted_raw_ostream &OS) { 517ec57ab37SNadav Rotem 518c1b1729bSDaniel Dunbar // Create the code generator passes. 519f4e22381SAlp Toker PassManager *PM = getCodeGenPasses(); 520c1b1729bSDaniel Dunbar 521b1cfc686SChad Rosier // Add LibraryInfo. 522aa437df9SDaniel Dunbar llvm::Triple TargetTriple(TheModule->getTargetTriple()); 523aa437df9SDaniel Dunbar TargetLibraryInfo *TLI = new TargetLibraryInfo(TargetTriple); 524b1cfc686SChad Rosier if (!CodeGenOpts.SimplifyLibCalls) 525b1cfc686SChad Rosier TLI->disableAllFunctions(); 526b1cfc686SChad Rosier PM->add(TLI); 527e9c233b1SNadav Rotem 528ed0f133bSChandler Carruth // Add Target specific analysis passes. 529ed0f133bSChandler Carruth TM->addAnalysisPasses(*PM); 530b1cfc686SChad Rosier 531c1b1729bSDaniel Dunbar // Normal mode, emit a .s or .o file by running the code generator. Note, 532c1b1729bSDaniel Dunbar // this also adds codegenerator level optimization passes. 533c1b1729bSDaniel Dunbar TargetMachine::CodeGenFileType CGFT = TargetMachine::CGFT_AssemblyFile; 534c1b1729bSDaniel Dunbar if (Action == Backend_EmitObj) 535c1b1729bSDaniel Dunbar CGFT = TargetMachine::CGFT_ObjectFile; 536c1b1729bSDaniel Dunbar else if (Action == Backend_EmitMCNull) 537c1b1729bSDaniel Dunbar CGFT = TargetMachine::CGFT_Null; 538c1b1729bSDaniel Dunbar else 539c1b1729bSDaniel Dunbar assert(Action == Backend_EmitAssembly && "Invalid action!"); 540fec0ff84SDan Gohman 541fec0ff84SDan Gohman // Add ObjC ARC final-cleanup optimizations. This is done as part of the 542fec0ff84SDan Gohman // "codegen" passes so that it isn't run multiple times when there is 543fec0ff84SDan Gohman // inlining happening. 5441170b088SDan Gohman if (LangOpts.ObjCAutoRefCount && 5451170b088SDan Gohman CodeGenOpts.OptimizationLevel > 0) 546fec0ff84SDan Gohman PM->add(createObjCARCContractPass()); 547fec0ff84SDan Gohman 548dd286bceSEvan Cheng if (TM->addPassesToEmitFile(*PM, OS, CGFT, 549c1b1729bSDaniel Dunbar /*DisableVerify=*/!CodeGenOpts.VerifyModule)) { 550c1b1729bSDaniel Dunbar Diags.Report(diag::err_fe_unable_to_interface_with_target); 551c1b1729bSDaniel Dunbar return false; 552c1b1729bSDaniel Dunbar } 553c1b1729bSDaniel Dunbar 554c1b1729bSDaniel Dunbar return true; 555c1b1729bSDaniel Dunbar } 556c1b1729bSDaniel Dunbar 557c1b1729bSDaniel Dunbar void EmitAssemblyHelper::EmitAssembly(BackendAction Action, raw_ostream *OS) { 558c1b1729bSDaniel Dunbar TimeRegion Region(llvm::TimePassesIsEnabled ? &CodeGenerationTime : 0); 559c1b1729bSDaniel Dunbar llvm::formatted_raw_ostream FormattedOS; 560c1b1729bSDaniel Dunbar 561dc06b2d3SNadav Rotem bool UsesCodeGen = (Action != Backend_EmitNothing && 562dc06b2d3SNadav Rotem Action != Backend_EmitBC && 563dc06b2d3SNadav Rotem Action != Backend_EmitLL); 564f4e22381SAlp Toker if (!TM) 565f4e22381SAlp Toker TM.reset(CreateTargetMachine(UsesCodeGen)); 566f4e22381SAlp Toker 567ecafbe60SChad Rosier if (UsesCodeGen && !TM) return; 568f4e22381SAlp Toker CreatePasses(); 569ec57ab37SNadav Rotem 570c1b1729bSDaniel Dunbar switch (Action) { 571c1b1729bSDaniel Dunbar case Backend_EmitNothing: 572c1b1729bSDaniel Dunbar break; 573c1b1729bSDaniel Dunbar 574c1b1729bSDaniel Dunbar case Backend_EmitBC: 575f4e22381SAlp Toker getPerModulePasses()->add(createBitcodeWriterPass(*OS)); 576c1b1729bSDaniel Dunbar break; 577c1b1729bSDaniel Dunbar 578c1b1729bSDaniel Dunbar case Backend_EmitLL: 579c1b1729bSDaniel Dunbar FormattedOS.setStream(*OS, formatted_raw_ostream::PRESERVE_STREAM); 580809403ffSChandler Carruth getPerModulePasses()->add(createPrintModulePass(FormattedOS)); 581c1b1729bSDaniel Dunbar break; 582c1b1729bSDaniel Dunbar 583c1b1729bSDaniel Dunbar default: 584c1b1729bSDaniel Dunbar FormattedOS.setStream(*OS, formatted_raw_ostream::PRESERVE_STREAM); 585f4e22381SAlp Toker if (!AddEmitPasses(Action, FormattedOS)) 586c1b1729bSDaniel Dunbar return; 587c1b1729bSDaniel Dunbar } 588c1b1729bSDaniel Dunbar 58915e36e8eSAndrew Trick // Before executing passes, print the final values of the LLVM options. 59015e36e8eSAndrew Trick cl::PrintOptionValues(); 59115e36e8eSAndrew Trick 592c1b1729bSDaniel Dunbar // Run passes. For now we do all passes at once, but eventually we 593c1b1729bSDaniel Dunbar // would like to have the option of streaming code generation. 594c1b1729bSDaniel Dunbar 595c1b1729bSDaniel Dunbar if (PerFunctionPasses) { 596c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Per-function optimization"); 597c1b1729bSDaniel Dunbar 598c1b1729bSDaniel Dunbar PerFunctionPasses->doInitialization(); 599c1b1729bSDaniel Dunbar for (Module::iterator I = TheModule->begin(), 600c1b1729bSDaniel Dunbar E = TheModule->end(); I != E; ++I) 601c1b1729bSDaniel Dunbar if (!I->isDeclaration()) 602c1b1729bSDaniel Dunbar PerFunctionPasses->run(*I); 603c1b1729bSDaniel Dunbar PerFunctionPasses->doFinalization(); 604c1b1729bSDaniel Dunbar } 605c1b1729bSDaniel Dunbar 606c1b1729bSDaniel Dunbar if (PerModulePasses) { 607c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Per-module optimization passes"); 608c1b1729bSDaniel Dunbar PerModulePasses->run(*TheModule); 609c1b1729bSDaniel Dunbar } 610c1b1729bSDaniel Dunbar 611c1b1729bSDaniel Dunbar if (CodeGenPasses) { 612c1b1729bSDaniel Dunbar PrettyStackTraceString CrashInfo("Code generation"); 613195fa003SDaniel Dunbar CodeGenPasses->run(*TheModule); 614c1b1729bSDaniel Dunbar } 615c1b1729bSDaniel Dunbar } 616c1b1729bSDaniel Dunbar 6179c902b55SDavid Blaikie void clang::EmitBackendOutput(DiagnosticsEngine &Diags, 6189c902b55SDavid Blaikie const CodeGenOptions &CGOpts, 619432add5bSNick Lewycky const clang::TargetOptions &TOpts, 620e83b9060SAlp Toker const LangOptions &LOpts, StringRef TDesc, 621e83b9060SAlp Toker Module *M, BackendAction Action, 622e83b9060SAlp Toker raw_ostream *OS) { 623fec0ff84SDan Gohman EmitAssemblyHelper AsmHelper(Diags, CGOpts, TOpts, LOpts, M); 624c1b1729bSDaniel Dunbar 625c1b1729bSDaniel Dunbar AsmHelper.EmitAssembly(Action, OS); 626e83b9060SAlp Toker 627e83b9060SAlp Toker // If an optional clang TargetInfo description string was passed in, use it to 628e83b9060SAlp Toker // verify the LLVM TargetMachine's DataLayout. 629e83b9060SAlp Toker if (AsmHelper.TM && !TDesc.empty()) { 630e83b9060SAlp Toker std::string DLDesc = 631e83b9060SAlp Toker AsmHelper.TM->getDataLayout()->getStringRepresentation(); 632e83b9060SAlp Toker if (DLDesc != TDesc) { 633e83b9060SAlp Toker unsigned DiagID = Diags.getCustomDiagID( 634e83b9060SAlp Toker DiagnosticsEngine::Error, "backend data layout '%0' does not match " 635e83b9060SAlp Toker "expected target description '%1'"); 636e83b9060SAlp Toker Diags.Report(DiagID) << DLDesc << TDesc; 637e83b9060SAlp Toker } 638e83b9060SAlp Toker } 639c1b1729bSDaniel Dunbar } 640