Blame __dist_sample/sample/test_ipv4_options.c

Packit Service b25606
/*
Packit Service b25606
 * Regression test for bugs in ipv4 ip_offset and h_len handling, such as
Packit Service b25606
 *   http://bugs.debian.org/cgi-bin/bugreport.cgi?bug=418975
Packit Service b25606
 *
Packit Service b25606
 * Copyright (c) 2009 Sam Roberts <sroberts@wurldtech.com>
Packit Service b25606
 * All rights reserved.
Packit Service b25606
 *
Packit Service b25606
 * Redistribution and use in source and binary forms, with or without
Packit Service b25606
 * modification, are permitted provided that the following conditions
Packit Service b25606
 * are met:
Packit Service b25606
 * 1. Redistributions of source code must retain the above copyright
Packit Service b25606
 *    notice, this list of conditions and the following disclaimer.
Packit Service b25606
 * 2. Redistributions in binary form must reproduce the above copyright
Packit Service b25606
 *    notice, this list of conditions and the following disclaimer in the
Packit Service b25606
 *    documentation and/or other materials provided with the distribution.
Packit Service b25606
 *
Packit Service b25606
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
Packit Service b25606
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
Packit Service b25606
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
Packit Service b25606
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
Packit Service b25606
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
Packit Service b25606
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
Packit Service b25606
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
Packit Service b25606
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
Packit Service b25606
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
Packit Service b25606
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
Packit Service b25606
 * SUCH DAMAGE.
Packit Service b25606
 *
Packit Service b25606
 */
