Home | History | Annotate | Line # | Download | only in common
getif.c revision 1.4
      1 /*	$NetBSD: getif.c,v 1.4 1998/03/14 04:39:54 lukem Exp $	*/
      2 
      3 #include <sys/cdefs.h>
      4 #ifndef lint
      5 __RCSID("$NetBSD: getif.c,v 1.4 1998/03/14 04:39:54 lukem Exp $");
      6 #endif
      7 
      8 /*
      9  * getif.c : get an interface structure
     10  */
     11 
     12 #include <sys/types.h>
     13 #include <sys/socket.h>
     14 #include <sys/ioctl.h>
     15 #include <sys/param.h>
     16 
     17 #if defined(SUNOS) || defined(SVR4)
     18 #include <sys/sockio.h>
     19 #endif
     20 #ifdef	SVR4
     21 #include <sys/stropts.h>
     22 #endif
     23 
     24 #include <net/if.h>				/* for struct ifreq */
     25 #include <netinet/in.h>
     26 
     27 #ifndef	NO_UNISTD
     28 #include <unistd.h>
     29 #endif
     30 #include <syslog.h>
     31 #include <errno.h>
     32 #include <assert.h>
     33 
     34 #include "getif.h"
     35 #include "report.h"
     36 
     37 #ifdef	__bsdi__
     38 #define BSD 43
     39 #endif
     40 
     41 static struct ifreq ifreq[10];	/* Holds interface configuration */
     42 static struct ifconf ifconf;	/* points to ifreq */
     43 
     44 #ifdef	__STDC__
     45 #define P(args) args
     46 #else
     47 #define P(args) ()
     48 #endif
     49 
     50 static int nmatch P((u_char *, u_char *));
     51 
     52 #undef P
     53 
     54 /* Return a pointer to the interface struct for the passed address. */
     55 struct ifreq *
     56 getif(s, addrp)
     57 	int s;						/* socket file descriptor */
     58 	struct in_addr *addrp;		/* destination address on interface */
     59 {
     60 	int maxmatch;
     61 	int len, m, incr;
     62 	struct ifreq *ifrq, *ifrmax;
     63 	struct sockaddr_in *sip;
     64 	char *p;
     65 
     66 	/* If no address was supplied, just return NULL. */
     67 	if (!addrp)
     68 		return (struct ifreq *) 0;
     69 
     70 	/* Get the interface config if not done already. */
     71 	if (ifconf.ifc_len == 0) {
     72 #ifdef	SVR4
     73 		/*
     74 		 * SysVr4 returns garbage if you do this the obvious way!
     75 		 * This one took a while to figure out... -gwr
     76 		 */
     77 		struct strioctl ioc;
     78 		ioc.ic_cmd = SIOCGIFCONF;
     79 		ioc.ic_timout = 0;
     80 		ioc.ic_len = sizeof(ifreq);
     81 		ioc.ic_dp = (char *) ifreq;
     82 		m = ioctl(s, I_STR, (char *) &ioc);
     83 		ifconf.ifc_len = ioc.ic_len;
     84 		ifconf.ifc_req = ifreq;
     85 #else	/* SVR4 */
     86 		ifconf.ifc_len = sizeof(ifreq);
     87 		ifconf.ifc_req = ifreq;
     88 		m = ioctl(s, SIOCGIFCONF, (caddr_t) & ifconf);
     89 #endif	/* SVR4 */
     90 		if ((m < 0) || (ifconf.ifc_len <= 0)) {
     91 			report(LOG_ERR, "ioctl SIOCGIFCONF");
     92 			return (struct ifreq *) 0;
     93 		}
     94 	}
     95 	maxmatch = 7;				/* this many bits or less... */
     96 	ifrmax = (struct ifreq *) 0;/* ... is not a valid match  */
     97 	p = (char *) ifreq;
     98 	len = ifconf.ifc_len;
     99 	while (len > 0) {
    100 		ifrq = (struct ifreq *) p;
    101 		sip = (struct sockaddr_in *) &ifrq->ifr_addr;
    102 		m = nmatch((u_char *)addrp, (u_char *)&(sip->sin_addr));
    103 		if (m > maxmatch) {
    104 			maxmatch = m;
    105 			ifrmax = ifrq;
    106 		}
    107 		/* XXX - Could this be just #ifndef IFNAMSIZ instead? -gwr */
    108 #if (BSD - 0) < 43
    109 		/* BSD not defined or earlier than 4.3 */
    110 		incr = sizeof(*ifrq);
    111 #else /* NetBSD */
    112 		incr = ifrq->ifr_addr.sa_len + IFNAMSIZ;
    113 #endif /* NetBSD */
    114 
    115 		p += incr;
    116 		len -= incr;
    117 	}
    118 
    119 	return ifrmax;
    120 }
    121 
    122 /*
    123  * Return the number of leading bits matching in the
    124  * internet addresses supplied.
    125  */
    126 static int
    127 nmatch(ca, cb)
    128 	u_char *ca, *cb;			/* ptrs to IP address, network order */
    129 {
    130 	u_int m = 0;				/* count of matching bits */
    131 	u_int n = 4;				/* bytes left, then bitmask */
    132 
    133 	/* Count matching bytes. */
    134 	while (n && (*ca == *cb)) {
    135 		ca++;
    136 		cb++;
    137 		m += 8;
    138 		n--;
    139 	}
    140 	/* Now count matching bits. */
    141 	if (n) {
    142 		n = 0x80;
    143 		while (n && ((*ca & n) == (*cb & n))) {
    144 			m++;
    145 			n >>= 1;
    146 		}
    147 	}
    148 	return (m);
    149 }
    150 
    151 /*
    152  * Local Variables:
    153  * tab-width: 4
    154  * c-indent-level: 4
    155  * c-argdecl-indent: 4
    156  * c-continued-statement-offset: 4
    157  * c-continued-brace-offset: -4
    158  * c-label-offset: -4
    159  * c-brace-offset: 0
    160  * End:
    161  */
    162