3

我一直在阅读并编写 Anthony Williams 的《Concurrency in Practice 》一书中的示例,并且需要为 gcc4.8 启用双字比较和交换,-mcx16以便可以在锁中操作包含指针 int 的结构- 自由原子方式。

Clang(任何版本)是否支持 x64 上的双字比较和交换?

以下代码在没有额外编译器选项的情况下给出了 GCC4.8 和 Clang 3.3 中的链接错误:

#include <atomic>
#include <thread>

struct ReferenceCountedPointer
{
  int referenceCount;
  void* data;
};

int main()
{
  std::atomic<ReferenceCountedPointer> arcp;
  ReferenceCountedPointer rcp;

  arcp.compare_exchange_weak(rcp, rcp);

  return 0;
}

上面的程序毫无意义,但说明了我看到的链接错误。

我用于 Clang 和 GCC 的编译命令是:

铿锵声 3.3:

clang++-mp-3.3 -std=c++11 -stdlib=libc++ CX16.cpp -o CX16

失败:

Undefined symbols for architecture x86_64:
"___atomic_compare_exchange", referenced from:
  _main in CX16-plVSvq.o
ld: symbol(s) not found for architecture x86_64

GCC4.8:

g++-mp-4.8 -std=c++11 CX16.cpp -o CX16

失败:

Undefined symbols for architecture x86_64:
"___atomic_compare_exchange_16", referenced from:
  std::atomic<ReferenceCountedPointer>::compare_exchange_weak(ReferenceCountedPointer&,        ReferenceCountedPointer, std::memory_order, std::memory_order) in ccOjp95s.o
ld: symbol(s) not found for architecture x86_64
4

2 回答 2

6

这里的问题是某些 64 位处理器型号没有cmpxchg16b. -mcx16告诉编译器“我知道这个处理器支持 cmpxchg16b 指令,所以你可以生成它” 。这是为了避免不支持此指令的较旧的 64 位处理器出现问题 - 它们会导致“非法操作码陷阱”。这与使用例如 SSE4 相同。

于 2013-08-25T19:53:42.173 回答
1

ReferenceCountedPointer是可简单复制的,因此代码在 C++11 下有效。显然该库不符合标准。

于 2013-08-25T20:07:58.373 回答