如何用'asm volatile'编写btr指令

时间:2018-08-28 23:05:07

标签: c++ assembly x86 c++14

在扩展的汇编语言(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汇编不是很熟悉。

1 个答案:

答案 0 :(得分:4)

这是一个用适当的读写操作数代替memory垃圾邮件的版本,而不是在寄存器中传递地址,并且还摆脱了setnc(需要gcc 6+)。添加了r以解决编译时未知bit的情况。将success的类型更改为更具可读性的was_setbool。注意,如果您希望这是原子的,则还需要添加一个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;
}