易截截图软件、单文件、免安装、纯绿色、仅160KB

C函数返回结构体在汇编下的实现

原文链接:http://blog.csdn.net/sytstarac/archive/2009/08/05/4411519.aspx
编译器:vc++6.0(因为此种实现依赖编译器处理)
此处只简要叙述一下机制。并附部分关键指令序列。
准备:
1,关于EBP:称做栈基址指针。为什么这样说呢?我们先来看看函数调用的过程:
参数从右到左压栈。
call指令执行,该指令将导致EIP压栈。
每个函数前两句必定是:push ebp   mov ebp,esp。则call指令后,跳到被调函数出开始执行。保存ebp,即ebp压栈。
局部变量压栈。一般是sub esp,xxx的形式。
这就是将ebp称为基指针的原因:ebp-xx访问的时局部变量;ebp+xx访问的是参数。最左边参数地址是ebp+8h。ebp坐镇中间为基准。
2,函数返回值统一放入eax中。只要放得下。
3,栈扩展方向为从高地址到低地址。结构内的变量存贮方式:低地址对应声明顺序靠前的成员。一定要注意这里的区别!它关系到反汇编生成的代码里面的数字是怎么算出来的。但如果你自己写汇编代码就不用考虑这些了。只取成员名即可。
4,不能直接在两个存储器变量间用mov指令。
主要原理:
    当调用一个返回结构体的函数时,在vc++下,是这样处理:
    首先sub esp,xx,在堆栈上开辟一个空间。大小为结构体大小。
    然后lea eax,[esp-xx],即将结构体在堆栈中的地址送eax。
    push 参数。
    push eax。
    call 函数。
    被调用函数内部:
    routine:
    push ebp     mov ebp,esp
    一般是定义一个结构体局部变量:
    struct aa a;
    sub esp, sizeof aa
    然后处理结构体,
.
.
.
.
最后return a。
首先mov eax,[ebp+8h]      ;将外面的调用函数在堆栈内开辟的结构体指针赋予eax。
然后将被调用函数在堆栈内开辟的结构体内的值赋到调用函数开辟的结构体内。
一般形式是:mov ecx,[ebp-结构体大小]        ;赋第一个成员
                mov [eax],ecx
                mov


相关文档:

加入crimson.jar包后不能使用C标签

 Crimson来自于Sun捐赠给Apache的ProjectX项目,Xerces来自IBM捐赠给Apache的XML4J项目,结果Xerces胜出,成了Apache XML小组全力开发的XML API,而Crimon已经早就不做了,如今Xerces名满天下,到处都是在用Xerces DOM和SAX解析器,只有Sun不服气,非要在JDK1.4里面使用过时的Crimson,让人感觉像是在赌气一样,真是让 ......

C/C++ 笔试、面试题目大汇总

1.求下面函数的返回值(微软)
int func(x)
{
    int countx = 0;
    while(x)
    {
          countx ++;
          x = x&(x-1);
   &n ......

C/C++招聘的一些感受和经验

前段时间刚参加了n多公司的C/C++软件工程师的面试,有国企,外企,私企(moto,飞思卡尔,港湾,中国卫星XXX(这个牛))等等等等。感受感想颇多,近日终于空闲,在此表述一下。
    本人基本条件:3年开发经验,2year+ Windows development experence,1year+ Linux experence. 熟悉C,C++,MFC/SDK/API,MiniG ......

#ifdef __cplusplus extern "C"

面试时被问到过,不甚明了,网上百度一下,整合了两个仁兄的文章,如下。:-)
时常在cpp的代码之中看到这样的代码:
#ifdef __cplusplus //c++编译环境中才会定义__cplusplus (plus就是"+"的意思)
extern "C" { //告诉编译器下面的函数是c语言函数(因为c++和c语言对函数的编译转换不一样,主要是c++中存在重载)
#en ......
© 2009 ej38.com All Rights Reserved. 关于E健网联系我们 | 站点地图 | 赣ICP备09004571号