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