2.4 OpenEuler中C语言中的函数调用测试(选做)
任务详情
- 在X86_64架构下实践2.4中的内容
- 通过GDB查看寄存器的内容,把教材中的图填入具体的值
- 把2.4的C代码在OpenEuler中重新实践一遍,绘制出ARM64的逻辑框图
- 实验内容要经过答辩才能得到相应分数
一、32位gcc中的运行时堆栈使用情况
main.c代码:
main(){
int a,b,c;
a=1;b=2;c=3;
c=sub(a,b);
printf("c=%d\n",c);
}
sub.c代码
int sub(int x,int y){
int u,v;
u=4;v=5;
return x+y+u+v;
}
使用指令gcc -g main.c sub.c -o main -m32
进入gdb调试
在main处设置断点
disas:查看汇编代码
i(info) r(registers):查看寄存器的值
可以看到U和v入栈(int的字节为4)
以及他们存放的地址
传递参数:此时pc指向的是sub()的入口地址
进入sub后的堆栈内容:现在我们已经知道u、v的内容以及物理地址
我们可以由此写出函数的调用序列。
long jump
代码:
include<stdio.h>
#include<setjmp.h>
jmp_buf env;
int main()
{
int r,a=100;
printf("call setjmp to save environmen\n");
if((r=setjmp(env))==0){
A();
printf("normal return\n");
}
else
printf("back to main() via long jump,r=%d,a=%d",r,a);
}
int A()
{
printf("enter A()\n");
B();
printf("exit A()\n");
}
int B()
{
printf("enter B()\n");
printf("long jump?(y|n) ");
if(getchar()=='y')
longjmp(env,1234);
printf("exit B()\n");
}
编译指令:gcc jump.c -o jump -m32
运行结果:
在main处设置断点,查看汇编代码:
二、64位gcc中的运行时堆栈使用情况
代码:
#include <stdio.h>
int sub(int a, int b, int c, int d, int e, int f, int g, int h)
{
int u, v, w;
u = 9;
v = 10;
w= 11;
return a+g+u+v;
}
int main()
{
int a, b, c, d, e,f, g, h,i;
a = 1;
b = 2;
c = 3;
d = 4;
e = 5;
f = 6;
g = 7;
h = 8;
i = sub(a,b,c,d,e,f,g,h);
}
运行结果:
不会产生结果。
生成.s文件,查看汇编代码:
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了