write
syscallには、次の機能プロトタイプがあります。
size_t write(int, const void *buf, size_t nbytes);
write
MacOS XのGCCでインラインアセンブラを使用してsyscallを呼び出すにはどうすればよいですか?
write
syscallには、次の機能プロトタイプがあります。
size_t write(int, const void *buf, size_t nbytes);
write
MacOS XのGCCでインラインアセンブラを使用してsyscallを呼び出すにはどうすればよいですか?
このタイプの質問に対する一般的な解決策:関心のあるwrite()呼び出しを実行する短いテストプログラムを作成し、gcc -S
アセンブリを生成するために使用otool
するか、バイナリを逆アセンブルするために使用します。write()呼び出しがどのようにアセンブルされたかを調べ、それを適切なインラインアセンブリに変換します。
(編集)実際のシステムコールを確認するには、アセンブリコードのライブラリコールに従います。の例では、インwrite()
ダイレクションに従うと、を介しlibSystem.B.dylib
て_write
inlibsystem_kernel.dylib
に移動します。これは、を使用して分解できますotool
。
(edit2)このテストプログラムを使用した以下の完全な例test.c
:
#include <stdio.h>
int main(void)
{
char buf[] = "test\n";
ssize_t n;
n = write(2, buf, sizeof(buf));
return n;
}
コンパイルしてテストし、-O1 -fverbose-asm
簡潔で読みやすいマシンコードを取得します。
% gcc -O1 -fverbose-asm -o test test.c
% ./test
test
バイナリでotool
分解するために使用しmain()
ます。test
% otool -p _main -tvV test
test:
(__TEXT,__text) section
_main:
0000000100000ef0 pushq %rbp
0000000100000ef1 movq %rsp,%rbp
0000000100000ef4 subq $0x10,%rsp
0000000100000ef8 leaq 0xfa(%rbp),%rsi
0000000100000efc movb $0x74,0xfa(%rbp)
0000000100000f00 movb $0x65,0xfb(%rbp)
0000000100000f04 movb $0x73,0xfc(%rbp)
0000000100000f08 movb $0x74,0xfd(%rbp)
0000000100000f0c movb $0x0a,0xfe(%rbp)
0000000100000f10 movb $0x00,0xff(%rbp)
0000000100000f14 movl $0x00000002,%edi
0000000100000f19 movl $0x00000006,%edx
0000000100000f1e xorb %al,%al
0000000100000f20 callq 0x100000f32 ; symbol stub for: _write
0000000100000f25 addq $0x10,%rsp
0000000100000f29 popq %rbp
0000000100000f2a ret
test
以下を見つけるために動的にリンクされているライブラリを一覧表示します_write
。
% otool -L test
test:
/usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 159.1.0)
で見つけてみて_write
くださいlibSystem.B.dylib
:
% otool -p _write -tvV /usr/lib/libSystem.B.dylib
/usr/lib/libSystem.B.dylib:
(__TEXT,__text) section
Can't find -p symbol: _write
の依存関係を確認してくださいlibSystem.B.dylib
:
% otool -L /usr/lib/libSystem.B.dylib
/usr/lib/libSystem.B.dylib:
/usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 159.1.0)
/usr/lib/system/libcache.dylib (compatibility version 1.0.0, current version 47.0.0)
/usr/lib/system/libcommonCrypto.dylib (compatibility version 1.0.0, current version 55010.0.0)
/usr/lib/system/libcompiler_rt.dylib (compatibility version 1.0.0, current version 6.0.0)
/usr/lib/system/libcopyfile.dylib (compatibility version 1.0.0, current version 85.1.0)
/usr/lib/system/libdispatch.dylib (compatibility version 1.0.0, current version 187.9.0)
/usr/lib/system/libdnsinfo.dylib (compatibility version 1.0.0, current version 395.11.0)
/usr/lib/system/libdyld.dylib (compatibility version 1.0.0, current version 195.6.0)
/usr/lib/system/libkeymgr.dylib (compatibility version 1.0.0, current version 23.0.0)
/usr/lib/system/liblaunch.dylib (compatibility version 1.0.0, current version 392.38.0)
/usr/lib/system/libmacho.dylib (compatibility version 1.0.0, current version 800.0.0)
/usr/lib/system/libmathCommon.A.dylib (compatibility version 1.0.0, current version 2026.0.0)
/usr/lib/system/libquarantine.dylib (compatibility version 1.0.0, current version 36.6.0)
/usr/lib/system/libremovefile.dylib (compatibility version 1.0.0, current version 21.1.0)
/usr/lib/system/libsystem_blocks.dylib (compatibility version 1.0.0, current version 53.0.0)
/usr/lib/system/libsystem_c.dylib (compatibility version 1.0.0, current version 763.13.0)
/usr/lib/system/libsystem_dnssd.dylib (compatibility version 1.0.0, current version 1.0.0)
/usr/lib/system/libsystem_info.dylib (compatibility version 1.0.0, current version 1.0.0)
/usr/lib/system/libsystem_kernel.dylib (compatibility version 1.0.0, current version 1699.26.8)
/usr/lib/system/libsystem_network.dylib (compatibility version 1.0.0, current version 1.0.0)
/usr/lib/system/libsystem_notify.dylib (compatibility version 1.0.0, current version 80.1.0)
/usr/lib/system/libsystem_sandbox.dylib (compatibility version 1.0.0, current version 1.0.0)
/usr/lib/system/libunc.dylib (compatibility version 1.0.0, current version 24.0.0)
/usr/lib/system/libunwind.dylib (compatibility version 1.0.0, current version 30.0.0)
/usr/lib/system/libxpc.dylib (compatibility version 1.0.0, current version 77.19.0)
_write
おそらく含まれていると思いますlibsystem_kernel.dylib
(または代わりにそれらすべてを試してください):
% otool -p _write -tvV /usr/lib/system/libsystem_kernel.dylib | head -20
(__TEXT,__text) section
_write:
0000000000017fd4 movl $0x02000004,%eax
0000000000017fd9 movq %rcx,%r10
0000000000017fdc syscall
0000000000017fde jae 0x00017fe5
0000000000017fe0 jmp cerror
0000000000017fe5 ret
0000000000017fe6 nop
0000000000017fe7 nop
[...]
これで、インラインアセンブリバージョンを構築するために必要なすべてのアセンブリができましたtest
。
#include <stdio.h>
int main(void)
{
char buf[] = "test\n";
ssize_t n;
asm volatile (
"movl $0x00000002, %%edi\n" /* first argument */
"movl $0x00000006, %%edx\n" /* third argument */
"movl $0x02000004, %%eax\n" /* syscall number */
"syscall\n"
: "=A"(n) /* %rax: return value */
: "S"(buf)); /* %rsi: second argument */
return n;
}
コンパイルとテスト:
% gcc -O1 -fverbose-asm -o test-asm test-asm.c
% ./test-asm
test
それはうまくいくようです。上記のインラインアセンブリはあまり洗練されていません。たとえば、アセンブリコードでハードコーディングする代わりに、最初と3番目の引数を動的に渡すこともできます。
Linuxソリューションは次のようなものです。
.data mytext:
.ascii "Hello World\n"
.text
_mywrite:
movl $0x04, %eax # Syscall No. 4 = write
movl $0x01, %ebx # File. 1 = stdout
movl $mytext, %ecx
movl $0x0c, %edx # text length (hope I counted correctly)
int $0x80 # Interrupt 0x80 -> make syscall
Macなどの他のUnixシステムでも同様であるはずですが、ドキュメントのシステムコール番号(0x04)を確認することをお勧めします。