- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
为什么 std::atomic
's store
:
std::atomic<int> my_atomic;
my_atomic.store(1, std::memory_order_seq_cst);
xchg
吗?
_ReadWriteBarrier(); // Or `asm volatile("" ::: "memory");` for gcc/clang
my_atomic.store(1, std::memory_order_acquire);
最佳答案
mov
-store + mfence
和 xchg
都是在x86上实现顺序一致性存储的有效方法。 带有内存的lock
上的隐式xchg
前缀使其成为完整的内存屏障,就像x86上的所有原子RMW操作一样。
(x86的内存排序规则实质上使全屏障效应成为任何原子RMW的唯一选择:它同时是负载和存储,并以全局顺序固定在一起。原子性要求负载和存储不只是通过将存储区排队到存储缓冲区中来分开,因此必须将其排空,并且负载侧的负载-负载排序要求它不必重新排序。)
普通的mov
是不够的;它仅具有发布语义,而不具有顺序发布。 (与AArch64的stlr
指令不同,后者执行顺序发布存储,该顺序发布存储无法与以后的ldar
顺序获取加载重新排序。此选择显然是由C++ 11将seq_cst作为默认内存顺序来驱动的。但是,AArch64的常规存储是弱得多;放松不释放。)
请参阅Jeff Preshing's article on acquire / release semantics,并请注意,常规发行版存储(如mov
或xchg以外的任何非锁定x86内存目标指令)允许对以后的操作进行重新排序,包括获取负载(如mov或任何x86内存源操作数)。例如如果发布商店正在释放锁,则可以在关键部分内部出现以后的内容。
在不同CPU上上的mfence
和xchg
之间存在性能差异,并且可能在热缓存与冷缓存以及竞争与非竞争的情况下。和/或在同一个线程中背靠背执行多个操作的吞吐量,而不是一个线程,并且允许周围的代码与原子操作重叠执行。
请参阅https://shipilev.net/blog/2014/on-the-fence-with-dependencies,以了解mfence
与lock addl $0, -8(%rsp)
与(%rsp)
的实际基准,将其作为一个完整的障碍(如果您还没有商店的话)。
在Intel Skylake硬件上,mfence
阻止无序执行独立的ALU指令,但xchg
不会。 (See my test asm + results in the bottom of this SO answer)。英特尔的手册并不要求它具有如此强大的功能。只有lfence
可以做到这一点。但是,作为实现细节,在Skylake上乱序执行周围的代码非常昂贵。
我尚未测试其他CPU,所以可能是a microcode fix for erratum SKL079 的结果,WC存储器中的SKL079 MOVNTDQA可能通过了早期的MFENCE指令。勘误表的存在基本上证明了SKL在MFENCE之后能够执行指令。如果他们通过使MFENCE在微代码中更强来解决问题,我将不会感到惊讶,这是一种钝器手段,可显着增加对周围代码的影响。
我只测试了L1d缓存中缓存行很热的单线程情况。 (当内存变冷或在另一个内核上处于Modified状态时,则不需要。)xchg
必须加载先前的值,从而对内存中的旧值创建“false”依赖项。但是mfence
强制CPU等待,直到先前的存储提交到L1d,这也需要高速缓存行到达(并处于M状态)。因此,它们在这方面可能大致相等,但英特尔的mfence
迫使所有内容等待,而不仅仅是加载。
AMD的优化手册为原子seq-cst存储建议使用xchg
。我以为Intel建议使用较老的gcc使用的mov
+ mfence
,但是英特尔的编译器在这里也使用xchg
。
当我进行测试时,在同一位置重复进行单线程循环时,在Skyt上xchg
的吞吐量要比mov
+ mfence
更好。有关详细信息,请参见Agner Fog's microarch guide and instruction tables,但他不会在锁定操作上花费很多时间。
有关C++ 11 seq-cst my_atomic = 4;
的信息,请参见gcc/clang/ICC/MSVC output on the Godbolt compiler explorer。 gcc在SSE2可用时使用mov
+ mfence
。 (使用-m32 -mno-sse2
也可以让gcc也使用xchg
)。其他3个编译器都喜欢使用默认调整的xchg
或znver1
(Ryzen)或skylake
。
Linux内核将xchg
用作 __smp_store_mb()
。
更新:最新的GCC(如GCC10)已更改为像其他编译器一样将xchg
用于seq-cst存储,即使mfence
的SSE2可用。
另一个有趣的问题是如何编译atomic_thread_fence(mo_seq_cst);
。显而易见的选项是mfence
,但是lock or dword [rsp], 0
是另一个有效的选项(当MFENCE不可用时,由gcc -m32
使用)。堆栈的底部通常在M状态的缓存中已经很热。缺点是如果本地存储在本地,则会引入延迟。 (如果只是返回地址,则返回地址预测通常非常好,因此延迟ret
的读取能力并不成问题。)因此,在某些情况下lock or dword [rsp-4], 0
值得考虑。 (gcc did consider it,但是将其还原,因为它会使valgrind感到不高兴。这是在知道即使mfence
可用时它也可能比mfence
更好的方法。)
当前,所有编译器都将mfence
用作独立的屏障(如果可用)。这些在C++ 11代码中很少见,但是对于真正的多线程代码(在无锁通信的线程内正在进行实际工作)真正最有效的方法,还需要进行更多的研究。
,但是有多个消息来源建议使用lock add
作为堆栈的屏障,而不是mfence
,因此,即使SSE2可用,Linux内核最近也将其用于x86上的smp_mb()
实现。
有关更多讨论,请参见https://groups.google.com/d/msg/fa.linux.kernel/hNOoIZc6I9E/pVO3hB5ABAAJ,其中包括有关HSW/BDW的一些勘误表,其中涉及通过早期movntdqa
ed指令从WC内存加载lock
的问题。 (在Skylake的对面,那里是mfence
而不是lock
ed指令是一个问题。但是与SKL不同,微代码没有修复。这就是为什么Linux仍将mfence
用作其mb()
驱动程序的原因,以防万一任何使用NT负载的情况从视频RAM或其他东西复制回来,但要等到可见较早的存储后才能进行读取。)
smp_mb()
使用mb()
。如果可用,则使用mfence,否则使用lock addl $0, 0(%esp)
。__smp_store_mb
(存储+内存屏障)使用xchg
(并且在以后的内核中不会更改)。smb_mb()
使用lock; addl $0,-4(%esp)
或%rsp
,而不是mb()
。 (即使在64位中,内核也不使用红色区域,因此-4
可能有助于避免本地var的额外延迟)。mb()
来订购对MMIO区域的访问,但是smp_mb()
在为单处理器系统编译时变成无操作。更改mb()
更具风险,因为它更难测试(影响驱动程序),并且CPU具有与锁vs.mfence有关的勘误表。但是无论如何,mb()
使用mfence(如果可用),否则使用lock addl $0, -4(%esp)
。唯一的变化是-4
。#if defined(CONFIG_X86_PPRO_FENCE)
以外,没有其他更改,该my_atomic.store(1, std::memory_order_acquire);
为比现代硬件实现的x86-TSO模型更弱的内存模型定义了东西。x86 & x86_64. Where a store has an implicit acquire fence
Or
asm volatile("" ::: "memory");
my_atomic.store(1, std::memory_order_release); // mov
// with no operations in between, there's nothing for the release-store to be delayed past
std::atomic_thread_fence(std::memory_order_seq_cst); // mfence
使用发布围栏还不够强大(它和发布存储都可能会延迟到以后的加载之后,这与说发布围栏不会阻止以后的加载尽早发生是一回事)。但是,通过发布获取隔离区可以解决问题,避免以后的加载尽早发生,并且本身无法通过发布存储重新排序。
关于c++ - 为什么具有顺序一致性的std::atomic存储区使用XCHG?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49107683/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!