// RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -fstrict-vtable-pointers -std=c++11 -disable-llvm-passes -O2 -emit-llvm -o %t.ll // RUN: FileCheck --check-prefix=CHECK-CTORS %s < %t.ll // RUN: FileCheck --check-prefix=CHECK-NEW %s < %t.ll // RUN: FileCheck --check-prefix=CHECK-DTORS %s < %t.ll // RUN: FileCheck --check-prefix=CHECK-LINK-REQ %s < %t.ll typedef __typeof__(sizeof(0)) size_t; void *operator new(size_t, void *) throw(); using uintptr_t = unsigned long long; struct NotTrivialDtor { ~NotTrivialDtor(); }; struct DynamicBase1 { NotTrivialDtor obj; virtual void foo(); }; struct DynamicDerived : DynamicBase1 { void foo() override; }; struct DynamicBase2 { virtual void bar(); ~DynamicBase2() { bar(); } }; struct DynamicDerivedMultiple : DynamicBase1, DynamicBase2 { void foo() override; void bar() override; }; struct StaticBase { NotTrivialDtor obj; void bar(); }; struct DynamicFromStatic : StaticBase { virtual void bar(); }; struct DynamicFromVirtualStatic1 : virtual StaticBase { }; struct DynamicFromVirtualStatic2 : virtual StaticBase { }; struct DynamicFrom2Virtuals : DynamicFromVirtualStatic1, DynamicFromVirtualStatic2 { }; // CHECK-NEW-LABEL: define{{.*}} void @_Z12LocalObjectsv() // CHECK-NEW-NOT: @llvm.launder.invariant.group.p0i8( // CHECK-NEW-LABEL: {{^}}} void LocalObjects() { DynamicBase1 DB; DB.foo(); DynamicDerived DD; DD.foo(); DynamicBase2 DB2; DB2.bar(); StaticBase SB; SB.bar(); DynamicDerivedMultiple DDM; DDM.foo(); DDM.bar(); DynamicFromStatic DFS; DFS.bar(); DynamicFromVirtualStatic1 DFVS1; DFVS1.bar(); DynamicFrom2Virtuals DF2V; DF2V.bar(); } struct DynamicFromVirtualStatic1; // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN25DynamicFromVirtualStatic1C1Ev // CHECK-CTORS-NOT: @llvm.launder.invariant.group.p0i8( // CHECK-CTORS-LABEL: {{^}}} struct DynamicFrom2Virtuals; // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN20DynamicFrom2VirtualsC1Ev // CHECK-CTORS: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-CTORS-LABEL: {{^}}} // CHECK-NEW-LABEL: define{{.*}} void @_Z9Pointers1v() // CHECK-NEW-NOT: @llvm.launder.invariant.group.p0i8( // CHECK-NEW-LABEL: call void @_ZN12DynamicBase1C1Ev( // CHECK-NEW: %[[THIS3:.*]] = call i8* @llvm.launder.invariant.group.p0i8(i8* %[[THIS2:.*]]) // CHECK-NEW: %[[THIS4:.*]] = bitcast i8* %[[THIS3]] to %[[DynamicDerived:.*]]* // CHECK-NEW: call void @_ZN14DynamicDerivedC1Ev(%[[DynamicDerived:.*]]* {{[^,]*}} %[[THIS4]]) // CHECK-NEW-LABEL: {{^}}} void Pointers1() { DynamicBase1 *DB = new DynamicBase1; DB->foo(); DynamicDerived *DD = new (DB) DynamicDerived; DD->foo(); DD->~DynamicDerived(); } // CHECK-NEW-LABEL: define{{.*}} void @_Z14HackingObjectsv() // CHECK-NEW: call void @_ZN12DynamicBase1C1Ev // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-NEW: call void @_ZN14DynamicDerivedC1Ev( // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-NEW: call void @_ZN12DynamicBase1C1Ev( // CHECK-NEW-LABEL: {{^}}} void HackingObjects() { DynamicBase1 DB; DB.foo(); DynamicDerived *DB2 = new (&DB) DynamicDerived; // Using DB now is prohibited. DB2->foo(); DB2->~DynamicDerived(); // We have to get back to the previous type to avoid calling wrong destructor new (&DB) DynamicBase1; DB.foo(); } /*** Testing Constructors ***/ struct DynamicBase1; // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase1C2Ev( // CHECK-CTORS-NOT: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-CTORS-LABEL: {{^}}} struct DynamicDerived; // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN14DynamicDerivedC2Ev( // CHECK-CTORS: %[[THIS0:.*]] = load %[[DynamicDerived:.*]]*, %[[DynamicDerived]]** {{.*}} // CHECK-CTORS: %[[THIS1:.*]] = bitcast %[[DynamicDerived:.*]]* %[[THIS0]] to i8* // CHECK-CTORS: %[[THIS2:.*]] = call i8* @llvm.launder.invariant.group.p0i8(i8* %[[THIS1:.*]]) // CHECK-CTORS: %[[THIS3:.*]] = bitcast i8* %[[THIS2]] to %[[DynamicDerived]]* // CHECK-CTORS: %[[THIS4:.*]] = bitcast %[[DynamicDerived]]* %[[THIS3]] to %[[DynamicBase:.*]]* // CHECK-CTORS: call void @_ZN12DynamicBase1C2Ev(%[[DynamicBase]]* {{[^,]*}} %[[THIS4]]) // CHECK-CTORS: %[[THIS5:.*]] = bitcast %struct.DynamicDerived* %[[THIS0]] to i32 (...)*** // CHECK-CTORS: store {{.*}} %[[THIS5]] // CHECK-CTORS-LABEL: {{^}}} struct DynamicDerivedMultiple; // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN22DynamicDerivedMultipleC2Ev( // CHECK-CTORS: %[[THIS0:.*]] = load %[[CLASS:.*]]*, %[[CLASS]]** {{.*}} // CHECK-CTORS: %[[THIS1:.*]] = bitcast %[[CLASS:.*]]* %[[THIS0]] to i8* // CHECK-CTORS: %[[THIS2:.*]] = call i8* @llvm.launder.invariant.group.p0i8(i8* %[[THIS1]]) // CHECK-CTORS: %[[THIS3:.*]] = bitcast i8* %[[THIS2]] to %[[CLASS]]* // CHECK-CTORS: %[[THIS4:.*]] = bitcast %[[CLASS]]* %[[THIS3]] to %[[BASE_CLASS:.*]]* // CHECK-CTORS: call void @_ZN12DynamicBase1C2Ev(%[[BASE_CLASS]]* {{[^,]*}} %[[THIS4]]) // CHECK-CTORS: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-CTORS: call void @_ZN12DynamicBase2C2Ev( // CHECK-CTORS-NOT: @llvm.launder.invariant.group.p0i8 // CHECK-CTORS: %[[THIS10:.*]] = bitcast %struct.DynamicDerivedMultiple* %[[THIS0]] to i32 (...)*** // CHECK-CTORS: store {{.*}} @_ZTV22DynamicDerivedMultiple, i32 0, inrange i32 0, i32 2) {{.*}} %[[THIS10]] // CHECK-CTORS: %[[THIS11:.*]] = bitcast %struct.DynamicDerivedMultiple* %[[THIS0]] to i8* // CHECK-CTORS: %[[THIS_ADD:.*]] = getelementptr inbounds i8, i8* %[[THIS11]], i64 16 // CHECK-CTORS: %[[THIS12:.*]] = bitcast i8* %[[THIS_ADD]] to i32 (...)*** // CHECK-CTORS: store {{.*}} @_ZTV22DynamicDerivedMultiple, i32 0, inrange i32 1, i32 2) {{.*}} %[[THIS12]] // CHECK-CTORS-LABEL: {{^}}} struct DynamicFromStatic; // CHECK-CTORS-LABEL: define linkonce_odr void @_ZN17DynamicFromStaticC2Ev( // CHECK-CTORS-NOT: @llvm.launder.invariant.group.p0i8( // CHECK-CTORS-LABEL: {{^}}} struct A { virtual void foo(); int m; }; struct B : A { void foo() override; }; union U { A a; B b; }; void changeToB(U *u); void changeToA(U *u); void g2(A *a) { a->foo(); } // We have to guard access to union fields with invariant.group, because // it is very easy to skip the barrier with unions. In this example the inlined // g2 will produce loads with the same !invariant.group metadata, and // u->a and u->b would use the same pointer. // CHECK-NEW-LABEL: define{{.*}} void @_Z14UnionsBarriersP1U void UnionsBarriers(U *u) { // CHECK-NEW: call void @_Z9changeToBP1U( changeToB(u); // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: call void @_Z2g2P1A(%struct.A* g2(&u->b); // CHECK-NEW: call void @_Z9changeToAP1U(%union.U* changeToA(u); // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* // call void @_Z2g2P1A(%struct.A* %a) g2(&u->a); // CHECK-NEW-NOT: call i8* @llvm.launder.invariant.group.p0i8(i8* } struct HoldingVirtuals { A a; }; struct Empty {}; struct AnotherEmpty { Empty e; }; union NoVptrs { int a; AnotherEmpty empty; }; void take(AnotherEmpty &); // CHECK-NEW-LABEL: noBarriers void noBarriers(NoVptrs &noVptrs) { // CHECK-NEW-NOT: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: 42 noVptrs.a += 42; // CHECK-NEW-NOT: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: call void @_Z4takeR12AnotherEmpty( take(noVptrs.empty); } union U2 { HoldingVirtuals h; int z; }; void take(HoldingVirtuals &); // CHECK-NEW-LABEL: define{{.*}} void @_Z15UnionsBarriers2R2U2 void UnionsBarriers2(U2 &u) { // CHECK-NEW-NOT: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: 42 u.z += 42; // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: call void @_Z4takeR15HoldingVirtuals( take(u.h); } struct VirtualInBase : HoldingVirtuals, Empty { }; struct VirtualInVBase : virtual Empty, virtual HoldingVirtuals { }; // It has vtable by virtual inheritance. struct VirtualInheritance : virtual Empty { }; union U3 { VirtualInBase v1; VirtualInBase v2; VirtualInheritance v3; int z; }; void take(VirtualInBase &); void take(VirtualInVBase &); void take(VirtualInheritance &); void UnionsBarrier3(U3 &u) { // CHECK-NEW-NOT: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: 42 u.z += 42; // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: call void @_Z4takeR13VirtualInBase( take(u.v1); // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: call void @_Z4takeR13VirtualInBase( take(u.v2); // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* // CHECK-NEW: call void @_Z4takeR18VirtualInheritance( take(u.v3); } // CHECK-NEW-LABEL: define{{.*}} void @_Z7comparev() void compare() { A *a = new A; a->foo(); // CHECK-NEW: call i8* @llvm.launder.invariant.group.p0i8(i8* A *b = new (a) B; // CHECK-NEW: %[[a:.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* // CHECK-NEW: %[[a2:.*]] = bitcast i8* %[[a]] to %struct.A* // CHECK-NEW: %[[b:.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* // CHECK-NEW: %[[b2:.*]] = bitcast i8* %[[b]] to %struct.A* // CHECK-NEW: %cmp = icmp eq %struct.A* %[[a2]], %[[b2]] if (a == b) b->foo(); } // CHECK-NEW-LABEL: compare2 bool compare2(A *a, A *a2) { // CHECK-NEW: %[[a:.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* // CHECK-NEW: %[[a2:.*]] = bitcast i8* %[[a]] to %struct.A* // CHECK-NEW: %[[b:.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* // CHECK-NEW: %[[b2:.*]] = bitcast i8* %[[b]] to %struct.A* // CHECK-NEW: %cmp = icmp ult %struct.A* %[[a2]], %[[b2]] return a < a2; } // CHECK-NEW-LABEL: compareIntPointers bool compareIntPointers(int *a, int *b) { // CHECK-NEW-NOT: call i8* @llvm.strip.invariant.group return a == b; } struct HoldingOtherVirtuals { B b; }; // There is no need to add barriers for comparision of pointer to classes // that are not dynamic. // CHECK-NEW-LABEL: compare5 bool compare5(HoldingOtherVirtuals *a, HoldingOtherVirtuals *b) { // CHECK-NEW-NOT: call i8* @llvm.strip.invariant.group return a == b; } // CHECK-NEW-LABEL: compareNull bool compareNull(A *a) { // CHECK-NEW-NOT: call i8* @llvm.strip.invariant.group if (a != nullptr) return false; if (!a) return false; return a == nullptr; } struct X; // We have to also introduce the barriers if comparing pointers to incomplete // objects // CHECK-NEW-LABEL: define{{.*}} zeroext i1 @_Z8compare4P1XS0_ bool compare4(X *x, X *x2) { // CHECK-NEW: %[[x:.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* // CHECK-NEW: %[[xp:.*]] = bitcast i8* %[[x]] to %struct.X* // CHECK-NEW: %[[x2:.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* // CHECK-NEW: %[[x2p:.*]] = bitcast i8* %[[x2]] to %struct.X* // CHECK-NEW: %cmp = icmp eq %struct.X* %[[xp]], %[[x2p]] return x == x2; } // CHECK-NEW-LABEL: define{{.*}} void @_Z7member1P20HoldingOtherVirtuals( void member1(HoldingOtherVirtuals *p) { // CHECK-NEW-NOT: call i8* @llvm.strip.invariant.group.p0i8( (void)p->b; } // CHECK-NEW-LABEL: member2 void member2(A *a) { // CHECK-NEW: call i8* @llvm.strip.invariant.group.p0i8 (void)a->m; } // Check if from comparison of addresses of member we can't infer the equality // of ap and bp. // CHECK-NEW-LABEL: @_Z18testCompareMembersv( void testCompareMembers() { // CHECK-NEW: [[AP:%.*]] = alloca %struct.A* // CHECK-NEW: [[APM:%.*]] = alloca i32* // CHECK-NEW: [[BP:%.*]] = alloca %struct.B* // CHECK-NEW: [[BPM:%.*]] = alloca i32* A *ap = new A; // CHECK-NEW: call void %{{.*}}(%struct.A* {{[^,]*}} %{{.*}}) ap->foo(); // CHECK-NEW: [[TMP7:%.*]] = load %struct.A*, %struct.A** [[AP]] // CHECK-NEW: [[TMP8:%.*]] = bitcast %struct.A* [[TMP7]] to i8* // CHECK-NEW: [[TMP9:%.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* [[TMP8]]) // CHECK-NEW: [[TMP10:%.*]] = bitcast i8* [[TMP9]] to %struct.A* // CHECK-NEW: [[M:%.*]] = getelementptr inbounds [[STRUCT_A:%.*]], %struct.A* [[TMP10]], i32 0, i32 1 // CHECK-NEW: store i32* [[M]], i32** [[APM]] int *const apm = &ap->m; B *bp = new (ap) B; // CHECK-NEW: [[TMP20:%.*]] = load %struct.B*, %struct.B** [[BP]] // CHECK-NEW: [[TMP21:%.*]] = bitcast %struct.B* [[TMP20]] to %struct.A* // CHECK-NEW: [[TMP22:%.*]] = bitcast %struct.A* [[TMP21]] to i8* // CHECK-NEW: [[TMP23:%.*]] = call i8* @llvm.strip.invariant.group.p0i8(i8* [[TMP22]]) // CHECK-NEW: [[TMP24:%.*]] = bitcast i8* [[TMP23]] to %struct.A* // CHECK-NEW: [[M4:%.*]] = getelementptr inbounds [[STRUCT_A]], %struct.A* [[TMP24]], i32 0, i32 1 // CHECK-NEW: store i32* [[M4]], i32** [[BPM]] int *const bpm = &bp->m; // CHECK-NEW: [[TMP25:%.*]] = load i32*, i32** [[APM]] // CHECK-NEW: [[TMP26:%.*]] = load i32*, i32** [[BPM]] // CHECK-NEW-NOT: strip.invariant.group // CHECK-NEW-NOT: launder.invariant.group // CHECK-NEW: [[CMP:%.*]] = icmp eq i32* [[TMP25]], [[TMP26]] if (apm == bpm) { bp->foo(); } } // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast1P1A(%struct.A* void testCast1(A *a) { // Here we get rid of dynamic info // CHECK-NEW: call i8* @llvm.strip.invariant.group auto *v = (void *)a; // CHECK-NEW: call i8* @llvm.strip.invariant.group auto i2 = (uintptr_t)a; (void)i2; // CHECK-NEW-NOT: @llvm.strip.invariant.group // CHECK-NEW-NOT: @llvm.launder.invariant.group // The information is already stripped auto i = (uintptr_t)v; } struct Incomplete; // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast2P10Incomplete(%struct.Incomplete* void testCast2(Incomplete *I) { // Here we get rid of potential dynamic info // CHECK-NEW: call i8* @llvm.strip.invariant.group auto *v = (void *)I; // CHECK-NEW: call i8* @llvm.strip.invariant.group auto i2 = (uintptr_t)I; (void)i2; // CHECK-NEW-NOT: @llvm.strip.invariant.group // CHECK-NEW-NOT: @llvm.launder.invariant.group // The information is already stripped auto i = (uintptr_t)v; } // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast3y( void testCast3(uintptr_t i) { // CHECK-NEW-NOT: @llvm.strip.invariant.group // CHECK-NEW: @llvm.launder.invariant.group A *a3 = (A *)i; (void)a3; auto *v2 = (void *)i; // CHECK-NEW: @llvm.launder.invariant.group A *a2 = (A *)v2; (void)a2; // CHECK-NEW-NOT: @llvm.launder.invariant.group auto *v3 = (void *)i; (void)v3; } // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast4y( void testCast4(uintptr_t i) { // CHECK-NEW-NOT: @llvm.strip.invariant.group // CHECK-NEW: @llvm.launder.invariant.group auto *a3 = (Incomplete *)i; (void)a3; // CHECK-NEW: @llvm.launder.invariant.group auto *v2 = (void *)i; // CHECK-NEW-NOT: @llvm.launder.invariant.group auto *a2 = (Incomplete *)v2; (void)a2; } // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast5P1B( void testCast5(B *b) { // CHECK-NEW-NOT: @llvm.strip.invariant.group // CHECK-NEW-NOT: @llvm.launder.invariant.group A *a = b; (void)a; auto *b2 = (B *)a; (void)b2; } // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast6P1A( void testCast6(A *a) { // CHECK-NEW: @llvm.strip.invariant.group auto *I = (Incomplete *)a; (void)I; // CHECK-NEW: @llvm.launder.invariant.group auto *a2 = (A *)I; (void)a2; // CHECK-NEW: @llvm.strip.invariant.group auto *E = (Empty *)a; (void)E; // CHECK-NEW: @llvm.launder.invariant.group auto *a3 = (A *)E; (void)a3; // CHECK-NEW-NOT: @llvm.strip.invariant.group auto i = (uintptr_t)E; (void)i; } class Incomplete2; // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast7P10Incomplete( void testCast7(Incomplete *I) { // CHECK-NEW-NOT: @llvm.strip.invariant.group // Incomplete2 could be dynamic where Incomplete may not be dynamic, thus // launder is needed. We don't strip firstly because launder is sufficient. // CHECK-NEW: @llvm.launder.invariant.group auto *I2 = (Incomplete2 *)I; (void)I2; // CHECK-NEW-LABEL: ret void } template struct PossiblyDerivingFromDynamicBase : Base { }; // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast8P10Incomplete( void testCast8(Incomplete *I) { // CHECK-NEW-NOT: @llvm.strip.invariant.group // CHECK-NEW: @llvm.launder.invariant.group auto *P = (PossiblyDerivingFromDynamicBase *)I; (void)P; // CHECK-NEW: @llvm.launder.invariant.group auto *P2 = (PossiblyDerivingFromDynamicBase *)I; (void)P2; // CHECK-NEW: @llvm.launder.invariant.group auto *P3 = (PossiblyDerivingFromDynamicBase *)I; (void)P3; // CHECK-NEW-NOT: @llvm.launder.invariant.group auto *a3 = (A *)P3; // CHECK-NEW-LABEL: ret void } // CHECK-NEW-LABEL: define{{.*}} void @_Z9testCast9 void testCast9(PossiblyDerivingFromDynamicBase *P) { // CHECK-NEW: @llvm.strip.invariant.group auto *V = (void *)P; // CHECK-NEW-LABEL: ret void } /** DTORS **/ // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN10StaticBaseD2Ev( // CHECK-DTORS-NOT: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-DTORS-LABEL: {{^}}} // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN25DynamicFromVirtualStatic2D2Ev( // CHECK-DTORS-NOT: invariant.barrier // CHECK-DTORS-LABEL: {{^}}} // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN17DynamicFromStaticD2Ev // CHECK-DTORS-NOT: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-DTORS-LABEL: {{^}}} // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN22DynamicDerivedMultipleD2Ev( // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase2D2Ev( // CHECK-DTORS: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-DTORS-LABEL: {{^}}} // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase1D2Ev // CHECK-DTORS: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-DTORS-LABEL: {{^}}} // CHECK-DTORS-LABEL: define linkonce_odr void @_ZN14DynamicDerivedD2Ev // CHECK-DTORS-NOT: call i8* @llvm.launder.invariant.group.p0i8( // CHECK-DTORS-LABEL: {{^}}} // CHECK-LINK-REQ: !llvm.module.flags = !{![[FIRST:[0-9]+]], ![[SEC:[0-9]+]]{{.*}}} // CHECK-LINK-REQ: ![[FIRST]] = !{i32 1, !"StrictVTablePointers", i32 1} // CHECK-LINK-REQ: ![[SEC]] = !{i32 3, !"StrictVTablePointersRequirement", ![[META:.*]]} // CHECK-LINK-REQ: ![[META]] = !{!"StrictVTablePointers", i32 1}