「现代C++设计魅力」虚函数继承-thunk技术初探

作者阿里云代理 文章分类 分类:linux图文教程 阅读次数 已被围观 980

作者 | 扬阜
来源 | 阿里技能大众号

一 问题背景

1 实践验证

工作中运用LLDB调试器调试这一段C++多承继程序的时分,发现经过lldb print(expression指令的别名) 指令获取的指针地址和实践了解的C++的内存模型的地址不相同。那么到底是什么原因呢?程序如下:

class Base { public:
    Base(){} protected: float x;
}; class VBase { public:
    VBase(){} virtual void test(){}; virtual void foo(){}; protected: float x;
}; class VBaseA: public VBase { public:
    VBaseA(){} virtual void test(){} virtual void foo(){}; protected: float x;
}; class VBaseB: public VBase  { public:
    VBaseB(){} virtual void test(){ printf("test \n");
    } virtual void foo(){}; protected: float x;
}; class VDerived : public VBaseA, public Base, public VBaseB { public:
    VDerived(){} virtual void test(){} virtual void foo(){}; protected: float x;
}; int main(int argc, char *argv[]) {
    VDerived *pDerived = new VDerived(); //0x0000000103407f30 Base  *pBase = (Base*)pDerived; //0x0000000103407f40 VBaseA *pvBaseA = static_cast< VBaseA*>(pDerived);//0x0000000103407f30 VBaseB  *pvBaseB = static_cast< VBaseB*>(pDerived);//0x0000000103407f30 这儿应该为0x0000000103407f48,但是显现的是0x0000000103407f30 unsigned long pBaseAddressbase = (unsigned long)pBase; unsigned long pvBaseAAddressbase = (unsigned long)pvBaseA; unsigned long pvBaseBAddressbase = (unsigned long)pvBaseB;
    pvBaseB->test();
}

经过lldb print指令获取的地址如下图:

image.png

image.png

正常了解的C++内存模型

image.png

由于我运用的是x86_64的mac系统,所以指针是8字节对齐,align=8。

按正常的了解的C++内存模型:pDerived转化为Base 类型pBase,地址偏移了16,是没问题的。

pDerived转化为VBaseA,由于共用了首地址为0x0000000103407f30,相同能够了解。pDerived转化为Base,地址偏移了16个字节(sizeof(VBaseA))为0x0000000103407f40,也是契合预期的。

但是pDerived转化为VBase 类型pBaseB内存地址应该偏移24,为0x0000000103407f48;而不是0x0000000103407f30(目标的首地址),这个到底是什么原因引起的的呢?

2 验证引发的猜想

关于上面的这段代码

Base 类中没有虚函数,VBaseB 中有虚函数test和foo,猜想如下

image.png

1.不含有虚函数的(不含有虚表的)基类的指针,在类型转化时编译器对地址依照实践偏移。

2.含有虚函数的(含有虚表的)基类指针,在类型转化时,编译器实践上没有做地址的偏移,仍是指向派生类,并没有指向实践的VBaseB类型。

二 现象带来的问题

1.有虚函数的(含有虚表的)基类指针,在派生类类型转化为有虚函数的基类时,编译器背面有做实在的地址偏移吗?

2.假如做了偏移

  • 那C++中在经过基类指针调用派生类重写的虚函数以及经过派生类指针调用虚函数的时分,编译器是怎么保证这两种调用this指针的值是相同的,以保证调用的正确性的?
  • 那为什么LLDB expression获取的地址是派生类目标的首地址呢?

3.假如没有做偏移,那是怎么经过派生类的指针调用基类成员变量和函数的?

