// 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}