1

我正在尝试使用Libnet11功能:

int libnet_write_raw_ipv6 (libnet_t *l, u_int8_t *packet, u_int32_t size)

在网络层注入 IPv6 数据包。我创建了 IPv6 数据包并使用 Wireshark 捕获它。Wireshark 报告: 格式错误的数据包(wireshark 说 IPv6 中的下一个标头值是错误的,并且我认为有效负载大小太大)

我希望有人可以通过最少的代码示例帮助我,展示如何使用 libnet11 (libnet_write_raw_ipv6()) 手动构建 IPv6 数据包(带有 ICMPv6 扩展标头)。

我假设最小的代码可能如下所示:

packet_len = 40 + 16; // 40B ~ IPv6 packet, 16B ~ ICMPv6 header
u_char *buf = NULL;

struct ip6_hdr *ip6 = NULL;
struct icmp6_hdr *icmp6 = NULL;

l = libnet_init();
if ( (buf = malloc(packet_len)) == NULL ) {
    // error
}

// create IPv6 header
ip6 = (struct ip6_hdr *) buf;
ip6->ip6_flow   = 0;
ip6->ip6_vfc    = 6 << 4;
ip6->ip6_plen   = 16;              // ICMPv6 packet size
ip6->ip6_nxt    = IPPROTO_ICMPV6;  // 0x3a
ip6->ip6_hlim   = 64;
memcpy(&(ip6->ip6_src), &src_addr, sizeof(struct in6_addr));
memcpy(&(ip6->ip6_dst), &dst_addr, sizeof(struct in6_addr));

// create ICMPv6 header
icmp6 = (struct icmp6_hdr *) (buf + 40); // 40B ~ IPv6 packet size
icmp6->icmp6_type = ICMP6_ECHO_REQUEST;
icmp6->icmp6_code = 0;
icmp6->icmp6_cksum= 0;
icmp6->icmp6_data32[0] = 0;

libnet_do_checksum(l, (u_int8_t *)buf, IPPROTO_ICMPV6, packet_len);

written = libnet_write_raw_ipv6(l, buf, packet_len);
if ( written != packet_len )
    perror("Failed to send packet");

libnet_destroy(l);
free(buf);

我试图找到代码示例但没有成功。先感谢您。

马丁

4

2 回答 2

0

如果您使用的是 C++,那么我会推荐您libtins,这是一个制作嗅探库的数据包。这个简短的片段完全符合您的要求:

#include <tins/tins.h>

using namespace Tins;

void test(const IPv6Address &dst, const IPv6Address &src) {
    PacketSender sender;
    IPv6 ipv6 = IPv6(dst, src) / ICMPv6();
    ipv6.hop_limit(64);
    sender.send(ipv6);
}

int main() {
    // now use it
    test("f0ef:1234::1", "f000::1");
}
于 2013-02-07T12:58:32.647 回答
0

不过,您可以使用原始套接字创建它。我也必须做类似的事情,但找不到任何参考。

要使用原始套接字,此链接为您提供了一个很好的解释

于 2014-03-17T16:48:58.123 回答