如何解决在 64 位 linux 上返回 C 代码 x86 编译中的汇编函数
C 代码
#include <stdio.h>
int fibonacci(int);
int main()
{
int x = fibonacci(3);
printf("Fibonacci is : %d",x);
return 0;
}
组装
section .text
global fibonacci
fibonacci:
push ebp;
mov ebp,esp;
; initialize
mov dword [prev],0x00000000;
mov dword [cur],0x00000001;
mov byte [it],0x01;
mov eax,dword [ebp + 8]; // n = 3
mov byte [n],al;
getfib:
xor edx,edx;
mov dl,byte [n];
cmp byte [it],dl;
jg loopend;
mov eax,dword [prev];
add eax,dword [cur];
mov ebx,dword [cur];
mov dword [prev],ebx;
mov dword [cur],eax;
inc byte [it];
jmp getfib;
loopend:
mov eax,dword [cur];
pop ebp;
ret;
section .bss
it resb 1
prev resd 1
cur resd 1
n resb 1
我试图在 C 代码和调试中运行这个汇编函数,我看到 C 代码中变量 x 中的值是正确的,但是当我使用 printf 函数时出现了一些错误
需要帮助
编译命令:
nasm -f elf32 asmcode.asm -o a.o
gcc -ggdb -no-pie -m32 a.o ccode.c -o a.out
解决方法
您的代码不保留 ebx
寄存器,它是一个被调用者保留的寄存器。 main
函数显然尝试进行一些 rip
相对寻址,以使用 printf
作为基址寄存器获取 ebx
的格式字符串的地址。这会失败,因为您的代码覆盖了 ebx
。
要解决这个问题,请确保在使用之前保存所有被调用者保存的寄存器,然后在返回时恢复它们的值。例如,你可以这样做
fibonacci:
push ebp
mov ebp,esp
push ebx ; <---
...
pop ebx ; <---
pop ebp
ret
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。