Linux C函数调用栈帧结构

2024-03-31 21:08

本文主要是介绍Linux C函数调用栈帧结构,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Linux C程序的反汇编,每个函数第一个指令都是push %rbp,即当caller者调callee时,指令callq <fun addr> callee返回后的下一个指令地址压入栈帧,然后push %rbp 保存 rbp寄存器,紧接着mov %rsp,%rbp 更新rbp 寄存器这样层层调用构成函数调用栈,每个栈帧开始就是存的上一个栈帧的rbp, 结尾就是调用callee后的下一个执行令地址。函数调用详细介绍可以看这篇文章从汇编看Linux C函数的调用约定和参数传递的细节。

通过前面分析,可以得出,rbp寄存器存放的是当前栈帧的基址,当前栈帧的基址即开始地址则存放的是上一个栈帧的基址,即*rbp就是上一个栈帧的基址。对于x64,每个栈帧开头的八个字节存放的就是上个栈帧的基址。调用栈的结构图大致如下:

call stack

下面用一个例子来说明栈帧结构的细节。

int fun3(int a3)
{int a = 3;int re = a3;while(1);return re;
}int fun2(int a2)
{int a = 2;int re = fun3(a2);return re;
}int fun1(int a1)
{int a = 1;int re = fun2(a1);return re;
}int main()
{int a = 7;int b = fun1(a);return 0;
}

利用gdb对上面里进行详细分析,编译gcc -g main.c, 执行./a.out, 然后新开一个shell窗口ps afxu | grep a.out查找进程PID,gdb - <PID>进行调试。可以很容易分析出函数的栈帧结构组织关系,详细如下:

fun3 (a3=7) at main.c:5
5       while(1);
(gdb) bt
#0  fun3 (a3=7) at main.c:5
#1  0x000000000040051f in fun2 (a2=7) at main.c:12
#2  0x0000000000400543 in fun1 (a1=7) at main.c:19
#3  0x0000000000400564 in main () at main.c:26
(gdb) p $rbp                    // 当前栈帧0 rbp
$1 = (void *) 0x7fff1c4d0ce0
(gdb) x/i *(long*)($1 + 8)      // 栈帧1 返回地址0x40051f <fun2+28>:  mov    %eax,-0x4(%rbp)
(gdb) p/x *(unsigned long*)$rbp // 栈帧1 rbp地址
$2 = 0x7fff1c4d0d08
(gdb) x/i *(long*)($2 + 8)      // 栈帧2 返回地址0x400543 <fun1+28>:  mov    %eax,-0x4(%rbp)
(gdb) p/x *(unsigned long*)$2   // 栈帧3 rbp
$3 = 0x7fff1c4d0d30
(gdb) x/i *(long*)($3 + 8)      // 栈帧3 返回地址0x400564 <main+25>:  mov    %eax,-0x4(%rbp)
(gdb) x/24x $rsp                // 从rsp开始24 DWORD的栈内容
0x7fff1c4d0ce0: 0x1c4d0d08  0x00007fff                          // rbp0x0040051f  0x00000000  // return addr
0x7fff1c4d0cf0: 0xa68271a8  0x00000007  0xa6dfe4c0  0x00007fe4
0x7fff1c4d0d00: 0x00000002  0x00007fe40x1c4d0d30  0x00007fff  // rbp
0x7fff1c4d0d10: 0x00400543  0x00000000                          // return addr0x004005bd  0x00000007
0x7fff1c4d0d20: 0x1c4d0d50  0x00007fff  0x00000001  0x00000000
0x7fff1c4d0d30: 0x1c4d0d50  0x00007fff                          // rbp0x00400564  0x00000000  // return addr

还可以用pmap命令看布局:

