У меня интересная проблема. Я забыл, что использую 64-битную машину и ОС, и написал 32-битный ассемблерный код. Я не знаю, как написать 64-битный код.
Это 32-битный код сборки x86 для Gnu Assembler (синтаксис AT & T) в Linux.
//hello.S
#include <asm/unistd.h>
#include <syscall.h>
#define STDOUT 1
.data
hellostr:
.ascii "hello wolrd\n";
helloend:
.text
.globl _start
_start:
movl $(SYS_write) , %eax //ssize_t write(int fd, const void *buf, size_t count);
movl $(STDOUT) , %ebx
movl $hellostr , %ecx
movl $(helloend-hellostr) , %edx
int $0x80
movl $(SYS_exit), %eax //void _exit(int status);
xorl %ebx, %ebx
int $0x80
ret
Теперь этот код должен нормально работать на 32-битном процессоре и 32-битной ОС, верно? Как мы знаем, 64-битные процессоры обратно совместимы с 32-битными процессорами. Так что это тоже не будет проблемой. Проблема возникает из-за различий в системных вызовах и механизме вызовов в 64-битной и 32-битной ОС. Я не знаю почему, но они изменили номера системных вызовов между 32-битным и 64-битным.
asm / unistd_32.h определяет:
#define __NR_write 4
#define __NR_exit 1
asm / unistd_64.h определяет:
#define __NR_write 1
#define __NR_exit 60
В любом случае использование макросов вместо прямых чисел окупается. Обеспечение правильных номеров системных вызовов.
когда я собираю, связываю и запускаю программу.
$cpp hello.S hello.s //pre-processor
$as hello.s -o hello.o //assemble
$ld hello.o // linker : converting relocatable to executable
Это не печать helloworld
.
В GDB его показ:
- Программа вышла с кодом 01.
Я не знаю, как отлаживать в GDB. используя учебник, я попытался отладить его и выполнить инструкцию, проверяя регистры команд на каждом шаге. это всегда показывает мне "программа вышла с 01". Было бы здорово, если бы кто-нибудь мог показать мне, как это отладить.
(gdb) break _start
Note: breakpoint -10 also set at pc 0x4000b0.
Breakpoint 8 at 0x4000b0
(gdb) start
Function "main" not defined.
Make breakpoint pending on future shared library load? (y or [n]) y
Temporary breakpoint 9 (main) pending.
Starting program: /home/claws/helloworld
Program exited with code 01.
(gdb) info breakpoints
Num Type Disp Enb Address What
8 breakpoint keep y 0x00000000004000b0 <_start>
9 breakpoint del y <PENDING> main
Я пытался запустить strace
. Это его вывод:
execve("./helloworld", ["./helloworld"], [/* 39 vars */]) = 0
write(0, NULL, 12 <unfinished ... exit status 1>
- Объясните параметры
write(0, NULL, 12)
системного вызова на выходе strace?
- Что точно происходит? Я хочу знать причину, по которой точно завершается с exitstatus = 1?
- Может кто-нибудь показать мне, как отлаживать эту программу с помощью GDB?
- Почему они изменили номера системных вызовов?
- Пожалуйста, измените эту программу соответствующим образом, чтобы она могла правильно работать на этом компьютере.
EDIT:
Прочитав ответ Пола Р. Я проверил мои файлы
claws@claws-desktop:~$ file ./hello.o
./hello.o: ELF 64-bit LSB relocatable, x86-64, version 1 (SYSV), not stripped
claws@claws-desktop:~$ file ./hello
./hello: ELF 64-bit LSB executable, x86-64, version 1 (SYSV), statically linked, not stripped
Я согласен с ним, что это должен быть 32-битный ELF-файл, который можно перемещать и выполнять. Но это не отвечает на мои мои вопросы. Все мои вопросы еще вопросы. Что именно происходит в этом случае? Может кто-нибудь ответить на мои вопросы и предоставить версию этого кода для x86-64?