在Python中计算struct的CRC

时间:2012-08-17 01:15:20

标签: python struct crc pack nrpe

我有以下结构,来自C:中的NRPE守护进程代码

typedef struct packet_struct {
  int16_t packet_version;
  int16_t packet_type;
  uint32_t crc32_value;
  int16_t result_code;
  char buffer[1024];
} packet;

我想将这种数据格式从Python发送到C守护进程。当crc32_value0时计算CRC,然后将其放入结构中。我的Python代码如下:

cmd = '_NRPE_CHECK'
pkt = struct.pack('hhIh1024s', 2, 1, 0, 0, cmd)
# pkt has length of 1034, as it should
checksum = zlib.crc32(pkt) & 0xFFFFFFFF
pkt = struct.pack('hhIh1024s', 2, 1, checksum, 0, cmd)
socket.send(....)

守护程序正在接收以下值:version=2 type=1 crc=FE4BBC49 result=0

但它正在计算crc=3731C3FD

计算CRC的实际C代码是:

https://github.com/KristianLyng/nrpe/blob/master/src/utils.c

通过以下方式调用:

calculate_crc32((char *)packet, sizeof(packet));

当我将这两个函数移植到Python时,我得到的内容与zlib.crc32返回的内容相同。

我的struct.pack电话是否正确?为什么我的CRC计算与服务器的计算不同?

1 个答案:

答案 0 :(得分:2)

来自Python struct documentation

  

处理与平台无关的数据格式或省略隐式填充   字节,使用标准大小和对齐而不是本机大小和   对齐:有关详细信息,请参阅Byte Order, Size, and Alignment

使用'!'作为第一个格式字符,使打包结构与平台无关。它强制使用big-endian,标准类型大小,并且没有填充字节。那么CRC应该是一致的。