You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			607 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			607 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===- lib/Linker/LinkModules.cpp - Module Linker Implementation ----------===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | //
 | ||
|  | // This file implements the LLVM module linker.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "LinkDiagnosticInfo.h"
 | ||
|  | #include "llvm-c/Linker.h"
 | ||
|  | #include "llvm/ADT/SetVector.h"
 | ||
|  | #include "llvm/IR/Comdat.h"
 | ||
|  | #include "llvm/IR/DiagnosticPrinter.h"
 | ||
|  | #include "llvm/IR/GlobalValue.h"
 | ||
|  | #include "llvm/IR/LLVMContext.h"
 | ||
|  | #include "llvm/IR/Module.h"
 | ||
|  | #include "llvm/Linker/Linker.h"
 | ||
|  | #include "llvm/Support/Error.h"
 | ||
|  | using namespace llvm; | ||
|  | 
 | ||
|  | namespace { | ||
|  | 
 | ||
|  | /// This is an implementation class for the LinkModules function, which is the
 | ||
|  | /// entrypoint for this file.
 | ||
|  | class ModuleLinker { | ||
|  |   IRMover &Mover; | ||
|  |   std::unique_ptr<Module> SrcM; | ||
|  | 
 | ||
|  |   SetVector<GlobalValue *> ValuesToLink; | ||
|  | 
 | ||
|  |   /// For symbol clashes, prefer those from Src.
 | ||
|  |   unsigned Flags; | ||
|  | 
 | ||
|  |   /// List of global value names that should be internalized.
 | ||
|  |   StringSet<> Internalize; | ||
|  | 
 | ||
|  |   /// Function that will perform the actual internalization. The reason for a
 | ||
|  |   /// callback is that the linker cannot call internalizeModule without
 | ||
|  |   /// creating a circular dependency between IPO and the linker.
 | ||
|  |   std::function<void(Module &, const StringSet<> &)> InternalizeCallback; | ||
|  | 
 | ||
|  |   /// Used as the callback for lazy linking.
 | ||
|  |   /// The mover has just hit GV and we have to decide if it, and other members
 | ||
|  |   /// of the same comdat, should be linked. Every member to be linked is passed
 | ||
|  |   /// to Add.
 | ||
|  |   void addLazyFor(GlobalValue &GV, const IRMover::ValueAdder &Add); | ||
|  | 
 | ||
|  |   bool shouldOverrideFromSrc() { return Flags & Linker::OverrideFromSrc; } | ||
|  |   bool shouldLinkOnlyNeeded() { return Flags & Linker::LinkOnlyNeeded; } | ||
|  | 
 | ||
|  |   bool shouldLinkFromSource(bool &LinkFromSrc, const GlobalValue &Dest, | ||
|  |                             const GlobalValue &Src); | ||
|  | 
 | ||
|  |   /// Should we have mover and linker error diag info?
 | ||
|  |   bool emitError(const Twine &Message) { | ||
|  |     SrcM->getContext().diagnose(LinkDiagnosticInfo(DS_Error, Message)); | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool getComdatLeader(Module &M, StringRef ComdatName, | ||
|  |                        const GlobalVariable *&GVar); | ||
|  |   bool computeResultingSelectionKind(StringRef ComdatName, | ||
|  |                                      Comdat::SelectionKind Src, | ||
|  |                                      Comdat::SelectionKind Dst, | ||
|  |                                      Comdat::SelectionKind &Result, | ||
|  |                                      bool &LinkFromSrc); | ||
|  |   std::map<const Comdat *, std::pair<Comdat::SelectionKind, bool>> | ||
|  |       ComdatsChosen; | ||
|  |   bool getComdatResult(const Comdat *SrcC, Comdat::SelectionKind &SK, | ||
|  |                        bool &LinkFromSrc); | ||
|  |   // Keep track of the lazy linked global members of each comdat in source.
 | ||
|  |   DenseMap<const Comdat *, std::vector<GlobalValue *>> LazyComdatMembers; | ||
|  | 
 | ||
|  |   /// Given a global in the source module, return the global in the
 | ||
|  |   /// destination module that is being linked to, if any.
 | ||
|  |   GlobalValue *getLinkedToGlobal(const GlobalValue *SrcGV) { | ||
|  |     Module &DstM = Mover.getModule(); | ||
|  |     // If the source has no name it can't link.  If it has local linkage,
 | ||
|  |     // there is no name match-up going on.
 | ||
|  |     if (!SrcGV->hasName() || GlobalValue::isLocalLinkage(SrcGV->getLinkage())) | ||
|  |       return nullptr; | ||
|  | 
 | ||
|  |     // Otherwise see if we have a match in the destination module's symtab.
 | ||
|  |     GlobalValue *DGV = DstM.getNamedValue(SrcGV->getName()); | ||
|  |     if (!DGV) | ||
|  |       return nullptr; | ||
|  | 
 | ||
|  |     // If we found a global with the same name in the dest module, but it has
 | ||
|  |     // internal linkage, we are really not doing any linkage here.
 | ||
|  |     if (DGV->hasLocalLinkage()) | ||
|  |       return nullptr; | ||
|  | 
 | ||
|  |     // Otherwise, we do in fact link to the destination global.
 | ||
|  |     return DGV; | ||
|  |   } | ||
|  | 
 | ||
|  |   /// Drop GV if it is a member of a comdat that we are dropping.
 | ||
|  |   /// This can happen with COFF's largest selection kind.
 | ||
|  |   void dropReplacedComdat(GlobalValue &GV, | ||
|  |                           const DenseSet<const Comdat *> &ReplacedDstComdats); | ||
|  | 
 | ||
|  |   bool linkIfNeeded(GlobalValue &GV); | ||
|  | 
 | ||
|  | public: | ||
|  |   ModuleLinker(IRMover &Mover, std::unique_ptr<Module> SrcM, unsigned Flags, | ||
|  |                std::function<void(Module &, const StringSet<> &)> | ||
|  |                    InternalizeCallback = {}) | ||
|  |       : Mover(Mover), SrcM(std::move(SrcM)), Flags(Flags), | ||
|  |         InternalizeCallback(std::move(InternalizeCallback)) {} | ||
|  | 
 | ||
|  |   bool run(); | ||
|  | }; | ||
|  | } | ||
|  | 
 | ||
|  | static GlobalValue::VisibilityTypes | ||
|  | getMinVisibility(GlobalValue::VisibilityTypes A, | ||
|  |                  GlobalValue::VisibilityTypes B) { | ||
|  |   if (A == GlobalValue::HiddenVisibility || B == GlobalValue::HiddenVisibility) | ||
|  |     return GlobalValue::HiddenVisibility; | ||
|  |   if (A == GlobalValue::ProtectedVisibility || | ||
|  |       B == GlobalValue::ProtectedVisibility) | ||
|  |     return GlobalValue::ProtectedVisibility; | ||
|  |   return GlobalValue::DefaultVisibility; | ||
|  | } | ||
|  | 
 | ||
|  | bool ModuleLinker::getComdatLeader(Module &M, StringRef ComdatName, | ||
|  |                                    const GlobalVariable *&GVar) { | ||
|  |   const GlobalValue *GVal = M.getNamedValue(ComdatName); | ||
|  |   if (const auto *GA = dyn_cast_or_null<GlobalAlias>(GVal)) { | ||
|  |     GVal = GA->getBaseObject(); | ||
|  |     if (!GVal) | ||
|  |       // We cannot resolve the size of the aliasee yet.
 | ||
|  |       return emitError("Linking COMDATs named '" + ComdatName + | ||
|  |                        "': COMDAT key involves incomputable alias size."); | ||
|  |   } | ||
|  | 
 | ||
|  |   GVar = dyn_cast_or_null<GlobalVariable>(GVal); | ||
|  |   if (!GVar) | ||
|  |     return emitError( | ||
|  |         "Linking COMDATs named '" + ComdatName + | ||
|  |         "': GlobalVariable required for data dependent selection!"); | ||
|  | 
 | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | bool ModuleLinker::computeResultingSelectionKind(StringRef ComdatName, | ||
|  |                                                  Comdat::SelectionKind Src, | ||
|  |                                                  Comdat::SelectionKind Dst, | ||
|  |                                                  Comdat::SelectionKind &Result, | ||
|  |                                                  bool &LinkFromSrc) { | ||
|  |   Module &DstM = Mover.getModule(); | ||
|  |   // The ability to mix Comdat::SelectionKind::Any with
 | ||
|  |   // Comdat::SelectionKind::Largest is a behavior that comes from COFF.
 | ||
|  |   bool DstAnyOrLargest = Dst == Comdat::SelectionKind::Any || | ||
|  |                          Dst == Comdat::SelectionKind::Largest; | ||
|  |   bool SrcAnyOrLargest = Src == Comdat::SelectionKind::Any || | ||
|  |                          Src == Comdat::SelectionKind::Largest; | ||
|  |   if (DstAnyOrLargest && SrcAnyOrLargest) { | ||
|  |     if (Dst == Comdat::SelectionKind::Largest || | ||
|  |         Src == Comdat::SelectionKind::Largest) | ||
|  |       Result = Comdat::SelectionKind::Largest; | ||
|  |     else | ||
|  |       Result = Comdat::SelectionKind::Any; | ||
|  |   } else if (Src == Dst) { | ||
|  |     Result = Dst; | ||
|  |   } else { | ||
|  |     return emitError("Linking COMDATs named '" + ComdatName + | ||
|  |                      "': invalid selection kinds!"); | ||
|  |   } | ||
|  | 
 | ||
|  |   switch (Result) { | ||
|  |   case Comdat::SelectionKind::Any: | ||
|  |     // Go with Dst.
 | ||
|  |     LinkFromSrc = false; | ||
|  |     break; | ||
|  |   case Comdat::SelectionKind::NoDuplicates: | ||
|  |     return emitError("Linking COMDATs named '" + ComdatName + | ||
|  |                      "': noduplicates has been violated!"); | ||
|  |   case Comdat::SelectionKind::ExactMatch: | ||
|  |   case Comdat::SelectionKind::Largest: | ||
|  |   case Comdat::SelectionKind::SameSize: { | ||
|  |     const GlobalVariable *DstGV; | ||
|  |     const GlobalVariable *SrcGV; | ||
|  |     if (getComdatLeader(DstM, ComdatName, DstGV) || | ||
|  |         getComdatLeader(*SrcM, ComdatName, SrcGV)) | ||
|  |       return true; | ||
|  | 
 | ||
|  |     const DataLayout &DstDL = DstM.getDataLayout(); | ||
|  |     const DataLayout &SrcDL = SrcM->getDataLayout(); | ||
|  |     uint64_t DstSize = DstDL.getTypeAllocSize(DstGV->getValueType()); | ||
|  |     uint64_t SrcSize = SrcDL.getTypeAllocSize(SrcGV->getValueType()); | ||
|  |     if (Result == Comdat::SelectionKind::ExactMatch) { | ||
|  |       if (SrcGV->getInitializer() != DstGV->getInitializer()) | ||
|  |         return emitError("Linking COMDATs named '" + ComdatName + | ||
|  |                          "': ExactMatch violated!"); | ||
|  |       LinkFromSrc = false; | ||
|  |     } else if (Result == Comdat::SelectionKind::Largest) { | ||
|  |       LinkFromSrc = SrcSize > DstSize; | ||
|  |     } else if (Result == Comdat::SelectionKind::SameSize) { | ||
|  |       if (SrcSize != DstSize) | ||
|  |         return emitError("Linking COMDATs named '" + ComdatName + | ||
|  |                          "': SameSize violated!"); | ||
|  |       LinkFromSrc = false; | ||
|  |     } else { | ||
|  |       llvm_unreachable("unknown selection kind"); | ||
|  |     } | ||
|  |     break; | ||
|  |   } | ||
|  |   } | ||
|  | 
 | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | bool ModuleLinker::getComdatResult(const Comdat *SrcC, | ||
|  |                                    Comdat::SelectionKind &Result, | ||
|  |                                    bool &LinkFromSrc) { | ||
|  |   Module &DstM = Mover.getModule(); | ||
|  |   Comdat::SelectionKind SSK = SrcC->getSelectionKind(); | ||
|  |   StringRef ComdatName = SrcC->getName(); | ||
|  |   Module::ComdatSymTabType &ComdatSymTab = DstM.getComdatSymbolTable(); | ||
|  |   Module::ComdatSymTabType::iterator DstCI = ComdatSymTab.find(ComdatName); | ||
|  | 
 | ||
|  |   if (DstCI == ComdatSymTab.end()) { | ||
|  |     // Use the comdat if it is only available in one of the modules.
 | ||
|  |     LinkFromSrc = true; | ||
|  |     Result = SSK; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   const Comdat *DstC = &DstCI->second; | ||
|  |   Comdat::SelectionKind DSK = DstC->getSelectionKind(); | ||
|  |   return computeResultingSelectionKind(ComdatName, SSK, DSK, Result, | ||
|  |                                        LinkFromSrc); | ||
|  | } | ||
|  | 
 | ||
|  | bool ModuleLinker::shouldLinkFromSource(bool &LinkFromSrc, | ||
|  |                                         const GlobalValue &Dest, | ||
|  |                                         const GlobalValue &Src) { | ||
|  | 
 | ||
|  |   // Should we unconditionally use the Src?
 | ||
|  |   if (shouldOverrideFromSrc()) { | ||
|  |     LinkFromSrc = true; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   // We always have to add Src if it has appending linkage.
 | ||
|  |   if (Src.hasAppendingLinkage()) { | ||
|  |     LinkFromSrc = true; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool SrcIsDeclaration = Src.isDeclarationForLinker(); | ||
|  |   bool DestIsDeclaration = Dest.isDeclarationForLinker(); | ||
|  | 
 | ||
|  |   if (SrcIsDeclaration) { | ||
|  |     // If Src is external or if both Src & Dest are external..  Just link the
 | ||
|  |     // external globals, we aren't adding anything.
 | ||
|  |     if (Src.hasDLLImportStorageClass()) { | ||
|  |       // If one of GVs is marked as DLLImport, result should be dllimport'ed.
 | ||
|  |       LinkFromSrc = DestIsDeclaration; | ||
|  |       return false; | ||
|  |     } | ||
|  |     // If the Dest is weak, use the source linkage.
 | ||
|  |     if (Dest.hasExternalWeakLinkage()) { | ||
|  |       LinkFromSrc = true; | ||
|  |       return false; | ||
|  |     } | ||
|  |     // Link an available_externally over a declaration.
 | ||
|  |     LinkFromSrc = !Src.isDeclaration() && Dest.isDeclaration(); | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (DestIsDeclaration) { | ||
|  |     // If Dest is external but Src is not:
 | ||
|  |     LinkFromSrc = true; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (Src.hasCommonLinkage()) { | ||
|  |     if (Dest.hasLinkOnceLinkage() || Dest.hasWeakLinkage()) { | ||
|  |       LinkFromSrc = true; | ||
|  |       return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!Dest.hasCommonLinkage()) { | ||
|  |       LinkFromSrc = false; | ||
|  |       return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     const DataLayout &DL = Dest.getParent()->getDataLayout(); | ||
|  |     uint64_t DestSize = DL.getTypeAllocSize(Dest.getValueType()); | ||
|  |     uint64_t SrcSize = DL.getTypeAllocSize(Src.getValueType()); | ||
|  |     LinkFromSrc = SrcSize > DestSize; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (Src.isWeakForLinker()) { | ||
|  |     assert(!Dest.hasExternalWeakLinkage()); | ||
|  |     assert(!Dest.hasAvailableExternallyLinkage()); | ||
|  | 
 | ||
|  |     if (Dest.hasLinkOnceLinkage() && Src.hasWeakLinkage()) { | ||
|  |       LinkFromSrc = true; | ||
|  |       return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     LinkFromSrc = false; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (Dest.isWeakForLinker()) { | ||
|  |     assert(Src.hasExternalLinkage()); | ||
|  |     LinkFromSrc = true; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   assert(!Src.hasExternalWeakLinkage()); | ||
|  |   assert(!Dest.hasExternalWeakLinkage()); | ||
|  |   assert(Dest.hasExternalLinkage() && Src.hasExternalLinkage() && | ||
|  |          "Unexpected linkage type!"); | ||
|  |   return emitError("Linking globals named '" + Src.getName() + | ||
|  |                    "': symbol multiply defined!"); | ||
|  | } | ||
|  | 
 | ||
|  | bool ModuleLinker::linkIfNeeded(GlobalValue &GV) { | ||
|  |   GlobalValue *DGV = getLinkedToGlobal(&GV); | ||
|  | 
 | ||
|  |   if (shouldLinkOnlyNeeded()) { | ||
|  |     // Always import variables with appending linkage.
 | ||
|  |     if (!GV.hasAppendingLinkage()) { | ||
|  |       // Don't import globals unless they are referenced by the destination
 | ||
|  |       // module.
 | ||
|  |       if (!DGV) | ||
|  |         return false; | ||
|  |       // Don't import globals that are already defined in the destination module
 | ||
|  |       if (!DGV->isDeclaration()) | ||
|  |         return false; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (DGV && !GV.hasLocalLinkage() && !GV.hasAppendingLinkage()) { | ||
|  |     auto *DGVar = dyn_cast<GlobalVariable>(DGV); | ||
|  |     auto *SGVar = dyn_cast<GlobalVariable>(&GV); | ||
|  |     if (DGVar && SGVar) { | ||
|  |       if (DGVar->isDeclaration() && SGVar->isDeclaration() && | ||
|  |           (!DGVar->isConstant() || !SGVar->isConstant())) { | ||
|  |         DGVar->setConstant(false); | ||
|  |         SGVar->setConstant(false); | ||
|  |       } | ||
|  |       if (DGVar->hasCommonLinkage() && SGVar->hasCommonLinkage()) { | ||
|  |         unsigned Align = std::max(DGVar->getAlignment(), SGVar->getAlignment()); | ||
|  |         SGVar->setAlignment(Align); | ||
|  |         DGVar->setAlignment(Align); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     GlobalValue::VisibilityTypes Visibility = | ||
|  |         getMinVisibility(DGV->getVisibility(), GV.getVisibility()); | ||
|  |     DGV->setVisibility(Visibility); | ||
|  |     GV.setVisibility(Visibility); | ||
|  | 
 | ||
|  |     GlobalValue::UnnamedAddr UnnamedAddr = GlobalValue::getMinUnnamedAddr( | ||
|  |         DGV->getUnnamedAddr(), GV.getUnnamedAddr()); | ||
|  |     DGV->setUnnamedAddr(UnnamedAddr); | ||
|  |     GV.setUnnamedAddr(UnnamedAddr); | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!DGV && !shouldOverrideFromSrc() && | ||
|  |       (GV.hasLocalLinkage() || GV.hasLinkOnceLinkage() || | ||
|  |        GV.hasAvailableExternallyLinkage())) | ||
|  |     return false; | ||
|  | 
 | ||
|  |   if (GV.isDeclaration()) | ||
|  |     return false; | ||
|  | 
 | ||
|  |   if (const Comdat *SC = GV.getComdat()) { | ||
|  |     bool LinkFromSrc; | ||
|  |     Comdat::SelectionKind SK; | ||
|  |     std::tie(SK, LinkFromSrc) = ComdatsChosen[SC]; | ||
|  |     if (!LinkFromSrc) | ||
|  |       return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool LinkFromSrc = true; | ||
|  |   if (DGV && shouldLinkFromSource(LinkFromSrc, *DGV, GV)) | ||
|  |     return true; | ||
|  |   if (LinkFromSrc) | ||
|  |     ValuesToLink.insert(&GV); | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | void ModuleLinker::addLazyFor(GlobalValue &GV, const IRMover::ValueAdder &Add) { | ||
|  |   // Add these to the internalize list
 | ||
|  |   if (!GV.hasLinkOnceLinkage() && !GV.hasAvailableExternallyLinkage() && | ||
|  |       !shouldLinkOnlyNeeded()) | ||
|  |     return; | ||
|  | 
 | ||
|  |   if (InternalizeCallback) | ||
|  |     Internalize.insert(GV.getName()); | ||
|  |   Add(GV); | ||
|  | 
 | ||
|  |   const Comdat *SC = GV.getComdat(); | ||
|  |   if (!SC) | ||
|  |     return; | ||
|  |   for (GlobalValue *GV2 : LazyComdatMembers[SC]) { | ||
|  |     GlobalValue *DGV = getLinkedToGlobal(GV2); | ||
|  |     bool LinkFromSrc = true; | ||
|  |     if (DGV && shouldLinkFromSource(LinkFromSrc, *DGV, *GV2)) | ||
|  |       return; | ||
|  |     if (!LinkFromSrc) | ||
|  |       continue; | ||
|  |     if (InternalizeCallback) | ||
|  |       Internalize.insert(GV2->getName()); | ||
|  |     Add(*GV2); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void ModuleLinker::dropReplacedComdat( | ||
|  |     GlobalValue &GV, const DenseSet<const Comdat *> &ReplacedDstComdats) { | ||
|  |   Comdat *C = GV.getComdat(); | ||
|  |   if (!C) | ||
|  |     return; | ||
|  |   if (!ReplacedDstComdats.count(C)) | ||
|  |     return; | ||
|  |   if (GV.use_empty()) { | ||
|  |     GV.eraseFromParent(); | ||
|  |     return; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (auto *F = dyn_cast<Function>(&GV)) { | ||
|  |     F->deleteBody(); | ||
|  |   } else if (auto *Var = dyn_cast<GlobalVariable>(&GV)) { | ||
|  |     Var->setInitializer(nullptr); | ||
|  |   } else { | ||
|  |     auto &Alias = cast<GlobalAlias>(GV); | ||
|  |     Module &M = *Alias.getParent(); | ||
|  |     PointerType &Ty = *cast<PointerType>(Alias.getType()); | ||
|  |     GlobalValue *Declaration; | ||
|  |     if (auto *FTy = dyn_cast<FunctionType>(Alias.getValueType())) { | ||
|  |       Declaration = Function::Create(FTy, GlobalValue::ExternalLinkage, "", &M); | ||
|  |     } else { | ||
|  |       Declaration = | ||
|  |           new GlobalVariable(M, Ty.getElementType(), /*isConstant*/ false, | ||
|  |                              GlobalValue::ExternalLinkage, | ||
|  |                              /*Initializer*/ nullptr); | ||
|  |     } | ||
|  |     Declaration->takeName(&Alias); | ||
|  |     Alias.replaceAllUsesWith(Declaration); | ||
|  |     Alias.eraseFromParent(); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | bool ModuleLinker::run() { | ||
|  |   Module &DstM = Mover.getModule(); | ||
|  |   DenseSet<const Comdat *> ReplacedDstComdats; | ||
|  | 
 | ||
|  |   for (const auto &SMEC : SrcM->getComdatSymbolTable()) { | ||
|  |     const Comdat &C = SMEC.getValue(); | ||
|  |     if (ComdatsChosen.count(&C)) | ||
|  |       continue; | ||
|  |     Comdat::SelectionKind SK; | ||
|  |     bool LinkFromSrc; | ||
|  |     if (getComdatResult(&C, SK, LinkFromSrc)) | ||
|  |       return true; | ||
|  |     ComdatsChosen[&C] = std::make_pair(SK, LinkFromSrc); | ||
|  | 
 | ||
|  |     if (!LinkFromSrc) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     Module::ComdatSymTabType &ComdatSymTab = DstM.getComdatSymbolTable(); | ||
|  |     Module::ComdatSymTabType::iterator DstCI = ComdatSymTab.find(C.getName()); | ||
|  |     if (DstCI == ComdatSymTab.end()) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // The source comdat is replacing the dest one.
 | ||
|  |     const Comdat *DstC = &DstCI->second; | ||
|  |     ReplacedDstComdats.insert(DstC); | ||
|  |   } | ||
|  | 
 | ||
|  |   // Alias have to go first, since we are not able to find their comdats
 | ||
|  |   // otherwise.
 | ||
|  |   for (auto I = DstM.alias_begin(), E = DstM.alias_end(); I != E;) { | ||
|  |     GlobalAlias &GV = *I++; | ||
|  |     dropReplacedComdat(GV, ReplacedDstComdats); | ||
|  |   } | ||
|  | 
 | ||
|  |   for (auto I = DstM.global_begin(), E = DstM.global_end(); I != E;) { | ||
|  |     GlobalVariable &GV = *I++; | ||
|  |     dropReplacedComdat(GV, ReplacedDstComdats); | ||
|  |   } | ||
|  | 
 | ||
|  |   for (auto I = DstM.begin(), E = DstM.end(); I != E;) { | ||
|  |     Function &GV = *I++; | ||
|  |     dropReplacedComdat(GV, ReplacedDstComdats); | ||
|  |   } | ||
|  | 
 | ||
|  |   for (GlobalVariable &GV : SrcM->globals()) | ||
|  |     if (GV.hasLinkOnceLinkage()) | ||
|  |       if (const Comdat *SC = GV.getComdat()) | ||
|  |         LazyComdatMembers[SC].push_back(&GV); | ||
|  | 
 | ||
|  |   for (Function &SF : *SrcM) | ||
|  |     if (SF.hasLinkOnceLinkage()) | ||
|  |       if (const Comdat *SC = SF.getComdat()) | ||
|  |         LazyComdatMembers[SC].push_back(&SF); | ||
|  | 
 | ||
|  |   for (GlobalAlias &GA : SrcM->aliases()) | ||
|  |     if (GA.hasLinkOnceLinkage()) | ||
|  |       if (const Comdat *SC = GA.getComdat()) | ||
|  |         LazyComdatMembers[SC].push_back(&GA); | ||
|  | 
 | ||
|  |   // Insert all of the globals in src into the DstM module... without linking
 | ||
|  |   // initializers (which could refer to functions not yet mapped over).
 | ||
|  |   for (GlobalVariable &GV : SrcM->globals()) | ||
|  |     if (linkIfNeeded(GV)) | ||
|  |       return true; | ||
|  | 
 | ||
|  |   for (Function &SF : *SrcM) | ||
|  |     if (linkIfNeeded(SF)) | ||
|  |       return true; | ||
|  | 
 | ||
|  |   for (GlobalAlias &GA : SrcM->aliases()) | ||
|  |     if (linkIfNeeded(GA)) | ||
|  |       return true; | ||
|  | 
 | ||
|  |   for (unsigned I = 0; I < ValuesToLink.size(); ++I) { | ||
|  |     GlobalValue *GV = ValuesToLink[I]; | ||
|  |     const Comdat *SC = GV->getComdat(); | ||
|  |     if (!SC) | ||
|  |       continue; | ||
|  |     for (GlobalValue *GV2 : LazyComdatMembers[SC]) { | ||
|  |       GlobalValue *DGV = getLinkedToGlobal(GV2); | ||
|  |       bool LinkFromSrc = true; | ||
|  |       if (DGV && shouldLinkFromSource(LinkFromSrc, *DGV, *GV2)) | ||
|  |         return true; | ||
|  |       if (LinkFromSrc) | ||
|  |         ValuesToLink.insert(GV2); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (InternalizeCallback) { | ||
|  |     for (GlobalValue *GV : ValuesToLink) | ||
|  |       Internalize.insert(GV->getName()); | ||
|  |   } | ||
|  | 
 | ||
|  |   // FIXME: Propagate Errors through to the caller instead of emitting
 | ||
|  |   // diagnostics.
 | ||
|  |   bool HasErrors = false; | ||
|  |   if (Error E = Mover.move(std::move(SrcM), ValuesToLink.getArrayRef(), | ||
|  |                            [this](GlobalValue &GV, IRMover::ValueAdder Add) { | ||
|  |                              addLazyFor(GV, Add); | ||
|  |                            }, | ||
|  |                            /* IsPerformingImport */ false)) { | ||
|  |     handleAllErrors(std::move(E), [&](ErrorInfoBase &EIB) { | ||
|  |       DstM.getContext().diagnose(LinkDiagnosticInfo(DS_Error, EIB.message())); | ||
|  |       HasErrors = true; | ||
|  |     }); | ||
|  |   } | ||
|  |   if (HasErrors) | ||
|  |     return true; | ||
|  | 
 | ||
|  |   if (InternalizeCallback) | ||
|  |     InternalizeCallback(DstM, Internalize); | ||
|  | 
 | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | Linker::Linker(Module &M) : Mover(M) {} | ||
|  | 
 | ||
|  | bool Linker::linkInModule( | ||
|  |     std::unique_ptr<Module> Src, unsigned Flags, | ||
|  |     std::function<void(Module &, const StringSet<> &)> InternalizeCallback) { | ||
|  |   ModuleLinker ModLinker(Mover, std::move(Src), Flags, | ||
|  |                          std::move(InternalizeCallback)); | ||
|  |   return ModLinker.run(); | ||
|  | } | ||
|  | 
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | // LinkModules entrypoint.
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | /// This function links two modules together, with the resulting Dest module
 | ||
|  | /// modified to be the composite of the two input modules. If an error occurs,
 | ||
|  | /// true is returned and ErrorMsg (if not null) is set to indicate the problem.
 | ||
|  | /// Upon failure, the Dest module could be in a modified state, and shouldn't be
 | ||
|  | /// relied on to be consistent.
 | ||
|  | bool Linker::linkModules( | ||
|  |     Module &Dest, std::unique_ptr<Module> Src, unsigned Flags, | ||
|  |     std::function<void(Module &, const StringSet<> &)> InternalizeCallback) { | ||
|  |   Linker L(Dest); | ||
|  |   return L.linkInModule(std::move(Src), Flags, std::move(InternalizeCallback)); | ||
|  | } | ||
|  | 
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | // C API.
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | LLVMBool LLVMLinkModules2(LLVMModuleRef Dest, LLVMModuleRef Src) { | ||
|  |   Module *D = unwrap(Dest); | ||
|  |   std::unique_ptr<Module> M(unwrap(Src)); | ||
|  |   return Linker::linkModules(*D, std::move(M)); | ||
|  | } |