xref: /xnu-12377.81.4/bsd/net/multicast_list.c (revision 043036a2b3718f7f0be807e2870f8f47d3fa0796)
1 /*
2  * Copyright (c) 2004-2021 Apple Inc. All rights reserved.
3  *
4  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5  *
6  * This file contains Original Code and/or Modifications of Original Code
7  * as defined in and that are subject to the Apple Public Source License
8  * Version 2.0 (the 'License'). You may not use this file except in
9  * compliance with the License. The rights granted to you under the License
10  * may not be used to create, or enable the creation or redistribution of,
11  * unlawful or unlicensed copies of an Apple operating system, or to
12  * circumvent, violate, or enable the circumvention or violation of, any
13  * terms of an Apple operating system software license agreement.
14  *
15  * Please obtain a copy of the License at
16  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17  *
18  * The Original Code and all software distributed under the License are
19  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23  * Please see the License for the specific language governing rights and
24  * limitations under the License.
25  *
26  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27  */
28 
29 /*
30  * multicast_util.c:
31  * - keep track of multicast addresses added to one interface based on the
32  *   actual multicast addresses in another
33  * - used by VLAN and BOND
34  */
35 
36 /*
37  * Modification History:
38  *
39  * April 29, 2004	Dieter Siegmund ([email protected])
40  * - created
41  */
42 
43 #include <net/multicast_list.h>
44 #include <sys/param.h>
45 #include <sys/systm.h>
46 #include <sys/malloc.h>
47 #include <net/if_dl.h>
48 
49 #include <net/sockaddr_utils.h>
50 
51 __private_extern__ void
multicast_list_init(struct multicast_list * mc_list)52 multicast_list_init(struct multicast_list * mc_list)
53 {
54 	SLIST_INIT(mc_list);
55 	return;
56 }
57 
58 /*
59  * Function: multicast_list_remove
60  * Purpose:
61  *   Remove the given list of multicast addresses from the interface and from
62  *   the multicast list structure.
63  */
64 __private_extern__ int
multicast_list_remove(struct multicast_list * mc_list)65 multicast_list_remove(struct multicast_list * mc_list)
66 {
67 	int                         error;
68 	struct multicast_entry *    mc;
69 	int                         result = 0;
70 
71 	while ((mc = SLIST_FIRST(mc_list)) != NULL) {
72 		error = ifnet_remove_multicast(mc->mc_ifma);
73 		if (error != 0) {
74 			result = error;
75 		}
76 		SLIST_REMOVE_HEAD(mc_list, mc_entries);
77 		ifmaddr_release(mc->mc_ifma);
78 		kfree_type(struct multicast_entry, mc);
79 	}
80 	return result;
81 }
82 
83 /*
84  * Function: multicast_list_program
85  * Purpose:
86  *   Program the multicast filter on "target_ifp" using the values from
87  *   "source_ifp", and saving the result in "mc_list"
88  *
89  *   We build a new list of multicast addresses while programming the new list.
90  *   If that completes successfully, we remove the old list, and return the
91  *   new list.
92  *
93  *   If it fails, we remove what we've added to the new list, and
94  *   return an error.
95  */
96 __private_extern__ int
multicast_list_program(struct multicast_list * mc_list,struct ifnet * source_ifp,struct ifnet * target_ifp)97 multicast_list_program(struct multicast_list * mc_list,
98     struct ifnet * source_ifp,
99     struct ifnet * target_ifp)
100 {
101 	u_char                      alen;
102 	int                         error = 0;
103 	struct multicast_entry *    mc = NULL;
104 	struct multicast_list       new_mc_list;
105 	struct sockaddr_dl          source_sdl = {};
106 	ifmultiaddr_t *__null_terminated source_multicast_list;
107 	struct sockaddr_dl          target_sdl;
108 
109 	alen = target_ifp->if_addrlen;
110 	bzero((char *)&target_sdl, sizeof(target_sdl));
111 	target_sdl.sdl_len = sizeof(target_sdl);
112 	target_sdl.sdl_family = AF_LINK;
113 	target_sdl.sdl_type = target_ifp->if_type;
114 	target_sdl.sdl_alen = alen;
115 	target_sdl.sdl_index = target_ifp->if_index;
116 
117 	/* build a new list */
118 	multicast_list_init(&new_mc_list);
119 	error = ifnet_get_multicast_list(source_ifp, &source_multicast_list);
120 	if (error != 0) {
121 		printf("multicast_list_program: "
122 		    "ifnet_get_multicast_list(%s%d) failed, %d\n",
123 		    source_ifp->if_name, source_ifp->if_unit, error);
124 		return error;
125 	}
126 	for (ifmultiaddr_t *__null_terminated ptr = source_multicast_list;
127 	    *ptr != NULL; ptr++) {
128 		if (ifmaddr_address(*ptr, SA(&source_sdl), sizeof(source_sdl)) != 0
129 		    || source_sdl.sdl_family != AF_LINK) {
130 			continue;
131 		}
132 		mc = kalloc_type(struct multicast_entry, Z_WAITOK | Z_NOFAIL);
133 		bcopy(LLADDR(&source_sdl), LLADDR(&target_sdl), alen);
134 		error = ifnet_add_multicast(target_ifp, SA(&target_sdl),
135 		    &mc->mc_ifma);
136 		if (error != 0) {
137 			kfree_type(struct multicast_entry, mc);
138 			break;
139 		}
140 		SLIST_INSERT_HEAD(&new_mc_list, mc, mc_entries);
141 	}
142 	if (error != 0) {
143 		/* restore previous state */
144 		(void)multicast_list_remove(&new_mc_list);
145 	} else {
146 		/* remove the old entries, and return the new list */
147 		(void)multicast_list_remove(mc_list);
148 		*mc_list = new_mc_list;
149 	}
150 	ifnet_free_multicast_list(source_multicast_list);
151 	return error;
152 }
153