【发布时间】:2020-11-11 08:09:12
【问题描述】:
我在 32 位 Linux 进程中需要 futex 系统调用,但不能使用 syscall 函数(标头不可用)。这仍然可以通过使用内联 asm 来完成,如下所示:
#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;
}
按预期编译。
但是,由于我们没有指定可以读取和/或写入的内存位置,因此可能会导致一些隐秘的错误。因此,我们可以使用虚拟内存输入和输出 (How can I indicate that the memory *pointed* to by an inline ASM argument may be used?)
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 以外的任何级别)编译时,它可以正常编译。
我看到了两个明显的解决方案:
- 改用
"memory"clobber,这可能过于严格,会阻止编译器将不相关的变量保存在寄存器中 - 使用
__attribute__((optimize("-O3"))),我想避免这种情况
还有其他解决办法吗?
【问题讨论】:
-
@NateEldredge 我忘了说,第一个版本编译得很好。当我在第二个代码块中进行更改时出现问题。 godbolt.org/z/4Ko1eY
-
快速说明:我认为
futex调用可以写出futex字,那你是不是也需要*uaddr作为输出操作数? -
@NateEldredge AFAIK 只有
FUTEX_WAKE_OP可以做到这一点,但它会写入*uaddr2,它已经被列为输出。我再看一下手册页。 -
我不确定,但似乎值得检查。
-
@NateEldredge 似乎只有
FUTEX_WAKE_OP和 PI-futex 操作修改*uaddr2。我不打算使用这些,但无论如何我都可以将其添加为输出,以防万一。
标签: c linux x86 inline-assembly cpu-registers