如何通过RAW套接字发送修改后的IPv6报文?

时间:2015-07-15 00:49:32

标签: c linux ipv6 raw-sockets sendto

我正在尝试通过C Linux中的RAW套接字发送自定义IPv6标头 我已经使用IP_HDRINCL套接字选项在IPv4中取得了成功,但是,IPv6并不等效。
我找到了一种解决方法here,建议使用socket(AF_INET6, SOCK_RAW, IPPROTO_RAW)与启用IP_HDRINCL套接字选项具有相同的效果。

套接字已成功创建,在我使用sendto函数修改标头之前,我没有收到任何错误。

我像这样设置套接字:

static int socketFd = 0;
static struct sockaddr_in6 remote;

int main()
{
    socketFd = socket (PF_INET6, SOCK_RAW, IPPROTO_RAW);

    if (socketFd < 0)
    {
        printf ("An error ocurred while creating the socket.\n");
        exit (2);
    }

    remote.sin6_family = AF_INET6;
    remote.sin6_port = htons (25000);

    if (inet_pton (AF_INET6, "fd00:c0de::70d6:4ab9:115d:8cda", &(remote.sin6_addr)) != 1)
    {
        close (socketFd);
        printf ("Unable to parse IPv6 address.\n");
        exit (2);
    }

 /*More code */
  ...

  return 0;
}

然后,我有这个回调函数应该发送我的自定义IPv6数据包,但sendto无法返回EINVAL。

static void sendPacket ()
{
    char buffer[BUFSIZ];
    const size_t len = sizeof(struct ip6_hdr) + sizeof(struct UDP_hdr);
    struct ip6_hdr *ip6 = (struct ip6_hdr*) (buffer);
    struct UDP_hdr *udp = (struct UDP_hdr *) (buffer + sizeof(struct ip6_hdr));

    memset (buffer, 0, BUFSIZ);

    ip6->ip6_ctlun.ip6_un2_vfc = 0x60;
    ip6->ip6_dst = remote.sin6_addr;
    ip6->ip6_flow = 60;
    ip6->ip6_hops = 64;
    ip6->ip6_nxt = 17;
    ip6->ip6_plen = sizeof(struct UDP_hdr);

    if (inet_pton (AF_INET6, "fd00:c0de::62a4:4cff:1234:5678", &(ip6->ip6_src)) != 1)
    {
        printf ("Error while parsing spoofed IPv6 address.\n");
        return;
    }

    // Fabricate the UDP header. Source port number, redundant
    udp->uh_sport = htons (21000);
    // Destination port number
    udp->uh_dport = remote.sin6_port;
    udp->uh_ulen = htons (sizeof(struct UDP_hdr));

    if (sendto (socketFd, buffer, len, 0, (struct sockaddr *) &remote, sizeof(remote)) < 0)
    {
        perror ("sendto");
        printf ("Error while sending packet.\n");
    }
}

我调试了程序,ip6_hdr结构中的所有值似乎都是正确的,也是struct UDP_hdr中的值。我错过了什么吗?

2 个答案:

答案 0 :(得分:3)

好的,所以我找到了解决方案。对于任何挣扎于此的人来说,我解决它的方式是使用Pcap库(遵循this示例)。但是对于那些必须使用AF_PACKETS的人,我找到了一个非常完整的例子,它使用RAW套接字来实现这个link

答案 1 :(得分:3)

对于它可能有用的人,我得到了类似的代码,但是用

打开套接字
int val = 1; ret = setsockopt(test->state.sockfd, IPPROTO_IPV6, IPV6_HDRINCL, &val, sizeof (val));

使用

设置选项
remote.sin6_port = 0;

使用EINVAL将端口设置为零,如here中所述,以避免sendto来电时出现Context错误。< / p>