4

我想计算要发送的 UDP 标头数据包的校验和:

packetosend = """60 00 00 00 00 24 3a 40 20 02 c0 a8 01 50 00 01 00 00 
00 00 00 00 09 38 20 02 c0 a8 01 50 00 01 00 00 00 00 00 00 09 6f"""

所以我需要加入这个 utf-16(不是问题)并计算这个特定数据包的校验和。我怎样才能做到这一点?

谢谢!

编辑:是的,它是 ICMPv6 数据包的 IPv6 标头,无论如何我想知道的是公式,以及它是如何工作的。

我再举一个 ICMP ping echo (v4) 数据包的例子:

packet = """
08 00 d1 15 76 0c 00 07 bf d3 55 4a ad b5 03 00   // "d1 15" is the packet checksum 
08 09 0a 0b 0c 0d 0e 0f 10 11 12 13 14 15 16 17
18 19 1a 1b 1c 1d 1e 1f 20 21 22 23 24 25 26 27
28 29 2a 2b 2c 2d 2e 2f 30 31 32 33 34 35 36 37"""

谢谢。

4

2 回答 2

10

对我来说,这看起来不像是 UDP 数据包。它看起来像 ICMPv6 数据包的 IPv6 标头,但缺少数据包的实际有效负载。

IPv6 标头不包含校验和

对于ICMP,校验和是“以 Type 字段开头的 ICMP 消息的反码和的 16 位反码”。一个补码算术涉及进位位的特殊处理。

def carry_around_add(a, b):
    c = a + b
    return (c & 0xffff) + (c >> 16)

def checksum(msg):
    s = 0
    for i in range(0, len(msg), 2):
        w = ord(msg[i]) + (ord(msg[i+1]) << 8)
        s = carry_around_add(s, w)
    return ~s & 0xffff

为了计算正确的校验和,您当然必须从 right 开始msg,这意味着至少首先将校验和字段清零,并且可能还需要添加“虚拟标头”,具体取决于您使用的协议。

于 2009-11-20T09:17:24.373 回答
5

有一个可靠的校验和函数可以正确处理 Scapy 中的字节顺序问题(http://www.secdev.org/projects/scapy,GPLv2)。

在 Python 2.7 中,scapy 的 utils.py:

if struct.pack("H",1) == "\x00\x01": # big endian
    def checksum(pkt):
        if len(pkt) % 2 == 1:
            pkt += "\0"
        s = sum(array.array("H", pkt))
        s = (s >> 16) + (s & 0xffff)
        s += s >> 16
        s = ~s
        return s & 0xffff
else:
    def checksum(pkt):
        if len(pkt) % 2 == 1:
            pkt += "\0"
        s = sum(array.array("H", pkt))
        s = (s >> 16) + (s & 0xffff)
        s += s >> 16
        s = ~s
        return (((s>>8)&0xff)|s<<8) & 0xffff

这对于任何使用 IP 标头校验和(IP、TCP、UDP、ICMP)的协议都是正确的。但是,UDP 也有特殊情况,计算为 0x0000 的校验和应作为 0xffff 传输。上述函数没有考虑到这一点,因此对于 UDP,您必须处理这种特殊情况。

于 2013-12-21T17:19:00.863 回答