免费注册 查看新帖 |

Chinaunix

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

[C] 以下函数为什么在O2下会产生下标越界的warning? [复制链接]

论坛徽章:
20
程序设计版块每日发帖之星
日期:2015-08-17 06:20:00程序设计版块每日发帖之星
日期:2016-07-16 06:20:00程序设计版块每日发帖之星
日期:2016-07-18 06:20:00每日论坛发贴之星
日期:2016-07-18 06:20:00黑曼巴
日期:2016-12-26 16:00:3215-16赛季CBA联赛之江苏
日期:2017-06-26 11:05:5615-16赛季CBA联赛之上海
日期:2017-07-21 18:12:5015-16赛季CBA联赛之青岛
日期:2017-09-04 17:32:0515-16赛季CBA联赛之吉林
日期:2018-03-26 10:02:16程序设计版块每日发帖之星
日期:2016-07-15 06:20:0015-16赛季CBA联赛之江苏
日期:2016-07-07 18:37:512015亚冠之萨济拖拉机
日期:2015-08-17 12:21:08
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-11-13 21:09 |只看该作者 |倒序浏览
gcc版本4.4.7
  1. typedef unsigned int u32;
  2. static inline u32 jhash2(const u32 *k, u32 len, u32 init)
  3. {
  4.         u32 c = 0;
  5.         while (len > 3) {
  6.                 c += k[2];
  7.                 len -= 3;
  8.                 k += 3;
  9.         }
  10.         return c;
  11. }
  12. struct Obj1 {
  13.         u32 a[5];
  14.         u32 b[5];
  15. } obj1;

  16. struct Obj2 {
  17.         u32 a[10];
  18. } obj2;

  19. #ifndef N
  20. #define N       8
  21. #endif

  22. #ifdef USE_OBJ2
  23. #define OBJ     obj2
  24. #else
  25. #define OBJ     obj1
  26. #endif

  27. u32 test()
  28. {
  29.         return jhash2((u32 *)&OBJ, N, 0);
  30. }
复制代码
以上代码是对jhash2的简单模拟。
Obj1和Obj2结构的内存布局应该一样的。
但是,当使用-Wall -O2 -fno-strict-aliasing编译的时候,Obj1在N取7/8/9的时候会产生warning: array subscript is above array bounds
使用Obj2就不会。

按理说,当N=7/8/9的时候,也只会访问到k[2]/k[5],并没有越界发生。
而真正会产生越界的N值,N>10,反而没有warning。

请问各位大牛,这该怎么解释?感觉跟循环展开有关系?
  1. for i in {0..50}
  2. do
  3.         gcc test.c -Wall -O2 -fno-strict-aliasing -c -DN=$i -Werror -DUSE_OBJ1 || echo error: N=$i, OBJ=obj1
  4.         gcc test.c -Wall -O2 -fno-strict-aliasing -c -DN=$i -Werror -DUSE_OBJ2 || echo error: N=$i, OBJ=obj2   
  5. done
复制代码

论坛徽章:
20
程序设计版块每日发帖之星
日期:2015-08-17 06:20:00程序设计版块每日发帖之星
日期:2016-07-16 06:20:00程序设计版块每日发帖之星
日期:2016-07-18 06:20:00每日论坛发贴之星
日期:2016-07-18 06:20:00黑曼巴
日期:2016-12-26 16:00:3215-16赛季CBA联赛之江苏
日期:2017-06-26 11:05:5615-16赛季CBA联赛之上海
日期:2017-07-21 18:12:5015-16赛季CBA联赛之青岛
日期:2017-09-04 17:32:0515-16赛季CBA联赛之吉林
日期:2018-03-26 10:02:16程序设计版块每日发帖之星
日期:2016-07-15 06:20:0015-16赛季CBA联赛之江苏
日期:2016-07-07 18:37:512015亚冠之萨济拖拉机
日期:2015-08-17 12:21:08
2 [报告]
发表于 2015-11-13 21:17 |只看该作者
让命令行更清晰一些,gcc -c  test.c -O2 -Wall -Werror -DN=$i -DUSE_OBJ1/USE_OBJ2

-fno-strict-aliasing不影响结果。

cc1: warnings being treated as errors
test.c: In function 'test':
test.c:6: error: array subscript is above array bounds
error: N=7, OBJ=obj1

cc1: warnings being treated as errors
test.c: In function 'test':
test.c:6: error: array subscript is above array bounds
error: N=8, OBJ=obj1

