1

我正在尝试在 Mac OS X 的 GNU 汇编器中编写 Hello World,但在它可以打印“Hello World”之前出现总线错误。

你好.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
  • 铿锵声4.1
  • Xcode 4.5
  • Mac OS X 10.8.2
  • MacBook Pro 2009
4

1 回答 1

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

你好.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

我希望这有帮助。

于 2012-10-15T22:20:50.917 回答