| //===- unittests/IR/MetadataTest.cpp - Metadata unit tests ----------------===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "llvm/IR/Metadata.h" |
| #include "llvm/ADT/STLExtras.h" |
| #include "llvm/IR/Constants.h" |
| #include "llvm/IR/DebugInfo.h" |
| #include "llvm/IR/DebugInfoMetadata.h" |
| #include "llvm/IR/Function.h" |
| #include "llvm/IR/Instructions.h" |
| #include "llvm/IR/LLVMContext.h" |
| #include "llvm/IR/Module.h" |
| #include "llvm/IR/ModuleSlotTracker.h" |
| #include "llvm/IR/Type.h" |
| #include "llvm/IR/Verifier.h" |
| #include "llvm/Support/raw_ostream.h" |
| #include "gtest/gtest.h" |
| using namespace llvm; |
| |
| namespace { |
| |
| TEST(ContextAndReplaceableUsesTest, FromContext) { |
| LLVMContext Context; |
| ContextAndReplaceableUses CRU(Context); |
| EXPECT_EQ(&Context, &CRU.getContext()); |
| EXPECT_FALSE(CRU.hasReplaceableUses()); |
| EXPECT_FALSE(CRU.getReplaceableUses()); |
| } |
| |
| TEST(ContextAndReplaceableUsesTest, FromReplaceableUses) { |
| LLVMContext Context; |
| ContextAndReplaceableUses CRU(make_unique<ReplaceableMetadataImpl>(Context)); |
| EXPECT_EQ(&Context, &CRU.getContext()); |
| EXPECT_TRUE(CRU.hasReplaceableUses()); |
| EXPECT_TRUE(CRU.getReplaceableUses()); |
| } |
| |
| TEST(ContextAndReplaceableUsesTest, makeReplaceable) { |
| LLVMContext Context; |
| ContextAndReplaceableUses CRU(Context); |
| CRU.makeReplaceable(make_unique<ReplaceableMetadataImpl>(Context)); |
| EXPECT_EQ(&Context, &CRU.getContext()); |
| EXPECT_TRUE(CRU.hasReplaceableUses()); |
| EXPECT_TRUE(CRU.getReplaceableUses()); |
| } |
| |
| TEST(ContextAndReplaceableUsesTest, takeReplaceableUses) { |
| LLVMContext Context; |
| auto ReplaceableUses = make_unique<ReplaceableMetadataImpl>(Context); |
| auto *Ptr = ReplaceableUses.get(); |
| ContextAndReplaceableUses CRU(std::move(ReplaceableUses)); |
| ReplaceableUses = CRU.takeReplaceableUses(); |
| EXPECT_EQ(&Context, &CRU.getContext()); |
| EXPECT_FALSE(CRU.hasReplaceableUses()); |
| EXPECT_FALSE(CRU.getReplaceableUses()); |
| EXPECT_EQ(Ptr, ReplaceableUses.get()); |
| } |
| |
| class MetadataTest : public testing::Test { |
| public: |
| MetadataTest() : M("test", Context), Counter(0) {} |
| |
| protected: |
| LLVMContext Context; |
| Module M; |
| int Counter; |
| |
| MDNode *getNode() { return MDNode::get(Context, None); } |
| MDNode *getNode(Metadata *MD) { return MDNode::get(Context, MD); } |
| MDNode *getNode(Metadata *MD1, Metadata *MD2) { |
| Metadata *MDs[] = {MD1, MD2}; |
| return MDNode::get(Context, MDs); |
| } |
| |
| MDTuple *getTuple() { return MDTuple::getDistinct(Context, None); } |
| DISubroutineType *getSubroutineType() { |
| return DISubroutineType::getDistinct(Context, DINode::FlagZero, 0, |
| getNode(nullptr)); |
| } |
| DISubprogram *getSubprogram() { |
| return DISubprogram::getDistinct(Context, nullptr, "", "", nullptr, 0, |
| nullptr, false, false, 0, nullptr, 0, 0, 0, |
| DINode::FlagZero, false, nullptr); |
| } |
| DIFile *getFile() { |
| return DIFile::getDistinct(Context, "file.c", "/path/to/dir"); |
| } |
| DICompileUnit *getUnit() { |
| return DICompileUnit::getDistinct( |
| Context, 1, getFile(), "clang", false, "-g", 2, "", |
| DICompileUnit::FullDebug, getTuple(), getTuple(), getTuple(), |
| getTuple(), getTuple(), 0, true, false, false); |
| } |
| DIType *getBasicType(StringRef Name) { |
| return DIBasicType::get(Context, dwarf::DW_TAG_unspecified_type, Name); |
| } |
| DIType *getDerivedType() { |
| return DIDerivedType::getDistinct( |
| Context, dwarf::DW_TAG_pointer_type, "", nullptr, 0, nullptr, |
| getBasicType("basictype"), 1, 2, 0, None, DINode::FlagZero); |
| } |
| Constant *getConstant() { |
| return ConstantInt::get(Type::getInt32Ty(Context), Counter++); |
| } |
| ConstantAsMetadata *getConstantAsMetadata() { |
| return ConstantAsMetadata::get(getConstant()); |
| } |
| DIType *getCompositeType() { |
| return DICompositeType::getDistinct( |
| Context, dwarf::DW_TAG_structure_type, "", nullptr, 0, nullptr, nullptr, |
| 32, 32, 0, DINode::FlagZero, nullptr, 0, nullptr, nullptr, ""); |
| } |
| Function *getFunction(StringRef Name) { |
| return cast<Function>(M.getOrInsertFunction( |
| Name, FunctionType::get(Type::getVoidTy(Context), None, false))); |
| } |
| }; |
| typedef MetadataTest MDStringTest; |
| |
| // Test that construction of MDString with different value produces different |
| // MDString objects, even with the same string pointer and nulls in the string. |
| TEST_F(MDStringTest, CreateDifferent) { |
| char x[3] = { 'f', 0, 'A' }; |
| MDString *s1 = MDString::get(Context, StringRef(&x[0], 3)); |
| x[2] = 'B'; |
| MDString *s2 = MDString::get(Context, StringRef(&x[0], 3)); |
| EXPECT_NE(s1, s2); |
| } |
| |
| // Test that creation of MDStrings with the same string contents produces the |
| // same MDString object, even with different pointers. |
| TEST_F(MDStringTest, CreateSame) { |
| char x[4] = { 'a', 'b', 'c', 'X' }; |
| char y[4] = { 'a', 'b', 'c', 'Y' }; |
| |
| MDString *s1 = MDString::get(Context, StringRef(&x[0], 3)); |
| MDString *s2 = MDString::get(Context, StringRef(&y[0], 3)); |
| EXPECT_EQ(s1, s2); |
| } |
| |
| // Test that MDString prints out the string we fed it. |
| TEST_F(MDStringTest, PrintingSimple) { |
| char *str = new char[13]; |
| strncpy(str, "testing 1 2 3", 13); |
| MDString *s = MDString::get(Context, StringRef(str, 13)); |
| strncpy(str, "aaaaaaaaaaaaa", 13); |
| delete[] str; |
| |
| std::string Str; |
| raw_string_ostream oss(Str); |
| s->print(oss); |
| EXPECT_STREQ("!\"testing 1 2 3\"", oss.str().c_str()); |
| } |
| |
| // Test printing of MDString with non-printable characters. |
| TEST_F(MDStringTest, PrintingComplex) { |
| char str[5] = {0, '\n', '"', '\\', (char)-1}; |
| MDString *s = MDString::get(Context, StringRef(str+0, 5)); |
| std::string Str; |
| raw_string_ostream oss(Str); |
| s->print(oss); |
| EXPECT_STREQ("!\"\\00\\0A\\22\\5C\\FF\"", oss.str().c_str()); |
| } |
| |
| typedef MetadataTest MDNodeTest; |
| |
| // Test the two constructors, and containing other Constants. |
| TEST_F(MDNodeTest, Simple) { |
| char x[3] = { 'a', 'b', 'c' }; |
| char y[3] = { '1', '2', '3' }; |
| |
| MDString *s1 = MDString::get(Context, StringRef(&x[0], 3)); |
| MDString *s2 = MDString::get(Context, StringRef(&y[0], 3)); |
| ConstantAsMetadata *CI = |
| ConstantAsMetadata::get(ConstantInt::get(Context, APInt(8, 0))); |
| |
| std::vector<Metadata *> V; |
| V.push_back(s1); |
| V.push_back(CI); |
| V.push_back(s2); |
| |
| MDNode *n1 = MDNode::get(Context, V); |
| Metadata *const c1 = n1; |
| MDNode *n2 = MDNode::get(Context, c1); |
| Metadata *const c2 = n2; |
| MDNode *n3 = MDNode::get(Context, V); |
| MDNode *n4 = MDNode::getIfExists(Context, V); |
| MDNode *n5 = MDNode::getIfExists(Context, c1); |
| MDNode *n6 = MDNode::getIfExists(Context, c2); |
| EXPECT_NE(n1, n2); |
| EXPECT_EQ(n1, n3); |
| EXPECT_EQ(n4, n1); |
| EXPECT_EQ(n5, n2); |
| EXPECT_EQ(n6, (Metadata *)nullptr); |
| |
| EXPECT_EQ(3u, n1->getNumOperands()); |
| EXPECT_EQ(s1, n1->getOperand(0)); |
| EXPECT_EQ(CI, n1->getOperand(1)); |
| EXPECT_EQ(s2, n1->getOperand(2)); |
| |
| EXPECT_EQ(1u, n2->getNumOperands()); |
| EXPECT_EQ(n1, n2->getOperand(0)); |
| } |
| |
| TEST_F(MDNodeTest, Delete) { |
| Constant *C = ConstantInt::get(Type::getInt32Ty(Context), 1); |
| Instruction *I = new BitCastInst(C, Type::getInt32Ty(Context)); |
| |
| Metadata *const V = LocalAsMetadata::get(I); |
| MDNode *n = MDNode::get(Context, V); |
| TrackingMDRef wvh(n); |
| |
| EXPECT_EQ(n, wvh); |
| |
| I->deleteValue(); |
| } |
| |
| TEST_F(MDNodeTest, SelfReference) { |
| // !0 = !{!0} |
| // !1 = !{!0} |
| { |
| auto Temp = MDNode::getTemporary(Context, None); |
| Metadata *Args[] = {Temp.get()}; |
| MDNode *Self = MDNode::get(Context, Args); |
| Self->replaceOperandWith(0, Self); |
| ASSERT_EQ(Self, Self->getOperand(0)); |
| |
| // Self-references should be distinct, so MDNode::get() should grab a |
| // uniqued node that references Self, not Self. |
| Args[0] = Self; |
| MDNode *Ref1 = MDNode::get(Context, Args); |
| MDNode *Ref2 = MDNode::get(Context, Args); |
| EXPECT_NE(Self, Ref1); |
| EXPECT_EQ(Ref1, Ref2); |
| } |
| |
| // !0 = !{!0, !{}} |
| // !1 = !{!0, !{}} |
| { |
| auto Temp = MDNode::getTemporary(Context, None); |
| Metadata *Args[] = {Temp.get(), MDNode::get(Context, None)}; |
| MDNode *Self = MDNode::get(Context, Args); |
| Self->replaceOperandWith(0, Self); |
| ASSERT_EQ(Self, Self->getOperand(0)); |
| |
| // Self-references should be distinct, so MDNode::get() should grab a |
| // uniqued node that references Self, not Self itself. |
| Args[0] = Self; |
| MDNode *Ref1 = MDNode::get(Context, Args); |
| MDNode *Ref2 = MDNode::get(Context, Args); |
| EXPECT_NE(Self, Ref1); |
| EXPECT_EQ(Ref1, Ref2); |
| } |
| } |
| |
| TEST_F(MDNodeTest, Print) { |
| Constant *C = ConstantInt::get(Type::getInt32Ty(Context), 7); |
| MDString *S = MDString::get(Context, "foo"); |
| MDNode *N0 = getNode(); |
| MDNode *N1 = getNode(N0); |
| MDNode *N2 = getNode(N0, N1); |
| |
| Metadata *Args[] = {ConstantAsMetadata::get(C), S, nullptr, N0, N1, N2}; |
| MDNode *N = MDNode::get(Context, Args); |
| |
| std::string Expected; |
| { |
| raw_string_ostream OS(Expected); |
| OS << "<" << (void *)N << "> = !{"; |
| C->printAsOperand(OS); |
| OS << ", "; |
| S->printAsOperand(OS); |
| OS << ", null"; |
| MDNode *Nodes[] = {N0, N1, N2}; |
| for (auto *Node : Nodes) |
| OS << ", <" << (void *)Node << ">"; |
| OS << "}"; |
| } |
| |
| std::string Actual; |
| { |
| raw_string_ostream OS(Actual); |
| N->print(OS); |
| } |
| |
| EXPECT_EQ(Expected, Actual); |
| } |
| |
| #define EXPECT_PRINTER_EQ(EXPECTED, PRINT) \ |
| do { \ |
| std::string Actual_; \ |
| raw_string_ostream OS(Actual_); \ |
| PRINT; \ |
| OS.flush(); \ |
| std::string Expected_(EXPECTED); \ |
| EXPECT_EQ(Expected_, Actual_); \ |
| } while (false) |
| |
| TEST_F(MDNodeTest, PrintTemporary) { |
| MDNode *Arg = getNode(); |
| TempMDNode Temp = MDNode::getTemporary(Context, Arg); |
| MDNode *N = getNode(Temp.get()); |
| Module M("test", Context); |
| NamedMDNode *NMD = M.getOrInsertNamedMetadata("named"); |
| NMD->addOperand(N); |
| |
| EXPECT_PRINTER_EQ("!0 = !{!1}", N->print(OS, &M)); |
| EXPECT_PRINTER_EQ("!1 = <temporary!> !{!2}", Temp->print(OS, &M)); |
| EXPECT_PRINTER_EQ("!2 = !{}", Arg->print(OS, &M)); |
| |
| // Cleanup. |
| Temp->replaceAllUsesWith(Arg); |
| } |
| |
| TEST_F(MDNodeTest, PrintFromModule) { |
| Constant *C = ConstantInt::get(Type::getInt32Ty(Context), 7); |
| MDString *S = MDString::get(Context, "foo"); |
| MDNode *N0 = getNode(); |
| MDNode *N1 = getNode(N0); |
| MDNode *N2 = getNode(N0, N1); |
| |
| Metadata *Args[] = {ConstantAsMetadata::get(C), S, nullptr, N0, N1, N2}; |
| MDNode *N = MDNode::get(Context, Args); |
| Module M("test", Context); |
| NamedMDNode *NMD = M.getOrInsertNamedMetadata("named"); |
| NMD->addOperand(N); |
| |
| std::string Expected; |
| { |
| raw_string_ostream OS(Expected); |
| OS << "!0 = !{"; |
| C->printAsOperand(OS); |
| OS << ", "; |
| S->printAsOperand(OS); |
| OS << ", null, !1, !2, !3}"; |
| } |
| |
| EXPECT_PRINTER_EQ(Expected, N->print(OS, &M)); |
| } |
| |
| TEST_F(MDNodeTest, PrintFromFunction) { |
| Module M("test", Context); |
| auto *FTy = FunctionType::get(Type::getVoidTy(Context), false); |
| auto *F0 = Function::Create(FTy, GlobalValue::ExternalLinkage, "F0", &M); |
| auto *F1 = Function::Create(FTy, GlobalValue::ExternalLinkage, "F1", &M); |
| auto *BB0 = BasicBlock::Create(Context, "entry", F0); |
| auto *BB1 = BasicBlock::Create(Context, "entry", F1); |
| auto *R0 = ReturnInst::Create(Context, BB0); |
| auto *R1 = ReturnInst::Create(Context, BB1); |
| auto *N0 = MDNode::getDistinct(Context, None); |
| auto *N1 = MDNode::getDistinct(Context, None); |
| R0->setMetadata("md", N0); |
| R1->setMetadata("md", N1); |
| |
| EXPECT_PRINTER_EQ("!0 = distinct !{}", N0->print(OS, &M)); |
| EXPECT_PRINTER_EQ("!1 = distinct !{}", N1->print(OS, &M)); |
| |
| ModuleSlotTracker MST(&M); |
| EXPECT_PRINTER_EQ("!0 = distinct !{}", N0->print(OS, MST)); |
| EXPECT_PRINTER_EQ("!1 = distinct !{}", N1->print(OS, MST)); |
| } |
| |
| TEST_F(MDNodeTest, PrintFromMetadataAsValue) { |
| Module M("test", Context); |
| |
| auto *Intrinsic = |
| Function::Create(FunctionType::get(Type::getVoidTy(Context), |
| Type::getMetadataTy(Context), false), |
| GlobalValue::ExternalLinkage, "llvm.intrinsic", &M); |
| |
| auto *FTy = FunctionType::get(Type::getVoidTy(Context), false); |
| auto *F0 = Function::Create(FTy, GlobalValue::ExternalLinkage, "F0", &M); |
| auto *F1 = Function::Create(FTy, GlobalValue::ExternalLinkage, "F1", &M); |
| auto *BB0 = BasicBlock::Create(Context, "entry", F0); |
| auto *BB1 = BasicBlock::Create(Context, "entry", F1); |
| auto *N0 = MDNode::getDistinct(Context, None); |
| auto *N1 = MDNode::getDistinct(Context, None); |
| auto *MAV0 = MetadataAsValue::get(Context, N0); |
| auto *MAV1 = MetadataAsValue::get(Context, N1); |
| CallInst::Create(Intrinsic, MAV0, "", BB0); |
| CallInst::Create(Intrinsic, MAV1, "", BB1); |
| |
| EXPECT_PRINTER_EQ("!0 = distinct !{}", MAV0->print(OS)); |
| EXPECT_PRINTER_EQ("!1 = distinct !{}", MAV1->print(OS)); |
| EXPECT_PRINTER_EQ("!0", MAV0->printAsOperand(OS, false)); |
| EXPECT_PRINTER_EQ("!1", MAV1->printAsOperand(OS, false)); |
| EXPECT_PRINTER_EQ("metadata !0", MAV0->printAsOperand(OS, true)); |
| EXPECT_PRINTER_EQ("metadata !1", MAV1->printAsOperand(OS, true)); |
| |
| ModuleSlotTracker MST(&M); |
| EXPECT_PRINTER_EQ("!0 = distinct !{}", MAV0->print(OS, MST)); |
| EXPECT_PRINTER_EQ("!1 = distinct !{}", MAV1->print(OS, MST)); |
| EXPECT_PRINTER_EQ("!0", MAV0->printAsOperand(OS, false, MST)); |
| EXPECT_PRINTER_EQ("!1", MAV1->printAsOperand(OS, false, MST)); |
| EXPECT_PRINTER_EQ("metadata !0", MAV0->printAsOperand(OS, true, MST)); |
| EXPECT_PRINTER_EQ("metadata !1", MAV1->printAsOperand(OS, true, MST)); |
| } |
| #undef EXPECT_PRINTER_EQ |
| |
| TEST_F(MDNodeTest, NullOperand) { |
| // metadata !{} |
| MDNode *Empty = MDNode::get(Context, None); |
| |
| // metadata !{metadata !{}} |
| Metadata *Ops[] = {Empty}; |
| MDNode *N = MDNode::get(Context, Ops); |
| ASSERT_EQ(Empty, N->getOperand(0)); |
| |
| // metadata !{metadata !{}} => metadata !{null} |
| N->replaceOperandWith(0, nullptr); |
| ASSERT_EQ(nullptr, N->getOperand(0)); |
| |
| // metadata !{null} |
| Ops[0] = nullptr; |
| MDNode *NullOp = MDNode::get(Context, Ops); |
| ASSERT_EQ(nullptr, NullOp->getOperand(0)); |
| EXPECT_EQ(N, NullOp); |
| } |
| |
| TEST_F(MDNodeTest, DistinctOnUniquingCollision) { |
| // !{} |
| MDNode *Empty = MDNode::get(Context, None); |
| ASSERT_TRUE(Empty->isResolved()); |
| EXPECT_FALSE(Empty->isDistinct()); |
| |
| // !{!{}} |
| Metadata *Wrapped1Ops[] = {Empty}; |
| MDNode *Wrapped1 = MDNode::get(Context, Wrapped1Ops); |
| ASSERT_EQ(Empty, Wrapped1->getOperand(0)); |
| ASSERT_TRUE(Wrapped1->isResolved()); |
| EXPECT_FALSE(Wrapped1->isDistinct()); |
| |
| // !{!{!{}}} |
| Metadata *Wrapped2Ops[] = {Wrapped1}; |
| MDNode *Wrapped2 = MDNode::get(Context, Wrapped2Ops); |
| ASSERT_EQ(Wrapped1, Wrapped2->getOperand(0)); |
| ASSERT_TRUE(Wrapped2->isResolved()); |
| EXPECT_FALSE(Wrapped2->isDistinct()); |
| |
| // !{!{!{}}} => !{!{}} |
| Wrapped2->replaceOperandWith(0, Empty); |
| ASSERT_EQ(Empty, Wrapped2->getOperand(0)); |
| EXPECT_TRUE(Wrapped2->isDistinct()); |
| EXPECT_FALSE(Wrapped1->isDistinct()); |
| } |
| |
| TEST_F(MDNodeTest, UniquedOnDeletedOperand) { |
| // temp !{} |
| TempMDTuple T = MDTuple::getTemporary(Context, None); |
| |
| // !{temp !{}} |
| Metadata *Ops[] = {T.get()}; |
| MDTuple *N = MDTuple::get(Context, Ops); |
| |
| // !{temp !{}} => !{null} |
| T.reset(); |
| ASSERT_TRUE(N->isUniqued()); |
| Metadata *NullOps[] = {nullptr}; |
| ASSERT_EQ(N, MDTuple::get(Context, NullOps)); |
| } |
| |
| TEST_F(MDNodeTest, DistinctOnDeletedValueOperand) { |
| // i1* @GV |
| Type *Ty = Type::getInt1PtrTy(Context); |
| std::unique_ptr<GlobalVariable> GV( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| ConstantAsMetadata *Op = ConstantAsMetadata::get(GV.get()); |
| |
| // !{i1* @GV} |
| Metadata *Ops[] = {Op}; |
| MDTuple *N = MDTuple::get(Context, Ops); |
| |
| // !{i1* @GV} => !{null} |
| GV.reset(); |
| ASSERT_TRUE(N->isDistinct()); |
| ASSERT_EQ(nullptr, N->getOperand(0)); |
| Metadata *NullOps[] = {nullptr}; |
| ASSERT_NE(N, MDTuple::get(Context, NullOps)); |
| } |
| |
| TEST_F(MDNodeTest, getDistinct) { |
| // !{} |
| MDNode *Empty = MDNode::get(Context, None); |
| ASSERT_TRUE(Empty->isResolved()); |
| ASSERT_FALSE(Empty->isDistinct()); |
| ASSERT_EQ(Empty, MDNode::get(Context, None)); |
| |
| // distinct !{} |
| MDNode *Distinct1 = MDNode::getDistinct(Context, None); |
| MDNode *Distinct2 = MDNode::getDistinct(Context, None); |
| EXPECT_TRUE(Distinct1->isResolved()); |
| EXPECT_TRUE(Distinct2->isDistinct()); |
| EXPECT_NE(Empty, Distinct1); |
| EXPECT_NE(Empty, Distinct2); |
| EXPECT_NE(Distinct1, Distinct2); |
| |
| // !{} |
| ASSERT_EQ(Empty, MDNode::get(Context, None)); |
| } |
| |
| TEST_F(MDNodeTest, isUniqued) { |
| MDNode *U = MDTuple::get(Context, None); |
| MDNode *D = MDTuple::getDistinct(Context, None); |
| auto T = MDTuple::getTemporary(Context, None); |
| EXPECT_TRUE(U->isUniqued()); |
| EXPECT_FALSE(D->isUniqued()); |
| EXPECT_FALSE(T->isUniqued()); |
| } |
| |
| TEST_F(MDNodeTest, isDistinct) { |
| MDNode *U = MDTuple::get(Context, None); |
| MDNode *D = MDTuple::getDistinct(Context, None); |
| auto T = MDTuple::getTemporary(Context, None); |
| EXPECT_FALSE(U->isDistinct()); |
| EXPECT_TRUE(D->isDistinct()); |
| EXPECT_FALSE(T->isDistinct()); |
| } |
| |
| TEST_F(MDNodeTest, isTemporary) { |
| MDNode *U = MDTuple::get(Context, None); |
| MDNode *D = MDTuple::getDistinct(Context, None); |
| auto T = MDTuple::getTemporary(Context, None); |
| EXPECT_FALSE(U->isTemporary()); |
| EXPECT_FALSE(D->isTemporary()); |
| EXPECT_TRUE(T->isTemporary()); |
| } |
| |
| TEST_F(MDNodeTest, getDistinctWithUnresolvedOperands) { |
| // temporary !{} |
| auto Temp = MDTuple::getTemporary(Context, None); |
| ASSERT_FALSE(Temp->isResolved()); |
| |
| // distinct !{temporary !{}} |
| Metadata *Ops[] = {Temp.get()}; |
| MDNode *Distinct = MDNode::getDistinct(Context, Ops); |
| EXPECT_TRUE(Distinct->isResolved()); |
| EXPECT_EQ(Temp.get(), Distinct->getOperand(0)); |
| |
| // temporary !{} => !{} |
| MDNode *Empty = MDNode::get(Context, None); |
| Temp->replaceAllUsesWith(Empty); |
| EXPECT_EQ(Empty, Distinct->getOperand(0)); |
| } |
| |
| TEST_F(MDNodeTest, handleChangedOperandRecursion) { |
| // !0 = !{} |
| MDNode *N0 = MDNode::get(Context, None); |
| |
| // !1 = !{!3, null} |
| auto Temp3 = MDTuple::getTemporary(Context, None); |
| Metadata *Ops1[] = {Temp3.get(), nullptr}; |
| MDNode *N1 = MDNode::get(Context, Ops1); |
| |
| // !2 = !{!3, !0} |
| Metadata *Ops2[] = {Temp3.get(), N0}; |
| MDNode *N2 = MDNode::get(Context, Ops2); |
| |
| // !3 = !{!2} |
| Metadata *Ops3[] = {N2}; |
| MDNode *N3 = MDNode::get(Context, Ops3); |
| Temp3->replaceAllUsesWith(N3); |
| |
| // !4 = !{!1} |
| Metadata *Ops4[] = {N1}; |
| MDNode *N4 = MDNode::get(Context, Ops4); |
| |
| // Confirm that the cycle prevented RAUW from getting dropped. |
| EXPECT_TRUE(N0->isResolved()); |
| EXPECT_FALSE(N1->isResolved()); |
| EXPECT_FALSE(N2->isResolved()); |
| EXPECT_FALSE(N3->isResolved()); |
| EXPECT_FALSE(N4->isResolved()); |
| |
| // Create a couple of distinct nodes to observe what's going on. |
| // |
| // !5 = distinct !{!2} |
| // !6 = distinct !{!3} |
| Metadata *Ops5[] = {N2}; |
| MDNode *N5 = MDNode::getDistinct(Context, Ops5); |
| Metadata *Ops6[] = {N3}; |
| MDNode *N6 = MDNode::getDistinct(Context, Ops6); |
| |
| // Mutate !2 to look like !1, causing a uniquing collision (and an RAUW). |
| // This will ripple up, with !3 colliding with !4, and RAUWing. Since !2 |
| // references !3, this can cause a re-entry of handleChangedOperand() when !3 |
| // is not ready for it. |
| // |
| // !2->replaceOperandWith(1, nullptr) |
| // !2: !{!3, !0} => !{!3, null} |
| // !2->replaceAllUsesWith(!1) |
| // !3: !{!2] => !{!1} |
| // !3->replaceAllUsesWith(!4) |
| N2->replaceOperandWith(1, nullptr); |
| |
| // If all has gone well, N2 and N3 will have been RAUW'ed and deleted from |
| // under us. Just check that the other nodes are sane. |
| // |
| // !1 = !{!4, null} |
| // !4 = !{!1} |
| // !5 = distinct !{!1} |
| // !6 = distinct !{!4} |
| EXPECT_EQ(N4, N1->getOperand(0)); |
| EXPECT_EQ(N1, N4->getOperand(0)); |
| EXPECT_EQ(N1, N5->getOperand(0)); |
| EXPECT_EQ(N4, N6->getOperand(0)); |
| } |
| |
| TEST_F(MDNodeTest, replaceResolvedOperand) { |
| // Check code for replacing one resolved operand with another. If doing this |
| // directly (via replaceOperandWith()) becomes illegal, change the operand to |
| // a global value that gets RAUW'ed. |
| // |
| // Use a temporary node to keep N from being resolved. |
| auto Temp = MDTuple::getTemporary(Context, None); |
| Metadata *Ops[] = {nullptr, Temp.get()}; |
| |
| MDNode *Empty = MDTuple::get(Context, ArrayRef<Metadata *>()); |
| MDNode *N = MDTuple::get(Context, Ops); |
| EXPECT_EQ(nullptr, N->getOperand(0)); |
| ASSERT_FALSE(N->isResolved()); |
| |
| // Check code for replacing resolved nodes. |
| N->replaceOperandWith(0, Empty); |
| EXPECT_EQ(Empty, N->getOperand(0)); |
| |
| // Check code for adding another unresolved operand. |
| N->replaceOperandWith(0, Temp.get()); |
| EXPECT_EQ(Temp.get(), N->getOperand(0)); |
| |
| // Remove the references to Temp; required for teardown. |
| Temp->replaceAllUsesWith(nullptr); |
| } |
| |
| TEST_F(MDNodeTest, replaceWithUniqued) { |
| auto *Empty = MDTuple::get(Context, None); |
| MDTuple *FirstUniqued; |
| { |
| Metadata *Ops[] = {Empty}; |
| auto Temp = MDTuple::getTemporary(Context, Ops); |
| EXPECT_TRUE(Temp->isTemporary()); |
| |
| // Don't expect a collision. |
| auto *Current = Temp.get(); |
| FirstUniqued = MDNode::replaceWithUniqued(std::move(Temp)); |
| EXPECT_TRUE(FirstUniqued->isUniqued()); |
| EXPECT_TRUE(FirstUniqued->isResolved()); |
| EXPECT_EQ(Current, FirstUniqued); |
| } |
| { |
| Metadata *Ops[] = {Empty}; |
| auto Temp = MDTuple::getTemporary(Context, Ops); |
| EXPECT_TRUE(Temp->isTemporary()); |
| |
| // Should collide with Uniqued above this time. |
| auto *Uniqued = MDNode::replaceWithUniqued(std::move(Temp)); |
| EXPECT_TRUE(Uniqued->isUniqued()); |
| EXPECT_TRUE(Uniqued->isResolved()); |
| EXPECT_EQ(FirstUniqued, Uniqued); |
| } |
| { |
| auto Unresolved = MDTuple::getTemporary(Context, None); |
| Metadata *Ops[] = {Unresolved.get()}; |
| auto Temp = MDTuple::getTemporary(Context, Ops); |
| EXPECT_TRUE(Temp->isTemporary()); |
| |
| // Shouldn't be resolved. |
| auto *Uniqued = MDNode::replaceWithUniqued(std::move(Temp)); |
| EXPECT_TRUE(Uniqued->isUniqued()); |
| EXPECT_FALSE(Uniqued->isResolved()); |
| |
| // Should be a different node. |
| EXPECT_NE(FirstUniqued, Uniqued); |
| |
| // Should resolve when we update its node (note: be careful to avoid a |
| // collision with any other nodes above). |
| Uniqued->replaceOperandWith(0, nullptr); |
| EXPECT_TRUE(Uniqued->isResolved()); |
| } |
| } |
| |
| TEST_F(MDNodeTest, replaceWithUniquedResolvingOperand) { |
| // temp !{} |
| MDTuple *Op = MDTuple::getTemporary(Context, None).release(); |
| EXPECT_FALSE(Op->isResolved()); |
| |
| // temp !{temp !{}} |
| Metadata *Ops[] = {Op}; |
| MDTuple *N = MDTuple::getTemporary(Context, Ops).release(); |
| EXPECT_FALSE(N->isResolved()); |
| |
| // temp !{temp !{}} => !{temp !{}} |
| ASSERT_EQ(N, MDNode::replaceWithUniqued(TempMDTuple(N))); |
| EXPECT_FALSE(N->isResolved()); |
| |
| // !{temp !{}} => !{!{}} |
| ASSERT_EQ(Op, MDNode::replaceWithUniqued(TempMDTuple(Op))); |
| EXPECT_TRUE(Op->isResolved()); |
| EXPECT_TRUE(N->isResolved()); |
| } |
| |
| TEST_F(MDNodeTest, replaceWithUniquedDeletedOperand) { |
| // i1* @GV |
| Type *Ty = Type::getInt1PtrTy(Context); |
| std::unique_ptr<GlobalVariable> GV( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| ConstantAsMetadata *Op = ConstantAsMetadata::get(GV.get()); |
| |
| // temp !{i1* @GV} |
| Metadata *Ops[] = {Op}; |
| MDTuple *N = MDTuple::getTemporary(Context, Ops).release(); |
| |
| // temp !{i1* @GV} => !{i1* @GV} |
| ASSERT_EQ(N, MDNode::replaceWithUniqued(TempMDTuple(N))); |
| ASSERT_TRUE(N->isUniqued()); |
| |
| // !{i1* @GV} => !{null} |
| GV.reset(); |
| ASSERT_TRUE(N->isDistinct()); |
| ASSERT_EQ(nullptr, N->getOperand(0)); |
| Metadata *NullOps[] = {nullptr}; |
| ASSERT_NE(N, MDTuple::get(Context, NullOps)); |
| } |
| |
| TEST_F(MDNodeTest, replaceWithUniquedChangedOperand) { |
| // i1* @GV |
| Type *Ty = Type::getInt1PtrTy(Context); |
| std::unique_ptr<GlobalVariable> GV( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| ConstantAsMetadata *Op = ConstantAsMetadata::get(GV.get()); |
| |
| // temp !{i1* @GV} |
| Metadata *Ops[] = {Op}; |
| MDTuple *N = MDTuple::getTemporary(Context, Ops).release(); |
| |
| // temp !{i1* @GV} => !{i1* @GV} |
| ASSERT_EQ(N, MDNode::replaceWithUniqued(TempMDTuple(N))); |
| ASSERT_TRUE(N->isUniqued()); |
| |
| // !{i1* @GV} => !{i1* @GV2} |
| std::unique_ptr<GlobalVariable> GV2( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| GV->replaceAllUsesWith(GV2.get()); |
| ASSERT_TRUE(N->isUniqued()); |
| Metadata *NullOps[] = {ConstantAsMetadata::get(GV2.get())}; |
| ASSERT_EQ(N, MDTuple::get(Context, NullOps)); |
| } |
| |
| TEST_F(MDNodeTest, replaceWithDistinct) { |
| { |
| auto *Empty = MDTuple::get(Context, None); |
| Metadata *Ops[] = {Empty}; |
| auto Temp = MDTuple::getTemporary(Context, Ops); |
| EXPECT_TRUE(Temp->isTemporary()); |
| |
| // Don't expect a collision. |
| auto *Current = Temp.get(); |
| auto *Distinct = MDNode::replaceWithDistinct(std::move(Temp)); |
| EXPECT_TRUE(Distinct->isDistinct()); |
| EXPECT_TRUE(Distinct->isResolved()); |
| EXPECT_EQ(Current, Distinct); |
| } |
| { |
| auto Unresolved = MDTuple::getTemporary(Context, None); |
| Metadata *Ops[] = {Unresolved.get()}; |
| auto Temp = MDTuple::getTemporary(Context, Ops); |
| EXPECT_TRUE(Temp->isTemporary()); |
| |
| // Don't expect a collision. |
| auto *Current = Temp.get(); |
| auto *Distinct = MDNode::replaceWithDistinct(std::move(Temp)); |
| EXPECT_TRUE(Distinct->isDistinct()); |
| EXPECT_TRUE(Distinct->isResolved()); |
| EXPECT_EQ(Current, Distinct); |
| |
| // Cleanup; required for teardown. |
| Unresolved->replaceAllUsesWith(nullptr); |
| } |
| } |
| |
| TEST_F(MDNodeTest, replaceWithPermanent) { |
| Metadata *Ops[] = {nullptr}; |
| auto Temp = MDTuple::getTemporary(Context, Ops); |
| auto *T = Temp.get(); |
| |
| // U is a normal, uniqued node that references T. |
| auto *U = MDTuple::get(Context, T); |
| EXPECT_TRUE(U->isUniqued()); |
| |
| // Make Temp self-referencing. |
| Temp->replaceOperandWith(0, T); |
| |
| // Try to uniquify Temp. This should, despite the name in the API, give a |
| // 'distinct' node, since self-references aren't allowed to be uniqued. |
| // |
| // Since it's distinct, N should have the same address as when it was a |
| // temporary (i.e., be equal to T not U). |
| auto *N = MDNode::replaceWithPermanent(std::move(Temp)); |
| EXPECT_EQ(N, T); |
| EXPECT_TRUE(N->isDistinct()); |
| |
| // U should be the canonical unique node with N as the argument. |
| EXPECT_EQ(U, MDTuple::get(Context, N)); |
| EXPECT_TRUE(U->isUniqued()); |
| |
| // This temporary should collide with U when replaced, but it should still be |
| // uniqued. |
| EXPECT_EQ(U, MDNode::replaceWithPermanent(MDTuple::getTemporary(Context, N))); |
| EXPECT_TRUE(U->isUniqued()); |
| |
| // This temporary should become a new uniqued node. |
| auto Temp2 = MDTuple::getTemporary(Context, U); |
| auto *V = Temp2.get(); |
| EXPECT_EQ(V, MDNode::replaceWithPermanent(std::move(Temp2))); |
| EXPECT_TRUE(V->isUniqued()); |
| EXPECT_EQ(U, V->getOperand(0)); |
| } |
| |
| TEST_F(MDNodeTest, deleteTemporaryWithTrackingRef) { |
| TrackingMDRef Ref; |
| EXPECT_EQ(nullptr, Ref.get()); |
| { |
| auto Temp = MDTuple::getTemporary(Context, None); |
| Ref.reset(Temp.get()); |
| EXPECT_EQ(Temp.get(), Ref.get()); |
| } |
| EXPECT_EQ(nullptr, Ref.get()); |
| } |
| |
| typedef MetadataTest DILocationTest; |
| |
| TEST_F(DILocationTest, Overflow) { |
| DISubprogram *N = getSubprogram(); |
| { |
| DILocation *L = DILocation::get(Context, 2, 7, N); |
| EXPECT_EQ(2u, L->getLine()); |
| EXPECT_EQ(7u, L->getColumn()); |
| } |
| unsigned U16 = 1u << 16; |
| { |
| DILocation *L = DILocation::get(Context, UINT32_MAX, U16 - 1, N); |
| EXPECT_EQ(UINT32_MAX, L->getLine()); |
| EXPECT_EQ(U16 - 1, L->getColumn()); |
| } |
| { |
| DILocation *L = DILocation::get(Context, UINT32_MAX, U16, N); |
| EXPECT_EQ(UINT32_MAX, L->getLine()); |
| EXPECT_EQ(0u, L->getColumn()); |
| } |
| { |
| DILocation *L = DILocation::get(Context, UINT32_MAX, U16 + 1, N); |
| EXPECT_EQ(UINT32_MAX, L->getLine()); |
| EXPECT_EQ(0u, L->getColumn()); |
| } |
| } |
| |
| TEST_F(DILocationTest, getDistinct) { |
| MDNode *N = getSubprogram(); |
| DILocation *L0 = DILocation::getDistinct(Context, 2, 7, N); |
| EXPECT_TRUE(L0->isDistinct()); |
| DILocation *L1 = DILocation::get(Context, 2, 7, N); |
| EXPECT_FALSE(L1->isDistinct()); |
| EXPECT_EQ(L1, DILocation::get(Context, 2, 7, N)); |
| } |
| |
| TEST_F(DILocationTest, getTemporary) { |
| MDNode *N = MDNode::get(Context, None); |
| auto L = DILocation::getTemporary(Context, 2, 7, N); |
| EXPECT_TRUE(L->isTemporary()); |
| EXPECT_FALSE(L->isResolved()); |
| } |
| |
| TEST_F(DILocationTest, cloneTemporary) { |
| MDNode *N = MDNode::get(Context, None); |
| auto L = DILocation::getTemporary(Context, 2, 7, N); |
| EXPECT_TRUE(L->isTemporary()); |
| auto L2 = L->clone(); |
| EXPECT_TRUE(L2->isTemporary()); |
| } |
| |
| typedef MetadataTest GenericDINodeTest; |
| |
| TEST_F(GenericDINodeTest, get) { |
| StringRef Header = "header"; |
| auto *Empty = MDNode::get(Context, None); |
| Metadata *Ops1[] = {Empty}; |
| auto *N = GenericDINode::get(Context, 15, Header, Ops1); |
| EXPECT_EQ(15u, N->getTag()); |
| EXPECT_EQ(2u, N->getNumOperands()); |
| EXPECT_EQ(Header, N->getHeader()); |
| EXPECT_EQ(MDString::get(Context, Header), N->getOperand(0)); |
| EXPECT_EQ(1u, N->getNumDwarfOperands()); |
| EXPECT_EQ(Empty, N->getDwarfOperand(0)); |
| EXPECT_EQ(Empty, N->getOperand(1)); |
| ASSERT_TRUE(N->isUniqued()); |
| |
| EXPECT_EQ(N, GenericDINode::get(Context, 15, Header, Ops1)); |
| |
| N->replaceOperandWith(1, nullptr); |
| EXPECT_EQ(15u, N->getTag()); |
| EXPECT_EQ(Header, N->getHeader()); |
| EXPECT_EQ(nullptr, N->getDwarfOperand(0)); |
| ASSERT_TRUE(N->isUniqued()); |
| |
| Metadata *Ops2[] = {nullptr}; |
| EXPECT_EQ(N, GenericDINode::get(Context, 15, Header, Ops2)); |
| |
| N->replaceDwarfOperandWith(0, Empty); |
| EXPECT_EQ(15u, N->getTag()); |
| EXPECT_EQ(Header, N->getHeader()); |
| EXPECT_EQ(Empty, N->getDwarfOperand(0)); |
| ASSERT_TRUE(N->isUniqued()); |
| EXPECT_EQ(N, GenericDINode::get(Context, 15, Header, Ops1)); |
| |
| TempGenericDINode Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(GenericDINodeTest, getEmptyHeader) { |
| // Canonicalize !"" to null. |
| auto *N = GenericDINode::get(Context, 15, StringRef(), None); |
| EXPECT_EQ(StringRef(), N->getHeader()); |
| EXPECT_EQ(nullptr, N->getOperand(0)); |
| } |
| |
| typedef MetadataTest DISubrangeTest; |
| |
| TEST_F(DISubrangeTest, get) { |
| auto *N = DISubrange::get(Context, 5, 7); |
| auto Count = N->getCount(); |
| EXPECT_EQ(dwarf::DW_TAG_subrange_type, N->getTag()); |
| ASSERT_TRUE(Count); |
| ASSERT_TRUE(Count.is<ConstantInt*>()); |
| EXPECT_EQ(5, Count.get<ConstantInt*>()->getSExtValue()); |
| EXPECT_EQ(7, N->getLowerBound()); |
| EXPECT_EQ(N, DISubrange::get(Context, 5, 7)); |
| EXPECT_EQ(DISubrange::get(Context, 5, 0), DISubrange::get(Context, 5)); |
| |
| TempDISubrange Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DISubrangeTest, getEmptyArray) { |
| auto *N = DISubrange::get(Context, -1, 0); |
| auto Count = N->getCount(); |
| EXPECT_EQ(dwarf::DW_TAG_subrange_type, N->getTag()); |
| ASSERT_TRUE(Count); |
| ASSERT_TRUE(Count.is<ConstantInt*>()); |
| EXPECT_EQ(-1, Count.get<ConstantInt*>()->getSExtValue()); |
| EXPECT_EQ(0, N->getLowerBound()); |
| EXPECT_EQ(N, DISubrange::get(Context, -1, 0)); |
| } |
| |
| TEST_F(DISubrangeTest, getVariableCount) { |
| DILocalScope *Scope = getSubprogram(); |
| DIFile *File = getFile(); |
| DIType *Type = getDerivedType(); |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(7); |
| auto *VlaExpr = DILocalVariable::get(Context, Scope, "vla_expr", File, 8, |
| Type, 2, Flags, 8); |
| |
| auto *N = DISubrange::get(Context, VlaExpr, 0); |
| auto Count = N->getCount(); |
| ASSERT_TRUE(Count); |
| ASSERT_TRUE(Count.is<DIVariable*>()); |
| EXPECT_EQ(VlaExpr, Count.get<DIVariable*>()); |
| ASSERT_TRUE(isa<DIVariable>(N->getRawCountNode())); |
| EXPECT_EQ(0, N->getLowerBound()); |
| EXPECT_EQ("vla_expr", Count.get<DIVariable*>()->getName()); |
| EXPECT_EQ(N, DISubrange::get(Context, VlaExpr, 0)); |
| } |
| |
| typedef MetadataTest DIEnumeratorTest; |
| |
| TEST_F(DIEnumeratorTest, get) { |
| auto *N = DIEnumerator::get(Context, 7, false, "name"); |
| EXPECT_EQ(dwarf::DW_TAG_enumerator, N->getTag()); |
| EXPECT_EQ(7, N->getValue()); |
| EXPECT_FALSE(N->isUnsigned()); |
| EXPECT_EQ("name", N->getName()); |
| EXPECT_EQ(N, DIEnumerator::get(Context, 7, false, "name")); |
| |
| EXPECT_NE(N, DIEnumerator::get(Context, 7, true, "name")); |
| EXPECT_NE(N, DIEnumerator::get(Context, 8, false, "name")); |
| EXPECT_NE(N, DIEnumerator::get(Context, 7, false, "nam")); |
| |
| TempDIEnumerator Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DIBasicTypeTest; |
| |
| TEST_F(DIBasicTypeTest, get) { |
| auto *N = |
| DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33, 26, 7); |
| EXPECT_EQ(dwarf::DW_TAG_base_type, N->getTag()); |
| EXPECT_EQ("special", N->getName()); |
| EXPECT_EQ(33u, N->getSizeInBits()); |
| EXPECT_EQ(26u, N->getAlignInBits()); |
| EXPECT_EQ(7u, N->getEncoding()); |
| EXPECT_EQ(0u, N->getLine()); |
| EXPECT_EQ(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33, |
| 26, 7)); |
| |
| EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_unspecified_type, |
| "special", 33, 26, 7)); |
| EXPECT_NE(N, |
| DIBasicType::get(Context, dwarf::DW_TAG_base_type, "s", 33, 26, 7)); |
| EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 32, |
| 26, 7)); |
| EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33, |
| 25, 7)); |
| EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33, |
| 26, 6)); |
| |
| TempDIBasicType Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DIBasicTypeTest, getWithLargeValues) { |
| auto *N = DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", |
| UINT64_MAX, UINT32_MAX - 1, 7); |
| EXPECT_EQ(UINT64_MAX, N->getSizeInBits()); |
| EXPECT_EQ(UINT32_MAX - 1, N->getAlignInBits()); |
| } |
| |
| TEST_F(DIBasicTypeTest, getUnspecified) { |
| auto *N = |
| DIBasicType::get(Context, dwarf::DW_TAG_unspecified_type, "unspecified"); |
| EXPECT_EQ(dwarf::DW_TAG_unspecified_type, N->getTag()); |
| EXPECT_EQ("unspecified", N->getName()); |
| EXPECT_EQ(0u, N->getSizeInBits()); |
| EXPECT_EQ(0u, N->getAlignInBits()); |
| EXPECT_EQ(0u, N->getEncoding()); |
| EXPECT_EQ(0u, N->getLine()); |
| } |
| |
| typedef MetadataTest DITypeTest; |
| |
| TEST_F(DITypeTest, clone) { |
| // Check that DIType has a specialized clone that returns TempDIType. |
| DIType *N = DIBasicType::get(Context, dwarf::DW_TAG_base_type, "int", 32, 32, |
| dwarf::DW_ATE_signed); |
| |
| TempDIType Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DITypeTest, cloneWithFlags) { |
| // void (void) |
| Metadata *TypesOps[] = {nullptr}; |
| Metadata *Types = MDTuple::get(Context, TypesOps); |
| |
| DIType *D = |
| DISubroutineType::getDistinct(Context, DINode::FlagZero, 0, Types); |
| EXPECT_EQ(DINode::FlagZero, D->getFlags()); |
| TempDIType D2 = D->cloneWithFlags(DINode::FlagRValueReference); |
| EXPECT_EQ(DINode::FlagRValueReference, D2->getFlags()); |
| EXPECT_EQ(DINode::FlagZero, D->getFlags()); |
| |
| TempDIType T = |
| DISubroutineType::getTemporary(Context, DINode::FlagZero, 0, Types); |
| EXPECT_EQ(DINode::FlagZero, T->getFlags()); |
| TempDIType T2 = T->cloneWithFlags(DINode::FlagRValueReference); |
| EXPECT_EQ(DINode::FlagRValueReference, T2->getFlags()); |
| EXPECT_EQ(DINode::FlagZero, T->getFlags()); |
| } |
| |
| typedef MetadataTest DIDerivedTypeTest; |
| |
| TEST_F(DIDerivedTypeTest, get) { |
| DIFile *File = getFile(); |
| DIScope *Scope = getSubprogram(); |
| DIType *BaseType = getBasicType("basic"); |
| MDTuple *ExtraData = getTuple(); |
| unsigned DWARFAddressSpace = 8; |
| DINode::DIFlags Flags5 = static_cast<DINode::DIFlags>(5); |
| DINode::DIFlags Flags4 = static_cast<DINode::DIFlags>(4); |
| |
| auto *N = |
| DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, "something", File, |
| 1, Scope, BaseType, 2, 3, 4, DWARFAddressSpace, Flags5, |
| ExtraData); |
| EXPECT_EQ(dwarf::DW_TAG_pointer_type, N->getTag()); |
| EXPECT_EQ("something", N->getName()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(1u, N->getLine()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(BaseType, N->getBaseType()); |
| EXPECT_EQ(2u, N->getSizeInBits()); |
| EXPECT_EQ(3u, N->getAlignInBits()); |
| EXPECT_EQ(4u, N->getOffsetInBits()); |
| EXPECT_EQ(DWARFAddressSpace, N->getDWARFAddressSpace().getValue()); |
| EXPECT_EQ(5u, N->getFlags()); |
| EXPECT_EQ(ExtraData, N->getExtraData()); |
| EXPECT_EQ(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace, Flags5, ExtraData)); |
| |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_reference_type, |
| "something", File, 1, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, "else", |
| File, 1, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", getFile(), 1, Scope, BaseType, 2, |
| 3, 4, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 2, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, getSubprogram(), |
| BaseType, 2, 3, 4, DWARFAddressSpace, Flags5, |
| ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get( |
| Context, dwarf::DW_TAG_pointer_type, "something", File, 1, |
| Scope, getBasicType("basic2"), 2, 3, 4, DWARFAddressSpace, |
| Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 3, 3, |
| 4, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 2, 2, |
| 4, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 2, 3, |
| 5, DWARFAddressSpace, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace + 1, Flags5, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace, Flags4, ExtraData)); |
| EXPECT_NE(N, DIDerivedType::get(Context, dwarf::DW_TAG_pointer_type, |
| "something", File, 1, Scope, BaseType, 2, 3, |
| 4, DWARFAddressSpace, Flags5, getTuple())); |
| |
| TempDIDerivedType Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DIDerivedTypeTest, getWithLargeValues) { |
| DIFile *File = getFile(); |
| DIScope *Scope = getSubprogram(); |
| DIType *BaseType = getBasicType("basic"); |
| MDTuple *ExtraData = getTuple(); |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(5); |
| |
| auto *N = DIDerivedType::get( |
| Context, dwarf::DW_TAG_pointer_type, "something", File, 1, Scope, |
| BaseType, UINT64_MAX, UINT32_MAX - 1, UINT64_MAX - 2, UINT32_MAX - 3, |
| Flags, ExtraData); |
| EXPECT_EQ(UINT64_MAX, N->getSizeInBits()); |
| EXPECT_EQ(UINT32_MAX - 1, N->getAlignInBits()); |
| EXPECT_EQ(UINT64_MAX - 2, N->getOffsetInBits()); |
| EXPECT_EQ(UINT32_MAX - 3, N->getDWARFAddressSpace().getValue()); |
| } |
| |
| typedef MetadataTest DICompositeTypeTest; |
| |
| TEST_F(DICompositeTypeTest, get) { |
| unsigned Tag = dwarf::DW_TAG_structure_type; |
| StringRef Name = "some name"; |
| DIFile *File = getFile(); |
| unsigned Line = 1; |
| DIScope *Scope = getSubprogram(); |
| DIType *BaseType = getCompositeType(); |
| uint64_t SizeInBits = 2; |
| uint32_t AlignInBits = 3; |
| uint64_t OffsetInBits = 4; |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(5); |
| MDTuple *Elements = getTuple(); |
| unsigned RuntimeLang = 6; |
| DIType *VTableHolder = getCompositeType(); |
| MDTuple *TemplateParams = getTuple(); |
| StringRef Identifier = "some id"; |
| |
| auto *N = DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier); |
| EXPECT_EQ(Tag, N->getTag()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(BaseType, N->getBaseType()); |
| EXPECT_EQ(SizeInBits, N->getSizeInBits()); |
| EXPECT_EQ(AlignInBits, N->getAlignInBits()); |
| EXPECT_EQ(OffsetInBits, N->getOffsetInBits()); |
| EXPECT_EQ(Flags, N->getFlags()); |
| EXPECT_EQ(Elements, N->getElements().get()); |
| EXPECT_EQ(RuntimeLang, N->getRuntimeLang()); |
| EXPECT_EQ(VTableHolder, N->getVTableHolder()); |
| EXPECT_EQ(TemplateParams, N->getTemplateParams().get()); |
| EXPECT_EQ(Identifier, N->getIdentifier()); |
| |
| EXPECT_EQ(N, DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| |
| EXPECT_NE(N, DICompositeType::get(Context, Tag + 1, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, "abc", File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, Name, getFile(), Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, Name, File, Line + 1, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, getSubprogram(), BaseType, |
| SizeInBits, AlignInBits, OffsetInBits, Flags, Elements, |
| RuntimeLang, VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, getBasicType("other"), |
| SizeInBits, AlignInBits, OffsetInBits, Flags, Elements, |
| RuntimeLang, VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits + 1, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits + 1, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, |
| AlignInBits, OffsetInBits + 1, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| DINode::DIFlags FlagsPOne = static_cast<DINode::DIFlags>(Flags + 1); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, |
| AlignInBits, OffsetInBits, FlagsPOne, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, |
| AlignInBits, OffsetInBits, Flags, getTuple(), RuntimeLang, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, |
| AlignInBits, OffsetInBits, Flags, Elements, RuntimeLang + 1, |
| VTableHolder, TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, |
| AlignInBits, OffsetInBits, Flags, Elements, RuntimeLang, |
| getCompositeType(), TemplateParams, Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, getTuple(), Identifier)); |
| EXPECT_NE(N, DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, "other")); |
| |
| // Be sure that missing identifiers get null pointers. |
| EXPECT_FALSE(DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, "") |
| ->getRawIdentifier()); |
| EXPECT_FALSE(DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams) |
| ->getRawIdentifier()); |
| |
| TempDICompositeType Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DICompositeTypeTest, getWithLargeValues) { |
| unsigned Tag = dwarf::DW_TAG_structure_type; |
| StringRef Name = "some name"; |
| DIFile *File = getFile(); |
| unsigned Line = 1; |
| DIScope *Scope = getSubprogram(); |
| DIType *BaseType = getCompositeType(); |
| uint64_t SizeInBits = UINT64_MAX; |
| uint32_t AlignInBits = UINT32_MAX - 1; |
| uint64_t OffsetInBits = UINT64_MAX - 2; |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(5); |
| MDTuple *Elements = getTuple(); |
| unsigned RuntimeLang = 6; |
| DIType *VTableHolder = getCompositeType(); |
| MDTuple *TemplateParams = getTuple(); |
| StringRef Identifier = "some id"; |
| |
| auto *N = DICompositeType::get(Context, Tag, Name, File, Line, Scope, |
| BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, Elements, RuntimeLang, |
| VTableHolder, TemplateParams, Identifier); |
| EXPECT_EQ(SizeInBits, N->getSizeInBits()); |
| EXPECT_EQ(AlignInBits, N->getAlignInBits()); |
| EXPECT_EQ(OffsetInBits, N->getOffsetInBits()); |
| } |
| |
| TEST_F(DICompositeTypeTest, replaceOperands) { |
| unsigned Tag = dwarf::DW_TAG_structure_type; |
| StringRef Name = "some name"; |
| DIFile *File = getFile(); |
| unsigned Line = 1; |
| DIScope *Scope = getSubprogram(); |
| DIType *BaseType = getCompositeType(); |
| uint64_t SizeInBits = 2; |
| uint32_t AlignInBits = 3; |
| uint64_t OffsetInBits = 4; |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(5); |
| unsigned RuntimeLang = 6; |
| StringRef Identifier = "some id"; |
| |
| auto *N = DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, nullptr, RuntimeLang, nullptr, nullptr, Identifier); |
| |
| auto *Elements = MDTuple::getDistinct(Context, None); |
| EXPECT_EQ(nullptr, N->getElements().get()); |
| N->replaceElements(Elements); |
| EXPECT_EQ(Elements, N->getElements().get()); |
| N->replaceElements(nullptr); |
| EXPECT_EQ(nullptr, N->getElements().get()); |
| |
| DIType *VTableHolder = getCompositeType(); |
| EXPECT_EQ(nullptr, N->getVTableHolder()); |
| N->replaceVTableHolder(VTableHolder); |
| EXPECT_EQ(VTableHolder, N->getVTableHolder()); |
| // As an extension, the containing type can be anything. This is |
| // used by Rust to associate vtables with their concrete type. |
| DIType *BasicType = getBasicType("basic"); |
| N->replaceVTableHolder(BasicType); |
| EXPECT_EQ(BasicType, N->getVTableHolder()); |
| N->replaceVTableHolder(nullptr); |
| EXPECT_EQ(nullptr, N->getVTableHolder()); |
| |
| auto *TemplateParams = MDTuple::getDistinct(Context, None); |
| EXPECT_EQ(nullptr, N->getTemplateParams().get()); |
| N->replaceTemplateParams(TemplateParams); |
| EXPECT_EQ(TemplateParams, N->getTemplateParams().get()); |
| N->replaceTemplateParams(nullptr); |
| EXPECT_EQ(nullptr, N->getTemplateParams().get()); |
| } |
| |
| TEST_F(DICompositeTypeTest, variant_part) { |
| unsigned Tag = dwarf::DW_TAG_variant_part; |
| StringRef Name = "some name"; |
| DIFile *File = getFile(); |
| unsigned Line = 1; |
| DIScope *Scope = getSubprogram(); |
| DIType *BaseType = getCompositeType(); |
| uint64_t SizeInBits = 2; |
| uint32_t AlignInBits = 3; |
| uint64_t OffsetInBits = 4; |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(5); |
| unsigned RuntimeLang = 6; |
| StringRef Identifier = "some id"; |
| DIDerivedType *Discriminator = cast<DIDerivedType>(getDerivedType()); |
| DIDerivedType *Discriminator2 = cast<DIDerivedType>(getDerivedType()); |
| |
| EXPECT_NE(Discriminator, Discriminator2); |
| |
| auto *N = DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, nullptr, RuntimeLang, nullptr, nullptr, Identifier, |
| Discriminator); |
| |
| // Test the hashing. |
| auto *Same = DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, nullptr, RuntimeLang, nullptr, nullptr, Identifier, |
| Discriminator); |
| auto *Other = DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, nullptr, RuntimeLang, nullptr, nullptr, Identifier, |
| Discriminator2); |
| auto *NoDisc = DICompositeType::get( |
| Context, Tag, Name, File, Line, Scope, BaseType, SizeInBits, AlignInBits, |
| OffsetInBits, Flags, nullptr, RuntimeLang, nullptr, nullptr, Identifier, |
| nullptr); |
| |
| EXPECT_EQ(N, Same); |
| EXPECT_NE(Same, Other); |
| EXPECT_NE(Same, NoDisc); |
| EXPECT_NE(Other, NoDisc); |
| |
| EXPECT_EQ(N->getDiscriminator(), Discriminator); |
| } |
| |
| typedef MetadataTest DISubroutineTypeTest; |
| |
| TEST_F(DISubroutineTypeTest, get) { |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(1); |
| DINode::DIFlags FlagsPOne = static_cast<DINode::DIFlags>(Flags + 1); |
| MDTuple *TypeArray = getTuple(); |
| |
| auto *N = DISubroutineType::get(Context, Flags, 0, TypeArray); |
| EXPECT_EQ(dwarf::DW_TAG_subroutine_type, N->getTag()); |
| EXPECT_EQ(Flags, N->getFlags()); |
| EXPECT_EQ(TypeArray, N->getTypeArray().get()); |
| EXPECT_EQ(N, DISubroutineType::get(Context, Flags, 0, TypeArray)); |
| |
| EXPECT_NE(N, DISubroutineType::get(Context, FlagsPOne, 0, TypeArray)); |
| EXPECT_NE(N, DISubroutineType::get(Context, Flags, 0, getTuple())); |
| |
| // Test the hashing of calling conventions. |
| auto *Fast = DISubroutineType::get( |
| Context, Flags, dwarf::DW_CC_BORLAND_msfastcall, TypeArray); |
| auto *Std = DISubroutineType::get(Context, Flags, |
| dwarf::DW_CC_BORLAND_stdcall, TypeArray); |
| EXPECT_EQ(Fast, |
| DISubroutineType::get(Context, Flags, |
| dwarf::DW_CC_BORLAND_msfastcall, TypeArray)); |
| EXPECT_EQ(Std, DISubroutineType::get( |
| Context, Flags, dwarf::DW_CC_BORLAND_stdcall, TypeArray)); |
| |
| EXPECT_NE(N, Fast); |
| EXPECT_NE(N, Std); |
| EXPECT_NE(Fast, Std); |
| |
| TempDISubroutineType Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| |
| // Test always-empty operands. |
| EXPECT_EQ(nullptr, N->getScope()); |
| EXPECT_EQ(nullptr, N->getFile()); |
| EXPECT_EQ("", N->getName()); |
| } |
| |
| typedef MetadataTest DIFileTest; |
| |
| TEST_F(DIFileTest, get) { |
| StringRef Filename = "file"; |
| StringRef Directory = "dir"; |
| DIFile::ChecksumKind CSKind = DIFile::ChecksumKind::CSK_MD5; |
| StringRef ChecksumString = "000102030405060708090a0b0c0d0e0f"; |
| DIFile::ChecksumInfo<StringRef> Checksum(CSKind, ChecksumString); |
| StringRef Source = "source"; |
| auto *N = DIFile::get(Context, Filename, Directory, Checksum, Source); |
| |
| EXPECT_EQ(dwarf::DW_TAG_file_type, N->getTag()); |
| EXPECT_EQ(Filename, N->getFilename()); |
| EXPECT_EQ(Directory, N->getDirectory()); |
| EXPECT_EQ(Checksum, N->getChecksum()); |
| EXPECT_EQ(Source, N->getSource()); |
| EXPECT_EQ(N, DIFile::get(Context, Filename, Directory, Checksum, Source)); |
| |
| EXPECT_NE(N, DIFile::get(Context, "other", Directory, Checksum, Source)); |
| EXPECT_NE(N, DIFile::get(Context, Filename, "other", Checksum, Source)); |
| DIFile::ChecksumInfo<StringRef> OtherChecksum(DIFile::ChecksumKind::CSK_SHA1, ChecksumString); |
| EXPECT_NE( |
| N, DIFile::get(Context, Filename, Directory, OtherChecksum)); |
| StringRef OtherSource = "other"; |
| EXPECT_NE(N, DIFile::get(Context, Filename, Directory, Checksum, OtherSource)); |
| EXPECT_NE(N, DIFile::get(Context, Filename, Directory, Checksum)); |
| EXPECT_NE(N, DIFile::get(Context, Filename, Directory)); |
| |
| TempDIFile Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DIFileTest, ScopeGetFile) { |
| // Ensure that DIScope::getFile() returns itself. |
| DIScope *N = DIFile::get(Context, "file", "dir"); |
| EXPECT_EQ(N, N->getFile()); |
| } |
| |
| typedef MetadataTest DICompileUnitTest; |
| |
| TEST_F(DICompileUnitTest, get) { |
| unsigned SourceLanguage = 1; |
| DIFile *File = getFile(); |
| StringRef Producer = "some producer"; |
| bool IsOptimized = false; |
| StringRef Flags = "flag after flag"; |
| unsigned RuntimeVersion = 2; |
| StringRef SplitDebugFilename = "another/file"; |
| auto EmissionKind = DICompileUnit::FullDebug; |
| MDTuple *EnumTypes = getTuple(); |
| MDTuple *RetainedTypes = getTuple(); |
| MDTuple *GlobalVariables = getTuple(); |
| MDTuple *ImportedEntities = getTuple(); |
| uint64_t DWOId = 0x10000000c0ffee; |
| MDTuple *Macros = getTuple(); |
| auto *N = DICompileUnit::getDistinct( |
| Context, SourceLanguage, File, Producer, IsOptimized, Flags, |
| RuntimeVersion, SplitDebugFilename, EmissionKind, EnumTypes, |
| RetainedTypes, GlobalVariables, ImportedEntities, Macros, DWOId, true, |
| false, false); |
| |
| EXPECT_EQ(dwarf::DW_TAG_compile_unit, N->getTag()); |
| EXPECT_EQ(SourceLanguage, N->getSourceLanguage()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Producer, N->getProducer()); |
| EXPECT_EQ(IsOptimized, N->isOptimized()); |
| EXPECT_EQ(Flags, N->getFlags()); |
| EXPECT_EQ(RuntimeVersion, N->getRuntimeVersion()); |
| EXPECT_EQ(SplitDebugFilename, N->getSplitDebugFilename()); |
| EXPECT_EQ(EmissionKind, N->getEmissionKind()); |
| EXPECT_EQ(EnumTypes, N->getEnumTypes().get()); |
| EXPECT_EQ(RetainedTypes, N->getRetainedTypes().get()); |
| EXPECT_EQ(GlobalVariables, N->getGlobalVariables().get()); |
| EXPECT_EQ(ImportedEntities, N->getImportedEntities().get()); |
| EXPECT_EQ(Macros, N->getMacros().get()); |
| EXPECT_EQ(DWOId, N->getDWOId()); |
| |
| TempDICompileUnit Temp = N->clone(); |
| EXPECT_EQ(dwarf::DW_TAG_compile_unit, Temp->getTag()); |
| EXPECT_EQ(SourceLanguage, Temp->getSourceLanguage()); |
| EXPECT_EQ(File, Temp->getFile()); |
| EXPECT_EQ(Producer, Temp->getProducer()); |
| EXPECT_EQ(IsOptimized, Temp->isOptimized()); |
| EXPECT_EQ(Flags, Temp->getFlags()); |
| EXPECT_EQ(RuntimeVersion, Temp->getRuntimeVersion()); |
| EXPECT_EQ(SplitDebugFilename, Temp->getSplitDebugFilename()); |
| EXPECT_EQ(EmissionKind, Temp->getEmissionKind()); |
| EXPECT_EQ(EnumTypes, Temp->getEnumTypes().get()); |
| EXPECT_EQ(RetainedTypes, Temp->getRetainedTypes().get()); |
| EXPECT_EQ(GlobalVariables, Temp->getGlobalVariables().get()); |
| EXPECT_EQ(ImportedEntities, Temp->getImportedEntities().get()); |
| EXPECT_EQ(Macros, Temp->getMacros().get()); |
| EXPECT_EQ(DWOId, Temp->getDWOId()); |
| |
| auto *TempAddress = Temp.get(); |
| auto *Clone = MDNode::replaceWithPermanent(std::move(Temp)); |
| EXPECT_TRUE(Clone->isDistinct()); |
| EXPECT_EQ(TempAddress, Clone); |
| } |
| |
| TEST_F(DICompileUnitTest, replaceArrays) { |
| unsigned SourceLanguage = 1; |
| DIFile *File = getFile(); |
| StringRef Producer = "some producer"; |
| bool IsOptimized = false; |
| StringRef Flags = "flag after flag"; |
| unsigned RuntimeVersion = 2; |
| StringRef SplitDebugFilename = "another/file"; |
| auto EmissionKind = DICompileUnit::FullDebug; |
| MDTuple *EnumTypes = MDTuple::getDistinct(Context, None); |
| MDTuple *RetainedTypes = MDTuple::getDistinct(Context, None); |
| MDTuple *ImportedEntities = MDTuple::getDistinct(Context, None); |
| uint64_t DWOId = 0xc0ffee; |
| auto *N = DICompileUnit::getDistinct( |
| Context, SourceLanguage, File, Producer, IsOptimized, Flags, |
| RuntimeVersion, SplitDebugFilename, EmissionKind, EnumTypes, |
| RetainedTypes, nullptr, ImportedEntities, nullptr, DWOId, true, false, |
| false); |
| |
| auto *GlobalVariables = MDTuple::getDistinct(Context, None); |
| EXPECT_EQ(nullptr, N->getGlobalVariables().get()); |
| N->replaceGlobalVariables(GlobalVariables); |
| EXPECT_EQ(GlobalVariables, N->getGlobalVariables().get()); |
| N->replaceGlobalVariables(nullptr); |
| EXPECT_EQ(nullptr, N->getGlobalVariables().get()); |
| |
| auto *Macros = MDTuple::getDistinct(Context, None); |
| EXPECT_EQ(nullptr, N->getMacros().get()); |
| N->replaceMacros(Macros); |
| EXPECT_EQ(Macros, N->getMacros().get()); |
| N->replaceMacros(nullptr); |
| EXPECT_EQ(nullptr, N->getMacros().get()); |
| } |
| |
| typedef MetadataTest DISubprogramTest; |
| |
| TEST_F(DISubprogramTest, get) { |
| DIScope *Scope = getCompositeType(); |
| StringRef Name = "name"; |
| StringRef LinkageName = "linkage"; |
| DIFile *File = getFile(); |
| unsigned Line = 2; |
| DISubroutineType *Type = getSubroutineType(); |
| bool IsLocalToUnit = false; |
| bool IsDefinition = true; |
| unsigned ScopeLine = 3; |
| DIType *ContainingType = getCompositeType(); |
| unsigned Virtuality = 2; |
| unsigned VirtualIndex = 5; |
| int ThisAdjustment = -3; |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(6); |
| bool IsOptimized = false; |
| MDTuple *TemplateParams = getTuple(); |
| DISubprogram *Declaration = getSubprogram(); |
| MDTuple *RetainedNodes = getTuple(); |
| MDTuple *ThrownTypes = getTuple(); |
| DICompileUnit *Unit = getUnit(); |
| |
| auto *N = DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, IsLocalToUnit, |
| IsDefinition, ScopeLine, ContainingType, Virtuality, VirtualIndex, |
| ThisAdjustment, Flags, IsOptimized, Unit, TemplateParams, Declaration, |
| RetainedNodes, ThrownTypes); |
| |
| EXPECT_EQ(dwarf::DW_TAG_subprogram, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(LinkageName, N->getLinkageName()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(Type, N->getType()); |
| EXPECT_EQ(IsLocalToUnit, N->isLocalToUnit()); |
| EXPECT_EQ(IsDefinition, N->isDefinition()); |
| EXPECT_EQ(ScopeLine, N->getScopeLine()); |
| EXPECT_EQ(ContainingType, N->getContainingType()); |
| EXPECT_EQ(Virtuality, N->getVirtuality()); |
| EXPECT_EQ(VirtualIndex, N->getVirtualIndex()); |
| EXPECT_EQ(ThisAdjustment, N->getThisAdjustment()); |
| EXPECT_EQ(Flags, N->getFlags()); |
| EXPECT_EQ(IsOptimized, N->isOptimized()); |
| EXPECT_EQ(Unit, N->getUnit()); |
| EXPECT_EQ(TemplateParams, N->getTemplateParams().get()); |
| EXPECT_EQ(Declaration, N->getDeclaration()); |
| EXPECT_EQ(RetainedNodes, N->getRetainedNodes().get()); |
| EXPECT_EQ(ThrownTypes, N->getThrownTypes().get()); |
| EXPECT_EQ(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| |
| EXPECT_NE(N, DISubprogram::get( |
| Context, getCompositeType(), Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, "other", LinkageName, File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, "other", File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, getFile(), Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line + 1, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| getSubroutineType(), IsLocalToUnit, |
| IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, |
| Flags, IsOptimized, Unit, TemplateParams, |
| Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| !IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| IsLocalToUnit, !IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine + 1, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, getCompositeType(), |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, |
| ContainingType, Virtuality + 1, VirtualIndex, |
| ThisAdjustment, Flags, IsOptimized, Unit, |
| TemplateParams, Declaration, RetainedNodes, |
| ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, |
| ContainingType, Virtuality, VirtualIndex + 1, |
| ThisAdjustment, Flags, IsOptimized, Unit, |
| TemplateParams, Declaration, RetainedNodes, |
| ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, |
| ContainingType, Virtuality, VirtualIndex, |
| ThisAdjustment, Flags, !IsOptimized, Unit, |
| TemplateParams, Declaration, RetainedNodes, |
| ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, |
| ContainingType, Virtuality, VirtualIndex, |
| ThisAdjustment, Flags, IsOptimized, nullptr, |
| TemplateParams, Declaration, RetainedNodes, |
| ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, getTuple(), Declaration, RetainedNodes, ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, |
| ContainingType, Virtuality, VirtualIndex, |
| ThisAdjustment, Flags, IsOptimized, Unit, |
| TemplateParams, getSubprogram(), RetainedNodes, |
| ThrownTypes)); |
| EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, ScopeLine, |
| ContainingType, Virtuality, VirtualIndex, |
| ThisAdjustment, Flags, IsOptimized, Unit, |
| TemplateParams, Declaration, getTuple())); |
| EXPECT_NE(N, DISubprogram::get( |
| Context, Scope, Name, LinkageName, File, Line, Type, |
| IsLocalToUnit, IsDefinition, ScopeLine, ContainingType, |
| Virtuality, VirtualIndex, ThisAdjustment, Flags, IsOptimized, |
| Unit, TemplateParams, Declaration, RetainedNodes, getTuple())); |
| |
| TempDISubprogram Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DILexicalBlockTest; |
| |
| TEST_F(DILexicalBlockTest, get) { |
| DILocalScope *Scope = getSubprogram(); |
| DIFile *File = getFile(); |
| unsigned Line = 5; |
| unsigned Column = 8; |
| |
| auto *N = DILexicalBlock::get(Context, Scope, File, Line, Column); |
| |
| EXPECT_EQ(dwarf::DW_TAG_lexical_block, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(Column, N->getColumn()); |
| EXPECT_EQ(N, DILexicalBlock::get(Context, Scope, File, Line, Column)); |
| |
| EXPECT_NE(N, |
| DILexicalBlock::get(Context, getSubprogram(), File, Line, Column)); |
| EXPECT_NE(N, DILexicalBlock::get(Context, Scope, getFile(), Line, Column)); |
| EXPECT_NE(N, DILexicalBlock::get(Context, Scope, File, Line + 1, Column)); |
| EXPECT_NE(N, DILexicalBlock::get(Context, Scope, File, Line, Column + 1)); |
| |
| TempDILexicalBlock Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DILexicalBlockTest, Overflow) { |
| DISubprogram *SP = getSubprogram(); |
| DIFile *F = getFile(); |
| { |
| auto *LB = DILexicalBlock::get(Context, SP, F, 2, 7); |
| EXPECT_EQ(2u, LB->getLine()); |
| EXPECT_EQ(7u, LB->getColumn()); |
| } |
| unsigned U16 = 1u << 16; |
| { |
| auto *LB = DILexicalBlock::get(Context, SP, F, UINT32_MAX, U16 - 1); |
| EXPECT_EQ(UINT32_MAX, LB->getLine()); |
| EXPECT_EQ(U16 - 1, LB->getColumn()); |
| } |
| { |
| auto *LB = DILexicalBlock::get(Context, SP, F, UINT32_MAX, U16); |
| EXPECT_EQ(UINT32_MAX, LB->getLine()); |
| EXPECT_EQ(0u, LB->getColumn()); |
| } |
| { |
| auto *LB = DILexicalBlock::get(Context, SP, F, UINT32_MAX, U16 + 1); |
| EXPECT_EQ(UINT32_MAX, LB->getLine()); |
| EXPECT_EQ(0u, LB->getColumn()); |
| } |
| } |
| |
| typedef MetadataTest DILexicalBlockFileTest; |
| |
| TEST_F(DILexicalBlockFileTest, get) { |
| DILocalScope *Scope = getSubprogram(); |
| DIFile *File = getFile(); |
| unsigned Discriminator = 5; |
| |
| auto *N = DILexicalBlockFile::get(Context, Scope, File, Discriminator); |
| |
| EXPECT_EQ(dwarf::DW_TAG_lexical_block, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Discriminator, N->getDiscriminator()); |
| EXPECT_EQ(N, DILexicalBlockFile::get(Context, Scope, File, Discriminator)); |
| |
| EXPECT_NE(N, DILexicalBlockFile::get(Context, getSubprogram(), File, |
| Discriminator)); |
| EXPECT_NE(N, |
| DILexicalBlockFile::get(Context, Scope, getFile(), Discriminator)); |
| EXPECT_NE(N, |
| DILexicalBlockFile::get(Context, Scope, File, Discriminator + 1)); |
| |
| TempDILexicalBlockFile Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DINamespaceTest; |
| |
| TEST_F(DINamespaceTest, get) { |
| DIScope *Scope = getFile(); |
| StringRef Name = "namespace"; |
| bool ExportSymbols = true; |
| |
| auto *N = DINamespace::get(Context, Scope, Name, ExportSymbols); |
| |
| EXPECT_EQ(dwarf::DW_TAG_namespace, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(N, DINamespace::get(Context, Scope, Name, ExportSymbols)); |
| EXPECT_NE(N, DINamespace::get(Context, getFile(), Name, ExportSymbols)); |
| EXPECT_NE(N, DINamespace::get(Context, Scope, "other", ExportSymbols)); |
| EXPECT_NE(N, DINamespace::get(Context, Scope, Name, !ExportSymbols)); |
| |
| TempDINamespace Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DIModuleTest; |
| |
| TEST_F(DIModuleTest, get) { |
| DIScope *Scope = getFile(); |
| StringRef Name = "module"; |
| StringRef ConfigMacro = "-DNDEBUG"; |
| StringRef Includes = "-I."; |
| StringRef Sysroot = "/"; |
| |
| auto *N = DIModule::get(Context, Scope, Name, ConfigMacro, Includes, Sysroot); |
| |
| EXPECT_EQ(dwarf::DW_TAG_module, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(ConfigMacro, N->getConfigurationMacros()); |
| EXPECT_EQ(Includes, N->getIncludePath()); |
| EXPECT_EQ(Sysroot, N->getISysRoot()); |
| EXPECT_EQ(N, DIModule::get(Context, Scope, Name, |
| ConfigMacro, Includes, Sysroot)); |
| EXPECT_NE(N, DIModule::get(Context, getFile(), Name, |
| ConfigMacro, Includes, Sysroot)); |
| EXPECT_NE(N, DIModule::get(Context, Scope, "other", |
| ConfigMacro, Includes, Sysroot)); |
| EXPECT_NE(N, DIModule::get(Context, Scope, Name, |
| "other", Includes, Sysroot)); |
| EXPECT_NE(N, DIModule::get(Context, Scope, Name, |
| ConfigMacro, "other", Sysroot)); |
| EXPECT_NE(N, DIModule::get(Context, Scope, Name, |
| ConfigMacro, Includes, "other")); |
| |
| TempDIModule Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DITemplateTypeParameterTest; |
| |
| TEST_F(DITemplateTypeParameterTest, get) { |
| StringRef Name = "template"; |
| DIType *Type = getBasicType("basic"); |
| |
| auto *N = DITemplateTypeParameter::get(Context, Name, Type); |
| |
| EXPECT_EQ(dwarf::DW_TAG_template_type_parameter, N->getTag()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(Type, N->getType()); |
| EXPECT_EQ(N, DITemplateTypeParameter::get(Context, Name, Type)); |
| |
| EXPECT_NE(N, DITemplateTypeParameter::get(Context, "other", Type)); |
| EXPECT_NE(N, |
| DITemplateTypeParameter::get(Context, Name, getBasicType("other"))); |
| |
| TempDITemplateTypeParameter Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DITemplateValueParameterTest; |
| |
| TEST_F(DITemplateValueParameterTest, get) { |
| unsigned Tag = dwarf::DW_TAG_template_value_parameter; |
| StringRef Name = "template"; |
| DIType *Type = getBasicType("basic"); |
| Metadata *Value = getConstantAsMetadata(); |
| |
| auto *N = DITemplateValueParameter::get(Context, Tag, Name, Type, Value); |
| EXPECT_EQ(Tag, N->getTag()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(Type, N->getType()); |
| EXPECT_EQ(Value, N->getValue()); |
| EXPECT_EQ(N, DITemplateValueParameter::get(Context, Tag, Name, Type, Value)); |
| |
| EXPECT_NE(N, DITemplateValueParameter::get( |
| Context, dwarf::DW_TAG_GNU_template_template_param, Name, |
| Type, Value)); |
| EXPECT_NE(N, |
| DITemplateValueParameter::get(Context, Tag, "other", Type, Value)); |
| EXPECT_NE(N, DITemplateValueParameter::get(Context, Tag, Name, |
| getBasicType("other"), Value)); |
| EXPECT_NE(N, DITemplateValueParameter::get(Context, Tag, Name, Type, |
| getConstantAsMetadata())); |
| |
| TempDITemplateValueParameter Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DIGlobalVariableTest; |
| |
| TEST_F(DIGlobalVariableTest, get) { |
| DIScope *Scope = getSubprogram(); |
| StringRef Name = "name"; |
| StringRef LinkageName = "linkage"; |
| DIFile *File = getFile(); |
| unsigned Line = 5; |
| DIType *Type = getDerivedType(); |
| bool IsLocalToUnit = false; |
| bool IsDefinition = true; |
| DIDerivedType *StaticDataMemberDeclaration = |
| cast<DIDerivedType>(getDerivedType()); |
| uint32_t AlignInBits = 8; |
| |
| auto *N = DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, Line, |
| Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits); |
| EXPECT_EQ(dwarf::DW_TAG_variable, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(LinkageName, N->getLinkageName()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(Type, N->getType()); |
| EXPECT_EQ(IsLocalToUnit, N->isLocalToUnit()); |
| EXPECT_EQ(IsDefinition, N->isDefinition()); |
| EXPECT_EQ(StaticDataMemberDeclaration, N->getStaticDataMemberDeclaration()); |
| EXPECT_EQ(AlignInBits, N->getAlignInBits()); |
| EXPECT_EQ(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| |
| EXPECT_NE(N, |
| DIGlobalVariable::get(Context, getSubprogram(), Name, LinkageName, |
| File, Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, "other", LinkageName, File, |
| Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, "other", File, Line, |
| Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, |
| DIGlobalVariable::get(Context, Scope, Name, LinkageName, getFile(), |
| Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, |
| DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line + 1, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, |
| DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, Line, |
| getDerivedType(), IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line, Type, !IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line, Type, IsLocalToUnit, !IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits)); |
| EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line, Type, IsLocalToUnit, IsDefinition, |
| cast<DIDerivedType>(getDerivedType()), |
| AlignInBits)); |
| EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, |
| (AlignInBits << 1))); |
| |
| TempDIGlobalVariable Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DIGlobalVariableExpressionTest; |
| |
| TEST_F(DIGlobalVariableExpressionTest, get) { |
| DIScope *Scope = getSubprogram(); |
| StringRef Name = "name"; |
| StringRef LinkageName = "linkage"; |
| DIFile *File = getFile(); |
| unsigned Line = 5; |
| DIType *Type = getDerivedType(); |
| bool IsLocalToUnit = false; |
| bool IsDefinition = true; |
| auto *Expr = DIExpression::get(Context, {1, 2}); |
| auto *Expr2 = DIExpression::get(Context, {1, 2, 3}); |
| DIDerivedType *StaticDataMemberDeclaration = |
| cast<DIDerivedType>(getDerivedType()); |
| uint32_t AlignInBits = 8; |
| |
| auto *Var = DIGlobalVariable::get(Context, Scope, Name, LinkageName, File, |
| Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits); |
| auto *Var2 = DIGlobalVariable::get(Context, Scope, "other", LinkageName, File, |
| Line, Type, IsLocalToUnit, IsDefinition, |
| StaticDataMemberDeclaration, AlignInBits); |
| auto *N = DIGlobalVariableExpression::get(Context, Var, Expr); |
| |
| EXPECT_EQ(Var, N->getVariable()); |
| EXPECT_EQ(Expr, N->getExpression()); |
| EXPECT_EQ(N, DIGlobalVariableExpression::get(Context, Var, Expr)); |
| EXPECT_NE(N, DIGlobalVariableExpression::get(Context, Var2, Expr)); |
| EXPECT_NE(N, DIGlobalVariableExpression::get(Context, Var, Expr2)); |
| |
| TempDIGlobalVariableExpression Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DILocalVariableTest; |
| |
| TEST_F(DILocalVariableTest, get) { |
| DILocalScope *Scope = getSubprogram(); |
| StringRef Name = "name"; |
| DIFile *File = getFile(); |
| unsigned Line = 5; |
| DIType *Type = getDerivedType(); |
| unsigned Arg = 6; |
| DINode::DIFlags Flags = static_cast<DINode::DIFlags>(7); |
| uint32_t AlignInBits = 8; |
| |
| auto *N = |
| DILocalVariable::get(Context, Scope, Name, File, Line, Type, Arg, Flags, |
| AlignInBits); |
| EXPECT_TRUE(N->isParameter()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(Type, N->getType()); |
| EXPECT_EQ(Arg, N->getArg()); |
| EXPECT_EQ(Flags, N->getFlags()); |
| EXPECT_EQ(AlignInBits, N->getAlignInBits()); |
| EXPECT_EQ(N, DILocalVariable::get(Context, Scope, Name, File, Line, Type, Arg, |
| Flags, AlignInBits)); |
| |
| EXPECT_FALSE( |
| DILocalVariable::get(Context, Scope, Name, File, Line, Type, 0, Flags, |
| AlignInBits)->isParameter()); |
| EXPECT_NE(N, DILocalVariable::get(Context, getSubprogram(), Name, File, Line, |
| Type, Arg, Flags, AlignInBits)); |
| EXPECT_NE(N, DILocalVariable::get(Context, Scope, "other", File, Line, Type, |
| Arg, Flags, AlignInBits)); |
| EXPECT_NE(N, DILocalVariable::get(Context, Scope, Name, getFile(), Line, Type, |
| Arg, Flags, AlignInBits)); |
| EXPECT_NE(N, DILocalVariable::get(Context, Scope, Name, File, Line + 1, Type, |
| Arg, Flags, AlignInBits)); |
| EXPECT_NE(N, DILocalVariable::get(Context, Scope, Name, File, Line, |
| getDerivedType(), Arg, Flags, AlignInBits)); |
| EXPECT_NE(N, DILocalVariable::get(Context, Scope, Name, File, Line, Type, |
| Arg + 1, Flags, AlignInBits)); |
| EXPECT_NE(N, DILocalVariable::get(Context, Scope, Name, File, Line, Type, |
| Arg, Flags, (AlignInBits << 1))); |
| |
| TempDILocalVariable Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| TEST_F(DILocalVariableTest, getArg256) { |
| EXPECT_EQ(255u, DILocalVariable::get(Context, getSubprogram(), "", getFile(), |
| 0, nullptr, 255, DINode::FlagZero, 0) |
| ->getArg()); |
| EXPECT_EQ(256u, DILocalVariable::get(Context, getSubprogram(), "", getFile(), |
| 0, nullptr, 256, DINode::FlagZero, 0) |
| ->getArg()); |
| EXPECT_EQ(257u, DILocalVariable::get(Context, getSubprogram(), "", getFile(), |
| 0, nullptr, 257, DINode::FlagZero, 0) |
| ->getArg()); |
| unsigned Max = UINT16_MAX; |
| EXPECT_EQ(Max, DILocalVariable::get(Context, getSubprogram(), "", getFile(), |
| 0, nullptr, Max, DINode::FlagZero, 0) |
| ->getArg()); |
| } |
| |
| typedef MetadataTest DIExpressionTest; |
| |
| TEST_F(DIExpressionTest, get) { |
| uint64_t Elements[] = {2, 6, 9, 78, 0}; |
| auto *N = DIExpression::get(Context, Elements); |
| EXPECT_EQ(makeArrayRef(Elements), N->getElements()); |
| EXPECT_EQ(N, DIExpression::get(Context, Elements)); |
| |
| EXPECT_EQ(5u, N->getNumElements()); |
| EXPECT_EQ(2u, N->getElement(0)); |
| EXPECT_EQ(6u, N->getElement(1)); |
| EXPECT_EQ(9u, N->getElement(2)); |
| EXPECT_EQ(78u, N->getElement(3)); |
| EXPECT_EQ(0u, N->getElement(4)); |
| |
| TempDIExpression Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| |
| // Test DIExpression::prepend(). |
| uint64_t Elts0[] = {dwarf::DW_OP_LLVM_fragment, 0, 32}; |
| auto *N0 = DIExpression::get(Context, Elts0); |
| auto *N0WithPrependedOps = DIExpression::prepend(N0, true, 64, true, true); |
| uint64_t Elts1[] = {dwarf::DW_OP_deref, |
| dwarf::DW_OP_plus_uconst, 64, |
| dwarf::DW_OP_deref, |
| dwarf::DW_OP_stack_value, |
| dwarf::DW_OP_LLVM_fragment, 0, 32}; |
| auto *N1 = DIExpression::get(Context, Elts1); |
| EXPECT_EQ(N0WithPrependedOps, N1); |
| |
| // Test DIExpression::append(). |
| uint64_t Elts2[] = {dwarf::DW_OP_deref, dwarf::DW_OP_plus_uconst, 64, |
| dwarf::DW_OP_deref, dwarf::DW_OP_stack_value}; |
| auto *N2 = DIExpression::append(N0, Elts2); |
| EXPECT_EQ(N0WithPrependedOps, N2); |
| } |
| |
| TEST_F(DIExpressionTest, isValid) { |
| #define EXPECT_VALID(...) \ |
| do { \ |
| uint64_t Elements[] = {__VA_ARGS__}; \ |
| EXPECT_TRUE(DIExpression::get(Context, Elements)->isValid()); \ |
| } while (false) |
| #define EXPECT_INVALID(...) \ |
| do { \ |
| uint64_t Elements[] = {__VA_ARGS__}; \ |
| EXPECT_FALSE(DIExpression::get(Context, Elements)->isValid()); \ |
| } while (false) |
| |
| // Empty expression should be valid. |
| EXPECT_TRUE(DIExpression::get(Context, None)); |
| |
| // Valid constructions. |
| EXPECT_VALID(dwarf::DW_OP_plus_uconst, 6); |
| EXPECT_VALID(dwarf::DW_OP_constu, 6, dwarf::DW_OP_plus); |
| EXPECT_VALID(dwarf::DW_OP_deref); |
| EXPECT_VALID(dwarf::DW_OP_LLVM_fragment, 3, 7); |
| EXPECT_VALID(dwarf::DW_OP_plus_uconst, 6, dwarf::DW_OP_deref); |
| EXPECT_VALID(dwarf::DW_OP_deref, dwarf::DW_OP_plus_uconst, 6); |
| EXPECT_VALID(dwarf::DW_OP_deref, dwarf::DW_OP_LLVM_fragment, 3, 7); |
| EXPECT_VALID(dwarf::DW_OP_deref, dwarf::DW_OP_plus_uconst, 6, |
| dwarf::DW_OP_LLVM_fragment, 3, 7); |
| |
| // Invalid constructions. |
| EXPECT_INVALID(~0u); |
| EXPECT_INVALID(dwarf::DW_OP_plus, 0); |
| EXPECT_INVALID(dwarf::DW_OP_plus_uconst); |
| EXPECT_INVALID(dwarf::DW_OP_LLVM_fragment); |
| EXPECT_INVALID(dwarf::DW_OP_LLVM_fragment, 3); |
| EXPECT_INVALID(dwarf::DW_OP_LLVM_fragment, 3, 7, dwarf::DW_OP_plus_uconst, 3); |
| EXPECT_INVALID(dwarf::DW_OP_LLVM_fragment, 3, 7, dwarf::DW_OP_deref); |
| |
| #undef EXPECT_VALID |
| #undef EXPECT_INVALID |
| } |
| |
| typedef MetadataTest DIObjCPropertyTest; |
| |
| TEST_F(DIObjCPropertyTest, get) { |
| StringRef Name = "name"; |
| DIFile *File = getFile(); |
| unsigned Line = 5; |
| StringRef GetterName = "getter"; |
| StringRef SetterName = "setter"; |
| unsigned Attributes = 7; |
| DIType *Type = getBasicType("basic"); |
| |
| auto *N = DIObjCProperty::get(Context, Name, File, Line, GetterName, |
| SetterName, Attributes, Type); |
| |
| EXPECT_EQ(dwarf::DW_TAG_APPLE_property, N->getTag()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(GetterName, N->getGetterName()); |
| EXPECT_EQ(SetterName, N->getSetterName()); |
| EXPECT_EQ(Attributes, N->getAttributes()); |
| EXPECT_EQ(Type, N->getType()); |
| EXPECT_EQ(N, DIObjCProperty::get(Context, Name, File, Line, GetterName, |
| SetterName, Attributes, Type)); |
| |
| EXPECT_NE(N, DIObjCProperty::get(Context, "other", File, Line, GetterName, |
| SetterName, Attributes, Type)); |
| EXPECT_NE(N, DIObjCProperty::get(Context, Name, getFile(), Line, GetterName, |
| SetterName, Attributes, Type)); |
| EXPECT_NE(N, DIObjCProperty::get(Context, Name, File, Line + 1, GetterName, |
| SetterName, Attributes, Type)); |
| EXPECT_NE(N, DIObjCProperty::get(Context, Name, File, Line, "other", |
| SetterName, Attributes, Type)); |
| EXPECT_NE(N, DIObjCProperty::get(Context, Name, File, Line, GetterName, |
| "other", Attributes, Type)); |
| EXPECT_NE(N, DIObjCProperty::get(Context, Name, File, Line, GetterName, |
| SetterName, Attributes + 1, Type)); |
| EXPECT_NE(N, DIObjCProperty::get(Context, Name, File, Line, GetterName, |
| SetterName, Attributes, |
| getBasicType("other"))); |
| |
| TempDIObjCProperty Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest DIImportedEntityTest; |
| |
| TEST_F(DIImportedEntityTest, get) { |
| unsigned Tag = dwarf::DW_TAG_imported_module; |
| DIScope *Scope = getSubprogram(); |
| DINode *Entity = getCompositeType(); |
| DIFile *File = getFile(); |
| unsigned Line = 5; |
| StringRef Name = "name"; |
| |
| auto *N = |
| DIImportedEntity::get(Context, Tag, Scope, Entity, File, Line, Name); |
| |
| EXPECT_EQ(Tag, N->getTag()); |
| EXPECT_EQ(Scope, N->getScope()); |
| EXPECT_EQ(Entity, N->getEntity()); |
| EXPECT_EQ(File, N->getFile()); |
| EXPECT_EQ(Line, N->getLine()); |
| EXPECT_EQ(Name, N->getName()); |
| EXPECT_EQ( |
| N, DIImportedEntity::get(Context, Tag, Scope, Entity, File, Line, Name)); |
| |
| EXPECT_NE(N, |
| DIImportedEntity::get(Context, dwarf::DW_TAG_imported_declaration, |
| Scope, Entity, File, Line, Name)); |
| EXPECT_NE(N, DIImportedEntity::get(Context, Tag, getSubprogram(), Entity, |
| File, Line, Name)); |
| EXPECT_NE(N, DIImportedEntity::get(Context, Tag, Scope, getCompositeType(), |
| File, Line, Name)); |
| EXPECT_NE(N, DIImportedEntity::get(Context, Tag, Scope, Entity, nullptr, Line, |
| Name)); |
| EXPECT_NE(N, DIImportedEntity::get(Context, Tag, Scope, Entity, File, |
| Line + 1, Name)); |
| EXPECT_NE(N, DIImportedEntity::get(Context, Tag, Scope, Entity, File, Line, |
| "other")); |
| |
| TempDIImportedEntity Temp = N->clone(); |
| EXPECT_EQ(N, MDNode::replaceWithUniqued(std::move(Temp))); |
| } |
| |
| typedef MetadataTest MetadataAsValueTest; |
| |
| TEST_F(MetadataAsValueTest, MDNode) { |
| MDNode *N = MDNode::get(Context, None); |
| auto *V = MetadataAsValue::get(Context, N); |
| EXPECT_TRUE(V->getType()->isMetadataTy()); |
| EXPECT_EQ(N, V->getMetadata()); |
| |
| auto *V2 = MetadataAsValue::get(Context, N); |
| EXPECT_EQ(V, V2); |
| } |
| |
| TEST_F(MetadataAsValueTest, MDNodeMDNode) { |
| MDNode *N = MDNode::get(Context, None); |
| Metadata *Ops[] = {N}; |
| MDNode *N2 = MDNode::get(Context, Ops); |
| auto *V = MetadataAsValue::get(Context, N2); |
| EXPECT_TRUE(V->getType()->isMetadataTy()); |
| EXPECT_EQ(N2, V->getMetadata()); |
| |
| auto *V2 = MetadataAsValue::get(Context, N2); |
| EXPECT_EQ(V, V2); |
| |
| auto *V3 = MetadataAsValue::get(Context, N); |
| EXPECT_TRUE(V3->getType()->isMetadataTy()); |
| EXPECT_NE(V, V3); |
| EXPECT_EQ(N, V3->getMetadata()); |
| } |
| |
| TEST_F(MetadataAsValueTest, MDNodeConstant) { |
| auto *C = ConstantInt::getTrue(Context); |
| auto *MD = ConstantAsMetadata::get(C); |
| Metadata *Ops[] = {MD}; |
| auto *N = MDNode::get(Context, Ops); |
| |
| auto *V = MetadataAsValue::get(Context, MD); |
| EXPECT_TRUE(V->getType()->isMetadataTy()); |
| EXPECT_EQ(MD, V->getMetadata()); |
| |
| auto *V2 = MetadataAsValue::get(Context, N); |
| EXPECT_EQ(MD, V2->getMetadata()); |
| EXPECT_EQ(V, V2); |
| } |
| |
| typedef MetadataTest ValueAsMetadataTest; |
| |
| TEST_F(ValueAsMetadataTest, UpdatesOnRAUW) { |
| Type *Ty = Type::getInt1PtrTy(Context); |
| std::unique_ptr<GlobalVariable> GV0( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| auto *MD = ValueAsMetadata::get(GV0.get()); |
| EXPECT_TRUE(MD->getValue() == GV0.get()); |
| ASSERT_TRUE(GV0->use_empty()); |
| |
| std::unique_ptr<GlobalVariable> GV1( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| GV0->replaceAllUsesWith(GV1.get()); |
| EXPECT_TRUE(MD->getValue() == GV1.get()); |
| } |
| |
| TEST_F(ValueAsMetadataTest, TempTempReplacement) { |
| // Create a constant. |
| ConstantAsMetadata *CI = |
| ConstantAsMetadata::get(ConstantInt::get(Context, APInt(8, 0))); |
| |
| auto Temp1 = MDTuple::getTemporary(Context, None); |
| auto Temp2 = MDTuple::getTemporary(Context, {CI}); |
| auto *N = MDTuple::get(Context, {Temp1.get()}); |
| |
| // Test replacing a temporary node with another temporary node. |
| Temp1->replaceAllUsesWith(Temp2.get()); |
| EXPECT_EQ(N->getOperand(0), Temp2.get()); |
| |
| // Clean up Temp2 for teardown. |
| Temp2->replaceAllUsesWith(nullptr); |
| } |
| |
| TEST_F(ValueAsMetadataTest, CollidingDoubleUpdates) { |
| // Create a constant. |
| ConstantAsMetadata *CI = |
| ConstantAsMetadata::get(ConstantInt::get(Context, APInt(8, 0))); |
| |
| // Create a temporary to prevent nodes from resolving. |
| auto Temp = MDTuple::getTemporary(Context, None); |
| |
| // When the first operand of N1 gets reset to nullptr, it'll collide with N2. |
| Metadata *Ops1[] = {CI, CI, Temp.get()}; |
| Metadata *Ops2[] = {nullptr, CI, Temp.get()}; |
| |
| auto *N1 = MDTuple::get(Context, Ops1); |
| auto *N2 = MDTuple::get(Context, Ops2); |
| ASSERT_NE(N1, N2); |
| |
| // Tell metadata that the constant is getting deleted. |
| // |
| // After this, N1 will be invalid, so don't touch it. |
| ValueAsMetadata::handleDeletion(CI->getValue()); |
| EXPECT_EQ(nullptr, N2->getOperand(0)); |
| EXPECT_EQ(nullptr, N2->getOperand(1)); |
| EXPECT_EQ(Temp.get(), N2->getOperand(2)); |
| |
| // Clean up Temp for teardown. |
| Temp->replaceAllUsesWith(nullptr); |
| } |
| |
| typedef MetadataTest TrackingMDRefTest; |
| |
| TEST_F(TrackingMDRefTest, UpdatesOnRAUW) { |
| Type *Ty = Type::getInt1PtrTy(Context); |
| std::unique_ptr<GlobalVariable> GV0( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| TypedTrackingMDRef<ValueAsMetadata> MD(ValueAsMetadata::get(GV0.get())); |
| EXPECT_TRUE(MD->getValue() == GV0.get()); |
| ASSERT_TRUE(GV0->use_empty()); |
| |
| std::unique_ptr<GlobalVariable> GV1( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| GV0->replaceAllUsesWith(GV1.get()); |
| EXPECT_TRUE(MD->getValue() == GV1.get()); |
| |
| // Reset it, so we don't inadvertently test deletion. |
| MD.reset(); |
| } |
| |
| TEST_F(TrackingMDRefTest, UpdatesOnDeletion) { |
| Type *Ty = Type::getInt1PtrTy(Context); |
| std::unique_ptr<GlobalVariable> GV( |
| new GlobalVariable(Ty, false, GlobalValue::ExternalLinkage)); |
| TypedTrackingMDRef<ValueAsMetadata> MD(ValueAsMetadata::get(GV.get())); |
| EXPECT_TRUE(MD->getValue() == GV.get()); |
| ASSERT_TRUE(GV->use_empty()); |
| |
| GV.reset(); |
| EXPECT_TRUE(!MD); |
| } |
| |
| TEST(NamedMDNodeTest, Search) { |
| LLVMContext Context; |
| ConstantAsMetadata *C = |
| ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Context), 1)); |
| ConstantAsMetadata *C2 = |
| ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Context), 2)); |
| |
| Metadata *const V = C; |
| Metadata *const V2 = C2; |
| MDNode *n = MDNode::get(Context, V); |
| MDNode *n2 = MDNode::get(Context, V2); |
| |
| Module M("MyModule", Context); |
| const char *Name = "llvm.NMD1"; |
| NamedMDNode *NMD = M.getOrInsertNamedMetadata(Name); |
| NMD->addOperand(n); |
| NMD->addOperand(n2); |
| |
| std::string Str; |
| raw_string_ostream oss(Str); |
| NMD->print(oss); |
| EXPECT_STREQ("!llvm.NMD1 = !{!0, !1}\n", |
| oss.str().c_str()); |
| } |
| |
| typedef MetadataTest FunctionAttachmentTest; |
| TEST_F(FunctionAttachmentTest, setMetadata) { |
| Function *F = getFunction("foo"); |
| ASSERT_FALSE(F->hasMetadata()); |
| EXPECT_EQ(nullptr, F->getMetadata(LLVMContext::MD_dbg)); |
| EXPECT_EQ(nullptr, F->getMetadata("dbg")); |
| EXPECT_EQ(nullptr, F->getMetadata("other")); |
| |
| DISubprogram *SP1 = getSubprogram(); |
| DISubprogram *SP2 = getSubprogram(); |
| ASSERT_NE(SP1, SP2); |
| |
| F->setMetadata("dbg", SP1); |
| EXPECT_TRUE(F->hasMetadata()); |
| EXPECT_EQ(SP1, F->getMetadata(LLVMContext::MD_dbg)); |
| EXPECT_EQ(SP1, F->getMetadata("dbg")); |
| EXPECT_EQ(nullptr, F->getMetadata("other")); |
| |
| F->setMetadata(LLVMContext::MD_dbg, SP2); |
| EXPECT_TRUE(F->hasMetadata()); |
| EXPECT_EQ(SP2, F->getMetadata(LLVMContext::MD_dbg)); |
| EXPECT_EQ(SP2, F->getMetadata("dbg")); |
| EXPECT_EQ(nullptr, F->getMetadata("other")); |
| |
| F->setMetadata("dbg", nullptr); |
| EXPECT_FALSE(F->hasMetadata()); |
| EXPECT_EQ(nullptr, F->getMetadata(LLVMContext::MD_dbg)); |
| EXPECT_EQ(nullptr, F->getMetadata("dbg")); |
| EXPECT_EQ(nullptr, F->getMetadata("other")); |
| |
| MDTuple *T1 = getTuple(); |
| MDTuple *T2 = getTuple(); |
| ASSERT_NE(T1, T2); |
| |
| F->setMetadata("other1", T1); |
| F->setMetadata("other2", T2); |
| EXPECT_TRUE(F->hasMetadata()); |
| EXPECT_EQ(T1, F->getMetadata("other1")); |
| EXPECT_EQ(T2, F->getMetadata("other2")); |
| EXPECT_EQ(nullptr, F->getMetadata("dbg")); |
| |
| F->setMetadata("other1", T2); |
| F->setMetadata("other2", T1); |
| EXPECT_EQ(T2, F->getMetadata("other1")); |
| EXPECT_EQ(T1, F->getMetadata("other2")); |
| |
| F->setMetadata("other1", nullptr); |
| F->setMetadata("other2", nullptr); |
| EXPECT_FALSE(F->hasMetadata()); |
| EXPECT_EQ(nullptr, F->getMetadata("other1")); |
| EXPECT_EQ(nullptr, F->getMetadata("other2")); |
| } |
| |
| TEST_F(FunctionAttachmentTest, getAll) { |
| Function *F = getFunction("foo"); |
| |
| MDTuple *T1 = getTuple(); |
| MDTuple *T2 = getTuple(); |
| MDTuple *P = getTuple(); |
| DISubprogram *SP = getSubprogram(); |
| |
| F->setMetadata("other1", T2); |
| F->setMetadata(LLVMContext::MD_dbg, SP); |
| F->setMetadata("other2", T1); |
| F->setMetadata(LLVMContext::MD_prof, P); |
| F->setMetadata("other2", T2); |
| F->setMetadata("other1", T1); |
| |
| SmallVector<std::pair<unsigned, MDNode *>, 4> MDs; |
| F->getAllMetadata(MDs); |
| ASSERT_EQ(4u, MDs.size()); |
| EXPECT_EQ(LLVMContext::MD_dbg, MDs[0].first); |
| EXPECT_EQ(LLVMContext::MD_prof, MDs[1].first); |
| EXPECT_EQ(Context.getMDKindID("other1"), MDs[2].first); |
| EXPECT_EQ(Context.getMDKindID("other2"), MDs[3].first); |
| EXPECT_EQ(SP, MDs[0].second); |
| EXPECT_EQ(P, MDs[1].second); |
| EXPECT_EQ(T1, MDs[2].second); |
| EXPECT_EQ(T2, MDs[3].second); |
| } |
| |
| TEST_F(FunctionAttachmentTest, Verifier) { |
| Function *F = getFunction("foo"); |
| F->setMetadata("attach", getTuple()); |
| F->setIsMaterializable(true); |
| |
| // Confirm this is materializable. |
| ASSERT_TRUE(F->isMaterializable()); |
| |
| // Materializable functions cannot have metadata attachments. |
| EXPECT_TRUE(verifyFunction(*F)); |
| |
| // Function declarations can. |
| F->setIsMaterializable(false); |
| EXPECT_FALSE(verifyModule(*F->getParent())); |
| EXPECT_FALSE(verifyFunction(*F)); |
| |
| // So can definitions. |
| (void)new UnreachableInst(Context, BasicBlock::Create(Context, "bb", F)); |
| EXPECT_FALSE(verifyModule(*F->getParent())); |
| EXPECT_FALSE(verifyFunction(*F)); |
| } |
| |
| TEST_F(FunctionAttachmentTest, EntryCount) { |
| Function *F = getFunction("foo"); |
| EXPECT_FALSE(F->getEntryCount().hasValue()); |
| F->setEntryCount(12304, Function::PCT_Real); |
| auto Count = F->getEntryCount(); |
| EXPECT_TRUE(Count.hasValue()); |
| EXPECT_EQ(12304u, Count.getCount()); |
| EXPECT_EQ(Function::PCT_Real, Count.getType()); |
| |
| // Repeat the same for synthetic counts. |
| F = getFunction("bar"); |
| EXPECT_FALSE(F->getEntryCount().hasValue()); |
| F->setEntryCount(123, Function::PCT_Synthetic); |
| Count = F->getEntryCount(); |
| EXPECT_TRUE(Count.hasValue()); |
| EXPECT_EQ(123u, Count.getCount()); |
| EXPECT_EQ(Function::PCT_Synthetic, Count.getType()); |
| } |
| |
| TEST_F(FunctionAttachmentTest, SubprogramAttachment) { |
| Function *F = getFunction("foo"); |
| DISubprogram *SP = getSubprogram(); |
| F->setSubprogram(SP); |
| |
| // Note that the static_cast confirms that F->getSubprogram() actually |
| // returns an DISubprogram. |
| EXPECT_EQ(SP, static_cast<DISubprogram *>(F->getSubprogram())); |
| EXPECT_EQ(SP, F->getMetadata("dbg")); |
| EXPECT_EQ(SP, F->getMetadata(LLVMContext::MD_dbg)); |
| } |
| |
| typedef MetadataTest DistinctMDOperandPlaceholderTest; |
| TEST_F(DistinctMDOperandPlaceholderTest, getID) { |
| EXPECT_EQ(7u, DistinctMDOperandPlaceholder(7).getID()); |
| } |
| |
| TEST_F(DistinctMDOperandPlaceholderTest, replaceUseWith) { |
| // Set up some placeholders. |
| DistinctMDOperandPlaceholder PH0(7); |
| DistinctMDOperandPlaceholder PH1(3); |
| DistinctMDOperandPlaceholder PH2(0); |
| Metadata *Ops[] = {&PH0, &PH1, &PH2}; |
| auto *D = MDTuple::getDistinct(Context, Ops); |
| ASSERT_EQ(&PH0, D->getOperand(0)); |
| ASSERT_EQ(&PH1, D->getOperand(1)); |
| ASSERT_EQ(&PH2, D->getOperand(2)); |
| |
| // Replace them. |
| auto *N0 = MDTuple::get(Context, None); |
| auto *N1 = MDTuple::get(Context, N0); |
| PH0.replaceUseWith(N0); |
| PH1.replaceUseWith(N1); |
| PH2.replaceUseWith(nullptr); |
| EXPECT_EQ(N0, D->getOperand(0)); |
| EXPECT_EQ(N1, D->getOperand(1)); |
| EXPECT_EQ(nullptr, D->getOperand(2)); |
| } |
| |
| TEST_F(DistinctMDOperandPlaceholderTest, replaceUseWithNoUser) { |
| // There is no user, but we can still call replace. |
| DistinctMDOperandPlaceholder(7).replaceUseWith(MDTuple::get(Context, None)); |
| } |
| |
| // Test various assertions in metadata tracking. Don't run these tests if gtest |
| // will use SEH to recover from them. Two of these tests get halfway through |
| // inserting metadata into DenseMaps for tracking purposes, and then they |
| // assert, and we attempt to destroy an LLVMContext with broken invariants, |
| // leading to infinite loops. |
| #if defined(GTEST_HAS_DEATH_TEST) && !defined(NDEBUG) && !defined(GTEST_HAS_SEH) |
| TEST_F(DistinctMDOperandPlaceholderTest, MetadataAsValue) { |
| // This shouldn't crash. |
| DistinctMDOperandPlaceholder PH(7); |
| EXPECT_DEATH(MetadataAsValue::get(Context, &PH), |
| "Unexpected callback to owner"); |
| } |
| |
| TEST_F(DistinctMDOperandPlaceholderTest, UniquedMDNode) { |
| // This shouldn't crash. |
| DistinctMDOperandPlaceholder PH(7); |
| EXPECT_DEATH(MDTuple::get(Context, &PH), "Unexpected callback to owner"); |
| } |
| |
| TEST_F(DistinctMDOperandPlaceholderTest, SecondDistinctMDNode) { |
| // This shouldn't crash. |
| DistinctMDOperandPlaceholder PH(7); |
| MDTuple::getDistinct(Context, &PH); |
| EXPECT_DEATH(MDTuple::getDistinct(Context, &PH), |
| "Placeholders can only be used once"); |
| } |
| |
| TEST_F(DistinctMDOperandPlaceholderTest, TrackingMDRefAndDistinctMDNode) { |
| // TrackingMDRef doesn't install an owner callback, so it can't be detected |
| // as an invalid use. However, using a placeholder in a TrackingMDRef *and* |
| // a distinct node isn't possible and we should assert. |
| // |
| // (There's no positive test for using TrackingMDRef because it's not a |
| // useful thing to do.) |
| { |
| DistinctMDOperandPlaceholder PH(7); |
| MDTuple::getDistinct(Context, &PH); |
| EXPECT_DEATH(TrackingMDRef Ref(&PH), "Placeholders can only be used once"); |
| } |
| { |
| DistinctMDOperandPlaceholder PH(7); |
| TrackingMDRef Ref(&PH); |
| EXPECT_DEATH(MDTuple::getDistinct(Context, &PH), |
| "Placeholders can only be used once"); |
| } |
| } |
| #endif |
| |
| } // end namespace |