编译选项导致的结构体字节参数异常

前言

在构建编译工程时,会有一些对应的编译配置选项,不同的编译器,会有对应的配置项。本文介绍GHS工程中编译选项配置不对应导致的异常。

问题描述

在S32K3集成工程中,核1的INP_SWC中使用E2E_P01Protect会导致核1 shutdown

排查原因是在调用Crc_CalculateCRC8函数时挂掉了

屏蔽里面的for循环后,可以跑起来

uint8 Crc_CalculateCRC8(const uint8* Crc_DataPtr, uint32 Crc_Length, uint8 Crc_StartValue8, boolean Crc_IsFirstCall)
{
    uint32  index;
    uint8  result = Crc_StartValue8;
    uint8  crcTemp;
    if(Crc_DataPtr != NULL_PTR)
    {
        crcTemp = (Crc_IsFirstCall != FALSE) ? ((uint8)CRC_INITIAL_VALUE8) : (Crc_StartValue8^ CRC_XOR_VALUE8);
        
        for (index = 0U; index < Crc_Length; ++index)
        {
            /* Impact of temporary rest on next crc rest */
            crcTemp ^= ((uint8)Crc_DataPtr[index]) ;
            /* Next temporary crc rest */        
            crcTemp = CRC_8_Tbl[crcTemp];    
        }
        result = crcTemp ^ CRC_XOR_VALUE8;
    }
    
    return (result);
}

原因分析

查看输入的参数,发现length异常

往前找,发现传入的config参数就不对

实际定义时的参数如下

image
传入E2E_P01Protect函数中的参数变成了
image
可以看到,DataLenth的地址在传递前为0x0046D0B4,而在传递后变成了0x0046D0B2

导致传递的参数乱了

为什么会出现这种情况呢?

有一点可以注意一下,出问题的地方在DataIDMode后面,他不是一个变量定义,而是一个枚举enum类型

为什么同一个枚举类型会有两种数据长度呢?

最终排查发现,由于两个源文件在不同的gpj配置文件中被编译,出现异常的文件中的配置文件中定义了–short_enum,开启了编译优化

导致参数在传递后,DataIDMode只占2个byte,而在未优化的文件中,占4个byte.所以导致了问题
查看ghs bulid手册:

–short_enum对枚举使用最小类型

控制枚举的分配。此选项允许的设置为:

•On(–short_enum) -以尽可能小的类型存储枚举。

•Off(–no_short_enum) -[默认]将枚举存储为整数
image
其实在手册中就已经明确指出了,需要确保所有文件配置一致!

问题解决

搞清楚问题出现的缘由,将两个文件中的配置统一之后,问题就解决了。

总结

编译器对应的编译选项,还是需要花时间好好学习下,不然出问题可能很难排查。

  • 5
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

赞哥哥s

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值