594 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			594 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
// 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}
 |