混合使用C、C++和汇编语之:内联汇编和嵌入型汇编的使用
7.相关基类的关键字
利用以下关键字可以确定从对象起始处到其相关基类的偏移量:
__offsetof_base(D,B)
其中,B必须是D的非虚拟基类。
该函数返回从D对象的起始处到其中B基子对象的起始处的偏移量。结果可能是零。必须将偏移量(以字节为单位)添加到D*p来执行。
static_castB*>(p)的等效功能,如下程序段所示:
__asmB*my_static_base_cast(D*/*p*/){
if__offsetof_base(D,B)>0 //排除偏移量为0的情况
ADDr0,r0,#__offsetof_base(D,B)
endif
MOVpc,lr
}
在汇编程序源代码中,这些关键字被转换为整数或逻辑常量。只能将它们用于__asm函数,而不能用于__cpp表达式。
8.成员函数类的关键字
以下关键字方便了从__asm函数中调用虚拟或非虚拟成员函数。以__mcall开头的关键字可用于虚拟和非虚拟函数。以__vcall开头的关键字仅能用于虚拟函数。在调用静态成员函数的过程中,这些关键字没有特别的作用。
下面详细介绍这些关键字的使用。
①__mcall_is_virtual(D,f)
如果f是D中的虚拟成员函数或是D的基类,结果是{TRUE},否则结果是{FALSE}。如果返回{TRUE},可用虚拟调度进行调用,否则必须直接进行调用。
②__mcall_is_in_vbase(D,f)
如果f是D虚拟基类中的非静态成员函数,结果是{TRUE},否则结果是{FALSE}。如果返回{TRUE},必须用__mcall_offsetof_vbaseptr(D,f)进行this调整,否则必须用__mcall_this_
offset(D,f)进行调整。
③__mcall_this_offset(D,f)
其中D是类,f是D中定义的非静态成员函数或是D的非虚拟基类。该函数返回从D对象的起始处到定义f的基的起始处的偏移量。在用指向D的指针调用f的过程中,这是必要的this调整。返回值在D中可找到f时为零,或者与__offsetof_base(D,B)相同,其中B为包含f的D非虚拟基类。在D的虚拟基类中找到f时,如果使用__mcall_this_offset(D,f),则返回任意值,在程序中使用该返回值,汇编器将报告__mcall_this_offset无效使用的错误。
④__vcall_offsetof_vfunc(D,f)
其中D是类,f是D中定义的虚拟函数或是D的基类。将偏移量返回到虚拟函数表,在该表中可以找到从虚拟函数表到虚拟函数的偏移量。在f不是虚拟成员函数时,如果使用__vcall_offsetof_vfunc(D,f),则返回任意值,而在设计上使用该值时会导致汇编错误。
9.调用非静态成员函数
本小节列出了可以从__asm函数中调用虚拟或非虚拟函数的关键字。静态成员函数的参数不相同(没有this),使得检测静态成员函数的关键字__mcall_is_static不可用,因此调用位置很可能已经专用于调用静态成员函数。
(1)调用非虚拟成员函数
例如,在虚拟基(virtualbase)或非虚拟基(non-virtualbase)中,以下代码可用于调用虚拟函数:
//rp包含指向D的指针,该程序的功能是实现在使用rp时调用D的非虚成员函数f
//所有参数准备好
//假设并不返回一个结构类型
if__mcall_is_in_vbase(D,f)
ASSERT{FALSE}//cantaccessvirtualbase
else
MOVr0,rp //使用指向D的指针rp*
ADDr0,r0,#__mcall_this_offset(D,f) //地址调整
endif
BL__cpp(D::f)
(2)调用虚拟成员函数
例如,在虚拟或非虚拟基中,以下代码可用于调用虚拟函数:
//rp包含指向D的指针,该程序的功能是在使用rp时调用D的虚拟函数f
//所有参数准备好
//假如函数并不返回一个结构类型
if__mcall_is_in_vbase(D,f)
ASSERT{FALSE} //不能调用虚拟基
else
MOVr0,rp //使用指向D的指针rp
LDRr12,[rp] //加载vtable表结构指针
ADDr0,r0,#__mcall_this_offset(D,f) //地址调整
endif
MOVlr,pc //保存返回地址到lr
LDRpc,[r12,#__vcall_offsetof_vfunc(D,f)] //调用函数rp→f()
c语言相关文章:c语言教程
c++相关文章:c++教程
评论