Blame src/libnet_if_addr.c

Packit 03b34a
/*
Packit 03b34a
 *  $Id: libnet_if_addr.c,v 1.23 2004/04/13 17:32:28 mike Exp $
Packit 03b34a
 *
Packit 03b34a
 *  libnet
Packit 03b34a
 *  libnet_if_addr.c - interface selection code
Packit 03b34a
 *
Packit 03b34a
 *  Copyright (c) 1998 - 2004 Mike D. Schiffman <mike@infonexus.com>
Packit 03b34a
 *  All rights reserved.
Packit 03b34a
 *
Packit 03b34a
 * Redistribution and use in source and binary forms, with or without
Packit 03b34a
 * modification, are permitted provided that the following conditions
Packit 03b34a
 * are met:
Packit 03b34a
 * 1. Redistributions of source code must retain the above copyright
Packit 03b34a
 *    notice, this list of conditions and the following disclaimer.
Packit 03b34a
 * 2. Redistributions in binary form must reproduce the above copyright
Packit 03b34a
 *    notice, this list of conditions and the following disclaimer in the
Packit 03b34a
 *    documentation and/or other materials provided with the distribution.
Packit 03b34a
 *
Packit 03b34a
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
Packit 03b34a
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
Packit 03b34a
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
Packit 03b34a
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
Packit 03b34a
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
Packit 03b34a
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
Packit 03b34a
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
Packit 03b34a
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
Packit 03b34a
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
Packit 03b34a
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
Packit 03b34a
 * SUCH DAMAGE.
Packit 03b34a
 *
Packit 03b34a
 */
Packit 03b34a
Packit 03b34a
#if (HAVE_CONFIG_H)
Packit 03b34a
#include "../include/config.h"
Packit 03b34a
#endif
Packit 03b34a
#if (!(_WIN32) || (__CYGWIN__)) 
Packit 03b34a
#include "../include/libnet.h"
Packit 03b34a
#else
Packit 03b34a
#include "../include/win32/libnet.h"
Packit 03b34a
#endif
Packit 03b34a
#ifdef HAVE_SYS_SOCKIO_H
Packit 03b34a
#include <sys/sockio.h>
Packit 03b34a
#endif
Packit 03b34a
#include "../include/ifaddrlist.h"
Packit 03b34a
Packit 03b34a
#define MAX_IPADDR 512
Packit 03b34a
Packit 03b34a
#if !(__WIN32__)
Packit 03b34a
Packit 03b34a
/*
Packit 03b34a
 * By testing if we can retrieve the FLAGS of an iface
Packit 03b34a
 * we can know if it exists or not and if it is up.
Packit 03b34a
 */
Packit 03b34a
int 
Packit 03b34a
libnet_check_iface(libnet_t *l)
Packit 03b34a
{
Packit 03b34a
    struct ifreq ifr;
Packit 03b34a
    int fd, res;
Packit 03b34a
Packit 03b34a
    fd = socket(AF_INET, SOCK_DGRAM, 0);
Packit 03b34a
    if (fd < 0)
Packit 03b34a
    {
Packit 03b34a
        snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s() socket: %s\n", __func__,
Packit 03b34a
                strerror(errno));
Packit 03b34a
        return (-1);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
    strncpy(ifr.ifr_name, l->device, sizeof(ifr.ifr_name) -1);
Packit 03b34a
    ifr.ifr_name[sizeof(ifr.ifr_name) - 1] = '\0';
Packit 03b34a
    
Packit 03b34a
    res = ioctl(fd, SIOCGIFFLAGS, (int8_t *)&ifr);
Packit 03b34a
Packit 03b34a
    if (res < 0)
Packit 03b34a
    {
Packit 03b34a
        snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s() ioctl: %s\n", __func__,
Packit 03b34a
                strerror(errno));
Packit 03b34a
    }
Packit 03b34a
    else
Packit 03b34a
    {
Packit 03b34a
        if ((ifr.ifr_flags & IFF_UP) == 0)
Packit 03b34a
        {
Packit 03b34a
            snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s(): %s is down\n",
Packit 03b34a
                    __func__, l->device);
Packit 03b34a
	    res = -1;
Packit 03b34a
        }
