- 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
。
当我进行测试时,在同一位置上反复进行单线程循环时,在Skylake上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/63724931/
这是代码片段。 请说出这种用小内存存储大数据的算法是什么。 public static void main(String[] args) { long longValue = 21474836
所以我使用 imap 从 gmail 和 outlook 接收电子邮件。 Gmail 像这样编码 =?UTF-8?B?UmU6IM69zq3OvyDOtc68zrHOuc67IG5ldyBlbWFpb
很久以前就学会了 C 代码;想用 Scheme 尝试一些新的和不同的东西。我正在尝试制作一个接受两个参数并返回两者中较大者的过程,例如 (define (larger x y) (if (> x
Azure 恢复服务保管库有两个备份配置选项 - LRS 与 GRS 这是一个有关 Azure 恢复服务保管库的问题。 当其驻留区域发生故障时,如何处理启用异地冗余的恢复服务保管库?如果未为恢复服务启
说,我有以下实体: @Entity public class A { @Id @GeneratedValue private Long id; @Embedded private
我有下一个问题。 我有下一个标准: criteria.add(Restrictions.in("entity.otherEntity", getOtherEntitiesList())); 如果我的
如果这是任何类型的重复,我会提前申请,但我找不到任何可以解决我的具体问题的内容。 这是我的程序: import java.util.Random; public class CarnivalGame{
我目前正在使用golang创建一个聚合管道,在其中使用“$ or”运算符查询文档。 结果是一堆需要分组的未分组文档,这样我就可以进入下一阶段,找到两个数据集之间的交集。 然后将其用于在单独的集合中进行
是否可以在正则表达式中创建 OR 条件。 我正在尝试查找包含此类模式的文件名列表的匹配项 第一个案例 xxxxx-hello.file 或者案例二 xxxx-hello-unasigned.file
该程序只是在用户输入行数时创建菱形的形状,因此它有 6 个 for 循环; 3 个循环创建第一个三角形,3 个循环创建另一个三角形,通过这 2 个三角形和 6 个循环,我们得到了一个菱形,这是整个程序
我有一个像这样的查询字符串 www.google.com?Department=Education & Finance&Department=Health 我有这些 li 标签,它们的查询字符串是这样
我有一个带有静态构造函数的类,我用它来读取 app.config 值。如何使用不同的配置值对类进行单元测试。我正在考虑在不同的应用程序域中运行每个测试,这样我就可以为每个测试执行静态构造函数 - 但我
我正在寻找一个可以容纳多个键的容器,如果我为其中一个键值输入保留值(例如 0),它会被视为“或”搜索。 map, int > myContainer; myContainer.insert(make_
我正在为 Web 应用程序创建数据库,并正在寻找一些建议来对可能具有多种类型的单个实体进行建模,每种类型具有不同的属性。 作为示例,假设我想为“数据源”对象创建一个关系模型。所有数据源都会有一些共享属
(1) =>CREATE TABLE T1(id BIGSERIAL PRIMARY KEY, name TEXT); CREATE TABLE (2) =>INSERT INTO T1 (name)
我不确定在使用别名时如何解决不明确的列引用。 假设有两个表,a 和 b,它们都有一个 name 列。如果我加入这两个表并为结果添加别名,我不知道如何为这两个表引用 name 列。我已经尝试了一些变体,
我的查询是: select * from table where id IN (1,5,4,3,2) 我想要的与这个顺序完全相同,不是从1...5,而是从1,5,4,3,2。我怎样才能做到这一点? 最
我正在使用 C# 代码执行动态生成的 MySQL 查询。抛出异常: CREATE TABLE dump ("@employee_OID" VARCHAR(50)); "{"You have an er
我有日期 2016-03-30T23:59:59.000000+0000。我可以知道它的格式是什么吗?因为如果我使用 yyyy-MM-dd'T'HH:mm:ss.SSS,它会抛出异常 最佳答案 Sim
我有一个示例模式,它的 SQL Fiddle 如下: http://sqlfiddle.com/#!2/6816b/2 这个 fiddle 只是根据 where 子句中的条件查询示例数据库,如下所示:
我是一名优秀的程序员,十分优秀!