如何使用SDCC创建位域的打包数组?

问题描述

我有一个嵌套的数据结构,其中包含一些位域数组,我需要使用这些位域用SDCC编译MCS-51目标。

这是一个简化的示例:

example.c

struct data {
    unsigned char a : 1;
    unsigned char b : 2;
};

struct data dummy[8];

void main()
{
}

由于struct data包含3位,所以8个实例总共包含24位,因此3个字节的内存足以存储它们。

但是SDCC分配了8个字节,正如我们在结果example.asm中看到的那样:

$ sdcc -c example.c
$ cat example.asm
[…]
; File Created by SDCC : free open source ANSI-C Compiler
; Version 3.5.0 #9253 (Mar 19 2016) (Linux)
[…]
        .module example
        .optsdcc -mmcs51 --model-small
[…]
        .area DSEG    (DATA)
_dummy::
        .ds 8

是否有办法让SDCC为dummy仅分配3个字节?

作为一种解决方法,我目前根本不使用struct,而是使用一些宏来计算数据结构的总大小,分配字节的平面数组并生成索引和位掩码以用于访问各个成员。我想改用dummy[5].b之类的语法。

某些编译器似乎具有启用位字段打包的选项(例如#pragma pack),但是我在SDCC手册中找不到类似的东西。

解决方法

我担心在任何编译器中都不可能。它将为每个结构分配至少一个字节

我会做这样的事情:

typedef struct {
    unsigned char a0 : 1;
    unsigned char b0 : 2;
    unsigned char a1 : 1;
    unsigned char b1 : 2;
    unsigned char a2 : 1;
    unsigned char b2 : 2;
    unsigned char a3 : 1;
    unsigned char b3 : 2;
    unsigned char a4 : 1;
    unsigned char b4 : 2;
    unsigned char a5 : 1;
    unsigned char b5 : 2;
    unsigned char a6 : 1;
    unsigned char b6 : 2;
    unsigned char a7 : 1;
    unsigned char b7 : 2;
}data;

data dx;

#define GT(dt,member,bit)   ((dt).member ##bit)

void foo()
{
    GT(dx,b,5) = 2;
}