三 现象中心原因

  1. 编译器背面和一般的非虚函数承继相同,也做了指针的偏移。
  2. 做了指针偏移,C++ 中基类目标指针调用派生类目标时,编译器经过thunk技能来完成每次参数调用和参数返回this地址的调整。
  3. LLDB expression显现的是派生类目标的首地址(0x0000000103407f30),而不是偏移后基类目标的首地址(0x0000000103407f48),是由于LLDB调试器在expression向用户展现的时分,关于虚函数承继的基类指针LLDB内部会经过summary format来对要获取的结果进行格式化。summary format时,会根据当时的内存地址获取C++运行时的动态类型和地址,来向用户展现。

四 证明结论进程

1 指针类型转化时编译器是否做了偏移?

汇编指令剖析

image.png

有虚函数的(含有虚表的)基类指针,在派生类类型转化为有虚函数的基类时,编译器背面有做实在的地址偏移吗?

基于上面的猜想,经过下面运行时反汇编的程序,来验证上面的猜想:

在开端反汇编程序之前,有一些下面要用到的汇编知识的遍及。假如了解,能够疏忽越过。

注意:由于小编运用的是mac操作系统,所以处理器运用的是AT&T语法;和Intel语法不相同。

AT&T语法的指令是从左到右,第一个是源操作数,第二个是意图操作数,比如:

movl %esp, %ebp //movl是指令称号。%则表明esp和ebp是寄存器.在AT&T语法中, 第一个是源操作数,第二个是意图操作数。

而Intel指令是从右到左,第二个是源操作数,第一个是意图操作数

MOVQ EBP, ESP //interl手册,你会看到是没有%的intel语法, 它的操作数顺序刚好相反

在x86_64的寄存器调用约定规定中

1.第一个参数基本上放在:RDI/edi寄存器,第二个参数:RSI/esi寄存器,第三个参数:RDX寄存器,第四个参数:RCD寄存器,第五个参数:R8寄存器,第六个参数:R9 寄存器;

2.假如超越六个参数在函数里就会经过栈来访问额外的参数;

3.函数返回值一般放在eax寄存器,或者rax寄存器。

下面运用的mac Unix操作系统,本文用到的汇编指令都是AT&T语法,在函数传参数时的第一个参数都放在RDI寄存器中。

下面是上面的main程序从开端履行到退出程序的所有汇编程序

image.png

经过上看的汇编代码咱们发现编译器在做类型转化的时分不管是承继的基类有虚函数,仍是没有虚函数,编译器都会做实践的指针偏移,偏移到实践的基类目标的地址,证明上面的猜想是错误的。编译器在类型转化的时分不区别有没有虚函数,都是实践做了偏移的。

内存剖析

上面的猜想,后来我经过LLDB调试器供给的:memory read ptr(memory read 指令缩写 x )得到了验证

(lldb) memory read pDerived 0x103407f30: 40 40 00 00 01 00 00 00 00 00 00 00 00 00 00 00 @@.............. 0x103407f40: 10 00 00 00 00 00 00 00 60 40 00 00 01 00 00 00 ........`@......
(lldb) memory read pvBaseB 0x103407f48: 60 40 00 00 01 00 00 00 00 00 00 00 00 00 00 00 `@.............. 0x103407f58: de 2d 05 10 00 00 00 00 00 00 00 00 00 00 00 00 .-..............

咱们发现不同类型的指针 在内存中的确读取到的内容分别是pDerived:0x103407f30 pvBaseB:0x103407f48内存地址都不相同;都是实践偏移后地址。

2 虚函数调用怎么保证this的值共同的呢?

image.png

那已然内容中的实在地址是偏移后的,派生类重写了基类的虚函数,在经过基类指针调用派生类从头的虚函数的时分和经过派生类调用自身完成的虚函数的时分,编译器是怎么保证这两种调用this指针的值是相同的,来保证调用的正确性的?

在网上查阅资料得知:C++在调用函数的时分, 编译器经过thunk技能对this指针的内容做了调整,使其指向正确的内存地址。那么什么是thunk技能?编译器是怎么完成的呢?

虚函数调用汇编指令剖析

经过上面main函数不难发现的pvBaseB->test() 的反汇编:

