From 21fca5ddcedd4c19303b08aaf1bf6b71399db7b4 Mon Sep 17 00:00:00 2001 From: Rafael Espindola Date: Wed, 2 Dec 2015 19:30:52 +0000 Subject: [PATCH] Fix linking when we copy over only a decl. We were failing to copy the fact that the GV is weak and in the case of an alias, producing invalid IR. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@254538 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/Linker/LinkModules.cpp | 56 +++++++++++++++++++++------------- test/Linker/Inputs/comdat14.ll | 12 ++++++++ test/Linker/comdat14.ll | 9 ++++++ 3 files changed, 55 insertions(+), 22 deletions(-) create mode 100644 test/Linker/Inputs/comdat14.ll create mode 100644 test/Linker/comdat14.ll diff --git a/lib/Linker/LinkModules.cpp b/lib/Linker/LinkModules.cpp index fba231100ee..1a82bbc7c90 100644 --- a/lib/Linker/LinkModules.cpp +++ b/lib/Linker/LinkModules.cpp @@ -453,7 +453,7 @@ private: /// Handles cloning of a global values from the source module into /// the destination module, including setting the attributes and visibility. GlobalValue *copyGlobalValueProto(TypeMapTy &TypeMap, const GlobalValue *SGV, - const GlobalValue *DGV = nullptr); + const GlobalValue *DGV, bool ForDefinition); /// Check if we should promote the given local value to global scope. bool doPromoteLocalToGlobal(const GlobalValue *SGV); @@ -594,14 +594,8 @@ static void forceRenaming(GlobalValue *GV, StringRef Name) { void ModuleLinker::copyGVAttributes(GlobalValue *NewGV, const GlobalValue *SrcGV) { auto *GA = dyn_cast(SrcGV); - // Check for the special case of converting an alias (definition) to a - // non-alias (declaration). This can happen when we are importing and - // encounter a weak_any alias (weak_any defs may not be imported, see - // comments in ModuleLinker::getLinkage) or an alias whose base object is - // being imported as a declaration. In that case copy the attributes from the - // base object. if (GA && !dyn_cast(NewGV)) { - assert(isPerformingImport() && !doImportAsDefinition(GA)); + // FIXME: this is likelly bogus: NewGV->copyAttributesFrom(GA->getBaseObject()); } else NewGV->copyAttributesFrom(SrcGV); @@ -779,11 +773,12 @@ ModuleLinker::copyGlobalVariableProto(TypeMapTy &TypeMap, // No linking to be performed or linking from the source: simply create an // identical version of the symbol over in the dest module... the // initializer will be filled in later by LinkGlobalInits. - GlobalVariable *NewDGV = new GlobalVariable( - DstM, TypeMap.get(SGVar->getType()->getElementType()), - SGVar->isConstant(), getLinkage(SGVar), /*init*/ nullptr, getName(SGVar), - /*insertbefore*/ nullptr, SGVar->getThreadLocalMode(), - SGVar->getType()->getAddressSpace()); + GlobalVariable *NewDGV = + new GlobalVariable(DstM, TypeMap.get(SGVar->getType()->getElementType()), + SGVar->isConstant(), GlobalValue::ExternalLinkage, + /*init*/ nullptr, getName(SGVar), + /*insertbefore*/ nullptr, SGVar->getThreadLocalMode(), + SGVar->getType()->getAddressSpace()); return NewDGV; } @@ -794,8 +789,8 @@ Function *ModuleLinker::copyFunctionProto(TypeMapTy &TypeMap, const Function *SF) { // If there is no linkage to be performed or we are linking from the source, // bring SF over. - return Function::Create(TypeMap.get(SF->getFunctionType()), getLinkage(SF), - getName(SF), &DstM); + return Function::Create(TypeMap.get(SF->getFunctionType()), + GlobalValue::ExternalLinkage, getName(SF), &DstM); } /// Set up prototypes for any aliases that come over from the source module. @@ -829,7 +824,7 @@ GlobalValue *ModuleLinker::copyGlobalAliasProto(TypeMapTy &TypeMap, // bring over SGA. auto *Ty = TypeMap.get(SGA->getValueType()); return GlobalAlias::create(Ty, SGA->getType()->getPointerAddressSpace(), - getLinkage(SGA), getName(SGA), &DstM); + GlobalValue::ExternalLinkage, getName(SGA), &DstM); } static GlobalValue::VisibilityTypes @@ -857,14 +852,31 @@ void ModuleLinker::setVisibility(GlobalValue *NewGV, const GlobalValue *SGV, GlobalValue *ModuleLinker::copyGlobalValueProto(TypeMapTy &TypeMap, const GlobalValue *SGV, - const GlobalValue *DGV) { + const GlobalValue *DGV, + bool ForDefinition) { GlobalValue *NewGV; - if (auto *SGVar = dyn_cast(SGV)) + if (auto *SGVar = dyn_cast(SGV)) { NewGV = copyGlobalVariableProto(TypeMap, SGVar); - else if (auto *SF = dyn_cast(SGV)) + } else if (auto *SF = dyn_cast(SGV)) { NewGV = copyFunctionProto(TypeMap, SF); - else - NewGV = copyGlobalAliasProto(TypeMap, cast(SGV)); + } else { + if (ForDefinition) + NewGV = copyGlobalAliasProto(TypeMap, cast(SGV)); + else + NewGV = new GlobalVariable( + DstM, TypeMap.get(SGV->getType()->getElementType()), + /*isConstant*/ false, GlobalValue::ExternalLinkage, + /*init*/ nullptr, getName(SGV), + /*insertbefore*/ nullptr, SGV->getThreadLocalMode(), + SGV->getType()->getAddressSpace()); + } + + if (ForDefinition) + NewGV->setLinkage(getLinkage(SGV)); + else if (SGV->hasAvailableExternallyLinkage() || SGV->hasWeakLinkage() || + SGV->hasLinkOnceLinkage()) + NewGV->setLinkage(GlobalValue::ExternalWeakLinkage); + copyGVAttributes(NewGV, SGV); setVisibility(NewGV, SGV, DGV); return NewGV; @@ -1446,7 +1458,7 @@ bool ModuleLinker::linkGlobalValueProto(GlobalValue *SGV) { // When linking from source we setVisibility from copyGlobalValueProto. setVisibility(NewGV, SGV, DGV); } else { - NewGV = copyGlobalValueProto(TypeMap, SGV, DGV); + NewGV = copyGlobalValueProto(TypeMap, SGV, DGV, LinkFromSrc); if (isPerformingImport() && !doImportAsDefinition(SGV)) DoNotLinkFromSource.insert(SGV); diff --git a/test/Linker/Inputs/comdat14.ll b/test/Linker/Inputs/comdat14.ll new file mode 100644 index 00000000000..7243ff86444 --- /dev/null +++ b/test/Linker/Inputs/comdat14.ll @@ -0,0 +1,12 @@ +$c = comdat any + +@v2 = weak global i32 0, comdat ($c) +define i32* @f2() { + ret i32* @v2 +} + +@v3 = weak alias i32, i32* @v2 +define i32* @f3() { + ret i32* @v3 +} + diff --git a/test/Linker/comdat14.ll b/test/Linker/comdat14.ll new file mode 100644 index 00000000000..70d1142bdf9 --- /dev/null +++ b/test/Linker/comdat14.ll @@ -0,0 +1,9 @@ +; RUN: llvm-link -S %s %p/Inputs/comdat14.ll -o - | FileCheck %s + +$c = comdat any + +@v = global i32 0, comdat ($c) + +; CHECK: @v = global i32 0, comdat($c) +; CHECK: @v2 = extern_weak global i32 +; CHECK: @v3 = extern_weak global i32 -- 2.34.1