1*8485SPeter.Memishian@Sun.COM /* 2*8485SPeter.Memishian@Sun.COM * CDDL HEADER START 3*8485SPeter.Memishian@Sun.COM * 4*8485SPeter.Memishian@Sun.COM * The contents of this file are subject to the terms of the 5*8485SPeter.Memishian@Sun.COM * Common Development and Distribution License (the "License"). 6*8485SPeter.Memishian@Sun.COM * You may not use this file except in compliance with the License. 7*8485SPeter.Memishian@Sun.COM * 8*8485SPeter.Memishian@Sun.COM * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9*8485SPeter.Memishian@Sun.COM * or http://www.opensolaris.org/os/licensing. 10*8485SPeter.Memishian@Sun.COM * See the License for the specific language governing permissions 11*8485SPeter.Memishian@Sun.COM * and limitations under the License. 12*8485SPeter.Memishian@Sun.COM * 13*8485SPeter.Memishian@Sun.COM * When distributing Covered Code, include this CDDL HEADER in each 14*8485SPeter.Memishian@Sun.COM * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15*8485SPeter.Memishian@Sun.COM * If applicable, add the following below this CDDL HEADER, with the 16*8485SPeter.Memishian@Sun.COM * fields enclosed by brackets "[]" replaced with your own identifying 17*8485SPeter.Memishian@Sun.COM * information: Portions Copyright [yyyy] [name of copyright owner] 18*8485SPeter.Memishian@Sun.COM * 19*8485SPeter.Memishian@Sun.COM * CDDL HEADER END 20*8485SPeter.Memishian@Sun.COM * 21*8485SPeter.Memishian@Sun.COM * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 22*8485SPeter.Memishian@Sun.COM * Use is subject to license terms. 23*8485SPeter.Memishian@Sun.COM */ 24*8485SPeter.Memishian@Sun.COM 25*8485SPeter.Memishian@Sun.COM #include <errno.h> 26*8485SPeter.Memishian@Sun.COM #include <libinetutil.h> 27*8485SPeter.Memishian@Sun.COM #include <stdio.h> 28*8485SPeter.Memishian@Sun.COM #include <stdlib.h> 29*8485SPeter.Memishian@Sun.COM #include <string.h> 30*8485SPeter.Memishian@Sun.COM #include <unistd.h> 31*8485SPeter.Memishian@Sun.COM #include <sys/socket.h> 32*8485SPeter.Memishian@Sun.COM #include <sys/sockio.h> 33*8485SPeter.Memishian@Sun.COM 34*8485SPeter.Memishian@Sun.COM /* 35*8485SPeter.Memishian@Sun.COM * Create a list of the addresses on physical interface `ifname' with at least 36*8485SPeter.Memishian@Sun.COM * one of the flags in `set' set and all of the flags in `clear' clear. 37*8485SPeter.Memishian@Sun.COM * Return the number of items in the list, or -1 on failure. 38*8485SPeter.Memishian@Sun.COM */ 39*8485SPeter.Memishian@Sun.COM int 40*8485SPeter.Memishian@Sun.COM ifaddrlistx(const char *ifname, uint64_t set, uint64_t clear, 41*8485SPeter.Memishian@Sun.COM ifaddrlistx_t **ifaddrsp) 42*8485SPeter.Memishian@Sun.COM { 43*8485SPeter.Memishian@Sun.COM struct lifconf lifc; 44*8485SPeter.Memishian@Sun.COM struct lifnum lifn; 45*8485SPeter.Memishian@Sun.COM struct lifreq *lifrp; 46*8485SPeter.Memishian@Sun.COM ifaddrlistx_t *ifaddrp, *ifaddrs = NULL; 47*8485SPeter.Memishian@Sun.COM int i, nlifr, naddr = 0; 48*8485SPeter.Memishian@Sun.COM char *cp; 49*8485SPeter.Memishian@Sun.COM uint_t flags; 50*8485SPeter.Memishian@Sun.COM int s4, s6 = -1; 51*8485SPeter.Memishian@Sun.COM boolean_t isv6; 52*8485SPeter.Memishian@Sun.COM int save_errno; 53*8485SPeter.Memishian@Sun.COM struct sockaddr_storage addr; 54*8485SPeter.Memishian@Sun.COM 55*8485SPeter.Memishian@Sun.COM (void) memset(&lifc, 0, sizeof (lifc)); 56*8485SPeter.Memishian@Sun.COM flags = LIFC_NOXMIT | LIFC_ALLZONES | LIFC_TEMPORARY | LIFC_UNDER_IPMP; 57*8485SPeter.Memishian@Sun.COM 58*8485SPeter.Memishian@Sun.COM /* 59*8485SPeter.Memishian@Sun.COM * We need both IPv4 and IPv6 sockets to query both IPv4 and IPv6 60*8485SPeter.Memishian@Sun.COM * interfaces below. 61*8485SPeter.Memishian@Sun.COM */ 62*8485SPeter.Memishian@Sun.COM if ((s4 = socket(AF_INET, SOCK_DGRAM, 0)) == -1 || 63*8485SPeter.Memishian@Sun.COM (s6 = socket(AF_INET6, SOCK_DGRAM, 0)) == -1) { 64*8485SPeter.Memishian@Sun.COM goto fail; 65*8485SPeter.Memishian@Sun.COM } 66*8485SPeter.Memishian@Sun.COM 67*8485SPeter.Memishian@Sun.COM /* 68*8485SPeter.Memishian@Sun.COM * Get the number of network interfaces of type `family'. 69*8485SPeter.Memishian@Sun.COM */ 70*8485SPeter.Memishian@Sun.COM lifn.lifn_family = AF_UNSPEC; 71*8485SPeter.Memishian@Sun.COM lifn.lifn_flags = flags; 72*8485SPeter.Memishian@Sun.COM again: 73*8485SPeter.Memishian@Sun.COM if (ioctl(s4, SIOCGLIFNUM, &lifn) == -1) 74*8485SPeter.Memishian@Sun.COM goto fail; 75*8485SPeter.Memishian@Sun.COM 76*8485SPeter.Memishian@Sun.COM /* 77*8485SPeter.Memishian@Sun.COM * Pad the interface count to detect when additional interfaces have 78*8485SPeter.Memishian@Sun.COM * been configured between SIOCGLIFNUM and SIOCGLIFCONF. 79*8485SPeter.Memishian@Sun.COM */ 80*8485SPeter.Memishian@Sun.COM lifn.lifn_count += 4; 81*8485SPeter.Memishian@Sun.COM 82*8485SPeter.Memishian@Sun.COM lifc.lifc_flags = flags; 83*8485SPeter.Memishian@Sun.COM lifc.lifc_family = AF_UNSPEC; 84*8485SPeter.Memishian@Sun.COM lifc.lifc_len = lifn.lifn_count * sizeof (struct lifreq); 85*8485SPeter.Memishian@Sun.COM if ((lifc.lifc_buf = realloc(lifc.lifc_buf, lifc.lifc_len)) == NULL) 86*8485SPeter.Memishian@Sun.COM goto fail; 87*8485SPeter.Memishian@Sun.COM 88*8485SPeter.Memishian@Sun.COM if (ioctl(s4, SIOCGLIFCONF, &lifc) == -1) 89*8485SPeter.Memishian@Sun.COM goto fail; 90*8485SPeter.Memishian@Sun.COM 91*8485SPeter.Memishian@Sun.COM /* 92*8485SPeter.Memishian@Sun.COM * If every lifr_req slot is taken, then additional interfaces must 93*8485SPeter.Memishian@Sun.COM * have been plumbed between the SIOCGLIFNUM and the SIOCGLIFCONF. 94*8485SPeter.Memishian@Sun.COM * Recalculate to make sure we didn't miss any interfaces. 95*8485SPeter.Memishian@Sun.COM */ 96*8485SPeter.Memishian@Sun.COM nlifr = lifc.lifc_len / sizeof (struct lifreq); 97*8485SPeter.Memishian@Sun.COM if (nlifr >= lifn.lifn_count) 98*8485SPeter.Memishian@Sun.COM goto again; 99*8485SPeter.Memishian@Sun.COM 100*8485SPeter.Memishian@Sun.COM /* 101*8485SPeter.Memishian@Sun.COM * Populate the ifaddrlistx by querying each matching interface. If a 102*8485SPeter.Memishian@Sun.COM * query ioctl returns ENXIO, then the interface must have been 103*8485SPeter.Memishian@Sun.COM * removed after the SIOCGLIFCONF completed -- so we just ignore it. 104*8485SPeter.Memishian@Sun.COM */ 105*8485SPeter.Memishian@Sun.COM for (lifrp = lifc.lifc_req, i = 0; i < nlifr; i++, lifrp++) { 106*8485SPeter.Memishian@Sun.COM if ((cp = strchr(lifrp->lifr_name, ':')) != NULL) 107*8485SPeter.Memishian@Sun.COM *cp = '\0'; 108*8485SPeter.Memishian@Sun.COM 109*8485SPeter.Memishian@Sun.COM if (strcmp(lifrp->lifr_name, ifname) != 0) 110*8485SPeter.Memishian@Sun.COM continue; 111*8485SPeter.Memishian@Sun.COM 112*8485SPeter.Memishian@Sun.COM if (cp != NULL) 113*8485SPeter.Memishian@Sun.COM *cp = ':'; 114*8485SPeter.Memishian@Sun.COM 115*8485SPeter.Memishian@Sun.COM addr = lifrp->lifr_addr; 116*8485SPeter.Memishian@Sun.COM isv6 = addr.ss_family == AF_INET6; 117*8485SPeter.Memishian@Sun.COM if (ioctl(isv6 ? s6 : s4, SIOCGLIFFLAGS, lifrp) == -1) { 118*8485SPeter.Memishian@Sun.COM if (errno == ENXIO) 119*8485SPeter.Memishian@Sun.COM continue; 120*8485SPeter.Memishian@Sun.COM goto fail; 121*8485SPeter.Memishian@Sun.COM } 122*8485SPeter.Memishian@Sun.COM 123*8485SPeter.Memishian@Sun.COM if (set != 0 && ((lifrp->lifr_flags & set) == 0) || 124*8485SPeter.Memishian@Sun.COM (lifrp->lifr_flags & clear) != 0) 125*8485SPeter.Memishian@Sun.COM continue; 126*8485SPeter.Memishian@Sun.COM 127*8485SPeter.Memishian@Sun.COM /* 128*8485SPeter.Memishian@Sun.COM * We've got a match; allocate a new record. 129*8485SPeter.Memishian@Sun.COM */ 130*8485SPeter.Memishian@Sun.COM if ((ifaddrp = malloc(sizeof (ifaddrlistx_t))) == NULL) 131*8485SPeter.Memishian@Sun.COM goto fail; 132*8485SPeter.Memishian@Sun.COM 133*8485SPeter.Memishian@Sun.COM (void) strlcpy(ifaddrp->ia_name, lifrp->lifr_name, LIFNAMSIZ); 134*8485SPeter.Memishian@Sun.COM ifaddrp->ia_flags = lifrp->lifr_flags; 135*8485SPeter.Memishian@Sun.COM ifaddrp->ia_addr = addr; 136*8485SPeter.Memishian@Sun.COM ifaddrp->ia_next = ifaddrs; 137*8485SPeter.Memishian@Sun.COM ifaddrs = ifaddrp; 138*8485SPeter.Memishian@Sun.COM naddr++; 139*8485SPeter.Memishian@Sun.COM } 140*8485SPeter.Memishian@Sun.COM 141*8485SPeter.Memishian@Sun.COM (void) close(s4); 142*8485SPeter.Memishian@Sun.COM (void) close(s6); 143*8485SPeter.Memishian@Sun.COM free(lifc.lifc_buf); 144*8485SPeter.Memishian@Sun.COM *ifaddrsp = ifaddrs; 145*8485SPeter.Memishian@Sun.COM return (naddr); 146*8485SPeter.Memishian@Sun.COM fail: 147*8485SPeter.Memishian@Sun.COM save_errno = errno; 148*8485SPeter.Memishian@Sun.COM (void) close(s4); 149*8485SPeter.Memishian@Sun.COM (void) close(s6); 150*8485SPeter.Memishian@Sun.COM free(lifc.lifc_buf); 151*8485SPeter.Memishian@Sun.COM ifaddrlistx_free(ifaddrs); 152*8485SPeter.Memishian@Sun.COM errno = save_errno; 153*8485SPeter.Memishian@Sun.COM return (-1); 154*8485SPeter.Memishian@Sun.COM } 155*8485SPeter.Memishian@Sun.COM 156*8485SPeter.Memishian@Sun.COM /* 157*8485SPeter.Memishian@Sun.COM * Free the provided ifaddrlistx_t. 158*8485SPeter.Memishian@Sun.COM */ 159*8485SPeter.Memishian@Sun.COM void 160*8485SPeter.Memishian@Sun.COM ifaddrlistx_free(ifaddrlistx_t *ifaddrp) 161*8485SPeter.Memishian@Sun.COM { 162*8485SPeter.Memishian@Sun.COM ifaddrlistx_t *next_ifaddrp; 163*8485SPeter.Memishian@Sun.COM 164*8485SPeter.Memishian@Sun.COM for (; ifaddrp != NULL; ifaddrp = next_ifaddrp) { 165*8485SPeter.Memishian@Sun.COM next_ifaddrp = ifaddrp->ia_next; 166*8485SPeter.Memishian@Sun.COM free(ifaddrp); 167*8485SPeter.Memishian@Sun.COM } 168*8485SPeter.Memishian@Sun.COM } 169