//===- Cloning.cpp - Unit tests for the Cloner ----------------------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/Transforms/Utils/Cloning.h" #include "llvm/ADT/STLExtras.h" #include "llvm/ADT/SmallPtrSet.h" #include "llvm/IR/Argument.h" #include "llvm/IR/Constant.h" #include "llvm/IR/DIBuilder.h" #include "llvm/IR/DebugInfo.h" #include "llvm/IR/Function.h" #include "llvm/IR/IRBuilder.h" #include "llvm/IR/InstIterator.h" #include "llvm/IR/Instructions.h" #include "llvm/IR/IntrinsicInst.h" #include "llvm/IR/LLVMContext.h" #include "llvm/IR/Module.h" #include "llvm/IR/Verifier.h" #include "gtest/gtest.h" using namespace llvm; namespace { class CloneInstruction : public ::testing::Test { protected: void SetUp() override { V = nullptr; } template T *clone(T *V1) { Value *V2 = V1->clone(); Orig.insert(V1); Clones.insert(V2); return cast(V2); } void eraseClones() { for (Value *V : Clones) V->deleteValue(); Clones.clear(); } void TearDown() override { eraseClones(); for (Value *V : Orig) V->deleteValue(); Orig.clear(); if (V) V->deleteValue(); } SmallPtrSet Orig; // Erase on exit SmallPtrSet Clones; // Erase in eraseClones LLVMContext context; Value *V; }; TEST_F(CloneInstruction, OverflowBits) { V = new Argument(Type::getInt32Ty(context)); BinaryOperator *Add = BinaryOperator::Create(Instruction::Add, V, V); BinaryOperator *Sub = BinaryOperator::Create(Instruction::Sub, V, V); BinaryOperator *Mul = BinaryOperator::Create(Instruction::Mul, V, V); BinaryOperator *AddClone = this->clone(Add); BinaryOperator *SubClone = this->clone(Sub); BinaryOperator *MulClone = this->clone(Mul); EXPECT_FALSE(AddClone->hasNoUnsignedWrap()); EXPECT_FALSE(AddClone->hasNoSignedWrap()); EXPECT_FALSE(SubClone->hasNoUnsignedWrap()); EXPECT_FALSE(SubClone->hasNoSignedWrap()); EXPECT_FALSE(MulClone->hasNoUnsignedWrap()); EXPECT_FALSE(MulClone->hasNoSignedWrap()); eraseClones(); Add->setHasNoUnsignedWrap(); Sub->setHasNoUnsignedWrap(); Mul->setHasNoUnsignedWrap(); AddClone = this->clone(Add); SubClone = this->clone(Sub); MulClone = this->clone(Mul); EXPECT_TRUE(AddClone->hasNoUnsignedWrap()); EXPECT_FALSE(AddClone->hasNoSignedWrap()); EXPECT_TRUE(SubClone->hasNoUnsignedWrap()); EXPECT_FALSE(SubClone->hasNoSignedWrap()); EXPECT_TRUE(MulClone->hasNoUnsignedWrap()); EXPECT_FALSE(MulClone->hasNoSignedWrap()); eraseClones(); Add->setHasNoSignedWrap(); Sub->setHasNoSignedWrap(); Mul->setHasNoSignedWrap(); AddClone = this->clone(Add); SubClone = this->clone(Sub); MulClone = this->clone(Mul); EXPECT_TRUE(AddClone->hasNoUnsignedWrap()); EXPECT_TRUE(AddClone->hasNoSignedWrap()); EXPECT_TRUE(SubClone->hasNoUnsignedWrap()); EXPECT_TRUE(SubClone->hasNoSignedWrap()); EXPECT_TRUE(MulClone->hasNoUnsignedWrap()); EXPECT_TRUE(MulClone->hasNoSignedWrap()); eraseClones(); Add->setHasNoUnsignedWrap(false); Sub->setHasNoUnsignedWrap(false); Mul->setHasNoUnsignedWrap(false); AddClone = this->clone(Add); SubClone = this->clone(Sub); MulClone = this->clone(Mul); EXPECT_FALSE(AddClone->hasNoUnsignedWrap()); EXPECT_TRUE(AddClone->hasNoSignedWrap()); EXPECT_FALSE(SubClone->hasNoUnsignedWrap()); EXPECT_TRUE(SubClone->hasNoSignedWrap()); EXPECT_FALSE(MulClone->hasNoUnsignedWrap()); EXPECT_TRUE(MulClone->hasNoSignedWrap()); } TEST_F(CloneInstruction, Inbounds) { V = new Argument(Type::getInt32PtrTy(context)); Constant *Z = Constant::getNullValue(Type::getInt32Ty(context)); std::vector ops; ops.push_back(Z); GetElementPtrInst *GEP = GetElementPtrInst::Create(Type::getInt32Ty(context), V, ops); EXPECT_FALSE(this->clone(GEP)->isInBounds()); GEP->setIsInBounds(); EXPECT_TRUE(this->clone(GEP)->isInBounds()); } TEST_F(CloneInstruction, Exact) { V = new Argument(Type::getInt32Ty(context)); BinaryOperator *SDiv = BinaryOperator::Create(Instruction::SDiv, V, V); EXPECT_FALSE(this->clone(SDiv)->isExact()); SDiv->setIsExact(true); EXPECT_TRUE(this->clone(SDiv)->isExact()); } TEST_F(CloneInstruction, Attributes) { Type *ArgTy1[] = { Type::getInt32PtrTy(context) }; FunctionType *FT1 = FunctionType::get(Type::getVoidTy(context), ArgTy1, false); Function *F1 = Function::Create(FT1, Function::ExternalLinkage); BasicBlock *BB = BasicBlock::Create(context, "", F1); IRBuilder<> Builder(BB); Builder.CreateRetVoid(); Function *F2 = Function::Create(FT1, Function::ExternalLinkage); Argument *A = &*F1->arg_begin(); A->addAttr(Attribute::NoCapture); SmallVector Returns; ValueToValueMapTy VMap; VMap[A] = UndefValue::get(A->getType()); CloneFunctionInto(F2, F1, VMap, false, Returns); EXPECT_FALSE(F2->arg_begin()->hasNoCaptureAttr()); delete F1; delete F2; } TEST_F(CloneInstruction, CallingConvention) { Type *ArgTy1[] = { Type::getInt32PtrTy(context) }; FunctionType *FT1 = FunctionType::get(Type::getVoidTy(context), ArgTy1, false); Function *F1 = Function::Create(FT1, Function::ExternalLinkage); F1->setCallingConv(CallingConv::Cold); BasicBlock *BB = BasicBlock::Create(context, "", F1); IRBuilder<> Builder(BB); Builder.CreateRetVoid(); Function *F2 = Function::Create(FT1, Function::ExternalLinkage); SmallVector Returns; ValueToValueMapTy VMap; VMap[&*F1->arg_begin()] = &*F2->arg_begin(); CloneFunctionInto(F2, F1, VMap, false, Returns); EXPECT_EQ(CallingConv::Cold, F2->getCallingConv()); delete F1; delete F2; } TEST_F(CloneInstruction, DuplicateInstructionsToSplit) { Type *ArgTy1[] = {Type::getInt32PtrTy(context)}; FunctionType *FT = FunctionType::get(Type::getVoidTy(context), ArgTy1, false); V = new Argument(Type::getInt32Ty(context)); Function *F = Function::Create(FT, Function::ExternalLinkage); BasicBlock *BB1 = BasicBlock::Create(context, "", F); IRBuilder<> Builder1(BB1); BasicBlock *BB2 = BasicBlock::Create(context, "", F); IRBuilder<> Builder2(BB2); Builder1.CreateBr(BB2); Instruction *AddInst = cast(Builder2.CreateAdd(V, V)); Instruction *MulInst = cast(Builder2.CreateMul(AddInst, V)); Instruction *SubInst = cast(Builder2.CreateSub(MulInst, V)); Builder2.CreateRetVoid(); ValueToValueMapTy Mapping; auto Split = DuplicateInstructionsInSplitBetween(BB2, BB1, SubInst, Mapping); EXPECT_TRUE(Split); EXPECT_EQ(Mapping.size(), 2u); EXPECT_TRUE(Mapping.find(AddInst) != Mapping.end()); EXPECT_TRUE(Mapping.find(MulInst) != Mapping.end()); auto AddSplit = dyn_cast(Mapping[AddInst]); EXPECT_TRUE(AddSplit); EXPECT_EQ(AddSplit->getOperand(0), V); EXPECT_EQ(AddSplit->getOperand(1), V); EXPECT_EQ(AddSplit->getParent(), Split); auto MulSplit = dyn_cast(Mapping[MulInst]); EXPECT_TRUE(MulSplit); EXPECT_EQ(MulSplit->getOperand(0), AddSplit); EXPECT_EQ(MulSplit->getOperand(1), V); EXPECT_EQ(MulSplit->getParent(), Split); EXPECT_EQ(AddSplit->getNextNode(), MulSplit); EXPECT_EQ(MulSplit->getNextNode(), Split->getTerminator()); delete F; } TEST_F(CloneInstruction, DuplicateInstructionsToSplitBlocksEq1) { Type *ArgTy1[] = {Type::getInt32PtrTy(context)}; FunctionType *FT = FunctionType::get(Type::getVoidTy(context), ArgTy1, false); V = new Argument(Type::getInt32Ty(context)); Function *F = Function::Create(FT, Function::ExternalLinkage); BasicBlock *BB1 = BasicBlock::Create(context, "", F); IRBuilder<> Builder1(BB1); BasicBlock *BB2 = BasicBlock::Create(context, "", F); IRBuilder<> Builder2(BB2); Builder1.CreateBr(BB2); Instruction *AddInst = cast(Builder2.CreateAdd(V, V)); Instruction *MulInst = cast(Builder2.CreateMul(AddInst, V)); Instruction *SubInst = cast(Builder2.CreateSub(MulInst, V)); Builder2.CreateBr(BB2); ValueToValueMapTy Mapping; auto Split = DuplicateInstructionsInSplitBetween(BB2, BB2, BB2->getTerminator(), Mapping); EXPECT_TRUE(Split); EXPECT_EQ(Mapping.size(), 3u); EXPECT_TRUE(Mapping.find(AddInst) != Mapping.end()); EXPECT_TRUE(Mapping.find(MulInst) != Mapping.end()); EXPECT_TRUE(Mapping.find(SubInst) != Mapping.end()); auto AddSplit = dyn_cast(Mapping[AddInst]); EXPECT_TRUE(AddSplit); EXPECT_EQ(AddSplit->getOperand(0), V); EXPECT_EQ(AddSplit->getOperand(1), V); EXPECT_EQ(AddSplit->getParent(), Split); auto MulSplit = dyn_cast(Mapping[MulInst]); EXPECT_TRUE(MulSplit); EXPECT_EQ(MulSplit->getOperand(0), AddSplit); EXPECT_EQ(MulSplit->getOperand(1), V); EXPECT_EQ(MulSplit->getParent(), Split); auto SubSplit = dyn_cast(Mapping[SubInst]); EXPECT_EQ(MulSplit->getNextNode(), SubSplit); EXPECT_EQ(SubSplit->getNextNode(), Split->getTerminator()); EXPECT_EQ(Split->getSingleSuccessor(), BB2); EXPECT_EQ(BB2->getSingleSuccessor(), Split); delete F; } TEST_F(CloneInstruction, DuplicateInstructionsToSplitBlocksEq2) { Type *ArgTy1[] = {Type::getInt32PtrTy(context)}; FunctionType *FT = FunctionType::get(Type::getVoidTy(context), ArgTy1, false); V = new Argument(Type::getInt32Ty(context)); Function *F = Function::Create(FT, Function::ExternalLinkage); BasicBlock *BB1 = BasicBlock::Create(context, "", F); IRBuilder<> Builder1(BB1); BasicBlock *BB2 = BasicBlock::Create(context, "", F); IRBuilder<> Builder2(BB2); Builder1.CreateBr(BB2); Instruction *AddInst = cast(Builder2.CreateAdd(V, V)); Instruction *MulInst = cast(Builder2.CreateMul(AddInst, V)); Instruction *SubInst = cast(Builder2.CreateSub(MulInst, V)); Builder2.CreateBr(BB2); ValueToValueMapTy Mapping; auto Split = DuplicateInstructionsInSplitBetween(BB2, BB2, SubInst, Mapping); EXPECT_TRUE(Split); EXPECT_EQ(Mapping.size(), 2u); EXPECT_TRUE(Mapping.find(AddInst) != Mapping.end()); EXPECT_TRUE(Mapping.find(MulInst) != Mapping.end()); auto AddSplit = dyn_cast(Mapping[AddInst]); EXPECT_TRUE(AddSplit); EXPECT_EQ(AddSplit->getOperand(0), V); EXPECT_EQ(AddSplit->getOperand(1), V); EXPECT_EQ(AddSplit->getParent(), Split); auto MulSplit = dyn_cast(Mapping[MulInst]); EXPECT_TRUE(MulSplit); EXPECT_EQ(MulSplit->getOperand(0), AddSplit); EXPECT_EQ(MulSplit->getOperand(1), V); EXPECT_EQ(MulSplit->getParent(), Split); EXPECT_EQ(MulSplit->getNextNode(), Split->getTerminator()); EXPECT_EQ(Split->getSingleSuccessor(), BB2); EXPECT_EQ(BB2->getSingleSuccessor(), Split); delete F; } class CloneFunc : public ::testing::Test { protected: void SetUp() override { SetupModule(); CreateOldFunc(); CreateNewFunc(); SetupFinder(); } void TearDown() override { delete Finder; } void SetupModule() { M = new Module("", C); } void CreateOldFunc() { FunctionType* FuncType = FunctionType::get(Type::getVoidTy(C), false); OldFunc = Function::Create(FuncType, GlobalValue::PrivateLinkage, "f", M); CreateOldFunctionBodyAndDI(); } void CreateOldFunctionBodyAndDI() { DIBuilder DBuilder(*M); IRBuilder<> IBuilder(C); // Function DI auto *File = DBuilder.createFile("filename.c", "/file/dir/"); DITypeRefArray ParamTypes = DBuilder.getOrCreateTypeArray(None); DISubroutineType *FuncType = DBuilder.createSubroutineType(ParamTypes); auto *CU = DBuilder.createCompileUnit(dwarf::DW_LANG_C99, DBuilder.createFile("filename.c", "/file/dir"), "CloneFunc", false, "", 0); auto *Subprogram = DBuilder.createFunction(CU, "f", "f", File, 4, FuncType, true, true, 3, DINode::FlagZero, false); OldFunc->setSubprogram(Subprogram); // Function body BasicBlock* Entry = BasicBlock::Create(C, "", OldFunc); IBuilder.SetInsertPoint(Entry); DebugLoc Loc = DebugLoc::get(3, 2, Subprogram); IBuilder.SetCurrentDebugLocation(Loc); AllocaInst* Alloca = IBuilder.CreateAlloca(IntegerType::getInt32Ty(C)); IBuilder.SetCurrentDebugLocation(DebugLoc::get(4, 2, Subprogram)); Value* AllocaContent = IBuilder.getInt32(1); Instruction* Store = IBuilder.CreateStore(AllocaContent, Alloca); IBuilder.SetCurrentDebugLocation(DebugLoc::get(5, 2, Subprogram)); // Create a local variable around the alloca auto *IntType = DBuilder.createBasicType("int", 32, dwarf::DW_ATE_signed); auto *E = DBuilder.createExpression(); auto *Variable = DBuilder.createAutoVariable(Subprogram, "x", File, 5, IntType, true); auto *DL = DILocation::get(Subprogram->getContext(), 5, 0, Subprogram); DBuilder.insertDeclare(Alloca, Variable, E, DL, Store); DBuilder.insertDbgValueIntrinsic(AllocaContent, Variable, E, DL, Entry); // Also create an inlined variable. // Create a distinct struct type that we should not duplicate during // cloning). auto *StructType = DICompositeType::getDistinct( C, dwarf::DW_TAG_structure_type, "some_struct", nullptr, 0, nullptr, nullptr, 32, 32, 0, DINode::FlagZero, nullptr, 0, nullptr, nullptr); auto *InlinedSP = DBuilder.createFunction(CU, "inlined", "inlined", File, 8, FuncType, true, true, 9, DINode::FlagZero, false); auto *InlinedVar = DBuilder.createAutoVariable(InlinedSP, "inlined", File, 5, StructType, true); auto *Scope = DBuilder.createLexicalBlock( DBuilder.createLexicalBlockFile(InlinedSP, File), File, 1, 1); auto InlinedDL = DebugLoc::get(9, 4, Scope, DebugLoc::get(5, 2, Subprogram)); IBuilder.SetCurrentDebugLocation(InlinedDL); DBuilder.insertDeclare(Alloca, InlinedVar, E, InlinedDL, Store); IBuilder.CreateStore(IBuilder.getInt32(2), Alloca); // Finalize the debug info. DBuilder.finalize(); IBuilder.CreateRetVoid(); // Create another, empty, compile unit. DIBuilder DBuilder2(*M); DBuilder2.createCompileUnit(dwarf::DW_LANG_C99, DBuilder.createFile("extra.c", "/file/dir"), "CloneFunc", false, "", 0); DBuilder2.finalize(); } void CreateNewFunc() { ValueToValueMapTy VMap; NewFunc = CloneFunction(OldFunc, VMap, nullptr); } void SetupFinder() { Finder = new DebugInfoFinder(); Finder->processModule(*M); } LLVMContext C; Function* OldFunc; Function* NewFunc; Module* M; DebugInfoFinder* Finder; }; // Test that a new, distinct function was created. TEST_F(CloneFunc, NewFunctionCreated) { EXPECT_NE(OldFunc, NewFunc); } // Test that a new subprogram entry was added and is pointing to the new // function, while the original subprogram still points to the old one. TEST_F(CloneFunc, Subprogram) { EXPECT_FALSE(verifyModule(*M, &errs())); EXPECT_EQ(3U, Finder->subprogram_count()); EXPECT_NE(NewFunc->getSubprogram(), OldFunc->getSubprogram()); } // Test that instructions in the old function still belong to it in the // metadata, while instruction in the new function belong to the new one. TEST_F(CloneFunc, InstructionOwnership) { EXPECT_FALSE(verifyModule(*M)); inst_iterator OldIter = inst_begin(OldFunc); inst_iterator OldEnd = inst_end(OldFunc); inst_iterator NewIter = inst_begin(NewFunc); inst_iterator NewEnd = inst_end(NewFunc); while (OldIter != OldEnd && NewIter != NewEnd) { Instruction& OldI = *OldIter; Instruction& NewI = *NewIter; EXPECT_NE(&OldI, &NewI); EXPECT_EQ(OldI.hasMetadata(), NewI.hasMetadata()); if (OldI.hasMetadata()) { const DebugLoc& OldDL = OldI.getDebugLoc(); const DebugLoc& NewDL = NewI.getDebugLoc(); // Verify that the debug location data is the same EXPECT_EQ(OldDL.getLine(), NewDL.getLine()); EXPECT_EQ(OldDL.getCol(), NewDL.getCol()); // But that they belong to different functions auto *OldSubprogram = cast(OldDL.getInlinedAtScope()); auto *NewSubprogram = cast(NewDL.getInlinedAtScope()); EXPECT_EQ(OldFunc->getSubprogram(), OldSubprogram); EXPECT_EQ(NewFunc->getSubprogram(), NewSubprogram); } ++OldIter; ++NewIter; } EXPECT_EQ(OldEnd, OldIter); EXPECT_EQ(NewEnd, NewIter); } // Test that the arguments for debug intrinsics in the new function were // properly cloned TEST_F(CloneFunc, DebugIntrinsics) { EXPECT_FALSE(verifyModule(*M)); inst_iterator OldIter = inst_begin(OldFunc); inst_iterator OldEnd = inst_end(OldFunc); inst_iterator NewIter = inst_begin(NewFunc); inst_iterator NewEnd = inst_end(NewFunc); while (OldIter != OldEnd && NewIter != NewEnd) { Instruction& OldI = *OldIter; Instruction& NewI = *NewIter; if (DbgDeclareInst* OldIntrin = dyn_cast(&OldI)) { DbgDeclareInst* NewIntrin = dyn_cast(&NewI); EXPECT_TRUE(NewIntrin); // Old address must belong to the old function EXPECT_EQ(OldFunc, cast(OldIntrin->getAddress())-> getParent()->getParent()); // New address must belong to the new function EXPECT_EQ(NewFunc, cast(NewIntrin->getAddress())-> getParent()->getParent()); if (OldIntrin->getDebugLoc()->getInlinedAt()) { // Inlined variable should refer to the same DILocalVariable as in the // Old Function EXPECT_EQ(OldIntrin->getVariable(), NewIntrin->getVariable()); } else { // Old variable must belong to the old function. EXPECT_EQ(OldFunc->getSubprogram(), cast(OldIntrin->getVariable()->getScope())); // New variable must belong to the new function. EXPECT_EQ(NewFunc->getSubprogram(), cast(NewIntrin->getVariable()->getScope())); } } else if (DbgValueInst* OldIntrin = dyn_cast(&OldI)) { DbgValueInst* NewIntrin = dyn_cast(&NewI); EXPECT_TRUE(NewIntrin); if (!OldIntrin->getDebugLoc()->getInlinedAt()) { // Old variable must belong to the old function. EXPECT_EQ(OldFunc->getSubprogram(), cast(OldIntrin->getVariable()->getScope())); // New variable must belong to the new function. EXPECT_EQ(NewFunc->getSubprogram(), cast(NewIntrin->getVariable()->getScope())); } } ++OldIter; ++NewIter; } } class CloneModule : public ::testing::Test { protected: void SetUp() override { SetupModule(); CreateOldModule(); CreateNewModule(); } void SetupModule() { OldM = new Module("", C); } void CreateOldModule() { auto *CD = OldM->getOrInsertComdat("comdat"); CD->setSelectionKind(Comdat::ExactMatch); auto GV = new GlobalVariable( *OldM, Type::getInt32Ty(C), false, GlobalValue::ExternalLinkage, ConstantInt::get(Type::getInt32Ty(C), 1), "gv"); GV->addMetadata(LLVMContext::MD_type, *MDNode::get(C, {})); GV->setComdat(CD); DIBuilder DBuilder(*OldM); IRBuilder<> IBuilder(C); auto *FuncType = FunctionType::get(Type::getVoidTy(C), false); auto *PersFn = Function::Create(FuncType, GlobalValue::ExternalLinkage, "persfn", OldM); auto *F = Function::Create(FuncType, GlobalValue::PrivateLinkage, "f", OldM); F->setPersonalityFn(PersFn); F->setComdat(CD); // Create debug info auto *File = DBuilder.createFile("filename.c", "/file/dir/"); DITypeRefArray ParamTypes = DBuilder.getOrCreateTypeArray(None); DISubroutineType *DFuncType = DBuilder.createSubroutineType(ParamTypes); auto *CU = DBuilder.createCompileUnit(dwarf::DW_LANG_C99, DBuilder.createFile("filename.c", "/file/dir"), "CloneModule", false, "", 0); // Function DI auto *Subprogram = DBuilder.createFunction(CU, "f", "f", File, 4, DFuncType, true, true, 3, DINode::FlagZero, false); F->setSubprogram(Subprogram); // Create and assign DIGlobalVariableExpression to gv auto GVExpression = DBuilder.createGlobalVariableExpression( Subprogram, "gv", "gv", File, 1, DBuilder.createNullPtrType(), false); GV->addDebugInfo(GVExpression); // DIGlobalVariableExpression not attached to any global variable auto Expr = DBuilder.createExpression( ArrayRef{dwarf::DW_OP_constu, 42U, dwarf::DW_OP_stack_value}); DBuilder.createGlobalVariableExpression( Subprogram, "unattached", "unattached", File, 1, DBuilder.createNullPtrType(), false, Expr); auto *Entry = BasicBlock::Create(C, "", F); IBuilder.SetInsertPoint(Entry); IBuilder.CreateRetVoid(); // Finalize the debug info DBuilder.finalize(); } void CreateNewModule() { NewM = llvm::CloneModule(*OldM).release(); } LLVMContext C; Module *OldM; Module *NewM; }; TEST_F(CloneModule, Verify) { EXPECT_FALSE(verifyModule(*NewM)); } TEST_F(CloneModule, OldModuleUnchanged) { DebugInfoFinder Finder; Finder.processModule(*OldM); EXPECT_EQ(1U, Finder.subprogram_count()); } TEST_F(CloneModule, Subprogram) { Function *NewF = NewM->getFunction("f"); DISubprogram *SP = NewF->getSubprogram(); EXPECT_TRUE(SP != nullptr); EXPECT_EQ(SP->getName(), "f"); EXPECT_EQ(SP->getFile()->getFilename(), "filename.c"); EXPECT_EQ(SP->getLine(), (unsigned)4); } TEST_F(CloneModule, GlobalMetadata) { GlobalVariable *NewGV = NewM->getGlobalVariable("gv"); EXPECT_NE(nullptr, NewGV->getMetadata(LLVMContext::MD_type)); } TEST_F(CloneModule, GlobalDebugInfo) { GlobalVariable *NewGV = NewM->getGlobalVariable("gv"); EXPECT_TRUE(NewGV != nullptr); // Find debug info expression assigned to global SmallVector GVs; NewGV->getDebugInfo(GVs); EXPECT_EQ(GVs.size(), 1U); DIGlobalVariableExpression *GVExpr = GVs[0]; DIGlobalVariable *GV = GVExpr->getVariable(); EXPECT_TRUE(GV != nullptr); EXPECT_EQ(GV->getName(), "gv"); EXPECT_EQ(GV->getLine(), 1U); // Assert that the scope of the debug info attached to // global variable matches the cloned function. DISubprogram *SP = NewM->getFunction("f")->getSubprogram(); EXPECT_TRUE(SP != nullptr); EXPECT_EQ(GV->getScope(), SP); } TEST_F(CloneModule, CompileUnit) { // Find DICompileUnit listed in llvm.dbg.cu auto *NMD = NewM->getNamedMetadata("llvm.dbg.cu"); EXPECT_TRUE(NMD != nullptr); EXPECT_EQ(NMD->getNumOperands(), 1U); DICompileUnit *CU = dyn_cast(NMD->getOperand(0)); EXPECT_TRUE(CU != nullptr); // Assert this CU is consistent with the cloned function debug info DISubprogram *SP = NewM->getFunction("f")->getSubprogram(); EXPECT_TRUE(SP != nullptr); EXPECT_EQ(SP->getUnit(), CU); // Check globals listed in CU have the correct scope DIGlobalVariableExpressionArray GlobalArray = CU->getGlobalVariables(); EXPECT_EQ(GlobalArray.size(), 2U); for (DIGlobalVariableExpression *GVExpr : GlobalArray) { DIGlobalVariable *GV = GVExpr->getVariable(); EXPECT_EQ(GV->getScope(), SP); } } TEST_F(CloneModule, Comdat) { GlobalVariable *NewGV = NewM->getGlobalVariable("gv"); auto *CD = NewGV->getComdat(); ASSERT_NE(nullptr, CD); EXPECT_EQ("comdat", CD->getName()); EXPECT_EQ(Comdat::ExactMatch, CD->getSelectionKind()); Function *NewF = NewM->getFunction("f"); EXPECT_EQ(CD, NewF->getComdat()); } }