xref: /xnu-11215.41.3/bsd/netinet/in.c (revision 33de042d024d46de5ff4e89f2471de6608e37fa4)
1*33de042dSApple OSS Distributions /*
2*33de042dSApple OSS Distributions  * Copyright (c) 2000-2021 Apple Inc. All rights reserved.
3*33de042dSApple OSS Distributions  *
4*33de042dSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*33de042dSApple OSS Distributions  *
6*33de042dSApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*33de042dSApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*33de042dSApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*33de042dSApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*33de042dSApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*33de042dSApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*33de042dSApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*33de042dSApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*33de042dSApple OSS Distributions  *
15*33de042dSApple OSS Distributions  * Please obtain a copy of the License at
16*33de042dSApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*33de042dSApple OSS Distributions  *
18*33de042dSApple OSS Distributions  * The Original Code and all software distributed under the License are
19*33de042dSApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*33de042dSApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*33de042dSApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*33de042dSApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*33de042dSApple OSS Distributions  * Please see the License for the specific language governing rights and
24*33de042dSApple OSS Distributions  * limitations under the License.
25*33de042dSApple OSS Distributions  *
26*33de042dSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*33de042dSApple OSS Distributions  */
28*33de042dSApple OSS Distributions /*
29*33de042dSApple OSS Distributions  * Copyright (c) 1982, 1986, 1991, 1993
30*33de042dSApple OSS Distributions  *	The Regents of the University of California.  All rights reserved.
31*33de042dSApple OSS Distributions  *
32*33de042dSApple OSS Distributions  * Redistribution and use in source and binary forms, with or without
33*33de042dSApple OSS Distributions  * modification, are permitted provided that the following conditions
34*33de042dSApple OSS Distributions  * are met:
35*33de042dSApple OSS Distributions  * 1. Redistributions of source code must retain the above copyright
36*33de042dSApple OSS Distributions  *    notice, this list of conditions and the following disclaimer.
37*33de042dSApple OSS Distributions  * 2. Redistributions in binary form must reproduce the above copyright
38*33de042dSApple OSS Distributions  *    notice, this list of conditions and the following disclaimer in the
39*33de042dSApple OSS Distributions  *    documentation and/or other materials provided with the distribution.
40*33de042dSApple OSS Distributions  * 3. All advertising materials mentioning features or use of this software
41*33de042dSApple OSS Distributions  *    must display the following acknowledgement:
42*33de042dSApple OSS Distributions  *	This product includes software developed by the University of
43*33de042dSApple OSS Distributions  *	California, Berkeley and its contributors.
44*33de042dSApple OSS Distributions  * 4. Neither the name of the University nor the names of its contributors
45*33de042dSApple OSS Distributions  *    may be used to endorse or promote products derived from this software
46*33de042dSApple OSS Distributions  *    without specific prior written permission.
47*33de042dSApple OSS Distributions  *
48*33de042dSApple OSS Distributions  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
49*33de042dSApple OSS Distributions  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
50*33de042dSApple OSS Distributions  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
51*33de042dSApple OSS Distributions  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
52*33de042dSApple OSS Distributions  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
53*33de042dSApple OSS Distributions  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
54*33de042dSApple OSS Distributions  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
55*33de042dSApple OSS Distributions  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
56*33de042dSApple OSS Distributions  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
57*33de042dSApple OSS Distributions  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
58*33de042dSApple OSS Distributions  * SUCH DAMAGE.
59*33de042dSApple OSS Distributions  *
60*33de042dSApple OSS Distributions  *	@(#)in.c	8.4 (Berkeley) 1/9/95
61*33de042dSApple OSS Distributions  */
62*33de042dSApple OSS Distributions 
63*33de042dSApple OSS Distributions #include <sys/param.h>
64*33de042dSApple OSS Distributions #include <sys/systm.h>
65*33de042dSApple OSS Distributions #include <sys/sockio.h>
66*33de042dSApple OSS Distributions #include <sys/socketvar.h>
67*33de042dSApple OSS Distributions #include <sys/malloc.h>
68*33de042dSApple OSS Distributions #include <sys/proc.h>
69*33de042dSApple OSS Distributions #include <sys/socket.h>
70*33de042dSApple OSS Distributions #include <sys/kernel.h>
71*33de042dSApple OSS Distributions #include <sys/sysctl.h>
72*33de042dSApple OSS Distributions #include <sys/kern_event.h>
73*33de042dSApple OSS Distributions #include <sys/syslog.h>
74*33de042dSApple OSS Distributions #include <sys/mcache.h>
75*33de042dSApple OSS Distributions #include <sys/protosw.h>
76*33de042dSApple OSS Distributions #include <sys/file.h>
77*33de042dSApple OSS Distributions 
78*33de042dSApple OSS Distributions #include <kern/zalloc.h>
79*33de042dSApple OSS Distributions #include <pexpert/pexpert.h>
80*33de042dSApple OSS Distributions #include <os/log.h>
81*33de042dSApple OSS Distributions 
82*33de042dSApple OSS Distributions #include <net/if.h>
83*33de042dSApple OSS Distributions #include <net/if_types.h>
84*33de042dSApple OSS Distributions #include <net/route.h>
85*33de042dSApple OSS Distributions #include <net/kpi_protocol.h>
86*33de042dSApple OSS Distributions #include <net/dlil.h>
87*33de042dSApple OSS Distributions #if PF
88*33de042dSApple OSS Distributions #include <net/pfvar.h>
89*33de042dSApple OSS Distributions #endif /* PF */
90*33de042dSApple OSS Distributions 
91*33de042dSApple OSS Distributions #include <netinet/in.h>
92*33de042dSApple OSS Distributions #include <netinet/in_var.h>
93*33de042dSApple OSS Distributions #include <netinet/in_pcb.h>
94*33de042dSApple OSS Distributions #include <netinet/igmp_var.h>
95*33de042dSApple OSS Distributions #include <netinet/ip_var.h>
96*33de042dSApple OSS Distributions #include <netinet/tcp.h>
97*33de042dSApple OSS Distributions #include <netinet/tcp_timer.h>
98*33de042dSApple OSS Distributions #include <netinet/tcp_var.h>
99*33de042dSApple OSS Distributions 
100*33de042dSApple OSS Distributions #include <net/sockaddr_utils.h>
101*33de042dSApple OSS Distributions 
102*33de042dSApple OSS Distributions static int inctl_associd(struct socket *, u_long, caddr_t __indexable);
103*33de042dSApple OSS Distributions static int inctl_connid(struct socket *, u_long, caddr_t __indexable);
104*33de042dSApple OSS Distributions static int inctl_conninfo(struct socket *, u_long, caddr_t __indexable);
105*33de042dSApple OSS Distributions 
106*33de042dSApple OSS Distributions static int inctl_autoaddr(struct ifnet *, struct ifreq *);
107*33de042dSApple OSS Distributions static int inctl_arpipll(struct ifnet *, struct ifreq *);
108*33de042dSApple OSS Distributions static int inctl_setrouter(struct ifnet *, struct ifreq *);
109*33de042dSApple OSS Distributions static int inctl_ifaddr(struct ifnet *, struct in_ifaddr *, u_long,
110*33de042dSApple OSS Distributions     struct ifreq *);
111*33de042dSApple OSS Distributions static int inctl_ifdstaddr(struct ifnet *, struct in_ifaddr *, u_long,
112*33de042dSApple OSS Distributions     struct ifreq *);
113*33de042dSApple OSS Distributions static int inctl_ifbrdaddr(struct ifnet *, struct in_ifaddr *, u_long,
114*33de042dSApple OSS Distributions     struct ifreq *);
115*33de042dSApple OSS Distributions static int inctl_ifnetmask(struct ifnet *, struct in_ifaddr *, u_long,
116*33de042dSApple OSS Distributions     struct ifreq *);
117*33de042dSApple OSS Distributions 
118*33de042dSApple OSS Distributions static void in_socktrim(struct sockaddr_in *);
119*33de042dSApple OSS Distributions static int in_ifinit(struct ifnet *, struct in_ifaddr *,
120*33de042dSApple OSS Distributions     struct sockaddr_in *, int);
121*33de042dSApple OSS Distributions 
122*33de042dSApple OSS Distributions #define IA_HASH_INIT(ia) {                                      \
123*33de042dSApple OSS Distributions 	(ia)->ia_hash.tqe_next = __unsafe_forge_single(void *, ~(uintptr_t)0); \
124*33de042dSApple OSS Distributions 	(ia)->ia_hash.tqe_prev = __unsafe_forge_single(void *, ~(uintptr_t)0); \
125*33de042dSApple OSS Distributions }
126*33de042dSApple OSS Distributions 
127*33de042dSApple OSS Distributions #define IA_IS_HASHED(ia)                                        \
128*33de042dSApple OSS Distributions 	(!((ia)->ia_hash.tqe_next == __unsafe_forge_single(void *, ~(uintptr_t)0) ||  \
129*33de042dSApple OSS Distributions 	(ia)->ia_hash.tqe_prev == __unsafe_forge_single(void *, ~(uintptr_t)0)))
130*33de042dSApple OSS Distributions 
131*33de042dSApple OSS Distributions static void in_iahash_remove(struct in_ifaddr *);
132*33de042dSApple OSS Distributions static void in_iahash_insert(struct in_ifaddr *);
133*33de042dSApple OSS Distributions static void in_iahash_insert_ptp(struct in_ifaddr *);
134*33de042dSApple OSS Distributions static struct in_ifaddr *in_ifaddr_alloc(void);
135*33de042dSApple OSS Distributions static void in_ifaddr_free(struct ifaddr *);
136*33de042dSApple OSS Distributions 
137*33de042dSApple OSS Distributions static int in_getassocids(struct socket *, uint32_t *, user_addr_t);
138*33de042dSApple OSS Distributions static int in_getconnids(struct socket *, sae_associd_t, uint32_t *, user_addr_t);
139*33de042dSApple OSS Distributions 
140*33de042dSApple OSS Distributions static int subnetsarelocal = 0;
141*33de042dSApple OSS Distributions SYSCTL_INT(_net_inet_ip, OID_AUTO, subnets_are_local,
142*33de042dSApple OSS Distributions     CTLFLAG_RW | CTLFLAG_LOCKED, &subnetsarelocal, 0, "");
143*33de042dSApple OSS Distributions 
144*33de042dSApple OSS Distributions /* Track whether or not the SIOCARPIPLL ioctl has been called */
145*33de042dSApple OSS Distributions u_int32_t ipv4_ll_arp_aware = 0;
146*33de042dSApple OSS Distributions 
147*33de042dSApple OSS Distributions /*
148*33de042dSApple OSS Distributions  * Return 1 if the address is
149*33de042dSApple OSS Distributions  * - loopback
150*33de042dSApple OSS Distributions  * - unicast or multicast link local
151*33de042dSApple OSS Distributions  * - routed via a link level gateway
152*33de042dSApple OSS Distributions  * - belongs to a directly connected (sub)net
153*33de042dSApple OSS Distributions  */
154*33de042dSApple OSS Distributions int
inaddr_local(struct in_addr in)155*33de042dSApple OSS Distributions inaddr_local(struct in_addr in)
156*33de042dSApple OSS Distributions {
157*33de042dSApple OSS Distributions 	struct rtentry *__single rt;
158*33de042dSApple OSS Distributions 	struct sockaddr_in sin;
159*33de042dSApple OSS Distributions 	int local = 0;
160*33de042dSApple OSS Distributions 
161*33de042dSApple OSS Distributions 	if (ntohl(in.s_addr) == INADDR_LOOPBACK ||
162*33de042dSApple OSS Distributions 	    IN_LINKLOCAL(ntohl(in.s_addr))) {
163*33de042dSApple OSS Distributions 		local = 1;
164*33de042dSApple OSS Distributions 	} else if (ntohl(in.s_addr) >= INADDR_UNSPEC_GROUP &&
165*33de042dSApple OSS Distributions 	    ntohl(in.s_addr) <= INADDR_MAX_LOCAL_GROUP) {
166*33de042dSApple OSS Distributions 		local = 1;
167*33de042dSApple OSS Distributions 	} else {
168*33de042dSApple OSS Distributions 		sin.sin_family = AF_INET;
169*33de042dSApple OSS Distributions 		sin.sin_len = sizeof(sin);
170*33de042dSApple OSS Distributions 		sin.sin_addr = in;
171*33de042dSApple OSS Distributions 		rt = rtalloc1(SA(&sin), 0, 0);
172*33de042dSApple OSS Distributions 
173*33de042dSApple OSS Distributions 		if (rt != NULL) {
174*33de042dSApple OSS Distributions 			RT_LOCK_SPIN(rt);
175*33de042dSApple OSS Distributions 			if (rt->rt_gateway->sa_family == AF_LINK ||
176*33de042dSApple OSS Distributions 			    (rt->rt_ifp->if_flags & IFF_LOOPBACK)) {
177*33de042dSApple OSS Distributions 				local = 1;
178*33de042dSApple OSS Distributions 			}
179*33de042dSApple OSS Distributions 			RT_UNLOCK(rt);
180*33de042dSApple OSS Distributions 			rtfree(rt);
181*33de042dSApple OSS Distributions 		} else {
182*33de042dSApple OSS Distributions 			local = in_localaddr(in);
183*33de042dSApple OSS Distributions 		}
184*33de042dSApple OSS Distributions 	}
185*33de042dSApple OSS Distributions 	return local;
186*33de042dSApple OSS Distributions }
187*33de042dSApple OSS Distributions 
188*33de042dSApple OSS Distributions /*
189*33de042dSApple OSS Distributions  * Return 1 if an internet address is for a ``local'' host
190*33de042dSApple OSS Distributions  * (one to which we have a connection).  If subnetsarelocal
191*33de042dSApple OSS Distributions  * is true, this includes other subnets of the local net,
192*33de042dSApple OSS Distributions  * otherwise, it includes the directly-connected (sub)nets.
193*33de042dSApple OSS Distributions  * The IPv4 link local prefix 169.254/16 is also included.
194*33de042dSApple OSS Distributions  */
195*33de042dSApple OSS Distributions int
in_localaddr(struct in_addr in)196*33de042dSApple OSS Distributions in_localaddr(struct in_addr in)
197*33de042dSApple OSS Distributions {
198*33de042dSApple OSS Distributions 	u_int32_t i = ntohl(in.s_addr);
199*33de042dSApple OSS Distributions 	struct in_ifaddr *__single ia;
200*33de042dSApple OSS Distributions 
201*33de042dSApple OSS Distributions 	if (IN_LINKLOCAL(i)) {
202*33de042dSApple OSS Distributions 		return 1;
203*33de042dSApple OSS Distributions 	}
204*33de042dSApple OSS Distributions 
205*33de042dSApple OSS Distributions 	if (subnetsarelocal) {
206*33de042dSApple OSS Distributions 		lck_rw_lock_shared(&in_ifaddr_rwlock);
207*33de042dSApple OSS Distributions 		for (ia = in_ifaddrhead.tqh_first; ia != NULL;
208*33de042dSApple OSS Distributions 		    ia = ia->ia_link.tqe_next) {
209*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
210*33de042dSApple OSS Distributions 			if ((i & ia->ia_netmask) == ia->ia_net) {
211*33de042dSApple OSS Distributions 				IFA_UNLOCK(&ia->ia_ifa);
212*33de042dSApple OSS Distributions 				lck_rw_done(&in_ifaddr_rwlock);
213*33de042dSApple OSS Distributions 				return 1;
214*33de042dSApple OSS Distributions 			}
215*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
216*33de042dSApple OSS Distributions 		}
217*33de042dSApple OSS Distributions 		lck_rw_done(&in_ifaddr_rwlock);
218*33de042dSApple OSS Distributions 	} else {
219*33de042dSApple OSS Distributions 		lck_rw_lock_shared(&in_ifaddr_rwlock);
220*33de042dSApple OSS Distributions 		for (ia = in_ifaddrhead.tqh_first; ia != NULL;
221*33de042dSApple OSS Distributions 		    ia = ia->ia_link.tqe_next) {
222*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
223*33de042dSApple OSS Distributions 			if ((i & ia->ia_subnetmask) == ia->ia_subnet) {
224*33de042dSApple OSS Distributions 				IFA_UNLOCK(&ia->ia_ifa);
225*33de042dSApple OSS Distributions 				lck_rw_done(&in_ifaddr_rwlock);
226*33de042dSApple OSS Distributions 				return 1;
227*33de042dSApple OSS Distributions 			}
228*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
229*33de042dSApple OSS Distributions 		}
230*33de042dSApple OSS Distributions 		lck_rw_done(&in_ifaddr_rwlock);
231*33de042dSApple OSS Distributions 	}
232*33de042dSApple OSS Distributions 	return 0;
233*33de042dSApple OSS Distributions }
234*33de042dSApple OSS Distributions 
235*33de042dSApple OSS Distributions /*
236*33de042dSApple OSS Distributions  * Determine whether an IP address is in a reserved set of addresses
237*33de042dSApple OSS Distributions  * that may not be forwarded, or whether datagrams to that destination
238*33de042dSApple OSS Distributions  * may be forwarded.
239*33de042dSApple OSS Distributions  */
240*33de042dSApple OSS Distributions boolean_t
in_canforward(struct in_addr in)241*33de042dSApple OSS Distributions in_canforward(struct in_addr in)
242*33de042dSApple OSS Distributions {
243*33de042dSApple OSS Distributions 	u_int32_t i = ntohl(in.s_addr);
244*33de042dSApple OSS Distributions 	u_int32_t net;
245*33de042dSApple OSS Distributions 
246*33de042dSApple OSS Distributions 	if (IN_EXPERIMENTAL(i) || IN_MULTICAST(i)) {
247*33de042dSApple OSS Distributions 		return FALSE;
248*33de042dSApple OSS Distributions 	}
249*33de042dSApple OSS Distributions 	if (IN_CLASSA(i)) {
250*33de042dSApple OSS Distributions 		net = i & IN_CLASSA_NET;
251*33de042dSApple OSS Distributions 		if (net == 0 || net == (IN_LOOPBACKNET << IN_CLASSA_NSHIFT)) {
252*33de042dSApple OSS Distributions 			return FALSE;
253*33de042dSApple OSS Distributions 		}
254*33de042dSApple OSS Distributions 	}
255*33de042dSApple OSS Distributions 	return TRUE;
256*33de042dSApple OSS Distributions }
257*33de042dSApple OSS Distributions 
258*33de042dSApple OSS Distributions /*
259*33de042dSApple OSS Distributions  * Trim a mask in a sockaddr
260*33de042dSApple OSS Distributions  */
261*33de042dSApple OSS Distributions static void
in_socktrim(struct sockaddr_in * ap)262*33de042dSApple OSS Distributions in_socktrim(struct sockaddr_in *ap)
263*33de042dSApple OSS Distributions {
264*33de042dSApple OSS Distributions 	char *cplim = (char *)&ap->sin_addr;
265*33de042dSApple OSS Distributions 	char *cp = (char *)(&ap->sin_addr + 1);
266*33de042dSApple OSS Distributions 
267*33de042dSApple OSS Distributions 	ap->sin_len = 0;
268*33de042dSApple OSS Distributions 	while (--cp >= cplim) {
269*33de042dSApple OSS Distributions 		if (*cp) {
270*33de042dSApple OSS Distributions 			(ap)->sin_len = (uint8_t)(cp - (char *)(ap) + 1);
271*33de042dSApple OSS Distributions 			break;
272*33de042dSApple OSS Distributions 		}
273*33de042dSApple OSS Distributions 	}
274*33de042dSApple OSS Distributions }
275*33de042dSApple OSS Distributions 
276*33de042dSApple OSS Distributions static int in_interfaces;       /* number of external internet interfaces */
277*33de042dSApple OSS Distributions 
278*33de042dSApple OSS Distributions static int
in_domifattach(struct ifnet * ifp)279*33de042dSApple OSS Distributions in_domifattach(struct ifnet *ifp)
280*33de042dSApple OSS Distributions {
281*33de042dSApple OSS Distributions 	int error = 0;
282*33de042dSApple OSS Distributions 
283*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
284*33de042dSApple OSS Distributions 
285*33de042dSApple OSS Distributions 	if ((error = proto_plumb(PF_INET, ifp)) && error != EEXIST) {
286*33de042dSApple OSS Distributions 		log(LOG_ERR, "%s: proto_plumb returned %d if=%s\n",
287*33de042dSApple OSS Distributions 		    __func__, error, if_name(ifp));
288*33de042dSApple OSS Distributions 		return error;
289*33de042dSApple OSS Distributions 	}
290*33de042dSApple OSS Distributions 
291*33de042dSApple OSS Distributions 	if (ifp->if_inetdata == NULL) {
292*33de042dSApple OSS Distributions 		ifp->if_inetdata = zalloc_permanent_type(struct in_ifextra);
293*33de042dSApple OSS Distributions 		error = 0;
294*33de042dSApple OSS Distributions 	} else if (error != EEXIST) {
295*33de042dSApple OSS Distributions 		/*
296*33de042dSApple OSS Distributions 		 * Since the structure is never freed, we need to
297*33de042dSApple OSS Distributions 		 * zero out its contents to avoid reusing stale data.
298*33de042dSApple OSS Distributions 		 * A little redundant with allocation above, but it
299*33de042dSApple OSS Distributions 		 * keeps the code simpler for all cases.
300*33de042dSApple OSS Distributions 		 */
301*33de042dSApple OSS Distributions 		IN_IFEXTRA(ifp)->netsig_len = 0;
302*33de042dSApple OSS Distributions 		bzero(IN_IFEXTRA(ifp)->netsig, sizeof(IN_IFEXTRA(ifp)->netsig));
303*33de042dSApple OSS Distributions 	}
304*33de042dSApple OSS Distributions 	return error;
305*33de042dSApple OSS Distributions }
306*33de042dSApple OSS Distributions 
307*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_associd(struct socket * so,u_long cmd,caddr_t __indexable data)308*33de042dSApple OSS Distributions inctl_associd(struct socket *so, u_long cmd, caddr_t __indexable data)
309*33de042dSApple OSS Distributions {
310*33de042dSApple OSS Distributions 	int error = 0;
311*33de042dSApple OSS Distributions 	union {
312*33de042dSApple OSS Distributions 		struct so_aidreq32 a32;
313*33de042dSApple OSS Distributions 		struct so_aidreq64 a64;
314*33de042dSApple OSS Distributions 	} u;
315*33de042dSApple OSS Distributions 
316*33de042dSApple OSS Distributions 	VERIFY(so != NULL);
317*33de042dSApple OSS Distributions 
318*33de042dSApple OSS Distributions 	switch (cmd) {
319*33de042dSApple OSS Distributions 	case SIOCGASSOCIDS32:           /* struct so_aidreq32 */
320*33de042dSApple OSS Distributions 		bcopy(data, &u.a32, sizeof(u.a32));
321*33de042dSApple OSS Distributions 		error = in_getassocids(so, &u.a32.sar_cnt, u.a32.sar_aidp);
322*33de042dSApple OSS Distributions 		if (error == 0) {
323*33de042dSApple OSS Distributions 			bcopy(&u.a32, data, sizeof(u.a32));
324*33de042dSApple OSS Distributions 		}
325*33de042dSApple OSS Distributions 		break;
326*33de042dSApple OSS Distributions 
327*33de042dSApple OSS Distributions 	case SIOCGASSOCIDS64:           /* struct so_aidreq64 */
328*33de042dSApple OSS Distributions 		bcopy(data, &u.a64, sizeof(u.a64));
329*33de042dSApple OSS Distributions 		error = in_getassocids(so, &u.a64.sar_cnt, (user_addr_t)u.a64.sar_aidp);
330*33de042dSApple OSS Distributions 		if (error == 0) {
331*33de042dSApple OSS Distributions 			bcopy(&u.a64, data, sizeof(u.a64));
332*33de042dSApple OSS Distributions 		}
333*33de042dSApple OSS Distributions 		break;
334*33de042dSApple OSS Distributions 
335*33de042dSApple OSS Distributions 	default:
336*33de042dSApple OSS Distributions 		VERIFY(0);
337*33de042dSApple OSS Distributions 		/* NOTREACHED */
338*33de042dSApple OSS Distributions 	}
339*33de042dSApple OSS Distributions 
340*33de042dSApple OSS Distributions 	return error;
341*33de042dSApple OSS Distributions }
342*33de042dSApple OSS Distributions 
343*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_connid(struct socket * so,u_long cmd,caddr_t __indexable data)344*33de042dSApple OSS Distributions inctl_connid(struct socket *so, u_long cmd, caddr_t __indexable data)
345*33de042dSApple OSS Distributions {
346*33de042dSApple OSS Distributions 	int error = 0;
347*33de042dSApple OSS Distributions 	union {
348*33de042dSApple OSS Distributions 		struct so_cidreq32 c32;
349*33de042dSApple OSS Distributions 		struct so_cidreq64 c64;
350*33de042dSApple OSS Distributions 	} u;
351*33de042dSApple OSS Distributions 
352*33de042dSApple OSS Distributions 	VERIFY(so != NULL);
353*33de042dSApple OSS Distributions 
354*33de042dSApple OSS Distributions 	switch (cmd) {
355*33de042dSApple OSS Distributions 	case SIOCGCONNIDS32:            /* struct so_cidreq32 */
356*33de042dSApple OSS Distributions 		bcopy(data, &u.c32, sizeof(u.c32));
357*33de042dSApple OSS Distributions 		error = in_getconnids(so, u.c32.scr_aid, &u.c32.scr_cnt,
358*33de042dSApple OSS Distributions 		    u.c32.scr_cidp);
359*33de042dSApple OSS Distributions 		if (error == 0) {
360*33de042dSApple OSS Distributions 			bcopy(&u.c32, data, sizeof(u.c32));
361*33de042dSApple OSS Distributions 		}
362*33de042dSApple OSS Distributions 		break;
363*33de042dSApple OSS Distributions 
364*33de042dSApple OSS Distributions 	case SIOCGCONNIDS64:            /* struct so_cidreq64 */
365*33de042dSApple OSS Distributions 		bcopy(data, &u.c64, sizeof(u.c64));
366*33de042dSApple OSS Distributions 		error = in_getconnids(so, u.c64.scr_aid, &u.c64.scr_cnt,
367*33de042dSApple OSS Distributions 		    (user_addr_t)u.c64.scr_cidp);
368*33de042dSApple OSS Distributions 		if (error == 0) {
369*33de042dSApple OSS Distributions 			bcopy(&u.c64, data, sizeof(u.c64));
370*33de042dSApple OSS Distributions 		}
371*33de042dSApple OSS Distributions 		break;
372*33de042dSApple OSS Distributions 
373*33de042dSApple OSS Distributions 	default:
374*33de042dSApple OSS Distributions 		VERIFY(0);
375*33de042dSApple OSS Distributions 		/* NOTREACHED */
376*33de042dSApple OSS Distributions 	}
377*33de042dSApple OSS Distributions 
378*33de042dSApple OSS Distributions 	return error;
379*33de042dSApple OSS Distributions }
380*33de042dSApple OSS Distributions 
381*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_conninfo(struct socket * so,u_long cmd,caddr_t __indexable data)382*33de042dSApple OSS Distributions inctl_conninfo(struct socket *so, u_long cmd, caddr_t __indexable data)
383*33de042dSApple OSS Distributions {
384*33de042dSApple OSS Distributions 	int error = 0;
385*33de042dSApple OSS Distributions 	union {
386*33de042dSApple OSS Distributions 		struct so_cinforeq32 ci32;
387*33de042dSApple OSS Distributions 		struct so_cinforeq64 ci64;
388*33de042dSApple OSS Distributions 	} u;
389*33de042dSApple OSS Distributions 
390*33de042dSApple OSS Distributions 	VERIFY(so != NULL);
391*33de042dSApple OSS Distributions 
392*33de042dSApple OSS Distributions 	switch (cmd) {
393*33de042dSApple OSS Distributions 	case SIOCGCONNINFO32:           /* struct so_cinforeq32 */
394*33de042dSApple OSS Distributions 		bcopy(data, &u.ci32, sizeof(u.ci32));
395*33de042dSApple OSS Distributions 		error = in_getconninfo(so, u.ci32.scir_cid, &u.ci32.scir_flags,
396*33de042dSApple OSS Distributions 		    &u.ci32.scir_ifindex, &u.ci32.scir_error, u.ci32.scir_src,
397*33de042dSApple OSS Distributions 		    &u.ci32.scir_src_len, u.ci32.scir_dst, &u.ci32.scir_dst_len,
398*33de042dSApple OSS Distributions 		    &u.ci32.scir_aux_type, u.ci32.scir_aux_data,
399*33de042dSApple OSS Distributions 		    &u.ci32.scir_aux_len);
400*33de042dSApple OSS Distributions 		if (error == 0) {
401*33de042dSApple OSS Distributions 			bcopy(&u.ci32, data, sizeof(u.ci32));
402*33de042dSApple OSS Distributions 		}
403*33de042dSApple OSS Distributions 		break;
404*33de042dSApple OSS Distributions 
405*33de042dSApple OSS Distributions 	case SIOCGCONNINFO64:           /* struct so_cinforeq64 */
406*33de042dSApple OSS Distributions 		bcopy(data, &u.ci64, sizeof(u.ci64));
407*33de042dSApple OSS Distributions 		error = in_getconninfo(so, u.ci64.scir_cid, &u.ci64.scir_flags,
408*33de042dSApple OSS Distributions 		    &u.ci64.scir_ifindex, &u.ci64.scir_error, (user_addr_t)u.ci64.scir_src,
409*33de042dSApple OSS Distributions 		    &u.ci64.scir_src_len, (user_addr_t)u.ci64.scir_dst, &u.ci64.scir_dst_len,
410*33de042dSApple OSS Distributions 		    &u.ci64.scir_aux_type, (user_addr_t)u.ci64.scir_aux_data,
411*33de042dSApple OSS Distributions 		    &u.ci64.scir_aux_len);
412*33de042dSApple OSS Distributions 		if (error == 0) {
413*33de042dSApple OSS Distributions 			bcopy(&u.ci64, data, sizeof(u.ci64));
414*33de042dSApple OSS Distributions 		}
415*33de042dSApple OSS Distributions 		break;
416*33de042dSApple OSS Distributions 
417*33de042dSApple OSS Distributions 	default:
418*33de042dSApple OSS Distributions 		VERIFY(0);
419*33de042dSApple OSS Distributions 		/* NOTREACHED */
420*33de042dSApple OSS Distributions 	}
421*33de042dSApple OSS Distributions 
422*33de042dSApple OSS Distributions 	return error;
423*33de042dSApple OSS Distributions }
424*33de042dSApple OSS Distributions 
425*33de042dSApple OSS Distributions /*
426*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
427*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
428*33de042dSApple OSS Distributions  * memory accesses.
429*33de042dSApple OSS Distributions  */
430*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_autoaddr(struct ifnet * ifp,struct ifreq * ifr)431*33de042dSApple OSS Distributions inctl_autoaddr(struct ifnet *ifp, struct ifreq *ifr)
432*33de042dSApple OSS Distributions {
433*33de042dSApple OSS Distributions 	int error = 0, intval;
434*33de042dSApple OSS Distributions 
435*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
436*33de042dSApple OSS Distributions 
437*33de042dSApple OSS Distributions 	bcopy(&ifr->ifr_intval, &intval, sizeof(intval));
438*33de042dSApple OSS Distributions 
439*33de042dSApple OSS Distributions 	ifnet_lock_exclusive(ifp);
440*33de042dSApple OSS Distributions 	if (intval) {
441*33de042dSApple OSS Distributions 		/*
442*33de042dSApple OSS Distributions 		 * An interface in IPv4 router mode implies that it
443*33de042dSApple OSS Distributions 		 * is configured with a static IP address and should
444*33de042dSApple OSS Distributions 		 * not act as a DHCP client; prevent SIOCAUTOADDR from
445*33de042dSApple OSS Distributions 		 * being set in that mode.
446*33de042dSApple OSS Distributions 		 */
447*33de042dSApple OSS Distributions 		if (ifp->if_eflags & IFEF_IPV4_ROUTER) {
448*33de042dSApple OSS Distributions 			intval = 0;     /* be safe; clear flag if set */
449*33de042dSApple OSS Distributions 			error = EBUSY;
450*33de042dSApple OSS Distributions 		} else {
451*33de042dSApple OSS Distributions 			if_set_eflags(ifp, IFEF_AUTOCONFIGURING);
452*33de042dSApple OSS Distributions 		}
453*33de042dSApple OSS Distributions 	}
454*33de042dSApple OSS Distributions 	if (!intval) {
455*33de042dSApple OSS Distributions 		if_clear_eflags(ifp, IFEF_AUTOCONFIGURING);
456*33de042dSApple OSS Distributions 	}
457*33de042dSApple OSS Distributions 	ifnet_lock_done(ifp);
458*33de042dSApple OSS Distributions 
459*33de042dSApple OSS Distributions 	return error;
460*33de042dSApple OSS Distributions }
461*33de042dSApple OSS Distributions 
462*33de042dSApple OSS Distributions /*
463*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
464*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
465*33de042dSApple OSS Distributions  * memory accesses.
466*33de042dSApple OSS Distributions  */
467*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_arpipll(struct ifnet * ifp,struct ifreq * ifr)468*33de042dSApple OSS Distributions inctl_arpipll(struct ifnet *ifp, struct ifreq *ifr)
469*33de042dSApple OSS Distributions {
470*33de042dSApple OSS Distributions 	int error = 0, intval;
471*33de042dSApple OSS Distributions 
472*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
473*33de042dSApple OSS Distributions 
474*33de042dSApple OSS Distributions 	bcopy(&ifr->ifr_intval, &intval, sizeof(intval));
475*33de042dSApple OSS Distributions 	ipv4_ll_arp_aware = 1;
476*33de042dSApple OSS Distributions 
477*33de042dSApple OSS Distributions 	ifnet_lock_exclusive(ifp);
478*33de042dSApple OSS Distributions 	if (intval) {
479*33de042dSApple OSS Distributions 		/*
480*33de042dSApple OSS Distributions 		 * An interface in IPv4 router mode implies that it
481*33de042dSApple OSS Distributions 		 * is configured with a static IP address and should
482*33de042dSApple OSS Distributions 		 * not have to deal with IPv4 Link-Local Address;
483*33de042dSApple OSS Distributions 		 * prevent SIOCARPIPLL from being set in that mode.
484*33de042dSApple OSS Distributions 		 */
485*33de042dSApple OSS Distributions 		if (ifp->if_eflags & IFEF_IPV4_ROUTER) {
486*33de042dSApple OSS Distributions 			intval = 0;     /* be safe; clear flag if set */
487*33de042dSApple OSS Distributions 			error = EBUSY;
488*33de042dSApple OSS Distributions 		} else {
489*33de042dSApple OSS Distributions 			if_set_eflags(ifp, IFEF_ARPLL);
490*33de042dSApple OSS Distributions 		}
491*33de042dSApple OSS Distributions 	}
492*33de042dSApple OSS Distributions 	if (!intval) {
493*33de042dSApple OSS Distributions 		if_clear_eflags(ifp, IFEF_ARPLL);
494*33de042dSApple OSS Distributions 	}
495*33de042dSApple OSS Distributions 	ifnet_lock_done(ifp);
496*33de042dSApple OSS Distributions 
497*33de042dSApple OSS Distributions 	return error;
498*33de042dSApple OSS Distributions }
499*33de042dSApple OSS Distributions 
500*33de042dSApple OSS Distributions /*
501*33de042dSApple OSS Distributions  * Handle SIOCSETROUTERMODE to set or clear the IPv4 router mode flag on
502*33de042dSApple OSS Distributions  * the interface.  When in this mode, IPv4 Link-Local Address support is
503*33de042dSApple OSS Distributions  * disabled in ARP, and DHCP client support is disabled in IP input; turning
504*33de042dSApple OSS Distributions  * any of them on would cause an error to be returned.  Entering or exiting
505*33de042dSApple OSS Distributions  * this mode will result in the removal of IPv4 addresses currently configured
506*33de042dSApple OSS Distributions  * on the interface.
507*33de042dSApple OSS Distributions  *
508*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
509*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
510*33de042dSApple OSS Distributions  * memory accesses.
511*33de042dSApple OSS Distributions  */
512*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_setrouter(struct ifnet * ifp,struct ifreq * ifr)513*33de042dSApple OSS Distributions inctl_setrouter(struct ifnet *ifp, struct ifreq *ifr)
514*33de042dSApple OSS Distributions {
515*33de042dSApple OSS Distributions 	int error = 0, intval;
516*33de042dSApple OSS Distributions 
517*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
518*33de042dSApple OSS Distributions 
519*33de042dSApple OSS Distributions 	/* Router mode isn't valid for loopback */
520*33de042dSApple OSS Distributions 	if (ifp->if_flags & IFF_LOOPBACK) {
521*33de042dSApple OSS Distributions 		return ENODEV;
522*33de042dSApple OSS Distributions 	}
523*33de042dSApple OSS Distributions 
524*33de042dSApple OSS Distributions 	bcopy(&ifr->ifr_intval, &intval, sizeof(intval));
525*33de042dSApple OSS Distributions 	switch (intval) {
526*33de042dSApple OSS Distributions 	case 0:
527*33de042dSApple OSS Distributions 	case 1:
528*33de042dSApple OSS Distributions 		break;
529*33de042dSApple OSS Distributions 	default:
530*33de042dSApple OSS Distributions 		return EINVAL;
531*33de042dSApple OSS Distributions 	}
532*33de042dSApple OSS Distributions 	ifnet_lock_exclusive(ifp);
533*33de042dSApple OSS Distributions 	if (intval != 0) {
534*33de042dSApple OSS Distributions 		if_set_eflags(ifp, IFEF_IPV4_ROUTER);
535*33de042dSApple OSS Distributions 		if_clear_eflags(ifp, (IFEF_ARPLL | IFEF_AUTOCONFIGURING));
536*33de042dSApple OSS Distributions 	} else {
537*33de042dSApple OSS Distributions 		if_clear_eflags(ifp, IFEF_IPV4_ROUTER);
538*33de042dSApple OSS Distributions 	}
539*33de042dSApple OSS Distributions 	ifnet_lock_done(ifp);
540*33de042dSApple OSS Distributions 
541*33de042dSApple OSS Distributions 	/* purge all IPv4 addresses configured on this interface */
542*33de042dSApple OSS Distributions 	in_purgeaddrs(ifp);
543*33de042dSApple OSS Distributions 
544*33de042dSApple OSS Distributions 	return error;
545*33de042dSApple OSS Distributions }
546*33de042dSApple OSS Distributions 
547*33de042dSApple OSS Distributions /*
548*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
549*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
550*33de042dSApple OSS Distributions  * memory accesses.
551*33de042dSApple OSS Distributions  */
552*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_ifaddr(struct ifnet * ifp,struct in_ifaddr * ia,u_long cmd,struct ifreq * ifr)553*33de042dSApple OSS Distributions inctl_ifaddr(struct ifnet *ifp, struct in_ifaddr *ia, u_long cmd,
554*33de042dSApple OSS Distributions     struct ifreq *ifr)
555*33de042dSApple OSS Distributions {
556*33de042dSApple OSS Distributions 	struct kev_in_data in_event_data;
557*33de042dSApple OSS Distributions 	struct kev_msg ev_msg;
558*33de042dSApple OSS Distributions 	struct sockaddr_in addr;
559*33de042dSApple OSS Distributions 	struct ifaddr *__single ifa;
560*33de042dSApple OSS Distributions 	int error = 0;
561*33de042dSApple OSS Distributions 
562*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
563*33de042dSApple OSS Distributions 
564*33de042dSApple OSS Distributions 	bzero(&in_event_data, sizeof(struct kev_in_data));
565*33de042dSApple OSS Distributions 	bzero(&ev_msg, sizeof(struct kev_msg));
566*33de042dSApple OSS Distributions 
567*33de042dSApple OSS Distributions 	switch (cmd) {
568*33de042dSApple OSS Distributions 	case SIOCGIFADDR:               /* struct ifreq */
569*33de042dSApple OSS Distributions 		if (ia == NULL) {
570*33de042dSApple OSS Distributions 			error = EADDRNOTAVAIL;
571*33de042dSApple OSS Distributions 			break;
572*33de042dSApple OSS Distributions 		}
573*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
574*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ia->ia_addr, &ifr->ifr_addr, sizeof(addr));
575*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
576*33de042dSApple OSS Distributions 		break;
577*33de042dSApple OSS Distributions 
578*33de042dSApple OSS Distributions 	case SIOCSIFADDR:               /* struct ifreq */
579*33de042dSApple OSS Distributions 		VERIFY(ia != NULL);
580*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ifr->ifr_addr, &addr, sizeof(addr));
581*33de042dSApple OSS Distributions 		/*
582*33de042dSApple OSS Distributions 		 * If this is a new address, the reference count for the
583*33de042dSApple OSS Distributions 		 * hash table has been taken at creation time above.
584*33de042dSApple OSS Distributions 		 */
585*33de042dSApple OSS Distributions 		error = in_ifinit(ifp, ia, &addr, 1);
586*33de042dSApple OSS Distributions 		if (error == 0) {
587*33de042dSApple OSS Distributions 			(void) ifnet_notify_address(ifp, AF_INET);
588*33de042dSApple OSS Distributions 		}
589*33de042dSApple OSS Distributions 		break;
590*33de042dSApple OSS Distributions 
591*33de042dSApple OSS Distributions 	case SIOCAIFADDR: {             /* struct {if,in_}aliasreq */
592*33de042dSApple OSS Distributions 		struct in_aliasreq *__single ifra = (struct in_aliasreq *)ifr;
593*33de042dSApple OSS Distributions 		struct sockaddr_in broadaddr, mask;
594*33de042dSApple OSS Distributions 		int hostIsNew, maskIsNew;
595*33de042dSApple OSS Distributions 
596*33de042dSApple OSS Distributions 		VERIFY(ia != NULL);
597*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ifra->ifra_addr, &addr, sizeof(addr));
598*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ifra->ifra_broadaddr, &broadaddr, sizeof(broadaddr));
599*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ifra->ifra_mask, &mask, sizeof(mask));
600*33de042dSApple OSS Distributions 
601*33de042dSApple OSS Distributions 		maskIsNew = 0;
602*33de042dSApple OSS Distributions 		hostIsNew = 1;
603*33de042dSApple OSS Distributions 		error = 0;
604*33de042dSApple OSS Distributions 
605*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
606*33de042dSApple OSS Distributions 		if (ia->ia_addr.sin_family == AF_INET) {
607*33de042dSApple OSS Distributions 			if (addr.sin_len == 0) {
608*33de042dSApple OSS Distributions 				addr = ia->ia_addr;
609*33de042dSApple OSS Distributions 				hostIsNew = 0;
610*33de042dSApple OSS Distributions 			} else if (addr.sin_addr.s_addr ==
611*33de042dSApple OSS Distributions 			    ia->ia_addr.sin_addr.s_addr) {
612*33de042dSApple OSS Distributions 				hostIsNew = 0;
613*33de042dSApple OSS Distributions 			}
614*33de042dSApple OSS Distributions 		}
615*33de042dSApple OSS Distributions 		if (mask.sin_len != 0) {
616*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
617*33de042dSApple OSS Distributions 			in_ifscrub(ifp, ia, 0);
618*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
619*33de042dSApple OSS Distributions 			ia->ia_sockmask.sin_len = sizeof(struct sockaddr_in);
620*33de042dSApple OSS Distributions 			ia->ia_sockmask.sin_family = AF_INET;
621*33de042dSApple OSS Distributions 			ia->ia_sockmask.sin_port = 0;
622*33de042dSApple OSS Distributions 			ia->ia_sockmask.sin_addr = mask.sin_addr;
623*33de042dSApple OSS Distributions 			bzero(&ia->ia_sockmask.sin_zero, sizeof(ia->ia_dstaddr.sin_zero));
624*33de042dSApple OSS Distributions 			ia->ia_subnetmask =
625*33de042dSApple OSS Distributions 			    ntohl(ia->ia_sockmask.sin_addr.s_addr);
626*33de042dSApple OSS Distributions 			maskIsNew = 1;
627*33de042dSApple OSS Distributions 		}
628*33de042dSApple OSS Distributions 		if ((ifp->if_flags & IFF_POINTOPOINT) &&
629*33de042dSApple OSS Distributions 		    (broadaddr.sin_family == AF_INET)) {
630*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
631*33de042dSApple OSS Distributions 			in_ifscrub(ifp, ia, 0);
632*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
633*33de042dSApple OSS Distributions 			ia->ia_dstaddr.sin_family = AF_INET;
634*33de042dSApple OSS Distributions 			ia->ia_dstaddr.sin_len = sizeof(struct sockaddr_in);
635*33de042dSApple OSS Distributions 			ia->ia_dstaddr.sin_port = 0;
636*33de042dSApple OSS Distributions 			ia->ia_dstaddr.sin_addr = broadaddr.sin_addr;
637*33de042dSApple OSS Distributions 			bzero(&ia->ia_dstaddr.sin_zero, sizeof(ia->ia_dstaddr.sin_zero));
638*33de042dSApple OSS Distributions 			maskIsNew  = 1; /* We lie; but the effect's the same */
639*33de042dSApple OSS Distributions 		}
640*33de042dSApple OSS Distributions 		if (addr.sin_family == AF_INET && (hostIsNew || maskIsNew)) {
641*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
642*33de042dSApple OSS Distributions 			error = in_ifinit(ifp, ia, &addr, 0);
643*33de042dSApple OSS Distributions 		} else {
644*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
645*33de042dSApple OSS Distributions 		}
646*33de042dSApple OSS Distributions 		if (error == 0) {
647*33de042dSApple OSS Distributions 			(void) ifnet_notify_address(ifp, AF_INET);
648*33de042dSApple OSS Distributions 		}
649*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
650*33de042dSApple OSS Distributions 		if ((ifp->if_flags & IFF_BROADCAST) &&
651*33de042dSApple OSS Distributions 		    (broadaddr.sin_family == AF_INET)) {
652*33de042dSApple OSS Distributions 			ia->ia_broadaddr.sin_family = AF_INET;
653*33de042dSApple OSS Distributions 			ia->ia_broadaddr.sin_len = sizeof(struct sockaddr_in);
654*33de042dSApple OSS Distributions 			ia->ia_broadaddr.sin_port = 0;
655*33de042dSApple OSS Distributions 			ia->ia_broadaddr.sin_addr = broadaddr.sin_addr;
656*33de042dSApple OSS Distributions 			bzero(&ia->ia_broadaddr.sin_zero, sizeof(ia->ia_broadaddr.sin_zero));
657*33de042dSApple OSS Distributions 		}
658*33de042dSApple OSS Distributions 
659*33de042dSApple OSS Distributions 		/*
660*33de042dSApple OSS Distributions 		 * Report event.
661*33de042dSApple OSS Distributions 		 */
662*33de042dSApple OSS Distributions 		if ((error == 0) || (error == EEXIST)) {
663*33de042dSApple OSS Distributions 			ev_msg.vendor_code      = KEV_VENDOR_APPLE;
664*33de042dSApple OSS Distributions 			ev_msg.kev_class        = KEV_NETWORK_CLASS;
665*33de042dSApple OSS Distributions 			ev_msg.kev_subclass     = KEV_INET_SUBCLASS;
666*33de042dSApple OSS Distributions 
667*33de042dSApple OSS Distributions 			if (hostIsNew) {
668*33de042dSApple OSS Distributions 				ev_msg.event_code = KEV_INET_NEW_ADDR;
669*33de042dSApple OSS Distributions 			} else {
670*33de042dSApple OSS Distributions 				ev_msg.event_code = KEV_INET_CHANGED_ADDR;
671*33de042dSApple OSS Distributions 			}
672*33de042dSApple OSS Distributions 
673*33de042dSApple OSS Distributions 			if (ia->ia_ifa.ifa_dstaddr) {
674*33de042dSApple OSS Distributions 				in_event_data.ia_dstaddr = SIN(ia->ia_ifa.ifa_dstaddr)->sin_addr;
675*33de042dSApple OSS Distributions 			} else {
676*33de042dSApple OSS Distributions 				in_event_data.ia_dstaddr.s_addr = INADDR_ANY;
677*33de042dSApple OSS Distributions 			}
678*33de042dSApple OSS Distributions 			in_event_data.ia_addr           = ia->ia_addr.sin_addr;
679*33de042dSApple OSS Distributions 			in_event_data.ia_net            = ia->ia_net;
680*33de042dSApple OSS Distributions 			in_event_data.ia_netmask        = ia->ia_netmask;
681*33de042dSApple OSS Distributions 			in_event_data.ia_subnet         = ia->ia_subnet;
682*33de042dSApple OSS Distributions 			in_event_data.ia_subnetmask     = ia->ia_subnetmask;
683*33de042dSApple OSS Distributions 			in_event_data.ia_netbroadcast   = ia->ia_netbroadcast;
684*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
685*33de042dSApple OSS Distributions 			(void) strlcpy(&in_event_data.link_data.if_name[0],
686*33de042dSApple OSS Distributions 			    ifp->if_name, IFNAMSIZ);
687*33de042dSApple OSS Distributions 			in_event_data.link_data.if_family = ifp->if_family;
688*33de042dSApple OSS Distributions 			in_event_data.link_data.if_unit = ifp->if_unit;
689*33de042dSApple OSS Distributions 
690*33de042dSApple OSS Distributions 			ev_msg.dv[0].data_ptr    = &in_event_data;
691*33de042dSApple OSS Distributions 			ev_msg.dv[0].data_length = sizeof(struct kev_in_data);
692*33de042dSApple OSS Distributions 			ev_msg.dv[1].data_length = 0;
693*33de042dSApple OSS Distributions 
694*33de042dSApple OSS Distributions 			dlil_post_complete_msg(ifp, &ev_msg);
695*33de042dSApple OSS Distributions 		} else {
696*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
697*33de042dSApple OSS Distributions 		}
698*33de042dSApple OSS Distributions 		break;
699*33de042dSApple OSS Distributions 	}
700*33de042dSApple OSS Distributions 
701*33de042dSApple OSS Distributions 	case SIOCDIFADDR:               /* struct ifreq */
702*33de042dSApple OSS Distributions 		VERIFY(ia != NULL);
703*33de042dSApple OSS Distributions 		error = ifnet_ioctl(ifp, PF_INET, SIOCDIFADDR, ia);
704*33de042dSApple OSS Distributions 		if (error == EOPNOTSUPP) {
705*33de042dSApple OSS Distributions 			error = 0;
706*33de042dSApple OSS Distributions 		}
707*33de042dSApple OSS Distributions 		if (error != 0) {
708*33de042dSApple OSS Distributions 			break;
709*33de042dSApple OSS Distributions 		}
710*33de042dSApple OSS Distributions 
711*33de042dSApple OSS Distributions 		/* Fill out the kernel event information */
712*33de042dSApple OSS Distributions 		ev_msg.vendor_code      = KEV_VENDOR_APPLE;
713*33de042dSApple OSS Distributions 		ev_msg.kev_class        = KEV_NETWORK_CLASS;
714*33de042dSApple OSS Distributions 		ev_msg.kev_subclass     = KEV_INET_SUBCLASS;
715*33de042dSApple OSS Distributions 
716*33de042dSApple OSS Distributions 		ev_msg.event_code       = KEV_INET_ADDR_DELETED;
717*33de042dSApple OSS Distributions 
718*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
719*33de042dSApple OSS Distributions 		if (ia->ia_ifa.ifa_dstaddr) {
720*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr = SIN(ia->ia_ifa.ifa_dstaddr)->sin_addr;
721*33de042dSApple OSS Distributions 		} else {
722*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr.s_addr = INADDR_ANY;
723*33de042dSApple OSS Distributions 		}
724*33de042dSApple OSS Distributions 		in_event_data.ia_addr           = ia->ia_addr.sin_addr;
725*33de042dSApple OSS Distributions 		in_event_data.ia_net            = ia->ia_net;
726*33de042dSApple OSS Distributions 		in_event_data.ia_netmask        = ia->ia_netmask;
727*33de042dSApple OSS Distributions 		in_event_data.ia_subnet         = ia->ia_subnet;
728*33de042dSApple OSS Distributions 		in_event_data.ia_subnetmask     = ia->ia_subnetmask;
729*33de042dSApple OSS Distributions 		in_event_data.ia_netbroadcast   = ia->ia_netbroadcast;
730*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
731*33de042dSApple OSS Distributions 		(void) strlcpy(&in_event_data.link_data.if_name[0],
732*33de042dSApple OSS Distributions 		    ifp->if_name, IFNAMSIZ);
733*33de042dSApple OSS Distributions 		in_event_data.link_data.if_family = ifp->if_family;
734*33de042dSApple OSS Distributions 		in_event_data.link_data.if_unit  = (u_int32_t)ifp->if_unit;
735*33de042dSApple OSS Distributions 
736*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_ptr    = &in_event_data;
737*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_length = sizeof(struct kev_in_data);
738*33de042dSApple OSS Distributions 		ev_msg.dv[1].data_length = 0;
739*33de042dSApple OSS Distributions 
740*33de042dSApple OSS Distributions 		ifa = &ia->ia_ifa;
741*33de042dSApple OSS Distributions 		lck_rw_lock_exclusive(&in_ifaddr_rwlock);
742*33de042dSApple OSS Distributions 		/* Release ia_link reference */
743*33de042dSApple OSS Distributions 		ifa_remref(ifa);
744*33de042dSApple OSS Distributions 		TAILQ_REMOVE(&in_ifaddrhead, ia, ia_link);
745*33de042dSApple OSS Distributions 		IFA_LOCK(ifa);
746*33de042dSApple OSS Distributions 		if (IA_IS_HASHED(ia)) {
747*33de042dSApple OSS Distributions 			in_iahash_remove(ia);
748*33de042dSApple OSS Distributions 		}
749*33de042dSApple OSS Distributions 		IFA_UNLOCK(ifa);
750*33de042dSApple OSS Distributions 		lck_rw_done(&in_ifaddr_rwlock);
751*33de042dSApple OSS Distributions 
752*33de042dSApple OSS Distributions 		/*
753*33de042dSApple OSS Distributions 		 * in_ifscrub kills the interface route.
754*33de042dSApple OSS Distributions 		 */
755*33de042dSApple OSS Distributions 		in_ifscrub(ifp, ia, 0);
756*33de042dSApple OSS Distributions 		ifnet_lock_exclusive(ifp);
757*33de042dSApple OSS Distributions 		IFA_LOCK(ifa);
758*33de042dSApple OSS Distributions 		/* if_detach_ifa() releases ifa_link reference */
759*33de042dSApple OSS Distributions 		if_detach_ifa(ifp, ifa);
760*33de042dSApple OSS Distributions 		/* Our reference to this address is dropped at the bottom */
761*33de042dSApple OSS Distributions 		IFA_UNLOCK(ifa);
762*33de042dSApple OSS Distributions 
763*33de042dSApple OSS Distributions 		/* invalidate route caches */
764*33de042dSApple OSS Distributions 		routegenid_inet_update();
765*33de042dSApple OSS Distributions 
766*33de042dSApple OSS Distributions 		/*
767*33de042dSApple OSS Distributions 		 * If the interface supports multicast, and no address is left,
768*33de042dSApple OSS Distributions 		 * remove the "all hosts" multicast group from that interface.
769*33de042dSApple OSS Distributions 		 */
770*33de042dSApple OSS Distributions 		if ((ifp->if_flags & IFF_MULTICAST) ||
771*33de042dSApple OSS Distributions 		    ifp->if_allhostsinm != NULL) {
772*33de042dSApple OSS Distributions 			TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
773*33de042dSApple OSS Distributions 				IFA_LOCK(ifa);
774*33de042dSApple OSS Distributions 				if (ifa->ifa_addr->sa_family == AF_INET) {
775*33de042dSApple OSS Distributions 					IFA_UNLOCK(ifa);
776*33de042dSApple OSS Distributions 					break;
777*33de042dSApple OSS Distributions 				}
778*33de042dSApple OSS Distributions 				IFA_UNLOCK(ifa);
779*33de042dSApple OSS Distributions 			}
780*33de042dSApple OSS Distributions 			ifnet_lock_done(ifp);
781*33de042dSApple OSS Distributions 
782*33de042dSApple OSS Distributions 			lck_mtx_lock(&ifp->if_addrconfig_lock);
783*33de042dSApple OSS Distributions 			if (ifa == NULL && ifp->if_allhostsinm != NULL) {
784*33de042dSApple OSS Distributions 				struct in_multi *__single inm = ifp->if_allhostsinm;
785*33de042dSApple OSS Distributions 				ifp->if_allhostsinm = NULL;
786*33de042dSApple OSS Distributions 
787*33de042dSApple OSS Distributions 				in_delmulti(inm);
788*33de042dSApple OSS Distributions 				/* release the reference for allhostsinm */
789*33de042dSApple OSS Distributions 				INM_REMREF(inm);
790*33de042dSApple OSS Distributions 			}
791*33de042dSApple OSS Distributions 			lck_mtx_unlock(&ifp->if_addrconfig_lock);
792*33de042dSApple OSS Distributions 		} else {
793*33de042dSApple OSS Distributions 			ifnet_lock_done(ifp);
794*33de042dSApple OSS Distributions 		}
795*33de042dSApple OSS Distributions 
796*33de042dSApple OSS Distributions 		/* Post the kernel event */
797*33de042dSApple OSS Distributions 		dlil_post_complete_msg(ifp, &ev_msg);
798*33de042dSApple OSS Distributions 
799*33de042dSApple OSS Distributions 		/*
800*33de042dSApple OSS Distributions 		 * See if there is any IPV4 address left and if so,
801*33de042dSApple OSS Distributions 		 * reconfigure KDP to use current primary address.
802*33de042dSApple OSS Distributions 		 */
803*33de042dSApple OSS Distributions 		ifa = ifa_ifpgetprimary(ifp, AF_INET);
804*33de042dSApple OSS Distributions 		if (ifa != NULL) {
805*33de042dSApple OSS Distributions 			/*
806*33de042dSApple OSS Distributions 			 * NOTE: SIOCSIFADDR is defined with struct ifreq
807*33de042dSApple OSS Distributions 			 * as parameter, but here we are sending it down
808*33de042dSApple OSS Distributions 			 * to the interface with a pointer to struct ifaddr,
809*33de042dSApple OSS Distributions 			 * for legacy reasons.
810*33de042dSApple OSS Distributions 			 */
811*33de042dSApple OSS Distributions 			error = ifnet_ioctl(ifp, PF_INET, SIOCSIFADDR, ifa);
812*33de042dSApple OSS Distributions 			if (error == EOPNOTSUPP) {
813*33de042dSApple OSS Distributions 				error = 0;
814*33de042dSApple OSS Distributions 			}
815*33de042dSApple OSS Distributions 
816*33de042dSApple OSS Distributions 			/* Release reference from ifa_ifpgetprimary() */
817*33de042dSApple OSS Distributions 			ifa_remref(ifa);
818*33de042dSApple OSS Distributions 		}
819*33de042dSApple OSS Distributions 		(void) ifnet_notify_address(ifp, AF_INET);
820*33de042dSApple OSS Distributions 		break;
821*33de042dSApple OSS Distributions 
822*33de042dSApple OSS Distributions 	default:
823*33de042dSApple OSS Distributions 		VERIFY(0);
824*33de042dSApple OSS Distributions 		/* NOTREACHED */
825*33de042dSApple OSS Distributions 	}
826*33de042dSApple OSS Distributions 
827*33de042dSApple OSS Distributions 	return error;
828*33de042dSApple OSS Distributions }
829*33de042dSApple OSS Distributions 
830*33de042dSApple OSS Distributions /*
831*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
832*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
833*33de042dSApple OSS Distributions  * memory accesses.
834*33de042dSApple OSS Distributions  */
835*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_ifdstaddr(struct ifnet * ifp,struct in_ifaddr * ia,u_long cmd,struct ifreq * ifr)836*33de042dSApple OSS Distributions inctl_ifdstaddr(struct ifnet *ifp, struct in_ifaddr *ia, u_long cmd,
837*33de042dSApple OSS Distributions     struct ifreq *ifr)
838*33de042dSApple OSS Distributions {
839*33de042dSApple OSS Distributions 	struct kev_in_data in_event_data;
840*33de042dSApple OSS Distributions 	struct kev_msg ev_msg;
841*33de042dSApple OSS Distributions 	struct sockaddr_in dstaddr;
842*33de042dSApple OSS Distributions 	int error = 0;
843*33de042dSApple OSS Distributions 
844*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
845*33de042dSApple OSS Distributions 
846*33de042dSApple OSS Distributions 	if (!(ifp->if_flags & IFF_POINTOPOINT)) {
847*33de042dSApple OSS Distributions 		return EINVAL;
848*33de042dSApple OSS Distributions 	}
849*33de042dSApple OSS Distributions 
850*33de042dSApple OSS Distributions 	bzero(&in_event_data, sizeof(struct kev_in_data));
851*33de042dSApple OSS Distributions 	bzero(&ev_msg, sizeof(struct kev_msg));
852*33de042dSApple OSS Distributions 
853*33de042dSApple OSS Distributions 	switch (cmd) {
854*33de042dSApple OSS Distributions 	case SIOCGIFDSTADDR:            /* struct ifreq */
855*33de042dSApple OSS Distributions 		if (ia == NULL) {
856*33de042dSApple OSS Distributions 			error = EADDRNOTAVAIL;
857*33de042dSApple OSS Distributions 			break;
858*33de042dSApple OSS Distributions 		}
859*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
860*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ia->ia_dstaddr, &ifr->ifr_dstaddr, sizeof(dstaddr));
861*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
862*33de042dSApple OSS Distributions 		break;
863*33de042dSApple OSS Distributions 
864*33de042dSApple OSS Distributions 	case SIOCSIFDSTADDR:            /* struct ifreq */
865*33de042dSApple OSS Distributions 		VERIFY(ia != NULL);
866*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
867*33de042dSApple OSS Distributions 		dstaddr = ia->ia_dstaddr;
868*33de042dSApple OSS Distributions 
869*33de042dSApple OSS Distributions 		ia->ia_dstaddr.sin_family = AF_INET;
870*33de042dSApple OSS Distributions 		ia->ia_dstaddr.sin_len = sizeof(struct sockaddr_in);
871*33de042dSApple OSS Distributions 		ia->ia_dstaddr.sin_port = 0;
872*33de042dSApple OSS Distributions 		bcopy(&SIN(&ifr->ifr_dstaddr)->sin_addr,
873*33de042dSApple OSS Distributions 		    &ia->ia_dstaddr.sin_addr, sizeof(ia->ia_dstaddr.sin_addr));
874*33de042dSApple OSS Distributions 		bzero(&ia->ia_dstaddr.sin_zero, sizeof(ia->ia_dstaddr.sin_zero));
875*33de042dSApple OSS Distributions 
876*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
877*33de042dSApple OSS Distributions 		/*
878*33de042dSApple OSS Distributions 		 * NOTE: SIOCSIFDSTADDR is defined with struct ifreq
879*33de042dSApple OSS Distributions 		 * as parameter, but here we are sending it down
880*33de042dSApple OSS Distributions 		 * to the interface with a pointer to struct ifaddr,
881*33de042dSApple OSS Distributions 		 * for legacy reasons.
882*33de042dSApple OSS Distributions 		 */
883*33de042dSApple OSS Distributions 		error = ifnet_ioctl(ifp, PF_INET, SIOCSIFDSTADDR, ia);
884*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
885*33de042dSApple OSS Distributions 		if (error == EOPNOTSUPP) {
886*33de042dSApple OSS Distributions 			error = 0;
887*33de042dSApple OSS Distributions 		}
888*33de042dSApple OSS Distributions 		if (error != 0) {
889*33de042dSApple OSS Distributions 			ia->ia_dstaddr = dstaddr;
890*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
891*33de042dSApple OSS Distributions 			break;
892*33de042dSApple OSS Distributions 		}
893*33de042dSApple OSS Distributions 		IFA_LOCK_ASSERT_HELD(&ia->ia_ifa);
894*33de042dSApple OSS Distributions 
895*33de042dSApple OSS Distributions 		ev_msg.vendor_code      = KEV_VENDOR_APPLE;
896*33de042dSApple OSS Distributions 		ev_msg.kev_class        = KEV_NETWORK_CLASS;
897*33de042dSApple OSS Distributions 		ev_msg.kev_subclass     = KEV_INET_SUBCLASS;
898*33de042dSApple OSS Distributions 
899*33de042dSApple OSS Distributions 		ev_msg.event_code       = KEV_INET_SIFDSTADDR;
900*33de042dSApple OSS Distributions 
901*33de042dSApple OSS Distributions 		if (ia->ia_ifa.ifa_dstaddr) {
902*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr = SIN(ia->ia_ifa.ifa_dstaddr)->sin_addr;
903*33de042dSApple OSS Distributions 		} else {
904*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr.s_addr = INADDR_ANY;
905*33de042dSApple OSS Distributions 		}
906*33de042dSApple OSS Distributions 
907*33de042dSApple OSS Distributions 		in_event_data.ia_addr           = ia->ia_addr.sin_addr;
908*33de042dSApple OSS Distributions 		in_event_data.ia_net            = ia->ia_net;
909*33de042dSApple OSS Distributions 		in_event_data.ia_netmask        = ia->ia_netmask;
910*33de042dSApple OSS Distributions 		in_event_data.ia_subnet         = ia->ia_subnet;
911*33de042dSApple OSS Distributions 		in_event_data.ia_subnetmask     = ia->ia_subnetmask;
912*33de042dSApple OSS Distributions 		in_event_data.ia_netbroadcast   = ia->ia_netbroadcast;
913*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
914*33de042dSApple OSS Distributions 		(void) strlcpy(&in_event_data.link_data.if_name[0],
915*33de042dSApple OSS Distributions 		    ifp->if_name, IFNAMSIZ);
916*33de042dSApple OSS Distributions 		in_event_data.link_data.if_family = ifp->if_family;
917*33de042dSApple OSS Distributions 		in_event_data.link_data.if_unit  = (u_int32_t)ifp->if_unit;
918*33de042dSApple OSS Distributions 
919*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_ptr    = &in_event_data;
920*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_length = sizeof(struct kev_in_data);
921*33de042dSApple OSS Distributions 		ev_msg.dv[1].data_length = 0;
922*33de042dSApple OSS Distributions 
923*33de042dSApple OSS Distributions 		dlil_post_complete_msg(ifp, &ev_msg);
924*33de042dSApple OSS Distributions 
925*33de042dSApple OSS Distributions 		lck_mtx_lock(rnh_lock);
926*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
927*33de042dSApple OSS Distributions 		if (ia->ia_flags & IFA_ROUTE) {
928*33de042dSApple OSS Distributions 			ia->ia_ifa.ifa_dstaddr = SA(&dstaddr);
929*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
930*33de042dSApple OSS Distributions 			rtinit_locked(&(ia->ia_ifa), RTM_DELETE, RTF_HOST);
931*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
932*33de042dSApple OSS Distributions 			ia->ia_ifa.ifa_dstaddr =
933*33de042dSApple OSS Distributions 			    SA(&ia->ia_dstaddr);
934*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
935*33de042dSApple OSS Distributions 			rtinit_locked(&(ia->ia_ifa), RTM_ADD,
936*33de042dSApple OSS Distributions 			    RTF_HOST | RTF_UP);
937*33de042dSApple OSS Distributions 		} else {
938*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
939*33de042dSApple OSS Distributions 		}
940*33de042dSApple OSS Distributions 		lck_mtx_unlock(rnh_lock);
941*33de042dSApple OSS Distributions 		break;
942*33de042dSApple OSS Distributions 
943*33de042dSApple OSS Distributions 
944*33de042dSApple OSS Distributions 
945*33de042dSApple OSS Distributions 	default:
946*33de042dSApple OSS Distributions 		VERIFY(0);
947*33de042dSApple OSS Distributions 		/* NOTREACHED */
948*33de042dSApple OSS Distributions 	}
949*33de042dSApple OSS Distributions 
950*33de042dSApple OSS Distributions 	return error;
951*33de042dSApple OSS Distributions }
952*33de042dSApple OSS Distributions 
953*33de042dSApple OSS Distributions /*
954*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
955*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
956*33de042dSApple OSS Distributions  * memory accesses.
957*33de042dSApple OSS Distributions  */
958*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_ifbrdaddr(struct ifnet * ifp,struct in_ifaddr * ia,u_long cmd,struct ifreq * ifr)959*33de042dSApple OSS Distributions inctl_ifbrdaddr(struct ifnet *ifp, struct in_ifaddr *ia, u_long cmd,
960*33de042dSApple OSS Distributions     struct ifreq *ifr)
961*33de042dSApple OSS Distributions {
962*33de042dSApple OSS Distributions 	struct kev_in_data in_event_data;
963*33de042dSApple OSS Distributions 	struct kev_msg ev_msg;
964*33de042dSApple OSS Distributions 	int error = 0;
965*33de042dSApple OSS Distributions 
966*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
967*33de042dSApple OSS Distributions 
968*33de042dSApple OSS Distributions 	if (ia == NULL) {
969*33de042dSApple OSS Distributions 		return EADDRNOTAVAIL;
970*33de042dSApple OSS Distributions 	}
971*33de042dSApple OSS Distributions 
972*33de042dSApple OSS Distributions 	if (!(ifp->if_flags & IFF_BROADCAST)) {
973*33de042dSApple OSS Distributions 		return EINVAL;
974*33de042dSApple OSS Distributions 	}
975*33de042dSApple OSS Distributions 
976*33de042dSApple OSS Distributions 	bzero(&in_event_data, sizeof(struct kev_in_data));
977*33de042dSApple OSS Distributions 	bzero(&ev_msg, sizeof(struct kev_msg));
978*33de042dSApple OSS Distributions 
979*33de042dSApple OSS Distributions 	switch (cmd) {
980*33de042dSApple OSS Distributions 	case SIOCGIFBRDADDR:            /* struct ifreq */
981*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
982*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ia->ia_broadaddr, &ifr->ifr_broadaddr,
983*33de042dSApple OSS Distributions 		    sizeof(struct sockaddr_in));
984*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
985*33de042dSApple OSS Distributions 		break;
986*33de042dSApple OSS Distributions 
987*33de042dSApple OSS Distributions 	case SIOCSIFBRDADDR:            /* struct ifreq */
988*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
989*33de042dSApple OSS Distributions 
990*33de042dSApple OSS Distributions 		ia->ia_broadaddr.sin_family = AF_INET;
991*33de042dSApple OSS Distributions 		ia->ia_broadaddr.sin_len = sizeof(struct sockaddr_in);
992*33de042dSApple OSS Distributions 		ia->ia_broadaddr.sin_port = 0;
993*33de042dSApple OSS Distributions 		bcopy(&SIN(&ifr->ifr_broadaddr)->sin_addr,
994*33de042dSApple OSS Distributions 		    &ia->ia_broadaddr.sin_addr, sizeof(ia->ia_broadaddr.sin_addr));
995*33de042dSApple OSS Distributions 		bzero(&ia->ia_broadaddr.sin_zero, sizeof(ia->ia_broadaddr.sin_zero));
996*33de042dSApple OSS Distributions 
997*33de042dSApple OSS Distributions 		ev_msg.vendor_code      = KEV_VENDOR_APPLE;
998*33de042dSApple OSS Distributions 		ev_msg.kev_class        = KEV_NETWORK_CLASS;
999*33de042dSApple OSS Distributions 		ev_msg.kev_subclass     = KEV_INET_SUBCLASS;
1000*33de042dSApple OSS Distributions 
1001*33de042dSApple OSS Distributions 		ev_msg.event_code = KEV_INET_SIFBRDADDR;
1002*33de042dSApple OSS Distributions 
1003*33de042dSApple OSS Distributions 		if (ia->ia_ifa.ifa_dstaddr) {
1004*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr = SIN(ia->ia_ifa.ifa_dstaddr)->sin_addr;
1005*33de042dSApple OSS Distributions 		} else {
1006*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr.s_addr = INADDR_ANY;
1007*33de042dSApple OSS Distributions 		}
1008*33de042dSApple OSS Distributions 		in_event_data.ia_addr           = ia->ia_addr.sin_addr;
1009*33de042dSApple OSS Distributions 		in_event_data.ia_net            = ia->ia_net;
1010*33de042dSApple OSS Distributions 		in_event_data.ia_netmask        = ia->ia_netmask;
1011*33de042dSApple OSS Distributions 		in_event_data.ia_subnet         = ia->ia_subnet;
1012*33de042dSApple OSS Distributions 		in_event_data.ia_subnetmask     = ia->ia_subnetmask;
1013*33de042dSApple OSS Distributions 		in_event_data.ia_netbroadcast   = ia->ia_netbroadcast;
1014*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1015*33de042dSApple OSS Distributions 		(void) strlcpy(&in_event_data.link_data.if_name[0],
1016*33de042dSApple OSS Distributions 		    ifp->if_name, IFNAMSIZ);
1017*33de042dSApple OSS Distributions 		in_event_data.link_data.if_family = ifp->if_family;
1018*33de042dSApple OSS Distributions 		in_event_data.link_data.if_unit  = (u_int32_t)ifp->if_unit;
1019*33de042dSApple OSS Distributions 
1020*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_ptr    = &in_event_data;
1021*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_length = sizeof(struct kev_in_data);
1022*33de042dSApple OSS Distributions 		ev_msg.dv[1].data_length = 0;
1023*33de042dSApple OSS Distributions 
1024*33de042dSApple OSS Distributions 		dlil_post_complete_msg(ifp, &ev_msg);
1025*33de042dSApple OSS Distributions 		break;
1026*33de042dSApple OSS Distributions 
1027*33de042dSApple OSS Distributions 	default:
1028*33de042dSApple OSS Distributions 		VERIFY(0);
1029*33de042dSApple OSS Distributions 		/* NOTREACHED */
1030*33de042dSApple OSS Distributions 	}
1031*33de042dSApple OSS Distributions 
1032*33de042dSApple OSS Distributions 	return error;
1033*33de042dSApple OSS Distributions }
1034*33de042dSApple OSS Distributions 
1035*33de042dSApple OSS Distributions /*
1036*33de042dSApple OSS Distributions  * Caller passes in the ioctl data pointer directly via "ifr", with the
1037*33de042dSApple OSS Distributions  * expectation that this routine always uses bcopy() or other byte-aligned
1038*33de042dSApple OSS Distributions  * memory accesses.
1039*33de042dSApple OSS Distributions  */
1040*33de042dSApple OSS Distributions static __attribute__((noinline)) int
inctl_ifnetmask(struct ifnet * ifp,struct in_ifaddr * ia,u_long cmd,struct ifreq * ifr)1041*33de042dSApple OSS Distributions inctl_ifnetmask(struct ifnet *ifp, struct in_ifaddr *ia, u_long cmd,
1042*33de042dSApple OSS Distributions     struct ifreq *ifr)
1043*33de042dSApple OSS Distributions {
1044*33de042dSApple OSS Distributions 	struct kev_in_data in_event_data;
1045*33de042dSApple OSS Distributions 	struct kev_msg ev_msg;
1046*33de042dSApple OSS Distributions 	struct sockaddr_in mask;
1047*33de042dSApple OSS Distributions 	int error = 0;
1048*33de042dSApple OSS Distributions 
1049*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
1050*33de042dSApple OSS Distributions 
1051*33de042dSApple OSS Distributions 	bzero(&in_event_data, sizeof(struct kev_in_data));
1052*33de042dSApple OSS Distributions 	bzero(&ev_msg, sizeof(struct kev_msg));
1053*33de042dSApple OSS Distributions 
1054*33de042dSApple OSS Distributions 	switch (cmd) {
1055*33de042dSApple OSS Distributions 	case SIOCGIFNETMASK:            /* struct ifreq */
1056*33de042dSApple OSS Distributions 		if (ia == NULL) {
1057*33de042dSApple OSS Distributions 			error = EADDRNOTAVAIL;
1058*33de042dSApple OSS Distributions 			break;
1059*33de042dSApple OSS Distributions 		}
1060*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
1061*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ia->ia_sockmask, &ifr->ifr_addr, sizeof(mask));
1062*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1063*33de042dSApple OSS Distributions 		break;
1064*33de042dSApple OSS Distributions 
1065*33de042dSApple OSS Distributions 	case SIOCSIFNETMASK: {          /* struct ifreq */
1066*33de042dSApple OSS Distributions 		in_addr_t i;
1067*33de042dSApple OSS Distributions 
1068*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ifr->ifr_addr, &mask, sizeof(mask));
1069*33de042dSApple OSS Distributions 		i = mask.sin_addr.s_addr;
1070*33de042dSApple OSS Distributions 
1071*33de042dSApple OSS Distributions 		VERIFY(ia != NULL);
1072*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
1073*33de042dSApple OSS Distributions 		ia->ia_subnetmask = ntohl(ia->ia_sockmask.sin_addr.s_addr = i);
1074*33de042dSApple OSS Distributions 		ev_msg.vendor_code      = KEV_VENDOR_APPLE;
1075*33de042dSApple OSS Distributions 		ev_msg.kev_class        = KEV_NETWORK_CLASS;
1076*33de042dSApple OSS Distributions 		ev_msg.kev_subclass     = KEV_INET_SUBCLASS;
1077*33de042dSApple OSS Distributions 
1078*33de042dSApple OSS Distributions 		ev_msg.event_code = KEV_INET_SIFNETMASK;
1079*33de042dSApple OSS Distributions 
1080*33de042dSApple OSS Distributions 		if (ia->ia_ifa.ifa_dstaddr) {
1081*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr = SIN(ia->ia_ifa.ifa_dstaddr)->sin_addr;
1082*33de042dSApple OSS Distributions 		} else {
1083*33de042dSApple OSS Distributions 			in_event_data.ia_dstaddr.s_addr = INADDR_ANY;
1084*33de042dSApple OSS Distributions 		}
1085*33de042dSApple OSS Distributions 		in_event_data.ia_addr           = ia->ia_addr.sin_addr;
1086*33de042dSApple OSS Distributions 		in_event_data.ia_net            = ia->ia_net;
1087*33de042dSApple OSS Distributions 		in_event_data.ia_netmask        = ia->ia_netmask;
1088*33de042dSApple OSS Distributions 		in_event_data.ia_subnet         = ia->ia_subnet;
1089*33de042dSApple OSS Distributions 		in_event_data.ia_subnetmask     = ia->ia_subnetmask;
1090*33de042dSApple OSS Distributions 		in_event_data.ia_netbroadcast   = ia->ia_netbroadcast;
1091*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1092*33de042dSApple OSS Distributions 		(void) strlcpy(&in_event_data.link_data.if_name[0],
1093*33de042dSApple OSS Distributions 		    ifp->if_name, IFNAMSIZ);
1094*33de042dSApple OSS Distributions 		in_event_data.link_data.if_family = ifp->if_family;
1095*33de042dSApple OSS Distributions 		in_event_data.link_data.if_unit  = (u_int32_t)ifp->if_unit;
1096*33de042dSApple OSS Distributions 
1097*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_ptr    = &in_event_data;
1098*33de042dSApple OSS Distributions 		ev_msg.dv[0].data_length = sizeof(struct kev_in_data);
1099*33de042dSApple OSS Distributions 		ev_msg.dv[1].data_length = 0;
1100*33de042dSApple OSS Distributions 
1101*33de042dSApple OSS Distributions 		dlil_post_complete_msg(ifp, &ev_msg);
1102*33de042dSApple OSS Distributions 		break;
1103*33de042dSApple OSS Distributions 	}
1104*33de042dSApple OSS Distributions 
1105*33de042dSApple OSS Distributions 	default:
1106*33de042dSApple OSS Distributions 		VERIFY(0);
1107*33de042dSApple OSS Distributions 		/* NOTREACHED */
1108*33de042dSApple OSS Distributions 	}
1109*33de042dSApple OSS Distributions 
1110*33de042dSApple OSS Distributions 	return error;
1111*33de042dSApple OSS Distributions }
1112*33de042dSApple OSS Distributions 
1113*33de042dSApple OSS Distributions /*
1114*33de042dSApple OSS Distributions  * Generic INET control operations (ioctl's).
1115*33de042dSApple OSS Distributions  *
1116*33de042dSApple OSS Distributions  * ifp is NULL if not an interface-specific ioctl.
1117*33de042dSApple OSS Distributions  *
1118*33de042dSApple OSS Distributions  * Most of the routines called to handle the ioctls would end up being
1119*33de042dSApple OSS Distributions  * tail-call optimized, which unfortunately causes this routine to
1120*33de042dSApple OSS Distributions  * consume too much stack space; this is the reason for the "noinline"
1121*33de042dSApple OSS Distributions  * attribute used on those routines.
1122*33de042dSApple OSS Distributions  *
1123*33de042dSApple OSS Distributions  * If called directly from within the networking stack (as opposed to via
1124*33de042dSApple OSS Distributions  * pru_control), the socket parameter may be NULL.
1125*33de042dSApple OSS Distributions  */
1126*33de042dSApple OSS Distributions int
in_control(struct socket * so,u_long cmd,caddr_t __sized_by (IOCPARM_LEN (cmd))data,struct ifnet * ifp,struct proc * p)1127*33de042dSApple OSS Distributions in_control(struct socket *so, u_long cmd, caddr_t __sized_by(IOCPARM_LEN(cmd)) data,
1128*33de042dSApple OSS Distributions     struct ifnet *ifp, struct proc *p)
1129*33de042dSApple OSS Distributions {
1130*33de042dSApple OSS Distributions 	struct ifreq *__single ifr = NULL;
1131*33de042dSApple OSS Distributions 	struct sockaddr_in addr, dstaddr;
1132*33de042dSApple OSS Distributions 	struct sockaddr_in sin;
1133*33de042dSApple OSS Distributions 	struct sockaddr_in *__single sa = NULL;
1134*33de042dSApple OSS Distributions 	boolean_t privileged = (proc_suser(p) == 0);
1135*33de042dSApple OSS Distributions 	boolean_t so_unlocked = FALSE;
1136*33de042dSApple OSS Distributions 	struct in_ifaddr *__single ia = NULL;
1137*33de042dSApple OSS Distributions 	struct ifaddr *__single ifa;
1138*33de042dSApple OSS Distributions 	int error = 0;
1139*33de042dSApple OSS Distributions 	int intval;
1140*33de042dSApple OSS Distributions 
1141*33de042dSApple OSS Distributions 	/* In case it's NULL, make sure it came from the kernel */
1142*33de042dSApple OSS Distributions 	VERIFY(so != NULL || p == kernproc);
1143*33de042dSApple OSS Distributions 
1144*33de042dSApple OSS Distributions 	/*
1145*33de042dSApple OSS Distributions 	 * ioctls which don't require ifp, but require socket.
1146*33de042dSApple OSS Distributions 	 */
1147*33de042dSApple OSS Distributions 	switch (cmd) {
1148*33de042dSApple OSS Distributions 	case SIOCGASSOCIDS32:           /* struct so_aidreq32 */
1149*33de042dSApple OSS Distributions 	case SIOCGASSOCIDS64:           /* struct so_aidreq64 */
1150*33de042dSApple OSS Distributions 		return inctl_associd(so, cmd, data);
1151*33de042dSApple OSS Distributions 	/* NOTREACHED */
1152*33de042dSApple OSS Distributions 
1153*33de042dSApple OSS Distributions 	case SIOCGCONNIDS32:            /* struct so_cidreq32 */
1154*33de042dSApple OSS Distributions 	case SIOCGCONNIDS64:            /* struct so_cidreq64 */
1155*33de042dSApple OSS Distributions 		return inctl_connid(so, cmd, data);
1156*33de042dSApple OSS Distributions 	/* NOTREACHED */
1157*33de042dSApple OSS Distributions 
1158*33de042dSApple OSS Distributions 	case SIOCGCONNINFO32:           /* struct so_cinforeq32 */
1159*33de042dSApple OSS Distributions 	case SIOCGCONNINFO64:           /* struct so_cinforeq64 */
1160*33de042dSApple OSS Distributions 		return inctl_conninfo(so, cmd, data);
1161*33de042dSApple OSS Distributions 		/* NOTREACHED */
1162*33de042dSApple OSS Distributions 	}
1163*33de042dSApple OSS Distributions 
1164*33de042dSApple OSS Distributions 	/*
1165*33de042dSApple OSS Distributions 	 * The rest of ioctls require ifp; reject if we don't have one;
1166*33de042dSApple OSS Distributions 	 * return ENXIO to be consistent with ifioctl().
1167*33de042dSApple OSS Distributions 	 */
1168*33de042dSApple OSS Distributions 	if (ifp == NULL) {
1169*33de042dSApple OSS Distributions 		return ENXIO;
1170*33de042dSApple OSS Distributions 	}
1171*33de042dSApple OSS Distributions 
1172*33de042dSApple OSS Distributions 	/*
1173*33de042dSApple OSS Distributions 	 * ioctls which require ifp but not interface address.
1174*33de042dSApple OSS Distributions 	 */
1175*33de042dSApple OSS Distributions 	switch (cmd) {
1176*33de042dSApple OSS Distributions 	case SIOCAUTOADDR:              /* struct ifreq */
1177*33de042dSApple OSS Distributions 		if (!privileged) {
1178*33de042dSApple OSS Distributions 			return EPERM;
1179*33de042dSApple OSS Distributions 		}
1180*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1181*33de042dSApple OSS Distributions 		return inctl_autoaddr(ifp, ifr);
1182*33de042dSApple OSS Distributions 	/* NOTREACHED */
1183*33de042dSApple OSS Distributions 
1184*33de042dSApple OSS Distributions 	case SIOCARPIPLL:               /* struct ifreq */
1185*33de042dSApple OSS Distributions 		if (!privileged) {
1186*33de042dSApple OSS Distributions 			return EPERM;
1187*33de042dSApple OSS Distributions 		}
1188*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1189*33de042dSApple OSS Distributions 		return inctl_arpipll(ifp, ifr);
1190*33de042dSApple OSS Distributions 	/* NOTREACHED */
1191*33de042dSApple OSS Distributions 
1192*33de042dSApple OSS Distributions 	case SIOCGETROUTERMODE:         /* struct ifreq */
1193*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1194*33de042dSApple OSS Distributions 		intval = (ifp->if_eflags & IFEF_IPV4_ROUTER) != 0 ? 1 : 0;
1195*33de042dSApple OSS Distributions 		bcopy(&intval, &ifr->ifr_intval, sizeof(intval));
1196*33de042dSApple OSS Distributions 		return 0;
1197*33de042dSApple OSS Distributions 	/* NOTREACHED */
1198*33de042dSApple OSS Distributions 
1199*33de042dSApple OSS Distributions 	case SIOCSETROUTERMODE:         /* struct ifreq */
1200*33de042dSApple OSS Distributions 		if (!privileged) {
1201*33de042dSApple OSS Distributions 			return EPERM;
1202*33de042dSApple OSS Distributions 		}
1203*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1204*33de042dSApple OSS Distributions 		return inctl_setrouter(ifp, ifr);
1205*33de042dSApple OSS Distributions 	/* NOTREACHED */
1206*33de042dSApple OSS Distributions 
1207*33de042dSApple OSS Distributions 	case SIOCPROTOATTACH:           /* struct ifreq */
1208*33de042dSApple OSS Distributions 		if (!privileged) {
1209*33de042dSApple OSS Distributions 			return EPERM;
1210*33de042dSApple OSS Distributions 		}
1211*33de042dSApple OSS Distributions 		return in_domifattach(ifp);
1212*33de042dSApple OSS Distributions 	/* NOTREACHED */
1213*33de042dSApple OSS Distributions 
1214*33de042dSApple OSS Distributions 	case SIOCPROTODETACH:           /* struct ifreq */
1215*33de042dSApple OSS Distributions 		if (!privileged) {
1216*33de042dSApple OSS Distributions 			return EPERM;
1217*33de042dSApple OSS Distributions 		}
1218*33de042dSApple OSS Distributions 
1219*33de042dSApple OSS Distributions 		/*
1220*33de042dSApple OSS Distributions 		 * If an IPv4 address is still present, refuse to detach.
1221*33de042dSApple OSS Distributions 		 */
1222*33de042dSApple OSS Distributions 		ifnet_lock_shared(ifp);
1223*33de042dSApple OSS Distributions 		TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
1224*33de042dSApple OSS Distributions 			IFA_LOCK(ifa);
1225*33de042dSApple OSS Distributions 			if (ifa->ifa_addr->sa_family == AF_INET) {
1226*33de042dSApple OSS Distributions 				IFA_UNLOCK(ifa);
1227*33de042dSApple OSS Distributions 				break;
1228*33de042dSApple OSS Distributions 			}
1229*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
1230*33de042dSApple OSS Distributions 		}
1231*33de042dSApple OSS Distributions 		ifnet_lock_done(ifp);
1232*33de042dSApple OSS Distributions 		return (ifa == NULL) ? proto_unplumb(PF_INET, ifp) : EBUSY;
1233*33de042dSApple OSS Distributions 		/* NOTREACHED */
1234*33de042dSApple OSS Distributions 	}
1235*33de042dSApple OSS Distributions 
1236*33de042dSApple OSS Distributions 	/*
1237*33de042dSApple OSS Distributions 	 * ioctls which require interface address; obtain sockaddr_in.
1238*33de042dSApple OSS Distributions 	 */
1239*33de042dSApple OSS Distributions 	switch (cmd) {
1240*33de042dSApple OSS Distributions 	case SIOCAIFADDR:               /* struct {if,in_}aliasreq */
1241*33de042dSApple OSS Distributions 		if (!privileged) {
1242*33de042dSApple OSS Distributions 			return EPERM;
1243*33de042dSApple OSS Distributions 		}
1244*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&((struct in_aliasreq *)(void *)data)->ifra_addr,
1245*33de042dSApple OSS Distributions 		    &sin, sizeof(sin));
1246*33de042dSApple OSS Distributions 		sa = &sin;
1247*33de042dSApple OSS Distributions 		break;
1248*33de042dSApple OSS Distributions 
1249*33de042dSApple OSS Distributions 	case SIOCDIFADDR:               /* struct ifreq */
1250*33de042dSApple OSS Distributions 	case SIOCSIFADDR:               /* struct ifreq */
1251*33de042dSApple OSS Distributions 	case SIOCSIFDSTADDR:            /* struct ifreq */
1252*33de042dSApple OSS Distributions 	case SIOCSIFNETMASK:            /* struct ifreq */
1253*33de042dSApple OSS Distributions 	case SIOCSIFBRDADDR:            /* struct ifreq */
1254*33de042dSApple OSS Distributions 		if (!privileged) {
1255*33de042dSApple OSS Distributions 			return EPERM;
1256*33de042dSApple OSS Distributions 		}
1257*33de042dSApple OSS Distributions 		OS_FALLTHROUGH;
1258*33de042dSApple OSS Distributions 	case SIOCGIFADDR:               /* struct ifreq */
1259*33de042dSApple OSS Distributions 	case SIOCGIFDSTADDR:            /* struct ifreq */
1260*33de042dSApple OSS Distributions 	case SIOCGIFNETMASK:            /* struct ifreq */
1261*33de042dSApple OSS Distributions 	case SIOCGIFBRDADDR:            /* struct ifreq */
1262*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1263*33de042dSApple OSS Distributions 		SOCKADDR_COPY(&ifr->ifr_addr, &sin, sizeof(sin));
1264*33de042dSApple OSS Distributions 		sa = &sin;
1265*33de042dSApple OSS Distributions 		break;
1266*33de042dSApple OSS Distributions 	}
1267*33de042dSApple OSS Distributions 
1268*33de042dSApple OSS Distributions 	/*
1269*33de042dSApple OSS Distributions 	 * Find address for this interface, if it exists.
1270*33de042dSApple OSS Distributions 	 *
1271*33de042dSApple OSS Distributions 	 * If an alias address was specified, find that one instead of
1272*33de042dSApple OSS Distributions 	 * the first one on the interface, if possible.
1273*33de042dSApple OSS Distributions 	 */
1274*33de042dSApple OSS Distributions 	VERIFY(ia == NULL);
1275*33de042dSApple OSS Distributions 	if (sa != NULL) {
1276*33de042dSApple OSS Distributions 		struct in_ifaddr *iap;
1277*33de042dSApple OSS Distributions 
1278*33de042dSApple OSS Distributions 		/*
1279*33de042dSApple OSS Distributions 		 * Any failures from this point on must take into account
1280*33de042dSApple OSS Distributions 		 * a non-NULL "ia" with an outstanding reference count, and
1281*33de042dSApple OSS Distributions 		 * therefore requires ifa_remref.  Jump to "done" label
1282*33de042dSApple OSS Distributions 		 * instead of calling return if "ia" is valid.
1283*33de042dSApple OSS Distributions 		 */
1284*33de042dSApple OSS Distributions 		lck_rw_lock_shared(&in_ifaddr_rwlock);
1285*33de042dSApple OSS Distributions 		TAILQ_FOREACH(iap, INADDR_HASH(sa->sin_addr.s_addr), ia_hash) {
1286*33de042dSApple OSS Distributions 			IFA_LOCK(&iap->ia_ifa);
1287*33de042dSApple OSS Distributions 			if (iap->ia_ifp == ifp &&
1288*33de042dSApple OSS Distributions 			    iap->ia_addr.sin_addr.s_addr ==
1289*33de042dSApple OSS Distributions 			    sa->sin_addr.s_addr) {
1290*33de042dSApple OSS Distributions 				ia = iap;
1291*33de042dSApple OSS Distributions 				ifa_addref(&iap->ia_ifa);
1292*33de042dSApple OSS Distributions 				IFA_UNLOCK(&iap->ia_ifa);
1293*33de042dSApple OSS Distributions 				break;
1294*33de042dSApple OSS Distributions 			}
1295*33de042dSApple OSS Distributions 			IFA_UNLOCK(&iap->ia_ifa);
1296*33de042dSApple OSS Distributions 		}
1297*33de042dSApple OSS Distributions 		lck_rw_done(&in_ifaddr_rwlock);
1298*33de042dSApple OSS Distributions 
1299*33de042dSApple OSS Distributions 		if (ia == NULL) {
1300*33de042dSApple OSS Distributions 			ifnet_lock_shared(ifp);
1301*33de042dSApple OSS Distributions 			TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
1302*33de042dSApple OSS Distributions 				iap = ifatoia(ifa);
1303*33de042dSApple OSS Distributions 				IFA_LOCK(&iap->ia_ifa);
1304*33de042dSApple OSS Distributions 				if (iap->ia_addr.sin_family == AF_INET) {
1305*33de042dSApple OSS Distributions 					ia = iap;
1306*33de042dSApple OSS Distributions 					ifa_addref(&iap->ia_ifa);
1307*33de042dSApple OSS Distributions 					IFA_UNLOCK(&iap->ia_ifa);
1308*33de042dSApple OSS Distributions 					break;
1309*33de042dSApple OSS Distributions 				}
1310*33de042dSApple OSS Distributions 				IFA_UNLOCK(&iap->ia_ifa);
1311*33de042dSApple OSS Distributions 			}
1312*33de042dSApple OSS Distributions 			ifnet_lock_done(ifp);
1313*33de042dSApple OSS Distributions 		}
1314*33de042dSApple OSS Distributions 	}
1315*33de042dSApple OSS Distributions 
1316*33de042dSApple OSS Distributions 	/*
1317*33de042dSApple OSS Distributions 	 * Unlock the socket since ifnet_ioctl() may be invoked by
1318*33de042dSApple OSS Distributions 	 * one of the ioctl handlers below.  Socket will be re-locked
1319*33de042dSApple OSS Distributions 	 * prior to returning.
1320*33de042dSApple OSS Distributions 	 */
1321*33de042dSApple OSS Distributions 	if (so != NULL) {
1322*33de042dSApple OSS Distributions 		socket_unlock(so, 0);
1323*33de042dSApple OSS Distributions 		so_unlocked = TRUE;
1324*33de042dSApple OSS Distributions 	}
1325*33de042dSApple OSS Distributions 
1326*33de042dSApple OSS Distributions 	switch (cmd) {
1327*33de042dSApple OSS Distributions 	case SIOCAIFADDR:               /* struct {if,in_}aliasreq */
1328*33de042dSApple OSS Distributions 	case SIOCDIFADDR:               /* struct ifreq */
1329*33de042dSApple OSS Distributions 		if (cmd == SIOCAIFADDR) {
1330*33de042dSApple OSS Distributions 			SOCKADDR_COPY(&((struct in_aliasreq *)(void *)data)->
1331*33de042dSApple OSS Distributions 			    ifra_addr, &addr, sizeof(addr));
1332*33de042dSApple OSS Distributions 			SOCKADDR_COPY(&((struct in_aliasreq *)(void *)data)->
1333*33de042dSApple OSS Distributions 			    ifra_dstaddr, &dstaddr, sizeof(dstaddr));
1334*33de042dSApple OSS Distributions 		} else {
1335*33de042dSApple OSS Distributions 			VERIFY(cmd == SIOCDIFADDR);
1336*33de042dSApple OSS Distributions 			SOCKADDR_COPY(&((struct ifreq *)(void *)data)->ifr_addr,
1337*33de042dSApple OSS Distributions 			    &addr, sizeof(addr));
1338*33de042dSApple OSS Distributions 			SOCKADDR_ZERO(&dstaddr, sizeof(dstaddr));
1339*33de042dSApple OSS Distributions 		}
1340*33de042dSApple OSS Distributions 
1341*33de042dSApple OSS Distributions 		if (addr.sin_family == AF_INET) {
1342*33de042dSApple OSS Distributions 			struct in_ifaddr *__single oia;
1343*33de042dSApple OSS Distributions 
1344*33de042dSApple OSS Distributions 			lck_rw_lock_shared(&in_ifaddr_rwlock);
1345*33de042dSApple OSS Distributions 			for (oia = ia; ia; ia = ia->ia_link.tqe_next) {
1346*33de042dSApple OSS Distributions 				IFA_LOCK(&ia->ia_ifa);
1347*33de042dSApple OSS Distributions 				if (ia->ia_ifp == ifp &&
1348*33de042dSApple OSS Distributions 				    ia->ia_addr.sin_addr.s_addr ==
1349*33de042dSApple OSS Distributions 				    addr.sin_addr.s_addr) {
1350*33de042dSApple OSS Distributions 					ifa_addref(&ia->ia_ifa);
1351*33de042dSApple OSS Distributions 					IFA_UNLOCK(&ia->ia_ifa);
1352*33de042dSApple OSS Distributions 					break;
1353*33de042dSApple OSS Distributions 				}
1354*33de042dSApple OSS Distributions 				IFA_UNLOCK(&ia->ia_ifa);
1355*33de042dSApple OSS Distributions 			}
1356*33de042dSApple OSS Distributions 			lck_rw_done(&in_ifaddr_rwlock);
1357*33de042dSApple OSS Distributions 			if (oia != NULL) {
1358*33de042dSApple OSS Distributions 				ifa_remref(&oia->ia_ifa);
1359*33de042dSApple OSS Distributions 			}
1360*33de042dSApple OSS Distributions 			if ((ifp->if_flags & IFF_POINTOPOINT) &&
1361*33de042dSApple OSS Distributions 			    (cmd == SIOCAIFADDR) &&
1362*33de042dSApple OSS Distributions 			    (dstaddr.sin_addr.s_addr == INADDR_ANY)) {
1363*33de042dSApple OSS Distributions 				error = EDESTADDRREQ;
1364*33de042dSApple OSS Distributions 				goto done;
1365*33de042dSApple OSS Distributions 			}
1366*33de042dSApple OSS Distributions 		} else if (cmd == SIOCAIFADDR) {
1367*33de042dSApple OSS Distributions 			error = EINVAL;
1368*33de042dSApple OSS Distributions 			goto done;
1369*33de042dSApple OSS Distributions 		}
1370*33de042dSApple OSS Distributions 		if (cmd == SIOCDIFADDR) {
1371*33de042dSApple OSS Distributions 			if (ia == NULL) {
1372*33de042dSApple OSS Distributions 				error = EADDRNOTAVAIL;
1373*33de042dSApple OSS Distributions 				goto done;
1374*33de042dSApple OSS Distributions 			}
1375*33de042dSApple OSS Distributions 
1376*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
1377*33de042dSApple OSS Distributions 			/*
1378*33de042dSApple OSS Distributions 			 * Avoid the race condition seen when two
1379*33de042dSApple OSS Distributions 			 * threads process SIOCDIFADDR command
1380*33de042dSApple OSS Distributions 			 * at the same time.
1381*33de042dSApple OSS Distributions 			 */
1382*33de042dSApple OSS Distributions 			while (ia->ia_ifa.ifa_debug & IFD_DETACHING) {
1383*33de042dSApple OSS Distributions 				os_log(OS_LOG_DEFAULT,
1384*33de042dSApple OSS Distributions 				    "Another thread is already attempting to "
1385*33de042dSApple OSS Distributions 				    "delete IPv4 address: %s on interface %s. "
1386*33de042dSApple OSS Distributions 				    "Go to sleep and check again after the operation is done",
1387*33de042dSApple OSS Distributions 				    inet_ntoa(sa->sin_addr), ia->ia_ifp->if_xname);
1388*33de042dSApple OSS Distributions 				ia->ia_ifa.ifa_del_waiters++;
1389*33de042dSApple OSS Distributions 				(void) msleep(ia->ia_ifa.ifa_del_wc, &ia->ia_ifa.ifa_lock, (PZERO - 1),
1390*33de042dSApple OSS Distributions 				    __func__, NULL);
1391*33de042dSApple OSS Distributions 				IFA_LOCK_ASSERT_HELD(&ia->ia_ifa);
1392*33de042dSApple OSS Distributions 			}
1393*33de042dSApple OSS Distributions 
1394*33de042dSApple OSS Distributions 			if ((ia->ia_ifa.ifa_debug & IFD_ATTACHED) == 0) {
1395*33de042dSApple OSS Distributions 				error = EADDRNOTAVAIL;
1396*33de042dSApple OSS Distributions 				IFA_UNLOCK(&ia->ia_ifa);
1397*33de042dSApple OSS Distributions 				goto done;
1398*33de042dSApple OSS Distributions 			}
1399*33de042dSApple OSS Distributions 
1400*33de042dSApple OSS Distributions 			ia->ia_ifa.ifa_debug |= IFD_DETACHING;
1401*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
1402*33de042dSApple OSS Distributions 		}
1403*33de042dSApple OSS Distributions 
1404*33de042dSApple OSS Distributions 		OS_FALLTHROUGH;
1405*33de042dSApple OSS Distributions 	case SIOCSIFADDR:               /* struct ifreq */
1406*33de042dSApple OSS Distributions 	case SIOCSIFDSTADDR:            /* struct ifreq */
1407*33de042dSApple OSS Distributions 	case SIOCSIFNETMASK:            /* struct ifreq */
1408*33de042dSApple OSS Distributions 		if (cmd == SIOCAIFADDR) {
1409*33de042dSApple OSS Distributions 			/* fell thru from above; just repeat it */
1410*33de042dSApple OSS Distributions 			SOCKADDR_COPY(&((struct in_aliasreq *)(void *)data)->
1411*33de042dSApple OSS Distributions 			    ifra_addr, &addr, sizeof(addr));
1412*33de042dSApple OSS Distributions 		} else {
1413*33de042dSApple OSS Distributions 			VERIFY(cmd == SIOCDIFADDR || cmd == SIOCSIFADDR ||
1414*33de042dSApple OSS Distributions 			    cmd == SIOCSIFNETMASK || cmd == SIOCSIFDSTADDR);
1415*33de042dSApple OSS Distributions 			SOCKADDR_COPY(&((struct ifreq *)(void *)data)->ifr_addr,
1416*33de042dSApple OSS Distributions 			    &addr, sizeof(addr));
1417*33de042dSApple OSS Distributions 		}
1418*33de042dSApple OSS Distributions 
1419*33de042dSApple OSS Distributions 		if (addr.sin_family != AF_INET && cmd == SIOCSIFADDR) {
1420*33de042dSApple OSS Distributions 			error = EINVAL;
1421*33de042dSApple OSS Distributions 			goto done;
1422*33de042dSApple OSS Distributions 		}
1423*33de042dSApple OSS Distributions 
1424*33de042dSApple OSS Distributions 		if ((cmd == SIOCAIFADDR || cmd == SIOCSIFADDR) &&
1425*33de042dSApple OSS Distributions 		    (IN_MULTICAST(ntohl(addr.sin_addr.s_addr)) ||
1426*33de042dSApple OSS Distributions 		    addr.sin_addr.s_addr == INADDR_BROADCAST ||
1427*33de042dSApple OSS Distributions 		    addr.sin_addr.s_addr == INADDR_ANY)) {
1428*33de042dSApple OSS Distributions 			error = EINVAL;
1429*33de042dSApple OSS Distributions 			goto done;
1430*33de042dSApple OSS Distributions 		}
1431*33de042dSApple OSS Distributions 
1432*33de042dSApple OSS Distributions 		if (ia == NULL) {
1433*33de042dSApple OSS Distributions 			ia = in_ifaddr_alloc();
1434*33de042dSApple OSS Distributions 			if (ia == NULL) {
1435*33de042dSApple OSS Distributions 				error = ENOBUFS;
1436*33de042dSApple OSS Distributions 				goto done;
1437*33de042dSApple OSS Distributions 			}
1438*33de042dSApple OSS Distributions 			ifnet_lock_exclusive(ifp);
1439*33de042dSApple OSS Distributions 			ifa = &ia->ia_ifa;
1440*33de042dSApple OSS Distributions 			IFA_LOCK(ifa);
1441*33de042dSApple OSS Distributions 			IA_HASH_INIT(ia);
1442*33de042dSApple OSS Distributions 			ifa->ifa_addr = SA(&ia->ia_addr);
1443*33de042dSApple OSS Distributions 			ifa->ifa_dstaddr = SA(&ia->ia_dstaddr);
1444*33de042dSApple OSS Distributions 			ifa->ifa_netmask = SA(&ia->ia_sockmask);
1445*33de042dSApple OSS Distributions 			ia->ia_sockmask.sin_len = offsetof(struct sockaddr_in, sin_zero);
1446*33de042dSApple OSS Distributions 			if (ifp->if_flags & IFF_BROADCAST) {
1447*33de042dSApple OSS Distributions 				ia->ia_broadaddr.sin_len = sizeof(ia->ia_addr);
1448*33de042dSApple OSS Distributions 				ia->ia_broadaddr.sin_family = AF_INET;
1449*33de042dSApple OSS Distributions 			}
1450*33de042dSApple OSS Distributions 			ia->ia_ifp = ifp;
1451*33de042dSApple OSS Distributions 			if (!(ifp->if_flags & IFF_LOOPBACK)) {
1452*33de042dSApple OSS Distributions 				in_interfaces++;
1453*33de042dSApple OSS Distributions 			}
1454*33de042dSApple OSS Distributions 			/* if_attach_ifa() holds a reference for ifa_link */
1455*33de042dSApple OSS Distributions 			if_attach_ifa(ifp, ifa);
1456*33de042dSApple OSS Distributions 			/*
1457*33de042dSApple OSS Distributions 			 * If we have to go through in_ifinit(), make sure
1458*33de042dSApple OSS Distributions 			 * to avoid installing route(s) based on this address
1459*33de042dSApple OSS Distributions 			 * via PFC_IFUP event, before the link resolver (ARP)
1460*33de042dSApple OSS Distributions 			 * initializes it.
1461*33de042dSApple OSS Distributions 			 */
1462*33de042dSApple OSS Distributions 			if (cmd == SIOCAIFADDR || cmd == SIOCSIFADDR) {
1463*33de042dSApple OSS Distributions 				ifa->ifa_debug |= IFD_NOTREADY;
1464*33de042dSApple OSS Distributions 			}
1465*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
1466*33de042dSApple OSS Distributions 			ifnet_lock_done(ifp);
1467*33de042dSApple OSS Distributions 			lck_rw_lock_exclusive(&in_ifaddr_rwlock);
1468*33de042dSApple OSS Distributions 			/* Hold a reference for ia_link */
1469*33de042dSApple OSS Distributions 			ifa_addref(ifa);
1470*33de042dSApple OSS Distributions 			TAILQ_INSERT_TAIL(&in_ifaddrhead, ia, ia_link);
1471*33de042dSApple OSS Distributions 			lck_rw_done(&in_ifaddr_rwlock);
1472*33de042dSApple OSS Distributions 			/* discard error */
1473*33de042dSApple OSS Distributions 			(void) in_domifattach(ifp);
1474*33de042dSApple OSS Distributions 			error = 0;
1475*33de042dSApple OSS Distributions 		}
1476*33de042dSApple OSS Distributions 		break;
1477*33de042dSApple OSS Distributions 	}
1478*33de042dSApple OSS Distributions 
1479*33de042dSApple OSS Distributions 	switch (cmd) {
1480*33de042dSApple OSS Distributions 	case SIOCGIFDSTADDR:            /* struct ifreq */
1481*33de042dSApple OSS Distributions 	case SIOCSIFDSTADDR:            /* struct ifreq */
1482*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1483*33de042dSApple OSS Distributions 		error = inctl_ifdstaddr(ifp, ia, cmd, ifr);
1484*33de042dSApple OSS Distributions 		break;
1485*33de042dSApple OSS Distributions 
1486*33de042dSApple OSS Distributions 	case SIOCGIFBRDADDR:            /* struct ifreq */
1487*33de042dSApple OSS Distributions 	case SIOCSIFBRDADDR:            /* struct ifreq */
1488*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1489*33de042dSApple OSS Distributions 		error = inctl_ifbrdaddr(ifp, ia, cmd, ifr);
1490*33de042dSApple OSS Distributions 		break;
1491*33de042dSApple OSS Distributions 
1492*33de042dSApple OSS Distributions 	case SIOCGIFNETMASK:            /* struct ifreq */
1493*33de042dSApple OSS Distributions 	case SIOCSIFNETMASK:            /* struct ifreq */
1494*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1495*33de042dSApple OSS Distributions 		error = inctl_ifnetmask(ifp, ia, cmd, ifr);
1496*33de042dSApple OSS Distributions 		break;
1497*33de042dSApple OSS Distributions 
1498*33de042dSApple OSS Distributions 	case SIOCGIFADDR:               /* struct ifreq */
1499*33de042dSApple OSS Distributions 	case SIOCSIFADDR:               /* struct ifreq */
1500*33de042dSApple OSS Distributions 	case SIOCAIFADDR:               /* struct {if,in_}aliasreq */
1501*33de042dSApple OSS Distributions 	case SIOCDIFADDR:               /* struct ifreq */
1502*33de042dSApple OSS Distributions 		ifr = (struct ifreq *)(void *)data;
1503*33de042dSApple OSS Distributions 		error = inctl_ifaddr(ifp, ia, cmd, ifr);
1504*33de042dSApple OSS Distributions 		break;
1505*33de042dSApple OSS Distributions 
1506*33de042dSApple OSS Distributions 	default:
1507*33de042dSApple OSS Distributions 		error = EOPNOTSUPP;
1508*33de042dSApple OSS Distributions 		break;
1509*33de042dSApple OSS Distributions 	}
1510*33de042dSApple OSS Distributions 
1511*33de042dSApple OSS Distributions done:
1512*33de042dSApple OSS Distributions 	if (ia != NULL) {
1513*33de042dSApple OSS Distributions 		if (cmd == SIOCDIFADDR) {
1514*33de042dSApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
1515*33de042dSApple OSS Distributions 			ia->ia_ifa.ifa_debug &= ~IFD_DETACHING;
1516*33de042dSApple OSS Distributions 			if (ia->ia_ifa.ifa_del_waiters > 0) {
1517*33de042dSApple OSS Distributions 				ia->ia_ifa.ifa_del_waiters = 0;
1518*33de042dSApple OSS Distributions 				wakeup(ia->ia_ifa.ifa_del_wc);
1519*33de042dSApple OSS Distributions 			}
1520*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
1521*33de042dSApple OSS Distributions 		}
1522*33de042dSApple OSS Distributions 		ifa_remref(&ia->ia_ifa);
1523*33de042dSApple OSS Distributions 	}
1524*33de042dSApple OSS Distributions 	if (so_unlocked) {
1525*33de042dSApple OSS Distributions 		socket_lock(so, 0);
1526*33de042dSApple OSS Distributions 	}
1527*33de042dSApple OSS Distributions 
1528*33de042dSApple OSS Distributions 	return error;
1529*33de042dSApple OSS Distributions }
1530*33de042dSApple OSS Distributions 
1531*33de042dSApple OSS Distributions /*
1532*33de042dSApple OSS Distributions  * Delete any existing route for an interface.
1533*33de042dSApple OSS Distributions  */
1534*33de042dSApple OSS Distributions void
in_ifscrub(struct ifnet * ifp,struct in_ifaddr * ia,int locked)1535*33de042dSApple OSS Distributions in_ifscrub(struct ifnet *ifp, struct in_ifaddr *ia, int locked)
1536*33de042dSApple OSS Distributions {
1537*33de042dSApple OSS Distributions 	IFA_LOCK(&ia->ia_ifa);
1538*33de042dSApple OSS Distributions 	if ((ia->ia_flags & IFA_ROUTE) == 0) {
1539*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1540*33de042dSApple OSS Distributions 		return;
1541*33de042dSApple OSS Distributions 	}
1542*33de042dSApple OSS Distributions 	IFA_UNLOCK(&ia->ia_ifa);
1543*33de042dSApple OSS Distributions 	if (!locked) {
1544*33de042dSApple OSS Distributions 		lck_mtx_lock(rnh_lock);
1545*33de042dSApple OSS Distributions 	}
1546*33de042dSApple OSS Distributions 	if (ifp->if_flags & (IFF_LOOPBACK | IFF_POINTOPOINT)) {
1547*33de042dSApple OSS Distributions 		rtinit_locked(&(ia->ia_ifa), RTM_DELETE, RTF_HOST);
1548*33de042dSApple OSS Distributions 	} else {
1549*33de042dSApple OSS Distributions 		rtinit_locked(&(ia->ia_ifa), RTM_DELETE, 0);
1550*33de042dSApple OSS Distributions 	}
1551*33de042dSApple OSS Distributions 	IFA_LOCK(&ia->ia_ifa);
1552*33de042dSApple OSS Distributions 	ia->ia_flags &= ~IFA_ROUTE;
1553*33de042dSApple OSS Distributions 	IFA_UNLOCK(&ia->ia_ifa);
1554*33de042dSApple OSS Distributions 	if (!locked) {
1555*33de042dSApple OSS Distributions 		lck_mtx_unlock(rnh_lock);
1556*33de042dSApple OSS Distributions 	}
1557*33de042dSApple OSS Distributions }
1558*33de042dSApple OSS Distributions 
1559*33de042dSApple OSS Distributions /*
1560*33de042dSApple OSS Distributions  * Caller must hold in_ifaddr_rwlock as writer.
1561*33de042dSApple OSS Distributions  */
1562*33de042dSApple OSS Distributions static void
in_iahash_remove(struct in_ifaddr * ia)1563*33de042dSApple OSS Distributions in_iahash_remove(struct in_ifaddr *ia)
1564*33de042dSApple OSS Distributions {
1565*33de042dSApple OSS Distributions 	LCK_RW_ASSERT(&in_ifaddr_rwlock, LCK_RW_ASSERT_EXCLUSIVE);
1566*33de042dSApple OSS Distributions 	IFA_LOCK_ASSERT_HELD(&ia->ia_ifa);
1567*33de042dSApple OSS Distributions 
1568*33de042dSApple OSS Distributions 	if (!IA_IS_HASHED(ia)) {
1569*33de042dSApple OSS Distributions 		panic("attempt to remove wrong ia %p from hash table", ia);
1570*33de042dSApple OSS Distributions 		/* NOTREACHED */
1571*33de042dSApple OSS Distributions 	}
1572*33de042dSApple OSS Distributions 	TAILQ_REMOVE(INADDR_HASH(ia->ia_addr.sin_addr.s_addr), ia, ia_hash);
1573*33de042dSApple OSS Distributions 	IA_HASH_INIT(ia);
1574*33de042dSApple OSS Distributions 	ifa_remref(&ia->ia_ifa);
1575*33de042dSApple OSS Distributions }
1576*33de042dSApple OSS Distributions 
1577*33de042dSApple OSS Distributions /*
1578*33de042dSApple OSS Distributions  * Caller must hold in_ifaddr_rwlock as writer.
1579*33de042dSApple OSS Distributions  */
1580*33de042dSApple OSS Distributions static void
in_iahash_insert(struct in_ifaddr * ia)1581*33de042dSApple OSS Distributions in_iahash_insert(struct in_ifaddr *ia)
1582*33de042dSApple OSS Distributions {
1583*33de042dSApple OSS Distributions 	LCK_RW_ASSERT(&in_ifaddr_rwlock, LCK_RW_ASSERT_EXCLUSIVE);
1584*33de042dSApple OSS Distributions 	IFA_LOCK_ASSERT_HELD(&ia->ia_ifa);
1585*33de042dSApple OSS Distributions 
1586*33de042dSApple OSS Distributions 	if (ia->ia_addr.sin_family != AF_INET) {
1587*33de042dSApple OSS Distributions 		panic("attempt to insert wrong ia %p into hash table", ia);
1588*33de042dSApple OSS Distributions 		/* NOTREACHED */
1589*33de042dSApple OSS Distributions 	} else if (IA_IS_HASHED(ia)) {
1590*33de042dSApple OSS Distributions 		panic("attempt to double-insert ia %p into hash table", ia);
1591*33de042dSApple OSS Distributions 		/* NOTREACHED */
1592*33de042dSApple OSS Distributions 	}
1593*33de042dSApple OSS Distributions 	TAILQ_INSERT_HEAD(INADDR_HASH(ia->ia_addr.sin_addr.s_addr),
1594*33de042dSApple OSS Distributions 	    ia, ia_hash);
1595*33de042dSApple OSS Distributions 	ifa_addref(&ia->ia_ifa);
1596*33de042dSApple OSS Distributions }
1597*33de042dSApple OSS Distributions 
1598*33de042dSApple OSS Distributions /*
1599*33de042dSApple OSS Distributions  * Some point to point interfaces that are tunnels borrow the address from
1600*33de042dSApple OSS Distributions  * an underlying interface (e.g. VPN server). In order for source address
1601*33de042dSApple OSS Distributions  * selection logic to find the underlying interface first, we add the address
1602*33de042dSApple OSS Distributions  * of borrowing point to point interfaces at the end of the list.
1603*33de042dSApple OSS Distributions  * (see rdar://6733789)
1604*33de042dSApple OSS Distributions  *
1605*33de042dSApple OSS Distributions  * Caller must hold in_ifaddr_rwlock as writer.
1606*33de042dSApple OSS Distributions  */
1607*33de042dSApple OSS Distributions static void
in_iahash_insert_ptp(struct in_ifaddr * ia)1608*33de042dSApple OSS Distributions in_iahash_insert_ptp(struct in_ifaddr *ia)
1609*33de042dSApple OSS Distributions {
1610*33de042dSApple OSS Distributions 	struct in_ifaddr *__single tmp_ifa;
1611*33de042dSApple OSS Distributions 	struct ifnet *__single tmp_ifp;
1612*33de042dSApple OSS Distributions 
1613*33de042dSApple OSS Distributions 	LCK_RW_ASSERT(&in_ifaddr_rwlock, LCK_RW_ASSERT_EXCLUSIVE);
1614*33de042dSApple OSS Distributions 	IFA_LOCK_ASSERT_HELD(&ia->ia_ifa);
1615*33de042dSApple OSS Distributions 
1616*33de042dSApple OSS Distributions 	if (ia->ia_addr.sin_family != AF_INET) {
1617*33de042dSApple OSS Distributions 		panic("attempt to insert wrong ia %p into hash table", ia);
1618*33de042dSApple OSS Distributions 		/* NOTREACHED */
1619*33de042dSApple OSS Distributions 	} else if (IA_IS_HASHED(ia)) {
1620*33de042dSApple OSS Distributions 		panic("attempt to double-insert ia %p into hash table", ia);
1621*33de042dSApple OSS Distributions 		/* NOTREACHED */
1622*33de042dSApple OSS Distributions 	}
1623*33de042dSApple OSS Distributions 	IFA_UNLOCK(&ia->ia_ifa);
1624*33de042dSApple OSS Distributions 	TAILQ_FOREACH(tmp_ifa, INADDR_HASH(ia->ia_addr.sin_addr.s_addr),
1625*33de042dSApple OSS Distributions 	    ia_hash) {
1626*33de042dSApple OSS Distributions 		IFA_LOCK(&tmp_ifa->ia_ifa);
1627*33de042dSApple OSS Distributions 		/* ia->ia_addr won't change, so check without lock */
1628*33de042dSApple OSS Distributions 		if (IA_SIN(tmp_ifa)->sin_addr.s_addr ==
1629*33de042dSApple OSS Distributions 		    ia->ia_addr.sin_addr.s_addr) {
1630*33de042dSApple OSS Distributions 			IFA_UNLOCK(&tmp_ifa->ia_ifa);
1631*33de042dSApple OSS Distributions 			break;
1632*33de042dSApple OSS Distributions 		}
1633*33de042dSApple OSS Distributions 		IFA_UNLOCK(&tmp_ifa->ia_ifa);
1634*33de042dSApple OSS Distributions 	}
1635*33de042dSApple OSS Distributions 	tmp_ifp = (tmp_ifa == NULL) ? NULL : tmp_ifa->ia_ifp;
1636*33de042dSApple OSS Distributions 
1637*33de042dSApple OSS Distributions 	IFA_LOCK(&ia->ia_ifa);
1638*33de042dSApple OSS Distributions 	if (tmp_ifp == NULL) {
1639*33de042dSApple OSS Distributions 		TAILQ_INSERT_HEAD(INADDR_HASH(ia->ia_addr.sin_addr.s_addr),
1640*33de042dSApple OSS Distributions 		    ia, ia_hash);
1641*33de042dSApple OSS Distributions 	} else {
1642*33de042dSApple OSS Distributions 		TAILQ_INSERT_TAIL(INADDR_HASH(ia->ia_addr.sin_addr.s_addr),
1643*33de042dSApple OSS Distributions 		    ia, ia_hash);
1644*33de042dSApple OSS Distributions 	}
1645*33de042dSApple OSS Distributions 	ifa_addref(&ia->ia_ifa);
1646*33de042dSApple OSS Distributions }
1647*33de042dSApple OSS Distributions 
1648*33de042dSApple OSS Distributions /*
1649*33de042dSApple OSS Distributions  * Initialize an interface's internet address
1650*33de042dSApple OSS Distributions  * and routing table entry.
1651*33de042dSApple OSS Distributions  */
1652*33de042dSApple OSS Distributions static int
in_ifinit(struct ifnet * ifp,struct in_ifaddr * ia,struct sockaddr_in * sin,int scrub)1653*33de042dSApple OSS Distributions in_ifinit(struct ifnet *ifp, struct in_ifaddr *ia, struct sockaddr_in *sin,
1654*33de042dSApple OSS Distributions     int scrub)
1655*33de042dSApple OSS Distributions {
1656*33de042dSApple OSS Distributions 	u_int32_t i = ntohl(sin->sin_addr.s_addr);
1657*33de042dSApple OSS Distributions 	struct sockaddr_in oldaddr;
1658*33de042dSApple OSS Distributions 	int flags = RTF_UP, error;
1659*33de042dSApple OSS Distributions 	struct ifaddr *__single ifa0;
1660*33de042dSApple OSS Distributions 	unsigned int cmd;
1661*33de042dSApple OSS Distributions 	int oldremoved = 0;
1662*33de042dSApple OSS Distributions 
1663*33de042dSApple OSS Distributions 	/* Take an extra reference for this routine */
1664*33de042dSApple OSS Distributions 	ifa_addref(&ia->ia_ifa);
1665*33de042dSApple OSS Distributions 
1666*33de042dSApple OSS Distributions 	lck_rw_lock_exclusive(&in_ifaddr_rwlock);
1667*33de042dSApple OSS Distributions 	IFA_LOCK(&ia->ia_ifa);
1668*33de042dSApple OSS Distributions 	oldaddr = ia->ia_addr;
1669*33de042dSApple OSS Distributions 	if (IA_IS_HASHED(ia)) {
1670*33de042dSApple OSS Distributions 		oldremoved = 1;
1671*33de042dSApple OSS Distributions 		in_iahash_remove(ia);
1672*33de042dSApple OSS Distributions 	}
1673*33de042dSApple OSS Distributions 	ia->ia_addr = *sin;
1674*33de042dSApple OSS Distributions 	/*
1675*33de042dSApple OSS Distributions 	 * Interface addresses should not contain port or sin_zero information.
1676*33de042dSApple OSS Distributions 	 */
1677*33de042dSApple OSS Distributions 	SIN(&ia->ia_addr)->sin_family = AF_INET;
1678*33de042dSApple OSS Distributions 	SIN(&ia->ia_addr)->sin_len = sizeof(struct sockaddr_in);
1679*33de042dSApple OSS Distributions 	SIN(&ia->ia_addr)->sin_port = 0;
1680*33de042dSApple OSS Distributions 	bzero(&SIN(&ia->ia_addr)->sin_zero, sizeof(sin->sin_zero));
1681*33de042dSApple OSS Distributions 	if ((ifp->if_flags & IFF_POINTOPOINT)) {
1682*33de042dSApple OSS Distributions 		in_iahash_insert_ptp(ia);
1683*33de042dSApple OSS Distributions 	} else {
1684*33de042dSApple OSS Distributions 		in_iahash_insert(ia);
1685*33de042dSApple OSS Distributions 	}
1686*33de042dSApple OSS Distributions 	IFA_UNLOCK(&ia->ia_ifa);
1687*33de042dSApple OSS Distributions 	lck_rw_done(&in_ifaddr_rwlock);
1688*33de042dSApple OSS Distributions 
1689*33de042dSApple OSS Distributions 	/*
1690*33de042dSApple OSS Distributions 	 * Give the interface a chance to initialize if this is its first
1691*33de042dSApple OSS Distributions 	 * address, and to validate the address if necessary.  Send down
1692*33de042dSApple OSS Distributions 	 * SIOCSIFADDR for first address, and SIOCAIFADDR for alias(es).
1693*33de042dSApple OSS Distributions 	 * We find the first IPV4 address assigned to it and check if this
1694*33de042dSApple OSS Distributions 	 * is the same as the one passed into this routine.
1695*33de042dSApple OSS Distributions 	 */
1696*33de042dSApple OSS Distributions 	ifa0 = ifa_ifpgetprimary(ifp, AF_INET);
1697*33de042dSApple OSS Distributions 	cmd = (&ia->ia_ifa == ifa0) ? SIOCSIFADDR : SIOCAIFADDR;
1698*33de042dSApple OSS Distributions 	error = ifnet_ioctl(ifp, PF_INET, cmd, ia);
1699*33de042dSApple OSS Distributions 	if (error == EOPNOTSUPP) {
1700*33de042dSApple OSS Distributions 		error = 0;
1701*33de042dSApple OSS Distributions 	}
1702*33de042dSApple OSS Distributions 	/*
1703*33de042dSApple OSS Distributions 	 * If we've just sent down SIOCAIFADDR, send another ioctl down
1704*33de042dSApple OSS Distributions 	 * for SIOCSIFADDR for the first IPV4 address of the interface,
1705*33de042dSApple OSS Distributions 	 * because an address change on one of the addresses will result
1706*33de042dSApple OSS Distributions 	 * in the removal of the previous first IPV4 address.  KDP needs
1707*33de042dSApple OSS Distributions 	 * be reconfigured with the current primary IPV4 address.
1708*33de042dSApple OSS Distributions 	 */
1709*33de042dSApple OSS Distributions 	if (error == 0 && cmd == SIOCAIFADDR) {
1710*33de042dSApple OSS Distributions 		/*
1711*33de042dSApple OSS Distributions 		 * NOTE: SIOCSIFADDR is defined with struct ifreq
1712*33de042dSApple OSS Distributions 		 * as parameter, but here we are sending it down
1713*33de042dSApple OSS Distributions 		 * to the interface with a pointer to struct ifaddr,
1714*33de042dSApple OSS Distributions 		 * for legacy reasons.
1715*33de042dSApple OSS Distributions 		 */
1716*33de042dSApple OSS Distributions 		error = ifnet_ioctl(ifp, PF_INET, SIOCSIFADDR, ifa0);
1717*33de042dSApple OSS Distributions 		if (error == EOPNOTSUPP) {
1718*33de042dSApple OSS Distributions 			error = 0;
1719*33de042dSApple OSS Distributions 		}
1720*33de042dSApple OSS Distributions 	}
1721*33de042dSApple OSS Distributions 
1722*33de042dSApple OSS Distributions 	/* Release reference from ifa_ifpgetprimary() */
1723*33de042dSApple OSS Distributions 	ifa_remref(ifa0);
1724*33de042dSApple OSS Distributions 
1725*33de042dSApple OSS Distributions 	if (error) {
1726*33de042dSApple OSS Distributions 		lck_rw_lock_exclusive(&in_ifaddr_rwlock);
1727*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
1728*33de042dSApple OSS Distributions 		if (IA_IS_HASHED(ia)) {
1729*33de042dSApple OSS Distributions 			in_iahash_remove(ia);
1730*33de042dSApple OSS Distributions 		}
1731*33de042dSApple OSS Distributions 		ia->ia_addr = oldaddr;
1732*33de042dSApple OSS Distributions 		if (oldremoved) {
1733*33de042dSApple OSS Distributions 			if ((ifp->if_flags & IFF_POINTOPOINT)) {
1734*33de042dSApple OSS Distributions 				in_iahash_insert_ptp(ia);
1735*33de042dSApple OSS Distributions 			} else {
1736*33de042dSApple OSS Distributions 				in_iahash_insert(ia);
1737*33de042dSApple OSS Distributions 			}
1738*33de042dSApple OSS Distributions 		}
1739*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1740*33de042dSApple OSS Distributions 		lck_rw_done(&in_ifaddr_rwlock);
1741*33de042dSApple OSS Distributions 		/* Release extra reference taken above */
1742*33de042dSApple OSS Distributions 		ifa_remref(&ia->ia_ifa);
1743*33de042dSApple OSS Distributions 		return error;
1744*33de042dSApple OSS Distributions 	}
1745*33de042dSApple OSS Distributions 	lck_mtx_lock(rnh_lock);
1746*33de042dSApple OSS Distributions 	IFA_LOCK(&ia->ia_ifa);
1747*33de042dSApple OSS Distributions 	/*
1748*33de042dSApple OSS Distributions 	 * Address has been initialized by the link resolver (ARP)
1749*33de042dSApple OSS Distributions 	 * via ifnet_ioctl() above; it may now generate route(s).
1750*33de042dSApple OSS Distributions 	 */
1751*33de042dSApple OSS Distributions 	ia->ia_ifa.ifa_debug &= ~IFD_NOTREADY;
1752*33de042dSApple OSS Distributions 	if (scrub) {
1753*33de042dSApple OSS Distributions 		ia->ia_ifa.ifa_addr = SA(&oldaddr);
1754*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1755*33de042dSApple OSS Distributions 		in_ifscrub(ifp, ia, 1);
1756*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
1757*33de042dSApple OSS Distributions 		ia->ia_ifa.ifa_addr = SA(&ia->ia_addr);
1758*33de042dSApple OSS Distributions 	}
1759*33de042dSApple OSS Distributions 	IFA_LOCK_ASSERT_HELD(&ia->ia_ifa);
1760*33de042dSApple OSS Distributions 	if (IN_CLASSA(i)) {
1761*33de042dSApple OSS Distributions 		ia->ia_netmask = IN_CLASSA_NET;
1762*33de042dSApple OSS Distributions 	} else if (IN_CLASSB(i)) {
1763*33de042dSApple OSS Distributions 		ia->ia_netmask = IN_CLASSB_NET;
1764*33de042dSApple OSS Distributions 	} else {
1765*33de042dSApple OSS Distributions 		ia->ia_netmask = IN_CLASSC_NET;
1766*33de042dSApple OSS Distributions 	}
1767*33de042dSApple OSS Distributions 	/*
1768*33de042dSApple OSS Distributions 	 * The subnet mask usually includes at least the standard network part,
1769*33de042dSApple OSS Distributions 	 * but may may be smaller in the case of supernetting.
1770*33de042dSApple OSS Distributions 	 * If it is set, we believe it.
1771*33de042dSApple OSS Distributions 	 */
1772*33de042dSApple OSS Distributions 	if (ia->ia_subnetmask == 0) {
1773*33de042dSApple OSS Distributions 		ia->ia_subnetmask = ia->ia_netmask;
1774*33de042dSApple OSS Distributions 		ia->ia_sockmask.sin_addr.s_addr = htonl(ia->ia_subnetmask);
1775*33de042dSApple OSS Distributions 	} else {
1776*33de042dSApple OSS Distributions 		ia->ia_netmask &= ia->ia_subnetmask;
1777*33de042dSApple OSS Distributions 	}
1778*33de042dSApple OSS Distributions 	ia->ia_net = i & ia->ia_netmask;
1779*33de042dSApple OSS Distributions 	ia->ia_subnet = i & ia->ia_subnetmask;
1780*33de042dSApple OSS Distributions 	in_socktrim(&ia->ia_sockmask);
1781*33de042dSApple OSS Distributions 	/*
1782*33de042dSApple OSS Distributions 	 * Add route for the network.
1783*33de042dSApple OSS Distributions 	 */
1784*33de042dSApple OSS Distributions 	ia->ia_ifa.ifa_metric = ifp->if_metric;
1785*33de042dSApple OSS Distributions 	if (ifp->if_flags & IFF_BROADCAST) {
1786*33de042dSApple OSS Distributions 		ia->ia_broadaddr.sin_addr.s_addr =
1787*33de042dSApple OSS Distributions 		    htonl(ia->ia_subnet | ~ia->ia_subnetmask);
1788*33de042dSApple OSS Distributions 		ia->ia_netbroadcast.s_addr =
1789*33de042dSApple OSS Distributions 		    htonl(ia->ia_net | ~ia->ia_netmask);
1790*33de042dSApple OSS Distributions 	} else if (ifp->if_flags & IFF_LOOPBACK) {
1791*33de042dSApple OSS Distributions 		ia->ia_ifa.ifa_dstaddr = ia->ia_ifa.ifa_addr;
1792*33de042dSApple OSS Distributions 		flags |= RTF_HOST;
1793*33de042dSApple OSS Distributions 	} else if (ifp->if_flags & IFF_POINTOPOINT) {
1794*33de042dSApple OSS Distributions 		if (ia->ia_dstaddr.sin_family != AF_INET) {
1795*33de042dSApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
1796*33de042dSApple OSS Distributions 			lck_mtx_unlock(rnh_lock);
1797*33de042dSApple OSS Distributions 			/* Release extra reference taken above */
1798*33de042dSApple OSS Distributions 			ifa_remref(&ia->ia_ifa);
1799*33de042dSApple OSS Distributions 			return 0;
1800*33de042dSApple OSS Distributions 		}
1801*33de042dSApple OSS Distributions 		ia->ia_dstaddr.sin_len = sizeof(struct sockaddr_in);
1802*33de042dSApple OSS Distributions 		flags |= RTF_HOST;
1803*33de042dSApple OSS Distributions 	}
1804*33de042dSApple OSS Distributions 	IFA_UNLOCK(&ia->ia_ifa);
1805*33de042dSApple OSS Distributions 
1806*33de042dSApple OSS Distributions 	if ((error = rtinit_locked(&(ia->ia_ifa), RTM_ADD, flags)) == 0) {
1807*33de042dSApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
1808*33de042dSApple OSS Distributions 		ia->ia_flags |= IFA_ROUTE;
1809*33de042dSApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1810*33de042dSApple OSS Distributions 	}
1811*33de042dSApple OSS Distributions 	lck_mtx_unlock(rnh_lock);
1812*33de042dSApple OSS Distributions 
1813*33de042dSApple OSS Distributions 	/* XXX check if the subnet route points to the same interface */
1814*33de042dSApple OSS Distributions 	if (error == EEXIST) {
1815*33de042dSApple OSS Distributions 		error = 0;
1816*33de042dSApple OSS Distributions 	}
1817*33de042dSApple OSS Distributions 
1818*33de042dSApple OSS Distributions 	/*
1819*33de042dSApple OSS Distributions 	 * If the interface supports multicast, join the "all hosts"
1820*33de042dSApple OSS Distributions 	 * multicast group on that interface.
1821*33de042dSApple OSS Distributions 	 */
1822*33de042dSApple OSS Distributions 	if (ifp->if_flags & IFF_MULTICAST) {
1823*33de042dSApple OSS Distributions 		struct in_addr addr;
1824*33de042dSApple OSS Distributions 
1825*33de042dSApple OSS Distributions 		lck_mtx_lock(&ifp->if_addrconfig_lock);
1826*33de042dSApple OSS Distributions 		addr.s_addr = htonl(INADDR_ALLHOSTS_GROUP);
1827*33de042dSApple OSS Distributions 		if (ifp->if_allhostsinm == NULL) {
1828*33de042dSApple OSS Distributions 			struct in_multi *__single inm;
1829*33de042dSApple OSS Distributions 			inm = in_addmulti(&addr, ifp);
1830*33de042dSApple OSS Distributions 
1831*33de042dSApple OSS Distributions 			if (inm != NULL) {
1832*33de042dSApple OSS Distributions 				/*
1833*33de042dSApple OSS Distributions 				 * Keep the reference on inm added by
1834*33de042dSApple OSS Distributions 				 * in_addmulti above for storing the
1835*33de042dSApple OSS Distributions 				 * pointer in allhostsinm.
1836*33de042dSApple OSS Distributions 				 */
1837*33de042dSApple OSS Distributions 				ifp->if_allhostsinm = inm;
1838*33de042dSApple OSS Distributions 			} else {
1839*33de042dSApple OSS Distributions 				printf("%s: failed to add membership to "
1840*33de042dSApple OSS Distributions 				    "all-hosts multicast address on %s\n",
1841*33de042dSApple OSS Distributions 				    __func__, if_name(ifp));
1842*33de042dSApple OSS Distributions 			}
1843*33de042dSApple OSS Distributions 		}
1844*33de042dSApple OSS Distributions 		lck_mtx_unlock(&ifp->if_addrconfig_lock);
1845*33de042dSApple OSS Distributions 	}
1846*33de042dSApple OSS Distributions 
1847*33de042dSApple OSS Distributions 	/* Release extra reference taken above */
1848*33de042dSApple OSS Distributions 	ifa_remref(&ia->ia_ifa);
1849*33de042dSApple OSS Distributions 
1850*33de042dSApple OSS Distributions 	if (error == 0) {
1851*33de042dSApple OSS Distributions 		/* invalidate route caches */
1852*33de042dSApple OSS Distributions 		routegenid_inet_update();
1853*33de042dSApple OSS Distributions 	}
1854*33de042dSApple OSS Distributions 
1855*33de042dSApple OSS Distributions 	return error;
1856*33de042dSApple OSS Distributions }
1857*33de042dSApple OSS Distributions 
1858*33de042dSApple OSS Distributions /*
1859*33de042dSApple OSS Distributions  * Return TRUE if the address might be a local broadcast address.
1860*33de042dSApple OSS Distributions  */
1861*33de042dSApple OSS Distributions boolean_t
in_broadcast(struct in_addr in,struct ifnet * ifp)1862*33de042dSApple OSS Distributions in_broadcast(struct in_addr in, struct ifnet *ifp)
1863*33de042dSApple OSS Distributions {
1864*33de042dSApple OSS Distributions 	struct ifaddr *__single ifa;
1865*33de042dSApple OSS Distributions 	u_int32_t t;
1866*33de042dSApple OSS Distributions 
1867*33de042dSApple OSS Distributions 	if (in.s_addr == INADDR_BROADCAST || in.s_addr == INADDR_ANY) {
1868*33de042dSApple OSS Distributions 		return TRUE;
1869*33de042dSApple OSS Distributions 	}
1870*33de042dSApple OSS Distributions 	if (!(ifp->if_flags & IFF_BROADCAST)) {
1871*33de042dSApple OSS Distributions 		return FALSE;
1872*33de042dSApple OSS Distributions 	}
1873*33de042dSApple OSS Distributions 	t = ntohl(in.s_addr);
1874*33de042dSApple OSS Distributions 
1875*33de042dSApple OSS Distributions 	/*
1876*33de042dSApple OSS Distributions 	 * Look through the list of addresses for a match
1877*33de042dSApple OSS Distributions 	 * with a broadcast address.
1878*33de042dSApple OSS Distributions 	 */
1879*33de042dSApple OSS Distributions 	ifnet_lock_shared(ifp);
1880*33de042dSApple OSS Distributions 	TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
1881*33de042dSApple OSS Distributions 		struct in_ifaddr *ia = ifatoia(ifa);
1882*33de042dSApple OSS Distributions 		IFA_LOCK(ifa);
1883*33de042dSApple OSS Distributions 		if (ifa->ifa_addr->sa_family == AF_INET &&
1884*33de042dSApple OSS Distributions 		    (in.s_addr == ia->ia_broadaddr.sin_addr.s_addr ||
1885*33de042dSApple OSS Distributions 		    in.s_addr == ia->ia_netbroadcast.s_addr ||
1886*33de042dSApple OSS Distributions 		    /*
1887*33de042dSApple OSS Distributions 		     * Check for old-style (host 0) broadcast.
1888*33de042dSApple OSS Distributions 		     */
1889*33de042dSApple OSS Distributions 		    t == ia->ia_subnet || t == ia->ia_net) &&
1890*33de042dSApple OSS Distributions 		    /*
1891*33de042dSApple OSS Distributions 		     * Check for an all one subnetmask. These
1892*33de042dSApple OSS Distributions 		     * only exist when an interface gets a secondary
1893*33de042dSApple OSS Distributions 		     * address.
1894*33de042dSApple OSS Distributions 		     */
1895*33de042dSApple OSS Distributions 		    ia->ia_subnetmask != (u_int32_t)0xffffffff) {
1896*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
1897*33de042dSApple OSS Distributions 			ifnet_lock_done(ifp);
1898*33de042dSApple OSS Distributions 			return TRUE;
1899*33de042dSApple OSS Distributions 		}
1900*33de042dSApple OSS Distributions 		IFA_UNLOCK(ifa);
1901*33de042dSApple OSS Distributions 	}
1902*33de042dSApple OSS Distributions 	ifnet_lock_done(ifp);
1903*33de042dSApple OSS Distributions 	return FALSE;
1904*33de042dSApple OSS Distributions #undef ia
1905*33de042dSApple OSS Distributions }
1906*33de042dSApple OSS Distributions 
1907*33de042dSApple OSS Distributions void
in_purgeaddrs(struct ifnet * ifp)1908*33de042dSApple OSS Distributions in_purgeaddrs(struct ifnet *ifp)
1909*33de042dSApple OSS Distributions {
1910*33de042dSApple OSS Distributions 	uint16_t addresses_count = 0;
1911*33de042dSApple OSS Distributions 	struct ifaddr **__counted_by(addresses_count) ifap = NULL;
1912*33de042dSApple OSS Distributions 	int err, i;
1913*33de042dSApple OSS Distributions 
1914*33de042dSApple OSS Distributions 	VERIFY(ifp != NULL);
1915*33de042dSApple OSS Distributions 
1916*33de042dSApple OSS Distributions 	/*
1917*33de042dSApple OSS Distributions 	 * Be nice, and try the civilized way first.  If we can't get
1918*33de042dSApple OSS Distributions 	 * rid of them this way, then do it the rough way.  We must
1919*33de042dSApple OSS Distributions 	 * only get here during detach time, after the ifnet has been
1920*33de042dSApple OSS Distributions 	 * removed from the global list and arrays.
1921*33de042dSApple OSS Distributions 	 */
1922*33de042dSApple OSS Distributions 	err = ifnet_get_address_list_family_internal(ifp, &ifap, &addresses_count,
1923*33de042dSApple OSS Distributions 	    AF_INET, 1, M_WAITOK, 0);
1924*33de042dSApple OSS Distributions 	if (err == 0 && ifap != NULL) {
1925*33de042dSApple OSS Distributions 		struct ifreq ifr;
1926*33de042dSApple OSS Distributions 
1927*33de042dSApple OSS Distributions 		bzero(&ifr, sizeof(ifr));
1928*33de042dSApple OSS Distributions 		(void) snprintf(ifr.ifr_name, sizeof(ifr.ifr_name),
1929*33de042dSApple OSS Distributions 		    "%s", if_name(ifp));
1930*33de042dSApple OSS Distributions 
1931*33de042dSApple OSS Distributions 		for (i = 0; ifap[i] != NULL; i++) {
1932*33de042dSApple OSS Distributions 			struct ifaddr *__single ifa;
1933*33de042dSApple OSS Distributions 
1934*33de042dSApple OSS Distributions 			ifa = ifap[i];
1935*33de042dSApple OSS Distributions 			IFA_LOCK(ifa);
1936*33de042dSApple OSS Distributions 			SOCKADDR_COPY(ifa->ifa_addr, &ifr.ifr_addr,
1937*33de042dSApple OSS Distributions 			    sizeof(struct sockaddr_in));
1938*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
1939*33de042dSApple OSS Distributions 			err = in_control(NULL, SIOCDIFADDR, (caddr_t)&ifr, ifp,
1940*33de042dSApple OSS Distributions 			    kernproc);
1941*33de042dSApple OSS Distributions 			/* if we lost the race, ignore it */
1942*33de042dSApple OSS Distributions 			if (err == EADDRNOTAVAIL) {
1943*33de042dSApple OSS Distributions 				err = 0;
1944*33de042dSApple OSS Distributions 			}
1945*33de042dSApple OSS Distributions 			if (err != 0) {
1946*33de042dSApple OSS Distributions 				char s_addr[MAX_IPv4_STR_LEN];
1947*33de042dSApple OSS Distributions 				char s_dstaddr[MAX_IPv4_STR_LEN];
1948*33de042dSApple OSS Distributions 				struct in_addr *__single s, *d;
1949*33de042dSApple OSS Distributions 
1950*33de042dSApple OSS Distributions 				IFA_LOCK(ifa);
1951*33de042dSApple OSS Distributions 				s = &SIN(ifa->ifa_addr)->sin_addr;
1952*33de042dSApple OSS Distributions 				d = &SIN(ifa->ifa_dstaddr)->sin_addr;
1953*33de042dSApple OSS Distributions 				(void) inet_ntop(AF_INET, &s->s_addr, s_addr,
1954*33de042dSApple OSS Distributions 				    sizeof(s_addr));
1955*33de042dSApple OSS Distributions 				(void) inet_ntop(AF_INET, &d->s_addr, s_dstaddr,
1956*33de042dSApple OSS Distributions 				    sizeof(s_dstaddr));
1957*33de042dSApple OSS Distributions 				IFA_UNLOCK(ifa);
1958*33de042dSApple OSS Distributions 
1959*33de042dSApple OSS Distributions 				printf("%s: SIOCDIFADDR ifp=%s ifa_addr=%s "
1960*33de042dSApple OSS Distributions 				    "ifa_dstaddr=%s (err=%d)\n", __func__,
1961*33de042dSApple OSS Distributions 				    ifp->if_xname, s_addr, s_dstaddr, err);
1962*33de042dSApple OSS Distributions 			}
1963*33de042dSApple OSS Distributions 		}
1964*33de042dSApple OSS Distributions 		ifnet_address_list_free_counted_by(ifap, addresses_count);
1965*33de042dSApple OSS Distributions 	} else if (err != 0 && err != ENXIO) {
1966*33de042dSApple OSS Distributions 		printf("%s: error retrieving list of AF_INET addresses for "
1967*33de042dSApple OSS Distributions 		    "ifp=%s (err=%d)\n", __func__, ifp->if_xname, err);
1968*33de042dSApple OSS Distributions 	}
1969*33de042dSApple OSS Distributions }
1970*33de042dSApple OSS Distributions 
1971*33de042dSApple OSS Distributions static struct in_ifaddr *
in_ifaddr_alloc(void)1972*33de042dSApple OSS Distributions in_ifaddr_alloc(void)
1973*33de042dSApple OSS Distributions {
1974*33de042dSApple OSS Distributions 	struct in_ifaddr *__single inifa;
1975*33de042dSApple OSS Distributions 
1976*33de042dSApple OSS Distributions 	inifa = kalloc_type(struct in_ifaddr, Z_ZERO | Z_WAITOK);
1977*33de042dSApple OSS Distributions 	if (inifa == NULL) {
1978*33de042dSApple OSS Distributions 		return NULL;
1979*33de042dSApple OSS Distributions 	}
1980*33de042dSApple OSS Distributions 
1981*33de042dSApple OSS Distributions 	inifa->ia_ifa.ifa_free = in_ifaddr_free;
1982*33de042dSApple OSS Distributions 	inifa->ia_ifa.ifa_debug |= IFD_ALLOC;
1983*33de042dSApple OSS Distributions 	inifa->ia_ifa.ifa_del_wc = &inifa->ia_ifa.ifa_debug;
1984*33de042dSApple OSS Distributions 	inifa->ia_ifa.ifa_del_waiters = 0;
1985*33de042dSApple OSS Distributions 	ifa_lock_init(&inifa->ia_ifa);
1986*33de042dSApple OSS Distributions 	ifa_initref(&inifa->ia_ifa);
1987*33de042dSApple OSS Distributions 
1988*33de042dSApple OSS Distributions 	return inifa;
1989*33de042dSApple OSS Distributions }
1990*33de042dSApple OSS Distributions 
1991*33de042dSApple OSS Distributions static void
in_ifaddr_free(struct ifaddr * ifa)1992*33de042dSApple OSS Distributions in_ifaddr_free(struct ifaddr *ifa)
1993*33de042dSApple OSS Distributions {
1994*33de042dSApple OSS Distributions 	struct in_ifaddr *__single inifa = ifatoia(ifa);
1995*33de042dSApple OSS Distributions 
1996*33de042dSApple OSS Distributions 	IFA_LOCK_ASSERT_HELD(ifa);
1997*33de042dSApple OSS Distributions 
1998*33de042dSApple OSS Distributions 	if (__improbable(!(ifa->ifa_debug & IFD_ALLOC))) {
1999*33de042dSApple OSS Distributions 		panic("%s: ifa %p cannot be freed", __func__, ifa);
2000*33de042dSApple OSS Distributions 		/* NOTREACHED */
2001*33de042dSApple OSS Distributions 	}
2002*33de042dSApple OSS Distributions 	IFA_UNLOCK(ifa);
2003*33de042dSApple OSS Distributions 	ifa_lock_destroy(ifa);
2004*33de042dSApple OSS Distributions 
2005*33de042dSApple OSS Distributions 	kfree_type(struct in_ifaddr, inifa);
2006*33de042dSApple OSS Distributions }
2007*33de042dSApple OSS Distributions 
2008*33de042dSApple OSS Distributions /*
2009*33de042dSApple OSS Distributions  * Handle SIOCGASSOCIDS ioctl for PF_INET domain.
2010*33de042dSApple OSS Distributions  */
2011*33de042dSApple OSS Distributions static int
in_getassocids(struct socket * so,uint32_t * cnt,user_addr_t aidp)2012*33de042dSApple OSS Distributions in_getassocids(struct socket *so, uint32_t *cnt, user_addr_t aidp)
2013*33de042dSApple OSS Distributions {
2014*33de042dSApple OSS Distributions 	struct inpcb *__single inp = sotoinpcb(so);
2015*33de042dSApple OSS Distributions 	sae_associd_t aid;
2016*33de042dSApple OSS Distributions 
2017*33de042dSApple OSS Distributions 	if (inp == NULL || inp->inp_state == INPCB_STATE_DEAD) {
2018*33de042dSApple OSS Distributions 		return EINVAL;
2019*33de042dSApple OSS Distributions 	}
2020*33de042dSApple OSS Distributions 
2021*33de042dSApple OSS Distributions 	/* INPCB has no concept of association */
2022*33de042dSApple OSS Distributions 	aid = SAE_ASSOCID_ANY;
2023*33de042dSApple OSS Distributions 	*cnt = 0;
2024*33de042dSApple OSS Distributions 
2025*33de042dSApple OSS Distributions 	/* just asking how many there are? */
2026*33de042dSApple OSS Distributions 	if (aidp == USER_ADDR_NULL) {
2027*33de042dSApple OSS Distributions 		return 0;
2028*33de042dSApple OSS Distributions 	}
2029*33de042dSApple OSS Distributions 
2030*33de042dSApple OSS Distributions 	return copyout(&aid, aidp, sizeof(aid));
2031*33de042dSApple OSS Distributions }
2032*33de042dSApple OSS Distributions 
2033*33de042dSApple OSS Distributions /*
2034*33de042dSApple OSS Distributions  * Handle SIOCGCONNIDS ioctl for PF_INET domain.
2035*33de042dSApple OSS Distributions  */
2036*33de042dSApple OSS Distributions static int
in_getconnids(struct socket * so,sae_associd_t aid,uint32_t * cnt,user_addr_t cidp)2037*33de042dSApple OSS Distributions in_getconnids(struct socket *so, sae_associd_t aid, uint32_t *cnt,
2038*33de042dSApple OSS Distributions     user_addr_t cidp)
2039*33de042dSApple OSS Distributions {
2040*33de042dSApple OSS Distributions 	struct inpcb *__single inp = sotoinpcb(so);
2041*33de042dSApple OSS Distributions 	sae_connid_t cid;
2042*33de042dSApple OSS Distributions 
2043*33de042dSApple OSS Distributions 	if (inp == NULL || inp->inp_state == INPCB_STATE_DEAD) {
2044*33de042dSApple OSS Distributions 		return EINVAL;
2045*33de042dSApple OSS Distributions 	}
2046*33de042dSApple OSS Distributions 
2047*33de042dSApple OSS Distributions 	if (aid != SAE_ASSOCID_ANY && aid != SAE_ASSOCID_ALL) {
2048*33de042dSApple OSS Distributions 		return EINVAL;
2049*33de042dSApple OSS Distributions 	}
2050*33de042dSApple OSS Distributions 
2051*33de042dSApple OSS Distributions 	/* if connected, return 1 connection count */
2052*33de042dSApple OSS Distributions 	*cnt = ((so->so_state & SS_ISCONNECTED) ? 1 : 0);
2053*33de042dSApple OSS Distributions 
2054*33de042dSApple OSS Distributions 	/* just asking how many there are? */
2055*33de042dSApple OSS Distributions 	if (cidp == USER_ADDR_NULL) {
2056*33de042dSApple OSS Distributions 		return 0;
2057*33de042dSApple OSS Distributions 	}
2058*33de042dSApple OSS Distributions 
2059*33de042dSApple OSS Distributions 	/* if INPCB is connected, assign it connid 1 */
2060*33de042dSApple OSS Distributions 	cid = ((*cnt != 0) ? 1 : SAE_CONNID_ANY);
2061*33de042dSApple OSS Distributions 
2062*33de042dSApple OSS Distributions 	return copyout(&cid, cidp, sizeof(cid));
2063*33de042dSApple OSS Distributions }
2064*33de042dSApple OSS Distributions 
2065*33de042dSApple OSS Distributions /*
2066*33de042dSApple OSS Distributions  * Handle SIOCGCONNINFO ioctl for PF_INET domain.
2067*33de042dSApple OSS Distributions  */
2068*33de042dSApple OSS Distributions int
in_getconninfo(struct socket * so,sae_connid_t cid,uint32_t * flags,uint32_t * ifindex,int32_t * soerror,user_addr_t src,socklen_t * src_len,user_addr_t dst,socklen_t * dst_len,uint32_t * aux_type,user_addr_t aux_data,uint32_t * aux_len)2069*33de042dSApple OSS Distributions in_getconninfo(struct socket *so, sae_connid_t cid, uint32_t *flags,
2070*33de042dSApple OSS Distributions     uint32_t *ifindex, int32_t *soerror, user_addr_t src, socklen_t *src_len,
2071*33de042dSApple OSS Distributions     user_addr_t dst, socklen_t *dst_len, uint32_t *aux_type,
2072*33de042dSApple OSS Distributions     user_addr_t aux_data, uint32_t *aux_len)
2073*33de042dSApple OSS Distributions {
2074*33de042dSApple OSS Distributions 	struct inpcb *__single inp = sotoinpcb(so);
2075*33de042dSApple OSS Distributions 	struct sockaddr_in sin;
2076*33de042dSApple OSS Distributions 	struct ifnet *__single ifp = NULL;
2077*33de042dSApple OSS Distributions 	int error = 0;
2078*33de042dSApple OSS Distributions 	u_int32_t copy_len = 0;
2079*33de042dSApple OSS Distributions 
2080*33de042dSApple OSS Distributions 	/*
2081*33de042dSApple OSS Distributions 	 * Don't test for INPCB_STATE_DEAD since this may be called
2082*33de042dSApple OSS Distributions 	 * after SOF_PCBCLEARING is set, e.g. after tcp_close().
2083*33de042dSApple OSS Distributions 	 */
2084*33de042dSApple OSS Distributions 	if (inp == NULL) {
2085*33de042dSApple OSS Distributions 		error = EINVAL;
2086*33de042dSApple OSS Distributions 		goto out;
2087*33de042dSApple OSS Distributions 	}
2088*33de042dSApple OSS Distributions 
2089*33de042dSApple OSS Distributions 	if (cid != SAE_CONNID_ANY && cid != SAE_CONNID_ALL && cid != 1) {
2090*33de042dSApple OSS Distributions 		error = EINVAL;
2091*33de042dSApple OSS Distributions 		goto out;
2092*33de042dSApple OSS Distributions 	}
2093*33de042dSApple OSS Distributions 
2094*33de042dSApple OSS Distributions 	ifp = inp->inp_last_outifp;
2095*33de042dSApple OSS Distributions 	*ifindex = ((ifp != NULL) ? ifp->if_index : 0);
2096*33de042dSApple OSS Distributions 	*soerror = so->so_error;
2097*33de042dSApple OSS Distributions 	*flags = 0;
2098*33de042dSApple OSS Distributions 	if (so->so_state & SS_ISCONNECTED) {
2099*33de042dSApple OSS Distributions 		*flags |= (CIF_CONNECTED | CIF_PREFERRED);
2100*33de042dSApple OSS Distributions 	}
2101*33de042dSApple OSS Distributions 	if (inp->inp_flags & INP_BOUND_IF) {
2102*33de042dSApple OSS Distributions 		*flags |= CIF_BOUND_IF;
2103*33de042dSApple OSS Distributions 	}
2104*33de042dSApple OSS Distributions 	if (!(inp->inp_flags & INP_INADDR_ANY)) {
2105*33de042dSApple OSS Distributions 		*flags |= CIF_BOUND_IP;
2106*33de042dSApple OSS Distributions 	}
2107*33de042dSApple OSS Distributions 	if (!(inp->inp_flags & INP_ANONPORT)) {
2108*33de042dSApple OSS Distributions 		*flags |= CIF_BOUND_PORT;
2109*33de042dSApple OSS Distributions 	}
2110*33de042dSApple OSS Distributions 
2111*33de042dSApple OSS Distributions 	SOCKADDR_ZERO(&sin, sizeof(sin));
2112*33de042dSApple OSS Distributions 	sin.sin_len = sizeof(sin);
2113*33de042dSApple OSS Distributions 	sin.sin_family = AF_INET;
2114*33de042dSApple OSS Distributions 
2115*33de042dSApple OSS Distributions 	/* source address and port */
2116*33de042dSApple OSS Distributions 	sin.sin_port = inp->inp_lport;
2117*33de042dSApple OSS Distributions 	sin.sin_addr.s_addr = inp->inp_laddr.s_addr;
2118*33de042dSApple OSS Distributions 	if (*src_len == 0) {
2119*33de042dSApple OSS Distributions 		*src_len = sin.sin_len;
2120*33de042dSApple OSS Distributions 	} else {
2121*33de042dSApple OSS Distributions 		if (src != USER_ADDR_NULL) {
2122*33de042dSApple OSS Distributions 			copy_len = min(*src_len, sizeof(sin));
2123*33de042dSApple OSS Distributions 			error = copyout(&sin, src, copy_len);
2124*33de042dSApple OSS Distributions 			if (error != 0) {
2125*33de042dSApple OSS Distributions 				goto out;
2126*33de042dSApple OSS Distributions 			}
2127*33de042dSApple OSS Distributions 			*src_len = copy_len;
2128*33de042dSApple OSS Distributions 		}
2129*33de042dSApple OSS Distributions 	}
2130*33de042dSApple OSS Distributions 
2131*33de042dSApple OSS Distributions 	/* destination address and port */
2132*33de042dSApple OSS Distributions 	sin.sin_port = inp->inp_fport;
2133*33de042dSApple OSS Distributions 	sin.sin_addr.s_addr = inp->inp_faddr.s_addr;
2134*33de042dSApple OSS Distributions 	if (*dst_len == 0) {
2135*33de042dSApple OSS Distributions 		*dst_len = sin.sin_len;
2136*33de042dSApple OSS Distributions 	} else {
2137*33de042dSApple OSS Distributions 		if (dst != USER_ADDR_NULL) {
2138*33de042dSApple OSS Distributions 			copy_len = min(*dst_len, sizeof(sin));
2139*33de042dSApple OSS Distributions 			error = copyout(&sin, dst, copy_len);
2140*33de042dSApple OSS Distributions 			if (error != 0) {
2141*33de042dSApple OSS Distributions 				goto out;
2142*33de042dSApple OSS Distributions 			}
2143*33de042dSApple OSS Distributions 			*dst_len = copy_len;
2144*33de042dSApple OSS Distributions 		}
2145*33de042dSApple OSS Distributions 	}
2146*33de042dSApple OSS Distributions 
2147*33de042dSApple OSS Distributions 	if (SOCK_PROTO(so) == IPPROTO_TCP) {
2148*33de042dSApple OSS Distributions 		struct conninfo_tcp tcp_ci;
2149*33de042dSApple OSS Distributions 
2150*33de042dSApple OSS Distributions 		*aux_type = CIAUX_TCP;
2151*33de042dSApple OSS Distributions 		if (*aux_len == 0) {
2152*33de042dSApple OSS Distributions 			*aux_len = sizeof(tcp_ci);
2153*33de042dSApple OSS Distributions 		} else {
2154*33de042dSApple OSS Distributions 			if (aux_data != USER_ADDR_NULL) {
2155*33de042dSApple OSS Distributions 				copy_len = min(*aux_len, sizeof(tcp_ci));
2156*33de042dSApple OSS Distributions 				bzero(&tcp_ci, sizeof(tcp_ci));
2157*33de042dSApple OSS Distributions 				tcp_getconninfo(so, &tcp_ci);
2158*33de042dSApple OSS Distributions 				error = copyout(&tcp_ci, aux_data, copy_len);
2159*33de042dSApple OSS Distributions 				if (error != 0) {
2160*33de042dSApple OSS Distributions 					goto out;
2161*33de042dSApple OSS Distributions 				}
2162*33de042dSApple OSS Distributions 				*aux_len = copy_len;
2163*33de042dSApple OSS Distributions 			}
2164*33de042dSApple OSS Distributions 		}
2165*33de042dSApple OSS Distributions 	} else {
2166*33de042dSApple OSS Distributions 		*aux_type = 0;
2167*33de042dSApple OSS Distributions 		*aux_len = 0;
2168*33de042dSApple OSS Distributions 	}
2169*33de042dSApple OSS Distributions 
2170*33de042dSApple OSS Distributions out:
2171*33de042dSApple OSS Distributions 	return error;
2172*33de042dSApple OSS Distributions }
2173*33de042dSApple OSS Distributions 
2174*33de042dSApple OSS Distributions struct in_ifaddr*
inifa_ifpwithflag(struct ifnet * ifp,uint32_t flag)2175*33de042dSApple OSS Distributions inifa_ifpwithflag(struct ifnet * ifp, uint32_t flag)
2176*33de042dSApple OSS Distributions {
2177*33de042dSApple OSS Distributions 	struct ifaddr *__single ifa;
2178*33de042dSApple OSS Distributions 
2179*33de042dSApple OSS Distributions 	ifnet_lock_shared(ifp);
2180*33de042dSApple OSS Distributions 	TAILQ_FOREACH(ifa, &ifp->if_addrlist, ifa_link)
2181*33de042dSApple OSS Distributions 	{
2182*33de042dSApple OSS Distributions 		IFA_LOCK_SPIN(ifa);
2183*33de042dSApple OSS Distributions 		if (ifa->ifa_addr->sa_family != AF_INET) {
2184*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
2185*33de042dSApple OSS Distributions 			continue;
2186*33de042dSApple OSS Distributions 		}
2187*33de042dSApple OSS Distributions 		if (((ifatoia(ifa))->ia_flags & flag) == flag) {
2188*33de042dSApple OSS Distributions 			ifa_addref(ifa);
2189*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
2190*33de042dSApple OSS Distributions 			break;
2191*33de042dSApple OSS Distributions 		}
2192*33de042dSApple OSS Distributions 		IFA_UNLOCK(ifa);
2193*33de042dSApple OSS Distributions 	}
2194*33de042dSApple OSS Distributions 	ifnet_lock_done(ifp);
2195*33de042dSApple OSS Distributions 
2196*33de042dSApple OSS Distributions 	return ifatoia(ifa);
2197*33de042dSApple OSS Distributions }
2198*33de042dSApple OSS Distributions 
2199*33de042dSApple OSS Distributions struct in_ifaddr *
inifa_ifpclatv4(struct ifnet * ifp)2200*33de042dSApple OSS Distributions inifa_ifpclatv4(struct ifnet * ifp)
2201*33de042dSApple OSS Distributions {
2202*33de042dSApple OSS Distributions 	struct ifaddr *__single ifa;
2203*33de042dSApple OSS Distributions 
2204*33de042dSApple OSS Distributions 	ifnet_lock_shared(ifp);
2205*33de042dSApple OSS Distributions 	TAILQ_FOREACH(ifa, &ifp->if_addrlist, ifa_link)
2206*33de042dSApple OSS Distributions 	{
2207*33de042dSApple OSS Distributions 		uint32_t addr = 0;
2208*33de042dSApple OSS Distributions 		IFA_LOCK_SPIN(ifa);
2209*33de042dSApple OSS Distributions 		if (ifa->ifa_addr->sa_family != AF_INET) {
2210*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
2211*33de042dSApple OSS Distributions 			continue;
2212*33de042dSApple OSS Distributions 		}
2213*33de042dSApple OSS Distributions 
2214*33de042dSApple OSS Distributions 		addr = ntohl(SIN(ifa->ifa_addr)->sin_addr.s_addr);
2215*33de042dSApple OSS Distributions 		if (!IN_LINKLOCAL(addr) &&
2216*33de042dSApple OSS Distributions 		    !IN_LOOPBACK(addr)) {
2217*33de042dSApple OSS Distributions 			ifa_addref(ifa);
2218*33de042dSApple OSS Distributions 			IFA_UNLOCK(ifa);
2219*33de042dSApple OSS Distributions 			break;
2220*33de042dSApple OSS Distributions 		}
2221*33de042dSApple OSS Distributions 		IFA_UNLOCK(ifa);
2222*33de042dSApple OSS Distributions 	}
2223*33de042dSApple OSS Distributions 	ifnet_lock_done(ifp);
2224*33de042dSApple OSS Distributions 
2225*33de042dSApple OSS Distributions 	return ifatoia(ifa);
2226*33de042dSApple OSS Distributions }
2227*33de042dSApple OSS Distributions 
2228*33de042dSApple OSS Distributions /*
2229*33de042dSApple OSS Distributions  * IPPROTO_xxx.
2230*33de042dSApple OSS Distributions  *
2231*33de042dSApple OSS Distributions  * The switch statement below does nothing at runtime, as it serves as a
2232*33de042dSApple OSS Distributions  * compile time check to ensure that all of the IPPROTO_xxx constants are
2233*33de042dSApple OSS Distributions  * unique.  This works as long as this routine gets updated each time a
2234*33de042dSApple OSS Distributions  * new IPPROTO_xxx constant gets added.
2235*33de042dSApple OSS Distributions  *
2236*33de042dSApple OSS Distributions  * Any failures at compile time indicates duplicated IPPROTO_xxx values.
2237*33de042dSApple OSS Distributions  */
2238*33de042dSApple OSS Distributions static __attribute__((unused)) void
ipproto_cassert(void)2239*33de042dSApple OSS Distributions ipproto_cassert(void)
2240*33de042dSApple OSS Distributions {
2241*33de042dSApple OSS Distributions 	/*
2242*33de042dSApple OSS Distributions 	 * This is equivalent to _CASSERT() and the compiler wouldn't
2243*33de042dSApple OSS Distributions 	 * generate any instructions, thus for compile time only.
2244*33de042dSApple OSS Distributions 	 */
2245*33de042dSApple OSS Distributions 	switch ((u_int16_t)0) {
2246*33de042dSApple OSS Distributions 	/* bsd/netinet/in.h */
2247*33de042dSApple OSS Distributions 	case IPPROTO_IP:
2248*33de042dSApple OSS Distributions //	case IPPROTO_HOPOPTS: // same value as IPPROTO_IP
2249*33de042dSApple OSS Distributions 	case IPPROTO_ICMP:
2250*33de042dSApple OSS Distributions 	case IPPROTO_IGMP:
2251*33de042dSApple OSS Distributions 	case IPPROTO_GGP:
2252*33de042dSApple OSS Distributions 	case IPPROTO_IPV4:
2253*33de042dSApple OSS Distributions 	// #define IPPROTO_IPIP            IPPROTO_IPV4
2254*33de042dSApple OSS Distributions 	case IPPROTO_TCP:
2255*33de042dSApple OSS Distributions 	case IPPROTO_ST:
2256*33de042dSApple OSS Distributions 	case IPPROTO_EGP:
2257*33de042dSApple OSS Distributions 	case IPPROTO_PIGP:
2258*33de042dSApple OSS Distributions 	case IPPROTO_RCCMON:
2259*33de042dSApple OSS Distributions 	case IPPROTO_NVPII:
2260*33de042dSApple OSS Distributions 	case IPPROTO_PUP:
2261*33de042dSApple OSS Distributions 	case IPPROTO_ARGUS:
2262*33de042dSApple OSS Distributions 	case IPPROTO_EMCON:
2263*33de042dSApple OSS Distributions 	case IPPROTO_XNET:
2264*33de042dSApple OSS Distributions 	case IPPROTO_CHAOS:
2265*33de042dSApple OSS Distributions 	case IPPROTO_UDP:
2266*33de042dSApple OSS Distributions 	case IPPROTO_MUX:
2267*33de042dSApple OSS Distributions 	case IPPROTO_MEAS:
2268*33de042dSApple OSS Distributions 	case IPPROTO_HMP:
2269*33de042dSApple OSS Distributions 	case IPPROTO_PRM:
2270*33de042dSApple OSS Distributions 	case IPPROTO_IDP:
2271*33de042dSApple OSS Distributions 	case IPPROTO_TRUNK1:
2272*33de042dSApple OSS Distributions 	case IPPROTO_TRUNK2:
2273*33de042dSApple OSS Distributions 	case IPPROTO_LEAF1:
2274*33de042dSApple OSS Distributions 	case IPPROTO_LEAF2:
2275*33de042dSApple OSS Distributions 	case IPPROTO_RDP:
2276*33de042dSApple OSS Distributions 	case IPPROTO_IRTP:
2277*33de042dSApple OSS Distributions 	case IPPROTO_TP:
2278*33de042dSApple OSS Distributions 	case IPPROTO_BLT:
2279*33de042dSApple OSS Distributions 	case IPPROTO_NSP:
2280*33de042dSApple OSS Distributions 	case IPPROTO_INP:
2281*33de042dSApple OSS Distributions 	case IPPROTO_SEP:
2282*33de042dSApple OSS Distributions 	case IPPROTO_3PC:
2283*33de042dSApple OSS Distributions 	case IPPROTO_IDPR:
2284*33de042dSApple OSS Distributions 	case IPPROTO_XTP:
2285*33de042dSApple OSS Distributions 	case IPPROTO_DDP:
2286*33de042dSApple OSS Distributions 	case IPPROTO_CMTP:
2287*33de042dSApple OSS Distributions 	case IPPROTO_TPXX:
2288*33de042dSApple OSS Distributions 	case IPPROTO_IL:
2289*33de042dSApple OSS Distributions 	case IPPROTO_IPV6:
2290*33de042dSApple OSS Distributions 	case IPPROTO_SDRP:
2291*33de042dSApple OSS Distributions 	case IPPROTO_ROUTING:
2292*33de042dSApple OSS Distributions 	case IPPROTO_FRAGMENT:
2293*33de042dSApple OSS Distributions 	case IPPROTO_IDRP:
2294*33de042dSApple OSS Distributions 	case IPPROTO_RSVP:
2295*33de042dSApple OSS Distributions 	case IPPROTO_GRE:
2296*33de042dSApple OSS Distributions 	case IPPROTO_MHRP:
2297*33de042dSApple OSS Distributions 	case IPPROTO_BHA:
2298*33de042dSApple OSS Distributions 	case IPPROTO_ESP:
2299*33de042dSApple OSS Distributions 	case IPPROTO_AH:
2300*33de042dSApple OSS Distributions 	case IPPROTO_INLSP:
2301*33de042dSApple OSS Distributions 	case IPPROTO_SWIPE:
2302*33de042dSApple OSS Distributions 	case IPPROTO_NHRP:
2303*33de042dSApple OSS Distributions 	case IPPROTO_ICMPV6:
2304*33de042dSApple OSS Distributions 	case IPPROTO_NONE:
2305*33de042dSApple OSS Distributions 	case IPPROTO_DSTOPTS:
2306*33de042dSApple OSS Distributions 	case IPPROTO_AHIP:
2307*33de042dSApple OSS Distributions 	case IPPROTO_CFTP:
2308*33de042dSApple OSS Distributions 	case IPPROTO_HELLO:
2309*33de042dSApple OSS Distributions 	case IPPROTO_SATEXPAK:
2310*33de042dSApple OSS Distributions 	case IPPROTO_KRYPTOLAN:
2311*33de042dSApple OSS Distributions 	case IPPROTO_RVD:
2312*33de042dSApple OSS Distributions 	case IPPROTO_IPPC:
2313*33de042dSApple OSS Distributions 	case IPPROTO_ADFS:
2314*33de042dSApple OSS Distributions 	case IPPROTO_SATMON:
2315*33de042dSApple OSS Distributions 	case IPPROTO_VISA:
2316*33de042dSApple OSS Distributions 	case IPPROTO_IPCV:
2317*33de042dSApple OSS Distributions 	case IPPROTO_CPNX:
2318*33de042dSApple OSS Distributions 	case IPPROTO_CPHB:
2319*33de042dSApple OSS Distributions 	case IPPROTO_WSN:
2320*33de042dSApple OSS Distributions 	case IPPROTO_PVP:
2321*33de042dSApple OSS Distributions 	case IPPROTO_BRSATMON:
2322*33de042dSApple OSS Distributions 	case IPPROTO_ND:
2323*33de042dSApple OSS Distributions 	case IPPROTO_WBMON:
2324*33de042dSApple OSS Distributions 	case IPPROTO_WBEXPAK:
2325*33de042dSApple OSS Distributions 	case IPPROTO_EON:
2326*33de042dSApple OSS Distributions 	case IPPROTO_VMTP:
2327*33de042dSApple OSS Distributions 	case IPPROTO_SVMTP:
2328*33de042dSApple OSS Distributions 	case IPPROTO_VINES:
2329*33de042dSApple OSS Distributions 	case IPPROTO_TTP:
2330*33de042dSApple OSS Distributions 	case IPPROTO_IGP:
2331*33de042dSApple OSS Distributions 	case IPPROTO_DGP:
2332*33de042dSApple OSS Distributions 	case IPPROTO_TCF:
2333*33de042dSApple OSS Distributions 	case IPPROTO_IGRP:
2334*33de042dSApple OSS Distributions 	case IPPROTO_OSPFIGP:
2335*33de042dSApple OSS Distributions 	case IPPROTO_SRPC:
2336*33de042dSApple OSS Distributions 	case IPPROTO_LARP:
2337*33de042dSApple OSS Distributions 	case IPPROTO_MTP:
2338*33de042dSApple OSS Distributions 	case IPPROTO_AX25:
2339*33de042dSApple OSS Distributions 	case IPPROTO_IPEIP:
2340*33de042dSApple OSS Distributions 	case IPPROTO_MICP:
2341*33de042dSApple OSS Distributions 	case IPPROTO_SCCSP:
2342*33de042dSApple OSS Distributions 	case IPPROTO_ETHERIP:
2343*33de042dSApple OSS Distributions 	case IPPROTO_ENCAP:
2344*33de042dSApple OSS Distributions 	case IPPROTO_APES:
2345*33de042dSApple OSS Distributions 	case IPPROTO_GMTP:
2346*33de042dSApple OSS Distributions 	case IPPROTO_PIM:
2347*33de042dSApple OSS Distributions 	case IPPROTO_IPCOMP:
2348*33de042dSApple OSS Distributions 	case IPPROTO_PGM:
2349*33de042dSApple OSS Distributions 	case IPPROTO_SCTP:
2350*33de042dSApple OSS Distributions 	case IPPROTO_DIVERT:
2351*33de042dSApple OSS Distributions 	case IPPROTO_RAW:
2352*33de042dSApple OSS Distributions 	case IPPROTO_MAX:
2353*33de042dSApple OSS Distributions 	case IPPROTO_DONE:
2354*33de042dSApple OSS Distributions 
2355*33de042dSApple OSS Distributions 	/* bsd/netinet/in_private.h */
2356*33de042dSApple OSS Distributions 	case IPPROTO_QUIC:
2357*33de042dSApple OSS Distributions 		;
2358*33de042dSApple OSS Distributions 	}
2359*33de042dSApple OSS Distributions }
2360*33de042dSApple OSS Distributions 
2361*33de042dSApple OSS Distributions static __attribute__((unused)) void
ipsockopt_cassert(void)2362*33de042dSApple OSS Distributions ipsockopt_cassert(void)
2363*33de042dSApple OSS Distributions {
2364*33de042dSApple OSS Distributions 	switch ((int)0) {
2365*33de042dSApple OSS Distributions 	case 0:
2366*33de042dSApple OSS Distributions 
2367*33de042dSApple OSS Distributions 	/* bsd/netinet/in.h */
2368*33de042dSApple OSS Distributions 	case IP_OPTIONS:
2369*33de042dSApple OSS Distributions 	case IP_HDRINCL:
2370*33de042dSApple OSS Distributions 	case IP_TOS:
2371*33de042dSApple OSS Distributions 	case IP_TTL:
2372*33de042dSApple OSS Distributions 	case IP_RECVOPTS:
2373*33de042dSApple OSS Distributions 	case IP_RECVRETOPTS:
2374*33de042dSApple OSS Distributions 	case IP_RECVDSTADDR:
2375*33de042dSApple OSS Distributions 	case IP_RETOPTS:
2376*33de042dSApple OSS Distributions 	case IP_MULTICAST_IF:
2377*33de042dSApple OSS Distributions 	case IP_MULTICAST_TTL:
2378*33de042dSApple OSS Distributions 	case IP_MULTICAST_LOOP:
2379*33de042dSApple OSS Distributions 	case IP_ADD_MEMBERSHIP:
2380*33de042dSApple OSS Distributions 	case IP_DROP_MEMBERSHIP:
2381*33de042dSApple OSS Distributions 	case IP_MULTICAST_VIF:
2382*33de042dSApple OSS Distributions 	case IP_RSVP_ON:
2383*33de042dSApple OSS Distributions 	case IP_RSVP_OFF:
2384*33de042dSApple OSS Distributions 	case IP_RSVP_VIF_ON:
2385*33de042dSApple OSS Distributions 	case IP_RSVP_VIF_OFF:
2386*33de042dSApple OSS Distributions 	case IP_PORTRANGE:
2387*33de042dSApple OSS Distributions 	case IP_RECVIF:
2388*33de042dSApple OSS Distributions 	case IP_IPSEC_POLICY:
2389*33de042dSApple OSS Distributions 	case IP_FAITH:
2390*33de042dSApple OSS Distributions #ifdef __APPLE__
2391*33de042dSApple OSS Distributions 	case IP_STRIPHDR:
2392*33de042dSApple OSS Distributions #endif
2393*33de042dSApple OSS Distributions 	case IP_RECVTTL:
2394*33de042dSApple OSS Distributions 	case IP_BOUND_IF:
2395*33de042dSApple OSS Distributions 	case IP_PKTINFO:
2396*33de042dSApple OSS Distributions // #define IP_RECVPKTINFO          IP_PKTINFO
2397*33de042dSApple OSS Distributions 	case IP_RECVTOS:
2398*33de042dSApple OSS Distributions 	case IP_DONTFRAG:
2399*33de042dSApple OSS Distributions 	case IP_FW_ADD:
2400*33de042dSApple OSS Distributions 	case IP_FW_DEL:
2401*33de042dSApple OSS Distributions 	case IP_FW_FLUSH:
2402*33de042dSApple OSS Distributions 	case IP_FW_ZERO:
2403*33de042dSApple OSS Distributions 	case IP_FW_GET:
2404*33de042dSApple OSS Distributions 	case IP_FW_RESETLOG:
2405*33de042dSApple OSS Distributions 	case IP_OLD_FW_ADD:
2406*33de042dSApple OSS Distributions 	case IP_OLD_FW_DEL:
2407*33de042dSApple OSS Distributions 	case IP_OLD_FW_FLUSH:
2408*33de042dSApple OSS Distributions 	case IP_OLD_FW_ZERO:
2409*33de042dSApple OSS Distributions 	case IP_OLD_FW_GET:
2410*33de042dSApple OSS Distributions 	case IP_NAT__XXX:
2411*33de042dSApple OSS Distributions 	case IP_OLD_FW_RESETLOG:
2412*33de042dSApple OSS Distributions 	case IP_DUMMYNET_CONFIGURE:
2413*33de042dSApple OSS Distributions 	case IP_DUMMYNET_DEL:
2414*33de042dSApple OSS Distributions 	case IP_DUMMYNET_FLUSH:
2415*33de042dSApple OSS Distributions 	case IP_DUMMYNET_GET:
2416*33de042dSApple OSS Distributions 	case IP_TRAFFIC_MGT_BACKGROUND:
2417*33de042dSApple OSS Distributions 	case IP_MULTICAST_IFINDEX:
2418*33de042dSApple OSS Distributions 	case IP_ADD_SOURCE_MEMBERSHIP:
2419*33de042dSApple OSS Distributions 	case IP_DROP_SOURCE_MEMBERSHIP:
2420*33de042dSApple OSS Distributions 	case IP_BLOCK_SOURCE:
2421*33de042dSApple OSS Distributions 	case IP_UNBLOCK_SOURCE:
2422*33de042dSApple OSS Distributions 	case IP_MSFILTER:
2423*33de042dSApple OSS Distributions 	case MCAST_JOIN_GROUP:
2424*33de042dSApple OSS Distributions 	case MCAST_LEAVE_GROUP:
2425*33de042dSApple OSS Distributions 	case MCAST_JOIN_SOURCE_GROUP:
2426*33de042dSApple OSS Distributions 	case MCAST_LEAVE_SOURCE_GROUP:
2427*33de042dSApple OSS Distributions 	case MCAST_BLOCK_SOURCE:
2428*33de042dSApple OSS Distributions 	case MCAST_UNBLOCK_SOURCE:
2429*33de042dSApple OSS Distributions 
2430*33de042dSApple OSS Distributions 	/* bsd/netinet/in_private.h */
2431*33de042dSApple OSS Distributions 	case IP_NO_IFT_CELLULAR:
2432*33de042dSApple OSS Distributions // #define IP_NO_IFT_PDP           IP_NO_IFT_CELLULAR /* deprecated */
2433*33de042dSApple OSS Distributions 	case IP_OUT_IF:
2434*33de042dSApple OSS Distributions 		;
2435*33de042dSApple OSS Distributions 	}
2436*33de042dSApple OSS Distributions }
2437