cc1: warnings being treated as errors
test.c: In function 'test':
test.c:6: error: array subscript is above array bounds
error: N=9, OBJ=obj1

上面的error实际都是warning,为了简化脚本才使用的-Werror

论坛徽章:
20
程序设计版块每日发帖之星
日期:2015-08-17 06:20:00程序设计版块每日发帖之星
日期:2016-07-16 06:20:00程序设计版块每日发帖之星
日期:2016-07-18 06:20:00每日论坛发贴之星
日期:2016-07-18 06:20:00黑曼巴
日期:2016-12-26 16:00:3215-16赛季CBA联赛之江苏
日期:2017-06-26 11:05:5615-16赛季CBA联赛之上海
日期:2017-07-21 18:12:5015-16赛季CBA联赛之青岛
日期:2017-09-04 17:32:0515-16赛季CBA联赛之吉林
日期:2018-03-26 10:02:16程序设计版块每日发帖之星
日期:2016-07-15 06:20:0015-16赛季CBA联赛之江苏
日期:2016-07-07 18:37:512015亚冠之萨济拖拉机
日期:2015-08-17 12:21:08
3 [报告]
发表于 2015-11-13 21:20 |只看该作者
补充,-O2/-O3/-Os下是这个结果,-O0/-O1下没有。

论坛徽章:
84
每日论坛发贴之星
日期:2015-12-29 06:20:00每日论坛发贴之星
日期:2016-01-16 06:20:00每周论坛发贴之星
日期:2016-01-17 22:22:00程序设计版块每日发帖之星
日期:2016-01-20 06:20:00每日论坛发贴之星
日期:2016-01-20 06:20:00程序设计版块每日发帖之星
日期:2016-01-21 06:20:00每日论坛发贴之星
日期:2016-01-21 06:20:00程序设计版块每日发帖之星
日期:2016-01-23 06:20:00程序设计版块每日发帖之星
日期:2016-01-31 06:20:00数据库技术版块每日发帖之星
日期:2016-01-16 06:20:00程序设计版块每日发帖之星
日期:2016-01-16 06:20:00程序设计版块每日发帖之星
日期:2016-01-14 06:20:00
4 [报告]
发表于 2015-11-13 23:05 |只看该作者
sizeof(struct Obj1)
sizeof(struct Obj2)

这两个打印出来看一样吗?

论坛徽章:
84
每日论坛发贴之星
日期:2015-12-29 06:20:00每日论坛发贴之星
日期:2016-01-16 06:20:00每周论坛发贴之星
日期:2016-01-17 22:22:00程序设计版块每日发帖之星
日期:2016-01-20 06:20:00每日论坛发贴之星
日期:2016-01-20 06:20:00程序设计版块每日发帖之星
日期:2016-01-21 06:20:00每日论坛发贴之星
日期:2016-01-21 06:20:00程序设计版块每日发帖之星
日期:2016-01-23 06:20:00程序设计版块每日发帖之星
日期:2016-01-31 06:20:00数据库技术版块每日发帖之星
日期:2016-01-16 06:20:00程序设计版块每日发帖之星
日期:2016-01-16 06:20:00程序设计版块每日发帖之星
日期:2016-01-14 06:20:00
5 [报告]
发表于 2015-11-16 13:15 |只看该作者
gcc 5.1.1 没有这个问题:

[yjh@jiyin-ws bkr-client-improved]$ gcc kkk.c -Wall -O2 -fno-strict-aliasing -c -DN=7  -DUSE_OBJ1
[yjh@jiyin-ws bkr-client-improved]$ gcc kkk.c -Wall -O2 -fno-strict-aliasing -c -DN=8  -DUSE_OBJ1
[yjh@jiyin-ws bkr-client-improved]$ gcc kkk.c -Wall -O2 -fno-strict-aliasing -c -DN=9  -DUSE_OBJ1
[yjh@jiyin-ws bkr-client-improved]$ LANG=C gcc --version
gcc (GCC) 5.1.1 20150618 (Red Hat 5.1.1-4)
Copyright (C) 2015 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

