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