我想知道是否有可能在C中强制对齐字段。使用下面代码中的变量我知道写入 _align_bytes 然后从位读取未定义(反之亦然)因为它是实现依赖的。下面的代码是否是一个有效的方法来“说服”位连续存储在unsigned short大小的东西中?我相信(减去任何endian问题)这段代码是正确的......但是bitfields和union是我最不熟悉的两个C主题。
我正在做一个低级别的微控制器项目,并且想要一种简单的方法来读取配置位而无需大量的位屏蔽。感谢您的任何提示和建议。
萨姆
P.S。请忽略我对有关字节序的任何假设,因为我正在处理的这个项目非常低级,并不打算移植到其他设备/平台。
#include <stdio.h>
#include <assert.h>
typedef union packet {
struct {
unsigned int bit0 : 1;
unsigned int bit1 : 1;
unsigned int bit2 : 1;
unsigned int bit3 : 1;
unsigned int bit4 : 1;
unsigned int bit5 : 1;
unsigned int bit6 : 1;
unsigned int bit7 : 1;
unsigned int bit8 : 1;
unsigned int bit9 : 1;
unsigned int bit10 : 1;
unsigned int bit11 : 1;
unsigned int bit12 : 1;
unsigned int bit13 : 1;
unsigned int bit14 : 1;
unsigned int bit15 : 1;
} bits;
unsigned short _align_bytes;
} packet_t;
int main(int argc, char *argv[]) {
assert(sizeof(unsigned short) == 2);
unsigned short data = 0xA05F;
packet_t *p = (packet_t *)&data;
printf("%u", p->bits.bit15);
printf("%u", p->bits.bit14);
printf("%u", p->bits.bit13);
printf("%u", p->bits.bit12);
printf("%u", p->bits.bit11);
printf("%u", p->bits.bit10);
printf("%u", p->bits.bit9);
printf("%u", p->bits.bit8);
printf("%u", p->bits.bit7);
printf("%u", p->bits.bit6);
printf("%u", p->bits.bit5);
printf("%u", p->bits.bit4);
printf("%u", p->bits.bit3);
printf("%u", p->bits.bit2);
printf("%u", p->bits.bit1);
printf("%u", p->bits.bit0);
return 0;
}
答案 0 :(得分:3)
这是一种常见的模式,据我所知,答案是肯定的:位字段将是连续的并占用与_align_bytes字段相同的内存。这就是工会的重点,对吗?查看同一记忆的不同方式。
我不确定你的意思是“写入_align_bytes然后从位读取是未定义的”。我看到的唯一问题是endianess:bit0可能是lsb或_align_bytes的msb。如果您不希望它是便携式的,那么您只需要进行快速测试以确定它是什么,并且您应该进行设置。
答案 1 :(得分:0)
我不确定,但这不会违反strict aliasing rules因为两个不同类型的指针指向同一个内存位置?在C89和C99中,您无法保证能够正确回归。
您可能需要对此进行测试,如果需要,请对编译器使用-fno-strict-aliasing
或类似内容,以便禁用可能导致问题的严格别名。