root@ubuntu:~# pmap -p 23620
23620:   ./a.out
0000000000400000      4K r-x-- /root/a.out
0000000000600000      4K r---- /root/a.out
0000000000601000      4K rw--- /root/a.out
00007fe4a6817000   1776K r-x-- /lib/x86_64-linux-gnu/libc-2.19.so
00007fe4a69d3000   2044K ----- /lib/x86_64-linux-gnu/libc-2.19.so
00007fe4a6bd2000     16K r---- /lib/x86_64-linux-gnu/libc-2.19.so
00007fe4a6bd6000      8K rw--- /lib/x86_64-linux-gnu/libc-2.19.so
00007fe4a6bd8000     20K rw---   [ anon ]
00007fe4a6bdd000    140K r-x-- /lib/x86_64-linux-gnu/ld-2.19.so
00007fe4a6de6000     12K rw---   [ anon ]
00007fe4a6dfd000      8K rw---   [ anon ]
00007fe4a6dff000      4K r---- /lib/x86_64-linux-gnu/ld-2.19.so
00007fe4a6e00000      4K rw--- /lib/x86_64-linux-gnu/ld-2.19.so
00007fe4a6e01000      4K rw---   [ anon ]
00007fff1c4b2000    132K rw---   [ stack ]
00007fff1c5dd000      8K r-x--   [ anon ]
ffffffffff600000      4K r-x--   [ anon ]

参考

Computer Systems: A Programmer’s Perspective, 3/E (CS:APP3e)
函数调用栈的获取原理分析

这篇关于Linux C函数调用栈帧结构的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/865351

相关文章

Linux内核定时器使用及说明

《Linux内核定时器使用及说明》文章详细介绍了Linux内核定时器的特性、核心数据结构、时间相关转换函数以及操作API,通过示例展示了如何编写和使用定时器,包括按键消抖的应用... 目录1.linux内核定时器特征2.Linux内核定时器核心数据结构3.Linux内核时间相关转换函数4.Linux内核定时

Linux镜像文件制作方式

《Linux镜像文件制作方式》本文介绍了Linux镜像文件制作的过程,包括确定磁盘空间布局、制作空白镜像文件、分区与格式化、复制引导分区和其他分区... 目录1.确定磁盘空间布局2.制作空白镜像文件3.分区与格式化1) 分区2) 格式化4.复制引导分区5.复制其它分区1) 挂载2) 复制bootfs分区3)

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

Linux服务器数据盘移除并重新挂载的全过程

《Linux服务器数据盘移除并重新挂载的全过程》:本文主要介绍在Linux服务器上移除并重新挂载数据盘的整个过程,分为三大步:卸载文件系统、分离磁盘和重新挂载,每一步都有详细的步骤和注意事项,确保... 目录引言第一步:卸载文件系统第二步:分离磁盘第三步:重新挂载引言在 linux 服务器上移除并重新挂p

Linux下屏幕亮度的调节方式

《Linux下屏幕亮度的调节方式》文章介绍了Linux下屏幕亮度调节的几种方法,包括图形界面、手动调节(使用ACPI内核模块)和外接显示屏调节,以及自动调节软件(CaliseRedshift和Reds... 目录1 概述2 手动调节http://www.chinasem.cn2.1 手动屏幕调节2.2 外接显

Linux(centos7)虚拟机没有IP问题及解决方案

《Linux(centos7)虚拟机没有IP问题及解决方案》文章介绍了在CentOS7中配置虚拟机网络并使用Xshell连接虚拟机的步骤,首先,检查并配置网卡ens33的ONBOOT属性为yes,然后... 目录输入查看ZFhrxIP命令:ip addr查看,没有虚拟机IP修改ens33配置文件重启网络Xh

linux实现对.jar文件的配置文件进行修改

《linux实现对.jar文件的配置文件进行修改》文章讲述了如何使用Linux系统修改.jar文件的配置文件,包括进入文件夹、编辑文件、保存并退出编辑器,以及重新启动项目... 目录linux对.jar文件的配置文件进行修改第一步第二步 第三步第四步总结linux对.jar文件的配置文件进行修改第一步进

linux ssh如何实现增加访问端口

《linuxssh如何实现增加访问端口》Linux中SSH默认使用22端口,为了增强安全性或满足特定需求,可以通过修改SSH配置来增加或更改SSH访问端口,具体步骤包括修改SSH配置文件、增加或修改... 目录1. 修改 SSH 配置文件2. 增加或修改端口3. 保存并退出编辑器4. 更新防火墙规则使用uf

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.