【问题标题】:Using libtins to recreate IP packet change its header使用 libtins 重新创建 IP 数据包更改其标头
【发布时间】:2020-11-10 10:04:43
【问题描述】:

我有一个uint8_t* 缓冲区,我通过buf.data() 获得。它是一个 IP 数据包的缓冲区。

我想更改源地址并重新计算校验和,因此我通过执行您在下面看到的操作在 libtins 中重新创建了数据包:

//prints original packet  
for (size_t i = 0; i < buf.size(); i++)
{
    printf("%x ", buf.data()[i]);
}
printf("\n");

Tins::IP pa = Tins::IP(buf.data(), buf.size());
pa.src_addr(newIpv4address);
uint8_t *packetData = pa.serialize().data();

//prints libtins packet created from original packet
for (size_t i = 0; i < buf.size(); i++)
{
    printf("%x ", packetData[i]);
}
printf("\n");

问题是从 libtins 打印的数据包出现了不同的标头:

45 0 0 34 0 0 40 0 40 6 6b 53 c0 a8 45 1 ac d9 1c ee 0 20 0 50 0 0 0 0 0 0 0 0 80 2 fd e8 a5 4d 0 0 2 4 5 b4 3 3 0 4 2 0 0 0 

60 b 0 34 62 76 0 0 d0 8 0 28 62 76 0 0 ac d9 1c ee 0 20 0 50 0 0 0 0 0 0 0 0 80 2 fd e8 eb 47 0 0 2 4 5 b4 3 3 0 4 2 0 0 0

然后我尝试用新数据包更改buf,即使它看起来不同:

uint8_t* b = buf.data();
uint8_t* o = pa.serialize().data();

for (int i = 0; i < buf.size(); i++)
{
    b[i] = o[i];
}

auto p = Tins::IP(buf.data(), buf.size());

但是在线auto p = Tins::IP(buf.data(), buf.size());我得到:

terminate called after throwing an instance of 'Tins::malformed_packet'
  what():  Malformed packet

这很奇怪,因为它实际上是从 libtins 生成的数据包。

我做错了什么?根据libtins documentation,我可以从缓冲区和大小创建一个 libtins 数据包。

更新:

#include <iostream>
#include <memory>
#include <tins/ip.h>

int main()
{
    const int packet_size = 52;
    uint8_t ip_packet[] = {
        69, 0, 0, 52, 0, 0, 64, 0, 64, 6, 107,
        83, 192, 168, 69, 1, 172, 217, 28, 238,
        0, 112, 0, 80, 0, 0, 0, 0, 0, 0, 0, 0, 128,
        2, 253, 232, 164, 253, 0, 0, 2, 4, 5, 180,
        3, 3, 0, 4, 2, 0, 0, 0};

    std::cout << "packet BEFORE " << std::endl;
    for (size_t i = 0; i < packet_size; i++)
    {
        printf("%x ", ip_packet[i]);
    }

    uint8_t b[52];
    for (int i = 0; i < packet_size; i++)
    {
        b[i] = ip_packet[i];
    }
    printf("\n");

    Tins::IP pa = Tins::IP(b, packet_size);
    pa.src_addr("172.78.123.76");
    uint8_t *packetData = pa.serialize().data();
    std::cout << "TINS packet: " << std::endl;
    for (size_t i = 0; i < packet_size; i++)
    {
        printf("%x ", packetData[i]);
    }
    printf("\n");

    return 0;
}

输出:

packet BEFORE 
45 0 0 34 0 0 40 0 40 6 6b 53 c0 a8 45 1 ac d9 1c ee 0 70 0 50 0 0 0 0 0 0 0 0 80 2 fd e8 a4 fd 0 0 2 4 5 b4 3 3 0 4 2 0 0 0 
TINS packet: 
0 0 0 0 0 0 0 0 10 90 68 1 0 0 0 0 ac d9 1c ee 0 70 0 50 0 0 0 0 0 0 0 0 80 2 fd e8 83 c 0 0 2 4 5 b4 3 3 0 4 2 0 0 0 

【问题讨论】:

  • 你的newIpv4address喜欢什么?
  • @NutCracker 一个简单的字符串“172.78.123.76”
  • 我尝试了与您在此处所做的相同的操作,但在 ip_test.cpp 中使用了 expected_packet 变量,并且一切正常
  • @NutCracker 对不起,expected_packet 这个名字和它有什么关系?
  • 和你使用 buf 数据包一样,我已经重用了位于 ip_test.cpp 中的 expected_packet 变量中的字节。你能告诉我你的buf 内容吗?

标签: c++ c++11 networking ip libtins


【解决方案1】:

使用您提供的示例 IP 数据包,我已完成以下操作:

uint8_t ip_packet[] = {
    69,  0,   0,   52,  0,   0,   64, 0, 
    64,  6,   107, 83,  192, 168, 69, 1,
    172, 217, 28,  238, 0,   112, 0,  80,
    0,   0,   0,   0,   0,   0,   0,  0,
    128, 2,   253, 232, 164, 253, 0,  0,
    2,   4,   5,   180, 3,   3,   0,  4,
    2,   0,   0,   0
};

Tins::IP ip(ip_packet, sizeof(ip_packet));
for (std::size_t i = 0; i < sizeof(ip_packet); i++)
    std::cout << +ip_packet[i] << " ";

std::cout << std::endl;

ip.src_addr("192.155.32.10");

auto data = ip.serialize();
for (std::size_t i = 0; i < data.size(); i++)
    std::cout << +data[i] << " ";

std::cout << std::endl;

我得到以下输出:

69 0 0 52 0 0 64 0 64 6 107 83 192 168 69 1 172 217 28 238 0 112 0 80 0 0 0 0 0 0 0 0 128 2 253 232 164 253 0 0 2 4 5 180 3 3 0 4 2 0 0 0
69 0 0 52 0 0 64 0 64 6 144 87 192 155 32 10 172 217 28 238 0 112 0 80 0 0 0 0 0 0 0 0 128 2 253 232 202 1 0 0 2 4 5 180 3 3 0 4 2 0 0 0

所以它按预期工作。

【讨论】:

  • 非常感谢!原来我所做的与你不同的是uint8_t *packetData = pa.serialize().data();,它很快就被破坏了,所以我们得到了未定义的行为
  • @LucasZanella 很高兴我能帮上忙
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2014-04-01
  • 2017-05-15
  • 2012-11-01
  • 2012-04-16
  • 2010-11-15
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多