1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857 |
- //===- unittests/IR/MetadataTest.cpp - Metadata unit tests ----------------===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- #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(std::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(std::make_unique<ReplaceableMetadataImpl>(Context));
- EXPECT_EQ(&Context, &CRU.getContext());
- EXPECT_TRUE(CRU.hasReplaceableUses());
- EXPECT_TRUE(CRU.getReplaceableUses());
- }
- TEST(ContextAndReplaceableUsesTest, takeReplaceableUses) {
- LLVMContext Context;
- auto ReplaceableUses = std::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, 0, nullptr, 0, 0,
- DINode::FlagZero, DISubprogram::SPFlagZero, 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,
- DICompileUnit::DebugNameTableKind::Default, 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 Function::Create(
- FunctionType::get(Type::getVoidTy(Context), None, false),
- Function::ExternalLinkage, Name, M);
- }
- };
- 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[14] = "testing 1 2 3";
- MDString *s = MDString::get(Context, StringRef(&str[0], 13));
- strncpy(str, "aaaaaaaaaaaaa", 14);
- 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\\\\\\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));
- }
- TEST_F(MDNodeTest, PrintWithDroppedCallOperand) {
- 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);
- CallInst *CI0 = CallInst::Create(F1, "", BB0);
- CI0->dropAllReferences();
- auto *R0 = ReturnInst::Create(Context, BB0);
- auto *N0 = MDNode::getDistinct(Context, None);
- R0->setMetadata("md", N0);
- // Printing the metadata node would previously result in a failed assertion
- // due to the call instruction's dropped function operand.
- ModuleSlotTracker MST(&M);
- EXPECT_PRINTER_EQ("!0 = distinct !{}", N0->print(OS, 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, Merge) {
- DISubprogram *N = getSubprogram();
- DIScope *S = DILexicalBlock::get(Context, N, getFile(), 3, 4);
- {
- // Identical.
- auto *A = DILocation::get(Context, 2, 7, N);
- auto *B = DILocation::get(Context, 2, 7, N);
- auto *M = DILocation::getMergedLocation(A, B);
- EXPECT_EQ(2u, M->getLine());
- EXPECT_EQ(7u, M->getColumn());
- EXPECT_EQ(N, M->getScope());
- }
- {
- // Identical, different scopes.
- auto *A = DILocation::get(Context, 2, 7, N);
- auto *B = DILocation::get(Context, 2, 7, S);
- auto *M = DILocation::getMergedLocation(A, B);
- EXPECT_EQ(0u, M->getLine()); // FIXME: Should this be 2?
- EXPECT_EQ(0u, M->getColumn()); // FIXME: Should this be 7?
- EXPECT_EQ(N, M->getScope());
- }
- {
- // Different lines, same scopes.
- auto *A = DILocation::get(Context, 1, 6, N);
- auto *B = DILocation::get(Context, 2, 7, N);
- auto *M = DILocation::getMergedLocation(A, B);
- EXPECT_EQ(0u, M->getLine());
- EXPECT_EQ(0u, M->getColumn());
- EXPECT_EQ(N, M->getScope());
- }
- {
- // Twisty locations, all different, same function.
- auto *A = DILocation::get(Context, 1, 6, N);
- auto *B = DILocation::get(Context, 2, 7, S);
- auto *M = DILocation::getMergedLocation(A, B);
- EXPECT_EQ(0u, M->getLine());
- EXPECT_EQ(0u, M->getColumn());
- EXPECT_EQ(N, M->getScope());
- }
- {
- // Different function, same inlined-at.
- auto *F = getFile();
- auto *SP1 = DISubprogram::getDistinct(Context, F, "a", "a", F, 0, nullptr,
- 0, nullptr, 0, 0, DINode::FlagZero,
- DISubprogram::SPFlagZero, nullptr);
- auto *SP2 = DISubprogram::getDistinct(Context, F, "b", "b", F, 0, nullptr,
- 0, nullptr, 0, 0, DINode::FlagZero,
- DISubprogram::SPFlagZero, nullptr);
- auto *I = DILocation::get(Context, 2, 7, N);
- auto *A = DILocation::get(Context, 1, 6, SP1, I);
- auto *B = DILocation::get(Context, 2, 7, SP2, I);
- auto *M = DILocation::getMergedLocation(A, B);
- EXPECT_EQ(0u, M->getLine());
- EXPECT_EQ(0u, M->getColumn());
- EXPECT_TRUE(isa<DILocalScope>(M->getScope()));
- EXPECT_EQ(I, M->getInlinedAt());
- }
- {
- // Completely different.
- auto *I = DILocation::get(Context, 2, 7, N);
- auto *A = DILocation::get(Context, 1, 6, S, I);
- auto *B = DILocation::get(Context, 2, 7, getSubprogram());
- auto *M = DILocation::getMergedLocation(A, B);
- EXPECT_EQ(0u, M->getLine());
- EXPECT_EQ(0u, M->getColumn());
- EXPECT_TRUE(isa<DILocalScope>(M->getScope()));
- EXPECT_EQ(S, M->getScope());
- EXPECT_EQ(nullptr, M->getInlinedAt());
- }
- }
- 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());
- }
- TEST_F(DILocationTest, discriminatorEncoding) {
- EXPECT_EQ(0U, DILocation::encodeDiscriminator(0, 0, 0).getValue());
- // Encode base discriminator as a component: lsb is 0, then the value.
- // The other components are all absent, so we leave all the other bits 0.
- EXPECT_EQ(2U, DILocation::encodeDiscriminator(1, 0, 0).getValue());
- // Base discriminator component is empty, so lsb is 1. Next component is not
- // empty, so its lsb is 0, then its value (1). Next component is empty.
- // So the bit pattern is 101.
- EXPECT_EQ(5U, DILocation::encodeDiscriminator(0, 1, 0).getValue());
- // First 2 components are empty, so the bit pattern is 11. Then the
- // next component - ending up with 1011.
- EXPECT_EQ(0xbU, DILocation::encodeDiscriminator(0, 0, 1).getValue());
- // The bit pattern for the first 2 components is 11. The next bit is 0,
- // because the last component is not empty. We have 29 bits usable for
- // encoding, but we cap it at 12 bits uniformously for all components. We
- // encode the last component over 14 bits.
- EXPECT_EQ(0xfffbU, DILocation::encodeDiscriminator(0, 0, 0xfff).getValue());
- EXPECT_EQ(0x102U, DILocation::encodeDiscriminator(1, 1, 0).getValue());
- EXPECT_EQ(0x13eU, DILocation::encodeDiscriminator(0x1f, 1, 0).getValue());
- EXPECT_EQ(0x87feU, DILocation::encodeDiscriminator(0x1ff, 1, 0).getValue());
- EXPECT_EQ(0x1f3eU, DILocation::encodeDiscriminator(0x1f, 0x1f, 0).getValue());
- EXPECT_EQ(0x3ff3eU,
- DILocation::encodeDiscriminator(0x1f, 0x1ff, 0).getValue());
- EXPECT_EQ(0x1ff87feU,
- DILocation::encodeDiscriminator(0x1ff, 0x1ff, 0).getValue());
- EXPECT_EQ(0xfff9f3eU,
- DILocation::encodeDiscriminator(0x1f, 0x1f, 0xfff).getValue());
- EXPECT_EQ(0xffc3ff3eU,
- DILocation::encodeDiscriminator(0x1f, 0x1ff, 0x1ff).getValue());
- EXPECT_EQ(0xffcf87feU,
- DILocation::encodeDiscriminator(0x1ff, 0x1f, 0x1ff).getValue());
- EXPECT_EQ(0xe1ff87feU,
- DILocation::encodeDiscriminator(0x1ff, 0x1ff, 7).getValue());
- }
- TEST_F(DILocationTest, discriminatorEncodingNegativeTests) {
- EXPECT_EQ(None, DILocation::encodeDiscriminator(0, 0, 0x1000));
- EXPECT_EQ(None, DILocation::encodeDiscriminator(0x1000, 0, 0));
- EXPECT_EQ(None, DILocation::encodeDiscriminator(0, 0x1000, 0));
- EXPECT_EQ(None, DILocation::encodeDiscriminator(0, 0, 0x1000));
- EXPECT_EQ(None, DILocation::encodeDiscriminator(0x1ff, 0x1ff, 8));
- EXPECT_EQ(None,
- DILocation::encodeDiscriminator(std::numeric_limits<uint32_t>::max(),
- std::numeric_limits<uint32_t>::max(),
- 0));
- }
- TEST_F(DILocationTest, discriminatorSpecialCases) {
- // We don't test getCopyIdentifier here because the only way
- // to set it is by constructing an encoded discriminator using
- // encodeDiscriminator, which is already tested.
- auto L1 = DILocation::get(Context, 1, 2, getSubprogram());
- EXPECT_EQ(0U, L1->getBaseDiscriminator());
- EXPECT_EQ(1U, L1->getDuplicationFactor());
- EXPECT_EQ(L1, L1->cloneWithBaseDiscriminator(0).getValue());
- EXPECT_EQ(L1, L1->cloneByMultiplyingDuplicationFactor(0).getValue());
- EXPECT_EQ(L1, L1->cloneByMultiplyingDuplicationFactor(1).getValue());
- auto L2 = L1->cloneWithBaseDiscriminator(1).getValue();
- EXPECT_EQ(0U, L1->getBaseDiscriminator());
- EXPECT_EQ(1U, L1->getDuplicationFactor());
- EXPECT_EQ(1U, L2->getBaseDiscriminator());
- EXPECT_EQ(1U, L2->getDuplicationFactor());
- auto L3 = L2->cloneByMultiplyingDuplicationFactor(2).getValue();
- EXPECT_EQ(1U, L3->getBaseDiscriminator());
- EXPECT_EQ(2U, L3->getDuplicationFactor());
- EXPECT_EQ(L2, L2->cloneByMultiplyingDuplicationFactor(1).getValue());
- auto L4 = L3->cloneByMultiplyingDuplicationFactor(4).getValue();
- EXPECT_EQ(1U, L4->getBaseDiscriminator());
- EXPECT_EQ(8U, L4->getDuplicationFactor());
- auto L5 = L4->cloneWithBaseDiscriminator(2).getValue();
- EXPECT_EQ(2U, L5->getBaseDiscriminator());
- EXPECT_EQ(8U, L5->getDuplicationFactor());
- // Check extreme cases
- auto L6 = L1->cloneWithBaseDiscriminator(0xfff).getValue();
- EXPECT_EQ(0xfffU, L6->getBaseDiscriminator());
- EXPECT_EQ(0xfffU, L6->cloneByMultiplyingDuplicationFactor(0xfff)
- .getValue()
- ->getDuplicationFactor());
- // Check we return None for unencodable cases.
- EXPECT_EQ(None, L4->cloneWithBaseDiscriminator(0x1000));
- EXPECT_EQ(None, L4->cloneByMultiplyingDuplicationFactor(0x1000));
- }
- 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,
- DINode::FlagZero);
- 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(DINode::FlagZero, N->getFlags());
- EXPECT_EQ(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33,
- 26, 7, DINode::FlagZero));
- EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_unspecified_type,
- "special", 33, 26, 7, DINode::FlagZero));
- EXPECT_NE(N,
- DIBasicType::get(Context, dwarf::DW_TAG_base_type, "s", 33, 26, 7,
- DINode::FlagZero));
- EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 32,
- 26, 7, DINode::FlagZero));
- EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33,
- 25, 7, DINode::FlagZero));
- EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33,
- 26, 6, DINode::FlagZero));
- EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33,
- 26, 7, DINode::FlagBigEndian));
- EXPECT_NE(N, DIBasicType::get(Context, dwarf::DW_TAG_base_type, "special", 33,
- 26, 7, DINode::FlagLittleEndian));
- 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, DINode::FlagZero);
- 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());
- EXPECT_EQ(DINode::FlagZero, N->getFlags());
- }
- 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, DINode::FlagZero);
- 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, DICompileUnit::DebugNameTableKind::Default, 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,
- DICompileUnit::DebugNameTableKind::Default, 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();
- DISubprogram::DISPFlags SPFlags =
- static_cast<DISubprogram::DISPFlags>(Virtuality);
- assert(!IsLocalToUnit && IsDefinition && !IsOptimized &&
- "bools and SPFlags have to match");
- SPFlags |= DISubprogram::SPFlagDefinition;
- auto *N = DISubprogram::get(
- Context, Scope, Name, LinkageName, File, Line, Type, ScopeLine,
- ContainingType, VirtualIndex, ThisAdjustment, Flags, SPFlags, 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, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, Unit,
- TemplateParams, Declaration, RetainedNodes,
- ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, getCompositeType(), Name, LinkageName,
- File, Line, Type, ScopeLine, ContainingType,
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, "other", LinkageName, File,
- Line, Type, ScopeLine, ContainingType,
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, "other", File, Line,
- Type, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, Unit,
- TemplateParams, Declaration, RetainedNodes,
- ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, getFile(),
- Line, Type, ScopeLine, ContainingType,
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File,
- Line + 1, Type, ScopeLine, ContainingType,
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- getSubroutineType(), ScopeLine, ContainingType,
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(
- Context, Scope, Name, LinkageName, File, Line, Type,
- ScopeLine, ContainingType, VirtualIndex, ThisAdjustment,
- Flags, SPFlags ^ DISubprogram::SPFlagLocalToUnit, Unit,
- TemplateParams, Declaration, RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(
- Context, Scope, Name, LinkageName, File, Line, Type,
- ScopeLine, ContainingType, VirtualIndex, ThisAdjustment,
- Flags, SPFlags ^ DISubprogram::SPFlagDefinition, Unit,
- TemplateParams, Declaration, RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine + 1, ContainingType,
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, getCompositeType(),
- VirtualIndex, ThisAdjustment, Flags, SPFlags,
- Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(
- Context, Scope, Name, LinkageName, File, Line, Type,
- ScopeLine, ContainingType, VirtualIndex, ThisAdjustment,
- Flags, SPFlags ^ DISubprogram::SPFlagVirtual, Unit,
- TemplateParams, Declaration, RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, ContainingType,
- VirtualIndex + 1, ThisAdjustment, Flags,
- SPFlags, Unit, TemplateParams, Declaration,
- RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(
- Context, Scope, Name, LinkageName, File, Line, Type,
- ScopeLine, ContainingType, VirtualIndex, ThisAdjustment,
- Flags, SPFlags ^ DISubprogram::SPFlagOptimized, Unit,
- TemplateParams, Declaration, RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, nullptr,
- TemplateParams, Declaration, RetainedNodes,
- ThrownTypes));
- EXPECT_NE(N,
- DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, Unit, getTuple(),
- Declaration, RetainedNodes, ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, Unit,
- TemplateParams, getSubprogram(), RetainedNodes,
- ThrownTypes));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, Unit,
- TemplateParams, Declaration, getTuple()));
- EXPECT_NE(N, DISubprogram::get(Context, Scope, Name, LinkageName, File, Line,
- Type, ScopeLine, ContainingType, VirtualIndex,
- ThisAdjustment, Flags, SPFlags, 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;
- MDTuple *templateParams = getTuple();
- DIDerivedType *StaticDataMemberDeclaration =
- cast<DIDerivedType>(getDerivedType());
- uint32_t AlignInBits = 8;
- auto *N = DIGlobalVariable::get(
- Context, Scope, Name, LinkageName, File, Line, Type, IsLocalToUnit,
- IsDefinition, StaticDataMemberDeclaration, templateParams, 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(templateParams, N->getTemplateParams());
- EXPECT_EQ(AlignInBits, N->getAlignInBits());
- EXPECT_EQ(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, Type, IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(
- Context, getSubprogram(), Name, LinkageName, File, Line,
- Type, IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration, templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, "other", LinkageName, File,
- Line, Type, IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, "other", File, Line,
- Type, IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName,
- getFile(), Line, Type, IsLocalToUnit,
- IsDefinition, StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line + 1, Type, IsLocalToUnit,
- IsDefinition, StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, getDerivedType(), IsLocalToUnit,
- IsDefinition, StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, Type, !IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, Type, IsLocalToUnit, !IsDefinition,
- StaticDataMemberDeclaration,
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, Type, IsLocalToUnit, IsDefinition,
- cast<DIDerivedType>(getDerivedType()),
- templateParams, AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, Type, IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration, nullptr,
- AlignInBits));
- EXPECT_NE(N, DIGlobalVariable::get(Context, Scope, Name, LinkageName, File,
- Line, Type, IsLocalToUnit, IsDefinition,
- StaticDataMemberDeclaration,
- templateParams, (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;
- MDTuple *templateParams = getTuple();
- 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, templateParams, AlignInBits);
- auto *Var2 = DIGlobalVariable::get(
- Context, Scope, "other", LinkageName, File, Line, Type, IsLocalToUnit,
- IsDefinition, StaticDataMemberDeclaration, templateParams, 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);
- uint8_t DIExprFlags = DIExpression::ApplyOffset;
- DIExprFlags |= DIExpression::DerefBefore;
- DIExprFlags |= DIExpression::DerefAfter;
- DIExprFlags |= DIExpression::StackValue;
- auto *N0WithPrependedOps = DIExpression::prepend(N0, DIExprFlags, 64);
- 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(true /*allow synthetic*/);
- 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
|