在扩展的汇编语言(asm volatile
)中,使用btr在C ++中使用btr重置整数的正确实现是什么?我需要在重置之前返回该位的值。
这是我的实现,对16位整数正确吗?
std::uint16_t reset(std::uint16_t& integer, std::uint32_t bit) {
auto success = false;
asm volatile("lock btrw %1, (%2); setnc %0"
: "=r"(success)
: "i"(bit), "r"(&integer)
: "memory", "flags");
return !success;
}
此实现正确吗?我错过了任何细节吗?我对asm()
语法或x86汇编不是很熟悉。
答案 0 :(得分:4)
这是一个用适当的读写操作数代替memory
垃圾邮件的版本,而不是在寄存器中传递地址,并且还摆脱了setnc
(需要gcc 6+)。添加了r
以解决编译时未知bit
的情况。将success
的类型更改为更具可读性的was_set
,bool
。注意,如果您希望这是原子的,则还需要添加一个lock
前缀。对于编译器内存障碍,您可能需要放回memory
约束。
bool reset(std::uint16_t& integer, std::uint32_t bit) {
bool was_set;
asm volatile("btrw %w2, %1"
: "=@ccc"(was_set), "+mr"(integer)
: "ri"(bit)
: "cc");
return was_set;
}
原子版本看起来像:
bool reset(std::uint16_t& integer, std::uint32_t bit) {
bool was_set;
asm volatile("lock btrw %w2, %1"
: "=@ccc"(was_set), "+m"(integer)
: "ri"(bit)
: "cc", "memory");
return was_set;
}