修改pcap数据包后checksum重新计算生成

1、最近用python修改完ip之后数据流出现问题,发现为修改完ip,未修改checksum的值,编写脚本留作后续使用

1.1 计算原理

1.发送IP数据报计算checksum

(1)将校验和字段置为0;

(2)对首部中(一般为20B)每个16位字进行二进制反码求和;

(3)将(2)中得到的和再取反码,即得checksum,写入校验和字段中。

抓个IP数据包,取IP数据报报头部分(20B),数据如下:

45 00 00 30 80 4c 40 00 80 06 b5 2e d3 43 11 7b cb 51 15 3d

下面我来计算一下校验和:

(1)将校验和字段置为0:

将b5 2e置为00 00,即变成:

45 00 00 30 80 4c 40 00 80 06 00 00 d3 43 11 7b cb 51 15 3d

(2)反码求和

4500+0030+804c+4000+8006+0000+d343+117b+cb51+153d=34ace

将将进位(3)加到低16位(4ace)上:0003+4ace=4ad1

(3)取反码

将4ad1取反得:checksum=b52e

2.接收IP数据报检验IP校验和

(1)对首部中每个16 bit进行二进制反码求和;

(2)将(1)中得到的和再取反码 ,看是否为0.

接收到的IP数据报首部:

45 00 00 30 80 4c 40 00 80 06 b5 2e d3 43 11 7b cb 51 15 3d

下面来验证下:

(1)反码求和

4500+0030+804c+4000+8006+b52e+d343+117b+cb51+153d=3fffc

0003+fffc=ffff

(2)取反码:~ffff=0 正确

1.2 出入修改后ip的数据流,返回正确checksum的数据流

def checksum(new_pkt):
    """
    :param new_pkt:pcap 包一条修改完 source_ip,dest_ip 的数据流
    :return: 计算好checksum的一条数据流
    """
    ip_header = new_pkt[14:34].hex()
    checksum = 0
    # 反码求和
    for i in range(0,37,4):
        if i == 20:continue
        checksum = checksum + int(ip_header[i:i+4], 16)
    checksum = "{:x}".format(checksum)
    # 将进位加到低16位上
    checksum = '{:016b}'.format(int(checksum[0:1], 16) + int(checksum[1:5], 16))
    # 取反码
    checksum_fanma = ''
    for i in checksum:
        if i == '1':
            checksum_fanma = checksum_fanma + str(0)
        else:
            checksum_fanma = checksum_fanma + str(1)

    checksum = bytes.fromhex("{:04x}".format(int(checksum_fanma, 2)))
    # 拼接成新的数据包返回
    new_pkt = new_pkt[0:24] + checksum + new_pkt[26:]
    return new_pkt

  

posted @ 2022-09-08 13:48  RoyFans  阅读(298)  评论(0编辑  收藏  举报