Как отправить измененный пакет IPv6 через сокет RAW?

Я пытаюсь отправить собственный заголовок IPv6 через сокет RAW в C Linux.
Мне уже удалось использовать IPv4 с помощью параметра сокета IP_HDRINCL, однако для IPv6 нет эквивалента.
Я нашел здесь обходной путь, предлагающий использовать 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 кажутся правильными, а также значения в структуре UDP_hdr. Я что-то упускаю?


person Pepedou    schedule 15.07.2015    source источник


Ответы (2)


Кому это может быть полезно, у меня есть аналогичный код для работы, но вместо этого я открываю сокет с помощью

socket(test->state.addr_family, SOCK_RAW, IPPROTO_UDP);

Настройка параметров с помощью

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

и обнуление порта с помощью remote.sin6_port = 0;, как описано в здесь, чтобы избежать EINVAL ошибки при вызове sendto.

person Pedro Fernandes    schedule 12.12.2017

Итак, я нашел решение. Для всех, кто борется с этим, я решил это с помощью библиотеки Pcap (следуйте this пример). Однако для тех, кто должен использовать AF_PACKETS, я нашел очень полный пример, в котором для этого используются сокеты RAW, в этом ссылка.

person Pepedou    schedule 16.07.2015