|  | // RUN: %clang_cc1 -fno-rtti -emit-llvm -fdump-vtable-layouts %s -o %t.ll -triple=i386-pc-win32 > %t | 
|  | // RUN: FileCheck %s < %t | 
|  | // RUN: FileCheck --check-prefix=MANGLING %s < %t.ll | 
|  |  | 
|  | // For now, just make sure x86_64 doesn't crash. | 
|  | // RUN: %clang_cc1 -fno-rtti -emit-llvm-only -fdump-vtable-layouts %s -triple=x86_64-pc-win32 > /dev/null | 
|  |  | 
|  | struct V1 { | 
|  | virtual void f(); | 
|  | virtual ~V1(); | 
|  | }; | 
|  |  | 
|  | struct V2 { | 
|  | virtual void f(); | 
|  | virtual ~V2(); | 
|  | int v; | 
|  | }; | 
|  |  | 
|  | struct Z { | 
|  | virtual void g(); | 
|  | virtual ~Z(); | 
|  | int x; | 
|  | }; | 
|  |  | 
|  | struct V3 : Z, V2 { | 
|  | }; | 
|  |  | 
|  | struct V4 : Z, V1, V2 { | 
|  | int y; | 
|  | }; | 
|  |  | 
|  | void use_somewhere_else(void*); | 
|  |  | 
|  | namespace simple { | 
|  | // In case of a single-layer virtual inheritance, the "this" adjustment is done | 
|  | // staically: | 
|  | //   struct A { | 
|  | //     virtual void f();  // Expects "(A*)this" in ECX | 
|  | //   }; | 
|  | //   struct B : virtual A { | 
|  | //     virtual void f();  // Expects "(char*)(B*)this + 12" in ECX | 
|  | //     virtual ~B();      // Might call f() | 
|  | //   }; | 
|  | // | 
|  | // If a class overrides a virtual function of its base and has a non-trivial | 
|  | // ctor/dtor that call(s) the virtual function (or may escape "this" to some | 
|  | // code that might call it), a virtual adjustment might be needed in case the | 
|  | // current class layout and the most derived class layout are different. | 
|  | // This is done using vtordisp thunks. | 
|  | // | 
|  | // A simple vtordisp{A,B} thunk for Method@Class is something like: | 
|  | //   sub  ecx, [ecx+A]  // apply the vtordisp adjustment | 
|  | //   sub  ecx, B        // apply the subobject adjustment, if needed. | 
|  | //   jmp Method@Class | 
|  |  | 
|  | struct A : virtual V1 { | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'simple::A' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::A::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  | // CHECK-NEXT: 1 | simple::A::~A() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'simple::A::~A()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::A::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | virtual void f(); | 
|  | // MANGLING-DAG: @"\01?f@A@simple@@$4PPPPPPPM@A@AEXXZ" | 
|  |  | 
|  | virtual ~A(); | 
|  | // MANGLING-DAG: @"\01??_EA@simple@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | A a; | 
|  | void use(A *obj) { obj->f(); } | 
|  |  | 
|  | struct B : virtual V3 { | 
|  | // CHECK-LABEL: VFTable for 'Z' in 'V3' in 'simple::B' (2 entries). | 
|  | // CHECK-NEXT: 0 | void Z::g() | 
|  | // CHECK-NEXT: 1 | simple::B::~B() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'simple::B::~B()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: VFTable for 'V2' in 'V3' in 'simple::B' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::B::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -12, 0 non-virtual] | 
|  | // CHECK-NEXT: 1 | simple::B::~B() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -12, -8 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'simple::B::~B()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -12, -8 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::B::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -12, 0 non-virtual] | 
|  |  | 
|  | // FIXME: The vtordisp thunk should only get emitted for a constructor | 
|  | // if "this" leaves scope. | 
|  | B() { use_somewhere_else(this); } | 
|  |  | 
|  | virtual void f(); | 
|  | // MANGLING-DAG: @"\01?f@B@simple@@$4PPPPPPPE@A@AEXXZ" | 
|  |  | 
|  | // Has an implicit destructor. | 
|  | // MANGLING-DAG: @"\01??_EB@simple@@$4PPPPPPPE@7AEPAXI@Z" | 
|  | // MANGLING-DAG: @"\01??_EB@simple@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | B b; | 
|  | void use(B *obj) { obj->f(); } | 
|  |  | 
|  | struct C : virtual V4 { | 
|  | // CHECK-LABEL: VFTable for 'Z' in 'V4' in 'simple::C' (2 entries). | 
|  | // CHECK-NEXT: 0 | void Z::g() | 
|  | // CHECK-NEXT: 1 | simple::C::~C() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'simple::C::~C()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'V4' in 'simple::C' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::C::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -12, 0 non-virtual] | 
|  | // CHECK-NEXT: 1 | simple::C::~C() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -12, -8 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'simple::C::~C()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -12, -8 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::C::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -12, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: VFTable for 'V2' in 'V4' in 'simple::C' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::C::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -16, -4 non-virtual] | 
|  | // CHECK-NEXT: 1 | simple::C::~C() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -16, -12 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'simple::C::~C()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -16, -12 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::C::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -16, -4 non-virtual] | 
|  |  | 
|  | int x; | 
|  | virtual void f(); | 
|  | // MANGLING-DAG: @"\01?f@C@simple@@$4PPPPPPPA@3AEXXZ" | 
|  | // MANGLING-DAG: @"\01?f@C@simple@@$4PPPPPPPE@A@AEXXZ" | 
|  | virtual ~C(); | 
|  | // MANGLING-DAG: @"\01??_EC@simple@@$4PPPPPPPA@M@AEPAXI@Z" | 
|  | // MANGLING-DAG: @"\01??_EC@simple@@$4PPPPPPPE@7AEPAXI@Z" | 
|  | // MANGLING-DAG: @"\01??_EC@simple@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | C c; | 
|  | void use(C *obj) { obj->f(); } | 
|  | } | 
|  |  | 
|  | namespace extended { | 
|  | // If a virtual function requires vtordisp adjustment and the final overrider | 
|  | // is defined in another vitual base of the most derived class, | 
|  | // we need to know two vbase offsets. | 
|  | // In this case, we should use the extended form of vtordisp thunks, called | 
|  | // vtordispex thunks. | 
|  | // | 
|  | // vtordispex{A,B,C,D} thunk for Method@Class is something like: | 
|  | //   sub  ecx, [ecx+C]  // apply the vtordisp adjustment | 
|  | //   sub  ecx, A        // jump to the vbtable of the most derived class | 
|  | //   mov  eax, [ecx]    // load the vbtable address | 
|  | //   add  ecx, [eax+B]  // lookup the final overrider's vbase offset | 
|  | //   add  ecx, D        // apphy the subobject offset if needed | 
|  | //   jmp Method@Class | 
|  |  | 
|  | struct A : virtual simple::A { | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'simple::A' in 'extended::A' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::A::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  | // CHECK-NEXT: 1 | extended::A::~A() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::A::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // `vtordispex{8,8,4294967292,8}' | 
|  | // MANGLING-DAG: @"\01?f@A@simple@@$R477PPPPPPPM@7AEXXZ" | 
|  |  | 
|  | virtual ~A(); | 
|  | // vtordisp{4294967292,0} | 
|  | // MANGLING-DAG: @"\01??_EA@extended@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | A a; | 
|  | void use(A *obj) { delete obj; } | 
|  |  | 
|  | struct B : virtual simple::A { | 
|  | // This class has an implicit dtor.  Vdtors don't require vtordispex thunks | 
|  | // as the most derived class always has an implicit dtor, | 
|  | // which is a final overrider. | 
|  |  | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'simple::A' in 'extended::B' (2 entries). | 
|  | //  ... | 
|  | // CHECK: 1 | extended::B::~B() [scalar deleting] | 
|  | // CHECK-NEXT: [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::A::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // vtordisp{4294967292,0} | 
|  | // MANGLING-DAG: @"\01??_EB@extended@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | B b; | 
|  | void use(B *obj) { delete obj; } | 
|  |  | 
|  | struct C : virtual simple::A { | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'simple::A' in 'extended::C' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::A::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, vbptr at 12 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::A::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 12 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // `vtordispex{12,8,4294967292,8}' | 
|  | // MANGLING-DAG: @"\01?f@A@simple@@$R4M@7PPPPPPPM@7AEXXZ" | 
|  | int x; | 
|  | virtual ~C(); | 
|  | // MANGLING-DAG: @"\01??_EC@extended@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | C c; | 
|  | void use(C *obj) { delete obj; } | 
|  |  | 
|  | struct D : virtual V2 { | 
|  | virtual void f(); | 
|  | virtual ~D(); | 
|  | int x; | 
|  | }; | 
|  |  | 
|  | struct E : virtual D { | 
|  | // CHECK-LABEL: VFTable for 'V2' in 'extended::D' in 'extended::E' (2 entries). | 
|  | // CHECK-NEXT: 0 | void extended::D::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 12 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void extended::D::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 12 non-virtual] | 
|  |  | 
|  | // `vtordispex{8,8,4294967292,12}' | 
|  | // MANGLING-DAG: @"\01?f@D@extended@@$R477PPPPPPPM@M@AEXXZ" | 
|  |  | 
|  | virtual ~E(); | 
|  | // MANGLING-DAG: @"\01??_EE@extended@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | E e; | 
|  | void use(E *obj) { delete obj; } | 
|  |  | 
|  | struct F : virtual Z, virtual D { | 
|  | // CHECK-LABEL: VFTable for 'V2' in 'extended::D' in 'extended::F' (2 entries). | 
|  | // CHECK-NEXT: 0 | void extended::D::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, vbptr at 20 to the left, | 
|  | // CHECK-NEXT:      vboffset at 12 in the vbtable, 12 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void extended::D::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 20 to the left, | 
|  | // CHECK-NEXT:      vboffset at 12 in the vbtable, 12 non-virtual] | 
|  |  | 
|  | // `vtordispex{20,12,4294967292,12}' | 
|  | // MANGLING-DAG: @"\01?f@D@extended@@$R4BE@M@PPPPPPPM@M@AEXXZ" | 
|  | int x; | 
|  | virtual ~F(); | 
|  | // MANGLING-DAG: @"\01??_EF@extended@@$4PPPPPPPM@M@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | F f; | 
|  | void use(F *obj) { delete obj; } | 
|  |  | 
|  | struct G : virtual simple::A { | 
|  | // CHECK-LABEL: VFTable for 'extended::G' (1 entry). | 
|  | // CHECK-NEXT: 0 | void extended::G::g() | 
|  |  | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'simple::A' in 'extended::G' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::A::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  | // CHECK-NEXT: 1 | extended::G::~G() [scalar deleting] | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, 0 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::A::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // Emits a G's own vfptr, thus moving the vbptr in the layout. | 
|  | virtual void g(); | 
|  |  | 
|  | virtual ~G(); | 
|  | // vtordisp{4294967292,0} | 
|  | // MANGLING-DAG: @"\01??_EG@extended@@$4PPPPPPPM@A@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | G g; | 
|  | void use(G *obj) { obj->g(); } | 
|  |  | 
|  | struct H : Z, A { | 
|  | // CHECK-LABEL: VFTable for 'Z' in 'extended::H' (2 entries). | 
|  | // CHECK-NEXT: 0 | void Z::g() | 
|  | // CHECK-NEXT: 1 | extended::H::~H() [scalar deleting] | 
|  |  | 
|  | // CHECK-LABEL: VFTable for 'V1' in 'simple::A' in 'extended::A' in 'extended::H' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::A::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::A::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -4, vbptr at 8 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 8 non-virtual] | 
|  |  | 
|  | // MANGLING-DAG: @"\01?f@A@simple@@$R477PPPPPPPM@7AEXXZ" | 
|  | // MANGLING-DAG: @"\01??_EH@extended@@$4PPPPPPPM@BA@AEPAXI@Z" | 
|  | }; | 
|  |  | 
|  | H h; | 
|  | void use(H *obj) { delete obj; } | 
|  | } | 
|  |  | 
|  | namespace pr17738 { | 
|  | // These classes should have vtordispex thunks but MSVS CL miscompiles them. | 
|  | // Just do the right thing. | 
|  |  | 
|  | struct A : virtual simple::B { | 
|  | // CHECK-LABEL: VFTable for 'V2' in 'V3' in 'simple::B' in 'pr17738::A' (2 entries). | 
|  | // CHECK-NEXT: 0 | void simple::B::f() | 
|  | // CHECK-NEXT:     [this adjustment: vtordisp at -12, vbptr at 20 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 16 non-virtual] | 
|  |  | 
|  | // CHECK-LABEL: Thunks for 'void simple::B::f()' (1 entry). | 
|  | // CHECK-NEXT: 0 | [this adjustment: vtordisp at -12, vbptr at 20 to the left, | 
|  | // CHECK-NEXT:      vboffset at 8 in the vbtable, 16 non-virtual] | 
|  |  | 
|  | // MANGLING-DAG: @"\01?f@B@simple@@$R4BE@7PPPPPPPE@BA@AEXXZ" | 
|  | int a; | 
|  | virtual ~A(); | 
|  | }; | 
|  |  | 
|  | A a; | 
|  | void use(A *obj) { delete obj; } | 
|  | } | 
|  |  | 
|  | namespace access { | 
|  | struct A { | 
|  | virtual ~A(); | 
|  | protected: | 
|  | virtual void prot(); | 
|  | private: | 
|  | virtual void priv(); | 
|  | }; | 
|  |  | 
|  | struct B : virtual A { | 
|  | virtual ~B(); | 
|  | protected: | 
|  | virtual void prot(); | 
|  | // MANGLING-DAG: @"\01?prot@B@access@@$2PPPPPPPM@A@AEXXZ" | 
|  | private: | 
|  | virtual void priv(); | 
|  | // MANGLING-DAG: @"\01?priv@B@access@@$0PPPPPPPM@A@AEXXZ" | 
|  | }; | 
|  |  | 
|  | B b; | 
|  |  | 
|  | struct C : virtual B { | 
|  | virtual ~C(); | 
|  |  | 
|  | // MANGLING-DAG: @"\01?prot@B@access@@$R277PPPPPPPM@7AEXXZ" | 
|  | // MANGLING-DAG: @"\01?priv@B@access@@$R077PPPPPPPM@7AEXXZ" | 
|  | }; | 
|  |  | 
|  | C c; | 
|  | } |