assembly - Macos 64 位程序集 - fork() test/jne 不起作用?

标签 assembly fork x86-64 system-calls shellcode

我正在尝试编写一个程序来执行 fork() 系统调用,子/父每个写入不同的字符串和 exit(0)

问题是 - 虽然 fork() 是成功的(因为我看到了 2 行输出),但出于某种原因,父进程和子进程都输出了父字符串。

代码如下:

BITS 64

section .data
msg1:    db    "Hello from child", 0x0a
len1:    equ   $-msg1
msg2:    db    "Hello from parent", 0x0a
len2:    equ   $-msg2

section .text
global start

start:
        xor rax,rax
        mov rax,0x2000002    ; fork() syscall
        syscall

        test eax,eax
        jne flow_parent

        mov rax, 0x2000004   ; SYS_write
        mov rdi, 1           ; stdout
        mov rsi, msg1
        mov rdx, len1
        syscall

        xor rdi,rdi
        mov rax,0x2000001
        syscall

flow_parent:
        mov rax, 0x2000004   ; SYS_write
        mov rdi, 1           ; stdout
        mov rsi, msg2
        mov rdx, len2
        syscall

        xor rdi,rdi
        mov rax,0x2000001
        syscall

我的输出:

$ nasm -f macho64 code.s -o code.o
$ ld -o code -e start -macosx_version_min 10.7 code.o
$ ./code
Hello from parent
Hello from parent
$

关于如何解决这个问题有什么想法吗?

最佳答案

首先,您在 Mac OS X 上进行系统调用时假设它的行为就像 Linux 内核一样,这显然是错误的(Linux fork syscall convention != XNU fork syscall convention)。 Mac OS X 不支持用户不通过 libSystem 库直接进行系统调用。

现在撇开这个不谈,我不确定你运行的是哪个版本的 XNU 内核,但如果你真的看一下 fork function 的源代码在 libsyscall 库中,您会发现使用 edx 寄存器代替 (orl %edx, %edx) 来确定进程是子进程还是父进程。

同样,这仍然不是实现它的可靠方式,因为 Apple 将来可能会随意更改界面。

关于assembly - Macos 64 位程序集 - fork() test/jne 不起作用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19940473/

相关文章:

assembly - 如何在 aarch64 程序集列表中轻松导航

c++ - C/С++。为什么 volatile 上的简单整数加法可以转换为 gcc 和 clang 上的不同 asm 指令?

在 fork 之后创建 pthread

performance - 为什么 32 字节的循环对齐会使代码更快?

linux - linux是如何保护内存的?

c++ - 为什么内联汇编程序在此代码中有时更快而有时更慢?每次运行的执行时间变化很大

perl - 在 Perl 中执行进程,使用 IPC 停止、恢复、终止选项

c - 为什么我的流程会这样?

assembly - REP INSW 做什么?

c - 在这种情况下我需要同步吗?