Skip to content

Commit

Permalink
CloneModule: Handling cloning ifuncs
Browse files Browse the repository at this point in the history
This is tested in a future llvm-reduce patch.
  • Loading branch information
arsenm committed Nov 23, 2022
1 parent 10eda33 commit f069327
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 0 deletions.
15 changes: 15 additions & 0 deletions llvm/lib/Transforms/Utils/CloneModule.cpp
Expand Up @@ -109,6 +109,15 @@ std::unique_ptr<Module> llvm::CloneModule(
VMap[&I] = GA;
}

for (const GlobalIFunc &I : M.ifuncs()) {
// Defer setting the resolver function until after functions are cloned.
auto *GI =
GlobalIFunc::create(I.getValueType(), I.getAddressSpace(),
I.getLinkage(), I.getName(), nullptr, New.get());
GI->copyAttributesFrom(&I);
VMap[&I] = GI;
}

// Now that all of the things that global variable initializer can refer to
// have been created, loop through and copy the global variable referrers
// over... We also set the attributes on the global now.
Expand Down Expand Up @@ -184,6 +193,12 @@ std::unique_ptr<Module> llvm::CloneModule(
GA->setAliasee(MapValue(C, VMap));
}

for (const GlobalIFunc &I : M.ifuncs()) {
GlobalIFunc *GI = cast<GlobalIFunc>(VMap[&I]);
if (const Constant *Resolver = I.getResolver())
GI->setResolver(MapValue(Resolver, VMap));
}

// And named metadata....
for (const NamedMDNode &NMD : M.named_metadata()) {
NamedMDNode *NewNMD = New->getOrInsertNamedMetadata(NMD.getName());
Expand Down
32 changes: 32 additions & 0 deletions llvm/unittests/Transforms/Utils/CloningTest.cpp
Expand Up @@ -922,6 +922,23 @@ class CloneModule : public ::testing::Test {
GV->addMetadata(LLVMContext::MD_type, *MDNode::get(C, {}));
GV->setComdat(CD);

// Add ifuncs
{
const unsigned AddrSpace = 123;
auto *FuncPtrTy = Type::getInt8Ty(C)->getPointerTo(123);
auto *FuncTy = FunctionType::get(FuncPtrTy, false);

auto *ResolverF = Function::Create(FuncTy, GlobalValue::PrivateLinkage,
AddrSpace, "resolver", OldM);
BasicBlock *ResolverBody = BasicBlock::Create(C, "", ResolverF);
ReturnInst::Create(C, ConstantPointerNull::get(FuncPtrTy), ResolverBody);

GlobalIFunc *GI = GlobalIFunc::create(FuncTy, AddrSpace,
GlobalValue::LinkOnceODRLinkage,
"an_ifunc", ResolverF, OldM);
GI->setVisibility(GlobalValue::ProtectedVisibility);
}

{
// Add an empty compile unit first that isn't otherwise referenced, to
// confirm that compile units get cloned in the correct order.
Expand Down Expand Up @@ -1087,4 +1104,19 @@ TEST_F(CloneModule, Comdat) {
Function *NewF = NewM->getFunction("f");
EXPECT_EQ(CD, NewF->getComdat());
}

TEST_F(CloneModule, IFunc) {
ASSERT_EQ(1u, NewM->ifunc_size());

const GlobalIFunc &IFunc = *NewM->ifunc_begin();
EXPECT_EQ("an_ifunc", IFunc.getName());
EXPECT_EQ(GlobalValue::LinkOnceODRLinkage, IFunc.getLinkage());
EXPECT_EQ(GlobalValue::ProtectedVisibility, IFunc.getVisibility());
EXPECT_EQ(123u, IFunc.getAddressSpace());

const Function *Resolver = IFunc.getResolverFunction();
ASSERT_NE(nullptr, Resolver);
EXPECT_EQ("resolver", Resolver->getName());
EXPECT_EQ(GlobalValue::PrivateLinkage, Resolver->getLinkage());
}
}

0 comments on commit f069327

Please sign in to comment.