c++ - static_cast 可以用于位字段吗?

标签 c++

您好,静态转换是否适用于位域?因为我在这里创建了一个 32 位 Long 数据的位字段,并且我试图将其转换为 u32。

void CO2_SpiDriver::CreateBuffer() {
    //#[ operation CreateBuffer()
    typedef struct scr_t
     {
         union {
            u32 data;
             struct {
                 u8 CRC : 4; 
                 u16 Datatosend : 16;
                 u8 DataLengthCode : 2;
                 u8 AddressReg : 4;
                 u8 ReadandWrite : 1;
                 u8 InstrucCode : 3;
                 u8 ExtChipId : 2;
             };
         };
     } scr_t;  

     //set of data bits
     scr_t se;
     se.CRC = 0x00;   //CRC BITS 0 to 3
     se.Datatosend = 0x0000000000000000;  //Data bits 19 to 4
     se.DataLengthCode = 0x00; //DLC bits 21 and 20
     se.AddressReg = 0x00;     // SMP580 registers to access 25-22
     se.ReadandWrite = 0x01;  //Read|Write Reg 26
     se.InstrucCode = 0x5;    //InstrCode 29-27
     se.ExtChipId = 0x2;      //ExtChipId 31-30       

     static_cast<u32>(se)

但在这里我得到一个错误,scr_t 不能转换为 u32。谁能给些建议

最佳答案

在 C++ 中, union 中只有一个字段在任何时候是事件的。重新解释内存布局的唯一可移植方法是使用 std::memcpy:

struct bf {
    u8 CRC : 4; 
    u16 Datatosend : 16;
    u8 DataLengthCode : 2;
    u8 AddressReg : 4;
    u8 ReadandWrite : 1;
    u8 InstrucCode : 3;
    u8 ExtChipId : 2;
};

bf se;
se.CRC = 0x00;   //CRC BITS 0 to 3
se.Datatosend = 0x0000000000000000;  //Data bits 19 to 4
se.DataLengthCode = 0x00; //DLC bits 21 and 20
se.AddressReg = 0x00;     // SMP580 registers to access 25-22
se.ReadandWrite = 0x01;  //Read|Write Reg 26
se.InstrucCode = 0x5;    //InstrCode 29-27
se.ExtChipId = 0x2;      //ExtChipId 31-30  

u32 se_as_u32;
static_assert(sizeof(se_as_u32) <= sizeof(se), "UB detector"); // avoid reading buffer overflow (see explanation below)
std::memcpy(&se_as_u32, &se, sizeof(se_as_u32));

注意:我添加了这一行

static_assert(sizeof(se_as_u32) <= sizeof(se), "UB detector");

为了保护 se 在其“边界”之外被读取:

std::memcpy(&se_as_u32, &se, sizeof(se_as_u32));

这从 se 中读取 sizeof(u32)(可能是 4)字节。如果 se 的长度小于 4 个字节,memcpy 将读取“在它之外”,这是一种缓冲区溢出并导致未定义的行为(这是坏的)。 如果不考虑不等式 (sizeof(se_as_u32) > sizeof(se)),将调用仁慈的 abort(),程序将立即崩溃可能会工作一段时间并在某一天破坏生产。

关于c++ - static_cast 可以用于位字段吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47711930/

相关文章:

C++ 分配 vector 指针

c++ - C++中的类型转换

c++ - 在 C++ 中转换 uchar

C++初始化一组结构

c++ - 如何在处理多个文件时组织 Vim 缓冲区、窗口和选项卡

c++ - 解决 lambda 表中的 VS2010 错误?

c++ - 找到变量定义的任何程序或技巧?

c++ - g++ libcurl 静态构建在 nghttp2 库上失败

c++ - C++中 protected 和私有(private)派生之间的区别是什么

c++ - pthread_cond_signal 或 pthread_cond_broadcast 调用是否意味着写入内存屏障?