微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

在 64 位 linux 上返回 C 代码 x86 编译中的汇编函数

如何解决在 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

如果图片看起来模糊,请单击下面的图片

下面是 printf 执行前的调试

下面是 printf 执行后

解决方法

您的代码不保留 ebx 寄存器,它是一个被调用者保留的寄存器。 main 函数显然尝试进行一些 rip 相对寻址,以使用 printf 作为基址寄存器获取 ebx 的格式字符串的地址。这会失败,因为您的代码覆盖了 ebx

要解决这个问题,请确保在使用之前保存所有被调用者保存的寄存器,然后在返回时恢复它们的值。例如,你可以这样做

fibonacci:

    push ebp
    mov ebp,esp
    push ebx ; <---

    ...

    pop ebx ; <---
    pop ebp
    ret

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。