无法让Hello World / gas / Mac OS X正常工作

时间:2012-10-15 18:59:05

标签: macos assembly clang gas

我正在尝试在Mac OS X的GNU汇编程序中编写Hello World,但是在它打印“Hello World”之前它的总线错误。

hello.s

.global start

.data

.equ stdout, 1

.equ sys_write, 4
.equ sys_exit, 1

.equ kernel, 0x80

msg: .asciz "Hello World!\n"
.equ len, .-msg

.text

start:
    push $len
    push $msg
    push $stdout
    mov $sys_write, %eax
    sub $4, %esp
    int $kernel
    add $4 + $4 * $3, %esp

    push $0
    mov $sys_exit, %eax
    sub $4, %esp
    int $kernel

跟踪:

$ clang -c -o hello.o hello.s
$ ld -o hello -macosx_version_min 10.6 hello.o
$ ./hello
Bus error: 10

$ gdb hello
(gdb) run
Starting program: /Users/andrew/Desktop/src/mcandre/gas/hello/freebsd/hello 

Program received signal EXC_BAD_ACCESS, Could not access memory.
Reason: KERN_PROTECTION_FAILURE at address: 0x0000000000002000
0x0000000000002000 in msg ()

我的代码基于几件事:

我从NASM / Mac OS X获取了语义,并使用了Gas / Linux的语法。

我也尝试过GNU Assembler Wikipedia page上的示例程序,但它也是总线错误。

我向Apple提交了一份错误报告,因为它的GNU汇编程序已经过时了(如1.38)。他们说要改用clang。

我做错了什么?

规格:

  • ld64-134.9
  • clang 4.1
  • Xcode 4.5
  • Mac OS X 10.8.2
  • MacBook Pro 2009

1 个答案:

答案 0 :(得分:2)

在回答你做错了什么的时候,我不确定。我已经使用了汇编已经有一段时间但是clang并不喜欢我的Mac上的.equ统计数据。当我更换它们时,运行正常。 (说实话,我更喜欢NASM,虽然我使用的是macports中的那个。)

我创建了一个简单的.s文件,用clang编译,我有一个字符串长度的问题.. (也许这是一个错误?)我在$ len中存储长度,但是当我使用$ len时它不起作用..但使用硬编码的数字呢。

我用以下代码编译了代码:

clang -c -o hello.o hello.s -arch i386
ld -o hello -macosx_version_min 10.6 -arch i386 hello.o
./hello

hello.s:

/* defines */
.set sys_exit, 1
.set sys_write, 4
.set stdout, 1
.set kernel, 0x80

// TEXT SECTION
.text

/* write function */
.macro write str, strSize
    // length of string
    push \strSize
    // the string
    push \str
    // to file descriptor
    push $stdout
    // what we want to call
    movl $sys_write, %eax
    // call the kernal
    call _syscall
    // free the stack
    addl $12, %esp
.endm

/* define main function */
.globl start

/* impl. main function */
start:

    // write(string, long)
    // if I use $len i get a bunch of letters
    write $msg, $14 //$len

    // exit(0)
    // if i print out $len it is 14
    push $0 //$len
    movl $sys_exit, %eax
    call _syscall

/* call kernal 
 * 
 * so you don't have to do sub $4, %esp
 */
_syscall:
    int $kernel
    ret

// DATA SECTION
.data
    /* our string */
    msg: .string "Hello World!\n"
    // using len: .long $14 causes the same issue as . - msg
    // where the string does not end at 14
    // perhaps someone else knows why..
    len: .long . - msg

我希望这会有所帮助。