我需要在 32 位 Linux 进程中使用 futex 系统调用,但无法使用syscall
函数(标头不可用)。这仍然可以通过使用内联汇编来完成,如下所示:
#include <time.h>
#define SYS_futex 0xf0
// We need -fomit-frame-pointer in order to set EBP
__attribute__((optimize("-fomit-frame-pointer")))
int futex(int* uaddr, int futex_op, int val, const struct timespec* timeout, int* uaddr2, int val3)
{
register int ebp asm ("ebp") = val3;
int result;
asm volatile("int $0x80"
: "=a"(result)
: "a"(SYS_futex), "b"(uaddr), "c"(futex_op), "d"(val), "S"(timeout), "D"(uaddr2), "r"(ebp)
// : "memory" // would make this safe, but could cause some unnecessary spills. THIS VERSION IS UNSAFE ON PURPOSE, DO NOT USE.
);
if (result < 0)
{
// Error handling
return -1;
}
return result;
}
编译成功,如预期的那样。
但是,由于我们没有指定可以读取和/或写入的内存位置,因此可能会导致一些偷偷摸摸的错误。因此,我们可以使用虚拟内存输入和输出(如何指示可以使用内联 ASM 参数*指向*的内存? https://stackoverflow.com/questions/56432259/how-can-i-indicate-that-the-memory-pointed-to-by-an-inline-asm-argument-may-be)
asm volatile("int $0x80"
: "=a"(result), "+m"(uaddr2)
: "a"(SYS_futex), "b"(uaddr), "c"(futex_op), "d"(val), "S"(timeout), "D"(uaddr2), "r"(ebp), "m"(*uaddr), "m"(*timeout));
当编译时gcc -m32
,它失败了'asm' operand has impossible constraints
。当编译时clang -fomit-frame-pointer -m32
,它失败了inline assembly requires more registers than available
。但我不明白为什么。
但是,当编译时-O1 -m32
(或除-O0
),编译得很好。
我看到两个明显的解决方案:
- Use the
"memory"
相反,这可能限制太多,阻止编译器将不相关的变量保留在寄存器中
- Use
__attribute__((optimize("-O3")))
,我想避免
还有其他解决办法吗?