怎样进行C++调用

进行C++调用时遇到了棘手的问题,Naked Call这是一个很少见的C++调用约定,建议程序设计者不要使用。编译器不会给这种函数增加初始化和清理代码,省的变成了白用工。

创新互联主要从事成都网站建设、网站建设、网页设计、企业做网站、公司建网站等业务。立足成都服务和田县,10年网站建设经验,价格优惠、服务专业,欢迎来电咨询建站服务:18982081108

调用处push 1push 2call functionadd esp,8 注意:这里C++调用者在恢复堆栈被调用函数_function处push ebp 保存ebp寄存器,该寄存器将用来保存堆栈的栈顶指针,可以在函数退出时恢复mov ebp,esp 保存堆栈指针mov eax,[ebp + 8H] 堆栈中ebp指向位置之前依次保存有ebp,cs:eip,a,b,ebp +8指向aadd eax,[ebp + 0CH] 堆栈中ebp + 12处保存了bmov esp,ebp 恢复esppop ebpret 注意,这里没有修改堆栈

MSDN中说,该修饰自动在函数名前加前导的下划线,因此函数名在符号表中被记录为_function,但是我在编译时似乎没有看到这种变化。由于参数按照从右向左顺序压栈,因此最开始的参数在最接近栈顶的位置,因此当采用不定个数参数时,***个参数在栈中的位置肯定能知道,只要不定的参数个数能够根据***个后者后续的明确的参数确定下来,就可以使用不定参数,例如对于CRT中的sprintf函数,定义为:

 
 
 
  1. class A{public:   int function1(int a,int b);   
  2. int function2(int a,...);};
  3. int A::function1 (int a,int b){  
  4.  return a+b;
  5. }#include int A::function2(int a,...)
  6. {   va_list ap;  
  7.  va_start(ap,a);  
  8.  int i;   
  9. int result = 0;   
  10. for(i = 0 ; i < a ; i ++)  
  11.  {     
  12. result += va_arg(ap,int);   } 

函数的***个和第二个DWORD参数(或者尺寸更小的)通过ecx和edx传递,其他参数通过从右向左的顺序压栈被调用函数清理堆栈函数名修改规则同这是一个很少见的调用约定,一般程序设计者建议不要使用。

编译器不会给这种函数增加初始化和清理代码,更特殊的是,你不能用return返回返回值,只能用插入汇编返回结果。这一般用于实模式驱动程序设计,假设定义一个求和的加法程序,可以定义为:

 
 
 
  1. __declspec(naked) int add(int a,int b){ __asm mov eax,a __asm add eax,b __asm ret }

注意,这个函数没有显式的return返回值,返回通过修改eax寄存器实现,而且连退出函数的ret指令都必须显式插入。上面代码被翻译成汇编以后变成:

 
 
 
  1. declspec(naked) int add(int a,int b){ __asm mov eax,a __asm add eax,b __asm ret }

由于调用者没有理解WINAPI的含义错误的增加了这个修饰,上述代码必然导致堆栈被破坏,MFC在编译时插入的checkesp函数将告诉你,堆栈被破坏了。

文章题目:怎样进行C++调用
URL链接:http://www.csdahua.cn/qtweb/news0/30550.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网