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