免费注册 查看新帖 |

Chinaunix

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

[内核入门] 连续两个printf输出变量不一致的情况 [复制链接]

论坛徽章:
1
2015年迎新春徽章
日期:2015-03-04 09:57:09
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-01-27 13:26 |只看该作者 |倒序浏览
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. //#include <stddef.h>
  5. #pragma pack(8)

  6. #define SETBIT(a,i)   ((a)[(i) / 8] |= (1 << (7 - (i) % 8)))

  7. typedef struct _testStru{
  8.     char chArr[8];
  9.     char *pCh;
  10. }testStru;

  11. int main()
  12. {
  13.     testStru* test=(testStru *)calloc(1, sizeof(testStru));
  14.     printf("test.chArr1 is %x.\n", test->pCh);
  15.     SETBIT(test->chArr, 65);   //故意内存越界

  16.     printf("test.chArr2 is %p.\n", test->pCh);
  17.     printf("test.chArr3 is %p.\n", test->pCh);

  18.     return 0;

  19. }
复制代码
只是模拟的我们程序中的一个用例,第二个printf打印的值竟然是没有SETBIT前的值,第三个printf才变成更新后的值,求解释。。。。


我在一个linux  server上用gcc version 4.1.2 20080704 (Red Hat 4.1.2-52)竟然模拟不出来。。。

论坛徽章:
0
2 [报告]
发表于 2013-01-27 23:10 |只看该作者
程序运行起来后就是一个进程了,进程空间里一般分为代码正文段、数据段、BSS段、heap及stack等五个部分,如果你的程序越界访问了那很可能会破坏进程空间的分配。像你以上的程序,要是想看到一致的效果,可以在
SETBIT(test->chArr, 65);   //故意内存越界
mb(); //设置屏障
再看看两个打印是否一致。

论坛徽章:
0
3 [报告]
发表于 2013-01-28 10:25 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽

论坛徽章:
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 11:02 |只看该作者
什么版本的编译器和运行环境能模拟出来你这个问题?

论坛徽章:
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
5 [报告]
发表于 2013-01-28 11:05 |只看该作者
我理解这只是编译器的优化行为,所以加barrier()即可,不需要mb()。

#define barrier() __asm__ __volatile__("": : :"memory")

论坛徽章:
1
2015年迎新春徽章
日期:2015-03-04 09:57:09
6 [报告]
发表于 2013-01-29 08:29 |只看该作者
typedef struct _testStru{
    char chArr[8];
    volatile char *pCh;
}testStru;

现在感觉是 pCh的值进行了优化
回复 3# stephen_du


   

论坛徽章:
1
2015年迎新春徽章
日期:2015-03-04 09:57:09
7 [报告]
发表于 2013-01-29 08:31 |只看该作者
回复 4# liuiang

这个问题我用 RedHat 和 GCC没有模拟出来。

问题重现是在 Octeon的硬件+Vxworks OS+ GCC


   

论坛徽章:
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
8 [报告]
发表于 2013-01-29 12:49 |只看该作者
我在gcc 4.4.5上,用-O3优化,能够模拟出问题。

问题可能是编译器对读写进行了乱序优化导致的。

论坛徽章:
0
9 [报告]
发表于 2013-01-29 14:45 |只看该作者
其实就是没有显式的对pCh就行写,只是SETBIT 对pCh有side effect, 如果pCh在一个寄存器里面,然后 SETBIT, 再去读pCh,这个过程中 编译器认为pCh的值没变,所以没有再去从内存取,所以是错的,这种情况只能加barrier或者volentile 人工保证不出问题,

论坛徽章:
0
10 [报告]
发表于 2013-01-30 09:06 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP