xref: /xnu-11417.140.69/bsd/netinet/igmp.c (revision 43a90889846e00bfb5cf1d255cdc0a701a1e05a4)
1*43a90889SApple OSS Distributions /*
2*43a90889SApple OSS Distributions  * Copyright (c) 2000-2020 Apple Inc. All rights reserved.
3*43a90889SApple OSS Distributions  *
4*43a90889SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*43a90889SApple OSS Distributions  *
6*43a90889SApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*43a90889SApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*43a90889SApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*43a90889SApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*43a90889SApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*43a90889SApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*43a90889SApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*43a90889SApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*43a90889SApple OSS Distributions  *
15*43a90889SApple OSS Distributions  * Please obtain a copy of the License at
16*43a90889SApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*43a90889SApple OSS Distributions  *
18*43a90889SApple OSS Distributions  * The Original Code and all software distributed under the License are
19*43a90889SApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*43a90889SApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*43a90889SApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*43a90889SApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*43a90889SApple OSS Distributions  * Please see the License for the specific language governing rights and
24*43a90889SApple OSS Distributions  * limitations under the License.
25*43a90889SApple OSS Distributions  *
26*43a90889SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*43a90889SApple OSS Distributions  */
28*43a90889SApple OSS Distributions /*-
29*43a90889SApple OSS Distributions  * Copyright (c) 2007-2009 Bruce Simpson.
30*43a90889SApple OSS Distributions  * Copyright (c) 1988 Stephen Deering.
31*43a90889SApple OSS Distributions  * Copyright (c) 1992, 1993
32*43a90889SApple OSS Distributions  *	The Regents of the University of California.  All rights reserved.
33*43a90889SApple OSS Distributions  *
34*43a90889SApple OSS Distributions  * This code is derived from software contributed to Berkeley by
35*43a90889SApple OSS Distributions  * Stephen Deering of Stanford University.
36*43a90889SApple OSS Distributions  *
37*43a90889SApple OSS Distributions  * Redistribution and use in source and binary forms, with or without
38*43a90889SApple OSS Distributions  * modification, are permitted provided that the following conditions
39*43a90889SApple OSS Distributions  * are met:
40*43a90889SApple OSS Distributions  * 1. Redistributions of source code must retain the above copyright
41*43a90889SApple OSS Distributions  *    notice, this list of conditions and the following disclaimer.
42*43a90889SApple OSS Distributions  * 2. Redistributions in binary form must reproduce the above copyright
43*43a90889SApple OSS Distributions  *    notice, this list of conditions and the following disclaimer in the
44*43a90889SApple OSS Distributions  *    documentation and/or other materials provided with the distribution.
45*43a90889SApple OSS Distributions  * 3. All advertising materials mentioning features or use of this software
46*43a90889SApple OSS Distributions  *    must display the following acknowledgement:
47*43a90889SApple OSS Distributions  *	This product includes software developed by the University of
48*43a90889SApple OSS Distributions  *	California, Berkeley and its contributors.
49*43a90889SApple OSS Distributions  * 4. Neither the name of the University nor the names of its contributors
50*43a90889SApple OSS Distributions  *    may be used to endorse or promote products derived from this software
51*43a90889SApple OSS Distributions  *    without specific prior written permission.
52*43a90889SApple OSS Distributions  *
53*43a90889SApple OSS Distributions  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
54*43a90889SApple OSS Distributions  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
55*43a90889SApple OSS Distributions  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
56*43a90889SApple OSS Distributions  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
57*43a90889SApple OSS Distributions  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
58*43a90889SApple OSS Distributions  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
59*43a90889SApple OSS Distributions  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
60*43a90889SApple OSS Distributions  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
61*43a90889SApple OSS Distributions  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
62*43a90889SApple OSS Distributions  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
63*43a90889SApple OSS Distributions  * SUCH DAMAGE.
64*43a90889SApple OSS Distributions  *
65*43a90889SApple OSS Distributions  *	@(#)igmp.c	8.1 (Berkeley) 7/19/93
66*43a90889SApple OSS Distributions  */
67*43a90889SApple OSS Distributions /*
68*43a90889SApple OSS Distributions  * NOTICE: This file was modified by SPARTA, Inc. in 2005 to introduce
69*43a90889SApple OSS Distributions  * support for mandatory and extensible security protections.  This notice
70*43a90889SApple OSS Distributions  * is included in support of clause 2.2 (b) of the Apple Public License,
71*43a90889SApple OSS Distributions  * Version 2.0.
72*43a90889SApple OSS Distributions  */
73*43a90889SApple OSS Distributions 
74*43a90889SApple OSS Distributions /*
75*43a90889SApple OSS Distributions  * Internet Group Management Protocol (IGMP) routines.
76*43a90889SApple OSS Distributions  * [RFC1112, RFC2236, RFC3376]
77*43a90889SApple OSS Distributions  *
78*43a90889SApple OSS Distributions  * Written by Steve Deering, Stanford, May 1988.
79*43a90889SApple OSS Distributions  * Modified by Rosen Sharma, Stanford, Aug 1994.
80*43a90889SApple OSS Distributions  * Modified by Bill Fenner, Xerox PARC, Feb 1995.
81*43a90889SApple OSS Distributions  * Modified to fully comply to IGMPv2 by Bill Fenner, Oct 1995.
82*43a90889SApple OSS Distributions  * Significantly rewritten for IGMPv3, VIMAGE, and SMP by Bruce Simpson.
83*43a90889SApple OSS Distributions  *
84*43a90889SApple OSS Distributions  * MULTICAST Revision: 3.5.1.4
85*43a90889SApple OSS Distributions  */
86*43a90889SApple OSS Distributions 
87*43a90889SApple OSS Distributions #include <sys/cdefs.h>
88*43a90889SApple OSS Distributions 
89*43a90889SApple OSS Distributions #include <sys/param.h>
90*43a90889SApple OSS Distributions #include <sys/systm.h>
91*43a90889SApple OSS Distributions #include <sys/malloc.h>
92*43a90889SApple OSS Distributions #include <sys/mbuf.h>
93*43a90889SApple OSS Distributions #include <sys/socket.h>
94*43a90889SApple OSS Distributions #include <sys/protosw.h>
95*43a90889SApple OSS Distributions #include <sys/kernel.h>
96*43a90889SApple OSS Distributions #include <sys/mcache.h>
97*43a90889SApple OSS Distributions 
98*43a90889SApple OSS Distributions #include <libkern/libkern.h>
99*43a90889SApple OSS Distributions #include <kern/zalloc.h>
100*43a90889SApple OSS Distributions 
101*43a90889SApple OSS Distributions #include <net/if.h>
102*43a90889SApple OSS Distributions #include <net/route.h>
103*43a90889SApple OSS Distributions #include <net/net_sysctl.h>
104*43a90889SApple OSS Distributions 
105*43a90889SApple OSS Distributions #include <netinet/in.h>
106*43a90889SApple OSS Distributions #include <netinet/in_var.h>
107*43a90889SApple OSS Distributions #include <netinet/in_systm.h>
108*43a90889SApple OSS Distributions #include <netinet/ip.h>
109*43a90889SApple OSS Distributions #include <netinet/ip_var.h>
110*43a90889SApple OSS Distributions #include <netinet/igmp.h>
111*43a90889SApple OSS Distributions #include <netinet/igmp_var.h>
112*43a90889SApple OSS Distributions #include <netinet/kpi_ipfilter_var.h>
113*43a90889SApple OSS Distributions 
114*43a90889SApple OSS Distributions #include <os/log.h>
115*43a90889SApple OSS Distributions 
116*43a90889SApple OSS Distributions #if SKYWALK
117*43a90889SApple OSS Distributions #include <skywalk/core/skywalk_var.h>
118*43a90889SApple OSS Distributions #endif /* SKYWALK */
119*43a90889SApple OSS Distributions 
120*43a90889SApple OSS Distributions SLIST_HEAD(igmp_inm_relhead, in_multi);
121*43a90889SApple OSS Distributions 
122*43a90889SApple OSS Distributions static void     igi_initvar(struct igmp_ifinfo *, struct ifnet *, int);
123*43a90889SApple OSS Distributions static struct igmp_ifinfo *igi_alloc(zalloc_flags_t);
124*43a90889SApple OSS Distributions static void     igi_free(struct igmp_ifinfo *);
125*43a90889SApple OSS Distributions static void     igi_delete(const struct ifnet *, struct igmp_inm_relhead *);
126*43a90889SApple OSS Distributions static void     igmp_dispatch_queue(struct igmp_ifinfo *, struct ifqueue *,
127*43a90889SApple OSS Distributions     int, const int);
128*43a90889SApple OSS Distributions static void     igmp_final_leave(struct in_multi *, struct igmp_ifinfo *,
129*43a90889SApple OSS Distributions     struct igmp_tparams *);
130*43a90889SApple OSS Distributions static int      igmp_handle_state_change(struct in_multi *,
131*43a90889SApple OSS Distributions     struct igmp_ifinfo *, struct igmp_tparams *);
132*43a90889SApple OSS Distributions static int      igmp_initial_join(struct in_multi *, struct igmp_ifinfo *,
133*43a90889SApple OSS Distributions     struct igmp_tparams *);
134*43a90889SApple OSS Distributions static int      igmp_input_v1_query(struct ifnet *, const struct ip *,
135*43a90889SApple OSS Distributions     const struct igmp *);
136*43a90889SApple OSS Distributions static int      igmp_input_v2_query(struct ifnet *, const struct ip *,
137*43a90889SApple OSS Distributions     const struct igmp *);
138*43a90889SApple OSS Distributions static int      igmp_input_v3_query(struct ifnet *, const struct ip *,
139*43a90889SApple OSS Distributions     /*const*/ struct igmpv3 *__indexable);
140*43a90889SApple OSS Distributions static int      igmp_input_v3_group_query(struct in_multi *,
141*43a90889SApple OSS Distributions     int, /*const*/ struct igmpv3 *__indexable);
142*43a90889SApple OSS Distributions static int      igmp_input_v1_report(struct ifnet *, struct mbuf *,
143*43a90889SApple OSS Distributions     /*const*/ struct ip *, /*const*/ struct igmp *);
144*43a90889SApple OSS Distributions static int      igmp_input_v2_report(struct ifnet *, struct mbuf *,
145*43a90889SApple OSS Distributions     /*const*/ struct ip *, /*const*/ struct igmp *);
146*43a90889SApple OSS Distributions static void     igmp_sendpkt(struct mbuf *);
147*43a90889SApple OSS Distributions static __inline__ int   igmp_isgroupreported(const struct in_addr);
148*43a90889SApple OSS Distributions static struct mbuf *igmp_ra_alloc(void);
149*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
150*43a90889SApple OSS Distributions static const char *igmp_rec_type_to_str(const int);
151*43a90889SApple OSS Distributions #endif
152*43a90889SApple OSS Distributions static uint32_t igmp_set_version(struct igmp_ifinfo *, const int);
153*43a90889SApple OSS Distributions static void     igmp_append_relq(struct igmp_ifinfo *, struct in_multi *);
154*43a90889SApple OSS Distributions static void     igmp_flush_relq(struct igmp_ifinfo *,
155*43a90889SApple OSS Distributions     struct igmp_inm_relhead *);
156*43a90889SApple OSS Distributions static int      igmp_v1v2_queue_report(struct in_multi *, const int);
157*43a90889SApple OSS Distributions static void     igmp_v1v2_process_group_timer(struct in_multi *, const int);
158*43a90889SApple OSS Distributions static void     igmp_v1v2_process_querier_timers(struct igmp_ifinfo *);
159*43a90889SApple OSS Distributions static uint32_t igmp_v2_update_group(struct in_multi *, const int);
160*43a90889SApple OSS Distributions static void     igmp_v3_cancel_link_timers(struct igmp_ifinfo *);
161*43a90889SApple OSS Distributions static uint32_t igmp_v3_dispatch_general_query(struct igmp_ifinfo *);
162*43a90889SApple OSS Distributions static struct mbuf *
163*43a90889SApple OSS Distributions igmp_v3_encap_report(struct ifnet *, struct mbuf *);
164*43a90889SApple OSS Distributions static int      igmp_v3_enqueue_group_record(struct ifqueue *,
165*43a90889SApple OSS Distributions     struct in_multi *, const int, const int, const int);
166*43a90889SApple OSS Distributions static int      igmp_v3_enqueue_filter_change(struct ifqueue *,
167*43a90889SApple OSS Distributions     struct in_multi *);
168*43a90889SApple OSS Distributions static void     igmp_v3_process_group_timers(struct igmp_ifinfo *,
169*43a90889SApple OSS Distributions     struct ifqueue *, struct ifqueue *, struct in_multi *,
170*43a90889SApple OSS Distributions     const unsigned int);
171*43a90889SApple OSS Distributions static int      igmp_v3_merge_state_changes(struct in_multi *,
172*43a90889SApple OSS Distributions     struct ifqueue *);
173*43a90889SApple OSS Distributions static void     igmp_v3_suppress_group_record(struct in_multi *);
174*43a90889SApple OSS Distributions static int      sysctl_igmp_ifinfo SYSCTL_HANDLER_ARGS;
175*43a90889SApple OSS Distributions static int      sysctl_igmp_gsr SYSCTL_HANDLER_ARGS;
176*43a90889SApple OSS Distributions static int      sysctl_igmp_default_version SYSCTL_HANDLER_ARGS;
177*43a90889SApple OSS Distributions 
178*43a90889SApple OSS Distributions static const uint32_t igmp_timeout_delay = 1000; /* in milliseconds */
179*43a90889SApple OSS Distributions static const uint32_t igmp_timeout_leeway = 500; /* in millseconds  */
180*43a90889SApple OSS Distributions static bool igmp_timeout_run;            /* IGMP timer is scheduled to run */
181*43a90889SApple OSS Distributions static bool igmp_fast_timeout_run;       /* IGMP fast timer is scheduled to run */
182*43a90889SApple OSS Distributions static void igmp_timeout(thread_call_param_t, thread_call_param_t);
183*43a90889SApple OSS Distributions static void igmp_sched_timeout(void);
184*43a90889SApple OSS Distributions static void igmp_sched_fast_timeout(void);
185*43a90889SApple OSS Distributions 
186*43a90889SApple OSS Distributions static struct mbuf *m_raopt;            /* Router Alert option */
187*43a90889SApple OSS Distributions 
188*43a90889SApple OSS Distributions static int querier_present_timers_running;      /* IGMPv1/v2 older version
189*43a90889SApple OSS Distributions                                                  * querier present */
190*43a90889SApple OSS Distributions static int interface_timers_running;            /* IGMPv3 general
191*43a90889SApple OSS Distributions                                                  * query response */
192*43a90889SApple OSS Distributions static int state_change_timers_running;         /* IGMPv3 state-change
193*43a90889SApple OSS Distributions                                                  * retransmit */
194*43a90889SApple OSS Distributions static int current_state_timers_running;        /* IGMPv1/v2 host
195*43a90889SApple OSS Distributions                                                  * report; IGMPv3 g/sg
196*43a90889SApple OSS Distributions                                                  * query response */
197*43a90889SApple OSS Distributions 
198*43a90889SApple OSS Distributions /*
199*43a90889SApple OSS Distributions  * Subsystem lock macros.
200*43a90889SApple OSS Distributions  */
201*43a90889SApple OSS Distributions #define IGMP_LOCK()                     \
202*43a90889SApple OSS Distributions 	lck_mtx_lock(&igmp_mtx)
203*43a90889SApple OSS Distributions #define IGMP_LOCK_ASSERT_HELD()         \
204*43a90889SApple OSS Distributions 	LCK_MTX_ASSERT(&igmp_mtx, LCK_MTX_ASSERT_OWNED)
205*43a90889SApple OSS Distributions #define IGMP_LOCK_ASSERT_NOTHELD()      \
206*43a90889SApple OSS Distributions 	LCK_MTX_ASSERT(&igmp_mtx, LCK_MTX_ASSERT_NOTOWNED)
207*43a90889SApple OSS Distributions #define IGMP_UNLOCK()                   \
208*43a90889SApple OSS Distributions 	lck_mtx_unlock(&igmp_mtx)
209*43a90889SApple OSS Distributions 
210*43a90889SApple OSS Distributions static LIST_HEAD(, igmp_ifinfo) igi_head;
211*43a90889SApple OSS Distributions static struct igmpstat_v3 igmpstat_v3 = {
212*43a90889SApple OSS Distributions 	.igps_version = IGPS_VERSION_3,
213*43a90889SApple OSS Distributions 	.igps_len = sizeof(struct igmpstat_v3),
214*43a90889SApple OSS Distributions };
215*43a90889SApple OSS Distributions static struct igmpstat igmpstat; /* old IGMPv2 stats structure */
216*43a90889SApple OSS Distributions static struct timeval igmp_gsrdelay = {.tv_sec = 10, .tv_usec = 0};
217*43a90889SApple OSS Distributions 
218*43a90889SApple OSS Distributions static int igmp_recvifkludge = 1;
219*43a90889SApple OSS Distributions static int igmp_sendra = 1;
220*43a90889SApple OSS Distributions static int igmp_sendlocal = 1;
221*43a90889SApple OSS Distributions static int igmp_v1enable = 1;
222*43a90889SApple OSS Distributions static int igmp_v2enable = 1;
223*43a90889SApple OSS Distributions static int igmp_legacysupp = 0;
224*43a90889SApple OSS Distributions static int igmp_default_version = IGMP_VERSION_3;
225*43a90889SApple OSS Distributions 
226*43a90889SApple OSS Distributions SYSCTL_STRUCT(_net_inet_igmp, IGMPCTL_STATS, stats, CTLFLAG_RD | CTLFLAG_LOCKED,
227*43a90889SApple OSS Distributions     &igmpstat, igmpstat, "");
228*43a90889SApple OSS Distributions SYSCTL_STRUCT(_net_inet_igmp, OID_AUTO, v3stats,
229*43a90889SApple OSS Distributions     CTLFLAG_RD | CTLFLAG_LOCKED, &igmpstat_v3, igmpstat_v3, "");
230*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO, recvifkludge, CTLFLAG_RW | CTLFLAG_LOCKED,
231*43a90889SApple OSS Distributions     &igmp_recvifkludge, 0,
232*43a90889SApple OSS Distributions     "Rewrite IGMPv1/v2 reports from 0.0.0.0 to contain subnet address");
233*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO, sendra, CTLFLAG_RW | CTLFLAG_LOCKED,
234*43a90889SApple OSS Distributions     &igmp_sendra, 0,
235*43a90889SApple OSS Distributions     "Send IP Router Alert option in IGMPv2/v3 messages");
236*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO, sendlocal, CTLFLAG_RW | CTLFLAG_LOCKED,
237*43a90889SApple OSS Distributions     &igmp_sendlocal, 0,
238*43a90889SApple OSS Distributions     "Send IGMP membership reports for 224.0.0.0/24 groups");
239*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO, v1enable, CTLFLAG_RW | CTLFLAG_LOCKED,
240*43a90889SApple OSS Distributions     &igmp_v1enable, 0,
241*43a90889SApple OSS Distributions     "Enable backwards compatibility with IGMPv1");
242*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO, v2enable, CTLFLAG_RW | CTLFLAG_LOCKED,
243*43a90889SApple OSS Distributions     &igmp_v2enable, 0,
244*43a90889SApple OSS Distributions     "Enable backwards compatibility with IGMPv2");
245*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO, legacysupp, CTLFLAG_RW | CTLFLAG_LOCKED,
246*43a90889SApple OSS Distributions     &igmp_legacysupp, 0,
247*43a90889SApple OSS Distributions     "Allow v1/v2 reports to suppress v3 group responses");
248*43a90889SApple OSS Distributions SYSCTL_PROC(_net_inet_igmp, OID_AUTO, default_version,
249*43a90889SApple OSS Distributions     CTLTYPE_INT | CTLFLAG_RW,
250*43a90889SApple OSS Distributions     &igmp_default_version, 0, sysctl_igmp_default_version, "I",
251*43a90889SApple OSS Distributions     "Default version of IGMP to run on each interface");
252*43a90889SApple OSS Distributions SYSCTL_PROC(_net_inet_igmp, OID_AUTO, gsrdelay,
253*43a90889SApple OSS Distributions     CTLTYPE_INT | CTLFLAG_RW,
254*43a90889SApple OSS Distributions     &igmp_gsrdelay.tv_sec, 0, sysctl_igmp_gsr, "I",
255*43a90889SApple OSS Distributions     "Rate limit for IGMPv3 Group-and-Source queries in seconds");
256*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
257*43a90889SApple OSS Distributions int igmp_debug = 0;
258*43a90889SApple OSS Distributions SYSCTL_INT(_net_inet_igmp, OID_AUTO,
259*43a90889SApple OSS Distributions     debug, CTLFLAG_RW | CTLFLAG_LOCKED, &igmp_debug, 0, "");
260*43a90889SApple OSS Distributions #endif
261*43a90889SApple OSS Distributions 
262*43a90889SApple OSS Distributions SYSCTL_NODE(_net_inet_igmp, OID_AUTO, ifinfo, CTLFLAG_RD | CTLFLAG_LOCKED,
263*43a90889SApple OSS Distributions     sysctl_igmp_ifinfo, "Per-interface IGMPv3 state");
264*43a90889SApple OSS Distributions 
265*43a90889SApple OSS Distributions /* Lock group and attribute for igmp_mtx */
266*43a90889SApple OSS Distributions static LCK_ATTR_DECLARE(igmp_mtx_attr, 0, 0);
267*43a90889SApple OSS Distributions static LCK_GRP_DECLARE(igmp_mtx_grp, "igmp_mtx");
268*43a90889SApple OSS Distributions 
269*43a90889SApple OSS Distributions /*
270*43a90889SApple OSS Distributions  * Locking and reference counting:
271*43a90889SApple OSS Distributions  *
272*43a90889SApple OSS Distributions  * igmp_mtx mainly protects igi_head.  In cases where both igmp_mtx and
273*43a90889SApple OSS Distributions  * in_multihead_lock must be held, the former must be acquired first in order
274*43a90889SApple OSS Distributions  * to maintain lock ordering.  It is not a requirement that igmp_mtx be
275*43a90889SApple OSS Distributions  * acquired first before in_multihead_lock, but in case both must be acquired
276*43a90889SApple OSS Distributions  * in succession, the correct lock ordering must be followed.
277*43a90889SApple OSS Distributions  *
278*43a90889SApple OSS Distributions  * Instead of walking the if_multiaddrs list at the interface and returning
279*43a90889SApple OSS Distributions  * the ifma_protospec value of a matching entry, we search the global list
280*43a90889SApple OSS Distributions  * of in_multi records and find it that way; this is done with in_multihead
281*43a90889SApple OSS Distributions  * lock held.  Doing so avoids the race condition issues that many other BSDs
282*43a90889SApple OSS Distributions  * suffer from (therefore in our implementation, ifma_protospec will never be
283*43a90889SApple OSS Distributions  * NULL for as long as the in_multi is valid.)
284*43a90889SApple OSS Distributions  *
285*43a90889SApple OSS Distributions  * The above creates a requirement for the in_multi to stay in in_multihead
286*43a90889SApple OSS Distributions  * list even after the final IGMP leave (in IGMPv3 mode) until no longer needs
287*43a90889SApple OSS Distributions  * be retransmitted (this is not required for IGMPv1/v2.)  In order to handle
288*43a90889SApple OSS Distributions  * this, the request and reference counts of the in_multi are bumped up when
289*43a90889SApple OSS Distributions  * the state changes to IGMP_LEAVING_MEMBER, and later dropped in the timeout
290*43a90889SApple OSS Distributions  * handler.  Each in_multi holds a reference to the underlying igmp_ifinfo.
291*43a90889SApple OSS Distributions  *
292*43a90889SApple OSS Distributions  * Thus, the permitted lock oder is:
293*43a90889SApple OSS Distributions  *
294*43a90889SApple OSS Distributions  *	igmp_mtx, in_multihead_lock, inm_lock, igi_lock
295*43a90889SApple OSS Distributions  *
296*43a90889SApple OSS Distributions  * Any may be taken independently, but if any are held at the same time,
297*43a90889SApple OSS Distributions  * the above lock order must be followed.
298*43a90889SApple OSS Distributions  */
299*43a90889SApple OSS Distributions static LCK_MTX_DECLARE_ATTR(igmp_mtx, &igmp_mtx_grp, &igmp_mtx_attr);
300*43a90889SApple OSS Distributions static int igmp_timers_are_running;
301*43a90889SApple OSS Distributions 
302*43a90889SApple OSS Distributions #define IGMP_ADD_DETACHED_INM(_head, _inm) {                            \
303*43a90889SApple OSS Distributions 	SLIST_INSERT_HEAD(_head, _inm, inm_dtle);                       \
304*43a90889SApple OSS Distributions }
305*43a90889SApple OSS Distributions 
306*43a90889SApple OSS Distributions #define IGMP_REMOVE_DETACHED_INM(_head) {                               \
307*43a90889SApple OSS Distributions 	struct in_multi *_inm, *_inm_tmp;                               \
308*43a90889SApple OSS Distributions 	SLIST_FOREACH_SAFE(_inm, _head, inm_dtle, _inm_tmp) {           \
309*43a90889SApple OSS Distributions 	        SLIST_REMOVE(_head, _inm, in_multi, inm_dtle);          \
310*43a90889SApple OSS Distributions 	        INM_REMREF(_inm);                                       \
311*43a90889SApple OSS Distributions 	}                                                               \
312*43a90889SApple OSS Distributions 	VERIFY(SLIST_EMPTY(_head));                                     \
313*43a90889SApple OSS Distributions }
314*43a90889SApple OSS Distributions 
315*43a90889SApple OSS Distributions static KALLOC_TYPE_DEFINE(igi_zone, struct igmp_ifinfo, NET_KT_DEFAULT);
316*43a90889SApple OSS Distributions 
317*43a90889SApple OSS Distributions /* Store IGMPv3 record count in the module private scratch space */
318*43a90889SApple OSS Distributions #define vt_nrecs        pkt_mpriv.__mpriv_u.__mpriv32[0].__mpriv32_u.__val16[0]
319*43a90889SApple OSS Distributions 
320*43a90889SApple OSS Distributions static __inline void
igmp_save_context(struct mbuf * m,struct ifnet * ifp)321*43a90889SApple OSS Distributions igmp_save_context(struct mbuf *m, struct ifnet *ifp)
322*43a90889SApple OSS Distributions {
323*43a90889SApple OSS Distributions 	m->m_pkthdr.rcvif = ifp;
324*43a90889SApple OSS Distributions }
325*43a90889SApple OSS Distributions 
326*43a90889SApple OSS Distributions static __inline void
igmp_scrub_context(struct mbuf * m)327*43a90889SApple OSS Distributions igmp_scrub_context(struct mbuf *m)
328*43a90889SApple OSS Distributions {
329*43a90889SApple OSS Distributions 	m->m_pkthdr.rcvif = NULL;
330*43a90889SApple OSS Distributions }
331*43a90889SApple OSS Distributions 
332*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
333*43a90889SApple OSS Distributions static __inline const char *
inet_ntop_haddr(in_addr_t haddr,char * buf __counted_by (size),socklen_t size)334*43a90889SApple OSS Distributions inet_ntop_haddr(in_addr_t haddr, char *buf __counted_by(size), socklen_t size)
335*43a90889SApple OSS Distributions {
336*43a90889SApple OSS Distributions 	struct in_addr ia;
337*43a90889SApple OSS Distributions 
338*43a90889SApple OSS Distributions 	ia.s_addr = htonl(haddr);
339*43a90889SApple OSS Distributions 	return inet_ntop(AF_INET, &ia, buf, size);
340*43a90889SApple OSS Distributions }
341*43a90889SApple OSS Distributions #endif
342*43a90889SApple OSS Distributions 
343*43a90889SApple OSS Distributions /*
344*43a90889SApple OSS Distributions  * Restore context from a queued IGMP output chain.
345*43a90889SApple OSS Distributions  * Return saved ifp.
346*43a90889SApple OSS Distributions  */
347*43a90889SApple OSS Distributions static __inline struct ifnet *
igmp_restore_context(struct mbuf * m)348*43a90889SApple OSS Distributions igmp_restore_context(struct mbuf *m)
349*43a90889SApple OSS Distributions {
350*43a90889SApple OSS Distributions 	return m->m_pkthdr.rcvif;
351*43a90889SApple OSS Distributions }
352*43a90889SApple OSS Distributions 
353*43a90889SApple OSS Distributions /*
354*43a90889SApple OSS Distributions  * Retrieve or set default IGMP version.
355*43a90889SApple OSS Distributions  */
356*43a90889SApple OSS Distributions static int
357*43a90889SApple OSS Distributions sysctl_igmp_default_version SYSCTL_HANDLER_ARGS
358*43a90889SApple OSS Distributions {
359*43a90889SApple OSS Distributions #pragma unused(oidp, arg2)
360*43a90889SApple OSS Distributions 	int      error;
361*43a90889SApple OSS Distributions 	int      new;
362*43a90889SApple OSS Distributions 
363*43a90889SApple OSS Distributions 	IGMP_LOCK();
364*43a90889SApple OSS Distributions 
365*43a90889SApple OSS Distributions 	error = SYSCTL_OUT(req, arg1, sizeof(int));
366*43a90889SApple OSS Distributions 	if (error || !req->newptr) {
367*43a90889SApple OSS Distributions 		goto out_locked;
368*43a90889SApple OSS Distributions 	}
369*43a90889SApple OSS Distributions 
370*43a90889SApple OSS Distributions 	new = igmp_default_version;
371*43a90889SApple OSS Distributions 
372*43a90889SApple OSS Distributions 	error = SYSCTL_IN(req, &new, sizeof(int));
373*43a90889SApple OSS Distributions 	if (error) {
374*43a90889SApple OSS Distributions 		goto out_locked;
375*43a90889SApple OSS Distributions 	}
376*43a90889SApple OSS Distributions 
377*43a90889SApple OSS Distributions 	if (new < IGMP_VERSION_1 || new > IGMP_VERSION_3) {
378*43a90889SApple OSS Distributions 		error = EINVAL;
379*43a90889SApple OSS Distributions 		goto out_locked;
380*43a90889SApple OSS Distributions 	}
381*43a90889SApple OSS Distributions 
382*43a90889SApple OSS Distributions 	os_log(OS_LOG_DEFAULT,
383*43a90889SApple OSS Distributions 	    "%s: changed igmp_default_version from %d to %d\n",
384*43a90889SApple OSS Distributions 	    __func__, igmp_default_version, new);
385*43a90889SApple OSS Distributions 
386*43a90889SApple OSS Distributions 	igmp_default_version = new;
387*43a90889SApple OSS Distributions 
388*43a90889SApple OSS Distributions out_locked:
389*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
390*43a90889SApple OSS Distributions 	return error;
391*43a90889SApple OSS Distributions }
392*43a90889SApple OSS Distributions 
393*43a90889SApple OSS Distributions /*
394*43a90889SApple OSS Distributions  * Retrieve or set threshold between group-source queries in seconds.
395*43a90889SApple OSS Distributions  *
396*43a90889SApple OSS Distributions  */
397*43a90889SApple OSS Distributions static int
398*43a90889SApple OSS Distributions sysctl_igmp_gsr SYSCTL_HANDLER_ARGS
399*43a90889SApple OSS Distributions {
400*43a90889SApple OSS Distributions #pragma unused(arg1, arg2)
401*43a90889SApple OSS Distributions 	int error;
402*43a90889SApple OSS Distributions 	int i;
403*43a90889SApple OSS Distributions 
404*43a90889SApple OSS Distributions 	IGMP_LOCK();
405*43a90889SApple OSS Distributions 
406*43a90889SApple OSS Distributions 	i = (int)igmp_gsrdelay.tv_sec;
407*43a90889SApple OSS Distributions 
408*43a90889SApple OSS Distributions 	error = sysctl_handle_int(oidp, &i, 0, req);
409*43a90889SApple OSS Distributions 	if (error || !req->newptr) {
410*43a90889SApple OSS Distributions 		goto out_locked;
411*43a90889SApple OSS Distributions 	}
412*43a90889SApple OSS Distributions 
413*43a90889SApple OSS Distributions 	if (i < -1 || i >= 60) {
414*43a90889SApple OSS Distributions 		error = EINVAL;
415*43a90889SApple OSS Distributions 		goto out_locked;
416*43a90889SApple OSS Distributions 	}
417*43a90889SApple OSS Distributions 
418*43a90889SApple OSS Distributions 	igmp_gsrdelay.tv_sec = i;
419*43a90889SApple OSS Distributions 
420*43a90889SApple OSS Distributions out_locked:
421*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
422*43a90889SApple OSS Distributions 	return error;
423*43a90889SApple OSS Distributions }
424*43a90889SApple OSS Distributions 
425*43a90889SApple OSS Distributions /*
426*43a90889SApple OSS Distributions  * Expose struct igmp_ifinfo to userland, keyed by ifindex.
427*43a90889SApple OSS Distributions  * For use by ifmcstat(8).
428*43a90889SApple OSS Distributions  *
429*43a90889SApple OSS Distributions  */
430*43a90889SApple OSS Distributions static int
431*43a90889SApple OSS Distributions sysctl_igmp_ifinfo SYSCTL_HANDLER_ARGS
432*43a90889SApple OSS Distributions {
433*43a90889SApple OSS Distributions #pragma unused(oidp)
434*43a90889SApple OSS Distributions 	DECLARE_SYSCTL_HANDLER_ARG_ARRAY(int, 1, name, namelen);
435*43a90889SApple OSS Distributions 	int                      error;
436*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
437*43a90889SApple OSS Distributions 	struct igmp_ifinfo      *igi;
438*43a90889SApple OSS Distributions 	struct igmp_ifinfo_u    igi_u;
439*43a90889SApple OSS Distributions 
440*43a90889SApple OSS Distributions 	if (req->newptr != USER_ADDR_NULL) {
441*43a90889SApple OSS Distributions 		return EPERM;
442*43a90889SApple OSS Distributions 	}
443*43a90889SApple OSS Distributions 
444*43a90889SApple OSS Distributions 	IGMP_LOCK();
445*43a90889SApple OSS Distributions 
446*43a90889SApple OSS Distributions 	if (name[0] <= 0 || name[0] > (u_int)if_index) {
447*43a90889SApple OSS Distributions 		error = ENOENT;
448*43a90889SApple OSS Distributions 		goto out_locked;
449*43a90889SApple OSS Distributions 	}
450*43a90889SApple OSS Distributions 
451*43a90889SApple OSS Distributions 	error = ENOENT;
452*43a90889SApple OSS Distributions 
453*43a90889SApple OSS Distributions 	ifnet_head_lock_shared();
454*43a90889SApple OSS Distributions 	ifp = ifindex2ifnet[name[0]];
455*43a90889SApple OSS Distributions 	ifnet_head_done();
456*43a90889SApple OSS Distributions 	if (ifp == NULL) {
457*43a90889SApple OSS Distributions 		goto out_locked;
458*43a90889SApple OSS Distributions 	}
459*43a90889SApple OSS Distributions 
460*43a90889SApple OSS Distributions 	bzero(&igi_u, sizeof(igi_u));
461*43a90889SApple OSS Distributions 
462*43a90889SApple OSS Distributions 	LIST_FOREACH(igi, &igi_head, igi_link) {
463*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
464*43a90889SApple OSS Distributions 		if (ifp != igi->igi_ifp) {
465*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
466*43a90889SApple OSS Distributions 			continue;
467*43a90889SApple OSS Distributions 		}
468*43a90889SApple OSS Distributions 		igi_u.igi_ifindex = igi->igi_ifp->if_index;
469*43a90889SApple OSS Distributions 		igi_u.igi_version = igi->igi_version;
470*43a90889SApple OSS Distributions 		igi_u.igi_v1_timer = igi->igi_v1_timer;
471*43a90889SApple OSS Distributions 		igi_u.igi_v2_timer = igi->igi_v2_timer;
472*43a90889SApple OSS Distributions 		igi_u.igi_v3_timer = igi->igi_v3_timer;
473*43a90889SApple OSS Distributions 		igi_u.igi_flags = igi->igi_flags;
474*43a90889SApple OSS Distributions 		igi_u.igi_rv = igi->igi_rv;
475*43a90889SApple OSS Distributions 		igi_u.igi_qi = igi->igi_qi;
476*43a90889SApple OSS Distributions 		igi_u.igi_qri = igi->igi_qri;
477*43a90889SApple OSS Distributions 		igi_u.igi_uri = igi->igi_uri;
478*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
479*43a90889SApple OSS Distributions 
480*43a90889SApple OSS Distributions 		error = SYSCTL_OUT(req, &igi_u, sizeof(igi_u));
481*43a90889SApple OSS Distributions 		break;
482*43a90889SApple OSS Distributions 	}
483*43a90889SApple OSS Distributions 
484*43a90889SApple OSS Distributions out_locked:
485*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
486*43a90889SApple OSS Distributions 	return error;
487*43a90889SApple OSS Distributions }
488*43a90889SApple OSS Distributions 
489*43a90889SApple OSS Distributions /*
490*43a90889SApple OSS Distributions  * Dispatch an entire queue of pending packet chains
491*43a90889SApple OSS Distributions  *
492*43a90889SApple OSS Distributions  * Must not be called with inm_lock held.
493*43a90889SApple OSS Distributions  */
494*43a90889SApple OSS Distributions static void
igmp_dispatch_queue(struct igmp_ifinfo * igi,struct ifqueue * ifq,int limit,const int loop)495*43a90889SApple OSS Distributions igmp_dispatch_queue(struct igmp_ifinfo *igi, struct ifqueue *ifq, int limit,
496*43a90889SApple OSS Distributions     const int loop)
497*43a90889SApple OSS Distributions {
498*43a90889SApple OSS Distributions 	struct mbuf *m;
499*43a90889SApple OSS Distributions 	struct ip *ip;
500*43a90889SApple OSS Distributions 
501*43a90889SApple OSS Distributions 	if (igi != NULL) {
502*43a90889SApple OSS Distributions 		IGI_LOCK_ASSERT_HELD(igi);
503*43a90889SApple OSS Distributions 	}
504*43a90889SApple OSS Distributions 
505*43a90889SApple OSS Distributions #if SKYWALK
506*43a90889SApple OSS Distributions 	/*
507*43a90889SApple OSS Distributions 	 * Since this function is called holding the igi lock, we need to ensure we
508*43a90889SApple OSS Distributions 	 * don't enter the driver directly because a deadlock can happen if another
509*43a90889SApple OSS Distributions 	 * thread holding the workloop lock tries to acquire the igi lock at
510*43a90889SApple OSS Distributions 	 * the same time.
511*43a90889SApple OSS Distributions 	 */
512*43a90889SApple OSS Distributions 	sk_protect_t __single protect = sk_async_transmit_protect();
513*43a90889SApple OSS Distributions #endif /* SKYWALK */
514*43a90889SApple OSS Distributions 
515*43a90889SApple OSS Distributions 	for (;;) {
516*43a90889SApple OSS Distributions 		IF_DEQUEUE(ifq, m);
517*43a90889SApple OSS Distributions 		if (m == NULL) {
518*43a90889SApple OSS Distributions 			break;
519*43a90889SApple OSS Distributions 		}
520*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: dispatch 0x%llx from 0x%llx\n", __func__,
521*43a90889SApple OSS Distributions 		    (uint64_t)VM_KERNEL_ADDRPERM(ifq),
522*43a90889SApple OSS Distributions 		    (uint64_t)VM_KERNEL_ADDRPERM(m)));
523*43a90889SApple OSS Distributions 		ip = mtod(m, struct ip *);
524*43a90889SApple OSS Distributions 		if (loop) {
525*43a90889SApple OSS Distributions 			m->m_flags |= M_IGMP_LOOP;
526*43a90889SApple OSS Distributions 		}
527*43a90889SApple OSS Distributions 		if (igi != NULL) {
528*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
529*43a90889SApple OSS Distributions 		}
530*43a90889SApple OSS Distributions 		igmp_sendpkt(m);
531*43a90889SApple OSS Distributions 		if (igi != NULL) {
532*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
533*43a90889SApple OSS Distributions 		}
534*43a90889SApple OSS Distributions 		if (--limit == 0) {
535*43a90889SApple OSS Distributions 			break;
536*43a90889SApple OSS Distributions 		}
537*43a90889SApple OSS Distributions 	}
538*43a90889SApple OSS Distributions 
539*43a90889SApple OSS Distributions #if SKYWALK
540*43a90889SApple OSS Distributions 	sk_async_transmit_unprotect(protect);
541*43a90889SApple OSS Distributions #endif /* SKYWALK */
542*43a90889SApple OSS Distributions 
543*43a90889SApple OSS Distributions 	if (igi != NULL) {
544*43a90889SApple OSS Distributions 		IGI_LOCK_ASSERT_HELD(igi);
545*43a90889SApple OSS Distributions 	}
546*43a90889SApple OSS Distributions }
547*43a90889SApple OSS Distributions 
548*43a90889SApple OSS Distributions /*
549*43a90889SApple OSS Distributions  * Filter outgoing IGMP report state by group.
550*43a90889SApple OSS Distributions  *
551*43a90889SApple OSS Distributions  * Reports are ALWAYS suppressed for ALL-HOSTS (224.0.0.1).
552*43a90889SApple OSS Distributions  * If the net.inet.igmp.sendlocal sysctl is 0, then IGMP reports are
553*43a90889SApple OSS Distributions  * disabled for all groups in the 224.0.0.0/24 link-local scope. However,
554*43a90889SApple OSS Distributions  * this may break certain IGMP snooping switches which rely on the old
555*43a90889SApple OSS Distributions  * report behaviour.
556*43a90889SApple OSS Distributions  *
557*43a90889SApple OSS Distributions  * Return zero if the given group is one for which IGMP reports
558*43a90889SApple OSS Distributions  * should be suppressed, or non-zero if reports should be issued.
559*43a90889SApple OSS Distributions  */
560*43a90889SApple OSS Distributions 
561*43a90889SApple OSS Distributions static __inline__
562*43a90889SApple OSS Distributions int
igmp_isgroupreported(const struct in_addr addr)563*43a90889SApple OSS Distributions igmp_isgroupreported(const struct in_addr addr)
564*43a90889SApple OSS Distributions {
565*43a90889SApple OSS Distributions 	if (in_allhosts(addr) ||
566*43a90889SApple OSS Distributions 	    ((!igmp_sendlocal && IN_LOCAL_GROUP(ntohl(addr.s_addr))))) {
567*43a90889SApple OSS Distributions 		return 0;
568*43a90889SApple OSS Distributions 	}
569*43a90889SApple OSS Distributions 
570*43a90889SApple OSS Distributions 	return 1;
571*43a90889SApple OSS Distributions }
572*43a90889SApple OSS Distributions 
573*43a90889SApple OSS Distributions /*
574*43a90889SApple OSS Distributions  * Construct a Router Alert option to use in outgoing packets.
575*43a90889SApple OSS Distributions  */
576*43a90889SApple OSS Distributions static struct mbuf *
igmp_ra_alloc(void)577*43a90889SApple OSS Distributions igmp_ra_alloc(void)
578*43a90889SApple OSS Distributions {
579*43a90889SApple OSS Distributions 	struct mbuf     *m;
580*43a90889SApple OSS Distributions 	struct ipoption *p;
581*43a90889SApple OSS Distributions 
582*43a90889SApple OSS Distributions 	MGET(m, M_WAITOK, MT_DATA);
583*43a90889SApple OSS Distributions 	p = mtod(m, struct ipoption *);
584*43a90889SApple OSS Distributions 	p->ipopt_dst.s_addr = INADDR_ANY;
585*43a90889SApple OSS Distributions 	p->ipopt_list[0] = (char)IPOPT_RA;      /* Router Alert Option */
586*43a90889SApple OSS Distributions 	p->ipopt_list[1] = 0x04;        /* 4 bytes long */
587*43a90889SApple OSS Distributions 	p->ipopt_list[2] = IPOPT_EOL;   /* End of IP option list */
588*43a90889SApple OSS Distributions 	p->ipopt_list[3] = 0x00;        /* pad byte */
589*43a90889SApple OSS Distributions 	m->m_len = sizeof(p->ipopt_dst) + p->ipopt_list[1];
590*43a90889SApple OSS Distributions 
591*43a90889SApple OSS Distributions 	return m;
592*43a90889SApple OSS Distributions }
593*43a90889SApple OSS Distributions 
594*43a90889SApple OSS Distributions /*
595*43a90889SApple OSS Distributions  * Attach IGMP when PF_INET is attached to an interface.
596*43a90889SApple OSS Distributions  */
597*43a90889SApple OSS Distributions struct igmp_ifinfo *
igmp_domifattach(struct ifnet * ifp,zalloc_flags_t how)598*43a90889SApple OSS Distributions igmp_domifattach(struct ifnet *ifp, zalloc_flags_t how)
599*43a90889SApple OSS Distributions {
600*43a90889SApple OSS Distributions 	struct igmp_ifinfo *igi;
601*43a90889SApple OSS Distributions 
602*43a90889SApple OSS Distributions 	os_log_debug(OS_LOG_DEFAULT, "%s: called for ifp %s\n",
603*43a90889SApple OSS Distributions 	    __func__, ifp->if_name);
604*43a90889SApple OSS Distributions 
605*43a90889SApple OSS Distributions 	igi = igi_alloc(how);
606*43a90889SApple OSS Distributions 	if (igi == NULL) {
607*43a90889SApple OSS Distributions 		return NULL;
608*43a90889SApple OSS Distributions 	}
609*43a90889SApple OSS Distributions 
610*43a90889SApple OSS Distributions 	IGMP_LOCK();
611*43a90889SApple OSS Distributions 
612*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
613*43a90889SApple OSS Distributions 	igi_initvar(igi, ifp, 0);
614*43a90889SApple OSS Distributions 	igi->igi_debug |= IFD_ATTACHED;
615*43a90889SApple OSS Distributions 	IGI_ADDREF_LOCKED(igi); /* hold a reference for igi_head */
616*43a90889SApple OSS Distributions 	IGI_ADDREF_LOCKED(igi); /* hold a reference for caller */
617*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
618*43a90889SApple OSS Distributions 	ifnet_lock_shared(ifp);
619*43a90889SApple OSS Distributions 	igmp_initsilent(ifp, igi);
620*43a90889SApple OSS Distributions 	ifnet_lock_done(ifp);
621*43a90889SApple OSS Distributions 
622*43a90889SApple OSS Distributions 	LIST_INSERT_HEAD(&igi_head, igi, igi_link);
623*43a90889SApple OSS Distributions 
624*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
625*43a90889SApple OSS Distributions 
626*43a90889SApple OSS Distributions 	os_log_info(OS_LOG_DEFAULT, "%s: allocated igmp_ifinfo for ifp %s\n",
627*43a90889SApple OSS Distributions 	    __func__, ifp->if_name);
628*43a90889SApple OSS Distributions 
629*43a90889SApple OSS Distributions 	return igi;
630*43a90889SApple OSS Distributions }
631*43a90889SApple OSS Distributions 
632*43a90889SApple OSS Distributions /*
633*43a90889SApple OSS Distributions  * Attach IGMP when PF_INET is reattached to an interface.  Caller is
634*43a90889SApple OSS Distributions  * expected to have an outstanding reference to the igi.
635*43a90889SApple OSS Distributions  */
636*43a90889SApple OSS Distributions void
igmp_domifreattach(struct igmp_ifinfo * igi)637*43a90889SApple OSS Distributions igmp_domifreattach(struct igmp_ifinfo *igi)
638*43a90889SApple OSS Distributions {
639*43a90889SApple OSS Distributions 	struct ifnet *ifp;
640*43a90889SApple OSS Distributions 
641*43a90889SApple OSS Distributions 	IGMP_LOCK();
642*43a90889SApple OSS Distributions 
643*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
644*43a90889SApple OSS Distributions 	VERIFY(!(igi->igi_debug & IFD_ATTACHED));
645*43a90889SApple OSS Distributions 	ifp = igi->igi_ifp;
646*43a90889SApple OSS Distributions 	VERIFY(ifp != NULL);
647*43a90889SApple OSS Distributions 	igi_initvar(igi, ifp, 1);
648*43a90889SApple OSS Distributions 	igi->igi_debug |= IFD_ATTACHED;
649*43a90889SApple OSS Distributions 	IGI_ADDREF_LOCKED(igi); /* hold a reference for igi_head */
650*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
651*43a90889SApple OSS Distributions 	ifnet_lock_shared(ifp);
652*43a90889SApple OSS Distributions 	igmp_initsilent(ifp, igi);
653*43a90889SApple OSS Distributions 	ifnet_lock_done(ifp);
654*43a90889SApple OSS Distributions 
655*43a90889SApple OSS Distributions 	LIST_INSERT_HEAD(&igi_head, igi, igi_link);
656*43a90889SApple OSS Distributions 
657*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
658*43a90889SApple OSS Distributions 
659*43a90889SApple OSS Distributions 	os_log_info(OS_LOG_DEFAULT, "%s: reattached igmp_ifinfo for ifp %s\n",
660*43a90889SApple OSS Distributions 	    __func__, ifp->if_name);
661*43a90889SApple OSS Distributions }
662*43a90889SApple OSS Distributions 
663*43a90889SApple OSS Distributions /*
664*43a90889SApple OSS Distributions  * Hook for domifdetach.
665*43a90889SApple OSS Distributions  */
666*43a90889SApple OSS Distributions void
igmp_domifdetach(struct ifnet * ifp)667*43a90889SApple OSS Distributions igmp_domifdetach(struct ifnet *ifp)
668*43a90889SApple OSS Distributions {
669*43a90889SApple OSS Distributions 	SLIST_HEAD(, in_multi) inm_dthead;
670*43a90889SApple OSS Distributions 
671*43a90889SApple OSS Distributions 	SLIST_INIT(&inm_dthead);
672*43a90889SApple OSS Distributions 
673*43a90889SApple OSS Distributions 	os_log_info(OS_LOG_DEFAULT, "%s: called for ifp %s\n", __func__,
674*43a90889SApple OSS Distributions 	    if_name(ifp));
675*43a90889SApple OSS Distributions 
676*43a90889SApple OSS Distributions 	IGMP_LOCK();
677*43a90889SApple OSS Distributions 	igi_delete(ifp, (struct igmp_inm_relhead *)&inm_dthead);
678*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
679*43a90889SApple OSS Distributions 
680*43a90889SApple OSS Distributions 	/* Now that we're dropped all locks, release detached records */
681*43a90889SApple OSS Distributions 	IGMP_REMOVE_DETACHED_INM(&inm_dthead);
682*43a90889SApple OSS Distributions }
683*43a90889SApple OSS Distributions 
684*43a90889SApple OSS Distributions /*
685*43a90889SApple OSS Distributions  * Called at interface detach time.  Note that we only flush all deferred
686*43a90889SApple OSS Distributions  * responses and record releases; all remaining inm records and their source
687*43a90889SApple OSS Distributions  * entries related to this interface are left intact, in order to handle
688*43a90889SApple OSS Distributions  * the reattach case.
689*43a90889SApple OSS Distributions  */
690*43a90889SApple OSS Distributions static void
igi_delete(const struct ifnet * ifp,struct igmp_inm_relhead * inm_dthead)691*43a90889SApple OSS Distributions igi_delete(const struct ifnet *ifp, struct igmp_inm_relhead *inm_dthead)
692*43a90889SApple OSS Distributions {
693*43a90889SApple OSS Distributions 	struct igmp_ifinfo *igi, *tigi;
694*43a90889SApple OSS Distributions 
695*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_HELD();
696*43a90889SApple OSS Distributions 
697*43a90889SApple OSS Distributions 	LIST_FOREACH_SAFE(igi, &igi_head, igi_link, tigi) {
698*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
699*43a90889SApple OSS Distributions 		if (igi->igi_ifp == ifp) {
700*43a90889SApple OSS Distributions 			/*
701*43a90889SApple OSS Distributions 			 * Free deferred General Query responses.
702*43a90889SApple OSS Distributions 			 */
703*43a90889SApple OSS Distributions 			IF_DRAIN(&igi->igi_gq);
704*43a90889SApple OSS Distributions 			IF_DRAIN(&igi->igi_v2q);
705*43a90889SApple OSS Distributions 			igmp_flush_relq(igi, inm_dthead);
706*43a90889SApple OSS Distributions 			igi->igi_debug &= ~IFD_ATTACHED;
707*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
708*43a90889SApple OSS Distributions 
709*43a90889SApple OSS Distributions 			LIST_REMOVE(igi, igi_link);
710*43a90889SApple OSS Distributions 			IGI_REMREF(igi); /* release igi_head reference */
711*43a90889SApple OSS Distributions 			return;
712*43a90889SApple OSS Distributions 		}
713*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
714*43a90889SApple OSS Distributions 	}
715*43a90889SApple OSS Distributions 	panic("%s: igmp_ifinfo not found for ifp %p(%s)", __func__,
716*43a90889SApple OSS Distributions 	    ifp, if_name(ifp));
717*43a90889SApple OSS Distributions }
718*43a90889SApple OSS Distributions 
719*43a90889SApple OSS Distributions __private_extern__ void
igmp_initsilent(struct ifnet * ifp,struct igmp_ifinfo * igi)720*43a90889SApple OSS Distributions igmp_initsilent(struct ifnet *ifp, struct igmp_ifinfo *igi)
721*43a90889SApple OSS Distributions {
722*43a90889SApple OSS Distributions 	ifnet_lock_assert(ifp, IFNET_LCK_ASSERT_OWNED);
723*43a90889SApple OSS Distributions 
724*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_NOTHELD(igi);
725*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
726*43a90889SApple OSS Distributions 	if (!(ifp->if_flags & IFF_MULTICAST)) {
727*43a90889SApple OSS Distributions 		igi->igi_flags |= IGIF_SILENT;
728*43a90889SApple OSS Distributions 	} else {
729*43a90889SApple OSS Distributions 		igi->igi_flags &= ~IGIF_SILENT;
730*43a90889SApple OSS Distributions 	}
731*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
732*43a90889SApple OSS Distributions }
733*43a90889SApple OSS Distributions 
734*43a90889SApple OSS Distributions static void
igi_initvar(struct igmp_ifinfo * igi,struct ifnet * ifp,int reattach)735*43a90889SApple OSS Distributions igi_initvar(struct igmp_ifinfo *igi, struct ifnet *ifp, int reattach)
736*43a90889SApple OSS Distributions {
737*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
738*43a90889SApple OSS Distributions 
739*43a90889SApple OSS Distributions 	igi->igi_ifp = ifp;
740*43a90889SApple OSS Distributions 	igi->igi_version = igmp_default_version;
741*43a90889SApple OSS Distributions 	igi->igi_flags = 0;
742*43a90889SApple OSS Distributions 	igi->igi_rv = IGMP_RV_INIT;
743*43a90889SApple OSS Distributions 	igi->igi_qi = IGMP_QI_INIT;
744*43a90889SApple OSS Distributions 	igi->igi_qri = IGMP_QRI_INIT;
745*43a90889SApple OSS Distributions 	igi->igi_uri = IGMP_URI_INIT;
746*43a90889SApple OSS Distributions 
747*43a90889SApple OSS Distributions 	if (!reattach) {
748*43a90889SApple OSS Distributions 		SLIST_INIT(&igi->igi_relinmhead);
749*43a90889SApple OSS Distributions 	}
750*43a90889SApple OSS Distributions 
751*43a90889SApple OSS Distributions 	/*
752*43a90889SApple OSS Distributions 	 * Responses to general queries are subject to bounds.
753*43a90889SApple OSS Distributions 	 */
754*43a90889SApple OSS Distributions 	igi->igi_gq.ifq_maxlen =  IGMP_MAX_RESPONSE_PACKETS;
755*43a90889SApple OSS Distributions 	igi->igi_v2q.ifq_maxlen = IGMP_MAX_RESPONSE_PACKETS;
756*43a90889SApple OSS Distributions }
757*43a90889SApple OSS Distributions 
758*43a90889SApple OSS Distributions static struct igmp_ifinfo *
igi_alloc(zalloc_flags_t how)759*43a90889SApple OSS Distributions igi_alloc(zalloc_flags_t how)
760*43a90889SApple OSS Distributions {
761*43a90889SApple OSS Distributions 	struct igmp_ifinfo *igi = zalloc_flags(igi_zone, how | Z_ZERO);
762*43a90889SApple OSS Distributions 	if (igi != NULL) {
763*43a90889SApple OSS Distributions 		lck_mtx_init(&igi->igi_lock, &igmp_mtx_grp, &igmp_mtx_attr);
764*43a90889SApple OSS Distributions 		igi->igi_debug |= IFD_ALLOC;
765*43a90889SApple OSS Distributions 	}
766*43a90889SApple OSS Distributions 	return igi;
767*43a90889SApple OSS Distributions }
768*43a90889SApple OSS Distributions 
769*43a90889SApple OSS Distributions static void
igi_free(struct igmp_ifinfo * igi)770*43a90889SApple OSS Distributions igi_free(struct igmp_ifinfo *igi)
771*43a90889SApple OSS Distributions {
772*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
773*43a90889SApple OSS Distributions 	if (igi->igi_debug & IFD_ATTACHED) {
774*43a90889SApple OSS Distributions 		panic("%s: attached igi=%p is being freed", __func__, igi);
775*43a90889SApple OSS Distributions 		/* NOTREACHED */
776*43a90889SApple OSS Distributions 	} else if (igi->igi_ifp != NULL) {
777*43a90889SApple OSS Distributions 		panic("%s: ifp not NULL for igi=%p", __func__, igi);
778*43a90889SApple OSS Distributions 		/* NOTREACHED */
779*43a90889SApple OSS Distributions 	} else if (!(igi->igi_debug & IFD_ALLOC)) {
780*43a90889SApple OSS Distributions 		panic("%s: igi %p cannot be freed", __func__, igi);
781*43a90889SApple OSS Distributions 		/* NOTREACHED */
782*43a90889SApple OSS Distributions 	} else if (igi->igi_refcnt != 0) {
783*43a90889SApple OSS Distributions 		panic("%s: non-zero refcnt igi=%p", __func__, igi);
784*43a90889SApple OSS Distributions 		/* NOTREACHED */
785*43a90889SApple OSS Distributions 	}
786*43a90889SApple OSS Distributions 	igi->igi_debug &= ~IFD_ALLOC;
787*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
788*43a90889SApple OSS Distributions 
789*43a90889SApple OSS Distributions 	lck_mtx_destroy(&igi->igi_lock, &igmp_mtx_grp);
790*43a90889SApple OSS Distributions 	zfree(igi_zone, igi);
791*43a90889SApple OSS Distributions }
792*43a90889SApple OSS Distributions 
793*43a90889SApple OSS Distributions void
igi_addref(struct igmp_ifinfo * igi,int locked)794*43a90889SApple OSS Distributions igi_addref(struct igmp_ifinfo *igi, int locked)
795*43a90889SApple OSS Distributions {
796*43a90889SApple OSS Distributions 	if (!locked) {
797*43a90889SApple OSS Distributions 		IGI_LOCK_SPIN(igi);
798*43a90889SApple OSS Distributions 	} else {
799*43a90889SApple OSS Distributions 		IGI_LOCK_ASSERT_HELD(igi);
800*43a90889SApple OSS Distributions 	}
801*43a90889SApple OSS Distributions 
802*43a90889SApple OSS Distributions 	if (++igi->igi_refcnt == 0) {
803*43a90889SApple OSS Distributions 		panic("%s: igi=%p wraparound refcnt", __func__, igi);
804*43a90889SApple OSS Distributions 		/* NOTREACHED */
805*43a90889SApple OSS Distributions 	}
806*43a90889SApple OSS Distributions 	if (!locked) {
807*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
808*43a90889SApple OSS Distributions 	}
809*43a90889SApple OSS Distributions }
810*43a90889SApple OSS Distributions 
811*43a90889SApple OSS Distributions void
igi_remref(struct igmp_ifinfo * igi)812*43a90889SApple OSS Distributions igi_remref(struct igmp_ifinfo *igi)
813*43a90889SApple OSS Distributions {
814*43a90889SApple OSS Distributions 	SLIST_HEAD(, in_multi) inm_dthead;
815*43a90889SApple OSS Distributions 	struct ifnet *ifp;
816*43a90889SApple OSS Distributions 
817*43a90889SApple OSS Distributions 	IGI_LOCK_SPIN(igi);
818*43a90889SApple OSS Distributions 
819*43a90889SApple OSS Distributions 	if (igi->igi_refcnt == 0) {
820*43a90889SApple OSS Distributions 		panic("%s: igi=%p negative refcnt", __func__, igi);
821*43a90889SApple OSS Distributions 		/* NOTREACHED */
822*43a90889SApple OSS Distributions 	}
823*43a90889SApple OSS Distributions 
824*43a90889SApple OSS Distributions 	--igi->igi_refcnt;
825*43a90889SApple OSS Distributions 	if (igi->igi_refcnt > 0) {
826*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
827*43a90889SApple OSS Distributions 		return;
828*43a90889SApple OSS Distributions 	}
829*43a90889SApple OSS Distributions 
830*43a90889SApple OSS Distributions 	ifp = igi->igi_ifp;
831*43a90889SApple OSS Distributions 	igi->igi_ifp = NULL;
832*43a90889SApple OSS Distributions 	IF_DRAIN(&igi->igi_gq);
833*43a90889SApple OSS Distributions 	IF_DRAIN(&igi->igi_v2q);
834*43a90889SApple OSS Distributions 	SLIST_INIT(&inm_dthead);
835*43a90889SApple OSS Distributions 	igmp_flush_relq(igi, (struct igmp_inm_relhead *)&inm_dthead);
836*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
837*43a90889SApple OSS Distributions 
838*43a90889SApple OSS Distributions 	/* Now that we're dropped all locks, release detached records */
839*43a90889SApple OSS Distributions 	IGMP_REMOVE_DETACHED_INM(&inm_dthead);
840*43a90889SApple OSS Distributions 
841*43a90889SApple OSS Distributions 	os_log_info(OS_LOG_DEFAULT, "%s: freeing igmp_ifinfo for ifp %s\n",
842*43a90889SApple OSS Distributions 	    __func__, if_name(ifp));
843*43a90889SApple OSS Distributions 
844*43a90889SApple OSS Distributions 	igi_free(igi);
845*43a90889SApple OSS Distributions }
846*43a90889SApple OSS Distributions 
847*43a90889SApple OSS Distributions /*
848*43a90889SApple OSS Distributions  * Process a received IGMPv1 query.
849*43a90889SApple OSS Distributions  * Return non-zero if the message should be dropped.
850*43a90889SApple OSS Distributions  */
851*43a90889SApple OSS Distributions static int
igmp_input_v1_query(struct ifnet * ifp,const struct ip * ip,const struct igmp * igmp)852*43a90889SApple OSS Distributions igmp_input_v1_query(struct ifnet *ifp, const struct ip *ip,
853*43a90889SApple OSS Distributions     const struct igmp *igmp)
854*43a90889SApple OSS Distributions {
855*43a90889SApple OSS Distributions 	struct igmp_ifinfo      *igi;
856*43a90889SApple OSS Distributions 	struct in_multi         *inm;
857*43a90889SApple OSS Distributions 	struct in_multistep     step;
858*43a90889SApple OSS Distributions 	struct igmp_tparams     itp = { .qpt = 0, .it = 0, .cst = 0, .sct = 0 };
859*43a90889SApple OSS Distributions 
860*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_NOTHELD();
861*43a90889SApple OSS Distributions 
862*43a90889SApple OSS Distributions 	/*
863*43a90889SApple OSS Distributions 	 * IGMPv1 Host Membership Queries SHOULD always be addressed to
864*43a90889SApple OSS Distributions 	 * 224.0.0.1. They are always treated as General Queries.
865*43a90889SApple OSS Distributions 	 * igmp_group is always ignored. Do not drop it as a userland
866*43a90889SApple OSS Distributions 	 * daemon may wish to see it.
867*43a90889SApple OSS Distributions 	 */
868*43a90889SApple OSS Distributions 	if (!in_allhosts(ip->ip_dst) || !in_nullhost(igmp->igmp_group)) {
869*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_badqueries);
870*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_badqueries);
871*43a90889SApple OSS Distributions 		goto done;
872*43a90889SApple OSS Distributions 	}
873*43a90889SApple OSS Distributions 	IGMPSTAT_INC(igps_rcv_gen_queries);
874*43a90889SApple OSS Distributions 
875*43a90889SApple OSS Distributions 	igi = IGMP_IFINFO(ifp);
876*43a90889SApple OSS Distributions 	VERIFY(igi != NULL);
877*43a90889SApple OSS Distributions 
878*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
879*43a90889SApple OSS Distributions 	if (igi->igi_flags & IGIF_LOOPBACK) {
880*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT,
881*43a90889SApple OSS Distributions 		    "%s: ignore v1 query on IGIF_LOOPBACK "
882*43a90889SApple OSS Distributions 		    "ifp %s\n", __func__,
883*43a90889SApple OSS Distributions 		    if_name(ifp));
884*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
885*43a90889SApple OSS Distributions 		goto done;
886*43a90889SApple OSS Distributions 	}
887*43a90889SApple OSS Distributions 	/*
888*43a90889SApple OSS Distributions 	 * Switch to IGMPv1 host compatibility mode.
889*43a90889SApple OSS Distributions 	 */
890*43a90889SApple OSS Distributions 	itp.qpt = igmp_set_version(igi, IGMP_VERSION_1);
891*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
892*43a90889SApple OSS Distributions 
893*43a90889SApple OSS Distributions 	os_log_debug(OS_LOG_DEFAULT, "%s: process v1 query on ifp %s\n", __func__,
894*43a90889SApple OSS Distributions 	    if_name(ifp));
895*43a90889SApple OSS Distributions 
896*43a90889SApple OSS Distributions 	/*
897*43a90889SApple OSS Distributions 	 * Start the timers in all of our group records
898*43a90889SApple OSS Distributions 	 * for the interface on which the query arrived,
899*43a90889SApple OSS Distributions 	 * except those which are already running.
900*43a90889SApple OSS Distributions 	 */
901*43a90889SApple OSS Distributions 	in_multihead_lock_shared();
902*43a90889SApple OSS Distributions 	IN_FIRST_MULTI(step, inm);
903*43a90889SApple OSS Distributions 	while (inm != NULL) {
904*43a90889SApple OSS Distributions 		INM_LOCK(inm);
905*43a90889SApple OSS Distributions 		if (inm->inm_ifp != ifp || inm->inm_timer != 0) {
906*43a90889SApple OSS Distributions 			goto next;
907*43a90889SApple OSS Distributions 		}
908*43a90889SApple OSS Distributions 
909*43a90889SApple OSS Distributions 		switch (inm->inm_state) {
910*43a90889SApple OSS Distributions 		case IGMP_NOT_MEMBER:
911*43a90889SApple OSS Distributions 		case IGMP_SILENT_MEMBER:
912*43a90889SApple OSS Distributions 			break;
913*43a90889SApple OSS Distributions 		case IGMP_G_QUERY_PENDING_MEMBER:
914*43a90889SApple OSS Distributions 		case IGMP_SG_QUERY_PENDING_MEMBER:
915*43a90889SApple OSS Distributions 		case IGMP_REPORTING_MEMBER:
916*43a90889SApple OSS Distributions 		case IGMP_IDLE_MEMBER:
917*43a90889SApple OSS Distributions 		case IGMP_LAZY_MEMBER:
918*43a90889SApple OSS Distributions 		case IGMP_SLEEPING_MEMBER:
919*43a90889SApple OSS Distributions 		case IGMP_AWAKENING_MEMBER:
920*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_REPORTING_MEMBER;
921*43a90889SApple OSS Distributions 			inm->inm_timer = IGMP_RANDOM_DELAY(IGMP_V1V2_MAX_RI);
922*43a90889SApple OSS Distributions 			itp.cst = 1;
923*43a90889SApple OSS Distributions 			break;
924*43a90889SApple OSS Distributions 		case IGMP_LEAVING_MEMBER:
925*43a90889SApple OSS Distributions 			break;
926*43a90889SApple OSS Distributions 		}
927*43a90889SApple OSS Distributions next:
928*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
929*43a90889SApple OSS Distributions 		IN_NEXT_MULTI(step, inm);
930*43a90889SApple OSS Distributions 	}
931*43a90889SApple OSS Distributions 	in_multihead_lock_done();
932*43a90889SApple OSS Distributions done:
933*43a90889SApple OSS Distributions 	igmp_set_timeout(&itp);
934*43a90889SApple OSS Distributions 
935*43a90889SApple OSS Distributions 	return 0;
936*43a90889SApple OSS Distributions }
937*43a90889SApple OSS Distributions 
938*43a90889SApple OSS Distributions /*
939*43a90889SApple OSS Distributions  * Process a received IGMPv2 general or group-specific query.
940*43a90889SApple OSS Distributions  */
941*43a90889SApple OSS Distributions static int
igmp_input_v2_query(struct ifnet * ifp,const struct ip * ip,const struct igmp * igmp)942*43a90889SApple OSS Distributions igmp_input_v2_query(struct ifnet *ifp, const struct ip *ip,
943*43a90889SApple OSS Distributions     const struct igmp *igmp)
944*43a90889SApple OSS Distributions {
945*43a90889SApple OSS Distributions 	struct igmp_ifinfo      *igi;
946*43a90889SApple OSS Distributions 	struct in_multi         *inm;
947*43a90889SApple OSS Distributions 	int                      is_general_query;
948*43a90889SApple OSS Distributions 	uint16_t                 timer;
949*43a90889SApple OSS Distributions 	struct igmp_tparams      itp = { .qpt = 0, .it = 0, .cst = 0, .sct = 0 };
950*43a90889SApple OSS Distributions 
951*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_NOTHELD();
952*43a90889SApple OSS Distributions 
953*43a90889SApple OSS Distributions 	is_general_query = 0;
954*43a90889SApple OSS Distributions 
955*43a90889SApple OSS Distributions 	/*
956*43a90889SApple OSS Distributions 	 * Validate address fields upfront.
957*43a90889SApple OSS Distributions 	 */
958*43a90889SApple OSS Distributions 	if (in_nullhost(igmp->igmp_group)) {
959*43a90889SApple OSS Distributions 		/*
960*43a90889SApple OSS Distributions 		 * IGMPv2 General Query.
961*43a90889SApple OSS Distributions 		 * If this was not sent to the all-hosts group, ignore it.
962*43a90889SApple OSS Distributions 		 */
963*43a90889SApple OSS Distributions 		if (!in_allhosts(ip->ip_dst)) {
964*43a90889SApple OSS Distributions 			goto done;
965*43a90889SApple OSS Distributions 		}
966*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_gen_queries);
967*43a90889SApple OSS Distributions 		is_general_query = 1;
968*43a90889SApple OSS Distributions 	} else {
969*43a90889SApple OSS Distributions 		/* IGMPv2 Group-Specific Query. */
970*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_group_queries);
971*43a90889SApple OSS Distributions 	}
972*43a90889SApple OSS Distributions 
973*43a90889SApple OSS Distributions 	igi = IGMP_IFINFO(ifp);
974*43a90889SApple OSS Distributions 	VERIFY(igi != NULL);
975*43a90889SApple OSS Distributions 
976*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
977*43a90889SApple OSS Distributions 	if (igi->igi_flags & IGIF_LOOPBACK) {
978*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: ignore v2 query on IGIF_LOOPBACK "
979*43a90889SApple OSS Distributions 		    "ifp %s\n", __func__, if_name(ifp));
980*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
981*43a90889SApple OSS Distributions 		goto done;
982*43a90889SApple OSS Distributions 	}
983*43a90889SApple OSS Distributions 	/*
984*43a90889SApple OSS Distributions 	 * Ignore v2 query if in v1 Compatibility Mode.
985*43a90889SApple OSS Distributions 	 */
986*43a90889SApple OSS Distributions 	if (igi->igi_version == IGMP_VERSION_1) {
987*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
988*43a90889SApple OSS Distributions 		goto done;
989*43a90889SApple OSS Distributions 	}
990*43a90889SApple OSS Distributions 	itp.qpt = igmp_set_version(igi, IGMP_VERSION_2);
991*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
992*43a90889SApple OSS Distributions 
993*43a90889SApple OSS Distributions 	timer = igmp->igmp_code / IGMP_TIMER_SCALE;
994*43a90889SApple OSS Distributions 	if (timer == 0) {
995*43a90889SApple OSS Distributions 		timer = 1;
996*43a90889SApple OSS Distributions 	}
997*43a90889SApple OSS Distributions 
998*43a90889SApple OSS Distributions 	if (is_general_query) {
999*43a90889SApple OSS Distributions 		struct in_multistep step;
1000*43a90889SApple OSS Distributions 
1001*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: process v2 general query on ifp %s\n",
1002*43a90889SApple OSS Distributions 		    __func__, if_name(ifp));
1003*43a90889SApple OSS Distributions 		/*
1004*43a90889SApple OSS Distributions 		 * For each reporting group joined on this
1005*43a90889SApple OSS Distributions 		 * interface, kick the report timer.
1006*43a90889SApple OSS Distributions 		 */
1007*43a90889SApple OSS Distributions 		in_multihead_lock_shared();
1008*43a90889SApple OSS Distributions 		IN_FIRST_MULTI(step, inm);
1009*43a90889SApple OSS Distributions 		while (inm != NULL) {
1010*43a90889SApple OSS Distributions 			INM_LOCK(inm);
1011*43a90889SApple OSS Distributions 			if (inm->inm_ifp == ifp) {
1012*43a90889SApple OSS Distributions 				itp.cst += igmp_v2_update_group(inm, timer);
1013*43a90889SApple OSS Distributions 			}
1014*43a90889SApple OSS Distributions 			INM_UNLOCK(inm);
1015*43a90889SApple OSS Distributions 			IN_NEXT_MULTI(step, inm);
1016*43a90889SApple OSS Distributions 		}
1017*43a90889SApple OSS Distributions 		in_multihead_lock_done();
1018*43a90889SApple OSS Distributions 	} else {
1019*43a90889SApple OSS Distributions 		/*
1020*43a90889SApple OSS Distributions 		 * Group-specific IGMPv2 query, we need only
1021*43a90889SApple OSS Distributions 		 * look up the single group to process it.
1022*43a90889SApple OSS Distributions 		 */
1023*43a90889SApple OSS Distributions 		in_multihead_lock_shared();
1024*43a90889SApple OSS Distributions 		IN_LOOKUP_MULTI(&igmp->igmp_group, ifp, inm);
1025*43a90889SApple OSS Distributions 		in_multihead_lock_done();
1026*43a90889SApple OSS Distributions 		if (inm != NULL) {
1027*43a90889SApple OSS Distributions 			INM_LOCK(inm);
1028*43a90889SApple OSS Distributions 			IGMP_INET_PRINTF(igmp->igmp_group,
1029*43a90889SApple OSS Distributions 			    ("process v2 query %s on ifp 0x%llx(%s)\n",
1030*43a90889SApple OSS Distributions 			    _igmp_inet_buf,
1031*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(ifp), if_name(ifp)));
1032*43a90889SApple OSS Distributions 			itp.cst = igmp_v2_update_group(inm, timer);
1033*43a90889SApple OSS Distributions 			INM_UNLOCK(inm);
1034*43a90889SApple OSS Distributions 			INM_REMREF(inm); /* from IN_LOOKUP_MULTI */
1035*43a90889SApple OSS Distributions 		}
1036*43a90889SApple OSS Distributions 	}
1037*43a90889SApple OSS Distributions done:
1038*43a90889SApple OSS Distributions 	igmp_set_timeout(&itp);
1039*43a90889SApple OSS Distributions 
1040*43a90889SApple OSS Distributions 	return 0;
1041*43a90889SApple OSS Distributions }
1042*43a90889SApple OSS Distributions 
1043*43a90889SApple OSS Distributions /*
1044*43a90889SApple OSS Distributions  * Update the report timer on a group in response to an IGMPv2 query.
1045*43a90889SApple OSS Distributions  *
1046*43a90889SApple OSS Distributions  * If we are becoming the reporting member for this group, start the timer.
1047*43a90889SApple OSS Distributions  * If we already are the reporting member for this group, and timer is
1048*43a90889SApple OSS Distributions  * below the threshold, reset it.
1049*43a90889SApple OSS Distributions  *
1050*43a90889SApple OSS Distributions  * We may be updating the group for the first time since we switched
1051*43a90889SApple OSS Distributions  * to IGMPv3. If we are, then we must clear any recorded source lists,
1052*43a90889SApple OSS Distributions  * and transition to REPORTING state; the group timer is overloaded
1053*43a90889SApple OSS Distributions  * for group and group-source query responses.
1054*43a90889SApple OSS Distributions  *
1055*43a90889SApple OSS Distributions  * Unlike IGMPv3, the delay per group should be jittered
1056*43a90889SApple OSS Distributions  * to avoid bursts of IGMPv2 reports.
1057*43a90889SApple OSS Distributions  */
1058*43a90889SApple OSS Distributions static uint32_t
igmp_v2_update_group(struct in_multi * inm,const int timer)1059*43a90889SApple OSS Distributions igmp_v2_update_group(struct in_multi *inm, const int timer)
1060*43a90889SApple OSS Distributions {
1061*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(inm->inm_addr, ("%s: %s/%s timer=%d\n",
1062*43a90889SApple OSS Distributions 	    __func__, _igmp_inet_buf, if_name(inm->inm_ifp),
1063*43a90889SApple OSS Distributions 	    timer));
1064*43a90889SApple OSS Distributions 
1065*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
1066*43a90889SApple OSS Distributions 
1067*43a90889SApple OSS Distributions 	switch (inm->inm_state) {
1068*43a90889SApple OSS Distributions 	case IGMP_NOT_MEMBER:
1069*43a90889SApple OSS Distributions 	case IGMP_SILENT_MEMBER:
1070*43a90889SApple OSS Distributions 		break;
1071*43a90889SApple OSS Distributions 	case IGMP_REPORTING_MEMBER:
1072*43a90889SApple OSS Distributions 		if (inm->inm_timer != 0 &&
1073*43a90889SApple OSS Distributions 		    inm->inm_timer <= timer) {
1074*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: REPORTING and timer running, "
1075*43a90889SApple OSS Distributions 			    "skipping.\n", __func__));
1076*43a90889SApple OSS Distributions 			break;
1077*43a90889SApple OSS Distributions 		}
1078*43a90889SApple OSS Distributions 		OS_FALLTHROUGH;
1079*43a90889SApple OSS Distributions 	case IGMP_SG_QUERY_PENDING_MEMBER:
1080*43a90889SApple OSS Distributions 	case IGMP_G_QUERY_PENDING_MEMBER:
1081*43a90889SApple OSS Distributions 	case IGMP_IDLE_MEMBER:
1082*43a90889SApple OSS Distributions 	case IGMP_LAZY_MEMBER:
1083*43a90889SApple OSS Distributions 	case IGMP_AWAKENING_MEMBER:
1084*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: ->REPORTING\n", __func__));
1085*43a90889SApple OSS Distributions 		inm->inm_state = IGMP_REPORTING_MEMBER;
1086*43a90889SApple OSS Distributions 		inm->inm_timer = IGMP_RANDOM_DELAY(timer);
1087*43a90889SApple OSS Distributions 		break;
1088*43a90889SApple OSS Distributions 	case IGMP_SLEEPING_MEMBER:
1089*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: ->AWAKENING\n", __func__));
1090*43a90889SApple OSS Distributions 		inm->inm_state = IGMP_AWAKENING_MEMBER;
1091*43a90889SApple OSS Distributions 		break;
1092*43a90889SApple OSS Distributions 	case IGMP_LEAVING_MEMBER:
1093*43a90889SApple OSS Distributions 		break;
1094*43a90889SApple OSS Distributions 	}
1095*43a90889SApple OSS Distributions 
1096*43a90889SApple OSS Distributions 	return inm->inm_timer;
1097*43a90889SApple OSS Distributions }
1098*43a90889SApple OSS Distributions 
1099*43a90889SApple OSS Distributions /*
1100*43a90889SApple OSS Distributions  * Process a received IGMPv3 general, group-specific or
1101*43a90889SApple OSS Distributions  * group-and-source-specific query.
1102*43a90889SApple OSS Distributions  * Assumes m has already been pulled up to the full IGMP message length.
1103*43a90889SApple OSS Distributions  * Return 0 if successful, otherwise an appropriate error code is returned.
1104*43a90889SApple OSS Distributions  */
1105*43a90889SApple OSS Distributions static int
igmp_input_v3_query(struct ifnet * ifp,const struct ip * ip,struct igmpv3 * __indexable igmpv3)1106*43a90889SApple OSS Distributions igmp_input_v3_query(struct ifnet *ifp, const struct ip *ip,
1107*43a90889SApple OSS Distributions     /*const*/ struct igmpv3 *__indexable igmpv3)
1108*43a90889SApple OSS Distributions {
1109*43a90889SApple OSS Distributions 	struct igmp_ifinfo      *igi;
1110*43a90889SApple OSS Distributions 	struct in_multi         *inm;
1111*43a90889SApple OSS Distributions 	int                      is_general_query;
1112*43a90889SApple OSS Distributions 	uint32_t                 maxresp, nsrc, qqi;
1113*43a90889SApple OSS Distributions 	uint32_t                 timer;
1114*43a90889SApple OSS Distributions 	uint8_t                  qrv;
1115*43a90889SApple OSS Distributions 	struct igmp_tparams      itp = { .qpt = 0, .it = 0, .cst = 0, .sct = 0 };
1116*43a90889SApple OSS Distributions 
1117*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_NOTHELD();
1118*43a90889SApple OSS Distributions 
1119*43a90889SApple OSS Distributions 	is_general_query = 0;
1120*43a90889SApple OSS Distributions 
1121*43a90889SApple OSS Distributions 	os_log_debug(OS_LOG_DEFAULT, "%s: process v3 query on ifp %s\n", __func__,
1122*43a90889SApple OSS Distributions 	    if_name(ifp));
1123*43a90889SApple OSS Distributions 
1124*43a90889SApple OSS Distributions 	maxresp = igmpv3->igmp_code;    /* in 1/10ths of a second */
1125*43a90889SApple OSS Distributions 	if (maxresp >= 128) {
1126*43a90889SApple OSS Distributions 		maxresp = IGMP_MANT(igmpv3->igmp_code) <<
1127*43a90889SApple OSS Distributions 		    (IGMP_EXP(igmpv3->igmp_code) + 3);
1128*43a90889SApple OSS Distributions 	}
1129*43a90889SApple OSS Distributions 
1130*43a90889SApple OSS Distributions 	/*
1131*43a90889SApple OSS Distributions 	 * Robustness must never be less than 2 for on-wire IGMPv3.
1132*43a90889SApple OSS Distributions 	 * FUTURE: Check if ifp has IGIF_LOOPBACK set, as we will make
1133*43a90889SApple OSS Distributions 	 * an exception for interfaces whose IGMPv3 state changes
1134*43a90889SApple OSS Distributions 	 * are redirected to loopback (e.g. MANET).
1135*43a90889SApple OSS Distributions 	 */
1136*43a90889SApple OSS Distributions 	qrv = IGMP_QRV(igmpv3->igmp_misc);
1137*43a90889SApple OSS Distributions 	if (qrv < 2) {
1138*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: clamping qrv %d to %d\n", __func__,
1139*43a90889SApple OSS Distributions 		    qrv, IGMP_RV_INIT));
1140*43a90889SApple OSS Distributions 		qrv = IGMP_RV_INIT;
1141*43a90889SApple OSS Distributions 	}
1142*43a90889SApple OSS Distributions 
1143*43a90889SApple OSS Distributions 	qqi = igmpv3->igmp_qqi;
1144*43a90889SApple OSS Distributions 	if (qqi >= 128) {
1145*43a90889SApple OSS Distributions 		qqi = IGMP_MANT(igmpv3->igmp_qqi) <<
1146*43a90889SApple OSS Distributions 		    (IGMP_EXP(igmpv3->igmp_qqi) + 3);
1147*43a90889SApple OSS Distributions 	}
1148*43a90889SApple OSS Distributions 
1149*43a90889SApple OSS Distributions 	timer = maxresp / IGMP_TIMER_SCALE;
1150*43a90889SApple OSS Distributions 	if (timer == 0) {
1151*43a90889SApple OSS Distributions 		timer = 1;
1152*43a90889SApple OSS Distributions 	}
1153*43a90889SApple OSS Distributions 
1154*43a90889SApple OSS Distributions 	nsrc = ntohs(igmpv3->igmp_numsrc);
1155*43a90889SApple OSS Distributions 
1156*43a90889SApple OSS Distributions 	/*
1157*43a90889SApple OSS Distributions 	 * Validate address fields and versions upfront before
1158*43a90889SApple OSS Distributions 	 * accepting v3 query.
1159*43a90889SApple OSS Distributions 	 */
1160*43a90889SApple OSS Distributions 	if (in_nullhost(igmpv3->igmp_group)) {
1161*43a90889SApple OSS Distributions 		/*
1162*43a90889SApple OSS Distributions 		 * IGMPv3 General Query.
1163*43a90889SApple OSS Distributions 		 *
1164*43a90889SApple OSS Distributions 		 * General Queries SHOULD be directed to 224.0.0.1.
1165*43a90889SApple OSS Distributions 		 * A general query with a source list has undefined
1166*43a90889SApple OSS Distributions 		 * behaviour; discard it.
1167*43a90889SApple OSS Distributions 		 */
1168*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_gen_queries);
1169*43a90889SApple OSS Distributions 		if (!in_allhosts(ip->ip_dst) || nsrc > 0) {
1170*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_badqueries);
1171*43a90889SApple OSS Distributions 			OIGMPSTAT_INC(igps_rcv_badqueries);
1172*43a90889SApple OSS Distributions 			goto done;
1173*43a90889SApple OSS Distributions 		}
1174*43a90889SApple OSS Distributions 		is_general_query = 1;
1175*43a90889SApple OSS Distributions 	} else {
1176*43a90889SApple OSS Distributions 		/* Group or group-source specific query. */
1177*43a90889SApple OSS Distributions 		if (nsrc == 0) {
1178*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_group_queries);
1179*43a90889SApple OSS Distributions 		} else {
1180*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_gsr_queries);
1181*43a90889SApple OSS Distributions 		}
1182*43a90889SApple OSS Distributions 	}
1183*43a90889SApple OSS Distributions 
1184*43a90889SApple OSS Distributions 	igi = IGMP_IFINFO(ifp);
1185*43a90889SApple OSS Distributions 	VERIFY(igi != NULL);
1186*43a90889SApple OSS Distributions 
1187*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
1188*43a90889SApple OSS Distributions 	if (igi->igi_flags & IGIF_LOOPBACK) {
1189*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: ignore v3 query on IGIF_LOOPBACK "
1190*43a90889SApple OSS Distributions 		    "ifp %s\n", __func__,
1191*43a90889SApple OSS Distributions 		    if_name(ifp));
1192*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1193*43a90889SApple OSS Distributions 		goto done;
1194*43a90889SApple OSS Distributions 	}
1195*43a90889SApple OSS Distributions 
1196*43a90889SApple OSS Distributions 	/*
1197*43a90889SApple OSS Distributions 	 * Discard the v3 query if we're in Compatibility Mode.
1198*43a90889SApple OSS Distributions 	 * The RFC is not obviously worded that hosts need to stay in
1199*43a90889SApple OSS Distributions 	 * compatibility mode until the Old Version Querier Present
1200*43a90889SApple OSS Distributions 	 * timer expires.
1201*43a90889SApple OSS Distributions 	 */
1202*43a90889SApple OSS Distributions 	if (igi->igi_version != IGMP_VERSION_3) {
1203*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: ignore v3 query in v%d mode on "
1204*43a90889SApple OSS Distributions 		    "ifp %s\n", __func__, igi->igi_version,
1205*43a90889SApple OSS Distributions 		    if_name(ifp));
1206*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1207*43a90889SApple OSS Distributions 		goto done;
1208*43a90889SApple OSS Distributions 	}
1209*43a90889SApple OSS Distributions 
1210*43a90889SApple OSS Distributions 	itp.qpt = igmp_set_version(igi, IGMP_VERSION_3);
1211*43a90889SApple OSS Distributions 	igi->igi_rv = qrv;
1212*43a90889SApple OSS Distributions 	igi->igi_qi = qqi;
1213*43a90889SApple OSS Distributions 	igi->igi_qri = MAX(timer, IGMP_QRI_MIN);
1214*43a90889SApple OSS Distributions 
1215*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: qrv %d qi %d qri %d\n", __func__, igi->igi_rv,
1216*43a90889SApple OSS Distributions 	    igi->igi_qi, igi->igi_qri));
1217*43a90889SApple OSS Distributions 
1218*43a90889SApple OSS Distributions 	if (is_general_query) {
1219*43a90889SApple OSS Distributions 		/*
1220*43a90889SApple OSS Distributions 		 * Schedule a current-state report on this ifp for
1221*43a90889SApple OSS Distributions 		 * all groups, possibly containing source lists.
1222*43a90889SApple OSS Distributions 		 * If there is a pending General Query response
1223*43a90889SApple OSS Distributions 		 * scheduled earlier than the selected delay, do
1224*43a90889SApple OSS Distributions 		 * not schedule any other reports.
1225*43a90889SApple OSS Distributions 		 * Otherwise, reset the interface timer.
1226*43a90889SApple OSS Distributions 		 */
1227*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: process v3 general query on ifp %s\n",
1228*43a90889SApple OSS Distributions 		    __func__, if_name(ifp));
1229*43a90889SApple OSS Distributions 		if (igi->igi_v3_timer == 0 || igi->igi_v3_timer >= timer) {
1230*43a90889SApple OSS Distributions 			itp.it = igi->igi_v3_timer = IGMP_RANDOM_DELAY(timer);
1231*43a90889SApple OSS Distributions 		}
1232*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1233*43a90889SApple OSS Distributions 	} else {
1234*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1235*43a90889SApple OSS Distributions 		/*
1236*43a90889SApple OSS Distributions 		 * Group-source-specific queries are throttled on
1237*43a90889SApple OSS Distributions 		 * a per-group basis to defeat denial-of-service attempts.
1238*43a90889SApple OSS Distributions 		 * Queries for groups we are not a member of on this
1239*43a90889SApple OSS Distributions 		 * link are simply ignored.
1240*43a90889SApple OSS Distributions 		 */
1241*43a90889SApple OSS Distributions 		in_multihead_lock_shared();
1242*43a90889SApple OSS Distributions 		IN_LOOKUP_MULTI(&igmpv3->igmp_group, ifp, inm);
1243*43a90889SApple OSS Distributions 		in_multihead_lock_done();
1244*43a90889SApple OSS Distributions 		if (inm == NULL) {
1245*43a90889SApple OSS Distributions 			goto done;
1246*43a90889SApple OSS Distributions 		}
1247*43a90889SApple OSS Distributions 
1248*43a90889SApple OSS Distributions 		INM_LOCK(inm);
1249*43a90889SApple OSS Distributions 		if (nsrc > 0) {
1250*43a90889SApple OSS Distributions 			if (!ratecheck(&inm->inm_lastgsrtv,
1251*43a90889SApple OSS Distributions 			    &igmp_gsrdelay)) {
1252*43a90889SApple OSS Distributions 				os_log_info(OS_LOG_DEFAULT, "%s: GS query throttled.\n",
1253*43a90889SApple OSS Distributions 				    __func__);
1254*43a90889SApple OSS Distributions 				IGMPSTAT_INC(igps_drop_gsr_queries);
1255*43a90889SApple OSS Distributions 				INM_UNLOCK(inm);
1256*43a90889SApple OSS Distributions 				INM_REMREF(inm); /* from IN_LOOKUP_MULTI */
1257*43a90889SApple OSS Distributions 				goto done;
1258*43a90889SApple OSS Distributions 			}
1259*43a90889SApple OSS Distributions 		}
1260*43a90889SApple OSS Distributions 		IGMP_INET_PRINTF(igmpv3->igmp_group,
1261*43a90889SApple OSS Distributions 		    ("process v3 %s query on ifp 0x%llx(%s)\n", _igmp_inet_buf,
1262*43a90889SApple OSS Distributions 		    (uint64_t)VM_KERNEL_ADDRPERM(ifp), if_name(ifp)));
1263*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: process v3 query on ifp %s\n",
1264*43a90889SApple OSS Distributions 		    __func__, if_name(ifp));
1265*43a90889SApple OSS Distributions 		/*
1266*43a90889SApple OSS Distributions 		 * If there is a pending General Query response
1267*43a90889SApple OSS Distributions 		 * scheduled sooner than the selected delay, no
1268*43a90889SApple OSS Distributions 		 * further report need be scheduled.
1269*43a90889SApple OSS Distributions 		 * Otherwise, prepare to respond to the
1270*43a90889SApple OSS Distributions 		 * group-specific or group-and-source query.
1271*43a90889SApple OSS Distributions 		 */
1272*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
1273*43a90889SApple OSS Distributions 		itp.it = igi->igi_v3_timer;
1274*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1275*43a90889SApple OSS Distributions 		if (itp.it == 0 || itp.it >= timer) {
1276*43a90889SApple OSS Distributions 			(void) igmp_input_v3_group_query(inm, timer, igmpv3);
1277*43a90889SApple OSS Distributions 			itp.cst = inm->inm_timer;
1278*43a90889SApple OSS Distributions 		}
1279*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
1280*43a90889SApple OSS Distributions 		INM_REMREF(inm); /* from IN_LOOKUP_MULTI */
1281*43a90889SApple OSS Distributions 	}
1282*43a90889SApple OSS Distributions done:
1283*43a90889SApple OSS Distributions 	if (itp.it > 0) {
1284*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: v3 general query response scheduled in "
1285*43a90889SApple OSS Distributions 		    "T+%d seconds on ifp %s\n", __func__, itp.it,
1286*43a90889SApple OSS Distributions 		    if_name(ifp));
1287*43a90889SApple OSS Distributions 	}
1288*43a90889SApple OSS Distributions 	igmp_set_timeout(&itp);
1289*43a90889SApple OSS Distributions 
1290*43a90889SApple OSS Distributions 	return 0;
1291*43a90889SApple OSS Distributions }
1292*43a90889SApple OSS Distributions 
1293*43a90889SApple OSS Distributions /*
1294*43a90889SApple OSS Distributions  * Process a recieved IGMPv3 group-specific or group-and-source-specific
1295*43a90889SApple OSS Distributions  * query.
1296*43a90889SApple OSS Distributions  * Return <0 if any error occured. Currently this is ignored.
1297*43a90889SApple OSS Distributions  */
1298*43a90889SApple OSS Distributions static int
igmp_input_v3_group_query(struct in_multi * inm,int timer,struct igmpv3 * __indexable igmpv3)1299*43a90889SApple OSS Distributions igmp_input_v3_group_query(struct in_multi *inm,
1300*43a90889SApple OSS Distributions     int timer, /*const*/ struct igmpv3 *__indexable igmpv3)
1301*43a90889SApple OSS Distributions {
1302*43a90889SApple OSS Distributions 	int                      retval;
1303*43a90889SApple OSS Distributions 	uint16_t                 nsrc;
1304*43a90889SApple OSS Distributions 
1305*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
1306*43a90889SApple OSS Distributions 
1307*43a90889SApple OSS Distributions 	retval = 0;
1308*43a90889SApple OSS Distributions 
1309*43a90889SApple OSS Distributions 	switch (inm->inm_state) {
1310*43a90889SApple OSS Distributions 	case IGMP_NOT_MEMBER:
1311*43a90889SApple OSS Distributions 	case IGMP_SILENT_MEMBER:
1312*43a90889SApple OSS Distributions 	case IGMP_SLEEPING_MEMBER:
1313*43a90889SApple OSS Distributions 	case IGMP_LAZY_MEMBER:
1314*43a90889SApple OSS Distributions 	case IGMP_AWAKENING_MEMBER:
1315*43a90889SApple OSS Distributions 	case IGMP_IDLE_MEMBER:
1316*43a90889SApple OSS Distributions 	case IGMP_LEAVING_MEMBER:
1317*43a90889SApple OSS Distributions 		return retval;
1318*43a90889SApple OSS Distributions 	case IGMP_REPORTING_MEMBER:
1319*43a90889SApple OSS Distributions 	case IGMP_G_QUERY_PENDING_MEMBER:
1320*43a90889SApple OSS Distributions 	case IGMP_SG_QUERY_PENDING_MEMBER:
1321*43a90889SApple OSS Distributions 		break;
1322*43a90889SApple OSS Distributions 	}
1323*43a90889SApple OSS Distributions 
1324*43a90889SApple OSS Distributions 	nsrc = ntohs(igmpv3->igmp_numsrc);
1325*43a90889SApple OSS Distributions 
1326*43a90889SApple OSS Distributions 	/*
1327*43a90889SApple OSS Distributions 	 * Deal with group-specific queries upfront.
1328*43a90889SApple OSS Distributions 	 * If any group query is already pending, purge any recorded
1329*43a90889SApple OSS Distributions 	 * source-list state if it exists, and schedule a query response
1330*43a90889SApple OSS Distributions 	 * for this group-specific query.
1331*43a90889SApple OSS Distributions 	 */
1332*43a90889SApple OSS Distributions 	if (nsrc == 0) {
1333*43a90889SApple OSS Distributions 		if (inm->inm_state == IGMP_G_QUERY_PENDING_MEMBER ||
1334*43a90889SApple OSS Distributions 		    inm->inm_state == IGMP_SG_QUERY_PENDING_MEMBER) {
1335*43a90889SApple OSS Distributions 			inm_clear_recorded(inm);
1336*43a90889SApple OSS Distributions 			timer = min(inm->inm_timer, timer);
1337*43a90889SApple OSS Distributions 		}
1338*43a90889SApple OSS Distributions 		inm->inm_state = IGMP_G_QUERY_PENDING_MEMBER;
1339*43a90889SApple OSS Distributions 		inm->inm_timer = IGMP_RANDOM_DELAY(timer);
1340*43a90889SApple OSS Distributions 		return retval;
1341*43a90889SApple OSS Distributions 	}
1342*43a90889SApple OSS Distributions 
1343*43a90889SApple OSS Distributions 	/*
1344*43a90889SApple OSS Distributions 	 * Deal with the case where a group-and-source-specific query has
1345*43a90889SApple OSS Distributions 	 * been received but a group-specific query is already pending.
1346*43a90889SApple OSS Distributions 	 */
1347*43a90889SApple OSS Distributions 	if (inm->inm_state == IGMP_G_QUERY_PENDING_MEMBER) {
1348*43a90889SApple OSS Distributions 		timer = min(inm->inm_timer, timer);
1349*43a90889SApple OSS Distributions 		inm->inm_timer = IGMP_RANDOM_DELAY(timer);
1350*43a90889SApple OSS Distributions 		return retval;
1351*43a90889SApple OSS Distributions 	}
1352*43a90889SApple OSS Distributions 
1353*43a90889SApple OSS Distributions 	/*
1354*43a90889SApple OSS Distributions 	 * Finally, deal with the case where a group-and-source-specific
1355*43a90889SApple OSS Distributions 	 * query has been received, where a response to a previous g-s-r
1356*43a90889SApple OSS Distributions 	 * query exists, or none exists.
1357*43a90889SApple OSS Distributions 	 * In this case, we need to parse the source-list which the Querier
1358*43a90889SApple OSS Distributions 	 * has provided us with and check if we have any source list filter
1359*43a90889SApple OSS Distributions 	 * entries at T1 for these sources. If we do not, there is no need
1360*43a90889SApple OSS Distributions 	 * schedule a report and the query may be dropped.
1361*43a90889SApple OSS Distributions 	 * If we do, we must record them and schedule a current-state
1362*43a90889SApple OSS Distributions 	 * report for those sources.
1363*43a90889SApple OSS Distributions 	 * FIXME: Handling source lists larger than 1 mbuf requires that
1364*43a90889SApple OSS Distributions 	 * we pass the mbuf chain pointer down to this function, and use
1365*43a90889SApple OSS Distributions 	 * m_getptr() to walk the chain.
1366*43a90889SApple OSS Distributions 	 */
1367*43a90889SApple OSS Distributions 	if (inm->inm_nsrc > 0) {
1368*43a90889SApple OSS Distributions 		const struct in_addr    *ap;
1369*43a90889SApple OSS Distributions 		int                      i, nrecorded;
1370*43a90889SApple OSS Distributions 
1371*43a90889SApple OSS Distributions 		ap = (const struct in_addr *)(igmpv3 + 1);
1372*43a90889SApple OSS Distributions 		nrecorded = 0;
1373*43a90889SApple OSS Distributions 		for (i = 0; i < nsrc; i++, ap++) {
1374*43a90889SApple OSS Distributions 			retval = inm_record_source(inm, ap->s_addr);
1375*43a90889SApple OSS Distributions 			if (retval < 0) {
1376*43a90889SApple OSS Distributions 				break;
1377*43a90889SApple OSS Distributions 			}
1378*43a90889SApple OSS Distributions 			nrecorded += retval;
1379*43a90889SApple OSS Distributions 		}
1380*43a90889SApple OSS Distributions 		if (nrecorded > 0) {
1381*43a90889SApple OSS Distributions 			os_log_debug(OS_LOG_DEFAULT, "%s: schedule response to SG query\n",
1382*43a90889SApple OSS Distributions 			    __func__);
1383*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_SG_QUERY_PENDING_MEMBER;
1384*43a90889SApple OSS Distributions 			inm->inm_timer = IGMP_RANDOM_DELAY(timer);
1385*43a90889SApple OSS Distributions 		}
1386*43a90889SApple OSS Distributions 	}
1387*43a90889SApple OSS Distributions 
1388*43a90889SApple OSS Distributions 	return retval;
1389*43a90889SApple OSS Distributions }
1390*43a90889SApple OSS Distributions 
1391*43a90889SApple OSS Distributions /*
1392*43a90889SApple OSS Distributions  * Process a received IGMPv1 host membership report.
1393*43a90889SApple OSS Distributions  *
1394*43a90889SApple OSS Distributions  * NOTE: 0.0.0.0 workaround breaks const correctness.
1395*43a90889SApple OSS Distributions  */
1396*43a90889SApple OSS Distributions static int
igmp_input_v1_report(struct ifnet * ifp,struct mbuf * m,struct ip * ip,struct igmp * igmp)1397*43a90889SApple OSS Distributions igmp_input_v1_report(struct ifnet *ifp, struct mbuf *m, /*const*/ struct ip *ip,
1398*43a90889SApple OSS Distributions     /*const*/ struct igmp *igmp)
1399*43a90889SApple OSS Distributions {
1400*43a90889SApple OSS Distributions 	struct in_ifaddr *ia;
1401*43a90889SApple OSS Distributions 	struct in_multi *inm;
1402*43a90889SApple OSS Distributions 
1403*43a90889SApple OSS Distributions 	IGMPSTAT_INC(igps_rcv_reports);
1404*43a90889SApple OSS Distributions 	OIGMPSTAT_INC(igps_rcv_reports);
1405*43a90889SApple OSS Distributions 
1406*43a90889SApple OSS Distributions 	if ((ifp->if_flags & IFF_LOOPBACK) ||
1407*43a90889SApple OSS Distributions 	    (m->m_pkthdr.pkt_flags & PKTF_LOOP)) {
1408*43a90889SApple OSS Distributions 		return 0;
1409*43a90889SApple OSS Distributions 	}
1410*43a90889SApple OSS Distributions 
1411*43a90889SApple OSS Distributions 	if (!IN_MULTICAST(ntohl(igmp->igmp_group.s_addr) ||
1412*43a90889SApple OSS Distributions 	    !in_hosteq(igmp->igmp_group, ip->ip_dst))) {
1413*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_badreports);
1414*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_badreports);
1415*43a90889SApple OSS Distributions 		return EINVAL;
1416*43a90889SApple OSS Distributions 	}
1417*43a90889SApple OSS Distributions 
1418*43a90889SApple OSS Distributions 	/*
1419*43a90889SApple OSS Distributions 	 * RFC 3376, Section 4.2.13, 9.2, 9.3:
1420*43a90889SApple OSS Distributions 	 * Booting clients may use the source address 0.0.0.0. Some
1421*43a90889SApple OSS Distributions 	 * IGMP daemons may not know how to use IP_RECVIF to determine
1422*43a90889SApple OSS Distributions 	 * the interface upon which this message was received.
1423*43a90889SApple OSS Distributions 	 * Replace 0.0.0.0 with the subnet address if told to do so.
1424*43a90889SApple OSS Distributions 	 */
1425*43a90889SApple OSS Distributions 	if (igmp_recvifkludge && in_nullhost(ip->ip_src)) {
1426*43a90889SApple OSS Distributions 		IFP_TO_IA(ifp, ia);
1427*43a90889SApple OSS Distributions 		if (ia != NULL) {
1428*43a90889SApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
1429*43a90889SApple OSS Distributions 			ip->ip_src.s_addr = htonl(ia->ia_subnet);
1430*43a90889SApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
1431*43a90889SApple OSS Distributions 			ifa_remref(&ia->ia_ifa);
1432*43a90889SApple OSS Distributions 		}
1433*43a90889SApple OSS Distributions 	}
1434*43a90889SApple OSS Distributions 
1435*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(igmp->igmp_group,
1436*43a90889SApple OSS Distributions 	    ("process v1 report %s on ifp 0x%llx(%s)\n", _igmp_inet_buf,
1437*43a90889SApple OSS Distributions 	    (uint64_t)VM_KERNEL_ADDRPERM(ifp), if_name(ifp)));
1438*43a90889SApple OSS Distributions 
1439*43a90889SApple OSS Distributions 	/*
1440*43a90889SApple OSS Distributions 	 * IGMPv1 report suppression.
1441*43a90889SApple OSS Distributions 	 * If we are a member of this group, and our membership should be
1442*43a90889SApple OSS Distributions 	 * reported, stop our group timer and transition to the 'lazy' state.
1443*43a90889SApple OSS Distributions 	 */
1444*43a90889SApple OSS Distributions 	in_multihead_lock_shared();
1445*43a90889SApple OSS Distributions 	IN_LOOKUP_MULTI(&igmp->igmp_group, ifp, inm);
1446*43a90889SApple OSS Distributions 	in_multihead_lock_done();
1447*43a90889SApple OSS Distributions 	if (inm != NULL) {
1448*43a90889SApple OSS Distributions 		struct igmp_ifinfo *igi;
1449*43a90889SApple OSS Distributions 
1450*43a90889SApple OSS Distributions 		INM_LOCK(inm);
1451*43a90889SApple OSS Distributions 
1452*43a90889SApple OSS Distributions 		igi = inm->inm_igi;
1453*43a90889SApple OSS Distributions 		VERIFY(igi != NULL);
1454*43a90889SApple OSS Distributions 
1455*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_ourreports);
1456*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_ourreports);
1457*43a90889SApple OSS Distributions 
1458*43a90889SApple OSS Distributions 		/*
1459*43a90889SApple OSS Distributions 		 * If we are in IGMPv3 host mode, do not allow the
1460*43a90889SApple OSS Distributions 		 * other host's IGMPv1 report to suppress our reports
1461*43a90889SApple OSS Distributions 		 * unless explicitly configured to do so.
1462*43a90889SApple OSS Distributions 		 */
1463*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
1464*43a90889SApple OSS Distributions 		if (igi->igi_version == IGMP_VERSION_3) {
1465*43a90889SApple OSS Distributions 			if (igmp_legacysupp) {
1466*43a90889SApple OSS Distributions 				igmp_v3_suppress_group_record(inm);
1467*43a90889SApple OSS Distributions 			}
1468*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
1469*43a90889SApple OSS Distributions 			INM_UNLOCK(inm);
1470*43a90889SApple OSS Distributions 			INM_REMREF(inm); /* from IN_LOOKUP_MULTI */
1471*43a90889SApple OSS Distributions 			return 0;
1472*43a90889SApple OSS Distributions 		}
1473*43a90889SApple OSS Distributions 
1474*43a90889SApple OSS Distributions 		INM_LOCK_ASSERT_HELD(inm);
1475*43a90889SApple OSS Distributions 		inm->inm_timer = 0;
1476*43a90889SApple OSS Distributions 
1477*43a90889SApple OSS Distributions 		switch (inm->inm_state) {
1478*43a90889SApple OSS Distributions 		case IGMP_NOT_MEMBER:
1479*43a90889SApple OSS Distributions 		case IGMP_SILENT_MEMBER:
1480*43a90889SApple OSS Distributions 			break;
1481*43a90889SApple OSS Distributions 		case IGMP_IDLE_MEMBER:
1482*43a90889SApple OSS Distributions 		case IGMP_LAZY_MEMBER:
1483*43a90889SApple OSS Distributions 		case IGMP_AWAKENING_MEMBER:
1484*43a90889SApple OSS Distributions 			IGMP_INET_PRINTF(igmp->igmp_group,
1485*43a90889SApple OSS Distributions 			    ("report suppressed for %s on ifp 0x%llx(%s)\n",
1486*43a90889SApple OSS Distributions 			    _igmp_inet_buf,
1487*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(ifp), if_name(ifp)));
1488*43a90889SApple OSS Distributions 			OS_FALLTHROUGH;
1489*43a90889SApple OSS Distributions 		case IGMP_SLEEPING_MEMBER:
1490*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_SLEEPING_MEMBER;
1491*43a90889SApple OSS Distributions 			break;
1492*43a90889SApple OSS Distributions 		case IGMP_REPORTING_MEMBER:
1493*43a90889SApple OSS Distributions 			IGMP_INET_PRINTF(igmp->igmp_group,
1494*43a90889SApple OSS Distributions 			    ("report suppressed for %s on ifp 0x%llx(%s)\n",
1495*43a90889SApple OSS Distributions 			    _igmp_inet_buf,
1496*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(ifp), if_name(ifp)));
1497*43a90889SApple OSS Distributions 			if (igi->igi_version == IGMP_VERSION_1) {
1498*43a90889SApple OSS Distributions 				inm->inm_state = IGMP_LAZY_MEMBER;
1499*43a90889SApple OSS Distributions 			} else if (igi->igi_version == IGMP_VERSION_2) {
1500*43a90889SApple OSS Distributions 				inm->inm_state = IGMP_SLEEPING_MEMBER;
1501*43a90889SApple OSS Distributions 			}
1502*43a90889SApple OSS Distributions 			break;
1503*43a90889SApple OSS Distributions 		case IGMP_G_QUERY_PENDING_MEMBER:
1504*43a90889SApple OSS Distributions 		case IGMP_SG_QUERY_PENDING_MEMBER:
1505*43a90889SApple OSS Distributions 		case IGMP_LEAVING_MEMBER:
1506*43a90889SApple OSS Distributions 			break;
1507*43a90889SApple OSS Distributions 		}
1508*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1509*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
1510*43a90889SApple OSS Distributions 		INM_REMREF(inm); /* from IN_LOOKUP_MULTI */
1511*43a90889SApple OSS Distributions 	}
1512*43a90889SApple OSS Distributions 
1513*43a90889SApple OSS Distributions 	return 0;
1514*43a90889SApple OSS Distributions }
1515*43a90889SApple OSS Distributions 
1516*43a90889SApple OSS Distributions /*
1517*43a90889SApple OSS Distributions  * Process a received IGMPv2 host membership report.
1518*43a90889SApple OSS Distributions  *
1519*43a90889SApple OSS Distributions  * NOTE: 0.0.0.0 workaround breaks const correctness.
1520*43a90889SApple OSS Distributions  */
1521*43a90889SApple OSS Distributions static int
igmp_input_v2_report(struct ifnet * ifp,struct mbuf * m,struct ip * ip,struct igmp * igmp)1522*43a90889SApple OSS Distributions igmp_input_v2_report(struct ifnet *ifp, struct mbuf *m, /*const*/ struct ip *ip,
1523*43a90889SApple OSS Distributions     /*const*/ struct igmp *igmp)
1524*43a90889SApple OSS Distributions {
1525*43a90889SApple OSS Distributions 	struct in_ifaddr *ia;
1526*43a90889SApple OSS Distributions 	struct in_multi *inm;
1527*43a90889SApple OSS Distributions 
1528*43a90889SApple OSS Distributions 	/*
1529*43a90889SApple OSS Distributions 	 * Make sure we don't hear our own membership report.  Fast
1530*43a90889SApple OSS Distributions 	 * leave requires knowing that we are the only member of a
1531*43a90889SApple OSS Distributions 	 * group.
1532*43a90889SApple OSS Distributions 	 */
1533*43a90889SApple OSS Distributions 	IFP_TO_IA(ifp, ia);
1534*43a90889SApple OSS Distributions 	if (ia != NULL) {
1535*43a90889SApple OSS Distributions 		IFA_LOCK(&ia->ia_ifa);
1536*43a90889SApple OSS Distributions 		if (in_hosteq(ip->ip_src, IA_SIN(ia)->sin_addr)) {
1537*43a90889SApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
1538*43a90889SApple OSS Distributions 			ifa_remref(&ia->ia_ifa);
1539*43a90889SApple OSS Distributions 			return 0;
1540*43a90889SApple OSS Distributions 		}
1541*43a90889SApple OSS Distributions 		IFA_UNLOCK(&ia->ia_ifa);
1542*43a90889SApple OSS Distributions 	}
1543*43a90889SApple OSS Distributions 
1544*43a90889SApple OSS Distributions 	IGMPSTAT_INC(igps_rcv_reports);
1545*43a90889SApple OSS Distributions 	OIGMPSTAT_INC(igps_rcv_reports);
1546*43a90889SApple OSS Distributions 
1547*43a90889SApple OSS Distributions 	if ((ifp->if_flags & IFF_LOOPBACK) ||
1548*43a90889SApple OSS Distributions 	    (m->m_pkthdr.pkt_flags & PKTF_LOOP)) {
1549*43a90889SApple OSS Distributions 		if (ia != NULL) {
1550*43a90889SApple OSS Distributions 			ifa_remref(&ia->ia_ifa);
1551*43a90889SApple OSS Distributions 		}
1552*43a90889SApple OSS Distributions 		return 0;
1553*43a90889SApple OSS Distributions 	}
1554*43a90889SApple OSS Distributions 
1555*43a90889SApple OSS Distributions 	if (!IN_MULTICAST(ntohl(igmp->igmp_group.s_addr)) ||
1556*43a90889SApple OSS Distributions 	    !in_hosteq(igmp->igmp_group, ip->ip_dst)) {
1557*43a90889SApple OSS Distributions 		if (ia != NULL) {
1558*43a90889SApple OSS Distributions 			ifa_remref(&ia->ia_ifa);
1559*43a90889SApple OSS Distributions 		}
1560*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_badreports);
1561*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_badreports);
1562*43a90889SApple OSS Distributions 		return EINVAL;
1563*43a90889SApple OSS Distributions 	}
1564*43a90889SApple OSS Distributions 
1565*43a90889SApple OSS Distributions 	/*
1566*43a90889SApple OSS Distributions 	 * RFC 3376, Section 4.2.13, 9.2, 9.3:
1567*43a90889SApple OSS Distributions 	 * Booting clients may use the source address 0.0.0.0. Some
1568*43a90889SApple OSS Distributions 	 * IGMP daemons may not know how to use IP_RECVIF to determine
1569*43a90889SApple OSS Distributions 	 * the interface upon which this message was received.
1570*43a90889SApple OSS Distributions 	 * Replace 0.0.0.0 with the subnet address if told to do so.
1571*43a90889SApple OSS Distributions 	 */
1572*43a90889SApple OSS Distributions 	if (igmp_recvifkludge && in_nullhost(ip->ip_src)) {
1573*43a90889SApple OSS Distributions 		if (ia != NULL) {
1574*43a90889SApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
1575*43a90889SApple OSS Distributions 			ip->ip_src.s_addr = htonl(ia->ia_subnet);
1576*43a90889SApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
1577*43a90889SApple OSS Distributions 		}
1578*43a90889SApple OSS Distributions 	}
1579*43a90889SApple OSS Distributions 	if (ia != NULL) {
1580*43a90889SApple OSS Distributions 		ifa_remref(&ia->ia_ifa);
1581*43a90889SApple OSS Distributions 	}
1582*43a90889SApple OSS Distributions 
1583*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(igmp->igmp_group,
1584*43a90889SApple OSS Distributions 	    ("process v2 report %s on ifp 0x%llx(%s)\n", _igmp_inet_buf,
1585*43a90889SApple OSS Distributions 	    (uint64_t)VM_KERNEL_ADDRPERM(ifp), if_name(ifp)));
1586*43a90889SApple OSS Distributions 	os_log_debug(OS_LOG_DEFAULT, "%s: process v2 report on ifp %s",
1587*43a90889SApple OSS Distributions 	    __func__, if_name(ifp));
1588*43a90889SApple OSS Distributions 
1589*43a90889SApple OSS Distributions 	/*
1590*43a90889SApple OSS Distributions 	 * IGMPv2 report suppression.
1591*43a90889SApple OSS Distributions 	 * If we are a member of this group, and our membership should be
1592*43a90889SApple OSS Distributions 	 * reported, and our group timer is pending or about to be reset,
1593*43a90889SApple OSS Distributions 	 * stop our group timer by transitioning to the 'lazy' state.
1594*43a90889SApple OSS Distributions 	 */
1595*43a90889SApple OSS Distributions 	in_multihead_lock_shared();
1596*43a90889SApple OSS Distributions 	IN_LOOKUP_MULTI(&igmp->igmp_group, ifp, inm);
1597*43a90889SApple OSS Distributions 	in_multihead_lock_done();
1598*43a90889SApple OSS Distributions 	if (inm != NULL) {
1599*43a90889SApple OSS Distributions 		struct igmp_ifinfo *igi;
1600*43a90889SApple OSS Distributions 
1601*43a90889SApple OSS Distributions 		INM_LOCK(inm);
1602*43a90889SApple OSS Distributions 		igi = inm->inm_igi;
1603*43a90889SApple OSS Distributions 		VERIFY(igi != NULL);
1604*43a90889SApple OSS Distributions 
1605*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_ourreports);
1606*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_ourreports);
1607*43a90889SApple OSS Distributions 
1608*43a90889SApple OSS Distributions 		/*
1609*43a90889SApple OSS Distributions 		 * If we are in IGMPv3 host mode, do not allow the
1610*43a90889SApple OSS Distributions 		 * other host's IGMPv1 report to suppress our reports
1611*43a90889SApple OSS Distributions 		 * unless explicitly configured to do so.
1612*43a90889SApple OSS Distributions 		 */
1613*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
1614*43a90889SApple OSS Distributions 		if (igi->igi_version == IGMP_VERSION_3) {
1615*43a90889SApple OSS Distributions 			if (igmp_legacysupp) {
1616*43a90889SApple OSS Distributions 				igmp_v3_suppress_group_record(inm);
1617*43a90889SApple OSS Distributions 			}
1618*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
1619*43a90889SApple OSS Distributions 			INM_UNLOCK(inm);
1620*43a90889SApple OSS Distributions 			INM_REMREF(inm);
1621*43a90889SApple OSS Distributions 			return 0;
1622*43a90889SApple OSS Distributions 		}
1623*43a90889SApple OSS Distributions 
1624*43a90889SApple OSS Distributions 		inm->inm_timer = 0;
1625*43a90889SApple OSS Distributions 
1626*43a90889SApple OSS Distributions 		switch (inm->inm_state) {
1627*43a90889SApple OSS Distributions 		case IGMP_NOT_MEMBER:
1628*43a90889SApple OSS Distributions 		case IGMP_SILENT_MEMBER:
1629*43a90889SApple OSS Distributions 		case IGMP_SLEEPING_MEMBER:
1630*43a90889SApple OSS Distributions 			break;
1631*43a90889SApple OSS Distributions 		case IGMP_REPORTING_MEMBER:
1632*43a90889SApple OSS Distributions 		case IGMP_IDLE_MEMBER:
1633*43a90889SApple OSS Distributions 		case IGMP_AWAKENING_MEMBER:
1634*43a90889SApple OSS Distributions 			IGMP_INET_PRINTF(igmp->igmp_group,
1635*43a90889SApple OSS Distributions 			    ("report suppressed for %s on ifp 0x%llx(%s)\n",
1636*43a90889SApple OSS Distributions 			    _igmp_inet_buf, (uint64_t)VM_KERNEL_ADDRPERM(ifp),
1637*43a90889SApple OSS Distributions 			    if_name(ifp)));
1638*43a90889SApple OSS Distributions 			OS_FALLTHROUGH;
1639*43a90889SApple OSS Distributions 		case IGMP_LAZY_MEMBER:
1640*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_LAZY_MEMBER;
1641*43a90889SApple OSS Distributions 			break;
1642*43a90889SApple OSS Distributions 		case IGMP_G_QUERY_PENDING_MEMBER:
1643*43a90889SApple OSS Distributions 		case IGMP_SG_QUERY_PENDING_MEMBER:
1644*43a90889SApple OSS Distributions 		case IGMP_LEAVING_MEMBER:
1645*43a90889SApple OSS Distributions 			break;
1646*43a90889SApple OSS Distributions 		}
1647*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
1648*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
1649*43a90889SApple OSS Distributions 		INM_REMREF(inm);
1650*43a90889SApple OSS Distributions 	}
1651*43a90889SApple OSS Distributions 
1652*43a90889SApple OSS Distributions 	return 0;
1653*43a90889SApple OSS Distributions }
1654*43a90889SApple OSS Distributions 
1655*43a90889SApple OSS Distributions void
igmp_input(struct mbuf * m,int off)1656*43a90889SApple OSS Distributions igmp_input(struct mbuf *m, int off)
1657*43a90889SApple OSS Distributions {
1658*43a90889SApple OSS Distributions 	int iphlen;
1659*43a90889SApple OSS Distributions 	struct ifnet *ifp;
1660*43a90889SApple OSS Distributions 	struct igmp *igmp;
1661*43a90889SApple OSS Distributions 	struct ip *ip;
1662*43a90889SApple OSS Distributions 	int igmplen;
1663*43a90889SApple OSS Distributions 	int minlen;
1664*43a90889SApple OSS Distributions 	int queryver;
1665*43a90889SApple OSS Distributions 
1666*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: called w/mbuf(0x%llx,%d)\n", __func__,
1667*43a90889SApple OSS Distributions 	    (uint64_t)VM_KERNEL_ADDRPERM(m), off));
1668*43a90889SApple OSS Distributions 
1669*43a90889SApple OSS Distributions 	ifp = m->m_pkthdr.rcvif;
1670*43a90889SApple OSS Distributions 
1671*43a90889SApple OSS Distributions 	IGMPSTAT_INC(igps_rcv_total);
1672*43a90889SApple OSS Distributions 	OIGMPSTAT_INC(igps_rcv_total);
1673*43a90889SApple OSS Distributions 
1674*43a90889SApple OSS Distributions 	/* Expect 32-bit aligned data pointer on strict-align platforms */
1675*43a90889SApple OSS Distributions 	MBUF_STRICT_DATA_ALIGNMENT_CHECK_32(m);
1676*43a90889SApple OSS Distributions 
1677*43a90889SApple OSS Distributions 	ip = mtod(m, struct ip *);
1678*43a90889SApple OSS Distributions 	iphlen = off;
1679*43a90889SApple OSS Distributions 
1680*43a90889SApple OSS Distributions 	/* By now, ip_len no longer contains the length of IP header */
1681*43a90889SApple OSS Distributions 	igmplen = ip->ip_len;
1682*43a90889SApple OSS Distributions 
1683*43a90889SApple OSS Distributions 	/*
1684*43a90889SApple OSS Distributions 	 * Validate lengths.
1685*43a90889SApple OSS Distributions 	 */
1686*43a90889SApple OSS Distributions 	if (igmplen < IGMP_MINLEN) {
1687*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_tooshort);
1688*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_tooshort);
1689*43a90889SApple OSS Distributions 		m_freem(m);
1690*43a90889SApple OSS Distributions 		return;
1691*43a90889SApple OSS Distributions 	}
1692*43a90889SApple OSS Distributions 
1693*43a90889SApple OSS Distributions 	/*
1694*43a90889SApple OSS Distributions 	 * Always pullup to the minimum size for v1/v2 or v3
1695*43a90889SApple OSS Distributions 	 * to amortize calls to m_pulldown().
1696*43a90889SApple OSS Distributions 	 */
1697*43a90889SApple OSS Distributions 	if (igmplen >= IGMP_V3_QUERY_MINLEN) {
1698*43a90889SApple OSS Distributions 		minlen = IGMP_V3_QUERY_MINLEN;
1699*43a90889SApple OSS Distributions 	} else {
1700*43a90889SApple OSS Distributions 		minlen = IGMP_MINLEN;
1701*43a90889SApple OSS Distributions 	}
1702*43a90889SApple OSS Distributions 
1703*43a90889SApple OSS Distributions 	/* A bit more expensive than M_STRUCT_GET, but ensures alignment */
1704*43a90889SApple OSS Distributions 	M_STRUCT_GET0(igmp, struct igmp *, m, off, minlen);
1705*43a90889SApple OSS Distributions 	if (igmp == NULL) {
1706*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_tooshort);
1707*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_tooshort);
1708*43a90889SApple OSS Distributions 		return;
1709*43a90889SApple OSS Distributions 	}
1710*43a90889SApple OSS Distributions 	/* N.B.: we assume the packet was correctly aligned in ip_input. */
1711*43a90889SApple OSS Distributions 
1712*43a90889SApple OSS Distributions 	/*
1713*43a90889SApple OSS Distributions 	 * Validate checksum.
1714*43a90889SApple OSS Distributions 	 */
1715*43a90889SApple OSS Distributions 	m->m_data += iphlen;
1716*43a90889SApple OSS Distributions 	m->m_len -= iphlen;
1717*43a90889SApple OSS Distributions 	if (in_cksum(m, igmplen)) {
1718*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_badsum);
1719*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_badsum);
1720*43a90889SApple OSS Distributions 		m_freem(m);
1721*43a90889SApple OSS Distributions 		return;
1722*43a90889SApple OSS Distributions 	}
1723*43a90889SApple OSS Distributions 	m->m_data -= iphlen;
1724*43a90889SApple OSS Distributions 	m->m_len += iphlen;
1725*43a90889SApple OSS Distributions 
1726*43a90889SApple OSS Distributions 	/*
1727*43a90889SApple OSS Distributions 	 * IGMP control traffic is link-scope, and must have a TTL of 1.
1728*43a90889SApple OSS Distributions 	 * DVMRP traffic (e.g. mrinfo, mtrace) is an exception;
1729*43a90889SApple OSS Distributions 	 * probe packets may come from beyond the LAN.
1730*43a90889SApple OSS Distributions 	 */
1731*43a90889SApple OSS Distributions 	if (igmp->igmp_type != IGMP_DVMRP && ip->ip_ttl != 1) {
1732*43a90889SApple OSS Distributions 		IGMPSTAT_INC(igps_rcv_badttl);
1733*43a90889SApple OSS Distributions 		m_freem(m);
1734*43a90889SApple OSS Distributions 		return;
1735*43a90889SApple OSS Distributions 	}
1736*43a90889SApple OSS Distributions 
1737*43a90889SApple OSS Distributions 	switch (igmp->igmp_type) {
1738*43a90889SApple OSS Distributions 	case IGMP_HOST_MEMBERSHIP_QUERY:
1739*43a90889SApple OSS Distributions 		if (igmplen == IGMP_MINLEN) {
1740*43a90889SApple OSS Distributions 			if (igmp->igmp_code == 0) {
1741*43a90889SApple OSS Distributions 				queryver = IGMP_VERSION_1;
1742*43a90889SApple OSS Distributions 			} else {
1743*43a90889SApple OSS Distributions 				queryver = IGMP_VERSION_2;
1744*43a90889SApple OSS Distributions 			}
1745*43a90889SApple OSS Distributions 		} else if (igmplen >= IGMP_V3_QUERY_MINLEN) {
1746*43a90889SApple OSS Distributions 			queryver = IGMP_VERSION_3;
1747*43a90889SApple OSS Distributions 		} else {
1748*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_tooshort);
1749*43a90889SApple OSS Distributions 			OIGMPSTAT_INC(igps_rcv_tooshort);
1750*43a90889SApple OSS Distributions 			m_freem(m);
1751*43a90889SApple OSS Distributions 			return;
1752*43a90889SApple OSS Distributions 		}
1753*43a90889SApple OSS Distributions 
1754*43a90889SApple OSS Distributions 		OIGMPSTAT_INC(igps_rcv_queries);
1755*43a90889SApple OSS Distributions 
1756*43a90889SApple OSS Distributions 		switch (queryver) {
1757*43a90889SApple OSS Distributions 		case IGMP_VERSION_1:
1758*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_v1v2_queries);
1759*43a90889SApple OSS Distributions 			if (!igmp_v1enable) {
1760*43a90889SApple OSS Distributions 				break;
1761*43a90889SApple OSS Distributions 			}
1762*43a90889SApple OSS Distributions 			if (igmp_input_v1_query(ifp, ip, igmp) != 0) {
1763*43a90889SApple OSS Distributions 				m_freem(m);
1764*43a90889SApple OSS Distributions 				return;
1765*43a90889SApple OSS Distributions 			}
1766*43a90889SApple OSS Distributions 			break;
1767*43a90889SApple OSS Distributions 
1768*43a90889SApple OSS Distributions 		case IGMP_VERSION_2:
1769*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_v1v2_queries);
1770*43a90889SApple OSS Distributions 			if (!igmp_v2enable) {
1771*43a90889SApple OSS Distributions 				break;
1772*43a90889SApple OSS Distributions 			}
1773*43a90889SApple OSS Distributions 			if (igmp_input_v2_query(ifp, ip, igmp) != 0) {
1774*43a90889SApple OSS Distributions 				m_freem(m);
1775*43a90889SApple OSS Distributions 				return;
1776*43a90889SApple OSS Distributions 			}
1777*43a90889SApple OSS Distributions 			break;
1778*43a90889SApple OSS Distributions 
1779*43a90889SApple OSS Distributions 		case IGMP_VERSION_3: {
1780*43a90889SApple OSS Distributions 			struct igmpv3 *igmpv3;
1781*43a90889SApple OSS Distributions 			uint16_t igmpv3len;
1782*43a90889SApple OSS Distributions 			uint16_t srclen;
1783*43a90889SApple OSS Distributions 			int nsrc;
1784*43a90889SApple OSS Distributions 
1785*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_v3_queries);
1786*43a90889SApple OSS Distributions 			igmpv3 = (struct igmpv3 *)igmp;
1787*43a90889SApple OSS Distributions 			/*
1788*43a90889SApple OSS Distributions 			 * Validate length based on source count.
1789*43a90889SApple OSS Distributions 			 */
1790*43a90889SApple OSS Distributions 			nsrc = ntohs(igmpv3->igmp_numsrc);
1791*43a90889SApple OSS Distributions 			/*
1792*43a90889SApple OSS Distributions 			 * The max vaue of nsrc is limited by the
1793*43a90889SApple OSS Distributions 			 * MTU of the network on which the datagram
1794*43a90889SApple OSS Distributions 			 * is received
1795*43a90889SApple OSS Distributions 			 */
1796*43a90889SApple OSS Distributions 			if (nsrc < 0 || nsrc > IGMP_V3_QUERY_MAX_SRCS) {
1797*43a90889SApple OSS Distributions 				IGMPSTAT_INC(igps_rcv_tooshort);
1798*43a90889SApple OSS Distributions 				OIGMPSTAT_INC(igps_rcv_tooshort);
1799*43a90889SApple OSS Distributions 				m_freem(m);
1800*43a90889SApple OSS Distributions 				return;
1801*43a90889SApple OSS Distributions 			}
1802*43a90889SApple OSS Distributions 			srclen = sizeof(struct in_addr) * (uint16_t)nsrc;
1803*43a90889SApple OSS Distributions 			if (igmplen < (IGMP_V3_QUERY_MINLEN + srclen)) {
1804*43a90889SApple OSS Distributions 				IGMPSTAT_INC(igps_rcv_tooshort);
1805*43a90889SApple OSS Distributions 				OIGMPSTAT_INC(igps_rcv_tooshort);
1806*43a90889SApple OSS Distributions 				m_freem(m);
1807*43a90889SApple OSS Distributions 				return;
1808*43a90889SApple OSS Distributions 			}
1809*43a90889SApple OSS Distributions 			igmpv3len = IGMP_V3_QUERY_MINLEN + srclen;
1810*43a90889SApple OSS Distributions 			/*
1811*43a90889SApple OSS Distributions 			 * A bit more expensive than M_STRUCT_GET,
1812*43a90889SApple OSS Distributions 			 * but ensures alignment.
1813*43a90889SApple OSS Distributions 			 */
1814*43a90889SApple OSS Distributions 			M_STRUCT_GET0(igmpv3, struct igmpv3 *, m,
1815*43a90889SApple OSS Distributions 			    off, igmpv3len);
1816*43a90889SApple OSS Distributions 			if (igmpv3 == NULL) {
1817*43a90889SApple OSS Distributions 				IGMPSTAT_INC(igps_rcv_tooshort);
1818*43a90889SApple OSS Distributions 				OIGMPSTAT_INC(igps_rcv_tooshort);
1819*43a90889SApple OSS Distributions 				return;
1820*43a90889SApple OSS Distributions 			}
1821*43a90889SApple OSS Distributions 			/*
1822*43a90889SApple OSS Distributions 			 * N.B.: we assume the packet was correctly
1823*43a90889SApple OSS Distributions 			 * aligned in ip_input.
1824*43a90889SApple OSS Distributions 			 */
1825*43a90889SApple OSS Distributions 			if (igmp_input_v3_query(ifp, ip, igmpv3) != 0) {
1826*43a90889SApple OSS Distributions 				m_freem(m);
1827*43a90889SApple OSS Distributions 				return;
1828*43a90889SApple OSS Distributions 			}
1829*43a90889SApple OSS Distributions 		}
1830*43a90889SApple OSS Distributions 		break;
1831*43a90889SApple OSS Distributions 		}
1832*43a90889SApple OSS Distributions 		break;
1833*43a90889SApple OSS Distributions 
1834*43a90889SApple OSS Distributions 	case IGMP_v1_HOST_MEMBERSHIP_REPORT:
1835*43a90889SApple OSS Distributions 		if (!igmp_v1enable) {
1836*43a90889SApple OSS Distributions 			break;
1837*43a90889SApple OSS Distributions 		}
1838*43a90889SApple OSS Distributions 		if (igmp_input_v1_report(ifp, m, ip, igmp) != 0) {
1839*43a90889SApple OSS Distributions 			m_freem(m);
1840*43a90889SApple OSS Distributions 			return;
1841*43a90889SApple OSS Distributions 		}
1842*43a90889SApple OSS Distributions 		break;
1843*43a90889SApple OSS Distributions 
1844*43a90889SApple OSS Distributions 	case IGMP_v2_HOST_MEMBERSHIP_REPORT:
1845*43a90889SApple OSS Distributions 		if (!igmp_v2enable) {
1846*43a90889SApple OSS Distributions 			break;
1847*43a90889SApple OSS Distributions 		}
1848*43a90889SApple OSS Distributions 		if (!ip_checkrouteralert(m)) {
1849*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_nora);
1850*43a90889SApple OSS Distributions 		}
1851*43a90889SApple OSS Distributions 		if (igmp_input_v2_report(ifp, m, ip, igmp) != 0) {
1852*43a90889SApple OSS Distributions 			m_freem(m);
1853*43a90889SApple OSS Distributions 			return;
1854*43a90889SApple OSS Distributions 		}
1855*43a90889SApple OSS Distributions 		break;
1856*43a90889SApple OSS Distributions 
1857*43a90889SApple OSS Distributions 	case IGMP_v3_HOST_MEMBERSHIP_REPORT:
1858*43a90889SApple OSS Distributions 		/*
1859*43a90889SApple OSS Distributions 		 * Hosts do not need to process IGMPv3 membership reports,
1860*43a90889SApple OSS Distributions 		 * as report suppression is no longer required.
1861*43a90889SApple OSS Distributions 		 */
1862*43a90889SApple OSS Distributions 		if (!ip_checkrouteralert(m)) {
1863*43a90889SApple OSS Distributions 			IGMPSTAT_INC(igps_rcv_nora);
1864*43a90889SApple OSS Distributions 		}
1865*43a90889SApple OSS Distributions 		break;
1866*43a90889SApple OSS Distributions 
1867*43a90889SApple OSS Distributions 	default:
1868*43a90889SApple OSS Distributions 		break;
1869*43a90889SApple OSS Distributions 	}
1870*43a90889SApple OSS Distributions 
1871*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_NOTHELD();
1872*43a90889SApple OSS Distributions 	/*
1873*43a90889SApple OSS Distributions 	 * Pass all valid IGMP packets up to any process(es) listening on a
1874*43a90889SApple OSS Distributions 	 * raw IGMP socket.
1875*43a90889SApple OSS Distributions 	 */
1876*43a90889SApple OSS Distributions 	rip_input(m, off);
1877*43a90889SApple OSS Distributions }
1878*43a90889SApple OSS Distributions 
1879*43a90889SApple OSS Distributions /*
1880*43a90889SApple OSS Distributions  * Schedule IGMP timer based on various parameters; caller must ensure that
1881*43a90889SApple OSS Distributions  * lock ordering is maintained as this routine acquires IGMP global lock.
1882*43a90889SApple OSS Distributions  */
1883*43a90889SApple OSS Distributions void
igmp_set_timeout(struct igmp_tparams * itp)1884*43a90889SApple OSS Distributions igmp_set_timeout(struct igmp_tparams *itp)
1885*43a90889SApple OSS Distributions {
1886*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_NOTHELD();
1887*43a90889SApple OSS Distributions 	VERIFY(itp != NULL);
1888*43a90889SApple OSS Distributions 
1889*43a90889SApple OSS Distributions 	if (itp->qpt != 0 || itp->it != 0 || itp->cst != 0 || itp->sct != 0) {
1890*43a90889SApple OSS Distributions 		IGMP_LOCK();
1891*43a90889SApple OSS Distributions 		if (itp->qpt != 0) {
1892*43a90889SApple OSS Distributions 			querier_present_timers_running = 1;
1893*43a90889SApple OSS Distributions 		}
1894*43a90889SApple OSS Distributions 		if (itp->it != 0) {
1895*43a90889SApple OSS Distributions 			interface_timers_running = 1;
1896*43a90889SApple OSS Distributions 		}
1897*43a90889SApple OSS Distributions 		if (itp->cst != 0) {
1898*43a90889SApple OSS Distributions 			current_state_timers_running = 1;
1899*43a90889SApple OSS Distributions 		}
1900*43a90889SApple OSS Distributions 		if (itp->sct != 0) {
1901*43a90889SApple OSS Distributions 			state_change_timers_running = 1;
1902*43a90889SApple OSS Distributions 		}
1903*43a90889SApple OSS Distributions 		if (itp->fast) {
1904*43a90889SApple OSS Distributions 			igmp_sched_fast_timeout();
1905*43a90889SApple OSS Distributions 		} else {
1906*43a90889SApple OSS Distributions 			igmp_sched_timeout();
1907*43a90889SApple OSS Distributions 		}
1908*43a90889SApple OSS Distributions 		IGMP_UNLOCK();
1909*43a90889SApple OSS Distributions 	}
1910*43a90889SApple OSS Distributions }
1911*43a90889SApple OSS Distributions 
1912*43a90889SApple OSS Distributions void
igmp_set_fast_timeout(struct igmp_tparams * itp)1913*43a90889SApple OSS Distributions igmp_set_fast_timeout(struct igmp_tparams *itp)
1914*43a90889SApple OSS Distributions {
1915*43a90889SApple OSS Distributions 	VERIFY(itp != NULL);
1916*43a90889SApple OSS Distributions 	itp->fast = true;
1917*43a90889SApple OSS Distributions 	igmp_set_timeout(itp);
1918*43a90889SApple OSS Distributions }
1919*43a90889SApple OSS Distributions 
1920*43a90889SApple OSS Distributions /*
1921*43a90889SApple OSS Distributions  * IGMP timer handler (per 1 second).
1922*43a90889SApple OSS Distributions  */
1923*43a90889SApple OSS Distributions static void
igmp_timeout(thread_call_param_t arg0,thread_call_param_t arg1 __unused)1924*43a90889SApple OSS Distributions igmp_timeout(thread_call_param_t arg0, thread_call_param_t arg1 __unused)
1925*43a90889SApple OSS Distributions {
1926*43a90889SApple OSS Distributions 	struct ifqueue           scq;   /* State-change packets */
1927*43a90889SApple OSS Distributions 	struct ifqueue           qrq;   /* Query response packets */
1928*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
1929*43a90889SApple OSS Distributions 	struct igmp_ifinfo      *igi;
1930*43a90889SApple OSS Distributions 	struct in_multi         *inm;
1931*43a90889SApple OSS Distributions 	unsigned int             loop = 0, uri_sec = 0;
1932*43a90889SApple OSS Distributions 	SLIST_HEAD(, in_multi)  inm_dthead;
1933*43a90889SApple OSS Distributions 	bool                     fast = arg0 != NULL;
1934*43a90889SApple OSS Distributions 
1935*43a90889SApple OSS Distributions 	SLIST_INIT(&inm_dthead);
1936*43a90889SApple OSS Distributions 
1937*43a90889SApple OSS Distributions 	/*
1938*43a90889SApple OSS Distributions 	 * Update coarse-grained networking timestamp (in sec.); the idea
1939*43a90889SApple OSS Distributions 	 * is to piggy-back on the timeout callout to update the counter
1940*43a90889SApple OSS Distributions 	 * returnable via net_uptime().
1941*43a90889SApple OSS Distributions 	 */
1942*43a90889SApple OSS Distributions 	net_update_uptime();
1943*43a90889SApple OSS Distributions 
1944*43a90889SApple OSS Distributions 	IGMP_LOCK();
1945*43a90889SApple OSS Distributions 
1946*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: qpt %d, it %d, cst %d, sct %d, fast %d\n", __func__,
1947*43a90889SApple OSS Distributions 	    querier_present_timers_running, interface_timers_running,
1948*43a90889SApple OSS Distributions 	    current_state_timers_running, state_change_timers_running,
1949*43a90889SApple OSS Distributions 	    fast));
1950*43a90889SApple OSS Distributions 
1951*43a90889SApple OSS Distributions 	if (fast) {
1952*43a90889SApple OSS Distributions 		/*
1953*43a90889SApple OSS Distributions 		 * When running the fast timer, skip processing
1954*43a90889SApple OSS Distributions 		 * of "querier present" timers since they are
1955*43a90889SApple OSS Distributions 		 * based on 1-second intervals.
1956*43a90889SApple OSS Distributions 		 */
1957*43a90889SApple OSS Distributions 		goto skip_query_timers;
1958*43a90889SApple OSS Distributions 	}
1959*43a90889SApple OSS Distributions 	/*
1960*43a90889SApple OSS Distributions 	 * IGMPv1/v2 querier present timer processing.
1961*43a90889SApple OSS Distributions 	 */
1962*43a90889SApple OSS Distributions 	if (querier_present_timers_running) {
1963*43a90889SApple OSS Distributions 		querier_present_timers_running = 0;
1964*43a90889SApple OSS Distributions 		LIST_FOREACH(igi, &igi_head, igi_link) {
1965*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
1966*43a90889SApple OSS Distributions 			igmp_v1v2_process_querier_timers(igi);
1967*43a90889SApple OSS Distributions 			if (igi->igi_v1_timer > 0 || igi->igi_v2_timer > 0) {
1968*43a90889SApple OSS Distributions 				querier_present_timers_running = 1;
1969*43a90889SApple OSS Distributions 			}
1970*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
1971*43a90889SApple OSS Distributions 		}
1972*43a90889SApple OSS Distributions 	}
1973*43a90889SApple OSS Distributions 
1974*43a90889SApple OSS Distributions 	/*
1975*43a90889SApple OSS Distributions 	 * IGMPv3 General Query response timer processing.
1976*43a90889SApple OSS Distributions 	 */
1977*43a90889SApple OSS Distributions 	if (interface_timers_running) {
1978*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: interface timers running\n", __func__));
1979*43a90889SApple OSS Distributions 		interface_timers_running = 0;
1980*43a90889SApple OSS Distributions 		LIST_FOREACH(igi, &igi_head, igi_link) {
1981*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
1982*43a90889SApple OSS Distributions 			if (igi->igi_version != IGMP_VERSION_3) {
1983*43a90889SApple OSS Distributions 				IGI_UNLOCK(igi);
1984*43a90889SApple OSS Distributions 				continue;
1985*43a90889SApple OSS Distributions 			}
1986*43a90889SApple OSS Distributions 			if (igi->igi_v3_timer == 0) {
1987*43a90889SApple OSS Distributions 				/* Do nothing. */
1988*43a90889SApple OSS Distributions 			} else if (--igi->igi_v3_timer == 0) {
1989*43a90889SApple OSS Distributions 				if (igmp_v3_dispatch_general_query(igi) > 0) {
1990*43a90889SApple OSS Distributions 					interface_timers_running = 1;
1991*43a90889SApple OSS Distributions 				}
1992*43a90889SApple OSS Distributions 			} else {
1993*43a90889SApple OSS Distributions 				interface_timers_running = 1;
1994*43a90889SApple OSS Distributions 			}
1995*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
1996*43a90889SApple OSS Distributions 		}
1997*43a90889SApple OSS Distributions 	}
1998*43a90889SApple OSS Distributions 
1999*43a90889SApple OSS Distributions skip_query_timers:
2000*43a90889SApple OSS Distributions 	if (!current_state_timers_running &&
2001*43a90889SApple OSS Distributions 	    !state_change_timers_running) {
2002*43a90889SApple OSS Distributions 		goto out_locked;
2003*43a90889SApple OSS Distributions 	}
2004*43a90889SApple OSS Distributions 
2005*43a90889SApple OSS Distributions 	current_state_timers_running = 0;
2006*43a90889SApple OSS Distributions 	state_change_timers_running = 0;
2007*43a90889SApple OSS Distributions 
2008*43a90889SApple OSS Distributions 	memset(&qrq, 0, sizeof(struct ifqueue));
2009*43a90889SApple OSS Distributions 	qrq.ifq_maxlen = IGMP_MAX_G_GS_PACKETS;
2010*43a90889SApple OSS Distributions 
2011*43a90889SApple OSS Distributions 	memset(&scq, 0, sizeof(struct ifqueue));
2012*43a90889SApple OSS Distributions 	scq.ifq_maxlen =  IGMP_MAX_STATE_CHANGE_PACKETS;
2013*43a90889SApple OSS Distributions 
2014*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: state change timers running\n", __func__));
2015*43a90889SApple OSS Distributions 
2016*43a90889SApple OSS Distributions 	/*
2017*43a90889SApple OSS Distributions 	 * IGMPv1/v2/v3 host report and state-change timer processing.
2018*43a90889SApple OSS Distributions 	 * Note: Processing a v3 group timer may remove a node.
2019*43a90889SApple OSS Distributions 	 */
2020*43a90889SApple OSS Distributions 	LIST_FOREACH(igi, &igi_head, igi_link) {
2021*43a90889SApple OSS Distributions 		struct in_multistep step;
2022*43a90889SApple OSS Distributions 
2023*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
2024*43a90889SApple OSS Distributions 		ifp = igi->igi_ifp;
2025*43a90889SApple OSS Distributions 		loop = (igi->igi_flags & IGIF_LOOPBACK) ? 1 : 0;
2026*43a90889SApple OSS Distributions 		uri_sec = IGMP_RANDOM_DELAY(igi->igi_uri);
2027*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
2028*43a90889SApple OSS Distributions 
2029*43a90889SApple OSS Distributions 		in_multihead_lock_shared();
2030*43a90889SApple OSS Distributions 		IN_FIRST_MULTI(step, inm);
2031*43a90889SApple OSS Distributions 		while (inm != NULL) {
2032*43a90889SApple OSS Distributions 			INM_LOCK(inm);
2033*43a90889SApple OSS Distributions 			if (inm->inm_ifp != ifp) {
2034*43a90889SApple OSS Distributions 				goto next;
2035*43a90889SApple OSS Distributions 			}
2036*43a90889SApple OSS Distributions 
2037*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
2038*43a90889SApple OSS Distributions 			switch (igi->igi_version) {
2039*43a90889SApple OSS Distributions 			case IGMP_VERSION_1:
2040*43a90889SApple OSS Distributions 			case IGMP_VERSION_2:
2041*43a90889SApple OSS Distributions 				igmp_v1v2_process_group_timer(inm,
2042*43a90889SApple OSS Distributions 				    igi->igi_version);
2043*43a90889SApple OSS Distributions 				break;
2044*43a90889SApple OSS Distributions 			case IGMP_VERSION_3:
2045*43a90889SApple OSS Distributions 				igmp_v3_process_group_timers(igi, &qrq,
2046*43a90889SApple OSS Distributions 				    &scq, inm, uri_sec);
2047*43a90889SApple OSS Distributions 				break;
2048*43a90889SApple OSS Distributions 			}
2049*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
2050*43a90889SApple OSS Distributions next:
2051*43a90889SApple OSS Distributions 			INM_UNLOCK(inm);
2052*43a90889SApple OSS Distributions 			IN_NEXT_MULTI(step, inm);
2053*43a90889SApple OSS Distributions 		}
2054*43a90889SApple OSS Distributions 		in_multihead_lock_done();
2055*43a90889SApple OSS Distributions 
2056*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
2057*43a90889SApple OSS Distributions 		if (igi->igi_version == IGMP_VERSION_1 ||
2058*43a90889SApple OSS Distributions 		    igi->igi_version == IGMP_VERSION_2) {
2059*43a90889SApple OSS Distributions 			igmp_dispatch_queue(igi, &igi->igi_v2q, 0, loop);
2060*43a90889SApple OSS Distributions 		} else if (igi->igi_version == IGMP_VERSION_3) {
2061*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
2062*43a90889SApple OSS Distributions 			igmp_dispatch_queue(NULL, &qrq, 0, loop);
2063*43a90889SApple OSS Distributions 			igmp_dispatch_queue(NULL, &scq, 0, loop);
2064*43a90889SApple OSS Distributions 			VERIFY(qrq.ifq_len == 0);
2065*43a90889SApple OSS Distributions 			VERIFY(scq.ifq_len == 0);
2066*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
2067*43a90889SApple OSS Distributions 		}
2068*43a90889SApple OSS Distributions 		/*
2069*43a90889SApple OSS Distributions 		 * In case there are still any pending membership reports
2070*43a90889SApple OSS Distributions 		 * which didn't get drained at version change time.
2071*43a90889SApple OSS Distributions 		 */
2072*43a90889SApple OSS Distributions 		IF_DRAIN(&igi->igi_v2q);
2073*43a90889SApple OSS Distributions 		/*
2074*43a90889SApple OSS Distributions 		 * Release all deferred inm records, and drain any locally
2075*43a90889SApple OSS Distributions 		 * enqueued packets; do it even if the current IGMP version
2076*43a90889SApple OSS Distributions 		 * for the link is no longer IGMPv3, in order to handle the
2077*43a90889SApple OSS Distributions 		 * version change case.
2078*43a90889SApple OSS Distributions 		 */
2079*43a90889SApple OSS Distributions 		igmp_flush_relq(igi, (struct igmp_inm_relhead *)&inm_dthead);
2080*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
2081*43a90889SApple OSS Distributions 
2082*43a90889SApple OSS Distributions 		IF_DRAIN(&qrq);
2083*43a90889SApple OSS Distributions 		IF_DRAIN(&scq);
2084*43a90889SApple OSS Distributions 	}
2085*43a90889SApple OSS Distributions 
2086*43a90889SApple OSS Distributions out_locked:
2087*43a90889SApple OSS Distributions 	/* re-arm the timer if there's work to do */
2088*43a90889SApple OSS Distributions 	if (fast) {
2089*43a90889SApple OSS Distributions 		igmp_fast_timeout_run = false;
2090*43a90889SApple OSS Distributions 	} else {
2091*43a90889SApple OSS Distributions 		igmp_timeout_run = false;
2092*43a90889SApple OSS Distributions 	}
2093*43a90889SApple OSS Distributions 	igmp_sched_timeout();
2094*43a90889SApple OSS Distributions 	IGMP_UNLOCK();
2095*43a90889SApple OSS Distributions 
2096*43a90889SApple OSS Distributions 	/* Now that we're dropped all locks, release detached records */
2097*43a90889SApple OSS Distributions 	IGMP_REMOVE_DETACHED_INM(&inm_dthead);
2098*43a90889SApple OSS Distributions }
2099*43a90889SApple OSS Distributions 
2100*43a90889SApple OSS Distributions static void
igmp_sched_timeout(void)2101*43a90889SApple OSS Distributions igmp_sched_timeout(void)
2102*43a90889SApple OSS Distributions {
2103*43a90889SApple OSS Distributions 	static thread_call_t igmp_timeout_tcall;
2104*43a90889SApple OSS Distributions 	uint64_t deadline = 0, leeway = 0;
2105*43a90889SApple OSS Distributions 
2106*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_HELD();
2107*43a90889SApple OSS Distributions 	if (igmp_timeout_tcall == NULL) {
2108*43a90889SApple OSS Distributions 		igmp_timeout_tcall =
2109*43a90889SApple OSS Distributions 		    thread_call_allocate_with_options(igmp_timeout,
2110*43a90889SApple OSS Distributions 		    NULL,
2111*43a90889SApple OSS Distributions 		    THREAD_CALL_PRIORITY_KERNEL,
2112*43a90889SApple OSS Distributions 		    THREAD_CALL_OPTIONS_ONCE);
2113*43a90889SApple OSS Distributions 	}
2114*43a90889SApple OSS Distributions 	if (!igmp_timeout_run &&
2115*43a90889SApple OSS Distributions 	    (querier_present_timers_running || current_state_timers_running ||
2116*43a90889SApple OSS Distributions 	    interface_timers_running || state_change_timers_running)) {
2117*43a90889SApple OSS Distributions 		igmp_timeout_run = true;
2118*43a90889SApple OSS Distributions 		clock_interval_to_deadline(igmp_timeout_delay, NSEC_PER_MSEC,
2119*43a90889SApple OSS Distributions 		    &deadline);
2120*43a90889SApple OSS Distributions 		clock_interval_to_absolutetime_interval(igmp_timeout_leeway,
2121*43a90889SApple OSS Distributions 		    NSEC_PER_MSEC, &leeway);
2122*43a90889SApple OSS Distributions 		thread_call_enter_delayed_with_leeway(igmp_timeout_tcall, NULL,
2123*43a90889SApple OSS Distributions 		    deadline, leeway,
2124*43a90889SApple OSS Distributions 		    THREAD_CALL_DELAY_LEEWAY);
2125*43a90889SApple OSS Distributions 	}
2126*43a90889SApple OSS Distributions }
2127*43a90889SApple OSS Distributions 
2128*43a90889SApple OSS Distributions static void
igmp_sched_fast_timeout(void)2129*43a90889SApple OSS Distributions igmp_sched_fast_timeout(void)
2130*43a90889SApple OSS Distributions {
2131*43a90889SApple OSS Distributions 	static thread_call_t igmp_fast_timeout_tcall;
2132*43a90889SApple OSS Distributions 
2133*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_HELD();
2134*43a90889SApple OSS Distributions 	if (igmp_fast_timeout_tcall == NULL) {
2135*43a90889SApple OSS Distributions 		igmp_fast_timeout_tcall =
2136*43a90889SApple OSS Distributions 		    thread_call_allocate_with_options(igmp_timeout,
2137*43a90889SApple OSS Distributions 		    igmp_sched_fast_timeout,
2138*43a90889SApple OSS Distributions 		    THREAD_CALL_PRIORITY_KERNEL,
2139*43a90889SApple OSS Distributions 		    THREAD_CALL_OPTIONS_ONCE);
2140*43a90889SApple OSS Distributions 	}
2141*43a90889SApple OSS Distributions 	if (!igmp_fast_timeout_run &&
2142*43a90889SApple OSS Distributions 	    (current_state_timers_running || state_change_timers_running)) {
2143*43a90889SApple OSS Distributions 		igmp_fast_timeout_run = true;
2144*43a90889SApple OSS Distributions 		thread_call_enter(igmp_fast_timeout_tcall);
2145*43a90889SApple OSS Distributions 	}
2146*43a90889SApple OSS Distributions }
2147*43a90889SApple OSS Distributions 
2148*43a90889SApple OSS Distributions /*
2149*43a90889SApple OSS Distributions  * Appends an in_multi to the list to be released later.
2150*43a90889SApple OSS Distributions  *
2151*43a90889SApple OSS Distributions  * Caller must be holding igi_lock.
2152*43a90889SApple OSS Distributions  */
2153*43a90889SApple OSS Distributions static void
igmp_append_relq(struct igmp_ifinfo * igi,struct in_multi * inm)2154*43a90889SApple OSS Distributions igmp_append_relq(struct igmp_ifinfo *igi, struct in_multi *inm)
2155*43a90889SApple OSS Distributions {
2156*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2157*43a90889SApple OSS Distributions 	if (inm->inm_in_nrele) {
2158*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: inm %llx already on relq ifp %s\n",
2159*43a90889SApple OSS Distributions 		    __func__, (uint64_t)VM_KERNEL_ADDRPERM(inm),
2160*43a90889SApple OSS Distributions 		    if_name(igi->igi_ifp));
2161*43a90889SApple OSS Distributions 		return;
2162*43a90889SApple OSS Distributions 	}
2163*43a90889SApple OSS Distributions 	os_log_debug(OS_LOG_DEFAULT, "%s: adding inm %llx on relq ifp %s\n",
2164*43a90889SApple OSS Distributions 	    __func__, (uint64_t)VM_KERNEL_ADDRPERM(inm),
2165*43a90889SApple OSS Distributions 	    if_name(igi->igi_ifp));
2166*43a90889SApple OSS Distributions 	inm->inm_in_nrele = true;
2167*43a90889SApple OSS Distributions 	SLIST_INSERT_HEAD(&igi->igi_relinmhead, inm, inm_nrele);
2168*43a90889SApple OSS Distributions }
2169*43a90889SApple OSS Distributions 
2170*43a90889SApple OSS Distributions /*
2171*43a90889SApple OSS Distributions  * Free the in_multi reference(s) for this IGMP lifecycle.
2172*43a90889SApple OSS Distributions  *
2173*43a90889SApple OSS Distributions  * Caller must be holding igi_lock.
2174*43a90889SApple OSS Distributions  */
2175*43a90889SApple OSS Distributions static void
igmp_flush_relq(struct igmp_ifinfo * igi,struct igmp_inm_relhead * inm_dthead)2176*43a90889SApple OSS Distributions igmp_flush_relq(struct igmp_ifinfo *igi, struct igmp_inm_relhead *inm_dthead)
2177*43a90889SApple OSS Distributions {
2178*43a90889SApple OSS Distributions 	struct in_multi *inm;
2179*43a90889SApple OSS Distributions 	SLIST_HEAD(, in_multi) temp_relinmhead;
2180*43a90889SApple OSS Distributions 
2181*43a90889SApple OSS Distributions 	/*
2182*43a90889SApple OSS Distributions 	 * Before dropping the igi_lock, copy all the items in the
2183*43a90889SApple OSS Distributions 	 * release list to a temporary list to prevent other threads
2184*43a90889SApple OSS Distributions 	 * from changing igi_relinmhead while we are traversing it.
2185*43a90889SApple OSS Distributions 	 */
2186*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2187*43a90889SApple OSS Distributions 	SLIST_INIT(&temp_relinmhead);
2188*43a90889SApple OSS Distributions 	while ((inm = SLIST_FIRST(&igi->igi_relinmhead)) != NULL) {
2189*43a90889SApple OSS Distributions 		SLIST_REMOVE_HEAD(&igi->igi_relinmhead, inm_nrele);
2190*43a90889SApple OSS Distributions 		SLIST_INSERT_HEAD(&temp_relinmhead, inm, inm_nrele);
2191*43a90889SApple OSS Distributions 	}
2192*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
2193*43a90889SApple OSS Distributions 	in_multihead_lock_exclusive();
2194*43a90889SApple OSS Distributions 	while ((inm = SLIST_FIRST(&temp_relinmhead)) != NULL) {
2195*43a90889SApple OSS Distributions 		int lastref;
2196*43a90889SApple OSS Distributions 
2197*43a90889SApple OSS Distributions 		SLIST_REMOVE_HEAD(&temp_relinmhead, inm_nrele);
2198*43a90889SApple OSS Distributions 		INM_LOCK(inm);
2199*43a90889SApple OSS Distributions 		os_log_debug(OS_LOG_DEFAULT, "%s: flushing %llx on relq ifp %s",
2200*43a90889SApple OSS Distributions 		    __func__,
2201*43a90889SApple OSS Distributions 		    (uint64_t)VM_KERNEL_ADDRPERM(inm),
2202*43a90889SApple OSS Distributions 		    if_name(inm->inm_ifp));
2203*43a90889SApple OSS Distributions 		VERIFY(inm->inm_in_nrele == true);
2204*43a90889SApple OSS Distributions 		inm->inm_in_nrele = false;
2205*43a90889SApple OSS Distributions 		VERIFY(inm->inm_nrelecnt != 0);
2206*43a90889SApple OSS Distributions 		inm->inm_nrelecnt--;
2207*43a90889SApple OSS Distributions 		lastref = in_multi_detach(inm);
2208*43a90889SApple OSS Distributions 		VERIFY(!lastref || (!(inm->inm_debug & IFD_ATTACHED) &&
2209*43a90889SApple OSS Distributions 		    inm->inm_reqcnt == 0));
2210*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
2211*43a90889SApple OSS Distributions 		/* from igi_relinmhead */
2212*43a90889SApple OSS Distributions 		INM_REMREF(inm);
2213*43a90889SApple OSS Distributions 		/* from in_multihead list */
2214*43a90889SApple OSS Distributions 		if (lastref) {
2215*43a90889SApple OSS Distributions 			/*
2216*43a90889SApple OSS Distributions 			 * Defer releasing our final reference, as we
2217*43a90889SApple OSS Distributions 			 * are holding the IGMP lock at this point, and
2218*43a90889SApple OSS Distributions 			 * we could end up with locking issues later on
2219*43a90889SApple OSS Distributions 			 * (while issuing SIOCDELMULTI) when this is the
2220*43a90889SApple OSS Distributions 			 * final reference count.  Let the caller do it
2221*43a90889SApple OSS Distributions 			 * when it is safe.
2222*43a90889SApple OSS Distributions 			 */
2223*43a90889SApple OSS Distributions 			IGMP_ADD_DETACHED_INM(inm_dthead, inm);
2224*43a90889SApple OSS Distributions 		}
2225*43a90889SApple OSS Distributions 	}
2226*43a90889SApple OSS Distributions 	in_multihead_lock_done();
2227*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
2228*43a90889SApple OSS Distributions }
2229*43a90889SApple OSS Distributions 
2230*43a90889SApple OSS Distributions /*
2231*43a90889SApple OSS Distributions  * Update host report group timer for IGMPv1/v2.
2232*43a90889SApple OSS Distributions  * Will update the global pending timer flags.
2233*43a90889SApple OSS Distributions  */
2234*43a90889SApple OSS Distributions static void
igmp_v1v2_process_group_timer(struct in_multi * inm,const int igmp_version)2235*43a90889SApple OSS Distributions igmp_v1v2_process_group_timer(struct in_multi *inm, const int igmp_version)
2236*43a90889SApple OSS Distributions {
2237*43a90889SApple OSS Distributions 	int report_timer_expired;
2238*43a90889SApple OSS Distributions 
2239*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_HELD();
2240*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2241*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(inm->inm_igi);
2242*43a90889SApple OSS Distributions 
2243*43a90889SApple OSS Distributions 	if (inm->inm_timer == 0) {
2244*43a90889SApple OSS Distributions 		report_timer_expired = 0;
2245*43a90889SApple OSS Distributions 	} else if (--inm->inm_timer == 0) {
2246*43a90889SApple OSS Distributions 		report_timer_expired = 1;
2247*43a90889SApple OSS Distributions 	} else {
2248*43a90889SApple OSS Distributions 		current_state_timers_running = 1;
2249*43a90889SApple OSS Distributions 		/* caller will schedule timer */
2250*43a90889SApple OSS Distributions 		return;
2251*43a90889SApple OSS Distributions 	}
2252*43a90889SApple OSS Distributions 
2253*43a90889SApple OSS Distributions 	switch (inm->inm_state) {
2254*43a90889SApple OSS Distributions 	case IGMP_NOT_MEMBER:
2255*43a90889SApple OSS Distributions 	case IGMP_SILENT_MEMBER:
2256*43a90889SApple OSS Distributions 	case IGMP_IDLE_MEMBER:
2257*43a90889SApple OSS Distributions 	case IGMP_LAZY_MEMBER:
2258*43a90889SApple OSS Distributions 	case IGMP_SLEEPING_MEMBER:
2259*43a90889SApple OSS Distributions 	case IGMP_AWAKENING_MEMBER:
2260*43a90889SApple OSS Distributions 		break;
2261*43a90889SApple OSS Distributions 	case IGMP_REPORTING_MEMBER:
2262*43a90889SApple OSS Distributions 		if (report_timer_expired) {
2263*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_IDLE_MEMBER;
2264*43a90889SApple OSS Distributions 			(void) igmp_v1v2_queue_report(inm,
2265*43a90889SApple OSS Distributions 			    (igmp_version == IGMP_VERSION_2) ?
2266*43a90889SApple OSS Distributions 			    IGMP_v2_HOST_MEMBERSHIP_REPORT :
2267*43a90889SApple OSS Distributions 			    IGMP_v1_HOST_MEMBERSHIP_REPORT);
2268*43a90889SApple OSS Distributions 			INM_LOCK_ASSERT_HELD(inm);
2269*43a90889SApple OSS Distributions 			IGI_LOCK_ASSERT_HELD(inm->inm_igi);
2270*43a90889SApple OSS Distributions 		}
2271*43a90889SApple OSS Distributions 		break;
2272*43a90889SApple OSS Distributions 	case IGMP_G_QUERY_PENDING_MEMBER:
2273*43a90889SApple OSS Distributions 	case IGMP_SG_QUERY_PENDING_MEMBER:
2274*43a90889SApple OSS Distributions 	case IGMP_LEAVING_MEMBER:
2275*43a90889SApple OSS Distributions 		break;
2276*43a90889SApple OSS Distributions 	}
2277*43a90889SApple OSS Distributions }
2278*43a90889SApple OSS Distributions 
2279*43a90889SApple OSS Distributions /*
2280*43a90889SApple OSS Distributions  * Update a group's timers for IGMPv3.
2281*43a90889SApple OSS Distributions  * Will update the global pending timer flags.
2282*43a90889SApple OSS Distributions  * Note: Unlocked read from igi.
2283*43a90889SApple OSS Distributions  */
2284*43a90889SApple OSS Distributions static void
igmp_v3_process_group_timers(struct igmp_ifinfo * igi,struct ifqueue * qrq,struct ifqueue * scq,struct in_multi * inm,const unsigned int uri_sec)2285*43a90889SApple OSS Distributions igmp_v3_process_group_timers(struct igmp_ifinfo *igi,
2286*43a90889SApple OSS Distributions     struct ifqueue *qrq, struct ifqueue *scq,
2287*43a90889SApple OSS Distributions     struct in_multi *inm, const unsigned int uri_sec)
2288*43a90889SApple OSS Distributions {
2289*43a90889SApple OSS Distributions 	int query_response_timer_expired;
2290*43a90889SApple OSS Distributions 	int state_change_retransmit_timer_expired;
2291*43a90889SApple OSS Distributions 
2292*43a90889SApple OSS Distributions 	IGMP_LOCK_ASSERT_HELD();
2293*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2294*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2295*43a90889SApple OSS Distributions 	VERIFY(igi == inm->inm_igi);
2296*43a90889SApple OSS Distributions 
2297*43a90889SApple OSS Distributions 	query_response_timer_expired = 0;
2298*43a90889SApple OSS Distributions 	state_change_retransmit_timer_expired = 0;
2299*43a90889SApple OSS Distributions 
2300*43a90889SApple OSS Distributions 	/*
2301*43a90889SApple OSS Distributions 	 * During a transition from v1/v2 compatibility mode back to v3,
2302*43a90889SApple OSS Distributions 	 * a group record in REPORTING state may still have its group
2303*43a90889SApple OSS Distributions 	 * timer active. This is a no-op in this function; it is easier
2304*43a90889SApple OSS Distributions 	 * to deal with it here than to complicate the timeout path.
2305*43a90889SApple OSS Distributions 	 */
2306*43a90889SApple OSS Distributions 	if (inm->inm_timer == 0) {
2307*43a90889SApple OSS Distributions 		query_response_timer_expired = 0;
2308*43a90889SApple OSS Distributions 	} else if (--inm->inm_timer == 0) {
2309*43a90889SApple OSS Distributions 		query_response_timer_expired = 1;
2310*43a90889SApple OSS Distributions 	} else {
2311*43a90889SApple OSS Distributions 		current_state_timers_running = 1;
2312*43a90889SApple OSS Distributions 		/* caller will schedule timer */
2313*43a90889SApple OSS Distributions 	}
2314*43a90889SApple OSS Distributions 
2315*43a90889SApple OSS Distributions 	if (inm->inm_sctimer == 0) {
2316*43a90889SApple OSS Distributions 		state_change_retransmit_timer_expired = 0;
2317*43a90889SApple OSS Distributions 	} else if (--inm->inm_sctimer == 0) {
2318*43a90889SApple OSS Distributions 		state_change_retransmit_timer_expired = 1;
2319*43a90889SApple OSS Distributions 	} else {
2320*43a90889SApple OSS Distributions 		state_change_timers_running = 1;
2321*43a90889SApple OSS Distributions 		/* caller will schedule timer */
2322*43a90889SApple OSS Distributions 	}
2323*43a90889SApple OSS Distributions 
2324*43a90889SApple OSS Distributions 	/* We are in timer callback, so be quick about it. */
2325*43a90889SApple OSS Distributions 	if (!state_change_retransmit_timer_expired &&
2326*43a90889SApple OSS Distributions 	    !query_response_timer_expired) {
2327*43a90889SApple OSS Distributions 		return;
2328*43a90889SApple OSS Distributions 	}
2329*43a90889SApple OSS Distributions 
2330*43a90889SApple OSS Distributions 	switch (inm->inm_state) {
2331*43a90889SApple OSS Distributions 	case IGMP_NOT_MEMBER:
2332*43a90889SApple OSS Distributions 	case IGMP_SILENT_MEMBER:
2333*43a90889SApple OSS Distributions 	case IGMP_SLEEPING_MEMBER:
2334*43a90889SApple OSS Distributions 	case IGMP_LAZY_MEMBER:
2335*43a90889SApple OSS Distributions 	case IGMP_AWAKENING_MEMBER:
2336*43a90889SApple OSS Distributions 	case IGMP_IDLE_MEMBER:
2337*43a90889SApple OSS Distributions 		break;
2338*43a90889SApple OSS Distributions 	case IGMP_G_QUERY_PENDING_MEMBER:
2339*43a90889SApple OSS Distributions 	case IGMP_SG_QUERY_PENDING_MEMBER:
2340*43a90889SApple OSS Distributions 		/*
2341*43a90889SApple OSS Distributions 		 * Respond to a previously pending Group-Specific
2342*43a90889SApple OSS Distributions 		 * or Group-and-Source-Specific query by enqueueing
2343*43a90889SApple OSS Distributions 		 * the appropriate Current-State report for
2344*43a90889SApple OSS Distributions 		 * immediate transmission.
2345*43a90889SApple OSS Distributions 		 */
2346*43a90889SApple OSS Distributions 		if (query_response_timer_expired) {
2347*43a90889SApple OSS Distributions 			int retval;
2348*43a90889SApple OSS Distributions 
2349*43a90889SApple OSS Distributions 			retval = igmp_v3_enqueue_group_record(qrq, inm, 0, 1,
2350*43a90889SApple OSS Distributions 			    (inm->inm_state == IGMP_SG_QUERY_PENDING_MEMBER));
2351*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: enqueue record = %d\n",
2352*43a90889SApple OSS Distributions 			    __func__, retval));
2353*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_REPORTING_MEMBER;
2354*43a90889SApple OSS Distributions 			/* XXX Clear recorded sources for next time. */
2355*43a90889SApple OSS Distributions 			inm_clear_recorded(inm);
2356*43a90889SApple OSS Distributions 		}
2357*43a90889SApple OSS Distributions 		OS_FALLTHROUGH;
2358*43a90889SApple OSS Distributions 	case IGMP_REPORTING_MEMBER:
2359*43a90889SApple OSS Distributions 	case IGMP_LEAVING_MEMBER:
2360*43a90889SApple OSS Distributions 		if (state_change_retransmit_timer_expired) {
2361*43a90889SApple OSS Distributions 			/*
2362*43a90889SApple OSS Distributions 			 * State-change retransmission timer fired.
2363*43a90889SApple OSS Distributions 			 * If there are any further pending retransmissions,
2364*43a90889SApple OSS Distributions 			 * set the global pending state-change flag, and
2365*43a90889SApple OSS Distributions 			 * reset the timer.
2366*43a90889SApple OSS Distributions 			 */
2367*43a90889SApple OSS Distributions 			if (--inm->inm_scrv > 0) {
2368*43a90889SApple OSS Distributions 				inm->inm_sctimer = (uint16_t)uri_sec;
2369*43a90889SApple OSS Distributions 				state_change_timers_running = 1;
2370*43a90889SApple OSS Distributions 				/* caller will schedule timer */
2371*43a90889SApple OSS Distributions 			}
2372*43a90889SApple OSS Distributions 			/*
2373*43a90889SApple OSS Distributions 			 * Retransmit the previously computed state-change
2374*43a90889SApple OSS Distributions 			 * report. If there are no further pending
2375*43a90889SApple OSS Distributions 			 * retransmissions, the mbuf queue will be consumed.
2376*43a90889SApple OSS Distributions 			 * Update T0 state to T1 as we have now sent
2377*43a90889SApple OSS Distributions 			 * a state-change.
2378*43a90889SApple OSS Distributions 			 */
2379*43a90889SApple OSS Distributions 			(void) igmp_v3_merge_state_changes(inm, scq);
2380*43a90889SApple OSS Distributions 
2381*43a90889SApple OSS Distributions 			inm_commit(inm);
2382*43a90889SApple OSS Distributions 			IGMP_INET_PRINTF(inm->inm_addr,
2383*43a90889SApple OSS Distributions 			    ("%s: T1 -> T0 for %s/%s\n", __func__,
2384*43a90889SApple OSS Distributions 			    _igmp_inet_buf, if_name(inm->inm_ifp)));
2385*43a90889SApple OSS Distributions 
2386*43a90889SApple OSS Distributions 			/*
2387*43a90889SApple OSS Distributions 			 * If we are leaving the group for good, make sure
2388*43a90889SApple OSS Distributions 			 * we release IGMP's reference to it.
2389*43a90889SApple OSS Distributions 			 * This release must be deferred using a SLIST,
2390*43a90889SApple OSS Distributions 			 * as we are called from a loop which traverses
2391*43a90889SApple OSS Distributions 			 * the in_multihead list.
2392*43a90889SApple OSS Distributions 			 */
2393*43a90889SApple OSS Distributions 			if (inm->inm_state == IGMP_LEAVING_MEMBER &&
2394*43a90889SApple OSS Distributions 			    inm->inm_scrv == 0) {
2395*43a90889SApple OSS Distributions 				inm->inm_state = IGMP_NOT_MEMBER;
2396*43a90889SApple OSS Distributions 				/*
2397*43a90889SApple OSS Distributions 				 * A reference has already been held in
2398*43a90889SApple OSS Distributions 				 * igmp_final_leave() for this inm, so
2399*43a90889SApple OSS Distributions 				 * no need to hold another one.  We also
2400*43a90889SApple OSS Distributions 				 * bumped up its request count then, so
2401*43a90889SApple OSS Distributions 				 * that it stays in in_multihead.  Both
2402*43a90889SApple OSS Distributions 				 * of them will be released when it is
2403*43a90889SApple OSS Distributions 				 * dequeued later on.
2404*43a90889SApple OSS Distributions 				 */
2405*43a90889SApple OSS Distributions 				VERIFY(inm->inm_nrelecnt != 0);
2406*43a90889SApple OSS Distributions 				igmp_append_relq(igi, inm);
2407*43a90889SApple OSS Distributions 			}
2408*43a90889SApple OSS Distributions 		}
2409*43a90889SApple OSS Distributions 		break;
2410*43a90889SApple OSS Distributions 	}
2411*43a90889SApple OSS Distributions }
2412*43a90889SApple OSS Distributions 
2413*43a90889SApple OSS Distributions /*
2414*43a90889SApple OSS Distributions  * Suppress a group's pending response to a group or source/group query.
2415*43a90889SApple OSS Distributions  *
2416*43a90889SApple OSS Distributions  * Do NOT suppress state changes. This leads to IGMPv3 inconsistency.
2417*43a90889SApple OSS Distributions  * Do NOT update ST1/ST0 as this operation merely suppresses
2418*43a90889SApple OSS Distributions  * the currently pending group record.
2419*43a90889SApple OSS Distributions  * Do NOT suppress the response to a general query. It is possible but
2420*43a90889SApple OSS Distributions  * it would require adding another state or flag.
2421*43a90889SApple OSS Distributions  */
2422*43a90889SApple OSS Distributions static void
igmp_v3_suppress_group_record(struct in_multi * inm)2423*43a90889SApple OSS Distributions igmp_v3_suppress_group_record(struct in_multi *inm)
2424*43a90889SApple OSS Distributions {
2425*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2426*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(inm->inm_igi);
2427*43a90889SApple OSS Distributions 
2428*43a90889SApple OSS Distributions 	VERIFY(inm->inm_igi->igi_version == IGMP_VERSION_3);
2429*43a90889SApple OSS Distributions 
2430*43a90889SApple OSS Distributions 	if (inm->inm_state != IGMP_G_QUERY_PENDING_MEMBER &&
2431*43a90889SApple OSS Distributions 	    inm->inm_state != IGMP_SG_QUERY_PENDING_MEMBER) {
2432*43a90889SApple OSS Distributions 		return;
2433*43a90889SApple OSS Distributions 	}
2434*43a90889SApple OSS Distributions 
2435*43a90889SApple OSS Distributions 	if (inm->inm_state == IGMP_SG_QUERY_PENDING_MEMBER) {
2436*43a90889SApple OSS Distributions 		inm_clear_recorded(inm);
2437*43a90889SApple OSS Distributions 	}
2438*43a90889SApple OSS Distributions 
2439*43a90889SApple OSS Distributions 	inm->inm_timer = 0;
2440*43a90889SApple OSS Distributions 	inm->inm_state = IGMP_REPORTING_MEMBER;
2441*43a90889SApple OSS Distributions }
2442*43a90889SApple OSS Distributions 
2443*43a90889SApple OSS Distributions /*
2444*43a90889SApple OSS Distributions  * Switch to a different IGMP version on the given interface,
2445*43a90889SApple OSS Distributions  * as per Section 7.2.1.
2446*43a90889SApple OSS Distributions  */
2447*43a90889SApple OSS Distributions static uint32_t
igmp_set_version(struct igmp_ifinfo * igi,const int igmp_version)2448*43a90889SApple OSS Distributions igmp_set_version(struct igmp_ifinfo *igi, const int igmp_version)
2449*43a90889SApple OSS Distributions {
2450*43a90889SApple OSS Distributions 	int old_version_timer;
2451*43a90889SApple OSS Distributions 
2452*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2453*43a90889SApple OSS Distributions 
2454*43a90889SApple OSS Distributions 	os_log(OS_LOG_DEFAULT, "%s: switching to v%d on ifp %s\n", __func__,
2455*43a90889SApple OSS Distributions 	    igmp_version, if_name(igi->igi_ifp));
2456*43a90889SApple OSS Distributions 
2457*43a90889SApple OSS Distributions 	if (igmp_version == IGMP_VERSION_1 || igmp_version == IGMP_VERSION_2) {
2458*43a90889SApple OSS Distributions 		/*
2459*43a90889SApple OSS Distributions 		 * Compute the "Older Version Querier Present" timer as per
2460*43a90889SApple OSS Distributions 		 * Section 8.12, in seconds.
2461*43a90889SApple OSS Distributions 		 */
2462*43a90889SApple OSS Distributions 		old_version_timer = igi->igi_rv * igi->igi_qi + igi->igi_qri;
2463*43a90889SApple OSS Distributions 
2464*43a90889SApple OSS Distributions 		if (igmp_version == IGMP_VERSION_1) {
2465*43a90889SApple OSS Distributions 			igi->igi_v1_timer = old_version_timer;
2466*43a90889SApple OSS Distributions 			igi->igi_v2_timer = 0;
2467*43a90889SApple OSS Distributions 		} else if (igmp_version == IGMP_VERSION_2) {
2468*43a90889SApple OSS Distributions 			igi->igi_v1_timer = 0;
2469*43a90889SApple OSS Distributions 			igi->igi_v2_timer = old_version_timer;
2470*43a90889SApple OSS Distributions 		}
2471*43a90889SApple OSS Distributions 	}
2472*43a90889SApple OSS Distributions 
2473*43a90889SApple OSS Distributions 	if (igi->igi_v1_timer == 0 && igi->igi_v2_timer > 0) {
2474*43a90889SApple OSS Distributions 		if (igi->igi_version != IGMP_VERSION_2) {
2475*43a90889SApple OSS Distributions 			igmp_v3_cancel_link_timers(igi);
2476*43a90889SApple OSS Distributions 			igi->igi_version = IGMP_VERSION_2;
2477*43a90889SApple OSS Distributions 		}
2478*43a90889SApple OSS Distributions 	} else if (igi->igi_v1_timer > 0) {
2479*43a90889SApple OSS Distributions 		if (igi->igi_version != IGMP_VERSION_1) {
2480*43a90889SApple OSS Distributions 			igmp_v3_cancel_link_timers(igi);
2481*43a90889SApple OSS Distributions 			igi->igi_version = IGMP_VERSION_1;
2482*43a90889SApple OSS Distributions 		}
2483*43a90889SApple OSS Distributions 	}
2484*43a90889SApple OSS Distributions 
2485*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2486*43a90889SApple OSS Distributions 
2487*43a90889SApple OSS Distributions 	return MAX(igi->igi_v1_timer, igi->igi_v2_timer);
2488*43a90889SApple OSS Distributions }
2489*43a90889SApple OSS Distributions 
2490*43a90889SApple OSS Distributions /*
2491*43a90889SApple OSS Distributions  * Cancel pending IGMPv3 timers for the given link and all groups
2492*43a90889SApple OSS Distributions  * joined on it; state-change, general-query, and group-query timers.
2493*43a90889SApple OSS Distributions  *
2494*43a90889SApple OSS Distributions  * Only ever called on a transition from v3 to Compatibility mode. Kill
2495*43a90889SApple OSS Distributions  * the timers stone dead (this may be expensive for large N groups), they
2496*43a90889SApple OSS Distributions  * will be restarted if Compatibility Mode deems that they must be due to
2497*43a90889SApple OSS Distributions  * query processing.
2498*43a90889SApple OSS Distributions  */
2499*43a90889SApple OSS Distributions static void
igmp_v3_cancel_link_timers(struct igmp_ifinfo * igi)2500*43a90889SApple OSS Distributions igmp_v3_cancel_link_timers(struct igmp_ifinfo *igi)
2501*43a90889SApple OSS Distributions {
2502*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
2503*43a90889SApple OSS Distributions 	struct in_multi         *inm;
2504*43a90889SApple OSS Distributions 	struct in_multistep     step;
2505*43a90889SApple OSS Distributions 
2506*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2507*43a90889SApple OSS Distributions 
2508*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: cancel v3 timers on ifp 0x%llx(%s)\n", __func__,
2509*43a90889SApple OSS Distributions 	    (uint64_t)VM_KERNEL_ADDRPERM(igi->igi_ifp), if_name(igi->igi_ifp)));
2510*43a90889SApple OSS Distributions 
2511*43a90889SApple OSS Distributions 	/*
2512*43a90889SApple OSS Distributions 	 * Stop the v3 General Query Response on this link stone dead.
2513*43a90889SApple OSS Distributions 	 * If timer is woken up due to interface_timers_running,
2514*43a90889SApple OSS Distributions 	 * the flag will be cleared if there are no pending link timers.
2515*43a90889SApple OSS Distributions 	 */
2516*43a90889SApple OSS Distributions 	igi->igi_v3_timer = 0;
2517*43a90889SApple OSS Distributions 
2518*43a90889SApple OSS Distributions 	/*
2519*43a90889SApple OSS Distributions 	 * Now clear the current-state and state-change report timers
2520*43a90889SApple OSS Distributions 	 * for all memberships scoped to this link.
2521*43a90889SApple OSS Distributions 	 */
2522*43a90889SApple OSS Distributions 	ifp = igi->igi_ifp;
2523*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
2524*43a90889SApple OSS Distributions 
2525*43a90889SApple OSS Distributions 	in_multihead_lock_shared();
2526*43a90889SApple OSS Distributions 	IN_FIRST_MULTI(step, inm);
2527*43a90889SApple OSS Distributions 	while (inm != NULL) {
2528*43a90889SApple OSS Distributions 		INM_LOCK(inm);
2529*43a90889SApple OSS Distributions 		if (inm->inm_ifp != ifp && inm->inm_igi != igi) {
2530*43a90889SApple OSS Distributions 			goto next;
2531*43a90889SApple OSS Distributions 		}
2532*43a90889SApple OSS Distributions 
2533*43a90889SApple OSS Distributions 		switch (inm->inm_state) {
2534*43a90889SApple OSS Distributions 		case IGMP_NOT_MEMBER:
2535*43a90889SApple OSS Distributions 		case IGMP_SILENT_MEMBER:
2536*43a90889SApple OSS Distributions 		case IGMP_IDLE_MEMBER:
2537*43a90889SApple OSS Distributions 		case IGMP_LAZY_MEMBER:
2538*43a90889SApple OSS Distributions 		case IGMP_SLEEPING_MEMBER:
2539*43a90889SApple OSS Distributions 		case IGMP_AWAKENING_MEMBER:
2540*43a90889SApple OSS Distributions 			/*
2541*43a90889SApple OSS Distributions 			 * These states are either not relevant in v3 mode,
2542*43a90889SApple OSS Distributions 			 * or are unreported. Do nothing.
2543*43a90889SApple OSS Distributions 			 */
2544*43a90889SApple OSS Distributions 			break;
2545*43a90889SApple OSS Distributions 		case IGMP_LEAVING_MEMBER:
2546*43a90889SApple OSS Distributions 			/*
2547*43a90889SApple OSS Distributions 			 * If we are leaving the group and switching to
2548*43a90889SApple OSS Distributions 			 * compatibility mode, we need to release the final
2549*43a90889SApple OSS Distributions 			 * reference held for issuing the INCLUDE {}, and
2550*43a90889SApple OSS Distributions 			 * transition to REPORTING to ensure the host leave
2551*43a90889SApple OSS Distributions 			 * message is sent upstream to the old querier --
2552*43a90889SApple OSS Distributions 			 * transition to NOT would lose the leave and race.
2553*43a90889SApple OSS Distributions 			 * During igmp_final_leave(), we bumped up both the
2554*43a90889SApple OSS Distributions 			 * request and reference counts.  Since we cannot
2555*43a90889SApple OSS Distributions 			 * call in_multi_detach() here, defer this task to
2556*43a90889SApple OSS Distributions 			 * the timer routine.
2557*43a90889SApple OSS Distributions 			 */
2558*43a90889SApple OSS Distributions 			VERIFY(inm->inm_nrelecnt != 0);
2559*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
2560*43a90889SApple OSS Distributions 			igmp_append_relq(igi, inm);
2561*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
2562*43a90889SApple OSS Distributions 			OS_FALLTHROUGH;
2563*43a90889SApple OSS Distributions 		case IGMP_G_QUERY_PENDING_MEMBER:
2564*43a90889SApple OSS Distributions 		case IGMP_SG_QUERY_PENDING_MEMBER:
2565*43a90889SApple OSS Distributions 			inm_clear_recorded(inm);
2566*43a90889SApple OSS Distributions 			OS_FALLTHROUGH;
2567*43a90889SApple OSS Distributions 		case IGMP_REPORTING_MEMBER:
2568*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_REPORTING_MEMBER;
2569*43a90889SApple OSS Distributions 			break;
2570*43a90889SApple OSS Distributions 		}
2571*43a90889SApple OSS Distributions 		/*
2572*43a90889SApple OSS Distributions 		 * Always clear state-change and group report timers.
2573*43a90889SApple OSS Distributions 		 * Free any pending IGMPv3 state-change records.
2574*43a90889SApple OSS Distributions 		 */
2575*43a90889SApple OSS Distributions 		inm->inm_sctimer = 0;
2576*43a90889SApple OSS Distributions 		inm->inm_timer = 0;
2577*43a90889SApple OSS Distributions 		IF_DRAIN(&inm->inm_scq);
2578*43a90889SApple OSS Distributions next:
2579*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
2580*43a90889SApple OSS Distributions 		IN_NEXT_MULTI(step, inm);
2581*43a90889SApple OSS Distributions 	}
2582*43a90889SApple OSS Distributions 	in_multihead_lock_done();
2583*43a90889SApple OSS Distributions 
2584*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
2585*43a90889SApple OSS Distributions }
2586*43a90889SApple OSS Distributions 
2587*43a90889SApple OSS Distributions /*
2588*43a90889SApple OSS Distributions  * Update the Older Version Querier Present timers for a link.
2589*43a90889SApple OSS Distributions  * See Section 7.2.1 of RFC 3376.
2590*43a90889SApple OSS Distributions  */
2591*43a90889SApple OSS Distributions static void
igmp_v1v2_process_querier_timers(struct igmp_ifinfo * igi)2592*43a90889SApple OSS Distributions igmp_v1v2_process_querier_timers(struct igmp_ifinfo *igi)
2593*43a90889SApple OSS Distributions {
2594*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
2595*43a90889SApple OSS Distributions 
2596*43a90889SApple OSS Distributions 	if (igi->igi_v1_timer == 0 && igi->igi_v2_timer == 0) {
2597*43a90889SApple OSS Distributions 		/*
2598*43a90889SApple OSS Distributions 		 * IGMPv1 and IGMPv2 Querier Present timers expired.
2599*43a90889SApple OSS Distributions 		 *
2600*43a90889SApple OSS Distributions 		 * Revert to IGMPv3.
2601*43a90889SApple OSS Distributions 		 */
2602*43a90889SApple OSS Distributions 		if (igi->igi_version != IGMP_VERSION_3) {
2603*43a90889SApple OSS Distributions 			os_log(OS_LOG_DEFAULT, "%s: transition from v%d->v%d "
2604*43a90889SApple OSS Distributions 			    "on %s\n", __func__,
2605*43a90889SApple OSS Distributions 			    igi->igi_version, IGMP_VERSION_3,
2606*43a90889SApple OSS Distributions 			    if_name(igi->igi_ifp));
2607*43a90889SApple OSS Distributions 			igi->igi_version = IGMP_VERSION_3;
2608*43a90889SApple OSS Distributions 			IF_DRAIN(&igi->igi_v2q);
2609*43a90889SApple OSS Distributions 		}
2610*43a90889SApple OSS Distributions 	} else if (igi->igi_v1_timer == 0 && igi->igi_v2_timer > 0) {
2611*43a90889SApple OSS Distributions 		/*
2612*43a90889SApple OSS Distributions 		 * IGMPv1 Querier Present timer expired,
2613*43a90889SApple OSS Distributions 		 * IGMPv2 Querier Present timer running.
2614*43a90889SApple OSS Distributions 		 * If IGMPv2 was disabled since last timeout,
2615*43a90889SApple OSS Distributions 		 * revert to IGMPv3.
2616*43a90889SApple OSS Distributions 		 * If IGMPv2 is enabled, revert to IGMPv2.
2617*43a90889SApple OSS Distributions 		 */
2618*43a90889SApple OSS Distributions 		if (!igmp_v2enable) {
2619*43a90889SApple OSS Distributions 			os_log(OS_LOG_DEFAULT, "%s: transition from v%d->v%d "
2620*43a90889SApple OSS Distributions 			    "on %s\n", __func__,
2621*43a90889SApple OSS Distributions 			    igi->igi_version, IGMP_VERSION_3,
2622*43a90889SApple OSS Distributions 			    if_name(igi->igi_ifp));
2623*43a90889SApple OSS Distributions 			igi->igi_v2_timer = 0;
2624*43a90889SApple OSS Distributions 			igi->igi_version = IGMP_VERSION_3;
2625*43a90889SApple OSS Distributions 			IF_DRAIN(&igi->igi_v2q);
2626*43a90889SApple OSS Distributions 		} else {
2627*43a90889SApple OSS Distributions 			--igi->igi_v2_timer;
2628*43a90889SApple OSS Distributions 			if (igi->igi_version != IGMP_VERSION_2) {
2629*43a90889SApple OSS Distributions 				os_log(OS_LOG_DEFAULT, "%s: transition from v%d->v%d "
2630*43a90889SApple OSS Distributions 				    "on %s\n", __func__,
2631*43a90889SApple OSS Distributions 				    igi->igi_version, IGMP_VERSION_2,
2632*43a90889SApple OSS Distributions 				    if_name(igi->igi_ifp));
2633*43a90889SApple OSS Distributions 				IF_DRAIN(&igi->igi_gq);
2634*43a90889SApple OSS Distributions 				igmp_v3_cancel_link_timers(igi);
2635*43a90889SApple OSS Distributions 				igi->igi_version = IGMP_VERSION_2;
2636*43a90889SApple OSS Distributions 			}
2637*43a90889SApple OSS Distributions 		}
2638*43a90889SApple OSS Distributions 	} else if (igi->igi_v1_timer > 0) {
2639*43a90889SApple OSS Distributions 		/*
2640*43a90889SApple OSS Distributions 		 * IGMPv1 Querier Present timer running.
2641*43a90889SApple OSS Distributions 		 * Stop IGMPv2 timer if running.
2642*43a90889SApple OSS Distributions 		 *
2643*43a90889SApple OSS Distributions 		 * If IGMPv1 was disabled since last timeout,
2644*43a90889SApple OSS Distributions 		 * revert to IGMPv3.
2645*43a90889SApple OSS Distributions 		 * If IGMPv1 is enabled, reset IGMPv2 timer if running.
2646*43a90889SApple OSS Distributions 		 */
2647*43a90889SApple OSS Distributions 		if (!igmp_v1enable) {
2648*43a90889SApple OSS Distributions 			os_log(OS_LOG_DEFAULT, "%s: transition from v%d->v%d "
2649*43a90889SApple OSS Distributions 			    "on %s\n", __func__,
2650*43a90889SApple OSS Distributions 			    igi->igi_version, IGMP_VERSION_3,
2651*43a90889SApple OSS Distributions 			    if_name(igi->igi_ifp));
2652*43a90889SApple OSS Distributions 			igi->igi_v1_timer = 0;
2653*43a90889SApple OSS Distributions 			igi->igi_version = IGMP_VERSION_3;
2654*43a90889SApple OSS Distributions 			IF_DRAIN(&igi->igi_v2q);
2655*43a90889SApple OSS Distributions 		} else {
2656*43a90889SApple OSS Distributions 			--igi->igi_v1_timer;
2657*43a90889SApple OSS Distributions 		}
2658*43a90889SApple OSS Distributions 		if (igi->igi_v2_timer > 0) {
2659*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: cancel v2 timer on 0x%llx(%s)\n",
2660*43a90889SApple OSS Distributions 			    __func__,
2661*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(igi->igi_ifp),
2662*43a90889SApple OSS Distributions 			    if_name(igi->igi_ifp)));
2663*43a90889SApple OSS Distributions 			igi->igi_v2_timer = 0;
2664*43a90889SApple OSS Distributions 		}
2665*43a90889SApple OSS Distributions 	}
2666*43a90889SApple OSS Distributions }
2667*43a90889SApple OSS Distributions 
2668*43a90889SApple OSS Distributions /*
2669*43a90889SApple OSS Distributions  * Dispatch an IGMPv1/v2 host report or leave message.
2670*43a90889SApple OSS Distributions  * These are always small enough to fit inside a single mbuf.
2671*43a90889SApple OSS Distributions  */
2672*43a90889SApple OSS Distributions static int
igmp_v1v2_queue_report(struct in_multi * inm,const int type)2673*43a90889SApple OSS Distributions igmp_v1v2_queue_report(struct in_multi *inm, const int type)
2674*43a90889SApple OSS Distributions {
2675*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
2676*43a90889SApple OSS Distributions 	struct igmp             *igmp;
2677*43a90889SApple OSS Distributions 	struct ip               *ip;
2678*43a90889SApple OSS Distributions 	struct mbuf             *m;
2679*43a90889SApple OSS Distributions 	int                     error = 0;
2680*43a90889SApple OSS Distributions 
2681*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2682*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(inm->inm_igi);
2683*43a90889SApple OSS Distributions 
2684*43a90889SApple OSS Distributions 	ifp = inm->inm_ifp;
2685*43a90889SApple OSS Distributions 
2686*43a90889SApple OSS Distributions 	MGETHDR(m, M_DONTWAIT, MT_DATA);
2687*43a90889SApple OSS Distributions 	if (m == NULL) {
2688*43a90889SApple OSS Distributions 		return ENOMEM;
2689*43a90889SApple OSS Distributions 	}
2690*43a90889SApple OSS Distributions 	MH_ALIGN(m, sizeof(struct ip) + sizeof(struct igmp));
2691*43a90889SApple OSS Distributions 
2692*43a90889SApple OSS Distributions 	m->m_pkthdr.len = sizeof(struct ip) + sizeof(struct igmp);
2693*43a90889SApple OSS Distributions 
2694*43a90889SApple OSS Distributions 	m->m_data += sizeof(struct ip);
2695*43a90889SApple OSS Distributions 	m->m_len = sizeof(struct igmp);
2696*43a90889SApple OSS Distributions 
2697*43a90889SApple OSS Distributions 	igmp = mtod(m, struct igmp *);
2698*43a90889SApple OSS Distributions 	igmp->igmp_type = (u_char)type;
2699*43a90889SApple OSS Distributions 	igmp->igmp_code = 0;
2700*43a90889SApple OSS Distributions 	igmp->igmp_group = inm->inm_addr;
2701*43a90889SApple OSS Distributions 	igmp->igmp_cksum = 0;
2702*43a90889SApple OSS Distributions 	igmp->igmp_cksum = in_cksum(m, sizeof(struct igmp));
2703*43a90889SApple OSS Distributions 
2704*43a90889SApple OSS Distributions 	m->m_data -= sizeof(struct ip);
2705*43a90889SApple OSS Distributions 	m->m_len += sizeof(struct ip);
2706*43a90889SApple OSS Distributions 
2707*43a90889SApple OSS Distributions 	ip = mtod(m, struct ip *);
2708*43a90889SApple OSS Distributions 	ip->ip_tos = 0;
2709*43a90889SApple OSS Distributions 	ip->ip_len = sizeof(struct ip) + sizeof(struct igmp);
2710*43a90889SApple OSS Distributions 	ip->ip_off = 0;
2711*43a90889SApple OSS Distributions 	ip->ip_p = IPPROTO_IGMP;
2712*43a90889SApple OSS Distributions 	ip->ip_src.s_addr = INADDR_ANY;
2713*43a90889SApple OSS Distributions 
2714*43a90889SApple OSS Distributions 	if (type == IGMP_HOST_LEAVE_MESSAGE) {
2715*43a90889SApple OSS Distributions 		ip->ip_dst.s_addr = htonl(INADDR_ALLRTRS_GROUP);
2716*43a90889SApple OSS Distributions 	} else {
2717*43a90889SApple OSS Distributions 		ip->ip_dst = inm->inm_addr;
2718*43a90889SApple OSS Distributions 	}
2719*43a90889SApple OSS Distributions 
2720*43a90889SApple OSS Distributions 	igmp_save_context(m, ifp);
2721*43a90889SApple OSS Distributions 
2722*43a90889SApple OSS Distributions 	m->m_flags |= M_IGMPV2;
2723*43a90889SApple OSS Distributions 	if (inm->inm_igi->igi_flags & IGIF_LOOPBACK) {
2724*43a90889SApple OSS Distributions 		m->m_flags |= M_IGMP_LOOP;
2725*43a90889SApple OSS Distributions 	}
2726*43a90889SApple OSS Distributions 
2727*43a90889SApple OSS Distributions 	/*
2728*43a90889SApple OSS Distributions 	 * Due to the fact that at this point we are possibly holding
2729*43a90889SApple OSS Distributions 	 * in_multihead_lock in shared or exclusive mode, we can't call
2730*43a90889SApple OSS Distributions 	 * igmp_sendpkt() here since that will eventually call ip_output(),
2731*43a90889SApple OSS Distributions 	 * which will try to lock in_multihead_lock and cause a deadlock.
2732*43a90889SApple OSS Distributions 	 * Instead we defer the work to the igmp_timeout() thread, thus
2733*43a90889SApple OSS Distributions 	 * avoiding unlocking in_multihead_lock here.
2734*43a90889SApple OSS Distributions 	 */
2735*43a90889SApple OSS Distributions 	if (IF_QFULL(&inm->inm_igi->igi_v2q)) {
2736*43a90889SApple OSS Distributions 		os_log_error(OS_LOG_DEFAULT,
2737*43a90889SApple OSS Distributions 		    "%s: v1 / v2 outbound queue full on %s\n",
2738*43a90889SApple OSS Distributions 		    __func__, if_name(ifp));
2739*43a90889SApple OSS Distributions 		error = ENOMEM;
2740*43a90889SApple OSS Distributions 		m_freem(m);
2741*43a90889SApple OSS Distributions 	} else {
2742*43a90889SApple OSS Distributions 		IF_ENQUEUE(&inm->inm_igi->igi_v2q, m);
2743*43a90889SApple OSS Distributions 		VERIFY(error == 0);
2744*43a90889SApple OSS Distributions 	}
2745*43a90889SApple OSS Distributions 	return error;
2746*43a90889SApple OSS Distributions }
2747*43a90889SApple OSS Distributions 
2748*43a90889SApple OSS Distributions /*
2749*43a90889SApple OSS Distributions  * Process a state change from the upper layer for the given IPv4 group.
2750*43a90889SApple OSS Distributions  *
2751*43a90889SApple OSS Distributions  * Each socket holds a reference on the in_multi in its own ip_moptions.
2752*43a90889SApple OSS Distributions  * The socket layer will have made the necessary updates to the group
2753*43a90889SApple OSS Distributions  * state, it is now up to IGMP to issue a state change report if there
2754*43a90889SApple OSS Distributions  * has been any change between T0 (when the last state-change was issued)
2755*43a90889SApple OSS Distributions  * and T1 (now).
2756*43a90889SApple OSS Distributions  *
2757*43a90889SApple OSS Distributions  * We use the IGMPv3 state machine at group level. The IGMP module
2758*43a90889SApple OSS Distributions  * however makes the decision as to which IGMP protocol version to speak.
2759*43a90889SApple OSS Distributions  * A state change *from* INCLUDE {} always means an initial join.
2760*43a90889SApple OSS Distributions  * A state change *to* INCLUDE {} always means a final leave.
2761*43a90889SApple OSS Distributions  *
2762*43a90889SApple OSS Distributions  * FUTURE: If IGIF_V3LITE is enabled for this interface, then we can
2763*43a90889SApple OSS Distributions  * save ourselves a bunch of work; any exclusive mode groups need not
2764*43a90889SApple OSS Distributions  * compute source filter lists.
2765*43a90889SApple OSS Distributions  */
2766*43a90889SApple OSS Distributions int
igmp_change_state(struct in_multi * inm,struct igmp_tparams * itp)2767*43a90889SApple OSS Distributions igmp_change_state(struct in_multi *inm, struct igmp_tparams *itp)
2768*43a90889SApple OSS Distributions {
2769*43a90889SApple OSS Distributions 	struct igmp_ifinfo *igi;
2770*43a90889SApple OSS Distributions 	struct ifnet *ifp;
2771*43a90889SApple OSS Distributions 	int error = 0;
2772*43a90889SApple OSS Distributions 
2773*43a90889SApple OSS Distributions 	VERIFY(itp != NULL);
2774*43a90889SApple OSS Distributions 	bzero(itp, sizeof(*itp));
2775*43a90889SApple OSS Distributions 
2776*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2777*43a90889SApple OSS Distributions 	VERIFY(inm->inm_igi != NULL);
2778*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_NOTHELD(inm->inm_igi);
2779*43a90889SApple OSS Distributions 
2780*43a90889SApple OSS Distributions 	/*
2781*43a90889SApple OSS Distributions 	 * Try to detect if the upper layer just asked us to change state
2782*43a90889SApple OSS Distributions 	 * for an interface which has now gone away.
2783*43a90889SApple OSS Distributions 	 */
2784*43a90889SApple OSS Distributions 	VERIFY(inm->inm_ifma != NULL);
2785*43a90889SApple OSS Distributions 	ifp = inm->inm_ifma->ifma_ifp;
2786*43a90889SApple OSS Distributions 	/*
2787*43a90889SApple OSS Distributions 	 * Sanity check that netinet's notion of ifp is the same as net's.
2788*43a90889SApple OSS Distributions 	 */
2789*43a90889SApple OSS Distributions 	VERIFY(inm->inm_ifp == ifp);
2790*43a90889SApple OSS Distributions 
2791*43a90889SApple OSS Distributions 	igi = IGMP_IFINFO(ifp);
2792*43a90889SApple OSS Distributions 	VERIFY(igi != NULL);
2793*43a90889SApple OSS Distributions 
2794*43a90889SApple OSS Distributions 	/*
2795*43a90889SApple OSS Distributions 	 * If we detect a state transition to or from MCAST_UNDEFINED
2796*43a90889SApple OSS Distributions 	 * for this group, then we are starting or finishing an IGMP
2797*43a90889SApple OSS Distributions 	 * life cycle for this group.
2798*43a90889SApple OSS Distributions 	 */
2799*43a90889SApple OSS Distributions 	if (inm->inm_st[1].iss_fmode != inm->inm_st[0].iss_fmode) {
2800*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: inm transition %d -> %d\n", __func__,
2801*43a90889SApple OSS Distributions 		    inm->inm_st[0].iss_fmode, inm->inm_st[1].iss_fmode));
2802*43a90889SApple OSS Distributions 		if (inm->inm_st[0].iss_fmode == MCAST_UNDEFINED) {
2803*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: initial join\n", __func__));
2804*43a90889SApple OSS Distributions 			error = igmp_initial_join(inm, igi, itp);
2805*43a90889SApple OSS Distributions 			goto out;
2806*43a90889SApple OSS Distributions 		} else if (inm->inm_st[1].iss_fmode == MCAST_UNDEFINED) {
2807*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: final leave\n", __func__));
2808*43a90889SApple OSS Distributions 			igmp_final_leave(inm, igi, itp);
2809*43a90889SApple OSS Distributions 			goto out;
2810*43a90889SApple OSS Distributions 		}
2811*43a90889SApple OSS Distributions 	} else {
2812*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: filter set change\n", __func__));
2813*43a90889SApple OSS Distributions 	}
2814*43a90889SApple OSS Distributions 
2815*43a90889SApple OSS Distributions 	error = igmp_handle_state_change(inm, igi, itp);
2816*43a90889SApple OSS Distributions out:
2817*43a90889SApple OSS Distributions 	return error;
2818*43a90889SApple OSS Distributions }
2819*43a90889SApple OSS Distributions 
2820*43a90889SApple OSS Distributions /*
2821*43a90889SApple OSS Distributions  * Perform the initial join for an IGMP group.
2822*43a90889SApple OSS Distributions  *
2823*43a90889SApple OSS Distributions  * When joining a group:
2824*43a90889SApple OSS Distributions  *  If the group should have its IGMP traffic suppressed, do nothing.
2825*43a90889SApple OSS Distributions  *  IGMPv1 starts sending IGMPv1 host membership reports.
2826*43a90889SApple OSS Distributions  *  IGMPv2 starts sending IGMPv2 host membership reports.
2827*43a90889SApple OSS Distributions  *  IGMPv3 will schedule an IGMPv3 state-change report containing the
2828*43a90889SApple OSS Distributions  *  initial state of the membership.
2829*43a90889SApple OSS Distributions  */
2830*43a90889SApple OSS Distributions static int
igmp_initial_join(struct in_multi * inm,struct igmp_ifinfo * igi,struct igmp_tparams * itp)2831*43a90889SApple OSS Distributions igmp_initial_join(struct in_multi *inm, struct igmp_ifinfo *igi,
2832*43a90889SApple OSS Distributions     struct igmp_tparams *itp)
2833*43a90889SApple OSS Distributions {
2834*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
2835*43a90889SApple OSS Distributions 	struct ifqueue          *ifq;
2836*43a90889SApple OSS Distributions 	int                      error, retval, syncstates;
2837*43a90889SApple OSS Distributions 
2838*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2839*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_NOTHELD(igi);
2840*43a90889SApple OSS Distributions 	VERIFY(itp != NULL);
2841*43a90889SApple OSS Distributions 
2842*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(inm->inm_addr,
2843*43a90889SApple OSS Distributions 	    ("%s: initial join %s on ifp 0x%llx(%s)\n", __func__,
2844*43a90889SApple OSS Distributions 	    _igmp_inet_buf, (uint64_t)VM_KERNEL_ADDRPERM(inm->inm_ifp),
2845*43a90889SApple OSS Distributions 	    if_name(inm->inm_ifp)));
2846*43a90889SApple OSS Distributions 
2847*43a90889SApple OSS Distributions 	error = 0;
2848*43a90889SApple OSS Distributions 	syncstates = 1;
2849*43a90889SApple OSS Distributions 
2850*43a90889SApple OSS Distributions 	ifp = inm->inm_ifp;
2851*43a90889SApple OSS Distributions 
2852*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
2853*43a90889SApple OSS Distributions 	VERIFY(igi->igi_ifp == ifp);
2854*43a90889SApple OSS Distributions 
2855*43a90889SApple OSS Distributions 	/*
2856*43a90889SApple OSS Distributions 	 * Groups joined on loopback or marked as 'not reported',
2857*43a90889SApple OSS Distributions 	 * e.g. 224.0.0.1, enter the IGMP_SILENT_MEMBER state and
2858*43a90889SApple OSS Distributions 	 * are never reported in any IGMP protocol exchanges.
2859*43a90889SApple OSS Distributions 	 * All other groups enter the appropriate IGMP state machine
2860*43a90889SApple OSS Distributions 	 * for the version in use on this link.
2861*43a90889SApple OSS Distributions 	 * A link marked as IGIF_SILENT causes IGMP to be completely
2862*43a90889SApple OSS Distributions 	 * disabled for the link.
2863*43a90889SApple OSS Distributions 	 */
2864*43a90889SApple OSS Distributions 	if ((ifp->if_flags & IFF_LOOPBACK) ||
2865*43a90889SApple OSS Distributions 	    (igi->igi_flags & IGIF_SILENT) ||
2866*43a90889SApple OSS Distributions 	    !igmp_isgroupreported(inm->inm_addr)) {
2867*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: not kicking state machine for silent group\n",
2868*43a90889SApple OSS Distributions 		    __func__));
2869*43a90889SApple OSS Distributions 		inm->inm_state = IGMP_SILENT_MEMBER;
2870*43a90889SApple OSS Distributions 		inm->inm_timer = 0;
2871*43a90889SApple OSS Distributions 	} else {
2872*43a90889SApple OSS Distributions 		/*
2873*43a90889SApple OSS Distributions 		 * Deal with overlapping in_multi lifecycle.
2874*43a90889SApple OSS Distributions 		 * If this group was LEAVING, then make sure
2875*43a90889SApple OSS Distributions 		 * we drop the reference we picked up to keep the
2876*43a90889SApple OSS Distributions 		 * group around for the final INCLUDE {} enqueue.
2877*43a90889SApple OSS Distributions 		 * Since we cannot call in_multi_detach() here,
2878*43a90889SApple OSS Distributions 		 * defer this task to the timer routine.
2879*43a90889SApple OSS Distributions 		 */
2880*43a90889SApple OSS Distributions 		if (igi->igi_version == IGMP_VERSION_3 &&
2881*43a90889SApple OSS Distributions 		    inm->inm_state == IGMP_LEAVING_MEMBER) {
2882*43a90889SApple OSS Distributions 			VERIFY(inm->inm_nrelecnt != 0);
2883*43a90889SApple OSS Distributions 			igmp_append_relq(igi, inm);
2884*43a90889SApple OSS Distributions 		}
2885*43a90889SApple OSS Distributions 
2886*43a90889SApple OSS Distributions 		inm->inm_state = IGMP_REPORTING_MEMBER;
2887*43a90889SApple OSS Distributions 
2888*43a90889SApple OSS Distributions 		switch (igi->igi_version) {
2889*43a90889SApple OSS Distributions 		case IGMP_VERSION_1:
2890*43a90889SApple OSS Distributions 		case IGMP_VERSION_2:
2891*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_IDLE_MEMBER;
2892*43a90889SApple OSS Distributions 			error = igmp_v1v2_queue_report(inm,
2893*43a90889SApple OSS Distributions 			    (igi->igi_version == IGMP_VERSION_2) ?
2894*43a90889SApple OSS Distributions 			    IGMP_v2_HOST_MEMBERSHIP_REPORT :
2895*43a90889SApple OSS Distributions 			    IGMP_v1_HOST_MEMBERSHIP_REPORT);
2896*43a90889SApple OSS Distributions 
2897*43a90889SApple OSS Distributions 			INM_LOCK_ASSERT_HELD(inm);
2898*43a90889SApple OSS Distributions 			IGI_LOCK_ASSERT_HELD(igi);
2899*43a90889SApple OSS Distributions 
2900*43a90889SApple OSS Distributions 			if (error == 0) {
2901*43a90889SApple OSS Distributions 				inm->inm_timer =
2902*43a90889SApple OSS Distributions 				    IGMP_RANDOM_DELAY(IGMP_V1V2_MAX_RI);
2903*43a90889SApple OSS Distributions 				itp->cst = 1;
2904*43a90889SApple OSS Distributions 			}
2905*43a90889SApple OSS Distributions 			break;
2906*43a90889SApple OSS Distributions 
2907*43a90889SApple OSS Distributions 		case IGMP_VERSION_3:
2908*43a90889SApple OSS Distributions 			/*
2909*43a90889SApple OSS Distributions 			 * Defer update of T0 to T1, until the first copy
2910*43a90889SApple OSS Distributions 			 * of the state change has been transmitted.
2911*43a90889SApple OSS Distributions 			 */
2912*43a90889SApple OSS Distributions 			syncstates = 0;
2913*43a90889SApple OSS Distributions 
2914*43a90889SApple OSS Distributions 			/*
2915*43a90889SApple OSS Distributions 			 * Immediately enqueue a State-Change Report for
2916*43a90889SApple OSS Distributions 			 * this interface, freeing any previous reports.
2917*43a90889SApple OSS Distributions 			 * Don't kick the timers if there is nothing to do,
2918*43a90889SApple OSS Distributions 			 * or if an error occurred.
2919*43a90889SApple OSS Distributions 			 */
2920*43a90889SApple OSS Distributions 			ifq = &inm->inm_scq;
2921*43a90889SApple OSS Distributions 			IF_DRAIN(ifq);
2922*43a90889SApple OSS Distributions 			retval = igmp_v3_enqueue_group_record(ifq, inm, 1,
2923*43a90889SApple OSS Distributions 			    0, 0);
2924*43a90889SApple OSS Distributions 			itp->cst = (ifq->ifq_len > 0);
2925*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: enqueue record = %d\n",
2926*43a90889SApple OSS Distributions 			    __func__, retval));
2927*43a90889SApple OSS Distributions 			if (retval <= 0) {
2928*43a90889SApple OSS Distributions 				error = retval * -1;
2929*43a90889SApple OSS Distributions 				break;
2930*43a90889SApple OSS Distributions 			}
2931*43a90889SApple OSS Distributions 
2932*43a90889SApple OSS Distributions 			/*
2933*43a90889SApple OSS Distributions 			 * Schedule transmission of pending state-change
2934*43a90889SApple OSS Distributions 			 * report up to RV times for this link. The timer
2935*43a90889SApple OSS Distributions 			 * will fire at the next igmp_timeout (1 second),
2936*43a90889SApple OSS Distributions 			 * giving us an opportunity to merge the reports.
2937*43a90889SApple OSS Distributions 			 */
2938*43a90889SApple OSS Distributions 			if (igi->igi_flags & IGIF_LOOPBACK) {
2939*43a90889SApple OSS Distributions 				inm->inm_scrv = 1;
2940*43a90889SApple OSS Distributions 			} else {
2941*43a90889SApple OSS Distributions 				VERIFY(igi->igi_rv > 1);
2942*43a90889SApple OSS Distributions 				inm->inm_scrv = (uint16_t)igi->igi_rv;
2943*43a90889SApple OSS Distributions 			}
2944*43a90889SApple OSS Distributions 			inm->inm_sctimer = 1;
2945*43a90889SApple OSS Distributions 			itp->sct = 1;
2946*43a90889SApple OSS Distributions 
2947*43a90889SApple OSS Distributions 			error = 0;
2948*43a90889SApple OSS Distributions 			break;
2949*43a90889SApple OSS Distributions 		}
2950*43a90889SApple OSS Distributions 	}
2951*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
2952*43a90889SApple OSS Distributions 
2953*43a90889SApple OSS Distributions 	/*
2954*43a90889SApple OSS Distributions 	 * Only update the T0 state if state change is atomic,
2955*43a90889SApple OSS Distributions 	 * i.e. we don't need to wait for a timer to fire before we
2956*43a90889SApple OSS Distributions 	 * can consider the state change to have been communicated.
2957*43a90889SApple OSS Distributions 	 */
2958*43a90889SApple OSS Distributions 	if (syncstates) {
2959*43a90889SApple OSS Distributions 		inm_commit(inm);
2960*43a90889SApple OSS Distributions 		IGMP_INET_PRINTF(inm->inm_addr,
2961*43a90889SApple OSS Distributions 		    ("%s: T1->T0 for %s / %s\n", __func__,
2962*43a90889SApple OSS Distributions 		    _igmp_inet_buf, if_name(inm->inm_ifp)));
2963*43a90889SApple OSS Distributions 	}
2964*43a90889SApple OSS Distributions 
2965*43a90889SApple OSS Distributions 	return error;
2966*43a90889SApple OSS Distributions }
2967*43a90889SApple OSS Distributions 
2968*43a90889SApple OSS Distributions /*
2969*43a90889SApple OSS Distributions  * Issue an intermediate state change during the IGMP life-cycle.
2970*43a90889SApple OSS Distributions  */
2971*43a90889SApple OSS Distributions static int
igmp_handle_state_change(struct in_multi * inm,struct igmp_ifinfo * igi,struct igmp_tparams * itp)2972*43a90889SApple OSS Distributions igmp_handle_state_change(struct in_multi *inm, struct igmp_ifinfo *igi,
2973*43a90889SApple OSS Distributions     struct igmp_tparams *itp)
2974*43a90889SApple OSS Distributions {
2975*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
2976*43a90889SApple OSS Distributions 	int                      retval = 0;
2977*43a90889SApple OSS Distributions 
2978*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
2979*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_NOTHELD(igi);
2980*43a90889SApple OSS Distributions 	VERIFY(itp != NULL);
2981*43a90889SApple OSS Distributions 
2982*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(inm->inm_addr,
2983*43a90889SApple OSS Distributions 	    ("%s: state change for %s on ifp 0x%llx(%s)\n", __func__,
2984*43a90889SApple OSS Distributions 	    _igmp_inet_buf, (uint64_t)VM_KERNEL_ADDRPERM(inm->inm_ifp),
2985*43a90889SApple OSS Distributions 	    if_name(inm->inm_ifp)));
2986*43a90889SApple OSS Distributions 
2987*43a90889SApple OSS Distributions 	ifp = inm->inm_ifp;
2988*43a90889SApple OSS Distributions 
2989*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
2990*43a90889SApple OSS Distributions 	VERIFY(igi->igi_ifp == ifp);
2991*43a90889SApple OSS Distributions 
2992*43a90889SApple OSS Distributions 	if ((ifp->if_flags & IFF_LOOPBACK) ||
2993*43a90889SApple OSS Distributions 	    (igi->igi_flags & IGIF_SILENT) ||
2994*43a90889SApple OSS Distributions 	    !igmp_isgroupreported(inm->inm_addr) ||
2995*43a90889SApple OSS Distributions 	    (igi->igi_version != IGMP_VERSION_3)) {
2996*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
2997*43a90889SApple OSS Distributions 		if (!igmp_isgroupreported(inm->inm_addr)) {
2998*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: not kicking state "
2999*43a90889SApple OSS Distributions 			    "machine for silent group\n", __func__));
3000*43a90889SApple OSS Distributions 		}
3001*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: nothing to do \n", __func__));
3002*43a90889SApple OSS Distributions 		inm_commit(inm);
3003*43a90889SApple OSS Distributions 		IGMP_INET_PRINTF(inm->inm_addr,
3004*43a90889SApple OSS Distributions 		    ("%s: T1 -> T0 for %s/%s\n", __func__,
3005*43a90889SApple OSS Distributions 		    _igmp_inet_buf, inm->inm_ifp->if_name));
3006*43a90889SApple OSS Distributions 		goto done;
3007*43a90889SApple OSS Distributions 	}
3008*43a90889SApple OSS Distributions 
3009*43a90889SApple OSS Distributions 	IF_DRAIN(&inm->inm_scq);
3010*43a90889SApple OSS Distributions 
3011*43a90889SApple OSS Distributions 	retval = igmp_v3_enqueue_group_record(&inm->inm_scq, inm, 1, 0, 0);
3012*43a90889SApple OSS Distributions 	itp->cst = (inm->inm_scq.ifq_len > 0);
3013*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: enqueue record = %d\n", __func__, retval));
3014*43a90889SApple OSS Distributions 	if (retval <= 0) {
3015*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
3016*43a90889SApple OSS Distributions 		retval *= -1;
3017*43a90889SApple OSS Distributions 		goto done;
3018*43a90889SApple OSS Distributions 	}
3019*43a90889SApple OSS Distributions 	/*
3020*43a90889SApple OSS Distributions 	 * If record(s) were enqueued, start the state-change
3021*43a90889SApple OSS Distributions 	 * report timer for this group.
3022*43a90889SApple OSS Distributions 	 */
3023*43a90889SApple OSS Distributions 	inm->inm_scrv = ((igi->igi_flags & IGIF_LOOPBACK) ? 1 : (uint16_t)igi->igi_rv);
3024*43a90889SApple OSS Distributions 	inm->inm_sctimer = 1;
3025*43a90889SApple OSS Distributions 	itp->sct = 1;
3026*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
3027*43a90889SApple OSS Distributions done:
3028*43a90889SApple OSS Distributions 	return retval;
3029*43a90889SApple OSS Distributions }
3030*43a90889SApple OSS Distributions 
3031*43a90889SApple OSS Distributions /*
3032*43a90889SApple OSS Distributions  * Perform the final leave for an IGMP group.
3033*43a90889SApple OSS Distributions  *
3034*43a90889SApple OSS Distributions  * When leaving a group:
3035*43a90889SApple OSS Distributions  *  IGMPv1 does nothing.
3036*43a90889SApple OSS Distributions  *  IGMPv2 sends a host leave message, if and only if we are the reporter.
3037*43a90889SApple OSS Distributions  *  IGMPv3 enqueues a state-change report containing a transition
3038*43a90889SApple OSS Distributions  *  to INCLUDE {} for immediate transmission.
3039*43a90889SApple OSS Distributions  */
3040*43a90889SApple OSS Distributions static void
igmp_final_leave(struct in_multi * inm,struct igmp_ifinfo * igi,struct igmp_tparams * itp)3041*43a90889SApple OSS Distributions igmp_final_leave(struct in_multi *inm, struct igmp_ifinfo *igi,
3042*43a90889SApple OSS Distributions     struct igmp_tparams *itp)
3043*43a90889SApple OSS Distributions {
3044*43a90889SApple OSS Distributions 	int syncstates = 1;
3045*43a90889SApple OSS Distributions 	bool retried_already = false;
3046*43a90889SApple OSS Distributions 
3047*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
3048*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_NOTHELD(igi);
3049*43a90889SApple OSS Distributions 	VERIFY(itp != NULL);
3050*43a90889SApple OSS Distributions 
3051*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(inm->inm_addr,
3052*43a90889SApple OSS Distributions 	    ("%s: final leave %s on ifp 0x%llx(%s)\n", __func__,
3053*43a90889SApple OSS Distributions 	    _igmp_inet_buf, (uint64_t)VM_KERNEL_ADDRPERM(inm->inm_ifp),
3054*43a90889SApple OSS Distributions 	    if_name(inm->inm_ifp)));
3055*43a90889SApple OSS Distributions 
3056*43a90889SApple OSS Distributions retry:
3057*43a90889SApple OSS Distributions 	switch (inm->inm_state) {
3058*43a90889SApple OSS Distributions 	case IGMP_NOT_MEMBER:
3059*43a90889SApple OSS Distributions 	case IGMP_SILENT_MEMBER:
3060*43a90889SApple OSS Distributions 	case IGMP_LEAVING_MEMBER:
3061*43a90889SApple OSS Distributions 		/* Already leaving or left; do nothing. */
3062*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: not kicking state machine for silent group\n",
3063*43a90889SApple OSS Distributions 		    __func__));
3064*43a90889SApple OSS Distributions 		break;
3065*43a90889SApple OSS Distributions 	case IGMP_REPORTING_MEMBER:
3066*43a90889SApple OSS Distributions 	case IGMP_IDLE_MEMBER:
3067*43a90889SApple OSS Distributions 	case IGMP_G_QUERY_PENDING_MEMBER:
3068*43a90889SApple OSS Distributions 	case IGMP_SG_QUERY_PENDING_MEMBER:
3069*43a90889SApple OSS Distributions 		IGI_LOCK(igi);
3070*43a90889SApple OSS Distributions 		if (igi->igi_version == IGMP_VERSION_2) {
3071*43a90889SApple OSS Distributions 			if (inm->inm_state == IGMP_G_QUERY_PENDING_MEMBER ||
3072*43a90889SApple OSS Distributions 			    inm->inm_state == IGMP_SG_QUERY_PENDING_MEMBER) {
3073*43a90889SApple OSS Distributions 				/*
3074*43a90889SApple OSS Distributions 				 * We may be in the process of downgrading to
3075*43a90889SApple OSS Distributions 				 * IGMPv2 but because we just grabbed the
3076*43a90889SApple OSS Distributions 				 * igi_lock we may have lost the race.
3077*43a90889SApple OSS Distributions 				 */
3078*43a90889SApple OSS Distributions 				if (!retried_already) {
3079*43a90889SApple OSS Distributions 					IGI_UNLOCK(igi);
3080*43a90889SApple OSS Distributions 					retried_already = true;
3081*43a90889SApple OSS Distributions 					goto retry;
3082*43a90889SApple OSS Distributions 				} else {
3083*43a90889SApple OSS Distributions 					/*
3084*43a90889SApple OSS Distributions 					 * Proceed with leaving the group
3085*43a90889SApple OSS Distributions 					 * as if it were IGMPv2 even though we
3086*43a90889SApple OSS Distributions 					 * may have an inconsistent multicast state.
3087*43a90889SApple OSS Distributions 					 */
3088*43a90889SApple OSS Distributions 				}
3089*43a90889SApple OSS Distributions 			}
3090*43a90889SApple OSS Distributions 			/* scheduler timer if enqueue is successful */
3091*43a90889SApple OSS Distributions 			itp->cst = (igmp_v1v2_queue_report(inm,
3092*43a90889SApple OSS Distributions 			    IGMP_HOST_LEAVE_MESSAGE) == 0);
3093*43a90889SApple OSS Distributions 
3094*43a90889SApple OSS Distributions 			INM_LOCK_ASSERT_HELD(inm);
3095*43a90889SApple OSS Distributions 			IGI_LOCK_ASSERT_HELD(igi);
3096*43a90889SApple OSS Distributions 
3097*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_NOT_MEMBER;
3098*43a90889SApple OSS Distributions 		} else if (igi->igi_version == IGMP_VERSION_3) {
3099*43a90889SApple OSS Distributions 			/*
3100*43a90889SApple OSS Distributions 			 * Stop group timer and all pending reports.
3101*43a90889SApple OSS Distributions 			 * Immediately enqueue a state-change report
3102*43a90889SApple OSS Distributions 			 * TO_IN {} to be sent on the next timeout,
3103*43a90889SApple OSS Distributions 			 * giving us an opportunity to merge reports.
3104*43a90889SApple OSS Distributions 			 */
3105*43a90889SApple OSS Distributions 			IF_DRAIN(&inm->inm_scq);
3106*43a90889SApple OSS Distributions 			inm->inm_timer = 0;
3107*43a90889SApple OSS Distributions 			if (igi->igi_flags & IGIF_LOOPBACK) {
3108*43a90889SApple OSS Distributions 				inm->inm_scrv = 1;
3109*43a90889SApple OSS Distributions 			} else {
3110*43a90889SApple OSS Distributions 				inm->inm_scrv = (uint16_t)igi->igi_rv;
3111*43a90889SApple OSS Distributions 			}
3112*43a90889SApple OSS Distributions 			IGMP_INET_PRINTF(inm->inm_addr,
3113*43a90889SApple OSS Distributions 			    ("%s: Leaving %s/%s with %d "
3114*43a90889SApple OSS Distributions 			    "pending retransmissions.\n", __func__,
3115*43a90889SApple OSS Distributions 			    _igmp_inet_buf, if_name(inm->inm_ifp),
3116*43a90889SApple OSS Distributions 			    inm->inm_scrv));
3117*43a90889SApple OSS Distributions 			if (inm->inm_scrv == 0) {
3118*43a90889SApple OSS Distributions 				inm->inm_state = IGMP_NOT_MEMBER;
3119*43a90889SApple OSS Distributions 				inm->inm_sctimer = 0;
3120*43a90889SApple OSS Distributions 			} else {
3121*43a90889SApple OSS Distributions 				int retval;
3122*43a90889SApple OSS Distributions 				/*
3123*43a90889SApple OSS Distributions 				 * Stick around in the in_multihead list;
3124*43a90889SApple OSS Distributions 				 * the final detach will be issued by
3125*43a90889SApple OSS Distributions 				 * igmp_v3_process_group_timers() when
3126*43a90889SApple OSS Distributions 				 * the retransmit timer expires.
3127*43a90889SApple OSS Distributions 				 */
3128*43a90889SApple OSS Distributions 				INM_ADDREF_LOCKED(inm);
3129*43a90889SApple OSS Distributions 				VERIFY(inm->inm_debug & IFD_ATTACHED);
3130*43a90889SApple OSS Distributions 				inm->inm_reqcnt++;
3131*43a90889SApple OSS Distributions 				VERIFY(inm->inm_reqcnt >= 1);
3132*43a90889SApple OSS Distributions 				inm->inm_nrelecnt++;
3133*43a90889SApple OSS Distributions 				VERIFY(inm->inm_nrelecnt != 0);
3134*43a90889SApple OSS Distributions 
3135*43a90889SApple OSS Distributions 				retval = igmp_v3_enqueue_group_record(
3136*43a90889SApple OSS Distributions 					&inm->inm_scq, inm, 1, 0, 0);
3137*43a90889SApple OSS Distributions 				itp->cst = (inm->inm_scq.ifq_len > 0);
3138*43a90889SApple OSS Distributions 				KASSERT(retval != 0,
3139*43a90889SApple OSS Distributions 				    ("%s: enqueue record = %d\n", __func__,
3140*43a90889SApple OSS Distributions 				    retval));
3141*43a90889SApple OSS Distributions 
3142*43a90889SApple OSS Distributions 				inm->inm_state = IGMP_LEAVING_MEMBER;
3143*43a90889SApple OSS Distributions 				inm->inm_sctimer = 1;
3144*43a90889SApple OSS Distributions 				itp->sct = 1;
3145*43a90889SApple OSS Distributions 				syncstates = 0;
3146*43a90889SApple OSS Distributions 			}
3147*43a90889SApple OSS Distributions 		}
3148*43a90889SApple OSS Distributions 		IGI_UNLOCK(igi);
3149*43a90889SApple OSS Distributions 		break;
3150*43a90889SApple OSS Distributions 	case IGMP_LAZY_MEMBER:
3151*43a90889SApple OSS Distributions 	case IGMP_SLEEPING_MEMBER:
3152*43a90889SApple OSS Distributions 	case IGMP_AWAKENING_MEMBER:
3153*43a90889SApple OSS Distributions 		/* Our reports are suppressed; do nothing. */
3154*43a90889SApple OSS Distributions 		break;
3155*43a90889SApple OSS Distributions 	}
3156*43a90889SApple OSS Distributions 
3157*43a90889SApple OSS Distributions 	if (syncstates) {
3158*43a90889SApple OSS Distributions 		inm_commit(inm);
3159*43a90889SApple OSS Distributions 		IGMP_INET_PRINTF(inm->inm_addr,
3160*43a90889SApple OSS Distributions 		    ("%s: T1 -> T0 for %s/%s\n", __func__,
3161*43a90889SApple OSS Distributions 		    _igmp_inet_buf, if_name(inm->inm_ifp)));
3162*43a90889SApple OSS Distributions 		inm->inm_st[1].iss_fmode = MCAST_UNDEFINED;
3163*43a90889SApple OSS Distributions 		IGMP_INET_PRINTF(inm->inm_addr,
3164*43a90889SApple OSS Distributions 		    ("%s: T1 now MCAST_UNDEFINED for %s/%s\n",
3165*43a90889SApple OSS Distributions 		    __func__, _igmp_inet_buf, if_name(inm->inm_ifp)));
3166*43a90889SApple OSS Distributions 	}
3167*43a90889SApple OSS Distributions }
3168*43a90889SApple OSS Distributions 
3169*43a90889SApple OSS Distributions /*
3170*43a90889SApple OSS Distributions  * Enqueue an IGMPv3 group record to the given output queue.
3171*43a90889SApple OSS Distributions  *
3172*43a90889SApple OSS Distributions  * XXX This function could do with having the allocation code
3173*43a90889SApple OSS Distributions  * split out, and the multiple-tree-walks coalesced into a single
3174*43a90889SApple OSS Distributions  * routine as has been done in igmp_v3_enqueue_filter_change().
3175*43a90889SApple OSS Distributions  *
3176*43a90889SApple OSS Distributions  * If is_state_change is zero, a current-state record is appended.
3177*43a90889SApple OSS Distributions  * If is_state_change is non-zero, a state-change report is appended.
3178*43a90889SApple OSS Distributions  *
3179*43a90889SApple OSS Distributions  * If is_group_query is non-zero, an mbuf packet chain is allocated.
3180*43a90889SApple OSS Distributions  * If is_group_query is zero, and if there is a packet with free space
3181*43a90889SApple OSS Distributions  * at the tail of the queue, it will be appended to providing there
3182*43a90889SApple OSS Distributions  * is enough free space.
3183*43a90889SApple OSS Distributions  * Otherwise a new mbuf packet chain is allocated.
3184*43a90889SApple OSS Distributions  *
3185*43a90889SApple OSS Distributions  * If is_source_query is non-zero, each source is checked to see if
3186*43a90889SApple OSS Distributions  * it was recorded for a Group-Source query, and will be omitted if
3187*43a90889SApple OSS Distributions  * it is not both in-mode and recorded.
3188*43a90889SApple OSS Distributions  *
3189*43a90889SApple OSS Distributions  * The function will attempt to allocate leading space in the packet
3190*43a90889SApple OSS Distributions  * for the IP/IGMP header to be prepended without fragmenting the chain.
3191*43a90889SApple OSS Distributions  *
3192*43a90889SApple OSS Distributions  * If successful the size of all data appended to the queue is returned,
3193*43a90889SApple OSS Distributions  * otherwise an error code less than zero is returned, or zero if
3194*43a90889SApple OSS Distributions  * no record(s) were appended.
3195*43a90889SApple OSS Distributions  */
3196*43a90889SApple OSS Distributions static int
igmp_v3_enqueue_group_record(struct ifqueue * ifq,struct in_multi * inm,const int is_state_change,const int is_group_query,const int is_source_query)3197*43a90889SApple OSS Distributions igmp_v3_enqueue_group_record(struct ifqueue *ifq, struct in_multi *inm,
3198*43a90889SApple OSS Distributions     const int is_state_change, const int is_group_query,
3199*43a90889SApple OSS Distributions     const int is_source_query)
3200*43a90889SApple OSS Distributions {
3201*43a90889SApple OSS Distributions 	struct igmp_grouprec     ig;
3202*43a90889SApple OSS Distributions 	struct igmp_grouprec    *pig;
3203*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
3204*43a90889SApple OSS Distributions 	struct ip_msource       *ims, *nims;
3205*43a90889SApple OSS Distributions 	mbuf_ref_t               m0, m, md;
3206*43a90889SApple OSS Distributions 	int                      error, is_filter_list_change;
3207*43a90889SApple OSS Distributions 	int                      minrec0len, m0srcs, nbytes, off;
3208*43a90889SApple OSS Distributions 	uint16_t                 msrcs;
3209*43a90889SApple OSS Distributions 	int                      record_has_sources;
3210*43a90889SApple OSS Distributions 	int                      now;
3211*43a90889SApple OSS Distributions 	int                      type;
3212*43a90889SApple OSS Distributions 	in_addr_t                naddr;
3213*43a90889SApple OSS Distributions 	uint16_t                 mode;
3214*43a90889SApple OSS Distributions 	u_int16_t                ig_numsrc;
3215*43a90889SApple OSS Distributions 
3216*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
3217*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(inm->inm_igi);
3218*43a90889SApple OSS Distributions 
3219*43a90889SApple OSS Distributions 	error = 0;
3220*43a90889SApple OSS Distributions 	ifp = inm->inm_ifp;
3221*43a90889SApple OSS Distributions 	is_filter_list_change = 0;
3222*43a90889SApple OSS Distributions 	m = NULL;
3223*43a90889SApple OSS Distributions 	m0 = NULL;
3224*43a90889SApple OSS Distributions 	m0srcs = 0;
3225*43a90889SApple OSS Distributions 	msrcs = 0;
3226*43a90889SApple OSS Distributions 	nbytes = 0;
3227*43a90889SApple OSS Distributions 	nims = NULL;
3228*43a90889SApple OSS Distributions 	record_has_sources = 1;
3229*43a90889SApple OSS Distributions 	pig = NULL;
3230*43a90889SApple OSS Distributions 	type = IGMP_DO_NOTHING;
3231*43a90889SApple OSS Distributions 	mode = inm->inm_st[1].iss_fmode;
3232*43a90889SApple OSS Distributions 
3233*43a90889SApple OSS Distributions 	/*
3234*43a90889SApple OSS Distributions 	 * If we did not transition out of ASM mode during t0->t1,
3235*43a90889SApple OSS Distributions 	 * and there are no source nodes to process, we can skip
3236*43a90889SApple OSS Distributions 	 * the generation of source records.
3237*43a90889SApple OSS Distributions 	 */
3238*43a90889SApple OSS Distributions 	if (inm->inm_st[0].iss_asm > 0 && inm->inm_st[1].iss_asm > 0 &&
3239*43a90889SApple OSS Distributions 	    inm->inm_nsrc == 0) {
3240*43a90889SApple OSS Distributions 		record_has_sources = 0;
3241*43a90889SApple OSS Distributions 	}
3242*43a90889SApple OSS Distributions 
3243*43a90889SApple OSS Distributions 	if (is_state_change) {
3244*43a90889SApple OSS Distributions 		/*
3245*43a90889SApple OSS Distributions 		 * Queue a state change record.
3246*43a90889SApple OSS Distributions 		 * If the mode did not change, and there are non-ASM
3247*43a90889SApple OSS Distributions 		 * listeners or source filters present,
3248*43a90889SApple OSS Distributions 		 * we potentially need to issue two records for the group.
3249*43a90889SApple OSS Distributions 		 * If we are transitioning to MCAST_UNDEFINED, we need
3250*43a90889SApple OSS Distributions 		 * not send any sources.
3251*43a90889SApple OSS Distributions 		 * If there are ASM listeners, and there was no filter
3252*43a90889SApple OSS Distributions 		 * mode transition of any kind, do nothing.
3253*43a90889SApple OSS Distributions 		 */
3254*43a90889SApple OSS Distributions 		if (mode != inm->inm_st[0].iss_fmode) {
3255*43a90889SApple OSS Distributions 			if (mode == MCAST_EXCLUDE) {
3256*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: change to EXCLUDE\n",
3257*43a90889SApple OSS Distributions 				    __func__));
3258*43a90889SApple OSS Distributions 				type = IGMP_CHANGE_TO_EXCLUDE_MODE;
3259*43a90889SApple OSS Distributions 			} else {
3260*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: change to INCLUDE\n",
3261*43a90889SApple OSS Distributions 				    __func__));
3262*43a90889SApple OSS Distributions 				type = IGMP_CHANGE_TO_INCLUDE_MODE;
3263*43a90889SApple OSS Distributions 				if (mode == MCAST_UNDEFINED) {
3264*43a90889SApple OSS Distributions 					record_has_sources = 0;
3265*43a90889SApple OSS Distributions 				}
3266*43a90889SApple OSS Distributions 			}
3267*43a90889SApple OSS Distributions 		} else {
3268*43a90889SApple OSS Distributions 			if (record_has_sources) {
3269*43a90889SApple OSS Distributions 				is_filter_list_change = 1;
3270*43a90889SApple OSS Distributions 			} else {
3271*43a90889SApple OSS Distributions 				type = IGMP_DO_NOTHING;
3272*43a90889SApple OSS Distributions 			}
3273*43a90889SApple OSS Distributions 		}
3274*43a90889SApple OSS Distributions 	} else {
3275*43a90889SApple OSS Distributions 		/*
3276*43a90889SApple OSS Distributions 		 * Queue a current state record.
3277*43a90889SApple OSS Distributions 		 */
3278*43a90889SApple OSS Distributions 		if (mode == MCAST_EXCLUDE) {
3279*43a90889SApple OSS Distributions 			type = IGMP_MODE_IS_EXCLUDE;
3280*43a90889SApple OSS Distributions 		} else if (mode == MCAST_INCLUDE) {
3281*43a90889SApple OSS Distributions 			type = IGMP_MODE_IS_INCLUDE;
3282*43a90889SApple OSS Distributions 			VERIFY(inm->inm_st[1].iss_asm == 0);
3283*43a90889SApple OSS Distributions 		}
3284*43a90889SApple OSS Distributions 	}
3285*43a90889SApple OSS Distributions 
3286*43a90889SApple OSS Distributions 	/*
3287*43a90889SApple OSS Distributions 	 * Generate the filter list changes using a separate function.
3288*43a90889SApple OSS Distributions 	 */
3289*43a90889SApple OSS Distributions 	if (is_filter_list_change) {
3290*43a90889SApple OSS Distributions 		return igmp_v3_enqueue_filter_change(ifq, inm);
3291*43a90889SApple OSS Distributions 	}
3292*43a90889SApple OSS Distributions 
3293*43a90889SApple OSS Distributions 	if (type == IGMP_DO_NOTHING) {
3294*43a90889SApple OSS Distributions 		IGMP_INET_PRINTF(inm->inm_addr,
3295*43a90889SApple OSS Distributions 		    ("%s: nothing to do for %s/%s\n",
3296*43a90889SApple OSS Distributions 		    __func__, _igmp_inet_buf,
3297*43a90889SApple OSS Distributions 		    if_name(inm->inm_ifp)));
3298*43a90889SApple OSS Distributions 		return 0;
3299*43a90889SApple OSS Distributions 	}
3300*43a90889SApple OSS Distributions 
3301*43a90889SApple OSS Distributions 	/*
3302*43a90889SApple OSS Distributions 	 * If any sources are present, we must be able to fit at least
3303*43a90889SApple OSS Distributions 	 * one in the trailing space of the tail packet's mbuf,
3304*43a90889SApple OSS Distributions 	 * ideally more.
3305*43a90889SApple OSS Distributions 	 */
3306*43a90889SApple OSS Distributions 	minrec0len = sizeof(struct igmp_grouprec);
3307*43a90889SApple OSS Distributions 	if (record_has_sources) {
3308*43a90889SApple OSS Distributions 		minrec0len += sizeof(in_addr_t);
3309*43a90889SApple OSS Distributions 	}
3310*43a90889SApple OSS Distributions 
3311*43a90889SApple OSS Distributions 	IGMP_INET_PRINTF(inm->inm_addr,
3312*43a90889SApple OSS Distributions 	    ("%s: queueing %s for %s/%s\n", __func__,
3313*43a90889SApple OSS Distributions 	    igmp_rec_type_to_str(type), _igmp_inet_buf,
3314*43a90889SApple OSS Distributions 	    if_name(inm->inm_ifp)));
3315*43a90889SApple OSS Distributions 
3316*43a90889SApple OSS Distributions 	/*
3317*43a90889SApple OSS Distributions 	 * Check if we have a packet in the tail of the queue for this
3318*43a90889SApple OSS Distributions 	 * group into which the first group record for this group will fit.
3319*43a90889SApple OSS Distributions 	 * Otherwise allocate a new packet.
3320*43a90889SApple OSS Distributions 	 * Always allocate leading space for IP+RA_OPT+IGMP+REPORT.
3321*43a90889SApple OSS Distributions 	 * Note: Group records for G/GSR query responses MUST be sent
3322*43a90889SApple OSS Distributions 	 * in their own packet.
3323*43a90889SApple OSS Distributions 	 */
3324*43a90889SApple OSS Distributions 	m0 = ifq->ifq_tail;
3325*43a90889SApple OSS Distributions 	if (!is_group_query &&
3326*43a90889SApple OSS Distributions 	    m0 != NULL &&
3327*43a90889SApple OSS Distributions 	    (m0->m_pkthdr.vt_nrecs + 1 <= IGMP_V3_REPORT_MAXRECS) &&
3328*43a90889SApple OSS Distributions 	    (m0->m_pkthdr.len + minrec0len) <
3329*43a90889SApple OSS Distributions 	    (ifp->if_mtu - IGMP_LEADINGSPACE)) {
3330*43a90889SApple OSS Distributions 		m0srcs = (ifp->if_mtu - m0->m_pkthdr.len -
3331*43a90889SApple OSS Distributions 		    sizeof(struct igmp_grouprec)) / sizeof(in_addr_t);
3332*43a90889SApple OSS Distributions 		m = m0;
3333*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: use existing packet\n", __func__));
3334*43a90889SApple OSS Distributions 	} else {
3335*43a90889SApple OSS Distributions 		if (IF_QFULL(ifq)) {
3336*43a90889SApple OSS Distributions 			os_log_error(OS_LOG_DEFAULT,
3337*43a90889SApple OSS Distributions 			    "%s: outbound queue full on %s\n", __func__, if_name(ifp));
3338*43a90889SApple OSS Distributions 			return -ENOMEM;
3339*43a90889SApple OSS Distributions 		}
3340*43a90889SApple OSS Distributions 		m = NULL;
3341*43a90889SApple OSS Distributions 		m0srcs = (ifp->if_mtu - IGMP_LEADINGSPACE -
3342*43a90889SApple OSS Distributions 		    sizeof(struct igmp_grouprec)) / sizeof(in_addr_t);
3343*43a90889SApple OSS Distributions 		if (!is_state_change && !is_group_query) {
3344*43a90889SApple OSS Distributions 			m = m_getcl(M_DONTWAIT, MT_DATA, M_PKTHDR);
3345*43a90889SApple OSS Distributions 			if (m) {
3346*43a90889SApple OSS Distributions 				m->m_data += IGMP_LEADINGSPACE;
3347*43a90889SApple OSS Distributions 			}
3348*43a90889SApple OSS Distributions 		}
3349*43a90889SApple OSS Distributions 		if (m == NULL) {
3350*43a90889SApple OSS Distributions 			m = m_gethdr(M_DONTWAIT, MT_DATA);
3351*43a90889SApple OSS Distributions 			if (m) {
3352*43a90889SApple OSS Distributions 				MH_ALIGN(m, IGMP_LEADINGSPACE);
3353*43a90889SApple OSS Distributions 			}
3354*43a90889SApple OSS Distributions 		}
3355*43a90889SApple OSS Distributions 		if (m == NULL) {
3356*43a90889SApple OSS Distributions 			return -ENOMEM;
3357*43a90889SApple OSS Distributions 		}
3358*43a90889SApple OSS Distributions 
3359*43a90889SApple OSS Distributions 		igmp_save_context(m, ifp);
3360*43a90889SApple OSS Distributions 
3361*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: allocated first packet\n", __func__));
3362*43a90889SApple OSS Distributions 	}
3363*43a90889SApple OSS Distributions 
3364*43a90889SApple OSS Distributions 	/*
3365*43a90889SApple OSS Distributions 	 * Append group record.
3366*43a90889SApple OSS Distributions 	 * If we have sources, we don't know how many yet.
3367*43a90889SApple OSS Distributions 	 */
3368*43a90889SApple OSS Distributions 	ig.ig_type = (u_char)type;
3369*43a90889SApple OSS Distributions 	ig.ig_datalen = 0;
3370*43a90889SApple OSS Distributions 	ig.ig_numsrc = 0;
3371*43a90889SApple OSS Distributions 	ig.ig_group = inm->inm_addr;
3372*43a90889SApple OSS Distributions 	if (!m_append(m, sizeof(struct igmp_grouprec), (void *)&ig)) {
3373*43a90889SApple OSS Distributions 		if (m != m0) {
3374*43a90889SApple OSS Distributions 			m_freem(m);
3375*43a90889SApple OSS Distributions 		}
3376*43a90889SApple OSS Distributions 		os_log_error(OS_LOG_DEFAULT, "%s: m_append() failed\n", __func__);
3377*43a90889SApple OSS Distributions 		return -ENOMEM;
3378*43a90889SApple OSS Distributions 	}
3379*43a90889SApple OSS Distributions 	nbytes += sizeof(struct igmp_grouprec);
3380*43a90889SApple OSS Distributions 
3381*43a90889SApple OSS Distributions 	/*
3382*43a90889SApple OSS Distributions 	 * Append as many sources as will fit in the first packet.
3383*43a90889SApple OSS Distributions 	 * If we are appending to a new packet, the chain allocation
3384*43a90889SApple OSS Distributions 	 * may potentially use clusters; use m_getptr() in this case.
3385*43a90889SApple OSS Distributions 	 * If we are appending to an existing packet, we need to obtain
3386*43a90889SApple OSS Distributions 	 * a pointer to the group record after m_append(), in case a new
3387*43a90889SApple OSS Distributions 	 * mbuf was allocated.
3388*43a90889SApple OSS Distributions 	 * Only append sources which are in-mode at t1. If we are
3389*43a90889SApple OSS Distributions 	 * transitioning to MCAST_UNDEFINED state on the group, do not
3390*43a90889SApple OSS Distributions 	 * include source entries.
3391*43a90889SApple OSS Distributions 	 * Only report recorded sources in our filter set when responding
3392*43a90889SApple OSS Distributions 	 * to a group-source query.
3393*43a90889SApple OSS Distributions 	 */
3394*43a90889SApple OSS Distributions 	if (record_has_sources) {
3395*43a90889SApple OSS Distributions 		if (m == m0) {
3396*43a90889SApple OSS Distributions 			md = m_last(m);
3397*43a90889SApple OSS Distributions 			pig = (struct igmp_grouprec *)(void *)
3398*43a90889SApple OSS Distributions 			    (mtod(md, uint8_t *) + md->m_len - nbytes);
3399*43a90889SApple OSS Distributions 		} else {
3400*43a90889SApple OSS Distributions 			md = m_getptr(m, 0, &off);
3401*43a90889SApple OSS Distributions 			pig = (struct igmp_grouprec *)(void *)
3402*43a90889SApple OSS Distributions 			    (mtod(md, uint8_t *) + off);
3403*43a90889SApple OSS Distributions 		}
3404*43a90889SApple OSS Distributions 		msrcs = 0;
3405*43a90889SApple OSS Distributions 		RB_FOREACH_SAFE(ims, ip_msource_tree, &inm->inm_srcs, nims) {
3406*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
3407*43a90889SApple OSS Distributions 			char buf[MAX_IPv4_STR_LEN];
3408*43a90889SApple OSS Distributions 
3409*43a90889SApple OSS Distributions 			inet_ntop_haddr(ims->ims_haddr, buf, sizeof(buf));
3410*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: visit node %s\n", __func__, buf));
3411*43a90889SApple OSS Distributions #endif
3412*43a90889SApple OSS Distributions 			now = ims_get_mode(inm, ims, 1);
3413*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: node is %d\n", __func__, now));
3414*43a90889SApple OSS Distributions 			if ((now != mode) ||
3415*43a90889SApple OSS Distributions 			    (now == mode && mode == MCAST_UNDEFINED)) {
3416*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: skip node\n", __func__));
3417*43a90889SApple OSS Distributions 				continue;
3418*43a90889SApple OSS Distributions 			}
3419*43a90889SApple OSS Distributions 			if (is_source_query && ims->ims_stp == 0) {
3420*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: skip unrecorded node\n",
3421*43a90889SApple OSS Distributions 				    __func__));
3422*43a90889SApple OSS Distributions 				continue;
3423*43a90889SApple OSS Distributions 			}
3424*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: append node\n", __func__));
3425*43a90889SApple OSS Distributions 			naddr = htonl(ims->ims_haddr);
3426*43a90889SApple OSS Distributions 			if (!m_append(m, sizeof(in_addr_t), (void *)&naddr)) {
3427*43a90889SApple OSS Distributions 				if (m != m0) {
3428*43a90889SApple OSS Distributions 					m_freem(m);
3429*43a90889SApple OSS Distributions 				}
3430*43a90889SApple OSS Distributions 				os_log_error(OS_LOG_DEFAULT, "%s: m_append() failed\n",
3431*43a90889SApple OSS Distributions 				    __func__);
3432*43a90889SApple OSS Distributions 				return -ENOMEM;
3433*43a90889SApple OSS Distributions 			}
3434*43a90889SApple OSS Distributions 			nbytes += sizeof(in_addr_t);
3435*43a90889SApple OSS Distributions 			++msrcs;
3436*43a90889SApple OSS Distributions 			if (msrcs == m0srcs) {
3437*43a90889SApple OSS Distributions 				break;
3438*43a90889SApple OSS Distributions 			}
3439*43a90889SApple OSS Distributions 		}
3440*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: msrcs is %d this packet\n", __func__,
3441*43a90889SApple OSS Distributions 		    msrcs));
3442*43a90889SApple OSS Distributions 		ig_numsrc = htons(msrcs);
3443*43a90889SApple OSS Distributions 		bcopy(&ig_numsrc, &pig->ig_numsrc, sizeof(ig_numsrc));
3444*43a90889SApple OSS Distributions 		nbytes += (msrcs * sizeof(in_addr_t));
3445*43a90889SApple OSS Distributions 	}
3446*43a90889SApple OSS Distributions 
3447*43a90889SApple OSS Distributions 	if (is_source_query && msrcs == 0) {
3448*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: no recorded sources to report\n", __func__));
3449*43a90889SApple OSS Distributions 		if (m != m0) {
3450*43a90889SApple OSS Distributions 			m_freem(m);
3451*43a90889SApple OSS Distributions 		}
3452*43a90889SApple OSS Distributions 		return 0;
3453*43a90889SApple OSS Distributions 	}
3454*43a90889SApple OSS Distributions 
3455*43a90889SApple OSS Distributions 	/*
3456*43a90889SApple OSS Distributions 	 * We are good to go with first packet.
3457*43a90889SApple OSS Distributions 	 */
3458*43a90889SApple OSS Distributions 	if (m != m0) {
3459*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: enqueueing first packet\n", __func__));
3460*43a90889SApple OSS Distributions 		m->m_pkthdr.vt_nrecs = 1;
3461*43a90889SApple OSS Distributions 		IF_ENQUEUE(ifq, m);
3462*43a90889SApple OSS Distributions 	} else {
3463*43a90889SApple OSS Distributions 		m->m_pkthdr.vt_nrecs++;
3464*43a90889SApple OSS Distributions 	}
3465*43a90889SApple OSS Distributions 	/*
3466*43a90889SApple OSS Distributions 	 * No further work needed if no source list in packet(s).
3467*43a90889SApple OSS Distributions 	 */
3468*43a90889SApple OSS Distributions 	if (!record_has_sources) {
3469*43a90889SApple OSS Distributions 		return nbytes;
3470*43a90889SApple OSS Distributions 	}
3471*43a90889SApple OSS Distributions 
3472*43a90889SApple OSS Distributions 	/*
3473*43a90889SApple OSS Distributions 	 * Whilst sources remain to be announced, we need to allocate
3474*43a90889SApple OSS Distributions 	 * a new packet and fill out as many sources as will fit.
3475*43a90889SApple OSS Distributions 	 * Always try for a cluster first.
3476*43a90889SApple OSS Distributions 	 */
3477*43a90889SApple OSS Distributions 	while (nims != NULL) {
3478*43a90889SApple OSS Distributions 		if (IF_QFULL(ifq)) {
3479*43a90889SApple OSS Distributions 			os_log_error(OS_LOG_DEFAULT, "%s: outbound queue full on %s\n",
3480*43a90889SApple OSS Distributions 			    __func__, if_name(ifp));
3481*43a90889SApple OSS Distributions 			return -ENOMEM;
3482*43a90889SApple OSS Distributions 		}
3483*43a90889SApple OSS Distributions 		m = m_getcl(M_DONTWAIT, MT_DATA, M_PKTHDR);
3484*43a90889SApple OSS Distributions 		if (m) {
3485*43a90889SApple OSS Distributions 			m->m_data += IGMP_LEADINGSPACE;
3486*43a90889SApple OSS Distributions 		}
3487*43a90889SApple OSS Distributions 		if (m == NULL) {
3488*43a90889SApple OSS Distributions 			m = m_gethdr(M_DONTWAIT, MT_DATA);
3489*43a90889SApple OSS Distributions 			if (m) {
3490*43a90889SApple OSS Distributions 				MH_ALIGN(m, IGMP_LEADINGSPACE);
3491*43a90889SApple OSS Distributions 			}
3492*43a90889SApple OSS Distributions 		}
3493*43a90889SApple OSS Distributions 		if (m == NULL) {
3494*43a90889SApple OSS Distributions 			return -ENOMEM;
3495*43a90889SApple OSS Distributions 		}
3496*43a90889SApple OSS Distributions 		igmp_save_context(m, ifp);
3497*43a90889SApple OSS Distributions 		md = m_getptr(m, 0, &off);
3498*43a90889SApple OSS Distributions 		pig = (struct igmp_grouprec *)(void *)
3499*43a90889SApple OSS Distributions 		    (mtod(md, uint8_t *) + off);
3500*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: allocated next packet\n", __func__));
3501*43a90889SApple OSS Distributions 
3502*43a90889SApple OSS Distributions 		if (!m_append(m, sizeof(struct igmp_grouprec), (void *)&ig)) {
3503*43a90889SApple OSS Distributions 			if (m != m0) {
3504*43a90889SApple OSS Distributions 				m_freem(m);
3505*43a90889SApple OSS Distributions 			}
3506*43a90889SApple OSS Distributions 			os_log_error(OS_LOG_DEFAULT, "%s: m_append() failed\n",
3507*43a90889SApple OSS Distributions 			    __func__);
3508*43a90889SApple OSS Distributions 			return -ENOMEM;
3509*43a90889SApple OSS Distributions 		}
3510*43a90889SApple OSS Distributions 		m->m_pkthdr.vt_nrecs = 1;
3511*43a90889SApple OSS Distributions 		nbytes += sizeof(struct igmp_grouprec);
3512*43a90889SApple OSS Distributions 
3513*43a90889SApple OSS Distributions 		m0srcs = (ifp->if_mtu - IGMP_LEADINGSPACE -
3514*43a90889SApple OSS Distributions 		    sizeof(struct igmp_grouprec)) / sizeof(in_addr_t);
3515*43a90889SApple OSS Distributions 
3516*43a90889SApple OSS Distributions 		msrcs = 0;
3517*43a90889SApple OSS Distributions 		RB_FOREACH_FROM(ims, ip_msource_tree, nims) {
3518*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
3519*43a90889SApple OSS Distributions 			char buf[MAX_IPv4_STR_LEN];
3520*43a90889SApple OSS Distributions 
3521*43a90889SApple OSS Distributions 			inet_ntop_haddr(ims->ims_haddr, buf, sizeof(buf));
3522*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: visit node %s\n", __func__, buf));
3523*43a90889SApple OSS Distributions #endif
3524*43a90889SApple OSS Distributions 			now = ims_get_mode(inm, ims, 1);
3525*43a90889SApple OSS Distributions 			if ((now != mode) ||
3526*43a90889SApple OSS Distributions 			    (now == mode && mode == MCAST_UNDEFINED)) {
3527*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: skip node\n", __func__));
3528*43a90889SApple OSS Distributions 				continue;
3529*43a90889SApple OSS Distributions 			}
3530*43a90889SApple OSS Distributions 			if (is_source_query && ims->ims_stp == 0) {
3531*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: skip unrecorded node\n",
3532*43a90889SApple OSS Distributions 				    __func__));
3533*43a90889SApple OSS Distributions 				continue;
3534*43a90889SApple OSS Distributions 			}
3535*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: append node\n", __func__));
3536*43a90889SApple OSS Distributions 			naddr = htonl(ims->ims_haddr);
3537*43a90889SApple OSS Distributions 			if (!m_append(m, sizeof(in_addr_t), (void *)&naddr)) {
3538*43a90889SApple OSS Distributions 				if (m != m0) {
3539*43a90889SApple OSS Distributions 					m_freem(m);
3540*43a90889SApple OSS Distributions 				}
3541*43a90889SApple OSS Distributions 				os_log_error(OS_LOG_DEFAULT, "%s: m_append() failed",
3542*43a90889SApple OSS Distributions 				    __func__);
3543*43a90889SApple OSS Distributions 				return -ENOMEM;
3544*43a90889SApple OSS Distributions 			}
3545*43a90889SApple OSS Distributions 			++msrcs;
3546*43a90889SApple OSS Distributions 			if (msrcs == m0srcs) {
3547*43a90889SApple OSS Distributions 				break;
3548*43a90889SApple OSS Distributions 			}
3549*43a90889SApple OSS Distributions 		}
3550*43a90889SApple OSS Distributions 		ig_numsrc = htons(msrcs);
3551*43a90889SApple OSS Distributions 		bcopy(&ig_numsrc, &pig->ig_numsrc, sizeof(ig_numsrc));
3552*43a90889SApple OSS Distributions 		nbytes += (msrcs * sizeof(in_addr_t));
3553*43a90889SApple OSS Distributions 
3554*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: enqueueing next packet\n", __func__));
3555*43a90889SApple OSS Distributions 		IF_ENQUEUE(ifq, m);
3556*43a90889SApple OSS Distributions 	}
3557*43a90889SApple OSS Distributions 
3558*43a90889SApple OSS Distributions 	return nbytes;
3559*43a90889SApple OSS Distributions }
3560*43a90889SApple OSS Distributions 
3561*43a90889SApple OSS Distributions /*
3562*43a90889SApple OSS Distributions  * Type used to mark record pass completion.
3563*43a90889SApple OSS Distributions  * We exploit the fact we can cast to this easily from the
3564*43a90889SApple OSS Distributions  * current filter modes on each ip_msource node.
3565*43a90889SApple OSS Distributions  */
3566*43a90889SApple OSS Distributions typedef enum {
3567*43a90889SApple OSS Distributions 	REC_NONE = 0x00,        /* MCAST_UNDEFINED */
3568*43a90889SApple OSS Distributions 	REC_ALLOW = 0x01,       /* MCAST_INCLUDE */
3569*43a90889SApple OSS Distributions 	REC_BLOCK = 0x02,       /* MCAST_EXCLUDE */
3570*43a90889SApple OSS Distributions 	REC_FULL = REC_ALLOW | REC_BLOCK
3571*43a90889SApple OSS Distributions } rectype_t;
3572*43a90889SApple OSS Distributions 
3573*43a90889SApple OSS Distributions /*
3574*43a90889SApple OSS Distributions  * Enqueue an IGMPv3 filter list change to the given output queue.
3575*43a90889SApple OSS Distributions  *
3576*43a90889SApple OSS Distributions  * Source list filter state is held in an RB-tree. When the filter list
3577*43a90889SApple OSS Distributions  * for a group is changed without changing its mode, we need to compute
3578*43a90889SApple OSS Distributions  * the deltas between T0 and T1 for each source in the filter set,
3579*43a90889SApple OSS Distributions  * and enqueue the appropriate ALLOW_NEW/BLOCK_OLD records.
3580*43a90889SApple OSS Distributions  *
3581*43a90889SApple OSS Distributions  * As we may potentially queue two record types, and the entire R-B tree
3582*43a90889SApple OSS Distributions  * needs to be walked at once, we break this out into its own function
3583*43a90889SApple OSS Distributions  * so we can generate a tightly packed queue of packets.
3584*43a90889SApple OSS Distributions  *
3585*43a90889SApple OSS Distributions  * XXX This could be written to only use one tree walk, although that makes
3586*43a90889SApple OSS Distributions  * serializing into the mbuf chains a bit harder. For now we do two walks
3587*43a90889SApple OSS Distributions  * which makes things easier on us, and it may or may not be harder on
3588*43a90889SApple OSS Distributions  * the L2 cache.
3589*43a90889SApple OSS Distributions  *
3590*43a90889SApple OSS Distributions  * If successful the size of all data appended to the queue is returned,
3591*43a90889SApple OSS Distributions  * otherwise an error code less than zero is returned, or zero if
3592*43a90889SApple OSS Distributions  * no record(s) were appended.
3593*43a90889SApple OSS Distributions  */
3594*43a90889SApple OSS Distributions static int
igmp_v3_enqueue_filter_change(struct ifqueue * ifq,struct in_multi * inm)3595*43a90889SApple OSS Distributions igmp_v3_enqueue_filter_change(struct ifqueue *ifq, struct in_multi *inm)
3596*43a90889SApple OSS Distributions {
3597*43a90889SApple OSS Distributions 	static const int MINRECLEN =
3598*43a90889SApple OSS Distributions 	    sizeof(struct igmp_grouprec) + sizeof(in_addr_t);
3599*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
3600*43a90889SApple OSS Distributions 	struct igmp_grouprec     ig;
3601*43a90889SApple OSS Distributions 	struct igmp_grouprec    *pig;
3602*43a90889SApple OSS Distributions 	struct ip_msource       *ims, *nims;
3603*43a90889SApple OSS Distributions 	mbuf_ref_t               m0, m, md;
3604*43a90889SApple OSS Distributions 	in_addr_t                naddr;
3605*43a90889SApple OSS Distributions 	int                      m0srcs, nbytes, npbytes, off, schanged;
3606*43a90889SApple OSS Distributions 	uint16_t                 rsrcs;
3607*43a90889SApple OSS Distributions 	int                      nallow, nblock;
3608*43a90889SApple OSS Distributions 	uint16_t                 mode;
3609*43a90889SApple OSS Distributions 	uint8_t                  now, then;
3610*43a90889SApple OSS Distributions 	rectype_t                crt, drt, nrt;
3611*43a90889SApple OSS Distributions 	u_int16_t                ig_numsrc;
3612*43a90889SApple OSS Distributions 
3613*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
3614*43a90889SApple OSS Distributions 
3615*43a90889SApple OSS Distributions 	if (inm->inm_nsrc == 0 ||
3616*43a90889SApple OSS Distributions 	    (inm->inm_st[0].iss_asm > 0 && inm->inm_st[1].iss_asm > 0)) {
3617*43a90889SApple OSS Distributions 		return 0;
3618*43a90889SApple OSS Distributions 	}
3619*43a90889SApple OSS Distributions 
3620*43a90889SApple OSS Distributions 	ifp = inm->inm_ifp;                     /* interface */
3621*43a90889SApple OSS Distributions 	mode = inm->inm_st[1].iss_fmode;        /* filter mode at t1 */
3622*43a90889SApple OSS Distributions 	crt = REC_NONE; /* current group record type */
3623*43a90889SApple OSS Distributions 	drt = REC_NONE; /* mask of completed group record types */
3624*43a90889SApple OSS Distributions 	nrt = REC_NONE; /* record type for current node */
3625*43a90889SApple OSS Distributions 	m0srcs = 0;     /* # source which will fit in current mbuf chain */
3626*43a90889SApple OSS Distributions 	nbytes = 0;     /* # of bytes appended to group's state-change queue */
3627*43a90889SApple OSS Distributions 	npbytes = 0;    /* # of bytes appended this packet */
3628*43a90889SApple OSS Distributions 	rsrcs = 0;      /* # sources encoded in current record */
3629*43a90889SApple OSS Distributions 	schanged = 0;   /* # nodes encoded in overall filter change */
3630*43a90889SApple OSS Distributions 	nallow = 0;     /* # of source entries in ALLOW_NEW */
3631*43a90889SApple OSS Distributions 	nblock = 0;     /* # of source entries in BLOCK_OLD */
3632*43a90889SApple OSS Distributions 	nims = NULL;    /* next tree node pointer */
3633*43a90889SApple OSS Distributions 
3634*43a90889SApple OSS Distributions 	/*
3635*43a90889SApple OSS Distributions 	 * For each possible filter record mode.
3636*43a90889SApple OSS Distributions 	 * The first kind of source we encounter tells us which
3637*43a90889SApple OSS Distributions 	 * is the first kind of record we start appending.
3638*43a90889SApple OSS Distributions 	 * If a node transitioned to UNDEFINED at t1, its mode is treated
3639*43a90889SApple OSS Distributions 	 * as the inverse of the group's filter mode.
3640*43a90889SApple OSS Distributions 	 */
3641*43a90889SApple OSS Distributions 	while (drt != REC_FULL) {
3642*43a90889SApple OSS Distributions 		do {
3643*43a90889SApple OSS Distributions 			m0 = ifq->ifq_tail;
3644*43a90889SApple OSS Distributions 			if (m0 != NULL &&
3645*43a90889SApple OSS Distributions 			    (m0->m_pkthdr.vt_nrecs + 1 <=
3646*43a90889SApple OSS Distributions 			    IGMP_V3_REPORT_MAXRECS) &&
3647*43a90889SApple OSS Distributions 			    (m0->m_pkthdr.len + MINRECLEN) <
3648*43a90889SApple OSS Distributions 			    (ifp->if_mtu - IGMP_LEADINGSPACE)) {
3649*43a90889SApple OSS Distributions 				m = m0;
3650*43a90889SApple OSS Distributions 				m0srcs = (ifp->if_mtu - m0->m_pkthdr.len -
3651*43a90889SApple OSS Distributions 				    sizeof(struct igmp_grouprec)) /
3652*43a90889SApple OSS Distributions 				    sizeof(in_addr_t);
3653*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: use previous packet\n",
3654*43a90889SApple OSS Distributions 				    __func__));
3655*43a90889SApple OSS Distributions 			} else {
3656*43a90889SApple OSS Distributions 				m = m_getcl(M_DONTWAIT, MT_DATA, M_PKTHDR);
3657*43a90889SApple OSS Distributions 				if (m) {
3658*43a90889SApple OSS Distributions 					m->m_data += IGMP_LEADINGSPACE;
3659*43a90889SApple OSS Distributions 				}
3660*43a90889SApple OSS Distributions 				if (m == NULL) {
3661*43a90889SApple OSS Distributions 					m = m_gethdr(M_DONTWAIT, MT_DATA);
3662*43a90889SApple OSS Distributions 					if (m) {
3663*43a90889SApple OSS Distributions 						MH_ALIGN(m, IGMP_LEADINGSPACE);
3664*43a90889SApple OSS Distributions 					}
3665*43a90889SApple OSS Distributions 				}
3666*43a90889SApple OSS Distributions 				if (m == NULL) {
3667*43a90889SApple OSS Distributions 					os_log_error(OS_LOG_DEFAULT, "%s: m_get*() failed",
3668*43a90889SApple OSS Distributions 					    __func__);
3669*43a90889SApple OSS Distributions 					return -ENOMEM;
3670*43a90889SApple OSS Distributions 				}
3671*43a90889SApple OSS Distributions 				m->m_pkthdr.vt_nrecs = 0;
3672*43a90889SApple OSS Distributions 				igmp_save_context(m, ifp);
3673*43a90889SApple OSS Distributions 				m0srcs = (ifp->if_mtu - IGMP_LEADINGSPACE -
3674*43a90889SApple OSS Distributions 				    sizeof(struct igmp_grouprec)) /
3675*43a90889SApple OSS Distributions 				    sizeof(in_addr_t);
3676*43a90889SApple OSS Distributions 				npbytes = 0;
3677*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: allocated new packet\n",
3678*43a90889SApple OSS Distributions 				    __func__));
3679*43a90889SApple OSS Distributions 			}
3680*43a90889SApple OSS Distributions 			/*
3681*43a90889SApple OSS Distributions 			 * Append the IGMP group record header to the
3682*43a90889SApple OSS Distributions 			 * current packet's data area.
3683*43a90889SApple OSS Distributions 			 * Recalculate pointer to free space for next
3684*43a90889SApple OSS Distributions 			 * group record, in case m_append() allocated
3685*43a90889SApple OSS Distributions 			 * a new mbuf or cluster.
3686*43a90889SApple OSS Distributions 			 */
3687*43a90889SApple OSS Distributions 			memset(&ig, 0, sizeof(ig));
3688*43a90889SApple OSS Distributions 			ig.ig_group = inm->inm_addr;
3689*43a90889SApple OSS Distributions 			if (!m_append(m, sizeof(ig), (void *)&ig)) {
3690*43a90889SApple OSS Distributions 				if (m != m0) {
3691*43a90889SApple OSS Distributions 					m_freem(m);
3692*43a90889SApple OSS Distributions 				}
3693*43a90889SApple OSS Distributions 				os_log_error(OS_LOG_DEFAULT,
3694*43a90889SApple OSS Distributions 				    "%s: m_append() failed\n",
3695*43a90889SApple OSS Distributions 				    __func__);
3696*43a90889SApple OSS Distributions 				return -ENOMEM;
3697*43a90889SApple OSS Distributions 			}
3698*43a90889SApple OSS Distributions 			npbytes += sizeof(struct igmp_grouprec);
3699*43a90889SApple OSS Distributions 			if (m != m0) {
3700*43a90889SApple OSS Distributions 				/* new packet; offset in c hain */
3701*43a90889SApple OSS Distributions 				md = m_getptr(m, npbytes -
3702*43a90889SApple OSS Distributions 				    sizeof(struct igmp_grouprec), &off);
3703*43a90889SApple OSS Distributions 				pig = (struct igmp_grouprec *)(void *)(mtod(md,
3704*43a90889SApple OSS Distributions 				    uint8_t *) + off);
3705*43a90889SApple OSS Distributions 			} else {
3706*43a90889SApple OSS Distributions 				/* current packet; offset from last append */
3707*43a90889SApple OSS Distributions 				md = m_last(m);
3708*43a90889SApple OSS Distributions 				pig = (struct igmp_grouprec *)(void *)(mtod(md,
3709*43a90889SApple OSS Distributions 				    uint8_t *) + md->m_len -
3710*43a90889SApple OSS Distributions 				    sizeof(struct igmp_grouprec));
3711*43a90889SApple OSS Distributions 			}
3712*43a90889SApple OSS Distributions 			/*
3713*43a90889SApple OSS Distributions 			 * Begin walking the tree for this record type
3714*43a90889SApple OSS Distributions 			 * pass, or continue from where we left off
3715*43a90889SApple OSS Distributions 			 * previously if we had to allocate a new packet.
3716*43a90889SApple OSS Distributions 			 * Only report deltas in-mode at t1.
3717*43a90889SApple OSS Distributions 			 * We need not report included sources as allowed
3718*43a90889SApple OSS Distributions 			 * if we are in inclusive mode on the group,
3719*43a90889SApple OSS Distributions 			 * however the converse is not true.
3720*43a90889SApple OSS Distributions 			 */
3721*43a90889SApple OSS Distributions 			rsrcs = 0;
3722*43a90889SApple OSS Distributions 			if (nims == NULL) {
3723*43a90889SApple OSS Distributions 				nims = RB_MIN(ip_msource_tree, &inm->inm_srcs);
3724*43a90889SApple OSS Distributions 			}
3725*43a90889SApple OSS Distributions 			RB_FOREACH_FROM(ims, ip_msource_tree, nims) {
3726*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
3727*43a90889SApple OSS Distributions 				char buf[MAX_IPv4_STR_LEN];
3728*43a90889SApple OSS Distributions 
3729*43a90889SApple OSS Distributions 				inet_ntop_haddr(ims->ims_haddr, buf, sizeof(buf));
3730*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: visit node %s\n", __func__, buf));
3731*43a90889SApple OSS Distributions #endif
3732*43a90889SApple OSS Distributions 				now = ims_get_mode(inm, ims, 1);
3733*43a90889SApple OSS Distributions 				then = ims_get_mode(inm, ims, 0);
3734*43a90889SApple OSS Distributions 				IGMP_PRINTF(("%s: mode: t0 %d, t1 %d\n",
3735*43a90889SApple OSS Distributions 				    __func__, then, now));
3736*43a90889SApple OSS Distributions 				if (now == then) {
3737*43a90889SApple OSS Distributions 					IGMP_PRINTF(("%s: skip unchanged\n",
3738*43a90889SApple OSS Distributions 					    __func__));
3739*43a90889SApple OSS Distributions 					continue;
3740*43a90889SApple OSS Distributions 				}
3741*43a90889SApple OSS Distributions 				if (mode == MCAST_EXCLUDE &&
3742*43a90889SApple OSS Distributions 				    now == MCAST_INCLUDE) {
3743*43a90889SApple OSS Distributions 					IGMP_PRINTF(("%s: skip IN src on EX "
3744*43a90889SApple OSS Distributions 					    "group\n", __func__));
3745*43a90889SApple OSS Distributions 					continue;
3746*43a90889SApple OSS Distributions 				}
3747*43a90889SApple OSS Distributions 				nrt = (rectype_t)now;
3748*43a90889SApple OSS Distributions 				if (nrt == REC_NONE) {
3749*43a90889SApple OSS Distributions 					nrt = (rectype_t)(~mode & REC_FULL);
3750*43a90889SApple OSS Distributions 				}
3751*43a90889SApple OSS Distributions 				if (schanged++ == 0) {
3752*43a90889SApple OSS Distributions 					crt = nrt;
3753*43a90889SApple OSS Distributions 				} else if (crt != nrt) {
3754*43a90889SApple OSS Distributions 					continue;
3755*43a90889SApple OSS Distributions 				}
3756*43a90889SApple OSS Distributions 				naddr = htonl(ims->ims_haddr);
3757*43a90889SApple OSS Distributions 				if (!m_append(m, sizeof(in_addr_t),
3758*43a90889SApple OSS Distributions 				    (void *)&naddr)) {
3759*43a90889SApple OSS Distributions 					if (m != m0) {
3760*43a90889SApple OSS Distributions 						m_freem(m);
3761*43a90889SApple OSS Distributions 					}
3762*43a90889SApple OSS Distributions 					os_log_error(OS_LOG_DEFAULT, "%s: m_append() failed\n",
3763*43a90889SApple OSS Distributions 					    __func__);
3764*43a90889SApple OSS Distributions 					return -ENOMEM;
3765*43a90889SApple OSS Distributions 				}
3766*43a90889SApple OSS Distributions 				nallow += !!(crt == REC_ALLOW);
3767*43a90889SApple OSS Distributions 				nblock += !!(crt == REC_BLOCK);
3768*43a90889SApple OSS Distributions 				if (++rsrcs == m0srcs) {
3769*43a90889SApple OSS Distributions 					break;
3770*43a90889SApple OSS Distributions 				}
3771*43a90889SApple OSS Distributions 			}
3772*43a90889SApple OSS Distributions 			/*
3773*43a90889SApple OSS Distributions 			 * If we did not append any tree nodes on this
3774*43a90889SApple OSS Distributions 			 * pass, back out of allocations.
3775*43a90889SApple OSS Distributions 			 */
3776*43a90889SApple OSS Distributions 			if (rsrcs == 0) {
3777*43a90889SApple OSS Distributions 				npbytes -= sizeof(struct igmp_grouprec);
3778*43a90889SApple OSS Distributions 				if (m != m0) {
3779*43a90889SApple OSS Distributions 					IGMP_PRINTF(("%s: m_free(m)\n",
3780*43a90889SApple OSS Distributions 					    __func__));
3781*43a90889SApple OSS Distributions 					m_freem(m);
3782*43a90889SApple OSS Distributions 				} else {
3783*43a90889SApple OSS Distributions 					IGMP_PRINTF(("%s: m_adj(m, -ig)\n",
3784*43a90889SApple OSS Distributions 					    __func__));
3785*43a90889SApple OSS Distributions 					m_adj(m, -((int)sizeof(
3786*43a90889SApple OSS Distributions 						    struct igmp_grouprec)));
3787*43a90889SApple OSS Distributions 				}
3788*43a90889SApple OSS Distributions 				continue;
3789*43a90889SApple OSS Distributions 			}
3790*43a90889SApple OSS Distributions 			npbytes += (rsrcs * sizeof(in_addr_t));
3791*43a90889SApple OSS Distributions 			if (crt == REC_ALLOW) {
3792*43a90889SApple OSS Distributions 				pig->ig_type = IGMP_ALLOW_NEW_SOURCES;
3793*43a90889SApple OSS Distributions 			} else if (crt == REC_BLOCK) {
3794*43a90889SApple OSS Distributions 				pig->ig_type = IGMP_BLOCK_OLD_SOURCES;
3795*43a90889SApple OSS Distributions 			}
3796*43a90889SApple OSS Distributions 			ig_numsrc = htons(rsrcs);
3797*43a90889SApple OSS Distributions 			bcopy(&ig_numsrc, &pig->ig_numsrc, sizeof(ig_numsrc));
3798*43a90889SApple OSS Distributions 			/*
3799*43a90889SApple OSS Distributions 			 * Count the new group record, and enqueue this
3800*43a90889SApple OSS Distributions 			 * packet if it wasn't already queued.
3801*43a90889SApple OSS Distributions 			 */
3802*43a90889SApple OSS Distributions 			m->m_pkthdr.vt_nrecs++;
3803*43a90889SApple OSS Distributions 			if (m != m0) {
3804*43a90889SApple OSS Distributions 				IF_ENQUEUE(ifq, m);
3805*43a90889SApple OSS Distributions 			}
3806*43a90889SApple OSS Distributions 			nbytes += npbytes;
3807*43a90889SApple OSS Distributions 		} while (nims != NULL);
3808*43a90889SApple OSS Distributions 		drt |= crt;
3809*43a90889SApple OSS Distributions 		crt = (~crt & REC_FULL);
3810*43a90889SApple OSS Distributions 	}
3811*43a90889SApple OSS Distributions 
3812*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: queued %d ALLOW_NEW, %d BLOCK_OLD\n", __func__,
3813*43a90889SApple OSS Distributions 	    nallow, nblock));
3814*43a90889SApple OSS Distributions 
3815*43a90889SApple OSS Distributions 	return nbytes;
3816*43a90889SApple OSS Distributions }
3817*43a90889SApple OSS Distributions 
3818*43a90889SApple OSS Distributions static int
igmp_v3_merge_state_changes(struct in_multi * inm,struct ifqueue * ifscq)3819*43a90889SApple OSS Distributions igmp_v3_merge_state_changes(struct in_multi *inm, struct ifqueue *ifscq)
3820*43a90889SApple OSS Distributions {
3821*43a90889SApple OSS Distributions 	struct ifqueue  *gq;
3822*43a90889SApple OSS Distributions 	mbuf_ref_t       m;             /* pending state-change */
3823*43a90889SApple OSS Distributions 	mbuf_ref_t       m0;            /* copy of pending state-change */
3824*43a90889SApple OSS Distributions 	mbuf_ref_t       mt;            /* last state-change in packet */
3825*43a90889SApple OSS Distributions 	mbuf_ref_t       n;;
3826*43a90889SApple OSS Distributions 	int              docopy, domerge;
3827*43a90889SApple OSS Distributions 	u_int            recslen;
3828*43a90889SApple OSS Distributions 
3829*43a90889SApple OSS Distributions 	INM_LOCK_ASSERT_HELD(inm);
3830*43a90889SApple OSS Distributions 
3831*43a90889SApple OSS Distributions 	docopy = 0;
3832*43a90889SApple OSS Distributions 	domerge = 0;
3833*43a90889SApple OSS Distributions 	recslen = 0;
3834*43a90889SApple OSS Distributions 
3835*43a90889SApple OSS Distributions 	/*
3836*43a90889SApple OSS Distributions 	 * If there are further pending retransmissions, make a writable
3837*43a90889SApple OSS Distributions 	 * copy of each queued state-change message before merging.
3838*43a90889SApple OSS Distributions 	 */
3839*43a90889SApple OSS Distributions 	if (inm->inm_scrv > 0) {
3840*43a90889SApple OSS Distributions 		docopy = 1;
3841*43a90889SApple OSS Distributions 	}
3842*43a90889SApple OSS Distributions 
3843*43a90889SApple OSS Distributions 	gq = &inm->inm_scq;
3844*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
3845*43a90889SApple OSS Distributions 	if (gq->ifq_head == NULL) {
3846*43a90889SApple OSS Distributions 		IGMP_PRINTF(("%s: WARNING: queue for inm 0x%llx is empty\n",
3847*43a90889SApple OSS Distributions 		    __func__, (uint64_t)VM_KERNEL_ADDRPERM(inm)));
3848*43a90889SApple OSS Distributions 	}
3849*43a90889SApple OSS Distributions #endif
3850*43a90889SApple OSS Distributions 
3851*43a90889SApple OSS Distributions 	/*
3852*43a90889SApple OSS Distributions 	 * Use IF_REMQUEUE() instead of IF_DEQUEUE() below, since the
3853*43a90889SApple OSS Distributions 	 * packet might not always be at the head of the ifqueue.
3854*43a90889SApple OSS Distributions 	 */
3855*43a90889SApple OSS Distributions 	m = gq->ifq_head;
3856*43a90889SApple OSS Distributions 	while (m != NULL) {
3857*43a90889SApple OSS Distributions 		/*
3858*43a90889SApple OSS Distributions 		 * Only merge the report into the current packet if
3859*43a90889SApple OSS Distributions 		 * there is sufficient space to do so; an IGMPv3 report
3860*43a90889SApple OSS Distributions 		 * packet may only contain 65,535 group records.
3861*43a90889SApple OSS Distributions 		 * Always use a simple mbuf chain concatentation to do this,
3862*43a90889SApple OSS Distributions 		 * as large state changes for single groups may have
3863*43a90889SApple OSS Distributions 		 * allocated clusters.
3864*43a90889SApple OSS Distributions 		 */
3865*43a90889SApple OSS Distributions 		domerge = 0;
3866*43a90889SApple OSS Distributions 		mt = ifscq->ifq_tail;
3867*43a90889SApple OSS Distributions 		if (mt != NULL) {
3868*43a90889SApple OSS Distributions 			recslen = m_length(m);
3869*43a90889SApple OSS Distributions 
3870*43a90889SApple OSS Distributions 			if ((mt->m_pkthdr.vt_nrecs +
3871*43a90889SApple OSS Distributions 			    m->m_pkthdr.vt_nrecs <=
3872*43a90889SApple OSS Distributions 			    IGMP_V3_REPORT_MAXRECS) &&
3873*43a90889SApple OSS Distributions 			    (mt->m_pkthdr.len + recslen <=
3874*43a90889SApple OSS Distributions 			    (inm->inm_ifp->if_mtu - IGMP_LEADINGSPACE))) {
3875*43a90889SApple OSS Distributions 				domerge = 1;
3876*43a90889SApple OSS Distributions 			}
3877*43a90889SApple OSS Distributions 		}
3878*43a90889SApple OSS Distributions 
3879*43a90889SApple OSS Distributions 		if (!domerge && IF_QFULL(gq)) {
3880*43a90889SApple OSS Distributions 			os_log_error(OS_LOG_DEFAULT,
3881*43a90889SApple OSS Distributions 			    "%s: outbound queue full on %s\n",
3882*43a90889SApple OSS Distributions 			    __func__, if_name(inm->inm_ifp));
3883*43a90889SApple OSS Distributions 			n = m->m_nextpkt;
3884*43a90889SApple OSS Distributions 			if (!docopy) {
3885*43a90889SApple OSS Distributions 				IF_REMQUEUE(gq, m);
3886*43a90889SApple OSS Distributions 				m_freem(m);
3887*43a90889SApple OSS Distributions 			}
3888*43a90889SApple OSS Distributions 			m = n;
3889*43a90889SApple OSS Distributions 			continue;
3890*43a90889SApple OSS Distributions 		}
3891*43a90889SApple OSS Distributions 
3892*43a90889SApple OSS Distributions 		if (!docopy) {
3893*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: dequeueing 0x%llx\n", __func__,
3894*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(m)));
3895*43a90889SApple OSS Distributions 			n = m->m_nextpkt;
3896*43a90889SApple OSS Distributions 			IF_REMQUEUE(gq, m);
3897*43a90889SApple OSS Distributions 			m0 = m;
3898*43a90889SApple OSS Distributions 			m = n;
3899*43a90889SApple OSS Distributions 		} else {
3900*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: copying 0x%llx\n", __func__,
3901*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(m)));
3902*43a90889SApple OSS Distributions 			m0 = m_dup(m, M_NOWAIT);
3903*43a90889SApple OSS Distributions 			if (m0 == NULL) {
3904*43a90889SApple OSS Distributions 				return ENOMEM;
3905*43a90889SApple OSS Distributions 			}
3906*43a90889SApple OSS Distributions 			m0->m_nextpkt = NULL;
3907*43a90889SApple OSS Distributions 			m = m->m_nextpkt;
3908*43a90889SApple OSS Distributions 		}
3909*43a90889SApple OSS Distributions 
3910*43a90889SApple OSS Distributions 		if (!domerge) {
3911*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: queueing 0x%llx to ifscq 0x%llx)\n",
3912*43a90889SApple OSS Distributions 			    __func__, (uint64_t)VM_KERNEL_ADDRPERM(m0),
3913*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(ifscq)));
3914*43a90889SApple OSS Distributions 			IF_ENQUEUE(ifscq, m0);
3915*43a90889SApple OSS Distributions 		} else {
3916*43a90889SApple OSS Distributions 			struct mbuf *mtl;       /* last mbuf of packet mt */
3917*43a90889SApple OSS Distributions 
3918*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: merging 0x%llx with ifscq tail "
3919*43a90889SApple OSS Distributions 			    "0x%llx)\n", __func__,
3920*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(m0),
3921*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(mt)));
3922*43a90889SApple OSS Distributions 
3923*43a90889SApple OSS Distributions 			mtl = m_last(mt);
3924*43a90889SApple OSS Distributions 			m0->m_flags &= ~M_PKTHDR;
3925*43a90889SApple OSS Distributions 			mt->m_pkthdr.len += recslen;
3926*43a90889SApple OSS Distributions 			mt->m_pkthdr.vt_nrecs +=
3927*43a90889SApple OSS Distributions 			    m0->m_pkthdr.vt_nrecs;
3928*43a90889SApple OSS Distributions 
3929*43a90889SApple OSS Distributions 			mtl->m_next = m0;
3930*43a90889SApple OSS Distributions 		}
3931*43a90889SApple OSS Distributions 	}
3932*43a90889SApple OSS Distributions 
3933*43a90889SApple OSS Distributions 	return 0;
3934*43a90889SApple OSS Distributions }
3935*43a90889SApple OSS Distributions 
3936*43a90889SApple OSS Distributions /*
3937*43a90889SApple OSS Distributions  * Respond to a pending IGMPv3 General Query.
3938*43a90889SApple OSS Distributions  */
3939*43a90889SApple OSS Distributions static uint32_t
igmp_v3_dispatch_general_query(struct igmp_ifinfo * igi)3940*43a90889SApple OSS Distributions igmp_v3_dispatch_general_query(struct igmp_ifinfo *igi)
3941*43a90889SApple OSS Distributions {
3942*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
3943*43a90889SApple OSS Distributions 	struct in_multi         *inm;
3944*43a90889SApple OSS Distributions 	struct in_multistep     step;
3945*43a90889SApple OSS Distributions 	int                      retval, loop;
3946*43a90889SApple OSS Distributions 
3947*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
3948*43a90889SApple OSS Distributions 
3949*43a90889SApple OSS Distributions 	VERIFY(igi->igi_version == IGMP_VERSION_3);
3950*43a90889SApple OSS Distributions 
3951*43a90889SApple OSS Distributions 	ifp = igi->igi_ifp;
3952*43a90889SApple OSS Distributions 	IGI_UNLOCK(igi);
3953*43a90889SApple OSS Distributions 
3954*43a90889SApple OSS Distributions 	in_multihead_lock_shared();
3955*43a90889SApple OSS Distributions 	IN_FIRST_MULTI(step, inm);
3956*43a90889SApple OSS Distributions 	while (inm != NULL) {
3957*43a90889SApple OSS Distributions 		INM_LOCK(inm);
3958*43a90889SApple OSS Distributions 		if (inm->inm_ifp != ifp) {
3959*43a90889SApple OSS Distributions 			goto next;
3960*43a90889SApple OSS Distributions 		}
3961*43a90889SApple OSS Distributions 
3962*43a90889SApple OSS Distributions 		switch (inm->inm_state) {
3963*43a90889SApple OSS Distributions 		case IGMP_NOT_MEMBER:
3964*43a90889SApple OSS Distributions 		case IGMP_SILENT_MEMBER:
3965*43a90889SApple OSS Distributions 			break;
3966*43a90889SApple OSS Distributions 		case IGMP_REPORTING_MEMBER:
3967*43a90889SApple OSS Distributions 		case IGMP_IDLE_MEMBER:
3968*43a90889SApple OSS Distributions 		case IGMP_LAZY_MEMBER:
3969*43a90889SApple OSS Distributions 		case IGMP_SLEEPING_MEMBER:
3970*43a90889SApple OSS Distributions 		case IGMP_AWAKENING_MEMBER:
3971*43a90889SApple OSS Distributions 			inm->inm_state = IGMP_REPORTING_MEMBER;
3972*43a90889SApple OSS Distributions 			IGI_LOCK(igi);
3973*43a90889SApple OSS Distributions 			retval = igmp_v3_enqueue_group_record(&igi->igi_gq,
3974*43a90889SApple OSS Distributions 			    inm, 0, 0, 0);
3975*43a90889SApple OSS Distributions 			IGI_UNLOCK(igi);
3976*43a90889SApple OSS Distributions 			IGMP_PRINTF(("%s: enqueue record = %d\n",
3977*43a90889SApple OSS Distributions 			    __func__, retval));
3978*43a90889SApple OSS Distributions 			break;
3979*43a90889SApple OSS Distributions 		case IGMP_G_QUERY_PENDING_MEMBER:
3980*43a90889SApple OSS Distributions 		case IGMP_SG_QUERY_PENDING_MEMBER:
3981*43a90889SApple OSS Distributions 		case IGMP_LEAVING_MEMBER:
3982*43a90889SApple OSS Distributions 			break;
3983*43a90889SApple OSS Distributions 		}
3984*43a90889SApple OSS Distributions next:
3985*43a90889SApple OSS Distributions 		INM_UNLOCK(inm);
3986*43a90889SApple OSS Distributions 		IN_NEXT_MULTI(step, inm);
3987*43a90889SApple OSS Distributions 	}
3988*43a90889SApple OSS Distributions 	in_multihead_lock_done();
3989*43a90889SApple OSS Distributions 
3990*43a90889SApple OSS Distributions 	IGI_LOCK(igi);
3991*43a90889SApple OSS Distributions 	loop = (igi->igi_flags & IGIF_LOOPBACK) ? 1 : 0;
3992*43a90889SApple OSS Distributions 	igmp_dispatch_queue(igi, &igi->igi_gq, IGMP_MAX_RESPONSE_BURST,
3993*43a90889SApple OSS Distributions 	    loop);
3994*43a90889SApple OSS Distributions 	IGI_LOCK_ASSERT_HELD(igi);
3995*43a90889SApple OSS Distributions 	/*
3996*43a90889SApple OSS Distributions 	 * Slew transmission of bursts over 1 second intervals.
3997*43a90889SApple OSS Distributions 	 */
3998*43a90889SApple OSS Distributions 	if (igi->igi_gq.ifq_head != NULL) {
3999*43a90889SApple OSS Distributions 		igi->igi_v3_timer = 1 + IGMP_RANDOM_DELAY(
4000*43a90889SApple OSS Distributions 			IGMP_RESPONSE_BURST_INTERVAL);
4001*43a90889SApple OSS Distributions 	}
4002*43a90889SApple OSS Distributions 
4003*43a90889SApple OSS Distributions 	return igi->igi_v3_timer;
4004*43a90889SApple OSS Distributions }
4005*43a90889SApple OSS Distributions 
4006*43a90889SApple OSS Distributions /*
4007*43a90889SApple OSS Distributions  * Transmit the next pending IGMP message in the output queue.
4008*43a90889SApple OSS Distributions  *
4009*43a90889SApple OSS Distributions  * Must not be called with inm_lock or igi_lock held.
4010*43a90889SApple OSS Distributions  */
4011*43a90889SApple OSS Distributions static void
igmp_sendpkt(struct mbuf * m)4012*43a90889SApple OSS Distributions igmp_sendpkt(struct mbuf *m)
4013*43a90889SApple OSS Distributions {
4014*43a90889SApple OSS Distributions 	struct ip_moptions      *imo;
4015*43a90889SApple OSS Distributions 	struct mbuf             *ipopts, *m0;
4016*43a90889SApple OSS Distributions 	int                     error;
4017*43a90889SApple OSS Distributions 	struct route            ro;
4018*43a90889SApple OSS Distributions 	struct ifnet            *ifp;
4019*43a90889SApple OSS Distributions 
4020*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: transmit 0x%llx\n", __func__,
4021*43a90889SApple OSS Distributions 	    (uint64_t)VM_KERNEL_ADDRPERM(m)));
4022*43a90889SApple OSS Distributions 
4023*43a90889SApple OSS Distributions 	ifp = igmp_restore_context(m);
4024*43a90889SApple OSS Distributions 	/*
4025*43a90889SApple OSS Distributions 	 * Check if the ifnet is still attached.
4026*43a90889SApple OSS Distributions 	 */
4027*43a90889SApple OSS Distributions 	if (ifp == NULL || !ifnet_is_attached(ifp, 0)) {
4028*43a90889SApple OSS Distributions 		os_log_error(OS_LOG_DEFAULT, "%s: dropped 0x%llx as interface went away\n",
4029*43a90889SApple OSS Distributions 		    __func__, (uint64_t)VM_KERNEL_ADDRPERM(m));
4030*43a90889SApple OSS Distributions 		m_freem(m);
4031*43a90889SApple OSS Distributions 		OSAddAtomic(1, &ipstat.ips_noroute);
4032*43a90889SApple OSS Distributions 		return;
4033*43a90889SApple OSS Distributions 	}
4034*43a90889SApple OSS Distributions 
4035*43a90889SApple OSS Distributions 	ipopts = igmp_sendra ? m_raopt : NULL;
4036*43a90889SApple OSS Distributions 
4037*43a90889SApple OSS Distributions 	imo = ip_allocmoptions(Z_WAITOK);
4038*43a90889SApple OSS Distributions 	if (imo == NULL) {
4039*43a90889SApple OSS Distributions 		m_freem(m);
4040*43a90889SApple OSS Distributions 		return;
4041*43a90889SApple OSS Distributions 	}
4042*43a90889SApple OSS Distributions 
4043*43a90889SApple OSS Distributions 	imo->imo_multicast_ttl  = 1;
4044*43a90889SApple OSS Distributions 	imo->imo_multicast_vif  = -1;
4045*43a90889SApple OSS Distributions 	imo->imo_multicast_loop = 0;
4046*43a90889SApple OSS Distributions 
4047*43a90889SApple OSS Distributions 	/*
4048*43a90889SApple OSS Distributions 	 * If the user requested that IGMP traffic be explicitly
4049*43a90889SApple OSS Distributions 	 * redirected to the loopback interface (e.g. they are running a
4050*43a90889SApple OSS Distributions 	 * MANET interface and the routing protocol needs to see the
4051*43a90889SApple OSS Distributions 	 * updates), handle this now.
4052*43a90889SApple OSS Distributions 	 */
4053*43a90889SApple OSS Distributions 	if (m->m_flags & M_IGMP_LOOP) {
4054*43a90889SApple OSS Distributions 		imo->imo_multicast_ifp = lo_ifp;
4055*43a90889SApple OSS Distributions 	} else {
4056*43a90889SApple OSS Distributions 		imo->imo_multicast_ifp = ifp;
4057*43a90889SApple OSS Distributions 	}
4058*43a90889SApple OSS Distributions 
4059*43a90889SApple OSS Distributions 	if (m->m_flags & M_IGMPV2) {
4060*43a90889SApple OSS Distributions 		m0 = m;
4061*43a90889SApple OSS Distributions 	} else {
4062*43a90889SApple OSS Distributions 		m0 = igmp_v3_encap_report(ifp, m);
4063*43a90889SApple OSS Distributions 		if (m0 == NULL) {
4064*43a90889SApple OSS Distributions 			/*
4065*43a90889SApple OSS Distributions 			 * If igmp_v3_encap_report() failed, then M_PREPEND()
4066*43a90889SApple OSS Distributions 			 * already freed the original mbuf chain.
4067*43a90889SApple OSS Distributions 			 * This means that we don't have to m_freem(m) here.
4068*43a90889SApple OSS Distributions 			 */
4069*43a90889SApple OSS Distributions 			os_log_error(OS_LOG_DEFAULT, "%s: dropped 0x%llx\n", __func__,
4070*43a90889SApple OSS Distributions 			    (uint64_t)VM_KERNEL_ADDRPERM(m));
4071*43a90889SApple OSS Distributions 			IMO_REMREF(imo);
4072*43a90889SApple OSS Distributions 			os_atomic_inc(&ipstat.ips_odropped, relaxed);
4073*43a90889SApple OSS Distributions 			return;
4074*43a90889SApple OSS Distributions 		}
4075*43a90889SApple OSS Distributions 	}
4076*43a90889SApple OSS Distributions 
4077*43a90889SApple OSS Distributions 	igmp_scrub_context(m0);
4078*43a90889SApple OSS Distributions 	m->m_flags &= ~(M_PROTOFLAGS | M_IGMP_LOOP);
4079*43a90889SApple OSS Distributions 	m0->m_pkthdr.rcvif = lo_ifp;
4080*43a90889SApple OSS Distributions 
4081*43a90889SApple OSS Distributions 	if (ifp->if_eflags & IFEF_TXSTART) {
4082*43a90889SApple OSS Distributions 		/*
4083*43a90889SApple OSS Distributions 		 * Use control service class if the interface supports
4084*43a90889SApple OSS Distributions 		 * transmit-start model.
4085*43a90889SApple OSS Distributions 		 */
4086*43a90889SApple OSS Distributions 		(void) m_set_service_class(m0, MBUF_SC_CTL);
4087*43a90889SApple OSS Distributions 	}
4088*43a90889SApple OSS Distributions 	bzero(&ro, sizeof(ro));
4089*43a90889SApple OSS Distributions 	error = ip_output(m0, ipopts, &ro, 0, imo, NULL);
4090*43a90889SApple OSS Distributions 	ROUTE_RELEASE(&ro);
4091*43a90889SApple OSS Distributions 
4092*43a90889SApple OSS Distributions 	IMO_REMREF(imo);
4093*43a90889SApple OSS Distributions 
4094*43a90889SApple OSS Distributions 	if (error) {
4095*43a90889SApple OSS Distributions 		os_log_error(OS_LOG_DEFAULT, "%s: ip_output(0x%llx) = %d\n", __func__,
4096*43a90889SApple OSS Distributions 		    (uint64_t)VM_KERNEL_ADDRPERM(m0), error);
4097*43a90889SApple OSS Distributions 		return;
4098*43a90889SApple OSS Distributions 	}
4099*43a90889SApple OSS Distributions 
4100*43a90889SApple OSS Distributions 	IGMPSTAT_INC(igps_snd_reports);
4101*43a90889SApple OSS Distributions 	OIGMPSTAT_INC(igps_snd_reports);
4102*43a90889SApple OSS Distributions }
4103*43a90889SApple OSS Distributions /*
4104*43a90889SApple OSS Distributions  * Encapsulate an IGMPv3 report.
4105*43a90889SApple OSS Distributions  *
4106*43a90889SApple OSS Distributions  * The internal mbuf flag M_IGMPV3_HDR is used to indicate that the mbuf
4107*43a90889SApple OSS Distributions  * chain has already had its IP/IGMPv3 header prepended. In this case
4108*43a90889SApple OSS Distributions  * the function will not attempt to prepend; the lengths and checksums
4109*43a90889SApple OSS Distributions  * will however be re-computed.
4110*43a90889SApple OSS Distributions  *
4111*43a90889SApple OSS Distributions  * Returns a pointer to the new mbuf chain head, or NULL if the
4112*43a90889SApple OSS Distributions  * allocation failed.
4113*43a90889SApple OSS Distributions  */
4114*43a90889SApple OSS Distributions static struct mbuf *
igmp_v3_encap_report(struct ifnet * ifp,struct mbuf * m)4115*43a90889SApple OSS Distributions igmp_v3_encap_report(struct ifnet *ifp, struct mbuf *m)
4116*43a90889SApple OSS Distributions {
4117*43a90889SApple OSS Distributions 	struct igmp_report      *igmp;
4118*43a90889SApple OSS Distributions 	struct ip               *ip;
4119*43a90889SApple OSS Distributions 	unsigned int             hdrlen, igmpreclen;
4120*43a90889SApple OSS Distributions 
4121*43a90889SApple OSS Distributions 	VERIFY((m->m_flags & M_PKTHDR));
4122*43a90889SApple OSS Distributions 
4123*43a90889SApple OSS Distributions 	igmpreclen = m_length(m);
4124*43a90889SApple OSS Distributions 	hdrlen = sizeof(struct ip) + sizeof(struct igmp_report);
4125*43a90889SApple OSS Distributions 
4126*43a90889SApple OSS Distributions 	if (m->m_flags & M_IGMPV3_HDR) {
4127*43a90889SApple OSS Distributions 		igmpreclen -= hdrlen;
4128*43a90889SApple OSS Distributions 	} else {
4129*43a90889SApple OSS Distributions 		M_PREPEND(m, hdrlen, M_DONTWAIT, 1);
4130*43a90889SApple OSS Distributions 		if (m == NULL) {
4131*43a90889SApple OSS Distributions 			return NULL;
4132*43a90889SApple OSS Distributions 		}
4133*43a90889SApple OSS Distributions 		m->m_flags |= M_IGMPV3_HDR;
4134*43a90889SApple OSS Distributions 	}
4135*43a90889SApple OSS Distributions 	if (hdrlen + igmpreclen > USHRT_MAX) {
4136*43a90889SApple OSS Distributions 		os_log_error(OS_LOG_DEFAULT, "%s: invalid length %d\n",
4137*43a90889SApple OSS Distributions 		    __func__, hdrlen + igmpreclen);
4138*43a90889SApple OSS Distributions 		m_freem(m);
4139*43a90889SApple OSS Distributions 		return NULL;
4140*43a90889SApple OSS Distributions 	}
4141*43a90889SApple OSS Distributions 
4142*43a90889SApple OSS Distributions 
4143*43a90889SApple OSS Distributions 	IGMP_PRINTF(("%s: igmpreclen is %d\n", __func__, igmpreclen));
4144*43a90889SApple OSS Distributions 
4145*43a90889SApple OSS Distributions 	m->m_data += sizeof(struct ip);
4146*43a90889SApple OSS Distributions 	m->m_len -= sizeof(struct ip);
4147*43a90889SApple OSS Distributions 
4148*43a90889SApple OSS Distributions 	igmp = mtod(m, struct igmp_report *);
4149*43a90889SApple OSS Distributions 	igmp->ir_type = IGMP_v3_HOST_MEMBERSHIP_REPORT;
4150*43a90889SApple OSS Distributions 	igmp->ir_rsv1 = 0;
4151*43a90889SApple OSS Distributions 	igmp->ir_rsv2 = 0;
4152*43a90889SApple OSS Distributions 	igmp->ir_numgrps = htons(m->m_pkthdr.vt_nrecs);
4153*43a90889SApple OSS Distributions 	igmp->ir_cksum = 0;
4154*43a90889SApple OSS Distributions 	igmp->ir_cksum = in_cksum(m, sizeof(struct igmp_report) + igmpreclen);
4155*43a90889SApple OSS Distributions 	m->m_pkthdr.vt_nrecs = 0;
4156*43a90889SApple OSS Distributions 
4157*43a90889SApple OSS Distributions 	m->m_data -= sizeof(struct ip);
4158*43a90889SApple OSS Distributions 	m->m_len += sizeof(struct ip);
4159*43a90889SApple OSS Distributions 
4160*43a90889SApple OSS Distributions 	ip = mtod(m, struct ip *);
4161*43a90889SApple OSS Distributions 	ip->ip_tos = IPTOS_PREC_INTERNETCONTROL;
4162*43a90889SApple OSS Distributions 	ip->ip_len = (u_short)(hdrlen + igmpreclen);
4163*43a90889SApple OSS Distributions 	ip->ip_off = IP_DF;
4164*43a90889SApple OSS Distributions 	ip->ip_p = IPPROTO_IGMP;
4165*43a90889SApple OSS Distributions 	ip->ip_sum = 0;
4166*43a90889SApple OSS Distributions 
4167*43a90889SApple OSS Distributions 	ip->ip_src.s_addr = INADDR_ANY;
4168*43a90889SApple OSS Distributions 
4169*43a90889SApple OSS Distributions 	if (m->m_flags & M_IGMP_LOOP) {
4170*43a90889SApple OSS Distributions 		struct in_ifaddr *ia;
4171*43a90889SApple OSS Distributions 
4172*43a90889SApple OSS Distributions 		IFP_TO_IA(ifp, ia);
4173*43a90889SApple OSS Distributions 		if (ia != NULL) {
4174*43a90889SApple OSS Distributions 			IFA_LOCK(&ia->ia_ifa);
4175*43a90889SApple OSS Distributions 			ip->ip_src = ia->ia_addr.sin_addr;
4176*43a90889SApple OSS Distributions 			IFA_UNLOCK(&ia->ia_ifa);
4177*43a90889SApple OSS Distributions 			ifa_remref(&ia->ia_ifa);
4178*43a90889SApple OSS Distributions 		}
4179*43a90889SApple OSS Distributions 	}
4180*43a90889SApple OSS Distributions 
4181*43a90889SApple OSS Distributions 	ip->ip_dst.s_addr = htonl(INADDR_ALLRPTS_GROUP);
4182*43a90889SApple OSS Distributions 
4183*43a90889SApple OSS Distributions 	return m;
4184*43a90889SApple OSS Distributions }
4185*43a90889SApple OSS Distributions 
4186*43a90889SApple OSS Distributions #ifdef IGMP_DEBUG
4187*43a90889SApple OSS Distributions static const char *
igmp_rec_type_to_str(const int type)4188*43a90889SApple OSS Distributions igmp_rec_type_to_str(const int type)
4189*43a90889SApple OSS Distributions {
4190*43a90889SApple OSS Distributions 	switch (type) {
4191*43a90889SApple OSS Distributions 	case IGMP_CHANGE_TO_EXCLUDE_MODE:
4192*43a90889SApple OSS Distributions 		return "TO_EX";
4193*43a90889SApple OSS Distributions 	case IGMP_CHANGE_TO_INCLUDE_MODE:
4194*43a90889SApple OSS Distributions 		return "TO_IN";
4195*43a90889SApple OSS Distributions 	case IGMP_MODE_IS_EXCLUDE:
4196*43a90889SApple OSS Distributions 		return "MODE_EX";
4197*43a90889SApple OSS Distributions 	case IGMP_MODE_IS_INCLUDE:
4198*43a90889SApple OSS Distributions 		return "MODE_IN";
4199*43a90889SApple OSS Distributions 	case IGMP_ALLOW_NEW_SOURCES:
4200*43a90889SApple OSS Distributions 		return "ALLOW_NEW";
4201*43a90889SApple OSS Distributions 	case IGMP_BLOCK_OLD_SOURCES:
4202*43a90889SApple OSS Distributions 		return "BLOCK_OLD";
4203*43a90889SApple OSS Distributions 	default:
4204*43a90889SApple OSS Distributions 		break;
4205*43a90889SApple OSS Distributions 	}
4206*43a90889SApple OSS Distributions 	return "unknown";
4207*43a90889SApple OSS Distributions }
4208*43a90889SApple OSS Distributions #endif
4209*43a90889SApple OSS Distributions 
4210*43a90889SApple OSS Distributions void
igmp_init(struct protosw * pp,struct domain * dp)4211*43a90889SApple OSS Distributions igmp_init(struct protosw *pp, struct domain *dp)
4212*43a90889SApple OSS Distributions {
4213*43a90889SApple OSS Distributions #pragma unused(dp)
4214*43a90889SApple OSS Distributions 	static int igmp_initialized = 0;
4215*43a90889SApple OSS Distributions 
4216*43a90889SApple OSS Distributions 	VERIFY((pp->pr_flags & (PR_INITIALIZED | PR_ATTACHED)) == PR_ATTACHED);
4217*43a90889SApple OSS Distributions 
4218*43a90889SApple OSS Distributions 	if (igmp_initialized) {
4219*43a90889SApple OSS Distributions 		return;
4220*43a90889SApple OSS Distributions 	}
4221*43a90889SApple OSS Distributions 	igmp_initialized = 1;
4222*43a90889SApple OSS Distributions 	os_log(OS_LOG_DEFAULT, "%s: initializing\n", __func__);
4223*43a90889SApple OSS Distributions 	igmp_timers_are_running = 0;
4224*43a90889SApple OSS Distributions 	LIST_INIT(&igi_head);
4225*43a90889SApple OSS Distributions 	m_raopt = igmp_ra_alloc();
4226*43a90889SApple OSS Distributions }
4227