xref: /freebsd/usr.sbin/ppp/route.c (revision af57ed9fdcc4e2a81940fc2243c0d3949fc0e00f)
1af57ed9fSAtsushi Murai /*
2af57ed9fSAtsushi Murai  *	      PPP Routing related Module
3af57ed9fSAtsushi Murai  *
4af57ed9fSAtsushi Murai  *	    Written by Toshiharu OHNO (tony-o@iij.ad.jp)
5af57ed9fSAtsushi Murai  *
6af57ed9fSAtsushi Murai  *   Copyright (C) 1994, Internet Initiative Japan, Inc. All rights reserverd.
7af57ed9fSAtsushi Murai  *
8af57ed9fSAtsushi Murai  * Redistribution and use in source and binary forms are permitted
9af57ed9fSAtsushi Murai  * provided that the above copyright notice and this paragraph are
10af57ed9fSAtsushi Murai  * duplicated in all such forms and that any documentation,
11af57ed9fSAtsushi Murai  * advertising materials, and other materials related to such
12af57ed9fSAtsushi Murai  * distribution and use acknowledge that the software was developed
13af57ed9fSAtsushi Murai  * by the Internet Initiative Japan, Inc.  The name of the
14af57ed9fSAtsushi Murai  * IIJ may not be used to endorse or promote products derived
15af57ed9fSAtsushi Murai  * from this software without specific prior written permission.
16af57ed9fSAtsushi Murai  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
17af57ed9fSAtsushi Murai  * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
18af57ed9fSAtsushi Murai  * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
19af57ed9fSAtsushi Murai  *
20af57ed9fSAtsushi Murai  * $Id:$
21af57ed9fSAtsushi Murai  */
22af57ed9fSAtsushi Murai #include <sys/types.h>
23af57ed9fSAtsushi Murai #include <machine/endian.h>
24af57ed9fSAtsushi Murai #include <sys/param.h>
25af57ed9fSAtsushi Murai #include <sys/socket.h>
26af57ed9fSAtsushi Murai #include <net/route.h>
27af57ed9fSAtsushi Murai #include <sys/ioctl.h>
28af57ed9fSAtsushi Murai #include <net/if.h>
29af57ed9fSAtsushi Murai #include <errno.h>
30af57ed9fSAtsushi Murai #include <netinet/in_systm.h>
31af57ed9fSAtsushi Murai #include <netinet/in.h>
32af57ed9fSAtsushi Murai #include <arpa/inet.h>
33af57ed9fSAtsushi Murai #if __FreeBSD__ >= 2
34af57ed9fSAtsushi Murai #include <osreldate.h>
35af57ed9fSAtsushi Murai #include <sys/sysctl.h>
36af57ed9fSAtsushi Murai #else
37af57ed9fSAtsushi Murai #include <sys/kinfo.h>
38af57ed9fSAtsushi Murai #endif
39af57ed9fSAtsushi Murai #include <stdlib.h>
40af57ed9fSAtsushi Murai #include <stdio.h>
41af57ed9fSAtsushi Murai #include <string.h>
42af57ed9fSAtsushi Murai #include <unistd.h>
43af57ed9fSAtsushi Murai 
44af57ed9fSAtsushi Murai static int IfIndex;
45af57ed9fSAtsushi Murai 
46af57ed9fSAtsushi Murai struct rtmsg {
47af57ed9fSAtsushi Murai   struct rt_msghdr m_rtm;
48af57ed9fSAtsushi Murai   char m_space[64];
49af57ed9fSAtsushi Murai };
50af57ed9fSAtsushi Murai 
51af57ed9fSAtsushi Murai static int seqno;
52af57ed9fSAtsushi Murai 
53af57ed9fSAtsushi Murai void
54af57ed9fSAtsushi Murai OsSetRoute(cmd, dst, gateway, mask)
55af57ed9fSAtsushi Murai int cmd;
56af57ed9fSAtsushi Murai struct in_addr dst;
57af57ed9fSAtsushi Murai struct in_addr gateway;
58af57ed9fSAtsushi Murai struct in_addr mask;
59af57ed9fSAtsushi Murai {
60af57ed9fSAtsushi Murai   struct rtmsg rtmes;
61af57ed9fSAtsushi Murai   int s, nb, wb;
62af57ed9fSAtsushi Murai   char *cp;
63af57ed9fSAtsushi Murai   u_long *lp;
64af57ed9fSAtsushi Murai   struct sockaddr_in rtdata;
65af57ed9fSAtsushi Murai 
66af57ed9fSAtsushi Murai   s = socket(PF_ROUTE, SOCK_RAW, 0);
67af57ed9fSAtsushi Murai   if (s < 0)
68af57ed9fSAtsushi Murai     logprintf("socket\n");
69af57ed9fSAtsushi Murai 
70af57ed9fSAtsushi Murai   bzero(&rtmes, sizeof(rtmes));
71af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_version = RTM_VERSION;
72af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_type = cmd;
73af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_addrs = RTA_DST | RTA_NETMASK;
74af57ed9fSAtsushi Murai   if (cmd == RTM_ADD) rtmes.m_rtm.rtm_addrs |= RTA_GATEWAY;
75af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_seq = ++seqno;
76af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_pid = getpid();
77af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_flags = RTF_UP | RTF_GATEWAY;
78af57ed9fSAtsushi Murai 
79af57ed9fSAtsushi Murai   bzero(&rtdata, sizeof(rtdata));
80af57ed9fSAtsushi Murai   rtdata.sin_len = 16;
81af57ed9fSAtsushi Murai   rtdata.sin_family = AF_INET;
82af57ed9fSAtsushi Murai   rtdata.sin_port = 0;
83af57ed9fSAtsushi Murai   rtdata.sin_addr = dst;
84af57ed9fSAtsushi Murai 
85af57ed9fSAtsushi Murai   cp = rtmes.m_space;
86af57ed9fSAtsushi Murai   bcopy(&rtdata, cp, 16);
87af57ed9fSAtsushi Murai   cp += 16;
88af57ed9fSAtsushi Murai   if (gateway.s_addr) {
89af57ed9fSAtsushi Murai     rtdata.sin_addr = gateway;
90af57ed9fSAtsushi Murai     bcopy(&rtdata, cp, 16);
91af57ed9fSAtsushi Murai     cp += 16;
92af57ed9fSAtsushi Murai   }
93af57ed9fSAtsushi Murai 
94af57ed9fSAtsushi Murai   if (dst.s_addr == INADDR_ANY)
95af57ed9fSAtsushi Murai     mask.s_addr = INADDR_ANY;
96af57ed9fSAtsushi Murai 
97af57ed9fSAtsushi Murai   lp = (u_long *)cp;
98af57ed9fSAtsushi Murai 
99af57ed9fSAtsushi Murai   if (mask.s_addr) {
100af57ed9fSAtsushi Murai     *lp++ = 8;
101af57ed9fSAtsushi Murai     cp += sizeof(int);
102af57ed9fSAtsushi Murai     *lp = mask.s_addr;
103af57ed9fSAtsushi Murai   } else
104af57ed9fSAtsushi Murai     *lp = 0;
105af57ed9fSAtsushi Murai   cp += sizeof(u_long);
106af57ed9fSAtsushi Murai 
107af57ed9fSAtsushi Murai   nb = cp - (char *)&rtmes;
108af57ed9fSAtsushi Murai   rtmes.m_rtm.rtm_msglen = nb;
109af57ed9fSAtsushi Murai   wb = write(s, &rtmes, nb);
110af57ed9fSAtsushi Murai   if (wb < 0) {
111af57ed9fSAtsushi Murai     perror("write");
112af57ed9fSAtsushi Murai   }
113af57ed9fSAtsushi Murai #ifdef DEBUG
114af57ed9fSAtsushi Murai   logprintf("wrote %d: dst = %x, gateway = %x\n", nb, dst.s_addr, gateway.s_addr);
115af57ed9fSAtsushi Murai #endif
116af57ed9fSAtsushi Murai   close(s);
117af57ed9fSAtsushi Murai }
118af57ed9fSAtsushi Murai 
119af57ed9fSAtsushi Murai static void
120af57ed9fSAtsushi Murai p_sockaddr(sa, width)
121af57ed9fSAtsushi Murai struct sockaddr *sa;
122af57ed9fSAtsushi Murai int width;
123af57ed9fSAtsushi Murai {
124af57ed9fSAtsushi Murai   register char *cp;
125af57ed9fSAtsushi Murai   register struct sockaddr_in *sin = (struct sockaddr_in *)sa;
126af57ed9fSAtsushi Murai 
127af57ed9fSAtsushi Murai   cp = (sin->sin_addr.s_addr == 0) ? "default" :
128af57ed9fSAtsushi Murai 	   inet_ntoa(sin->sin_addr);
129af57ed9fSAtsushi Murai   printf("%-*.*s ", width, width, cp);
130af57ed9fSAtsushi Murai }
131af57ed9fSAtsushi Murai 
132af57ed9fSAtsushi Murai struct bits {
133af57ed9fSAtsushi Murai   short b_mask;
134af57ed9fSAtsushi Murai   char  b_val;
135af57ed9fSAtsushi Murai } bits[] = {
136af57ed9fSAtsushi Murai   { RTF_UP,	  'U' },
137af57ed9fSAtsushi Murai   { RTF_GATEWAY,  'G' },
138af57ed9fSAtsushi Murai   { RTF_HOST,	  'H' },
139af57ed9fSAtsushi Murai   { RTF_DYNAMIC,  'D' },
140af57ed9fSAtsushi Murai   { RTF_MODIFIED, 'M' },
141af57ed9fSAtsushi Murai   { RTF_CLONING,  'C' },
142af57ed9fSAtsushi Murai   { RTF_XRESOLVE, 'X' },
143af57ed9fSAtsushi Murai   { RTF_LLINFO,   'L' },
144af57ed9fSAtsushi Murai   { RTF_REJECT,   'R' },
145af57ed9fSAtsushi Murai   { 0 }
146af57ed9fSAtsushi Murai };
147af57ed9fSAtsushi Murai 
148af57ed9fSAtsushi Murai static void
149af57ed9fSAtsushi Murai p_flags(f, format)
150af57ed9fSAtsushi Murai register int f;
151af57ed9fSAtsushi Murai char *format;
152af57ed9fSAtsushi Murai {
153af57ed9fSAtsushi Murai   char name[33], *flags;
154af57ed9fSAtsushi Murai   register struct bits *p = bits;
155af57ed9fSAtsushi Murai 
156af57ed9fSAtsushi Murai   for (flags = name; p->b_mask; p++)
157af57ed9fSAtsushi Murai     if (p->b_mask & f)
158af57ed9fSAtsushi Murai       *flags++ = p->b_val;
159af57ed9fSAtsushi Murai   *flags = '\0';
160af57ed9fSAtsushi Murai   printf(format, name);
161af57ed9fSAtsushi Murai }
162af57ed9fSAtsushi Murai 
163af57ed9fSAtsushi Murai int
164af57ed9fSAtsushi Murai ShowRoute()
165af57ed9fSAtsushi Murai {
166af57ed9fSAtsushi Murai   struct rt_msghdr *rtm;
167af57ed9fSAtsushi Murai   struct sockaddr *sa;
168af57ed9fSAtsushi Murai   char *sp, *ep, *cp;
169af57ed9fSAtsushi Murai   u_char *wp;
170af57ed9fSAtsushi Murai   int *lp;
171af57ed9fSAtsushi Murai   int needed, nb;
172af57ed9fSAtsushi Murai   u_long mask;
173af57ed9fSAtsushi Murai #if ( __FreeBSD_version >= 199412 )
174af57ed9fSAtsushi Murai   int 	mib[6];
175af57ed9fSAtsushi Murai #endif
176af57ed9fSAtsushi Murai 
177af57ed9fSAtsushi Murai #if ( __FreeBSD_version >= 199412 )
178af57ed9fSAtsushi Murai   mib[0] = CTL_NET;
179af57ed9fSAtsushi Murai   mib[1] = PF_ROUTE;
180af57ed9fSAtsushi Murai   mib[2] = 0;             /* protocol */
181af57ed9fSAtsushi Murai   mib[3] = 0;             /* wildcard address family */
182af57ed9fSAtsushi Murai   mib[4] = NET_RT_DUMP;
183af57ed9fSAtsushi Murai   mib[5] = 0;             /* no flags */
184af57ed9fSAtsushi Murai   needed = sysctl(mib, 6, NULL, &needed, NULL, 0 );
185af57ed9fSAtsushi Murai #else
186af57ed9fSAtsushi Murai   needed = getkerninfo(KINFO_RT_DUMP, 0, 0, 0);
187af57ed9fSAtsushi Murai #endif
188af57ed9fSAtsushi Murai   if (needed < 0)
189af57ed9fSAtsushi Murai     return(1);
190af57ed9fSAtsushi Murai   sp = malloc(needed);
191af57ed9fSAtsushi Murai   if (sp == NULL)
192af57ed9fSAtsushi Murai     return(1);
193af57ed9fSAtsushi Murai #if ( __FreeBSD_version >= 199412 )
194af57ed9fSAtsushi Murai   if (sysctl(mib, 6, sp, &needed, NULL, 0 ) < 0)
195af57ed9fSAtsushi Murai     return(1);
196af57ed9fSAtsushi Murai #else
197af57ed9fSAtsushi Murai   if (getkerninfo(KINFO_RT_DUMP, sp, &needed, 0) < 0)
198af57ed9fSAtsushi Murai     return(1);
199af57ed9fSAtsushi Murai #endif
200af57ed9fSAtsushi Murai   ep = sp + needed;
201af57ed9fSAtsushi Murai 
202af57ed9fSAtsushi Murai   for (cp = sp; cp < ep; cp += rtm->rtm_msglen) {
203af57ed9fSAtsushi Murai     rtm = (struct rt_msghdr *)cp;
204af57ed9fSAtsushi Murai     sa = (struct sockaddr *)(rtm + 1);
205af57ed9fSAtsushi Murai     mask = 0xffffffff;
206af57ed9fSAtsushi Murai     if (rtm->rtm_addrs == RTA_DST)
207af57ed9fSAtsushi Murai       p_sockaddr(sa, 36);
208af57ed9fSAtsushi Murai     else {
209af57ed9fSAtsushi Murai       wp = (u_char *)cp + rtm->rtm_msglen;
210af57ed9fSAtsushi Murai       p_sockaddr(sa, 16);
211af57ed9fSAtsushi Murai       if (sa->sa_len == 0)
212af57ed9fSAtsushi Murai 	sa->sa_len = sizeof(long);
213af57ed9fSAtsushi Murai       sa = (struct sockaddr *)(sa->sa_len + (char *)sa);
214af57ed9fSAtsushi Murai       p_sockaddr(sa, 18);
215af57ed9fSAtsushi Murai       lp = (int *)(sa->sa_len + (char *)sa);
216af57ed9fSAtsushi Murai       if ((char *)lp < (char *)wp && *lp) {
217af57ed9fSAtsushi Murai #ifdef DEBUG
218af57ed9fSAtsushi Murai 	logprintf(" flag = %x, rest = %d", rtm->rtm_flags, *lp);
219af57ed9fSAtsushi Murai #endif
220af57ed9fSAtsushi Murai 	wp = (u_char *)(lp + 1);
221af57ed9fSAtsushi Murai 	mask = 0;
222af57ed9fSAtsushi Murai 	for (nb = *lp; nb > 4; nb--) {
223af57ed9fSAtsushi Murai 	  mask <<= 8;
224af57ed9fSAtsushi Murai 	  mask |= *wp++;
225af57ed9fSAtsushi Murai 	}
226af57ed9fSAtsushi Murai 	for (nb = 8 - *lp; nb > 0; nb--)
227af57ed9fSAtsushi Murai 	  mask <<= 8;
228af57ed9fSAtsushi Murai       }
229af57ed9fSAtsushi Murai     }
230af57ed9fSAtsushi Murai     printf("%08x  ", mask);
231af57ed9fSAtsushi Murai     p_flags(rtm->rtm_flags & (RTF_UP|RTF_GATEWAY|RTF_HOST), "%-6.6s ");
232af57ed9fSAtsushi Murai     printf("(%d)\n", rtm->rtm_index);
233af57ed9fSAtsushi Murai   }
234af57ed9fSAtsushi Murai 
235af57ed9fSAtsushi Murai   return(1);
236af57ed9fSAtsushi Murai }
237af57ed9fSAtsushi Murai 
238af57ed9fSAtsushi Murai /*
239af57ed9fSAtsushi Murai  *  Delete routes associated with our interface
240af57ed9fSAtsushi Murai  */
241af57ed9fSAtsushi Murai void
242af57ed9fSAtsushi Murai DeleteIfRoutes(all)
243af57ed9fSAtsushi Murai int all;
244af57ed9fSAtsushi Murai {
245af57ed9fSAtsushi Murai   struct rt_msghdr *rtm;
246af57ed9fSAtsushi Murai   struct sockaddr *sa;
247af57ed9fSAtsushi Murai   struct in_addr dstnet, gateway;
248af57ed9fSAtsushi Murai   int needed;
249af57ed9fSAtsushi Murai   char *sp, *cp, *ep;
250af57ed9fSAtsushi Murai   u_long mask;
251af57ed9fSAtsushi Murai   int *lp, nb;
252af57ed9fSAtsushi Murai   u_char *wp;
253af57ed9fSAtsushi Murai #if ( __FreeBSD_version >= 199412 )
254af57ed9fSAtsushi Murai   int 	mib[6];
255af57ed9fSAtsushi Murai #endif
256af57ed9fSAtsushi Murai 
257af57ed9fSAtsushi Murai #ifdef DEBUG
258af57ed9fSAtsushi Murai   logprintf("DeleteIfRoutes (%d)\n", IfIndex);
259af57ed9fSAtsushi Murai #endif
260af57ed9fSAtsushi Murai #if ( __FreeBSD_version >= 199412 )
261af57ed9fSAtsushi Murai   mib[0] = CTL_NET;
262af57ed9fSAtsushi Murai   mib[1] = PF_ROUTE;
263af57ed9fSAtsushi Murai   mib[2] = 0;             /* protocol */
264af57ed9fSAtsushi Murai   mib[3] = 0;             /* wildcard address family */
265af57ed9fSAtsushi Murai   mib[4] = NET_RT_DUMP;
266af57ed9fSAtsushi Murai   mib[5] = 0;             /* no flags */
267af57ed9fSAtsushi Murai   needed = sysctl(mib, 6, NULL, &needed, NULL, 0 );
268af57ed9fSAtsushi Murai #else
269af57ed9fSAtsushi Murai   needed = getkerninfo(KINFO_RT_DUMP, 0, 0, 0);
270af57ed9fSAtsushi Murai #endif
271af57ed9fSAtsushi Murai   if (needed < 0)
272af57ed9fSAtsushi Murai     return;
273af57ed9fSAtsushi Murai 
274af57ed9fSAtsushi Murai   sp = malloc(needed);
275af57ed9fSAtsushi Murai   if (sp == NULL)
276af57ed9fSAtsushi Murai     return;
277af57ed9fSAtsushi Murai 
278af57ed9fSAtsushi Murai #if ( __FreeBSD_version >= 199412 )
279af57ed9fSAtsushi Murai   if (sysctl(mib, 6, sp, &needed, NULL, 0 ) < 0) {
280af57ed9fSAtsushi Murai     free(sp);
281af57ed9fSAtsushi Murai     return;
282af57ed9fSAtsushi Murai   }
283af57ed9fSAtsushi Murai #else
284af57ed9fSAtsushi Murai   if (getkerninfo(KINFO_RT_DUMP, sp, &needed, 0) < 0) {
285af57ed9fSAtsushi Murai     free(sp);
286af57ed9fSAtsushi Murai     return;
287af57ed9fSAtsushi Murai   }
288af57ed9fSAtsushi Murai #endif
289af57ed9fSAtsushi Murai   ep = sp + needed;
290af57ed9fSAtsushi Murai 
291af57ed9fSAtsushi Murai   for (cp = sp; cp < ep; cp += rtm->rtm_msglen) {
292af57ed9fSAtsushi Murai     rtm = (struct rt_msghdr *)cp;
293af57ed9fSAtsushi Murai     sa = (struct sockaddr *)(rtm + 1);
294af57ed9fSAtsushi Murai #ifdef DEBUG
295af57ed9fSAtsushi Murai     logprintf("addrs: %x, index: %d, flags: %x, dstnet: %x\n",
296af57ed9fSAtsushi Murai 	rtm->rtm_addrs, rtm->rtm_index, rtm->rtm_flags,
297af57ed9fSAtsushi Murai 	((struct sockaddr_in *)sa)->sin_addr);
298af57ed9fSAtsushi Murai #endif
299af57ed9fSAtsushi Murai     if (rtm->rtm_addrs != RTA_DST &&
300af57ed9fSAtsushi Murai        (rtm->rtm_index == IfIndex) &&
301af57ed9fSAtsushi Murai        (all || (rtm->rtm_flags & RTF_GATEWAY))) {
302af57ed9fSAtsushi Murai       dstnet = ((struct sockaddr_in *)sa)->sin_addr;
303af57ed9fSAtsushi Murai       wp = (u_char *)cp + rtm->rtm_msglen;
304af57ed9fSAtsushi Murai       if (sa->sa_len == 0)
305af57ed9fSAtsushi Murai 	sa->sa_len = sizeof(long);
306af57ed9fSAtsushi Murai       sa = (struct sockaddr *)(sa->sa_len + (char *)sa);
307af57ed9fSAtsushi Murai       gateway = ((struct sockaddr_in *)sa)->sin_addr;
308af57ed9fSAtsushi Murai       lp = (int *)(sa->sa_len + (char *)sa);
309af57ed9fSAtsushi Murai       mask = 0;
310af57ed9fSAtsushi Murai       if ((char *)lp < (char *)wp && *lp) {
311af57ed9fSAtsushi Murai #ifdef DEBUG
312af57ed9fSAtsushi Murai 	printf(" flag = %x, rest = %d", rtm->rtm_flags, *lp);
313af57ed9fSAtsushi Murai #endif
314af57ed9fSAtsushi Murai 	wp = (u_char *)(lp + 1);
315af57ed9fSAtsushi Murai 	for (nb = *lp; nb > 4; nb--) {
316af57ed9fSAtsushi Murai 	  mask <<= 8;
317af57ed9fSAtsushi Murai 	  mask |= *wp++;
318af57ed9fSAtsushi Murai 	}
319af57ed9fSAtsushi Murai 	for (nb = 8 - *lp; nb > 0; nb--)
320af57ed9fSAtsushi Murai 	  mask <<= 8;
321af57ed9fSAtsushi Murai       }
322af57ed9fSAtsushi Murai #ifdef DEBUG
323af57ed9fSAtsushi Murai       logprintf("## %s ", inet_ntoa(dstnet));
324af57ed9fSAtsushi Murai       logprintf(" %s  %d\n", inet_ntoa(gateway), rtm->rtm_index);
325af57ed9fSAtsushi Murai #endif
326af57ed9fSAtsushi Murai       if (dstnet.s_addr == INADDR_ANY) {
327af57ed9fSAtsushi Murai         gateway.s_addr = INADDR_ANY;
328af57ed9fSAtsushi Murai         mask = INADDR_ANY;
329af57ed9fSAtsushi Murai       }
330af57ed9fSAtsushi Murai       OsSetRoute(RTM_DELETE, dstnet, gateway, htonl(mask));
331af57ed9fSAtsushi Murai     }
332af57ed9fSAtsushi Murai #ifdef DEBUG
333af57ed9fSAtsushi Murai     else if (rtm->rtm_index == IfIndex) {
334af57ed9fSAtsushi Murai       logprintf("??? addrs: %x, flags = %x\n", rtm->rtm_addrs, rtm->rtm_flags);
335af57ed9fSAtsushi Murai     }
336af57ed9fSAtsushi Murai #endif
337af57ed9fSAtsushi Murai   }
338af57ed9fSAtsushi Murai   free(sp);
339af57ed9fSAtsushi Murai }
340af57ed9fSAtsushi Murai 
341af57ed9fSAtsushi Murai int
342af57ed9fSAtsushi Murai GetIfIndex(name)
343af57ed9fSAtsushi Murai char *name;
344af57ed9fSAtsushi Murai {
345af57ed9fSAtsushi Murai   struct ifreq *ifrp;
346af57ed9fSAtsushi Murai   int s, len, elen, index;
347af57ed9fSAtsushi Murai   struct ifconf ifconfs;
348af57ed9fSAtsushi Murai   struct ifreq reqbuf[32];
349af57ed9fSAtsushi Murai 
350af57ed9fSAtsushi Murai   s = socket(AF_INET, SOCK_DGRAM, 0);
351af57ed9fSAtsushi Murai   if (s < 0) {
352af57ed9fSAtsushi Murai     perror("socket");
353af57ed9fSAtsushi Murai     return(-1);
354af57ed9fSAtsushi Murai   }
355af57ed9fSAtsushi Murai 
356af57ed9fSAtsushi Murai   ifconfs.ifc_len = sizeof(reqbuf);
357af57ed9fSAtsushi Murai   ifconfs.ifc_buf = (caddr_t)reqbuf;
358af57ed9fSAtsushi Murai   if (ioctl(s, SIOCGIFCONF, &ifconfs) < 0) {
359af57ed9fSAtsushi Murai     perror("IFCONF");
360af57ed9fSAtsushi Murai     return(-1);
361af57ed9fSAtsushi Murai   }
362af57ed9fSAtsushi Murai 
363af57ed9fSAtsushi Murai   ifrp = ifconfs.ifc_req;
364af57ed9fSAtsushi Murai 
365af57ed9fSAtsushi Murai   index = 1;
366af57ed9fSAtsushi Murai   for (len = ifconfs.ifc_len; len > 0; len -= sizeof(struct ifreq)) {
367af57ed9fSAtsushi Murai     elen = ifrp->ifr_addr.sa_len - sizeof(struct sockaddr);
368af57ed9fSAtsushi Murai     if (ifrp->ifr_addr.sa_family == AF_LINK) {
369af57ed9fSAtsushi Murai #ifdef DEBUG
370af57ed9fSAtsushi Murai       logprintf("%d: %-*.*s, %d, %d\n", index, IFNAMSIZ, IFNAMSIZ, ifrp->ifr_name,
371af57ed9fSAtsushi Murai 	   ifrp->ifr_addr.sa_family, elen);
372af57ed9fSAtsushi Murai #endif
373af57ed9fSAtsushi Murai       if (strcmp(ifrp->ifr_name, name) == 0) {
374af57ed9fSAtsushi Murai         IfIndex = index;
375af57ed9fSAtsushi Murai         return(index);
376af57ed9fSAtsushi Murai       }
377af57ed9fSAtsushi Murai #if defined(__FreeBSD__) || (_BSDI_VERSION >= 199312)
378af57ed9fSAtsushi Murai       index++;
379af57ed9fSAtsushi Murai #endif
380af57ed9fSAtsushi Murai     }
381af57ed9fSAtsushi Murai 
382af57ed9fSAtsushi Murai     len -= elen;
383af57ed9fSAtsushi Murai     ifrp = (struct ifreq *)((char *)ifrp + elen);
384af57ed9fSAtsushi Murai     ifrp++;
385af57ed9fSAtsushi Murai #if defined(_BSDI_VERSION) && (_BSDI_VERSION < 199312)
386af57ed9fSAtsushi Murai     index++;
387af57ed9fSAtsushi Murai #endif
388af57ed9fSAtsushi Murai   }
389af57ed9fSAtsushi Murai 
390af57ed9fSAtsushi Murai   close(s);
391af57ed9fSAtsushi Murai   return(-1);
392af57ed9fSAtsushi Murai }
393