一个由自增运算符以及C语法顺序细节引起的bug

本文通过一个具体的Modbus代码实例,详细分析了C语言中自增运算符及表达式解析顺序引起的bug,展示了如何通过反汇编理解问题所在,并总结了C语言运算符的优先级和结合方向。
 一、问题描述
在编写modbus代码时发生一件由语法细节引起的bug,起因是自增运算符以及C语法顺序。
输入的数据是2233=0X08B9,高低字节顺序是0x08 0xB9, 使用modbus poll向92寄存器写入十进制数据2233.
但是经过(*reg++)*256+(reg++)之后,结果变成了0xB908。
检查内存也是0xB908. 说明reg_value写入了错误的数。
 
二、问题调查
反汇编后查看.
1)LDRB R2,[R5],#0X01 
--> 把R5数据的低字节放入R2,并且R5+1字节装入R5
--> R2 = 0X08, R5=0XB9
2)LDRB R0,[R5],#0X01  
--> 把R5数据的低字节放入R2,并且R5+1字节装入R5
--> R0 = 0XB9, R5=xxxx
3) ADD R0,R2,R0,LSL,#8
--> R0 = R2+(R0<<8) = 0xB908.
显然(*reg++)*256+(reg++)的解析顺序变为了:
(1)(reg++)=0x08, (2)(*reg++)*256=0xB9*256
 
 
三、举一反三
上述问题是由于编译器以及c语法解析顺序造成的。
很多这样的问题,比如近期遇到的一个宏定义变量没有加括号导致内部展开时候优先级变化等。
 
一般的编译器是从右到左如fun(a,b)这个函数调用,是先计算参数b,入栈,再计算参数a,入栈。
例1:
int a=1;
printf("%d %d", a++,++a);
printf(" %d\n",a);
--》:结果2 2 3
printf("%d %d", a++,++a);  //先计算++a,先自增,a的值变为2,将2入栈   再来计算a++,将a的值2入栈,再使a自增,a的值变为3
printf(" %d\n",a); //a的值已经变为3了
 
再次复习c基础,运算符优先级和编译器基本常识必须正确掌握。
 
优先级运算符名称或含义使用形式结合方向说明
1[ ]数组下标数组名[整型表达式]左到右 
( )圆括号(表达式)/函数名(形参表) 
.成员选择(对象)对象.成员名 
->成员选择(指针)对象指针->成员名 
2-负号运算符-表达式右到左单目运算符
(类型)强制类型转换(数据类型)表达式 
++自增运算符++变量名/变量名++单目运算符
--自减运算符--变量名/变量名--单目运算符
*取值运算符*指针表达式单目运算符
&取地址运算符&左值表达式单目运算符
!逻辑非运算符!表达式单目运算符
~按位取反运算符~表达式单目运算符
sizeof长度运算符sizeof 表达式/sizeof(类型) 
3/表达式/表达式左到右双目运算符
*表达式*表达式双目运算符
%余数(取模)整型表达式%整型表达式双目运算符
4+表达式+表达式左到右双目运算符
-表达式-表达式双目运算符
5<<左移表达式<<表达式左到右双目运算符
>>右移表达式>>表达式双目运算符
6>大于表达式>表达式左到右双目运算符
>=大于等于表达式>=表达式双目运算符
<小于表达式<表达式双目运算符
<=小于等于表达式<=表达式双目运算符
7==等于表达式==表达式左到右双目运算符
!=不等于表达式!= 表达式双目运算符
8&按位与整型表达式&整型表达式左到右双目运算符
9^按位异或整型表达式^整型表达式左到右双目运算符
10|按位或整型表达式|整型表达式左到右双目运算符
11&&逻辑与表达式&&表达式左到右双目运算符
12||逻辑或表达式||表达式左到右双目运算符
13?:条件运算符表达式1? 表达式2: 表达式3右到左三目运算符
14=赋值运算符变量=表达式右到左 
/=除后赋值变量/=表达式 
*=乘后赋值变量*=表达式 
%=取模后赋值变量%=表达式 
+=加后赋值变量+=表达式 
-=减后赋值变量-=表达式 
<<=左移后赋值变量<<=表达式 
>>=右移后赋值变量>>=表达式 
&=按位与后赋值变量&=表达式 
^=按位异或后赋值变量^=表达式 
|=按位或后赋值变量|=表达式 
15,逗号运算符表达式,表达式,…左到右从左向右顺序运算
 

转载于:https://www.cnblogs.com/pingwen/p/7260399.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值