私は興味深い問題に直面しています.64ビットのマシンとOSを使用していることを忘れて、32ビットのアセンブリコードを書きました. 64ビットコードの書き方がわかりません。
これは、Linux 上の Gnu アセンブラー (AT&T 構文) 用の x86 32 ビット アセンブリ コードです。
//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 ビット OS で問題なく動作するはずですよね? ご存知のように、64 ビット プロセッサは 32 ビット プロセッサと下位互換性があります。ですから、それも問題にはなりません。この問題は、64 ビット OS と 32 ビット OS でシステム コールと呼び出しメカニズムが異なるために発生します。理由はわかりませんが、32 ビット Linux と 64 ビット Linux の間でシステム コール番号が変更されました。
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 を使用してこのプログラムをデバッグする方法を教えてください。
- なぜ彼らはシステムコール番号を変更したのですか?
- このマシンで正しく実行できるように、このプログラムを適切に変更してください。
編集:
Paul Rの答えを読んだ後。ファイルをチェックしました
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
私は、これらが ELF 32 ビットの再配置可能で実行可能であるべきだという彼の意見に同意します。しかし、それは私の質問に答えません。私の質問はすべてまだ質問です。この場合、正確には何が起こっているのでしょうか?誰かが私の質問に答えて、このコードの x86-64 バージョンを提供してもらえますか?