论坛徽章:
36
CU大牛徽章
日期:2013-09-18 15:24:20NBA常规赛纪念章
日期:2015-05-04 22:32:03牛市纪念徽章
日期:2015-07-24 12:48:5515-16赛季CBA联赛之辽宁
日期:2016-03-30 09:26:4715-16赛季CBA联赛之北控
日期:2016-03-30 11:26:2315-16赛季CBA联赛之广夏
日期:2016-05-20 15:46:5715-16赛季CBA联赛之吉林
日期:2016-05-24 11:38:0615-16赛季CBA联赛之青岛
日期:2016-05-30 13:41:3215-16赛季CBA联赛之同曦
日期:2016-06-23 16:41:052015年亚洲杯之巴林
日期:2015-02-03 15:05:04CU大牛徽章
日期:2013-09-18 15:24:52CU十二周年纪念徽章
日期:2013-10-24 15:46:53
6 [报告]
发表于 2015-11-16 14:15 |只看该作者
看楼上的,说明你gcc版本太老了

论坛徽章:
324
射手座
日期:2013-08-23 12:04:38射手座
日期:2013-08-23 16:18:12未羊
日期:2013-08-30 14:33:15水瓶座
日期:2013-09-02 16:44:31摩羯座
日期:2013-09-25 09:33:52双子座
日期:2013-09-26 12:21:10金牛座
日期:2013-10-14 09:08:49申猴
日期:2013-10-16 13:09:43子鼠
日期:2013-10-17 23:23:19射手座
日期:2013-10-18 13:00:27金牛座
日期:2013-10-18 15:47:57午马
日期:2013-10-18 21:43:38
7 [报告]
发表于 2015-11-16 15:03 |只看该作者
测试了几下,感觉是以Obj1中大的那个数组作为越界判断了

论坛徽章:
36
CU大牛徽章
日期:2013-09-18 15:24:20NBA常规赛纪念章
日期:2015-05-04 22:32:03牛市纪念徽章
日期:2015-07-24 12:48:5515-16赛季CBA联赛之辽宁
日期:2016-03-30 09:26:4715-16赛季CBA联赛之北控
日期:2016-03-30 11:26:2315-16赛季CBA联赛之广夏
日期:2016-05-20 15:46:5715-16赛季CBA联赛之吉林
日期:2016-05-24 11:38:0615-16赛季CBA联赛之青岛
日期:2016-05-30 13:41:3215-16赛季CBA联赛之同曦
日期:2016-06-23 16:41:052015年亚洲杯之巴林
日期:2015-02-03 15:05:04CU大牛徽章
日期:2013-09-18 15:24:52CU十二周年纪念徽章
日期:2013-10-24 15:46:53
8 [报告]
发表于 2015-11-16 17:32 |只看该作者
回复 7# hellioncu


    Obj1中不是两数组一样大么,如果以a[]来判断,当7,8,9时,会取到k[5], k[5]应该算是越界了,最多应该到a[4]。是这样吗? 可这些编译时能检查到的吗?

论坛徽章:
324
射手座
日期:2013-08-23 12:04:38射手座
日期:2013-08-23 16:18:12未羊
日期:2013-08-30 14:33:15水瓶座
日期:2013-09-02 16:44:31摩羯座
日期:2013-09-25 09:33:52双子座
日期:2013-09-26 12:21:10金牛座
日期:2013-10-14 09:08:49申猴
日期:2013-10-16 13:09:43子鼠
日期:2013-10-17 23:23:19射手座
日期:2013-10-18 13:00:27金牛座
日期:2013-10-18 15:47:57午马
日期:2013-10-18 21:43:38
9 [报告]
发表于 2015-11-16 17:41 |只看该作者
idi0t 发表于 2015-11-16 17:32
回复 7# hellioncu


我也是猜的,因为我试着改成

        u32 a[3];
        u32 b[7]; 之类的,然后 N取8就没警告。

不管怎样,这个警告看来是不应该的。

论坛徽章:
20
程序设计版块每日发帖之星
日期:2015-08-17 06:20:00程序设计版块每日发帖之星
日期:2016-07-16 06:20:00程序设计版块每日发帖之星
日期:2016-07-18 06:20:00每日论坛发贴之星
日期:2016-07-18 06:20:00黑曼巴
日期:2016-12-26 16:00:3215-16赛季CBA联赛之江苏
日期:2017-06-26 11:05:5615-16赛季CBA联赛之上海
日期:2017-07-21 18:12:5015-16赛季CBA联赛之青岛
日期:2017-09-04 17:32:0515-16赛季CBA联赛之吉林
日期:2018-03-26 10:02:16程序设计版块每日发帖之星
日期:2016-07-15 06:20:0015-16赛季CBA联赛之江苏
日期:2016-07-07 18:37:512015亚冠之萨济拖拉机
日期:2015-08-17 12:21:08
10 [报告]
发表于 2015-11-19 11:39 |只看该作者
回复 4# yjh777

是一样的。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP