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:$ 2153c9f6c0SAtsushi Murai * 22af57ed9fSAtsushi Murai */ 23af57ed9fSAtsushi Murai #include <sys/types.h> 24af57ed9fSAtsushi Murai #include <machine/endian.h> 25af57ed9fSAtsushi Murai #include <sys/param.h> 26af57ed9fSAtsushi Murai #include <sys/socket.h> 27af57ed9fSAtsushi Murai #include <net/route.h> 28af57ed9fSAtsushi Murai #include <sys/ioctl.h> 29af57ed9fSAtsushi Murai #include <net/if.h> 30af57ed9fSAtsushi Murai #include <errno.h> 31af57ed9fSAtsushi Murai #include <netinet/in_systm.h> 32af57ed9fSAtsushi Murai #include <netinet/in.h> 33af57ed9fSAtsushi Murai #include <arpa/inet.h> 3453c9f6c0SAtsushi Murai #if (BSD >= 199306) 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; 17353c9f6c0SAtsushi Murai #if (BSD >= 199306) 174af57ed9fSAtsushi Murai int mib[6]; 175af57ed9fSAtsushi Murai #endif 176af57ed9fSAtsushi Murai 17753c9f6c0SAtsushi Murai #if (BSD >= 199306) 178af57ed9fSAtsushi Murai mib[0] = CTL_NET; 179af57ed9fSAtsushi Murai mib[1] = PF_ROUTE; 18053c9f6c0SAtsushi Murai mib[2] = 0; 18153c9f6c0SAtsushi Murai mib[3] = 0; 182af57ed9fSAtsushi Murai mib[4] = NET_RT_DUMP; 18353c9f6c0SAtsushi Murai mib[5] = 0; 18453c9f6c0SAtsushi Murai if (sysctl(mib, 6, NULL, &needed, NULL, 0) < 0) { 18553c9f6c0SAtsushi Murai perror("sysctl-estimate"); 18653c9f6c0SAtsushi Murai return(1); 18753c9f6c0SAtsushi Murai } 188af57ed9fSAtsushi Murai #else 189af57ed9fSAtsushi Murai needed = getkerninfo(KINFO_RT_DUMP, 0, 0, 0); 190af57ed9fSAtsushi Murai #endif 191af57ed9fSAtsushi Murai if (needed < 0) 192af57ed9fSAtsushi Murai return(1); 193af57ed9fSAtsushi Murai sp = malloc(needed); 194af57ed9fSAtsushi Murai if (sp == NULL) 195af57ed9fSAtsushi Murai return(1); 19653c9f6c0SAtsushi Murai #if (BSD >= 199306) 19753c9f6c0SAtsushi Murai if (sysctl(mib, 6, sp, &needed, NULL, 0) < 0) { 19853c9f6c0SAtsushi Murai perror("sysctl-getroute"); 199af57ed9fSAtsushi Murai return(1); 20053c9f6c0SAtsushi Murai } 201af57ed9fSAtsushi Murai #else 202af57ed9fSAtsushi Murai if (getkerninfo(KINFO_RT_DUMP, sp, &needed, 0) < 0) 203af57ed9fSAtsushi Murai return(1); 204af57ed9fSAtsushi Murai #endif 205af57ed9fSAtsushi Murai ep = sp + needed; 206af57ed9fSAtsushi Murai 207af57ed9fSAtsushi Murai for (cp = sp; cp < ep; cp += rtm->rtm_msglen) { 208af57ed9fSAtsushi Murai rtm = (struct rt_msghdr *)cp; 209af57ed9fSAtsushi Murai sa = (struct sockaddr *)(rtm + 1); 210af57ed9fSAtsushi Murai mask = 0xffffffff; 211af57ed9fSAtsushi Murai if (rtm->rtm_addrs == RTA_DST) 212af57ed9fSAtsushi Murai p_sockaddr(sa, 36); 213af57ed9fSAtsushi Murai else { 214af57ed9fSAtsushi Murai wp = (u_char *)cp + rtm->rtm_msglen; 215af57ed9fSAtsushi Murai p_sockaddr(sa, 16); 216af57ed9fSAtsushi Murai if (sa->sa_len == 0) 217af57ed9fSAtsushi Murai sa->sa_len = sizeof(long); 218af57ed9fSAtsushi Murai sa = (struct sockaddr *)(sa->sa_len + (char *)sa); 219af57ed9fSAtsushi Murai p_sockaddr(sa, 18); 220af57ed9fSAtsushi Murai lp = (int *)(sa->sa_len + (char *)sa); 221af57ed9fSAtsushi Murai if ((char *)lp < (char *)wp && *lp) { 222af57ed9fSAtsushi Murai #ifdef DEBUG 223af57ed9fSAtsushi Murai logprintf(" flag = %x, rest = %d", rtm->rtm_flags, *lp); 224af57ed9fSAtsushi Murai #endif 225af57ed9fSAtsushi Murai wp = (u_char *)(lp + 1); 226af57ed9fSAtsushi Murai mask = 0; 227af57ed9fSAtsushi Murai for (nb = *lp; nb > 4; nb--) { 228af57ed9fSAtsushi Murai mask <<= 8; 229af57ed9fSAtsushi Murai mask |= *wp++; 230af57ed9fSAtsushi Murai } 231af57ed9fSAtsushi Murai for (nb = 8 - *lp; nb > 0; nb--) 232af57ed9fSAtsushi Murai mask <<= 8; 233af57ed9fSAtsushi Murai } 234af57ed9fSAtsushi Murai } 235af57ed9fSAtsushi Murai printf("%08x ", mask); 236af57ed9fSAtsushi Murai p_flags(rtm->rtm_flags & (RTF_UP|RTF_GATEWAY|RTF_HOST), "%-6.6s "); 237af57ed9fSAtsushi Murai printf("(%d)\n", rtm->rtm_index); 238af57ed9fSAtsushi Murai } 239af57ed9fSAtsushi Murai 240af57ed9fSAtsushi Murai return(1); 241af57ed9fSAtsushi Murai } 242af57ed9fSAtsushi Murai 243af57ed9fSAtsushi Murai /* 244af57ed9fSAtsushi Murai * Delete routes associated with our interface 245af57ed9fSAtsushi Murai */ 246af57ed9fSAtsushi Murai void 247af57ed9fSAtsushi Murai DeleteIfRoutes(all) 248af57ed9fSAtsushi Murai int all; 249af57ed9fSAtsushi Murai { 250af57ed9fSAtsushi Murai struct rt_msghdr *rtm; 251af57ed9fSAtsushi Murai struct sockaddr *sa; 252af57ed9fSAtsushi Murai struct in_addr dstnet, gateway; 253af57ed9fSAtsushi Murai int needed; 254af57ed9fSAtsushi Murai char *sp, *cp, *ep; 255af57ed9fSAtsushi Murai u_long mask; 256af57ed9fSAtsushi Murai int *lp, nb; 257af57ed9fSAtsushi Murai u_char *wp; 25853c9f6c0SAtsushi Murai #if (BSD >= 199306) 259af57ed9fSAtsushi Murai int mib[6]; 260af57ed9fSAtsushi Murai #endif 261af57ed9fSAtsushi Murai 262af57ed9fSAtsushi Murai #ifdef DEBUG 263af57ed9fSAtsushi Murai logprintf("DeleteIfRoutes (%d)\n", IfIndex); 264af57ed9fSAtsushi Murai #endif 26553c9f6c0SAtsushi Murai #if (BSD >= 199306) 266af57ed9fSAtsushi Murai mib[0] = CTL_NET; 267af57ed9fSAtsushi Murai mib[1] = PF_ROUTE; 26853c9f6c0SAtsushi Murai mib[2] = 0; 26953c9f6c0SAtsushi Murai mib[3] = 0; 270af57ed9fSAtsushi Murai mib[4] = NET_RT_DUMP; 27153c9f6c0SAtsushi Murai mib[5] = 0; 27253c9f6c0SAtsushi Murai if (sysctl(mib, 6, NULL, &needed, NULL, 0) < 0) { 27353c9f6c0SAtsushi Murai perror("sysctl-estimate"); 27453c9f6c0SAtsushi Murai return; 27553c9f6c0SAtsushi Murai } 276af57ed9fSAtsushi Murai #else 277af57ed9fSAtsushi Murai needed = getkerninfo(KINFO_RT_DUMP, 0, 0, 0); 278af57ed9fSAtsushi Murai #endif 27953c9f6c0SAtsushi Murai 280af57ed9fSAtsushi Murai if (needed < 0) 281af57ed9fSAtsushi Murai return; 282af57ed9fSAtsushi Murai 283af57ed9fSAtsushi Murai sp = malloc(needed); 284af57ed9fSAtsushi Murai if (sp == NULL) 285af57ed9fSAtsushi Murai return; 286af57ed9fSAtsushi Murai 28753c9f6c0SAtsushi Murai #if (BSD >= 199306) 288af57ed9fSAtsushi Murai if (sysctl(mib, 6, sp, &needed, NULL, 0) < 0) { 289af57ed9fSAtsushi Murai free(sp); 29053c9f6c0SAtsushi Murai perror("sysctl-getroute"); 291af57ed9fSAtsushi Murai return; 292af57ed9fSAtsushi Murai } 293af57ed9fSAtsushi Murai #else 294af57ed9fSAtsushi Murai if (getkerninfo(KINFO_RT_DUMP, sp, &needed, 0) < 0) { 295af57ed9fSAtsushi Murai free(sp); 296af57ed9fSAtsushi Murai return; 297af57ed9fSAtsushi Murai } 298af57ed9fSAtsushi Murai #endif 299af57ed9fSAtsushi Murai ep = sp + needed; 300af57ed9fSAtsushi Murai 301af57ed9fSAtsushi Murai for (cp = sp; cp < ep; cp += rtm->rtm_msglen) { 302af57ed9fSAtsushi Murai rtm = (struct rt_msghdr *)cp; 303af57ed9fSAtsushi Murai sa = (struct sockaddr *)(rtm + 1); 304af57ed9fSAtsushi Murai #ifdef DEBUG 305af57ed9fSAtsushi Murai logprintf("addrs: %x, index: %d, flags: %x, dstnet: %x\n", 306af57ed9fSAtsushi Murai rtm->rtm_addrs, rtm->rtm_index, rtm->rtm_flags, 307af57ed9fSAtsushi Murai ((struct sockaddr_in *)sa)->sin_addr); 308af57ed9fSAtsushi Murai #endif 309af57ed9fSAtsushi Murai if (rtm->rtm_addrs != RTA_DST && 310af57ed9fSAtsushi Murai (rtm->rtm_index == IfIndex) && 311af57ed9fSAtsushi Murai (all || (rtm->rtm_flags & RTF_GATEWAY))) { 312af57ed9fSAtsushi Murai dstnet = ((struct sockaddr_in *)sa)->sin_addr; 313af57ed9fSAtsushi Murai wp = (u_char *)cp + rtm->rtm_msglen; 314af57ed9fSAtsushi Murai if (sa->sa_len == 0) 315af57ed9fSAtsushi Murai sa->sa_len = sizeof(long); 316af57ed9fSAtsushi Murai sa = (struct sockaddr *)(sa->sa_len + (char *)sa); 317af57ed9fSAtsushi Murai gateway = ((struct sockaddr_in *)sa)->sin_addr; 318af57ed9fSAtsushi Murai lp = (int *)(sa->sa_len + (char *)sa); 319af57ed9fSAtsushi Murai mask = 0; 320af57ed9fSAtsushi Murai if ((char *)lp < (char *)wp && *lp) { 321af57ed9fSAtsushi Murai #ifdef DEBUG 322af57ed9fSAtsushi Murai printf(" flag = %x, rest = %d", rtm->rtm_flags, *lp); 323af57ed9fSAtsushi Murai #endif 324af57ed9fSAtsushi Murai wp = (u_char *)(lp + 1); 325af57ed9fSAtsushi Murai for (nb = *lp; nb > 4; nb--) { 326af57ed9fSAtsushi Murai mask <<= 8; 327af57ed9fSAtsushi Murai mask |= *wp++; 328af57ed9fSAtsushi Murai } 329af57ed9fSAtsushi Murai for (nb = 8 - *lp; nb > 0; nb--) 330af57ed9fSAtsushi Murai mask <<= 8; 331af57ed9fSAtsushi Murai } 332af57ed9fSAtsushi Murai #ifdef DEBUG 333af57ed9fSAtsushi Murai logprintf("## %s ", inet_ntoa(dstnet)); 334af57ed9fSAtsushi Murai logprintf(" %s %d\n", inet_ntoa(gateway), rtm->rtm_index); 335af57ed9fSAtsushi Murai #endif 336af57ed9fSAtsushi Murai if (dstnet.s_addr == INADDR_ANY) { 337af57ed9fSAtsushi Murai gateway.s_addr = INADDR_ANY; 338af57ed9fSAtsushi Murai mask = INADDR_ANY; 339af57ed9fSAtsushi Murai } 340af57ed9fSAtsushi Murai OsSetRoute(RTM_DELETE, dstnet, gateway, htonl(mask)); 341af57ed9fSAtsushi Murai } 342af57ed9fSAtsushi Murai #ifdef DEBUG 343af57ed9fSAtsushi Murai else if (rtm->rtm_index == IfIndex) { 344af57ed9fSAtsushi Murai logprintf("??? addrs: %x, flags = %x\n", rtm->rtm_addrs, rtm->rtm_flags); 345af57ed9fSAtsushi Murai } 346af57ed9fSAtsushi Murai #endif 347af57ed9fSAtsushi Murai } 348af57ed9fSAtsushi Murai free(sp); 349af57ed9fSAtsushi Murai } 350af57ed9fSAtsushi Murai 351af57ed9fSAtsushi Murai int 352af57ed9fSAtsushi Murai GetIfIndex(name) 353af57ed9fSAtsushi Murai char *name; 354af57ed9fSAtsushi Murai { 355af57ed9fSAtsushi Murai struct ifreq *ifrp; 356af57ed9fSAtsushi Murai int s, len, elen, index; 357af57ed9fSAtsushi Murai struct ifconf ifconfs; 358af57ed9fSAtsushi Murai struct ifreq reqbuf[32]; 359af57ed9fSAtsushi Murai 360af57ed9fSAtsushi Murai s = socket(AF_INET, SOCK_DGRAM, 0); 361af57ed9fSAtsushi Murai if (s < 0) { 362af57ed9fSAtsushi Murai perror("socket"); 363af57ed9fSAtsushi Murai return(-1); 364af57ed9fSAtsushi Murai } 365af57ed9fSAtsushi Murai 366af57ed9fSAtsushi Murai ifconfs.ifc_len = sizeof(reqbuf); 367af57ed9fSAtsushi Murai ifconfs.ifc_buf = (caddr_t)reqbuf; 368af57ed9fSAtsushi Murai if (ioctl(s, SIOCGIFCONF, &ifconfs) < 0) { 369af57ed9fSAtsushi Murai perror("IFCONF"); 370af57ed9fSAtsushi Murai return(-1); 371af57ed9fSAtsushi Murai } 372af57ed9fSAtsushi Murai 373af57ed9fSAtsushi Murai ifrp = ifconfs.ifc_req; 374af57ed9fSAtsushi Murai 375af57ed9fSAtsushi Murai index = 1; 376af57ed9fSAtsushi Murai for (len = ifconfs.ifc_len; len > 0; len -= sizeof(struct ifreq)) { 377af57ed9fSAtsushi Murai elen = ifrp->ifr_addr.sa_len - sizeof(struct sockaddr); 378af57ed9fSAtsushi Murai if (ifrp->ifr_addr.sa_family == AF_LINK) { 379af57ed9fSAtsushi Murai #ifdef DEBUG 380af57ed9fSAtsushi Murai logprintf("%d: %-*.*s, %d, %d\n", index, IFNAMSIZ, IFNAMSIZ, ifrp->ifr_name, 381af57ed9fSAtsushi Murai ifrp->ifr_addr.sa_family, elen); 382af57ed9fSAtsushi Murai #endif 383af57ed9fSAtsushi Murai if (strcmp(ifrp->ifr_name, name) == 0) { 384af57ed9fSAtsushi Murai IfIndex = index; 385af57ed9fSAtsushi Murai return(index); 386af57ed9fSAtsushi Murai } 387af57ed9fSAtsushi Murai index++; 388af57ed9fSAtsushi Murai } 389af57ed9fSAtsushi Murai 390af57ed9fSAtsushi Murai len -= elen; 391af57ed9fSAtsushi Murai ifrp = (struct ifreq *)((char *)ifrp + elen); 392af57ed9fSAtsushi Murai ifrp++; 393af57ed9fSAtsushi Murai } 394af57ed9fSAtsushi Murai 395af57ed9fSAtsushi Murai close(s); 396af57ed9fSAtsushi Murai return(-1); 397af57ed9fSAtsushi Murai } 398