汇编中的 sigprocmask returns -22

sigprocmask returns -22 in assembly

我想在汇编中使用 sigprocmask 来阻止函数中的所有信号。

以下代码适用于 C:

#include <stdio.h>
#include <signal.h>
int main() {
    sigset_t n={(unsigned long int) 0xffffffff};
    sigprocmask (SIG_BLOCK, &n, 0);

    for (int i=0; i<0x8ffff; i++) printf(".");
}

当代码执行并开始在终端上打印点时,我无法使用 Ctrl+C 中断它。到目前为止一切顺利。

SIG_BLOCK的值显然是0; sys_rt_sigprocmask 的系统调用号是 14: http://blog.rchapman.org/post/36801038863/linux-system-call-table-for-x86-64

所以我写:

[BITS 64]
    [section .text align=1]

    global main

        main:

            mov r10, 32
            mov rdx, 0
            mov rsi, newmask
            mov rdi, 0

            mov rax, 14
            syscall


            dotPrintLoop:
                mov rdi, dotstring
                mov rax, 0
                syscall
                jmp dotPrintLoop


    [section .data align=1]

        dotstring:  db ".",0

        newmask:    dd 0xffffffff
                    dd 0xffffffff
                    dd 0xffffffff
                    ...

而且它不起作用。 gdb 显示 rax 在第一次系统调用后的值为 -22 (EINVAL - "invalid parameter");而第二个系统调用(sys_write)工作正常。

我做错了什么?

显然 r10 应该保持值 8 而不是 32。

我在内核代码中遇到了 4 个不同的 sigset_t 定义;对于它们中的每一个,sizeof() 函数 returns 都有不同的结果(我记得是 32、128、4 和 8)。只有最后一个与系统调用相关。

内核首先检查$r10 == sizeof(sigset_t);和 returns EINVAL (-22) 如果不成立。对于 32 位和 64 位版本,sizeof(sigset_t) 的值都等于 8。