免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2014 | 回复: 7
打印 上一主题 下一主题

[C] 如何理解 volatile [复制链接]

论坛徽章:
11
未羊
日期:2013-12-16 12:45:4615-16赛季CBA联赛之青岛
日期:2016-04-11 19:17:4715-16赛季CBA联赛之广夏
日期:2016-04-06 16:34:012015亚冠之卡尔希纳萨夫
日期:2015-11-10 10:04:522015亚冠之大阪钢巴
日期:2015-07-30 18:29:402015亚冠之城南
日期:2015-06-15 17:56:392015亚冠之卡尔希纳萨夫
日期:2015-05-15 15:19:272015亚冠之山东鲁能
日期:2015-05-14 12:38:13金牛座
日期:2014-12-04 15:34:06子鼠
日期:2014-10-16 13:40:4715-16赛季CBA联赛之八一
日期:2016-07-22 09:41:40
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-01-28 18:29 |只看该作者 |倒序浏览

如下
#define ACCESS_ONCE(x) (*(volatile typeof(x) *)&(x))

如果说这样定义的目的是强迫 x 从内存里面取值, 而忽略已经缓存的值的话, 那么如果在MESI 角度, 它本身保有的值就是最新值, 而且和内存中的不一致呢? 或者其他 CPU 中保有最新值, 但不在内存中呢?
如果说如果自己保有的是最新值, 则不从内存查的话, 那么为什么不这样定义呢:
#define ACCESS_ONCE(x) (*(typeof(x) *)&(x))

这样同样是产生出x地址, 并尝试 read 这个地址, 难道这样在 MESI 角度就不能取出最新值吗?
如果考虑到 invalidate request 在 queue 中的延迟, 那么使用 rmb 可以达到同样效果吗? 莫非这样定义的目的, 就是为了少用一个rmb? 但是, 他怎么就能替代  rmb 呢, 而且 volatile 是 C 层面的东西吧, CPU 根本见不到

那么, 其实是为了在寄存器层次防止直接从寄存器而不是缓存层次取值, 那么又有一个问题了, 难道
#define ACCESS_ONCE(x) (*(typeof(x) *)&(x))
就不能达到这个目的, 莫非 CPU 还记录者哪个寄存器中的值是之前从哪个地址中读出来的, 这个很显然也不太可能

论坛徽章:
59
2015年亚洲杯之约旦
日期:2015-01-27 21:27:392015年亚洲杯之日本
日期:2015-02-06 22:09:41拜羊年徽章
日期:2015-03-03 16:15:432015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:50:282015元宵节徽章
日期:2015-03-06 15:50:392015年亚洲杯之阿联酋
日期:2015-03-19 17:39:302015年亚洲杯之中国
日期:2015-03-23 18:52:23巳蛇
日期:2014-12-14 22:44:03双子座
日期:2014-12-10 21:39:16处女座
日期:2014-12-02 08:03:17天蝎座
日期:2014-07-21 19:08:47
2 [报告]
发表于 2013-01-28 18:35 |只看该作者
volatile只管到Register
管不了Cache。Over

论坛徽章:
11
未羊
日期:2013-12-16 12:45:4615-16赛季CBA联赛之青岛
日期:2016-04-11 19:17:4715-16赛季CBA联赛之广夏
日期:2016-04-06 16:34:012015亚冠之卡尔希纳萨夫
日期:2015-11-10 10:04:522015亚冠之大阪钢巴
日期:2015-07-30 18:29:402015亚冠之城南
日期:2015-06-15 17:56:392015亚冠之卡尔希纳萨夫
日期:2015-05-15 15:19:272015亚冠之山东鲁能
日期:2015-05-14 12:38:13金牛座
日期:2014-12-04 15:34:06子鼠
日期:2014-10-16 13:40:4715-16赛季CBA联赛之八一
日期:2016-07-22 09:41:40
3 [报告]
发表于 2013-01-28 18:47 |只看该作者
已经半明白了, 妈的就在定义的头顶上就有注释:
303/*
304 * Prevent the compiler from merging or refetching accesses.  The compiler
305 * is also forbidden from reordering successive instances of ACCESS_ONCE(),
306 * but only when the compiler is aware of some particular ordering.  One way
307 * to make the compiler aware of ordering is to put the two invocations of
308 * ACCESS_ONCE() in different C statements.
309 *
310 * This macro does absolutely -nothing- to prevent the CPU from reordering,
311 * merging, or refetching absolutely anything at any time.  Its main intended
312 * use is to mediate communication between process-level code and irq/NMI
313 * handlers, all running on the same CPU.
314 */

论坛徽章:
4
天秤座
日期:2013-10-18 13:58:33金牛座
日期:2013-11-28 16:17:01辰龙
日期:2014-01-14 09:54:32戌狗
日期:2014-01-24 09:23:27
4 [报告]
发表于 2013-01-28 18:51 |只看该作者
并非CPU记录那个寄存器是从哪个地址读出来,而是编译器记录哪个寄存器是从哪个地址(变量名)里读出来。

论坛徽章:
3
15-16赛季CBA联赛之山东
日期:2016-10-30 08:47:3015-16赛季CBA联赛之佛山
日期:2016-12-17 00:06:31CU十四周年纪念徽章
日期:2017-12-03 01:04:02
5 [报告]
发表于 2013-01-29 12:32 |只看该作者

扔个链接给LZ... 有空看看也不错.
http://ridiculousfish.com/blog/posts/barrier.html

论坛徽章:
3
寅虎
日期:2013-11-27 07:53:29申猴
日期:2014-09-12 09:24:152015年迎新春徽章
日期:2015-03-04 09:48:31
6 [报告]
发表于 2013-01-29 21:09 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽

论坛徽章:
0
7 [报告]
发表于 2013-01-30 09:10 |只看该作者
明显所有关键字都是作用于编译器的,硬件可不认识什么关键字。

论坛徽章:
0
8 [报告]
发表于 2013-02-05 14:49 |只看该作者
volatile确实只在编译器层器作用.

硬件的缓存的问题(内存映射的输入输出端口,或者多处理器)是由硬件设计来处理的。
可以用总线监听(一般是MMU来进行的),内存被写时,让缓存无效就可以了。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP