我正在使用libtrace来解析网络数据包,但我认为这是一个endian问题。
以下是Radiotap数据包的libtrace定义:
typedef struct libtrace_radiotap_t {
uint8_t it_version; /**< Radiotap version */
uint8_t it_pad; /**< Padding for natural alignment */
uint16_t it_len; /**< Length in bytes of the entire Radiotap header */
uint32_t it_present; /**< Which Radiotap fields are present */
} PACKED libtrace_radiotap_t;因此,我将我的libtrace_packet_t转换为此Radiotap结构,并检查结果:
link = (char *) trace_get_packet_buffer(packet, &linktype, NULL);
if (linktype != TRACE_TYPE_80211_RADIO)
return;
rtap = (libtrace_radiotap_t *) link;
printf("%d %d %d %d\n", rtap->it_present, rtap->it_pad, rtap->it_len,
rtap->it_present);在我的开发机器上,来自我的pcap文件中的包的Radiotap数据是:
806959 0 72 806959这是正确的。我的开发机器正在成功地解析我希望从pcap文件中看到的数据。
当在我的生产机器上运行时,我看到了不同的值:
793775104 0 18432 793775104相同pcap文件中的相同数据包。不同的Radiotap值。我怀疑问题出在两台机器的字节顺序不同上。然而,rtap.it_version是一个uint8_t,它是单字节的,不应该受到endian问题的影响,不是吗?
发布于 2013-08-29 11:42:42
这应该是一个字节序问题。对于72,十六进制是0x48,它是一个uint16_t,所以在不同的字节顺序中是0x4800 = 18432。没错。对于806959 = 0xC502F,在不同的字节顺序中为0x2F50C000 = 793775104。
这可能会有所帮助:
#define T(x) (((x&0xff)<<24)|((x&0xff00)<<8)|((x&0xff0000)>>8)|((x&0xff000000)>>24))发布于 2013-08-30 06:12:43
libtrace返回的头结构指向出现在"wire“上的头,即不尝试转换为主机字节顺序。
Radiotap报头字段始终是小端的(与传统的网络字节顺序相反,传统的网络字节顺序是大端的),因此如果您要尝试手动解析报头,则需要对此进行补偿。仅仅使用ntohl是行不通的,因为这需要您转换的值是大端的。
然而,一种更好的方法是使用内置的libtrace函数来访问各种无线电抽头字段,例如trace_get_wireless_rate、trace_get_wireless_signal_strength_dbm和许多其他字段。这些函数将根据您的主机架构进行字节顺序转换,因此您不必担心。
至于你对struct ieee80211_frame_control的问题,这看起来像是一个bug。我推荐filing a bug ticket with the libtrace developers。
https://stackoverflow.com/questions/18501516
复制相似问题