1*8b26181fSAndroid Build Coastguard Worker /* -*- Mode: c; tab-width: 8; indent-tabs-mode: 1; c-basic-offset: 8; -*- */
2*8b26181fSAndroid Build Coastguard Worker /*
3*8b26181fSAndroid Build Coastguard Worker * Copyright (c) 1994, 1995, 1996, 1997, 1998
4*8b26181fSAndroid Build Coastguard Worker * The Regents of the University of California. All rights reserved.
5*8b26181fSAndroid Build Coastguard Worker *
6*8b26181fSAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
7*8b26181fSAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
8*8b26181fSAndroid Build Coastguard Worker * are met:
9*8b26181fSAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
10*8b26181fSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
11*8b26181fSAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
12*8b26181fSAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
13*8b26181fSAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
14*8b26181fSAndroid Build Coastguard Worker * 3. All advertising materials mentioning features or use of this software
15*8b26181fSAndroid Build Coastguard Worker * must display the following acknowledgement:
16*8b26181fSAndroid Build Coastguard Worker * This product includes software developed by the Computer Systems
17*8b26181fSAndroid Build Coastguard Worker * Engineering Group at Lawrence Berkeley Laboratory.
18*8b26181fSAndroid Build Coastguard Worker * 4. Neither the name of the University nor of the Laboratory may be used
19*8b26181fSAndroid Build Coastguard Worker * to endorse or promote products derived from this software without
20*8b26181fSAndroid Build Coastguard Worker * specific prior written permission.
21*8b26181fSAndroid Build Coastguard Worker *
22*8b26181fSAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23*8b26181fSAndroid Build Coastguard Worker * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24*8b26181fSAndroid Build Coastguard Worker * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25*8b26181fSAndroid Build Coastguard Worker * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26*8b26181fSAndroid Build Coastguard Worker * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27*8b26181fSAndroid Build Coastguard Worker * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28*8b26181fSAndroid Build Coastguard Worker * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29*8b26181fSAndroid Build Coastguard Worker * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30*8b26181fSAndroid Build Coastguard Worker * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31*8b26181fSAndroid Build Coastguard Worker * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32*8b26181fSAndroid Build Coastguard Worker * SUCH DAMAGE.
33*8b26181fSAndroid Build Coastguard Worker */
34*8b26181fSAndroid Build Coastguard Worker
35*8b26181fSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
36*8b26181fSAndroid Build Coastguard Worker #include <config.h>
37*8b26181fSAndroid Build Coastguard Worker #endif
38*8b26181fSAndroid Build Coastguard Worker
39*8b26181fSAndroid Build Coastguard Worker #include <sys/param.h>
40*8b26181fSAndroid Build Coastguard Worker #include <sys/file.h>
41*8b26181fSAndroid Build Coastguard Worker #include <sys/ioctl.h>
42*8b26181fSAndroid Build Coastguard Worker #include <sys/socket.h>
43*8b26181fSAndroid Build Coastguard Worker #ifdef HAVE_SYS_SOCKIO_H
44*8b26181fSAndroid Build Coastguard Worker #include <sys/sockio.h>
45*8b26181fSAndroid Build Coastguard Worker #endif
46*8b26181fSAndroid Build Coastguard Worker #include <sys/time.h> /* concession to AIX */
47*8b26181fSAndroid Build Coastguard Worker
48*8b26181fSAndroid Build Coastguard Worker struct mbuf; /* Squelch compiler warnings on some platforms for */
49*8b26181fSAndroid Build Coastguard Worker struct rtentry; /* declarations in <net/if.h> */
50*8b26181fSAndroid Build Coastguard Worker #include <net/if.h>
51*8b26181fSAndroid Build Coastguard Worker #include <netinet/in.h>
52*8b26181fSAndroid Build Coastguard Worker
53*8b26181fSAndroid Build Coastguard Worker #include <errno.h>
54*8b26181fSAndroid Build Coastguard Worker #include <memory.h>
55*8b26181fSAndroid Build Coastguard Worker #include <stdio.h>
56*8b26181fSAndroid Build Coastguard Worker #include <stdlib.h>
57*8b26181fSAndroid Build Coastguard Worker #include <string.h>
58*8b26181fSAndroid Build Coastguard Worker #include <unistd.h>
59*8b26181fSAndroid Build Coastguard Worker
60*8b26181fSAndroid Build Coastguard Worker #include "pcap-int.h"
61*8b26181fSAndroid Build Coastguard Worker
62*8b26181fSAndroid Build Coastguard Worker #ifdef HAVE_OS_PROTO_H
63*8b26181fSAndroid Build Coastguard Worker #include "os-proto.h"
64*8b26181fSAndroid Build Coastguard Worker #endif
65*8b26181fSAndroid Build Coastguard Worker
66*8b26181fSAndroid Build Coastguard Worker /*
67*8b26181fSAndroid Build Coastguard Worker * Get a list of all interfaces that are up and that we can open.
68*8b26181fSAndroid Build Coastguard Worker * Returns -1 on error, 0 otherwise.
69*8b26181fSAndroid Build Coastguard Worker * The list, as returned through "alldevsp", may be null if no interfaces
70*8b26181fSAndroid Build Coastguard Worker * were up and could be opened.
71*8b26181fSAndroid Build Coastguard Worker *
72*8b26181fSAndroid Build Coastguard Worker * This is the implementation used on platforms that have SIOCGLIFCONF
73*8b26181fSAndroid Build Coastguard Worker * but don't have "getifaddrs()". (Solaris 8 and later; we use
74*8b26181fSAndroid Build Coastguard Worker * SIOCGLIFCONF rather than SIOCGIFCONF in order to get IPv6 addresses.)
75*8b26181fSAndroid Build Coastguard Worker */
76*8b26181fSAndroid Build Coastguard Worker int
pcap_findalldevs_interfaces(pcap_if_list_t * devlistp,char * errbuf,int (* check_usable)(const char *),get_if_flags_func get_flags_func)77*8b26181fSAndroid Build Coastguard Worker pcap_findalldevs_interfaces(pcap_if_list_t *devlistp, char *errbuf,
78*8b26181fSAndroid Build Coastguard Worker int (*check_usable)(const char *), get_if_flags_func get_flags_func)
79*8b26181fSAndroid Build Coastguard Worker {
80*8b26181fSAndroid Build Coastguard Worker register int fd4, fd6, fd;
81*8b26181fSAndroid Build Coastguard Worker register struct lifreq *ifrp, *ifend;
82*8b26181fSAndroid Build Coastguard Worker struct lifnum ifn;
83*8b26181fSAndroid Build Coastguard Worker struct lifconf ifc;
84*8b26181fSAndroid Build Coastguard Worker char *buf = NULL;
85*8b26181fSAndroid Build Coastguard Worker unsigned buf_size;
86*8b26181fSAndroid Build Coastguard Worker #ifdef HAVE_SOLARIS
87*8b26181fSAndroid Build Coastguard Worker char *p, *q;
88*8b26181fSAndroid Build Coastguard Worker #endif
89*8b26181fSAndroid Build Coastguard Worker struct lifreq ifrflags, ifrnetmask, ifrbroadaddr, ifrdstaddr;
90*8b26181fSAndroid Build Coastguard Worker struct sockaddr *netmask, *broadaddr, *dstaddr;
91*8b26181fSAndroid Build Coastguard Worker int ret = 0;
92*8b26181fSAndroid Build Coastguard Worker
93*8b26181fSAndroid Build Coastguard Worker /*
94*8b26181fSAndroid Build Coastguard Worker * Create a socket from which to fetch the list of interfaces,
95*8b26181fSAndroid Build Coastguard Worker * and from which to fetch IPv4 information.
96*8b26181fSAndroid Build Coastguard Worker */
97*8b26181fSAndroid Build Coastguard Worker fd4 = socket(AF_INET, SOCK_DGRAM, 0);
98*8b26181fSAndroid Build Coastguard Worker if (fd4 < 0) {
99*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf, PCAP_ERRBUF_SIZE,
100*8b26181fSAndroid Build Coastguard Worker errno, "socket: AF_INET");
101*8b26181fSAndroid Build Coastguard Worker return (-1);
102*8b26181fSAndroid Build Coastguard Worker }
103*8b26181fSAndroid Build Coastguard Worker
104*8b26181fSAndroid Build Coastguard Worker /*
105*8b26181fSAndroid Build Coastguard Worker * Create a socket from which to fetch IPv6 information.
106*8b26181fSAndroid Build Coastguard Worker */
107*8b26181fSAndroid Build Coastguard Worker fd6 = socket(AF_INET6, SOCK_DGRAM, 0);
108*8b26181fSAndroid Build Coastguard Worker if (fd6 < 0) {
109*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf, PCAP_ERRBUF_SIZE,
110*8b26181fSAndroid Build Coastguard Worker errno, "socket: AF_INET6");
111*8b26181fSAndroid Build Coastguard Worker (void)close(fd4);
112*8b26181fSAndroid Build Coastguard Worker return (-1);
113*8b26181fSAndroid Build Coastguard Worker }
114*8b26181fSAndroid Build Coastguard Worker
115*8b26181fSAndroid Build Coastguard Worker /*
116*8b26181fSAndroid Build Coastguard Worker * How many entries will SIOCGLIFCONF return?
117*8b26181fSAndroid Build Coastguard Worker */
118*8b26181fSAndroid Build Coastguard Worker ifn.lifn_family = AF_UNSPEC;
119*8b26181fSAndroid Build Coastguard Worker ifn.lifn_flags = 0;
120*8b26181fSAndroid Build Coastguard Worker ifn.lifn_count = 0;
121*8b26181fSAndroid Build Coastguard Worker if (ioctl(fd4, SIOCGLIFNUM, (char *)&ifn) < 0) {
122*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf, PCAP_ERRBUF_SIZE,
123*8b26181fSAndroid Build Coastguard Worker errno, "SIOCGLIFNUM");
124*8b26181fSAndroid Build Coastguard Worker (void)close(fd6);
125*8b26181fSAndroid Build Coastguard Worker (void)close(fd4);
126*8b26181fSAndroid Build Coastguard Worker return (-1);
127*8b26181fSAndroid Build Coastguard Worker }
128*8b26181fSAndroid Build Coastguard Worker
129*8b26181fSAndroid Build Coastguard Worker /*
130*8b26181fSAndroid Build Coastguard Worker * Allocate a buffer for those entries.
131*8b26181fSAndroid Build Coastguard Worker */
132*8b26181fSAndroid Build Coastguard Worker buf_size = ifn.lifn_count * sizeof (struct lifreq);
133*8b26181fSAndroid Build Coastguard Worker buf = malloc(buf_size);
134*8b26181fSAndroid Build Coastguard Worker if (buf == NULL) {
135*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf, PCAP_ERRBUF_SIZE,
136*8b26181fSAndroid Build Coastguard Worker errno, "malloc");
137*8b26181fSAndroid Build Coastguard Worker (void)close(fd6);
138*8b26181fSAndroid Build Coastguard Worker (void)close(fd4);
139*8b26181fSAndroid Build Coastguard Worker return (-1);
140*8b26181fSAndroid Build Coastguard Worker }
141*8b26181fSAndroid Build Coastguard Worker
142*8b26181fSAndroid Build Coastguard Worker /*
143*8b26181fSAndroid Build Coastguard Worker * Get the entries.
144*8b26181fSAndroid Build Coastguard Worker */
145*8b26181fSAndroid Build Coastguard Worker ifc.lifc_len = buf_size;
146*8b26181fSAndroid Build Coastguard Worker ifc.lifc_buf = buf;
147*8b26181fSAndroid Build Coastguard Worker ifc.lifc_family = AF_UNSPEC;
148*8b26181fSAndroid Build Coastguard Worker ifc.lifc_flags = 0;
149*8b26181fSAndroid Build Coastguard Worker memset(buf, 0, buf_size);
150*8b26181fSAndroid Build Coastguard Worker if (ioctl(fd4, SIOCGLIFCONF, (char *)&ifc) < 0) {
151*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf, PCAP_ERRBUF_SIZE,
152*8b26181fSAndroid Build Coastguard Worker errno, "SIOCGLIFCONF");
153*8b26181fSAndroid Build Coastguard Worker (void)close(fd6);
154*8b26181fSAndroid Build Coastguard Worker (void)close(fd4);
155*8b26181fSAndroid Build Coastguard Worker free(buf);
156*8b26181fSAndroid Build Coastguard Worker return (-1);
157*8b26181fSAndroid Build Coastguard Worker }
158*8b26181fSAndroid Build Coastguard Worker
159*8b26181fSAndroid Build Coastguard Worker /*
160*8b26181fSAndroid Build Coastguard Worker * Loop over the entries.
161*8b26181fSAndroid Build Coastguard Worker */
162*8b26181fSAndroid Build Coastguard Worker ifrp = (struct lifreq *)buf;
163*8b26181fSAndroid Build Coastguard Worker ifend = (struct lifreq *)(buf + ifc.lifc_len);
164*8b26181fSAndroid Build Coastguard Worker
165*8b26181fSAndroid Build Coastguard Worker for (; ifrp < ifend; ifrp++) {
166*8b26181fSAndroid Build Coastguard Worker /*
167*8b26181fSAndroid Build Coastguard Worker * Skip entries that begin with "dummy".
168*8b26181fSAndroid Build Coastguard Worker * XXX - what are these? Is this Linux-specific?
169*8b26181fSAndroid Build Coastguard Worker * Are there platforms on which we shouldn't do this?
170*8b26181fSAndroid Build Coastguard Worker */
171*8b26181fSAndroid Build Coastguard Worker if (strncmp(ifrp->lifr_name, "dummy", 5) == 0)
172*8b26181fSAndroid Build Coastguard Worker continue;
173*8b26181fSAndroid Build Coastguard Worker
174*8b26181fSAndroid Build Coastguard Worker /*
175*8b26181fSAndroid Build Coastguard Worker * Can we capture on this device?
176*8b26181fSAndroid Build Coastguard Worker */
177*8b26181fSAndroid Build Coastguard Worker if (!(*check_usable)(ifrp->lifr_name)) {
178*8b26181fSAndroid Build Coastguard Worker /*
179*8b26181fSAndroid Build Coastguard Worker * No.
180*8b26181fSAndroid Build Coastguard Worker */
181*8b26181fSAndroid Build Coastguard Worker continue;
182*8b26181fSAndroid Build Coastguard Worker }
183*8b26181fSAndroid Build Coastguard Worker
184*8b26181fSAndroid Build Coastguard Worker /*
185*8b26181fSAndroid Build Coastguard Worker * IPv6 or not?
186*8b26181fSAndroid Build Coastguard Worker */
187*8b26181fSAndroid Build Coastguard Worker if (((struct sockaddr *)&ifrp->lifr_addr)->sa_family == AF_INET6)
188*8b26181fSAndroid Build Coastguard Worker fd = fd6;
189*8b26181fSAndroid Build Coastguard Worker else
190*8b26181fSAndroid Build Coastguard Worker fd = fd4;
191*8b26181fSAndroid Build Coastguard Worker
192*8b26181fSAndroid Build Coastguard Worker /*
193*8b26181fSAndroid Build Coastguard Worker * Get the flags for this interface.
194*8b26181fSAndroid Build Coastguard Worker */
195*8b26181fSAndroid Build Coastguard Worker strncpy(ifrflags.lifr_name, ifrp->lifr_name,
196*8b26181fSAndroid Build Coastguard Worker sizeof(ifrflags.lifr_name));
197*8b26181fSAndroid Build Coastguard Worker if (ioctl(fd, SIOCGLIFFLAGS, (char *)&ifrflags) < 0) {
198*8b26181fSAndroid Build Coastguard Worker if (errno == ENXIO)
199*8b26181fSAndroid Build Coastguard Worker continue;
200*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf, PCAP_ERRBUF_SIZE,
201*8b26181fSAndroid Build Coastguard Worker errno, "SIOCGLIFFLAGS: %.*s",
202*8b26181fSAndroid Build Coastguard Worker (int)sizeof(ifrflags.lifr_name),
203*8b26181fSAndroid Build Coastguard Worker ifrflags.lifr_name);
204*8b26181fSAndroid Build Coastguard Worker ret = -1;
205*8b26181fSAndroid Build Coastguard Worker break;
206*8b26181fSAndroid Build Coastguard Worker }
207*8b26181fSAndroid Build Coastguard Worker
208*8b26181fSAndroid Build Coastguard Worker /*
209*8b26181fSAndroid Build Coastguard Worker * Get the netmask for this address on this interface.
210*8b26181fSAndroid Build Coastguard Worker */
211*8b26181fSAndroid Build Coastguard Worker strncpy(ifrnetmask.lifr_name, ifrp->lifr_name,
212*8b26181fSAndroid Build Coastguard Worker sizeof(ifrnetmask.lifr_name));
213*8b26181fSAndroid Build Coastguard Worker memcpy(&ifrnetmask.lifr_addr, &ifrp->lifr_addr,
214*8b26181fSAndroid Build Coastguard Worker sizeof(ifrnetmask.lifr_addr));
215*8b26181fSAndroid Build Coastguard Worker if (ioctl(fd, SIOCGLIFNETMASK, (char *)&ifrnetmask) < 0) {
216*8b26181fSAndroid Build Coastguard Worker if (errno == EADDRNOTAVAIL) {
217*8b26181fSAndroid Build Coastguard Worker /*
218*8b26181fSAndroid Build Coastguard Worker * Not available.
219*8b26181fSAndroid Build Coastguard Worker */
220*8b26181fSAndroid Build Coastguard Worker netmask = NULL;
221*8b26181fSAndroid Build Coastguard Worker } else {
222*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf,
223*8b26181fSAndroid Build Coastguard Worker PCAP_ERRBUF_SIZE, errno,
224*8b26181fSAndroid Build Coastguard Worker "SIOCGLIFNETMASK: %.*s",
225*8b26181fSAndroid Build Coastguard Worker (int)sizeof(ifrnetmask.lifr_name),
226*8b26181fSAndroid Build Coastguard Worker ifrnetmask.lifr_name);
227*8b26181fSAndroid Build Coastguard Worker ret = -1;
228*8b26181fSAndroid Build Coastguard Worker break;
229*8b26181fSAndroid Build Coastguard Worker }
230*8b26181fSAndroid Build Coastguard Worker } else
231*8b26181fSAndroid Build Coastguard Worker netmask = (struct sockaddr *)&ifrnetmask.lifr_addr;
232*8b26181fSAndroid Build Coastguard Worker
233*8b26181fSAndroid Build Coastguard Worker /*
234*8b26181fSAndroid Build Coastguard Worker * Get the broadcast address for this address on this
235*8b26181fSAndroid Build Coastguard Worker * interface (if any).
236*8b26181fSAndroid Build Coastguard Worker */
237*8b26181fSAndroid Build Coastguard Worker if (ifrflags.lifr_flags & IFF_BROADCAST) {
238*8b26181fSAndroid Build Coastguard Worker strncpy(ifrbroadaddr.lifr_name, ifrp->lifr_name,
239*8b26181fSAndroid Build Coastguard Worker sizeof(ifrbroadaddr.lifr_name));
240*8b26181fSAndroid Build Coastguard Worker memcpy(&ifrbroadaddr.lifr_addr, &ifrp->lifr_addr,
241*8b26181fSAndroid Build Coastguard Worker sizeof(ifrbroadaddr.lifr_addr));
242*8b26181fSAndroid Build Coastguard Worker if (ioctl(fd, SIOCGLIFBRDADDR,
243*8b26181fSAndroid Build Coastguard Worker (char *)&ifrbroadaddr) < 0) {
244*8b26181fSAndroid Build Coastguard Worker if (errno == EADDRNOTAVAIL) {
245*8b26181fSAndroid Build Coastguard Worker /*
246*8b26181fSAndroid Build Coastguard Worker * Not available.
247*8b26181fSAndroid Build Coastguard Worker */
248*8b26181fSAndroid Build Coastguard Worker broadaddr = NULL;
249*8b26181fSAndroid Build Coastguard Worker } else {
250*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf,
251*8b26181fSAndroid Build Coastguard Worker PCAP_ERRBUF_SIZE, errno,
252*8b26181fSAndroid Build Coastguard Worker "SIOCGLIFBRDADDR: %.*s",
253*8b26181fSAndroid Build Coastguard Worker (int)sizeof(ifrbroadaddr.lifr_name),
254*8b26181fSAndroid Build Coastguard Worker ifrbroadaddr.lifr_name);
255*8b26181fSAndroid Build Coastguard Worker ret = -1;
256*8b26181fSAndroid Build Coastguard Worker break;
257*8b26181fSAndroid Build Coastguard Worker }
258*8b26181fSAndroid Build Coastguard Worker } else
259*8b26181fSAndroid Build Coastguard Worker broadaddr = (struct sockaddr *)&ifrbroadaddr.lifr_broadaddr;
260*8b26181fSAndroid Build Coastguard Worker } else {
261*8b26181fSAndroid Build Coastguard Worker /*
262*8b26181fSAndroid Build Coastguard Worker * Not a broadcast interface, so no broadcast
263*8b26181fSAndroid Build Coastguard Worker * address.
264*8b26181fSAndroid Build Coastguard Worker */
265*8b26181fSAndroid Build Coastguard Worker broadaddr = NULL;
266*8b26181fSAndroid Build Coastguard Worker }
267*8b26181fSAndroid Build Coastguard Worker
268*8b26181fSAndroid Build Coastguard Worker /*
269*8b26181fSAndroid Build Coastguard Worker * Get the destination address for this address on this
270*8b26181fSAndroid Build Coastguard Worker * interface (if any).
271*8b26181fSAndroid Build Coastguard Worker */
272*8b26181fSAndroid Build Coastguard Worker if (ifrflags.lifr_flags & IFF_POINTOPOINT) {
273*8b26181fSAndroid Build Coastguard Worker strncpy(ifrdstaddr.lifr_name, ifrp->lifr_name,
274*8b26181fSAndroid Build Coastguard Worker sizeof(ifrdstaddr.lifr_name));
275*8b26181fSAndroid Build Coastguard Worker memcpy(&ifrdstaddr.lifr_addr, &ifrp->lifr_addr,
276*8b26181fSAndroid Build Coastguard Worker sizeof(ifrdstaddr.lifr_addr));
277*8b26181fSAndroid Build Coastguard Worker if (ioctl(fd, SIOCGLIFDSTADDR,
278*8b26181fSAndroid Build Coastguard Worker (char *)&ifrdstaddr) < 0) {
279*8b26181fSAndroid Build Coastguard Worker if (errno == EADDRNOTAVAIL) {
280*8b26181fSAndroid Build Coastguard Worker /*
281*8b26181fSAndroid Build Coastguard Worker * Not available.
282*8b26181fSAndroid Build Coastguard Worker */
283*8b26181fSAndroid Build Coastguard Worker dstaddr = NULL;
284*8b26181fSAndroid Build Coastguard Worker } else {
285*8b26181fSAndroid Build Coastguard Worker pcap_fmt_errmsg_for_errno(errbuf,
286*8b26181fSAndroid Build Coastguard Worker PCAP_ERRBUF_SIZE, errno,
287*8b26181fSAndroid Build Coastguard Worker "SIOCGLIFDSTADDR: %.*s",
288*8b26181fSAndroid Build Coastguard Worker (int)sizeof(ifrdstaddr.lifr_name),
289*8b26181fSAndroid Build Coastguard Worker ifrdstaddr.lifr_name);
290*8b26181fSAndroid Build Coastguard Worker ret = -1;
291*8b26181fSAndroid Build Coastguard Worker break;
292*8b26181fSAndroid Build Coastguard Worker }
293*8b26181fSAndroid Build Coastguard Worker } else
294*8b26181fSAndroid Build Coastguard Worker dstaddr = (struct sockaddr *)&ifrdstaddr.lifr_dstaddr;
295*8b26181fSAndroid Build Coastguard Worker } else
296*8b26181fSAndroid Build Coastguard Worker dstaddr = NULL;
297*8b26181fSAndroid Build Coastguard Worker
298*8b26181fSAndroid Build Coastguard Worker #ifdef HAVE_SOLARIS
299*8b26181fSAndroid Build Coastguard Worker /*
300*8b26181fSAndroid Build Coastguard Worker * If this entry has a colon followed by a number at
301*8b26181fSAndroid Build Coastguard Worker * the end, it's a logical interface. Those are just
302*8b26181fSAndroid Build Coastguard Worker * the way you assign multiple IP addresses to a real
303*8b26181fSAndroid Build Coastguard Worker * interface, so an entry for a logical interface should
304*8b26181fSAndroid Build Coastguard Worker * be treated like the entry for the real interface;
305*8b26181fSAndroid Build Coastguard Worker * we do that by stripping off the ":" and the number.
306*8b26181fSAndroid Build Coastguard Worker */
307*8b26181fSAndroid Build Coastguard Worker p = strchr(ifrp->lifr_name, ':');
308*8b26181fSAndroid Build Coastguard Worker if (p != NULL) {
309*8b26181fSAndroid Build Coastguard Worker /*
310*8b26181fSAndroid Build Coastguard Worker * We have a ":"; is it followed by a number?
311*8b26181fSAndroid Build Coastguard Worker */
312*8b26181fSAndroid Build Coastguard Worker q = p + 1;
313*8b26181fSAndroid Build Coastguard Worker while (PCAP_ISDIGIT(*q))
314*8b26181fSAndroid Build Coastguard Worker q++;
315*8b26181fSAndroid Build Coastguard Worker if (*q == '\0') {
316*8b26181fSAndroid Build Coastguard Worker /*
317*8b26181fSAndroid Build Coastguard Worker * All digits after the ":" until the end.
318*8b26181fSAndroid Build Coastguard Worker * Strip off the ":" and everything after
319*8b26181fSAndroid Build Coastguard Worker * it.
320*8b26181fSAndroid Build Coastguard Worker */
321*8b26181fSAndroid Build Coastguard Worker *p = '\0';
322*8b26181fSAndroid Build Coastguard Worker }
323*8b26181fSAndroid Build Coastguard Worker }
324*8b26181fSAndroid Build Coastguard Worker #endif
325*8b26181fSAndroid Build Coastguard Worker
326*8b26181fSAndroid Build Coastguard Worker /*
327*8b26181fSAndroid Build Coastguard Worker * Add information for this address to the list.
328*8b26181fSAndroid Build Coastguard Worker */
329*8b26181fSAndroid Build Coastguard Worker if (add_addr_to_if(devlistp, ifrp->lifr_name,
330*8b26181fSAndroid Build Coastguard Worker ifrflags.lifr_flags, get_flags_func,
331*8b26181fSAndroid Build Coastguard Worker (struct sockaddr *)&ifrp->lifr_addr,
332*8b26181fSAndroid Build Coastguard Worker sizeof (struct sockaddr_storage),
333*8b26181fSAndroid Build Coastguard Worker netmask, sizeof (struct sockaddr_storage),
334*8b26181fSAndroid Build Coastguard Worker broadaddr, sizeof (struct sockaddr_storage),
335*8b26181fSAndroid Build Coastguard Worker dstaddr, sizeof (struct sockaddr_storage), errbuf) < 0) {
336*8b26181fSAndroid Build Coastguard Worker ret = -1;
337*8b26181fSAndroid Build Coastguard Worker break;
338*8b26181fSAndroid Build Coastguard Worker }
339*8b26181fSAndroid Build Coastguard Worker }
340*8b26181fSAndroid Build Coastguard Worker free(buf);
341*8b26181fSAndroid Build Coastguard Worker (void)close(fd6);
342*8b26181fSAndroid Build Coastguard Worker (void)close(fd4);
343*8b26181fSAndroid Build Coastguard Worker
344*8b26181fSAndroid Build Coastguard Worker return (ret);
345*8b26181fSAndroid Build Coastguard Worker }
346