Packit 03b34a
    }
Packit 03b34a
    close(fd);
Packit 03b34a
    return (res);
Packit 03b34a
}
Packit 03b34a
Packit 03b34a
Packit 03b34a
/*
Packit 03b34a
 *  Return the interface list
Packit 03b34a
 */
Packit 03b34a
Packit 03b34a
#ifdef HAVE_SOCKADDR_SA_LEN
Packit 03b34a
#define NEXTIFR(i) \
Packit 03b34a
((struct ifreq *)((u_char *)&i->ifr_addr + i->ifr_addr.sa_len))
Packit 03b34a
#else
Packit 03b34a
#define NEXTIFR(i) (i + 1)
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
#ifndef BUFSIZE
Packit 03b34a
#define BUFSIZE 2048
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
#define PROC_DEV_FILE "/proc/net/dev"
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
int
Packit 03b34a
libnet_ifaddrlist(register struct libnet_ifaddr_list **ipaddrp, char *dev,
Packit 03b34a
register char *errbuf)
Packit 03b34a
{
Packit 03b34a
    register struct libnet_ifaddr_list *al;
Packit 03b34a
    struct ifreq *ifr, *lifr, *pifr, nifr;
Packit 03b34a
    char device[sizeof(nifr.ifr_name)];
Packit 03b34a
    static struct libnet_ifaddr_list ifaddrlist[MAX_IPADDR];
Packit 03b34a
    
Packit 03b34a
    char *p;
Packit 03b34a
    struct ifconf ifc;
Packit 03b34a
    struct ifreq ibuf[MAX_IPADDR];
Packit 03b34a
    register int fd, nipaddr;
Packit 03b34a
    
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
    FILE *fp;
Packit 03b34a
    char buf[2048];
Packit 03b34a
#endif
Packit 03b34a
    
Packit 03b34a
    fd = socket(AF_INET, SOCK_DGRAM, 0);
Packit 03b34a
    if (fd < 0)
Packit 03b34a
    {
Packit 03b34a
	snprintf(errbuf, LIBNET_ERRBUF_SIZE, "%s(): socket error: %s\n",
Packit 03b34a
                __func__, strerror(errno));
Packit 03b34a
	return (-1);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
    if ((fp = fopen(PROC_DEV_FILE, "r")) == NULL)
Packit 03b34a
    {
Packit 03b34a
	snprintf(errbuf, LIBNET_ERRBUF_SIZE,
Packit 03b34a
                "%s(): fopen(proc_dev_file) failed: %s\n",  __func__,
Packit 03b34a
                strerror(errno));
Packit 03b34a
	return (-1);
Packit 03b34a
    }
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
    memset(&ifc, 0, sizeof(ifc));
Packit 03b34a
    ifc.ifc_len = sizeof(ibuf);
Packit 03b34a
    ifc.ifc_buf = (caddr_t)ibuf;
Packit 03b34a
Packit 03b34a
    if(ioctl(fd, SIOCGIFCONF, &ifc) < 0)
Packit 03b34a
    {
Packit 03b34a
	snprintf(errbuf, LIBNET_ERRBUF_SIZE,
Packit 03b34a
                "%s(): ioctl(SIOCGIFCONF) error: %s\n", 
Packit 03b34a
                __func__, strerror(errno));
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
	fclose(fp);
Packit 03b34a
#endif
Packit 03b34a
	return(-1);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
    pifr = NULL;
Packit 03b34a
    lifr = (struct ifreq *)&ifc.ifc_buf[ifc.ifc_len];
Packit 03b34a
    
Packit 03b34a
    al = ifaddrlist;
Packit 03b34a
    nipaddr = 0;
Packit 03b34a
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
    while (fgets(buf, sizeof(buf), fp))
Packit 03b34a
    {
Packit 03b34a
	if ((p = strchr(buf, ':')) == NULL)
Packit 03b34a
        {
Packit 03b34a
            continue;
Packit 03b34a
        }
Packit 03b34a
        *p = '\0';
Packit 03b34a
        for(p = buf; *p == ' '; p++) ;
Packit 03b34a
	
Packit 03b34a
        strncpy(nifr.ifr_name, p, sizeof(nifr.ifr_name) - 1);
Packit 03b34a
        nifr.ifr_name[sizeof(nifr.ifr_name) - 1] = '\0';
Packit 03b34a
	
Packit 03b34a
#else /* !HAVE_LINUX_PROCFS */
Packit 03b34a
Packit 03b34a
    for (ifr = ifc.ifc_req; ifr < lifr; ifr = NEXTIFR(ifr))
Packit 03b34a
    {
Packit 03b34a
	/* XXX LINUX SOLARIS ifalias */
Packit 03b34a
	if((p = strchr(ifr->ifr_name, ':')))
Packit 03b34a
        {
Packit 03b34a
            *p='\0';
Packit 03b34a
        }
Packit 03b34a
	if (pifr && strcmp(ifr->ifr_name, pifr->ifr_name) == 0)
Packit 03b34a
        {
Packit 03b34a
            continue;
Packit 03b34a
        }
Packit 03b34a
	strncpy(nifr.ifr_name, ifr->ifr_name, sizeof(nifr.ifr_name) - 1);
Packit 03b34a
	nifr.ifr_name[sizeof(nifr.ifr_name) - 1] = '\0';
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
        /* save device name */
Packit 03b34a
        strncpy(device, nifr.ifr_name, sizeof(device) - 1);
Packit 03b34a
        device[sizeof(device) - 1] = '\0';
Packit 03b34a
Packit 03b34a
        if (ioctl(fd, SIOCGIFFLAGS, &nifr) < 0)
Packit 03b34a
        {
Packit 03b34a
            pifr = ifr;
Packit 03b34a
            continue;
Packit 03b34a
	}
Packit 03b34a
        if ((nifr.ifr_flags & IFF_UP) == 0)
Packit 03b34a
	{
Packit 03b34a
            pifr = ifr;
Packit 03b34a
            continue;	
Packit 03b34a
	}
Packit 03b34a
Packit 03b34a
        if (dev == NULL && LIBNET_ISLOOPBACK(&nifr))
Packit 03b34a
	{
Packit 03b34a
            pifr = ifr;
Packit 03b34a
            continue;
Packit 03b34a
	}
Packit 03b34a
	
Packit 03b34a
        strncpy(nifr.ifr_name, device, sizeof(device) - 1);
Packit 03b34a
        nifr.ifr_name[sizeof(nifr.ifr_name) - 1] = '\0';
Packit 03b34a
        if (ioctl(fd, SIOCGIFADDR, (int8_t *)&nifr) < 0)
Packit 03b34a
        {
Packit 03b34a
            if (errno != EADDRNOTAVAIL)
Packit 03b34a
            {
Packit 03b34a
                snprintf(errbuf, LIBNET_ERRBUF_SIZE,
Packit 03b34a
                        "%s(): SIOCGIFADDR: dev=%s: %s\n", __func__, device,
Packit 03b34a
                        strerror(errno));
Packit 03b34a
                close(fd);
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
                fclose(fp);
Packit 03b34a
#endif
Packit 03b34a
                return (-1);
Packit 03b34a
	    }
Packit 03b34a
            else /* device has no IP address => set to 0 */
Packit 03b34a
            {
Packit 03b34a
                al->addr = 0;
Packit 03b34a
            }
Packit 03b34a
        }
Packit 03b34a
        else
Packit 03b34a
        {
Packit 03b34a
            al->addr = ((struct sockaddr_in *)&nifr.ifr_addr)->sin_addr.s_addr;
Packit 03b34a
        }
Packit 03b34a
        
Packit 03b34a
        free(al->device);
Packit 03b34a
        al->device = NULL;
Packit 03b34a
Packit 03b34a
        if ((al->device = strdup(device)) == NULL)
Packit 03b34a
        {
Packit 03b34a
            snprintf(errbuf, LIBNET_ERRBUF_SIZE, 
Packit 03b34a
                    "%s(): strdup not enough memory\n", __func__);
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
            fclose(fp);
Packit 03b34a
#endif
Packit 03b34a
            return(-1);
Packit 03b34a
        }
Packit 03b34a
Packit 03b34a
        ++al;
Packit 03b34a
        ++nipaddr;
Packit 03b34a
Packit 03b34a
#ifndef HAVE_LINUX_PROCFS
Packit 03b34a
        pifr = ifr;
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
    } /* while|for */
Packit 03b34a
	
Packit 03b34a
#ifdef HAVE_LINUX_PROCFS
Packit 03b34a
    if (ferror(fp))
Packit 03b34a
    {
Packit 03b34a
        snprintf(errbuf, LIBNET_ERRBUF_SIZE,
Packit 03b34a
                "%s(): ferror: %s\n", __func__, strerror(errno));
Packit 03b34a
	fclose(fp);
Packit 03b34a
	return (-1);
Packit 03b34a
    }
Packit 03b34a
    fclose(fp);
Packit 03b34a
#endif
Packit 03b34a
Packit 03b34a
    *ipaddrp = ifaddrlist;
Packit 03b34a
    return (nipaddr);
Packit 03b34a
}
Packit 03b34a
#else
Packit 03b34a
/* From tcptraceroute, convert a numeric IP address to a string */
Packit 03b34a
#define IPTOSBUFFERS    12
Packit 03b34a
static int8_t *iptos(uint32_t in)
Packit 03b34a
{
Packit 03b34a
    static int8_t output[IPTOSBUFFERS][ 3 * 4 + 3 + 1];
Packit 03b34a
    static int16_t which;
Packit 03b34a
    uint8_t *p;
Packit 03b34a
Packit 03b34a
    p = (uint8_t *)∈
Packit 03b34a
    which = (which + 1 == IPTOSBUFFERS ? 0 : which + 1);
Packit 03b34a
    snprintf(output[which], IPTOSBUFFERS, "%d.%d.%d.%d", 
Packit 03b34a
            p[0], p[1], p[2], p[3]);
Packit 03b34a
    return output[which];
Packit 03b34a
}
Packit 03b34a
Packit 03b34a
int
Packit 03b34a
libnet_ifaddrlist(register struct libnet_ifaddr_list **ipaddrp, char *dev,
Packit 03b34a
register char *errbuf)
Packit 03b34a
{
Packit 03b34a
    int nipaddr = 0;    int i = 0;
Packit 03b34a
Packit 03b34a
    static struct libnet_ifaddr_list ifaddrlist[MAX_IPADDR];
Packit 03b34a
    pcap_if_t *alldevs;
Packit 03b34a
    pcap_if_t *d;
Packit 03b34a
    int8_t err[PCAP_ERRBUF_SIZE];
Packit 03b34a
Packit 03b34a
    /* Retrieve the interfaces list */
Packit 03b34a
    if (pcap_findalldevs(&alldevs, err) == -1)
Packit 03b34a
    {
Packit 03b34a
        snprintf(errbuf, LIBNET_ERRBUF_SIZE, 
Packit 03b34a
                "%s(): error in pcap_findalldevs: %s\n", __func__, err);
Packit 03b34a
        return (-1);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
    /* Scan the list printing every entry */
Packit 03b34a
	for (d = alldevs; d; d = d->next)
Packit 03b34a
    {
Packit 03b34a
		if((!d->addresses) || (d->addresses->addr->sa_family != AF_INET))
Packit 03b34a
            continue;
Packit 03b34a
        if(d->flags & PCAP_IF_LOOPBACK)
Packit 03b34a
            continue;
Packit 03b34a
    
Packit 03b34a
		/* XXX - strdup */
Packit 03b34a
        ifaddrlist[i].device = strdup(d->name);
Packit 03b34a
        ifaddrlist[i].addr = (uint32_t)
Packit 03b34a
                strdup(iptos(((struct sockaddr_in *)
Packit 03b34a
                d->addresses->addr)->sin_addr.s_addr));
Packit 03b34a
        ++i;
Packit 03b34a
        ++nipaddr;
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
    *ipaddrp = ifaddrlist;
Packit 03b34a
    return (nipaddr);
Packit 03b34a
}
Packit 03b34a
#endif /* __WIN32__ */
Packit 03b34a
Packit 03b34a
int
Packit 03b34a
libnet_select_device(libnet_t *l)
Packit 03b34a
{
Packit 03b34a
    int c, i;
Packit 03b34a
    char err_buf[LIBNET_ERRBUF_SIZE];
Packit 03b34a
    struct libnet_ifaddr_list *address_list, *al;
Packit 03b34a
    uint32_t addr;
Packit 03b34a
Packit 03b34a
Packit 03b34a
    if (l == NULL)
Packit 03b34a
    { 
Packit 03b34a
        return (-1);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
    if (l->device && !isdigit(l->device[0]))
Packit 03b34a
    {
Packit 03b34a
#if !(__WIN32__)
Packit 03b34a
	if (libnet_check_iface(l) < 0)
Packit 03b34a
	{
Packit 03b34a
            /* err msg set in libnet_check_iface() */
Packit 03b34a
	    return (-1);
Packit 03b34a
	}
Packit 03b34a
#endif
Packit 03b34a
	return (1);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
    /*
Packit 03b34a
     *  Number of interfaces.
Packit 03b34a
     */
Packit 03b34a
    c = libnet_ifaddrlist(&address_list, l->device, err_buf);
Packit 03b34a
    if (c < 0)
Packit 03b34a
    {
Packit 03b34a
        /* err msg set in libnet_ifaddrlist() */
Packit 03b34a
        return (-1);
Packit 03b34a
    }
Packit 03b34a
    else if (c == 0)
Packit 03b34a
    {
Packit 03b34a
        snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
Packit 03b34a
                "%s(): no network interface found\n", __func__);
Packit 03b34a
        return (-1);
Packit 03b34a
    }
Packit 03b34a
	
Packit 03b34a
    al = address_list;
Packit 03b34a
    if (l->device)
Packit 03b34a
    {
Packit 03b34a
        /*
Packit 03b34a
         *  Then we have an IP address in l->device => do lookup
Packit 03b34a
         */
Packit 03b34a
	addr = libnet_name2addr4(l, l->device, 0);
Packit 03b34a
Packit 03b34a
        for (i = c; i; --i, ++address_list)
Packit 03b34a
        {
Packit 03b34a
            if (((addr == -1) && !(strncmp(l->device, address_list->device,
Packit 03b34a
                   strlen(l->device)))) || 
Packit 03b34a
                    (address_list->addr == addr))
Packit 03b34a
            {
Packit 03b34a
                /* free the "user supplied device" - see libnet_init() */
Packit 03b34a
                free(l->device);
Packit 03b34a
                l->device =  strdup(address_list->device);
Packit 03b34a
                goto good;
Packit 03b34a
            }
Packit 03b34a
        }
Packit 03b34a
        if (i <= 0)
Packit 03b34a
        {
Packit 03b34a
            snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
Packit 03b34a
                    "%s(): can't find interface for IP %s\n", __func__,
Packit 03b34a
                    l->device);
Packit 03b34a
	    goto bad;
Packit 03b34a
        }
Packit 03b34a
    }
Packit 03b34a
    else
Packit 03b34a
    {
Packit 03b34a
        l->device = strdup(address_list->device);
Packit 03b34a
    }
Packit 03b34a
Packit 03b34a
good:
Packit 03b34a
    for (i = 0; i < c; i++)
Packit 03b34a
    {
Packit 03b34a
        free(al[i].device);
Packit 03b34a
        al[i].device = NULL;
Packit 03b34a
    }
Packit 03b34a
    return (1);
Packit 03b34a
Packit 03b34a
bad:
Packit 03b34a
    for (i = 0; i < c; i++)
Packit 03b34a
    {
Packit 03b34a
        free(al[i].device);
Packit 03b34a
        al[i].device = NULL;
Packit 03b34a
    }
Packit 03b34a
    return (-1);
Packit 03b34a
}
Packit 03b34a
Packit 03b34a
/* EOF */