| // 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 <typename Base> |
| 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<Incomplete> *)I; |
| (void)P; |
| |
| // CHECK-NEW: @llvm.launder.invariant.group |
| auto *P2 = (PossiblyDerivingFromDynamicBase<Empty> *)I; |
| (void)P2; |
| |
| // CHECK-NEW: @llvm.launder.invariant.group |
| auto *P3 = (PossiblyDerivingFromDynamicBase<A> *)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<Incomplete> *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} |