forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			220 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			220 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
// RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -fstrict-vtable-pointers -disable-llvm-optzns -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();
 | 
						|
 | 
						|
struct NotTrivialDtor {
 | 
						|
  ~NotTrivialDtor();
 | 
						|
};
 | 
						|
 | 
						|
struct DynamicBase1 {
 | 
						|
  NotTrivialDtor obj;
 | 
						|
  virtual void foo();
 | 
						|
};
 | 
						|
 | 
						|
struct DynamicDerived : DynamicBase1 {
 | 
						|
  void foo();
 | 
						|
};
 | 
						|
 | 
						|
struct DynamicBase2 {
 | 
						|
  virtual void bar();
 | 
						|
  ~DynamicBase2() {
 | 
						|
    bar();
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
struct DynamicDerivedMultiple : DynamicBase1, DynamicBase2 {
 | 
						|
  virtual void foo();
 | 
						|
  virtual void bar();
 | 
						|
};
 | 
						|
 | 
						|
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.invariant.group.barrier(
 | 
						|
// 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.invariant.group.barrier(
 | 
						|
// CHECK-CTORS-LABEL: }
 | 
						|
 | 
						|
struct DynamicFrom2Virtuals;
 | 
						|
// CHECK-CTORS-LABEL: define linkonce_odr void @_ZN20DynamicFrom2VirtualsC1Ev
 | 
						|
// CHECK-CTORS: call i8* @llvm.invariant.group.barrier(
 | 
						|
// CHECK-CTORS-LABEL: }
 | 
						|
 | 
						|
 | 
						|
// CHECK-NEW-LABEL: define void @_Z9Pointers1v()
 | 
						|
// CHECK-NEW-NOT: @llvm.invariant.group.barrier(
 | 
						|
// CHECK-NEW-LABEL: call void @_ZN12DynamicBase1C1Ev(
 | 
						|
 | 
						|
// CHECK-NEW: %[[THIS3:.*]] = call i8* @llvm.invariant.group.barrier(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.invariant.group.barrier(
 | 
						|
// CHECK-NEW:  call void @_ZN14DynamicDerivedC1Ev(
 | 
						|
// CHECK-NEW:  call i8* @llvm.invariant.group.barrier(
 | 
						|
// 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.invariant.group.barrier(
 | 
						|
// 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.invariant.group.barrier(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.invariant.group.barrier(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.invariant.group.barrier(
 | 
						|
 | 
						|
// CHECK-CTORS: call void @_ZN12DynamicBase2C2Ev(
 | 
						|
// CHECK-CTORS-NOT: @llvm.invariant.group.barrier
 | 
						|
 | 
						|
 | 
						|
// CHECK-CTORS: %[[THIS10:.*]] = bitcast %struct.DynamicDerivedMultiple* %[[THIS0]] to i32 (...)***
 | 
						|
// CHECK-CTORS: store {{.*}} @_ZTV22DynamicDerivedMultiple, 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, i32 6) {{.*}} %[[THIS12]]
 | 
						|
// CHECK-CTORS-LABEL: }
 | 
						|
 | 
						|
struct DynamicFromStatic;
 | 
						|
// CHECK-CTORS-LABEL: define linkonce_odr void @_ZN17DynamicFromStaticC2Ev(
 | 
						|
// CHECK-CTORS-NOT: @llvm.invariant.group.barrier(
 | 
						|
// CHECK-CTORS-LABEL: }
 | 
						|
 | 
						|
 | 
						|
/** DTORS **/
 | 
						|
// CHECK-DTORS-LABEL: define linkonce_odr void @_ZN10StaticBaseD2Ev(
 | 
						|
// CHECK-DTORS-NOT: call i8* @llvm.invariant.group.barrier(
 | 
						|
// 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.invariant.group.barrier(
 | 
						|
// CHECK-DTORS-LABEL: }
 | 
						|
 | 
						|
 | 
						|
// CHECK-DTORS-LABEL: define linkonce_odr void @_ZN22DynamicDerivedMultipleD2Ev(
 | 
						|
 | 
						|
// CHECK-DTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase2D2Ev(
 | 
						|
// CHECK-DTORS: call i8* @llvm.invariant.group.barrier(
 | 
						|
// CHECK-DTORS-LABEL: }
 | 
						|
 | 
						|
// CHECK-DTORS-LABEL: define linkonce_odr void @_ZN12DynamicBase1D2Ev
 | 
						|
// CHECK-DTORS: call i8* @llvm.invariant.group.barrier(
 | 
						|
// CHECK-DTORS-LABEL: }
 | 
						|
 | 
						|
// CHECK-DTORS-LABEL: define linkonce_odr void @_ZN14DynamicDerivedD2Ev
 | 
						|
// CHECK-DTORS-NOT: call i8* @llvm.invariant.group.barrier(
 | 
						|
// CHECK-DTORS-LABEL: }
 | 
						|
 | 
						|
 | 
						|
// CHECK-LINK-REQ: !llvm.module.flags = !{![[FIRST:.*]], ![[SEC:.*]]{{.*}}}
 | 
						|
 | 
						|
// CHECK-LINK-REQ: ![[FIRST]] = !{i32 1, !"StrictVTablePointers", i32 1}
 | 
						|
// CHECK-LINK-REQ: ![[SEC]] = !{i32 3, !"StrictVTablePointersRequirement", ![[META:.*]]}
 | 
						|
// CHECK-LINK-REQ: ![[META]] = !{!"StrictVTablePointers", i32 1}
 | 
						|
 |