现在的位置: 首页 > 综合 > 正文

不使用调试器查看堆栈的调用情况(CallTrace)

2019年08月07日 ⁄ 综合 ⁄ 共 929字 ⁄ 字号 评论关闭

转自:http://www.xue5.com/itedu/200707/127638.html

程序不大,调了三个小时,自己有必要在强化下对指针的理解:(

原理:函数调用的时候会建立栈帧,其大致存放如下

压栈传入的参数2

压栈传入的参数1

------------------------

函数返回时的地址

------------------------

栈帧

-----------------------

局部变量1

局部变量2

注意:上面表示的内存方式为上面是高地址,下面为低地址。堆栈的生长方向向下。

ebp保存当前栈帧 ,当前栈帧指向的内存里面存放调用函数的栈帧,紧接着的就是返回地址

根据这个原则编写代码如下:



#define GetCurFrame( r )__asm mov r , ebp 

#define GetNextFrame( m , r )r = ((void**)m)[0]

#define GetRetIns( m , r )r = ((void**)m)[1]

long Array[8];

void ShowFrame(long * npIns,int nDeep )

{

 int i =0;

 long* nCurFrame;

 long* nNextFrame;

 long* nRetIns;

 GetCurFrame(nCurFrame);

 for (i =0; i<nDeep; i++)

 { 

  GetNextFrame(nCurFrame,nNextFrame);

  GetRetIns(nCurFrame,nRetIns);

  npIns[i*2] = (long)nNextFrame;

  npIns[i*2 +1] = (long)nRetIns;

  nCurFrame = nNextFrame;

  printf("\nStack Frame\t:0x%08x\r\n",npIns[i*2]);

  printf("\nIns Ret At\t:0x%08x\r\n",npIns[i*2+1]);

 }

 return;

}

栈帧存放在 npIns的偶索引,函数的返回地址存放在对应的奇索引。

由于程序中无法知道链接器和加载器所使用的符号表,无法在程序中

直接函数返回地址转换为函数名。需要使用外部工具,Addr2line 将函数返回地址转化为函数名。


抱歉!评论已关闭.