NASM Hello World 在 Mac OS X 中出现段错误或总线错误



我正在用 NASM 编写 Hello World,我可以让它Hello World回显到控制台,但如果我不使用 Make 运行它,程序就会出现段错误。

使用生成文件进行跟踪:

$ make
nasm -f macho -o hello.o --prefix _ hello.asm
ld -o hello hello.o -arch i386 -lc -macosx_version_min 10.6 -e _start -no_pie
./hello
Hello World!

使用手动命令进行跟踪:

$ nasm -f macho -o hello.o --prefix _ hello.asm
$ ld -o hello hello.o -arch i386 -lc -macosx_version_min 10.6 -e _start -no_pie
$ ./hello 
Segmentation fault: 11

你好.asm:

[bits 32]
section .data
msg: db "Hello World!", 0
section .text
global start
extern puts
extern exit
start:
push msg
call puts
add esp, 4
push 0
call exit

制作文件:

# Linux defaults
FORMAT=-f elf
MINV=
ARCH=-arch i386
LIBS=
RUN=./
EXECUTABLE=hello
PREFIX=
ENTRY=
PIE=
# Windows
ifeq (${MSYSTEM},MINGW32)
    FORMAT=-f win32
    EXECUTABLE=hello.exe
    PREFIX=--prefix _
    ENTRY=-e _start
    ARCH=
    LIBS=c:/strawberry/c/i686-w64-mingw32/lib/crt2.o -Lc:/strawberry/c/i686-w64-mingw32/lib -lmingw32 -lmingwex -lmsvcrt -lkernel32
    ENTRY=
    RUN=
endif
# Mac OS X
ifeq ($(shell uname -s),Darwin)
    FORMAT=-f macho
    PREFIX=--prefix _
    ENTRY=-e _start
    LIBS=-lc
    MINV=-macosx_version_min 10.6
    PIE=-no_pie
endif
all: test
test: $(EXECUTABLE)
    $(RUN)$(EXECUTABLE)
$(EXECUTABLE): hello.o
    ld -o $(EXECUTABLE) hello.o $(ARCH) $(LIBS) $(MINV) $(ENTRY) $(PIE)
hello.o: hello.asm
    nasm $(FORMAT) -o hello.o $(PREFIX) hello.asm
clean:
    -rm $(EXECUTABLE)
    -rm hello.o

规格:

  • LD 64-134.9
  • LLVM 3.1svn
  • 纳斯姆 0.98.40
  • 制作 3.81
  • Xcode 4.5
  • Mac OS X 10.8.1
  • MacBook Pro 2009

2件事,你的hello world字符串不是NULL 终止的,正如我在另一篇文章中提到的,当你使用 C 函数时,你必须在每次调用后调整 esp

你撕掉了两次堆栈帧:

mov esp, ebp
pop ebp
...
leave

您只需要其中之一,因为leave等同于mov esp, ebp; pop ebp .

有关几个示例 hello world 程序,请参阅 http://michaux.ca/articles/assembly-hello-world-for-os-x。请注意,它们都显式退出程序

; 2a prepare the argument for the sys call to exit
push dword 0              ; exit status returned to the operating system
; 2b make the call to sys call to exit
mov eax, 0x1              ; system call number for exit
sub esp, 4                ; OS X (and BSD) system calls needs "extra space" on stack
int 0x80                  ; make the system call

因为您无法从入口点ret(没有什么可返回的)。

另请注意,如果调用函数main并且没有向ld提供e选项,则将调用libc 的入口点。在这种情况下,允许ret,因为您将控制权归还给libc(代表您调用exit)。

相关内容

  • 没有找到相关文章

最新更新