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
ifaddrlistx(const char * ifname,uint64_t set,uint64_t clear,ifaddrlistx_t ** ifaddrsp)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
ifaddrlistx_free(ifaddrlistx_t * ifaddrp)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