- 论坛徽章:
- 0
|
我想举一个自己最近在项目中犯的错误来说明要踏踏实实做人,不要做装B青年 
在代码中,我需要在一个library和一个daemon之间通过socket传送数据包,包的格式定义如下(为了简化,我就用最简单的数据类型举例):
- typedef struct {
- int head;
- int size; //指明整个包的长度
- char reply;
- char data[0];
- } packet;
- packet* cmd = malloc (sizeof(packet) + 20);
- memcpy (packet->data, some_data, 20);
复制代码
daemon将上面分配的cmd包发送给library,library接收到包后,需要将data字段中的数据取出来。size指明了整个包的长度,但没有字段指明数据的长度。我需要这么一个指明数据长度的字段吗?作为一个装B青年,我认为当然不需要,于是我这样来计算数据的长度:
- #define offsetof(type, element) ((int)&((type *)0)->element)
- static inline size_t packet_data_len(packet* cmd) {
- assert(cmd);
- return cmd->size - offsetof(packet, data);
- }
- memcpy (buffer_to_receive_data, cmd->data, packet_data_len (cmd));
复制代码
于是乎,这段程序成功的给我带来了无数的bug,莫名奇妙的segfault,奇怪的数据错误,还是有部分时间的正常工作。当然,最终我还是找到了问题:
sizeof (packet) == 12;
这是合理的,char reply被padding成了4个字节,而char data[0]字节为0。
但,offsetof(packet, data) == 9,在计算偏移时,char reply为一个字节,没有padding。
所以packet_data_len每次都会返回比真实的数据多3个字节 ……
最后我还是老老实实加了个data_len字段指明数据的长度,并告诫自己,本本分分写程序,莫装B …… |
|