2

NASM で Hello World を作成してHello Worldいます。コンソールにエコーすることはできますが、Make で実行しないと、プログラムがセグメンテーション違反を起こします。

Makefile でトレースします。

$ 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
  • NASM 0.98.40
  • 3.81を作る
  • Xcode 4.5
  • マック OS X 10.8.1
  • MacBook Pro 2009
4

2 に答える 2

3

2 つのこと、あなたの hello world 文字列は NULL で終了していません。別の投稿で述べたように、C 関数を使用する場合は、呼び出しのたびに esp を調整する必要があります。

于 2012-09-30T01:16:01.573 に答える
2

スタック フレームを 2 回破棄しました。

mov esp, ebp
pop ebp
...
leave

leaveは と同等であるため、これらのうちの 1 つだけが必要です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しない場合、のエントリ ポイントが呼び出されることにも注意してください。その場合、制御を(あなたに代わって呼び出す) に戻すので、許容されます。eldlibcretlibcexit

于 2012-09-30T01:14:08.720 に答える