pBaseB->test(); 0x100003c84 < +244>: movq -0x40(%rbp), %rax    //-x40存方的是pBaseB指针的内容,这儿取出pBaseB指向的地址 0x100003c88 < +248>: movq (%rax), %rcx         //然后将 rax的内容赋值给rcx 0x100003c8b < +251>: movq %rax, %rdi           // 之后再将rax的值给到rdi寄存器:咱们都知道,rdi寄存器是函数调用的第一个参数,这儿的this是基类的地址
-> 0x100003c8e < +254>: callq  *(%rcx)              // 在这儿取出rcx的地址,然后经过*(rcx) 直接调用rcx中存的地址

咱们再跳到VDerived::test函数的汇编完成, 在这儿经过lldb的指令:register read rdi 检查函数的第一个传参,也便是 this的地址,已经是派生类的地址了,不是调用前基类的地址

testCPPVirtualMemeory`VDerived::test: 0x100003e00 < +0>:  pushq  %rbp //   栈低指针压栈  0x100003e01 < +1>:  movq   %rsp, %rbp //  将BP指针指向SP,由于上一级函数的栈顶指针是下一级函数的栈底指针 0x100003e04 < +4>:  subq   $0x10, %rsp // 开端函数栈帧空间 0x100003e08 < +8>:  movq   %rdi, -0x8(%rbp) //  将函数第一个参数入栈,也便是this 指针 -> 0x100003e0c < +12>: leaq 0x15c(%rip), %rdi         ; "test\n" 0x100003e13 < +19>: movb   $0x0, %al 0x100003e15 < +21>: callq 0x100003efc ; symbol stub for: printf 0x100003e1a < +26>: addq   $0x10, %rsp //回收栈空间 0x100003e1e < +30>: popq   %rbp //出栈 指回上一层 rbp 0x100003e1f < +31>: retq //指向下一条指令

image.png

经过上面的汇编咱们剖析,编译器在调用虚函数表中的函数时,是经过 *(%rcx) 直接寻址,然后中间做了某一个操作,跳到 test的完成,那么这个进程中thunk做了什么操作呢?

llvm-thunk源代码剖析

小编运用的IDE都运用的是LLVM编译器,于是经过翻看LLVM的源码找到了答案: 在VTableBuilder.cpp的AddMethods函数,小编找到了答案,描绘如下:

// Now go through all virtual member functions and add them to the current
  // vftable. This is done by
  //  - replacing overridden methods in their existing slots, as long as they
  //    don't require return adjustment; calculating This adjustment if needed.
  //  - adding new slots for methods of the current base not present in any
  //    sub-bases;
  //  - adding new slots for methods that require Return adjustment.
  // We keep track of the methods visited in the sub-bases in MethodInfoMap.

编译器在编译的时分会判断基类的虚函数派生类有没有覆盖,假如有完成的时分,则动态替换虚函数表中的地址为派生类的地址,一起:

1.会计算调用时this指针的地址是否需求调整,假如需求调整的话,会为当时的办法拓荒一块新的内存空间;

2.也会为需求this返回值的函数拓荒一块新的内存空间;

代码如下:

void VFTableBuilder::AddMethods(BaseSubobject Base, unsigned BaseDepth, const CXXRecordDecl *LastVBase,
                                BasesSetVectorTy &VisitedBases) { const CXXRecordDecl *RD = Base.getBase(); if (!RD->isPolymorphic()) return; const ASTRecordLayout &Layout = Context.getASTRecordLayout(RD); // See if this class expands a vftable of the base we look at, which is either // the one defined by the vfptr base path or the primary base of the current // class. const CXXRecordDecl *NextBase = nullptr, *NextLastVBase = LastVBase;
  CharUnits NextBaseOffset; if (BaseDepth < WhichVFPtr.PathToIntroducingObject.size()) {
    NextBase = WhichVFPtr.PathToIntroducingObject[BaseDepth]; if (isDirectVBase(NextBase, RD)) {
      NextLastVBase = NextBase;
      NextBaseOffset = MostDerivedClassLayout.getVBaseClassOffset(NextBase);
    } else {
      NextBaseOffset =
          Base.getBaseOffset() + Layout.getBaseClassOffset(NextBase);
    }
  } else if (const CXXRecordDecl *PrimaryBase = Layout.getPrimaryBase()) {
    assert(!Layout.isPrimaryBaseVirtual() && "No primary virtual bases in this ABI");
    NextBase = PrimaryBase;
    NextBaseOffset = Base.getBaseOffset();
  } if (NextBase) {
    AddMethods(BaseSubobject(NextBase, NextBaseOffset), BaseDepth + 1,
               NextLastVBase, VisitedBases); if (!VisitedBases.insert(NextBase))
      llvm_unreachable("Found a duplicate primary base!");
  }

  SmallVector< const CXXMethodDecl*, 10> VirtualMethods; // Put virtual methods in the proper order. GroupNewVirtualOverloads(RD, VirtualMethods); // Now go through all virtual member functions and add them to the current // vftable. This is done by //  - replacing overridden methods in their existing slots, as long as they //    don't require return adjustment; calculating This adjustment if needed. //  - adding new slots for methods of the current base not present in any //    sub-bases; //  - adding new slots for methods that require Return adjustment. // We keep track of the methods visited in the sub-bases in MethodInfoMap. for (const CXXMethodDecl *MD : VirtualMethods) {
    FinalOverriders::OverriderInfo FinalOverrider =
        Overriders.getOverrider(MD, Base.getBaseOffset()); const CXXMethodDecl *FinalOverriderMD = FinalOverrider.Method; const CXXMethodDecl *OverriddenMD =
        FindNearestOverriddenMethod(MD, VisitedBases);

    ThisAdjustment ThisAdjustmentOffset; bool ReturnAdjustingThunk = false, ForceReturnAdjustmentMangling = false;
    CharUnits ThisOffset = ComputeThisOffset(FinalOverrider);
    ThisAdjustmentOffset.NonVirtual =
        (ThisOffset - WhichVFPtr.FullOffsetInMDC).getQuantity(); if ((OverriddenMD || FinalOverriderMD != MD) &&
        WhichVFPtr.getVBaseWithVPtr())
      CalculateVtordispAdjustment(FinalOverrider, ThisOffset,
                                  ThisAdjustmentOffset); unsigned VBIndex =
        LastVBase ? VTables.getVBTableIndex(MostDerivedClass, LastVBase) : 0; if (OverriddenMD) { // If MD overrides anything in this vftable, we need to update the // entries. MethodInfoMapTy::iterator OverriddenMDIterator =
          MethodInfoMap.find(OverriddenMD); // If the overridden method went to a different vftable, skip it. if (OverriddenMDIterator == MethodInfoMap.end()) continue;

      MethodInfo &OverriddenMethodInfo = OverriddenMDIterator->second;

      VBIndex = OverriddenMethodInfo.VBTableIndex; // Let's check if the overrider requires any return adjustments. // We must create a new slot if the MD's return type is not trivially // convertible to the OverriddenMD's one. // Once a chain of method overrides adds a return adjusting vftable slot, // all subsequent overrides will also use an extra method slot. ReturnAdjustingThunk = !ComputeReturnAdjustmentBaseOffset(
                                  Context, MD, OverriddenMD).isEmpty() ||
                             OverriddenMethodInfo.UsesExtraSlot; if (!ReturnAdjustingThunk) { // No return adjustment needed - just replace the overridden method info // with the current info. MethodInfo MI(VBIndex, OverriddenMethodInfo.VFTableIndex);
        MethodInfoMap.erase(OverriddenMDIterator);

        assert(!MethodInfoMap.count(MD) && "Should not have method info for this method yet!");
        MethodInfoMap.insert(std::make_pair(MD, MI)); continue;
      } // In case we need a return adjustment, we'll add a new slot for // the overrider. Mark the overridden method as shadowed by the new slot. OverriddenMethodInfo.Shadowed = true; // Force a special name mangling for a return-adjusting thunk // unless the method is the final overrider without this adjustment. ForceReturnAdjustmentMangling =
          !(MD == FinalOverriderMD && ThisAdjustmentOffset.isEmpty());
    } else if (Base.getBaseOffset() != WhichVFPtr.FullOffsetInMDC ||
               MD->size_overridden_methods()) { // Skip methods that don't belong to the vftable of the current class, // e.g. each method that wasn't seen in any of the visited sub-bases // but overrides multiple methods of other sub-bases. continue;
    } // If we got here, MD is a method not seen in any of the sub-bases or // it requires return adjustment. Insert the method info for this method. MethodInfo MI(VBIndex,
                  HasRTTIComponent ? Components.size() - 1 : Components.size(),
                  ReturnAdjustingThunk);

    assert(!MethodInfoMap.count(MD) && "Should not have method info for this method yet!");
    MethodInfoMap.insert(std::make_pair(MD, MI)); // Check if this overrider needs a return adjustment. // We don't want to do this for pure virtual member functions. BaseOffset ReturnAdjustmentOffset;
    ReturnAdjustment ReturnAdjustment; if (!FinalOverriderMD->isPure()) {
      ReturnAdjustmentOffset =
          ComputeReturnAdjustmentBaseOffset(Context, FinalOverriderMD, MD);
    } if (!ReturnAdjustmentOffset.isEmpty()) {
      ForceReturnAdjustmentMangling = true;
      ReturnAdjustment.NonVirtual =
          ReturnAdjustmentOffset.NonVirtualOffset.getQuantity(); if (ReturnAdjustmentOffset.VirtualBase) { const ASTRecordLayout &DerivedLayout =
            Context.getASTRecordLayout(ReturnAdjustmentOffset.DerivedClass);
        ReturnAdjustment.Virtual.Microsoft.VBPtrOffset =
            DerivedLayout.getVBPtrOffset().getQuantity();
        ReturnAdjustment.Virtual.Microsoft.VBIndex =
            VTables.getVBTableIndex(ReturnAdjustmentOffset.DerivedClass,
                                    ReturnAdjustmentOffset.VirtualBase);
      }
    }

    AddMethod(FinalOverriderMD,
              ThunkInfo(ThisAdjustmentOffset, ReturnAdjustment,
                        ForceReturnAdjustmentMangling ? MD : nullptr));
  }
}

经过上面代码剖析,在this 需求调整的时分,都是经过AddMethod(FinalOverriderMD,ThunkInfo(ThisAdjustmentOffset, ReturnAdjustment,ForceReturnAdjustmentMangling ? MD : nullptr))函数来增加一个ThunkInfo的结构体,ThunkInfo在结构体(完成在ABI.h)如下:

struct ThunkInfo {
  /// The \c this pointer adjustment.
  ThisAdjustment This;

  /// The return adjustment.
  ReturnAdjustment Return;

  /// Holds a pointer to the overridden method this thunk is for, /// if needed by the ABI to distinguish different thunks with equal
  /// adjustments. Otherwise, null.
  /// CAUTION: In the unlikely event you need to sort ThunkInfos, consider using
  /// an ABI-specific comparator. const CXXMethodDecl *Method; ThunkInfo() : Method(nullptr) { }

  ThunkInfo(const ThisAdjustment &This, const ReturnAdjustment &Return, const CXXMethodDecl *Method = nullptr)
      : This(This), Return(Return), Method(Method) {} friend bool operator==(const ThunkInfo &LHS, const ThunkInfo &RHS) { return LHS.This == RHS.This && LHS.Return == RHS.Return &&
           LHS.Method == RHS.Method; }

  bool isEmpty() const { return This.isEmpty() && Return.isEmpty() && Method == nullptr;
  }
};

}

Thunkinfo的结构体有一个method,寄存函数的实在完成,This和Return记录this需求调整的信息,然后在生成办法的时分,根据这些信息,编译器主动刺进thunk函数的信息,经过ItaniumMangleContextImpl::mangleThunk(const CXXMethodDecl *MD,const ThunkInfo &Thunk,raw_ostream &Out)的函数,咱们得到了证明,函数如下:

(mangle和demangle:将C++源程序标识符(original C++ source identifier)转化成C++ ABI标识符(C++ ABI identifier)的进程称为mangle;相反的进程称为demangle。wiki)

void ItaniumMangleContextImpl::mangleThunk(const CXXMethodDecl *MD,
                                           const ThunkInfo &Thunk,
                                           raw_ostream &Out) {
  //  < special-name> ::= T < call-offset> < base encoding>
  //                      # base is the nominal target function of thunk
  //  < special-name> ::= Tc < call-offset> < call-offset> < base encoding>
  //                      # base is the nominal target function of thunk
  //                      # first call-offset is 'this' adjustment
  //                      # second call-offset is result adjustment assert(!isa< CXXDestructorDecl>(MD) &&
         "Use mangleCXXDtor for destructor decls!");
  CXXNameMangler Mangler(*this, Out);
  Mangler.getStream() << "_ZT"; if (!Thunk.Return.isEmpty())
    Mangler.getStream() << 'c';

  // Mangle the 'this' pointer adjustment.
  Mangler.mangleCallOffset(Thunk.This.NonVirtual,
                           Thunk.This.Virtual.Itanium.VCallOffsetOffset);

  // Mangle the return pointer adjustment if there is one. if (!Thunk.Return.isEmpty())
    Mangler.mangleCallOffset(Thunk.Return.NonVirtual,
                             Thunk.Return.Virtual.Itanium.VBaseOffsetOffset);

  Mangler.mangleFunctionEncoding(MD);
}

thunk汇编指令剖析

至此,经过LLVM源码咱们解开了thunk技能的真面目,那么咱们经过反汇编程序来验证证明一下, 这儿运用objdump 或者逆向利器 hopper都能够,小编运用的是hopper,汇编代码如下:

1.咱们先来看编译器完成的thunk 版的test函数

派生类完成的test函数

image.png

编译器完成的thunk版的test函数

image.png

2.经过上面两张截图咱们发现

编译器完成的thunk的test函数地址为0x100003e30

派生类完成的test函数地址为0x100003e00

下面咱们来看下派生类的虚表中存的实在地址是那一个

image.png

经过上图咱们能够看到:派生类的虚表中存的实在地址为编译器动态增加的thunk函数的地址0x100003e30。

上面剖析的*(rcx)直接寻址:便是调用thunk函数的完成,然后在thunk中去调用实在的派生类覆盖的函数。

在这儿咱们能够确认的 thunk技能:

便是编译器在编译的时分,遇到调用this和返回值this需求调整的地方,动态的参加对应的thunk版的函数,在thunk函数的内部完成this的偏移调整,和调用派生类完成的虚函数;并将编译器完成的thunk函数的地址存入虚表中,而不是派生类完成的虚函数的地址。

thunk函数的内存布局

也能够确认对应的内存布局如下:

image.png

故(承继链中不是第一个)虚函数承继的基类指针的调用顺序为:

image.png

virtual-thunk和non-virtual-thunk

注意:在这儿能够看到,内存中有两份VBase,在多承继中分为一般承继、虚函数承继、虚承继。虚承继主要是为了解决上面看到的问题:在内存中一起有两份Vbase 的内存,将上面的代码改动一下就会保证内存中的实例只要一份:

class VBaseA: public VBase 改成 class VBaseA: public virtual VBase

class VBaseB: public VBase 改成 class VBaseB: public virtual VBase

这样内存中的VBase就只要一分内存了。

到这儿还有问题没有回答,便是上面截图里的thunk函数类型是:

咱们发现thunk函数是 non-virtual-thunk类型,那对应的virtual-thunk是什么类型呢?

在回答这个问题之前咱们现看下下面的例子?

public A { virtual void test() {
    }
} public B { virtual void test1() {
    }
} public C { virtual void test2() {
    }
} public D : public virtual A, public virtual B, public C { virtual void test1() { // 这儿完成的test1函数在 B类的虚函数表里便是virtual-trunk的类型 } virtual void test2() { // 这儿完成的test2函数在 C类的虚函数表示便是no-virtual-trunk的类型 }
}

虚函数承继和虚承继相结合,且该类在派生类的承继链中不是第一个基类的时分,则该派生类完成的虚函数在编译器编译的时分,虚表里寄存便是virtual-trunk类型。

只要虚函数承继的时分,且该类在派生类的承继链中不是第一个基类的时分,则该派生类完成的虚函数在编译器编译的时分,虚表里寄存便是no-virtual-trunk类型。

3 为什么LLDB调试器显现的地址相同呢?

image.png

假如做了偏移,那为什么LLDB expression显现的地址是派生类目标的首地址呢?

到了现在了解了什么是thunk技能,还没有一个问题没有解决:便是LLDB调试的时分,显现的this的地址是基类偏移后的(派生类的地址),前面经过汇编剖析编译器在类型转化的时分,做了实在的偏移,经过读取内存地址也发现是偏移后的实在地址,那lldb expression获取的地址为啥仍是派生类的地址呢?由此能够猜想是LLDB调试器经过exppress 指令履行的时分做了类型的转化。

经过翻阅LLDB调试器的源码和LLDB阐明文档,经过文档得知LLDB在每次拿到一个地址,需求向用户友好的展现的时分,首先需求经过summary format()进行格式化转化,格式化转化的依据是动态类型(lldb-getdynamictypeandaddress)的获取,在LLDB源码的bool ItaniumABILanguageRuntime::GetDynamicTypeAndAddress (lldb-summary-format)函数中找到了答案,代码如下

// For Itanium, if the type has a vtable pointer in the object, it will be at // offset 0 // in the object.  That will point to the "address point" within the vtable // (not the beginning of the // vtable.)  We can then look up the symbol containing this "address point" // and that symbol's name // demangled will contain the full class name. // The second pointer above the "address point" is the "offset_to_top".  We'll // use that to get the // start of the value object which holds the dynamic type.
bool ItaniumABILanguageRuntime::GetDynamicTypeAndAddress(
    ValueObject &in_value, lldb::DynamicValueType use_dynamic,
    TypeAndOrName &class_type_or_name, Address &dynamic_address,
    Value::ValueType &value_type) { // For Itanium, if the type has a vtable pointer in the object, it will be at // offset 0 // in the object.  That will point to the "address point" within the vtable // (not the beginning of the // vtable.)  We can then look up the symbol containing this "address point" // and that symbol's name // demangled will contain the full class name. // The second pointer above the "address point" is the "offset_to_top".  We'll // use that to get the // start of the value object which holds the dynamic type. // class_type_or_name.Clear();
  value_type = Value::ValueType::eValueTypeScalar; // Only a pointer or reference type can have a different dynamic and static // type: if (CouldHaveDynamicValue(in_value)) { // First job, pull out the address at 0 offset from the object. AddressType address_type;
    lldb::addr_t original_ptr = in_value.GetPointerValue(&address_type); if (original_ptr == LLDB_INVALID_ADDRESS) return false; ExecutionContext exe_ctx(in_value.GetExecutionContextRef()); Process *process = exe_ctx.GetProcessPtr(); if (process == nullptr) return false;

    Status error; const lldb::addr_t vtable_address_point = process->ReadPointerFromMemory(original_ptr, error); if (!error.Success() || vtable_address_point == LLDB_INVALID_ADDRESS) { return false;
    }

    class_type_or_name = GetTypeInfoFromVTableAddress(in_value, original_ptr,
                                                      vtable_address_point); if (class_type_or_name) {
      TypeSP type_sp = class_type_or_name.GetTypeSP(); // There can only be one type with a given name, // so we've just found duplicate definitions, and this // one will do as well as any other. // We don't consider something to have a dynamic type if // it is the same as the static type.  So compare against // the value we were handed. if (type_sp) { if (ClangASTContext::AreTypesSame(in_value.GetCompilerType(),
                                          type_sp->GetForwardCompilerType())) { // The dynamic type we found was the same type, // so we don't have a dynamic type here... return false;
        } // The offset_to_top is two pointers above the vtable pointer. const uint32_t addr_byte_size = process->GetAddressByteSize(); const lldb::addr_t offset_to_top_location =
            vtable_address_point - 2 * addr_byte_size; // Watch for underflow, offset_to_top_location should be less than // vtable_address_point if (offset_to_top_location >= vtable_address_point) return false; const int64_t offset_to_top = process->ReadSignedIntegerFromMemory(
            offset_to_top_location, addr_byte_size, INT64_MIN, error); if (offset_to_top == INT64_MIN) return false; // So the dynamic type is a value that starts at offset_to_top // above the original address. lldb::addr_t dynamic_addr = original_ptr + offset_to_top; if (!process->GetTarget().GetSectionLoadList().ResolveLoadAddress(
                dynamic_addr, dynamic_address)) {
          dynamic_address.SetRawAddress(dynamic_addr);
        } return true;
      }
    }
  } return class_type_or_name.IsEmpty() == false;
}

经过上面代码剖析可知,每次在经过LLDB 指令expression动态调用 指针地址的时分,LLDB 会去依照调试器默认的格式进行格式化,格式化的条件是动态获取到对应的类型和偏移后的地址;在碰到C++有虚表的时分,且不是虚表中的第一个基类指针的时分,就会运用指针上头的offset_to_top 获取到这个对应动态的类型和返回动态获取的该类型目标开端的地址。

五 总结

image.png

  1. 上面主要验证了在指针类型转化的时分,编译器内部做了实在的地址偏移;
  2. 经过上面的剖析,咱们得知编译器在函数调用时经过thunk技能动态调整入参this指针和返回值this指针,保证C++调用时this的正确性;
  3. 在经过LLDB expression获取非虚函数基类指针内容时,LLDB内部经过summary format进行格式化转化,格式化转化时会进行动态类型的获取。

六 东西篇

1 获取汇编程序

预处理->汇编

clang++ -E main.cpp -o main.i
clang++ -S main.i

objdump

objdump -S -C 可履行程序

反汇编利器: hopper

下载hopper,可履行程序拖入即可

Xcode

Xcode->Debug->Debug WorkFlow->Show disassembly

2 导出C++内存布局

Clang++编译器

clang++ -cc1 -emit-llvm -fdump-record-layouts -fdump-vtable-layouts main.cpp

七 参考文献

https://matklad.github.io/2017/10/21/lldb-dynamic-type.html
https://lldb.llvm.org/use/variable.html
https://github.com/llvm-mirror/lldb/blob/bc19e289f759c26e4840aab450443d4a85071139/source/Plugins/LanguageRuntime/CPlusPlus/ItaniumABI/ItaniumABILanguageRuntime.cpp#L185
https://clang.llvm.org/doxygen/VTableBuilder_8cpp_source.html#l03109
https://clang.llvm.org/doxygen/ABI_8h_source.html

相关技能:

llvm-virtual-thunk
llvm-no-virtual-thunk
lldb-summary-format
lldb-getdynamictypeandaddress


数据库安全

 


本公司销售:阿里云新/老客户,只要购买阿里云,即可享受折上折优惠!>

留言:

  1. [回复]阿里云代理发表于 2022-01-26 16:57
    棒棒棒!

我有话说: