- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
std::atomic_uint64_t writing_ {0};
std::atomic_uint64_t reading_ {0};
std::array<type, size> storage_ {};
bool try_enqueue(type t) noexcept
{
const std::uint64_t writing {
writing_.load(std::memory_order::memory_order_relaxed)};
const auto last_read {reading_.load(std::memory_order::memory_order_relaxed)};
if (writing - last_read < size) {
storage_.at(writing & (size - 1)) = t;
writing_.store(writing + 1, std::memory_order::memory_order_release);
return true;
}
else
return false;
}
在上面的代码中,据我了解,如果条件评估为假,则任何线程都不可能观察到对共享存储的写入。一个操作不能被认为是在它排序之后的条件之前发生的,这是否正确?还是我完全误读了这一点,这样的事情实际上可能会发生(可能通过推测执行?)?
更具体一点,处理器是否可以推测性地执行写入(当条件最终评估为 false 时),另一个线程观察到写入已经发生,然后然后第一个线程丢弃推测性的写?
(注意:这是单生产者单消费者)
最佳答案
Is it correct that an operation cannot be perceived as having occurred before a conditional it is sequenced after?
C++ 编译器绝对不允许发明写入atomic
(或volatile
)对象。
编译器甚至不允许发明对非原子对象的写入(例如,将条件写入转换为读取 + cmov + 写入),因为 C++11 引入了一种内存模型,该模型明确定义了两个线程运行同时编写这样的代码,只要其中最多一个 实际上 写入(然后读取顺序)。但是两个非原子 RMW 可能会相互踩踏,所以不要“好像”C++ 抽象机正在运行源代码一样工作,因此编译器发出执行此操作的 asm 不是一个选项。
但如果编译器知道一个对象总是被写入,它几乎可以做任何它想做的事,因为合法的程序无法观察到差异:这将涉及数据争用 UB。
A little more specifically, could the processor speculatively execute the write (when the condition will eventually evaluate to false), another thread observe the write as having occurred, and then the first thread discarding the speculative write?
不,推测无法逃脱进行推测的核心。否则,当检测到错误推测时,所有核心都必须回滚其状态!
这是存在存储缓冲区的主要原因之一:将存储的 OoO 推测执行从提交解耦到 L1d 缓存(这是存储变得全局可见时到其他核心)。并将执行与缓存未命中存储分离,这甚至在有序的非推测 CPU 上也很有用。
直到存储指令从无序核心退出(即已知是非推测性的)之后,存储才提交给 L1d。尚未提交的退役存储有时被称为“已毕业”,以将它们与其他存储缓冲区条目区分开来,如果核心需要回滚到退役状态,这些条目可能会被丢弃。
这允许硬件推测执行而无需发明写入。
(另请参阅 Can a speculatively executed CPU branch contain opcodes that access RAM? 以获取更多详细信息。有趣的事实:一些 CPU,特别是 PowerPC,可以在同一物理内核上的 SMT 线程之间执行分级存储的存储转发,使存储在某些内核成为 globally 可见。但仅限于分级商店,否则可能的错误猜测可能会泄露。)
在 C++ 中,std::mo_release
存储强制编译器使用足够的屏障或发布存储指令(例如,x86 上的普通 mov
是发布存储,或 AArch64 上的 STLr
是顺序发布存储)。或者任何其他机制来确保 asm 保证运行时排序至少与 C++ 抽象机保证一样强。
C++ 根据前序/后序而非障碍来定义其标准,但在任何给定的平台实现上/ABI 对从 std::atomic 操作到 asm 序列的某些映射进行了标准化。 (例如 https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html)
关于c++ - 加载或存储可以在有条件之前重新排序吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57937532/
这个问题在这里已经有了答案: How does Scala's apply() method magic work? (3 个回答) 9年前关闭。 假设我在 scala 中有一个 MyList 类,其
这个问题在这里已经有了答案: What is a non-capturing group in regular expressions? (18 个回答) Reference - What does
这个问题是针对嵌入式系统的! 我有以下选项来初始化一个对象: Object* o = new Object(arg); 这会将对象放入堆中并返回指向它的指针。我不喜欢在嵌入式软件中使用动态分配。 Ob
我自己搜索过,没能成功的正则表达式。 我有一个 html 文件,其中包含 [] 之间的变量我想把每一个字都写进去。 [client_name][client_company] [cl
我是 Python 新手。我不明白为什么这段代码不起作用: reOptions = re.search( "[\s+@twitter\s+(?P\w+):(?P.*?)\s+]", d
在过去 7 个月左右的时间里,我几乎一直在使用 .NET C# 进行编程。在那之前,我的大部分编程都是用 C++(从学校里学的)。在工作中,我可能需要在接下来的几个月里做一大堆 C 语言。我对 C 的
我是 RE 的新手,我正在尝试获取歌词并分离出歌词标题、和声和主唱: 下面是一些歌词的例子: [Intro] D.A. got that dope! [Chorus: Travis Scott] Ic
这可能是不可能的,但我想检查是否可以用一种简单的方式表达这样的事情: // obviously doesn't work class Foo : IFoo where T: Bar {
我们的应用程序中有“user”和“study”实体,存储在它们各自的表中。一项研究代表一种研究和已收集的数据。它们是多对多的关系,所以我们需要一个链接表:studies_users。 我们为用户分配角
将测试条件添加到 Visual Studio 2010 数据库单元测试(对于 SQL Server 2008)时,这些条件称为例如rowCountCondition1、rowCountConditio
在模拟器上,我可以从设置中卸载 SD 卡。 然后我可以将它安装到我的操作系统上,然后正常卸载它。 我一直无法弄清楚如何在模拟器上重新安装它(无需重新启动)。 提示: adb 命令 remount 是无
假设在一个分支上执行了一系列提交,但该分支尚未与主干重新同步。是否可以从提交中生成全局补丁?是否可以从一系列提交中生成“分组”补丁?如果是,如何? 最佳答案 svn diff -rXXX:YYY UR
在某些情况下,我想在我的应用程序中锁定调整大小功能,为此我尝试对属性进行数据绑定(bind),并且不允许在某些情况下更改它,但没有成功。 有没有办法这样做? 这是我不成功的尝试: XAML: Vie
当我的计算机连接多个显示器时,我可以检测它们,并根据从获取的值设置位置来向它们绘制图形 get(0, 'MonitorPositions') 但是,当我在 MATLAB 运行时断开监视器时,此属性不会
我们有一个grails应用程序,该应用程序在grails数据库中存储了各种域对象。该应用程序连接到第二个数据库,运行一些原始sql,并在表中显示结果。它基本上是一个报告服务器。 我们通过在DataSo
无法比较来自不同容器的迭代器(参见这里的示例: https://stackoverflow.com/a/4664519/225186 )(或者从技术上讲,它不需要有意义。) 这就提出了另一个问题,来自
我有以下情况: 家长 Activity : ParentActivityClass { private Intent intent; @Override public void onCreate(Bu
我经常将元素与附加功能 Hook ,例如: $('.myfav').autocomplete(); $('.myfav').datepicker(); $('.myfav').click(somefu
因此,我将 tooltipster.js 库用于工具提示,并尝试更改工具提示在不同屏幕尺寸上的默认距离。 所以这是默认的 init 的样子: $(inputTooltipTrigger).tool
我在 ARM7 嵌入式环境中工作。我使用的编译器不支持完整的 C++ 功能。它不支持的一项功能是动态类型转换。 有没有办法实现dynamic_cast<>() ? 我使用 Google 寻找代码,但到
我是一名优秀的程序员,十分优秀!