2014-01-15 03:35:09 +08:00
|
|
|
// RUN: %clang_cc1 -fno-rtti -emit-llvm %s -o - -triple=i386-pc-win32 >%t 2>&1
|
2013-10-09 17:23:58 +08:00
|
|
|
// RUN: FileCheck --check-prefix=MANGLING %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=XMANGLING %s < %t
|
|
|
|
// RUN: FileCheck --check-prefix=CODEGEN %s < %t
|
2014-01-15 03:35:09 +08:00
|
|
|
// RUN: %clang_cc1 -fno-rtti -emit-llvm %s -o - -triple=x86_64-pc-win32 2>&1 | FileCheck --check-prefix=MANGLING-X64 %s
|
2013-10-09 17:23:58 +08:00
|
|
|
|
|
|
|
void foo(void *);
|
|
|
|
|
|
|
|
struct A {
|
|
|
|
virtual ~A();
|
|
|
|
virtual void public_f();
|
|
|
|
// Make sure we don't emit unneeded thunks:
|
|
|
|
// XMANGLING-NOT: @"\01?public_f@A@@QAEXXZ"
|
|
|
|
protected:
|
|
|
|
virtual void protected_f();
|
|
|
|
private:
|
|
|
|
virtual void private_f();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct B {
|
|
|
|
virtual ~B();
|
|
|
|
virtual void public_f();
|
|
|
|
protected:
|
|
|
|
virtual void protected_f();
|
|
|
|
private:
|
|
|
|
virtual void private_f();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct C : A, B {
|
|
|
|
C();
|
|
|
|
|
|
|
|
virtual ~C();
|
2015-01-19 21:59:31 +08:00
|
|
|
// MANGLING-DAG: declare {{.*}} @"\01??1C@@UAE@XZ"({{.*}})
|
|
|
|
// MANGLING-DAG: define {{.*}} @"\01??_GC@@UAEPAXI@Z"({{.*}})
|
2015-01-19 22:02:14 +08:00
|
|
|
// MANGLING-DAG: define {{.*}} @"\01??_EC@@W3AEPAXI@Z"({{.*}}) {{.*}} comdat
|
2015-01-19 21:59:31 +08:00
|
|
|
// MANGLING-X64-DAG: declare {{.*}} @"\01??1C@@UEAA@XZ"({{.*}})
|
|
|
|
// MANGLING-X64-DAG: define {{.*}} @"\01??_GC@@UEAAPEAXI@Z"({{.*}})
|
2015-01-19 22:02:14 +08:00
|
|
|
// MANGLING-X64-DAG: define {{.*}} @"\01??_EC@@W7EAAPEAXI@Z"({{.*}}) {{.*}} comdat
|
2013-10-09 17:23:58 +08:00
|
|
|
|
|
|
|
// Overrides public_f() of two subobjects with distinct vfptrs, thus needs a thunk.
|
|
|
|
virtual void public_f();
|
|
|
|
// MANGLING-DAG: @"\01?public_f@C@@UAEXXZ"
|
|
|
|
// MANGLING-DAG: @"\01?public_f@C@@W3AEXXZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?public_f@C@@UEAAXXZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?public_f@C@@W7EAAXXZ"
|
|
|
|
protected:
|
|
|
|
virtual void protected_f();
|
|
|
|
// MANGLING-DAG: @"\01?protected_f@C@@MAEXXZ"
|
|
|
|
// MANGLING-DAG: @"\01?protected_f@C@@O3AEXXZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?protected_f@C@@MEAAXXZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?protected_f@C@@O7EAAXXZ"
|
|
|
|
|
|
|
|
private:
|
|
|
|
virtual void private_f();
|
|
|
|
// MANGLING-DAG: @"\01?private_f@C@@EAEXXZ"
|
|
|
|
// MANGLING-DAG: @"\01?private_f@C@@G3AEXXZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?private_f@C@@EEAAXXZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?private_f@C@@G7EAAXXZ"
|
|
|
|
};
|
|
|
|
|
|
|
|
C::C() {} // Emits vftable and forces thunk generation.
|
|
|
|
|
2015-01-19 22:02:14 +08:00
|
|
|
// CODEGEN-LABEL: define linkonce_odr x86_thiscallcc i8* @"\01??_EC@@W3AEPAXI@Z"(%struct.C* %this, i32 %should_call_delete) {{.*}} comdat
|
2015-02-28 03:18:17 +08:00
|
|
|
// CODEGEN: getelementptr i8, i8* {{.*}}, i32 -4
|
2013-10-09 17:23:58 +08:00
|
|
|
// FIXME: should actually call _EC, not _GC.
|
2014-11-01 04:09:12 +08:00
|
|
|
// CODEGEN: call x86_thiscallcc i8* @"\01??_GC@@UAEPAXI@Z"
|
2013-10-09 17:23:58 +08:00
|
|
|
// CODEGEN: ret
|
|
|
|
|
2014-06-07 04:04:01 +08:00
|
|
|
// CODEGEN-LABEL: define linkonce_odr x86_thiscallcc void @"\01?public_f@C@@W3AEXXZ"(%struct.C*
|
2015-02-28 03:18:17 +08:00
|
|
|
// CODEGEN: getelementptr i8, i8* {{.*}}, i32 -4
|
2013-10-09 17:23:58 +08:00
|
|
|
// CODEGEN: call x86_thiscallcc void @"\01?public_f@C@@UAEXXZ"(%struct.C*
|
|
|
|
// CODEGEN: ret
|
|
|
|
|
|
|
|
void zoo(C* obj) {
|
|
|
|
delete obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct D {
|
|
|
|
virtual B* goo();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct E : D {
|
2013-10-30 19:55:43 +08:00
|
|
|
E();
|
2013-10-09 17:23:58 +08:00
|
|
|
virtual C* goo();
|
|
|
|
// MANGLING-DAG: @"\01?goo@E@@UAEPAUC@@XZ"
|
|
|
|
// MANGLING-DAG: @"\01?goo@E@@QAEPAUB@@XZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?goo@E@@UEAAPEAUC@@XZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?goo@E@@QEAAPEAUB@@XZ"
|
|
|
|
};
|
|
|
|
|
2013-10-30 19:55:43 +08:00
|
|
|
E::E() {} // Emits vftable and forces thunk generation.
|
2013-10-09 17:23:58 +08:00
|
|
|
|
2014-06-07 04:04:01 +08:00
|
|
|
// CODEGEN-LABEL: define weak_odr x86_thiscallcc %struct.C* @"\01?goo@E@@QAEPAUB@@XZ"
|
2013-10-09 17:23:58 +08:00
|
|
|
// CODEGEN: call x86_thiscallcc %struct.C* @"\01?goo@E@@UAEPAUC@@XZ"
|
2015-02-28 03:18:17 +08:00
|
|
|
// CODEGEN: getelementptr inbounds i8, i8* {{.*}}, i32 4
|
2013-10-09 17:23:58 +08:00
|
|
|
// CODEGEN: ret
|
|
|
|
|
|
|
|
struct F : virtual A, virtual B {
|
2013-10-30 19:55:43 +08:00
|
|
|
virtual void own_method();
|
2013-10-09 17:23:58 +08:00
|
|
|
virtual ~F();
|
|
|
|
};
|
|
|
|
|
|
|
|
F f; // Just make sure we don't crash, e.g. mangling the complete dtor.
|
|
|
|
|
|
|
|
struct G : C { };
|
|
|
|
|
|
|
|
struct H : E {
|
|
|
|
virtual G* goo();
|
|
|
|
// MANGLING-DAG: @"\01?goo@H@@UAEPAUG@@XZ"
|
|
|
|
// MANGLING-DAG: @"\01?goo@H@@QAEPAUB@@XZ"
|
|
|
|
// MANGLING-DAG: @"\01?goo@H@@QAEPAUC@@XZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?goo@H@@UEAAPEAUG@@XZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?goo@H@@QEAAPEAUB@@XZ"
|
|
|
|
// MANGLING-X64-DAG: @"\01?goo@H@@QEAAPEAUC@@XZ"
|
|
|
|
};
|
|
|
|
|
|
|
|
H h;
|
|
|
|
|
2013-10-30 19:55:43 +08:00
|
|
|
struct I : D {
|
|
|
|
I();
|
|
|
|
virtual F* goo();
|
|
|
|
};
|
|
|
|
|
|
|
|
I::I() {} // Emits vftable and forces thunk generation.
|
|
|
|
|
2014-06-07 04:04:01 +08:00
|
|
|
// CODEGEN-LABEL: define weak_odr x86_thiscallcc %struct.{{[BF]}}* @"\01?goo@I@@QAEPAUB@@XZ"
|
2013-10-30 19:55:43 +08:00
|
|
|
// CODEGEN: %[[ORIG_RET:.*]] = call x86_thiscallcc %struct.F* @"\01?goo@I@@UAEPAUF@@XZ"
|
|
|
|
// CODEGEN: %[[ORIG_RET_i8:.*]] = bitcast %struct.F* %[[ORIG_RET]] to i8*
|
2015-02-28 03:18:17 +08:00
|
|
|
// CODEGEN: %[[VBPTR_i8:.*]] = getelementptr inbounds i8, i8* %[[ORIG_RET_i8]], i32 4
|
2014-10-23 01:26:00 +08:00
|
|
|
// CODEGEN: %[[VBPTR:.*]] = bitcast i8* %[[VBPTR_i8]] to i32**
|
2015-02-28 05:19:58 +08:00
|
|
|
// CODEGEN: %[[VBTABLE:.*]] = load i32*, i32** %[[VBPTR]]
|
2015-02-28 03:18:17 +08:00
|
|
|
// CODEGEN: %[[VBASE_OFFSET_PTR:.*]] = getelementptr inbounds i32, i32* %[[VBTABLE]], i32 2
|
2015-02-28 05:19:58 +08:00
|
|
|
// CODEGEN: %[[VBASE_OFFSET:.*]] = load i32, i32* %[[VBASE_OFFSET_PTR]]
|
2015-02-28 03:18:17 +08:00
|
|
|
// CODEGEN: %[[RES_i8:.*]] = getelementptr inbounds i8, i8* %[[VBPTR_i8]], i32 %[[VBASE_OFFSET]]
|
2013-10-30 19:55:43 +08:00
|
|
|
// CODEGEN: %[[RES:.*]] = bitcast i8* %[[RES_i8]] to %struct.F*
|
|
|
|
// CODEGEN: phi %struct.F* {{.*}} %[[RES]]
|
|
|
|
// CODEGEN: ret %struct.{{[BF]}}*
|
|
|
|
|
2013-10-12 04:46:27 +08:00
|
|
|
namespace CrashOnThunksForAttributedType {
|
|
|
|
// We used to crash on this because the type of foo is an AttributedType, not
|
|
|
|
// FunctionType, and we had to look through the sugar.
|
|
|
|
struct A {
|
|
|
|
virtual void __stdcall foo();
|
|
|
|
};
|
|
|
|
struct B {
|
|
|
|
virtual void __stdcall foo();
|
|
|
|
};
|
|
|
|
struct C : A, B {
|
|
|
|
virtual void __stdcall foo();
|
|
|
|
};
|
|
|
|
C c;
|
|
|
|
}
|
2014-06-07 04:04:01 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
struct E : D {
|
|
|
|
E();
|
|
|
|
virtual C* goo();
|
|
|
|
};
|
|
|
|
E::E() {}
|
|
|
|
E e;
|
|
|
|
// Class with internal linkage has internal linkage thunks.
|
|
|
|
// CODEGEN: define internal x86_thiscallcc %struct.C* @"\01?goo@E@?A@@QAEPAUB@@XZ"
|
|
|
|
}
|