我正在尝试学习x86汇编程序,我想用C调用NASM函数。当我运行我的程序时,我收到此错误:
分段错误(核心转储)
我已经尝试了数十种简单的测试函数变体,但它每次都在同一位置停止。
以下是我的asm
和c
文件:
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
是与ldiv
和lldiv
一起在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