Packit Service b25606
#if (HAVE_CONFIG_H)
Packit Service b25606
#include "../include/config.h"
Packit Service b25606
#endif
Packit Service b25606
#include "./libnet_test.h"
Packit Service b25606
Packit Service b25606
#include <assert.h>
Packit Service b25606
Packit Service b25606
static void assert_eq_(long have, long want, const char* file, int line) {
Packit Service b25606
    if(have != want) {
Packit Service b25606
        printf("%s:%d: fail - have %ld want %ld\n", file, line, have, want);
Packit Service b25606
        abort();
Packit Service b25606
    }
Packit Service b25606
}
Packit Service b25606
#define assert_eq(have, want) assert_eq_(have, want, __FILE__, __LINE__)
Packit Service b25606
Packit Service b25606
Packit Service b25606
static void print_pblocks(libnet_t* l)
Packit Service b25606
{
Packit Service b25606
    libnet_pblock_t* p = l->protocol_blocks;
Packit Service b25606
Packit Service b25606
    while(p) {
Packit Service b25606
        printf("  tag %2d flags %d type %20s/%#x buf %p b_len %2u h_len %2u copied %2u\n",
Packit Service b25606
                p->ptag, p->flags,
Packit Service b25606
                libnet_diag_dump_pblock_type(p->type), p->type,
Packit Service b25606
                p->buf, p->b_len, p->h_len, p->copied);
Packit Service b25606
        p = p->next;
Packit Service b25606
    }
Packit Service b25606
    printf("  link_offset %d aligner %d total_size %u nblocks %d\n",
Packit Service b25606
            l->link_offset, l->aligner, l->total_size, l->n_pblocks);
Packit Service b25606
Packit Service b25606
}
Packit Service b25606
Packit Service b25606
static void ptag_error(libnet_t* l, int ptag)
Packit Service b25606
{
Packit Service b25606
    if(ptag <= 0) {
Packit Service b25606
        printf("error: %s\n", libnet_geterror(l));
Packit Service b25606
    }
Packit Service b25606
    assert(ptag > 0);
Packit Service b25606
}
Packit Service b25606
Packit Service b25606
static int build_ipo(libnet_t* l, libnet_ptag_t ptag, int payload_s)
Packit Service b25606
{
Packit Service b25606
    uint8_t* payload = malloc(payload_s);
Packit Service b25606
    assert(payload);
Packit Service b25606
    memset(payload, '\x88', payload_s);
Packit Service b25606
Packit Service b25606
    ptag = libnet_build_ipv4_options(payload, payload_s, l, ptag);
Packit Service b25606
Packit Service b25606
    ptag_error(l, ptag);
Packit Service b25606
Packit Service b25606
    free(payload);
Packit Service b25606
Packit Service b25606
    return ptag;
Packit Service b25606
}
Packit Service b25606
Packit Service b25606
static int build_ipv4(libnet_t* l, libnet_ptag_t ptag, int payload_s, int ip_len)
Packit Service b25606
{
Packit Service b25606
    u_long src_ip = 0xf101f1f1;
Packit Service b25606
    u_long dst_ip = 0xf102f1f1;
Packit Service b25606
    uint8_t* payload = malloc(payload_s);
Packit Service b25606
    assert(payload);
Packit Service b25606
    memset(payload, '\x99', payload_s);
Packit Service b25606
Packit Service b25606
    if(!ip_len) {
Packit Service b25606
        ip_len = LIBNET_IPV4_H + payload_s;
Packit Service b25606
    }
Packit Service b25606
Packit Service b25606
    ptag = libnet_build_ipv4(
Packit Service b25606
        ip_len,                                     /* length */
Packit Service b25606
        0,                                          /* TOS */
Packit Service b25606
        0xbbbb,                                     /* IP ID */
Packit Service b25606
        0,                                          /* IP Frag */
Packit Service b25606
        0xcc,                                       /* TTL */
Packit Service b25606
        IPPROTO_UDP,                                /* protocol */
Packit Service b25606
        0,                                          /* checksum */
Packit Service b25606
        src_ip,                                     /* source IP */
Packit Service b25606
        dst_ip,                                     /* destination IP */
Packit Service b25606
        payload_s ? payload : NULL,                 /* payload */
Packit Service b25606
        payload_s,                                  /* payload size */
Packit Service b25606
        l,                                          /* libnet handle */
Packit Service b25606
        ptag);                                      /* libnet id */
Packit Service b25606
Packit Service b25606
    ptag_error(l, ptag);
Packit Service b25606
Packit Service b25606
    free(payload);
Packit Service b25606
Packit Service b25606
    return ptag;
Packit Service b25606
}
Packit Service b25606
Packit Service b25606
static int build_ethernet(libnet_t* l, libnet_ptag_t ptag)
Packit Service b25606
{
Packit Service b25606
    uint8_t enet_src[6] = {0x11, 0x11, 0x11, 0x11, 0x11, 0x11};
Packit Service b25606
    uint8_t enet_dst[6] = {0x22, 0x22, 0x22, 0x22, 0x22, 0x22};
Packit Service b25606
Packit Service b25606
    ptag = libnet_build_ethernet(
Packit Service b25606
        enet_dst,                                   /* ethernet destination */
Packit Service b25606
        enet_src,                                   /* ethernet source */
Packit Service b25606
        ETHERTYPE_IP,                               /* protocol type */
Packit Service b25606
        NULL,                                       /* payload */
Packit Service b25606
        0,                                          /* payload size */
Packit Service b25606
        l,                                          /* libnet handle */
Packit Service b25606
        ptag);                                      /* libnet id */
Packit Service b25606
Packit Service b25606
    ptag_error(l, ptag);
Packit Service b25606
Packit Service b25606
    return ptag;
Packit Service b25606
}
Packit Service b25606
Packit Service b25606
static
Packit Service b25606
void assert_lengths(libnet_t* l, int ip_len, int ip_ihl, int payload_s)
Packit Service b25606
{
Packit Service b25606
    uint8_t* pkt1 = NULL;
Packit Service b25606
    uint32_t pkt1_sz = 0;
Packit Service b25606
    struct libnet_ipv4_hdr* h1;
Packit Service b25606
    uint8_t* payload = NULL;
Packit Service b25606
Packit Service b25606
Packit Service b25606
    int r = libnet_pblock_coalesce(l, &pkt1, &pkt1_sz);
Packit Service b25606
    assert(r >= 0);
Packit Service b25606
Packit Service b25606
    print_pblocks(l);
Packit Service b25606
Packit Service b25606
    libnet_diag_dump_hex(pkt1, 14, 1, stdout);
Packit Service b25606
    libnet_diag_dump_hex(pkt1+14, pkt1_sz-14, 1, stdout);
Packit Service b25606
Packit Service b25606
    /* check ip IHL value, total ip pkt length, and options value */
Packit Service b25606
    h1 = (struct libnet_ipv4_hdr*) (pkt1+14);
Packit Service b25606
    assert_eq(h1->ip_hl, ip_ihl); 
Packit Service b25606
    assert_eq(ntohs(h1->ip_len), ip_len);
Packit Service b25606
Packit Service b25606
    payload = ((uint8_t*) h1) + ip_ihl * 4;
Packit Service b25606
    if(payload_s > 0) {
Packit Service b25606
        assert(payload[0] == (uint8_t)'\x99');
Packit Service b25606
        assert(payload[payload_s-1] == (uint8_t)'\x99');
Packit Service b25606
    }
Packit Service b25606
}
Packit Service b25606
Packit Service b25606
int
Packit Service b25606
main(int argc, char *argv[])
Packit Service b25606
{
Packit Service b25606
    libnet_t *l;
Packit Service b25606
    char *device = "eth0";
Packit Service b25606
    char errbuf[LIBNET_ERRBUF_SIZE];
Packit Service b25606
    libnet_ptag_t ipo_ptag = 0;
Packit Service b25606
    libnet_ptag_t ip_ptag = 0;
Packit Service b25606
    libnet_ptag_t eth_ptag = 0;
Packit Service b25606
    int ip_len = 0;
Packit Service b25606
Packit Service b25606
    l = libnet_init( LIBNET_LINK, device, errbuf);
Packit Service b25606
Packit Service b25606
    assert(l);
Packit Service b25606
Packit Service b25606
    printf("Packet: options=4, payload=0\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 4 + 0; /* ip + options + payload */
Packit Service b25606
    ipo_ptag = build_ipo(l, ipo_ptag, 4);
Packit Service b25606
    ip_ptag = build_ipv4(l, ip_ptag, 0, 24);
Packit Service b25606
    eth_ptag = build_ethernet(l, eth_ptag);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, 24, 6, 0);
Packit Service b25606
Packit Service b25606
    ipo_ptag = ip_ptag = eth_ptag = 0;
Packit Service b25606
Packit Service b25606
    libnet_clear_packet(l);
Packit Service b25606
Packit Service b25606
    printf("Packet: options=3, payload=1\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 4 + 1; /* ip + options + payload */
Packit Service b25606
    ipo_ptag = build_ipo(l, ipo_ptag, 3);
Packit Service b25606
    ip_ptag = build_ipv4(l, ip_ptag, 1, 25);
Packit Service b25606
    eth_ptag = build_ethernet(l, eth_ptag);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, 25, 6, 1);
Packit Service b25606
Packit Service b25606
    ipo_ptag = ip_ptag = eth_ptag = 0;
Packit Service b25606
Packit Service b25606
    libnet_clear_packet(l);
Packit Service b25606
Packit Service b25606
    printf("Packet: options=3, payload=1\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 4 + 1; /* ip + options + payload */
Packit Service b25606
    ipo_ptag = build_ipo(l, ipo_ptag, 3);
Packit Service b25606
    ip_ptag = build_ipv4(l, ip_ptag, 1, ip_len);
Packit Service b25606
    eth_ptag = build_ethernet(l, eth_ptag);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, 25, 6, 1);
Packit Service b25606
Packit Service b25606
    printf("... modify -> options=40\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 40 + 1; /* ip + options + payload */
Packit Service b25606
    ipo_ptag = build_ipo(l, ipo_ptag, 40);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, ip_len, 15, 1);
Packit Service b25606
Packit Service b25606
    printf("... modify -> options=0\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 0 + 1; /* ip + options + payload */
Packit Service b25606
    ipo_ptag = build_ipo(l, ipo_ptag, 0);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, ip_len, 5, 1);
Packit Service b25606
Packit Service b25606
    printf("... modify -> options=5\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 8 + 1; /* ip + options + payload */
Packit Service b25606
    ipo_ptag = build_ipo(l, ipo_ptag, 5);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, ip_len, 7, 1);
Packit Service b25606
Packit Service b25606
    printf("... modify -> ip_payload=5\n");
Packit Service b25606
Packit Service b25606
    ip_len = 20 + 8 + 5; /* ip + options + payload */
Packit Service b25606
    ip_ptag = build_ipv4(l, ip_ptag, 5, ip_len);
Packit Service b25606
Packit Service b25606
    assert_lengths(l, ip_len, 7, 1);
Packit Service b25606
Packit Service b25606
    ipo_ptag = ip_ptag = eth_ptag = 0;
Packit Service b25606
Packit Service b25606
    libnet_clear_packet(l);
Packit Service b25606
Packit Service b25606
Packit Service b25606
    return (EXIT_SUCCESS);
Packit Service b25606
}
Packit Service b25606