问题描述
我有一个嵌套的数据结构,其中包含一些位域数组,我需要使用这些位域用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;
}