在 C 中调用 NASM 函数



我正在尝试学习x86汇编程序,我想用C调用NASM函数。当我运行我的程序时,我收到此错误:

分段错误(核心转储)

我已经尝试了数十种简单的测试函数变体,但它每次都在同一位置停止。

以下是我的asmc文件:

div.asm:

global _test
_test:
    push    ebp
    mov     ebp, esp
    push    ebx
    mov     eax, [ebp+8]
    mov     ebx, [ebp+12]
    div     ebx
    pop     ebp
    ret

主.c:

#include <stdio.h>
extern unsigned int test (unsigned int, unsigned int);
int main(void)
{
    printf("%dn", div(85,5));
    return 0;
}

我编译并链接文件:

nasm -f elf -o div.o div.asm
gcc -m32 -c -o main.o main.c
gcc -m32 -o run div.o main.o

我在Virtual Machine中使用64 Bit Linux.

在这里犯了什么错误,我该如何解决?

你忘了弹出 ebx(或者至少按顺序排列堆栈):

push    ebp
mov     ebp, esp
push    ebx         ; you push it here
mov     eax, [ebp+8]
mov     ebx, [ebp+12]
xor     edx,edx     ; ..and you must zero edx
div     ebx
pop     ebx         ; forgot to pop it here
pop     ebp
ret

目前还不清楚你是否曾经解决过你的问题。除了其他问题之外,您还需要在main.c中进行函数调用,以匹配div.asm中的调用。例如,如果你创建了一个汇编函数_test,你需要将其声明为extern,并在main中实际使用该函数。例如:

#include <stdio.h>
extern unsigned int _test (unsigned int, unsigned int);
int main(void)
{
    printf("%dn", _test (85,5));    /* you are calling div here, not _test */
    return 0;
}

(您的函数名称不是汇编对象文件的名称 div.o - 正如注释中指出的,div是与ldivlldiv一起在stdlib.h中声明的无符号除法)

汇编函数文件中的global声明必须与您在 main 中声明的名称匹配,例如:

    global _test
_test:
    push    ebp
    mov     ebp, esp
    mov     eax, [ebp+8]
    xor     edx, edx
    div     dword [ebp+12]
    mov     esp, ebp
    pop     ebp
    ret

现在,您可以编译、链接和运行测试文件:

$ nasm -f elf -o div.o div.asm
$ gcc -m32 -c -o main.o main.c
$ gcc -m32 -o run div.o main.o
$./run
17

或者对于编译/链接,只需:

$ nasm -f elf -o div.o div.asm
$ gcc -m32 -o run main.c div.o

相关内容

  • 没有找到相关文章

最新更新