2012-09-30 36 views
2

我在NASM中編寫Hello World,並且我可以將它回顯給控制檯Hello World,但是如果我不使用Make運行它,程序會發生段錯誤。NASM Hello World在Mac OS X中發生段錯誤或總線錯誤

跟蹤與生成文件:

$ 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 

hello.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
  • NASM 0.98.40
  • 讓3.81
  • 的Xcode 4.5
  • 的Mac OS X 10.8.1
  • 的MacBook Pro 2009
+0

更新:我被利用擺脫總線錯誤的libc的'exit(0)'。然而,除非它由Make運行,否則該程序仍然是段錯誤。 – mcandre

+0

這很奇怪,我關閉了Terminal.app,並嘗試再次構建hello.asm。現在,當我手動運行它時不再發生段錯誤。 – mcandre

回答

2

你扯下你的棧幀兩次:

mov esp, ebp 
pop ebp 
... 
leave 

你只需要其中的一個,因爲leave相當於mov esp, ebp; pop ebp

請參閱http://michaux.ca/articles/assembly-hello-world-for-os-x瞭解幾個示例hello world程序。需要注意的是所有的人都用

; 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不提供e選項ld,然後libc的入口點會被調用。在這種情況下,由於您將控制返回到libc(代表您呼叫exit),因此可以使用ret

+0

你能提供一個完整的hello.asm,它不會在Mac OS X上發生段錯誤或總線錯誤嗎? – mcandre

+0

好的,看編輯(我用這個信息添加了一個鏈接到一個網站) – nneonneo

+0

我用libc的'exit(0)'解決了總線錯誤,但是我仍然得到一段錯誤,除非我從Make運行程序。在原文中查看我的更新代碼。 – mcandre

3

兩件事情,你的hello world的字符串不是空終止,正如我在另一篇文章中所提到的,當你使用C函數,你必須ESP每次通話後調整

+0

他不需要在這裏調整'esp',因爲他在之後破壞了堆棧框架。 – nneonneo

+1

他正在學習大會,在開始時學會正確的方式,無論多麼微不足道,它都會在以後挽救頭痛。 – Gunner

+0

@Gunner,upvote,但請提供改進的代碼。當我終止Hello World字符串並在'puts'調用後調整esp時,我的程序仍然存在段錯誤。 – mcandre