144099b7bSPaul Traina /*
244099b7bSPaul Traina * getif.c : get an interface structure
344099b7bSPaul Traina */
444099b7bSPaul Traina
544099b7bSPaul Traina #include <sys/types.h>
644099b7bSPaul Traina #include <sys/socket.h>
744099b7bSPaul Traina #include <sys/ioctl.h>
844099b7bSPaul Traina
944099b7bSPaul Traina #if defined(SUNOS) || defined(SVR4)
1044099b7bSPaul Traina #include <sys/sockio.h>
1144099b7bSPaul Traina #endif
1244099b7bSPaul Traina #ifdef SVR4
1344099b7bSPaul Traina #include <sys/stropts.h>
1444099b7bSPaul Traina #endif
1544099b7bSPaul Traina
16e08ac58bSPaul Traina #include <sys/time.h> /* for struct timeval in net/if.h */
1744099b7bSPaul Traina #include <net/if.h> /* for struct ifreq */
1844099b7bSPaul Traina #include <netinet/in.h>
1944099b7bSPaul Traina
2044099b7bSPaul Traina #ifndef NO_UNISTD
2144099b7bSPaul Traina #include <unistd.h>
2244099b7bSPaul Traina #endif
2344099b7bSPaul Traina #include <syslog.h>
2444099b7bSPaul Traina #include <errno.h>
2544099b7bSPaul Traina #include <assert.h>
2644099b7bSPaul Traina
2744099b7bSPaul Traina #include "getif.h"
2844099b7bSPaul Traina #include "report.h"
2944099b7bSPaul Traina
3044099b7bSPaul Traina #ifdef __bsdi__
3144099b7bSPaul Traina #define BSD 43
3244099b7bSPaul Traina #endif
3344099b7bSPaul Traina
3444099b7bSPaul Traina static struct ifreq ifreq[10]; /* Holds interface configuration */
3544099b7bSPaul Traina static struct ifconf ifconf; /* points to ifreq */
3644099b7bSPaul Traina
3745c0d45bSRobert Watson static int nmatch(u_char *ca, u_char *cb);
3844099b7bSPaul Traina
3944099b7bSPaul Traina /* Return a pointer to the interface struct for the passed address. */
4044099b7bSPaul Traina struct ifreq *
getif(int s,struct in_addr * addrp)41*8b356c88SJohn Baldwin getif(int s, struct in_addr *addrp)
4244099b7bSPaul Traina {
4344099b7bSPaul Traina int maxmatch;
4444099b7bSPaul Traina int len, m, incr;
4544099b7bSPaul Traina struct ifreq *ifrq, *ifrmax;
4644099b7bSPaul Traina struct sockaddr_in *sip;
4744099b7bSPaul Traina char *p;
4844099b7bSPaul Traina
4944099b7bSPaul Traina /* If no address was supplied, just return NULL. */
5044099b7bSPaul Traina if (!addrp)
5144099b7bSPaul Traina return (struct ifreq *) 0;
5244099b7bSPaul Traina
5344099b7bSPaul Traina /* Get the interface config if not done already. */
5444099b7bSPaul Traina if (ifconf.ifc_len == 0) {
5544099b7bSPaul Traina #ifdef SVR4
5644099b7bSPaul Traina /*
5744099b7bSPaul Traina * SysVr4 returns garbage if you do this the obvious way!
5844099b7bSPaul Traina * This one took a while to figure out... -gwr
5944099b7bSPaul Traina */
6044099b7bSPaul Traina struct strioctl ioc;
6144099b7bSPaul Traina ioc.ic_cmd = SIOCGIFCONF;
6244099b7bSPaul Traina ioc.ic_timout = 0;
6344099b7bSPaul Traina ioc.ic_len = sizeof(ifreq);
6444099b7bSPaul Traina ioc.ic_dp = (char *) ifreq;
6544099b7bSPaul Traina m = ioctl(s, I_STR, (char *) &ioc);
6644099b7bSPaul Traina ifconf.ifc_len = ioc.ic_len;
6744099b7bSPaul Traina ifconf.ifc_req = ifreq;
6844099b7bSPaul Traina #else /* SVR4 */
6944099b7bSPaul Traina ifconf.ifc_len = sizeof(ifreq);
7044099b7bSPaul Traina ifconf.ifc_req = ifreq;
7144099b7bSPaul Traina m = ioctl(s, SIOCGIFCONF, (caddr_t) & ifconf);
7244099b7bSPaul Traina #endif /* SVR4 */
7344099b7bSPaul Traina if ((m < 0) || (ifconf.ifc_len <= 0)) {
7444099b7bSPaul Traina report(LOG_ERR, "ioctl SIOCGIFCONF");
7544099b7bSPaul Traina return (struct ifreq *) 0;
7644099b7bSPaul Traina }
7744099b7bSPaul Traina }
7844099b7bSPaul Traina maxmatch = 7; /* this many bits or less... */
7944099b7bSPaul Traina ifrmax = (struct ifreq *) 0;/* ... is not a valid match */
8044099b7bSPaul Traina p = (char *) ifreq;
8144099b7bSPaul Traina len = ifconf.ifc_len;
8244099b7bSPaul Traina while (len > 0) {
8344099b7bSPaul Traina ifrq = (struct ifreq *) p;
8444099b7bSPaul Traina sip = (struct sockaddr_in *) &ifrq->ifr_addr;
85c27501e7SAlexander Kabaev m = nmatch((u_char *)addrp, (u_char *)&(sip->sin_addr));
8644099b7bSPaul Traina if (m > maxmatch) {
8744099b7bSPaul Traina maxmatch = m;
8844099b7bSPaul Traina ifrmax = ifrq;
8944099b7bSPaul Traina }
904f5241c6SPaul Traina #ifndef IFNAMSIZ
9144099b7bSPaul Traina /* BSD not defined or earlier than 4.3 */
9244099b7bSPaul Traina incr = sizeof(*ifrq);
934f5241c6SPaul Traina #else
9444099b7bSPaul Traina incr = ifrq->ifr_addr.sa_len + IFNAMSIZ;
954f5241c6SPaul Traina #endif
9644099b7bSPaul Traina
9744099b7bSPaul Traina p += incr;
9844099b7bSPaul Traina len -= incr;
9944099b7bSPaul Traina }
10044099b7bSPaul Traina
10144099b7bSPaul Traina return ifrmax;
10244099b7bSPaul Traina }
10344099b7bSPaul Traina
10444099b7bSPaul Traina /*
10544099b7bSPaul Traina * Return the number of leading bits matching in the
10644099b7bSPaul Traina * internet addresses supplied.
10744099b7bSPaul Traina */
10844099b7bSPaul Traina static int
nmatch(u_char * ca,u_char * cb)109*8b356c88SJohn Baldwin nmatch(u_char *ca, u_char *cb)
11044099b7bSPaul Traina {
11144099b7bSPaul Traina u_int m = 0; /* count of matching bits */
11244099b7bSPaul Traina u_int n = 4; /* bytes left, then bitmask */
11344099b7bSPaul Traina
11444099b7bSPaul Traina /* Count matching bytes. */
11544099b7bSPaul Traina while (n && (*ca == *cb)) {
11644099b7bSPaul Traina ca++;
11744099b7bSPaul Traina cb++;
11844099b7bSPaul Traina m += 8;
11944099b7bSPaul Traina n--;
12044099b7bSPaul Traina }
12144099b7bSPaul Traina /* Now count matching bits. */
12244099b7bSPaul Traina if (n) {
12344099b7bSPaul Traina n = 0x80;
12444099b7bSPaul Traina while (n && ((*ca & n) == (*cb & n))) {
12544099b7bSPaul Traina m++;
12644099b7bSPaul Traina n >>= 1;
12744099b7bSPaul Traina }
12844099b7bSPaul Traina }
12944099b7bSPaul Traina return (m);
13044099b7bSPaul Traina }
13144099b7bSPaul Traina
13244099b7bSPaul Traina /*
13344099b7bSPaul Traina * Local Variables:
13444099b7bSPaul Traina * tab-width: 4
13544099b7bSPaul Traina * c-indent-level: 4
13644099b7bSPaul Traina * c-argdecl-indent: 4
13744099b7bSPaul Traina * c-continued-statement-offset: 4
13844099b7bSPaul Traina * c-continued-brace-offset: -4
13944099b7bSPaul Traina * c-label-offset: -4
14044099b7bSPaul Traina * c-brace-offset: 0
14144099b7bSPaul Traina * End:
14244099b7bSPaul Traina */
143