xref: /openbmc/linux/drivers/net/amt.c (revision 1188f7f111c61394ec56beb8e30322305a8220b6)
1b9022b53STaehee Yoo // SPDX-License-Identifier: GPL-2.0-or-later
2b9022b53STaehee Yoo /* Copyright (c) 2021 Taehee Yoo <ap420073@gmail.com> */
3b9022b53STaehee Yoo 
4b9022b53STaehee Yoo #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
5b9022b53STaehee Yoo 
6b9022b53STaehee Yoo #include <linux/module.h>
7b9022b53STaehee Yoo #include <linux/skbuff.h>
8b9022b53STaehee Yoo #include <linux/udp.h>
9b9022b53STaehee Yoo #include <linux/jhash.h>
10b9022b53STaehee Yoo #include <linux/if_tunnel.h>
11b9022b53STaehee Yoo #include <linux/net.h>
12b9022b53STaehee Yoo #include <linux/igmp.h>
13b9022b53STaehee Yoo #include <linux/workqueue.h>
14*05f7e346STaehee Yoo #include <net/pkt_sched.h>
15b9022b53STaehee Yoo #include <net/net_namespace.h>
16b9022b53STaehee Yoo #include <net/ip.h>
17b9022b53STaehee Yoo #include <net/udp.h>
18b9022b53STaehee Yoo #include <net/udp_tunnel.h>
19b9022b53STaehee Yoo #include <net/icmp.h>
20b9022b53STaehee Yoo #include <net/mld.h>
21b9022b53STaehee Yoo #include <net/amt.h>
22b9022b53STaehee Yoo #include <uapi/linux/amt.h>
23b9022b53STaehee Yoo #include <linux/security.h>
24b9022b53STaehee Yoo #include <net/gro_cells.h>
25b9022b53STaehee Yoo #include <net/ipv6.h>
26b9022b53STaehee Yoo #include <net/if_inet6.h>
27b9022b53STaehee Yoo #include <net/ndisc.h>
28b9022b53STaehee Yoo #include <net/addrconf.h>
29b9022b53STaehee Yoo #include <net/ip6_route.h>
30b9022b53STaehee Yoo #include <net/inet_common.h>
31b75f7095STaehee Yoo #include <net/ip6_checksum.h>
32b9022b53STaehee Yoo 
33b9022b53STaehee Yoo static struct workqueue_struct *amt_wq;
34b9022b53STaehee Yoo 
35bc54e49cSTaehee Yoo static HLIST_HEAD(source_gc_list);
36bc54e49cSTaehee Yoo /* Lock for source_gc_list */
37bc54e49cSTaehee Yoo static spinlock_t source_gc_lock;
38bc54e49cSTaehee Yoo static struct delayed_work source_gc_wq;
39cbc21dc1STaehee Yoo static char *status_str[] = {
40cbc21dc1STaehee Yoo 	"AMT_STATUS_INIT",
41cbc21dc1STaehee Yoo 	"AMT_STATUS_SENT_DISCOVERY",
42cbc21dc1STaehee Yoo 	"AMT_STATUS_RECEIVED_DISCOVERY",
43cbc21dc1STaehee Yoo 	"AMT_STATUS_SENT_ADVERTISEMENT",
44cbc21dc1STaehee Yoo 	"AMT_STATUS_RECEIVED_ADVERTISEMENT",
45cbc21dc1STaehee Yoo 	"AMT_STATUS_SENT_REQUEST",
46cbc21dc1STaehee Yoo 	"AMT_STATUS_RECEIVED_REQUEST",
47cbc21dc1STaehee Yoo 	"AMT_STATUS_SENT_QUERY",
48cbc21dc1STaehee Yoo 	"AMT_STATUS_RECEIVED_QUERY",
49cbc21dc1STaehee Yoo 	"AMT_STATUS_SENT_UPDATE",
50cbc21dc1STaehee Yoo 	"AMT_STATUS_RECEIVED_UPDATE",
51cbc21dc1STaehee Yoo };
52cbc21dc1STaehee Yoo 
53cbc21dc1STaehee Yoo static char *type_str[] = {
54d7970039STaehee Yoo 	"", /* Type 0 is not defined */
55cbc21dc1STaehee Yoo 	"AMT_MSG_DISCOVERY",
56cbc21dc1STaehee Yoo 	"AMT_MSG_ADVERTISEMENT",
57cbc21dc1STaehee Yoo 	"AMT_MSG_REQUEST",
58cbc21dc1STaehee Yoo 	"AMT_MSG_MEMBERSHIP_QUERY",
59cbc21dc1STaehee Yoo 	"AMT_MSG_MEMBERSHIP_UPDATE",
60cbc21dc1STaehee Yoo 	"AMT_MSG_MULTICAST_DATA",
614934609dSTaehee Yoo 	"AMT_MSG_TEARDOWN",
62cbc21dc1STaehee Yoo };
63cbc21dc1STaehee Yoo 
64bc54e49cSTaehee Yoo static char *action_str[] = {
65bc54e49cSTaehee Yoo 	"AMT_ACT_GMI",
66bc54e49cSTaehee Yoo 	"AMT_ACT_GMI_ZERO",
67bc54e49cSTaehee Yoo 	"AMT_ACT_GT",
68bc54e49cSTaehee Yoo 	"AMT_ACT_STATUS_FWD_NEW",
69bc54e49cSTaehee Yoo 	"AMT_ACT_STATUS_D_FWD_NEW",
70bc54e49cSTaehee Yoo 	"AMT_ACT_STATUS_NONE_NEW",
71bc54e49cSTaehee Yoo };
72bc54e49cSTaehee Yoo 
73b75f7095STaehee Yoo static struct igmpv3_grec igmpv3_zero_grec;
74b75f7095STaehee Yoo 
75b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
76b75f7095STaehee Yoo #define MLD2_ALL_NODE_INIT { { { 0xff, 0x02, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x01 } } }
77b75f7095STaehee Yoo static struct in6_addr mld2_all_node = MLD2_ALL_NODE_INIT;
78b75f7095STaehee Yoo static struct mld2_grec mldv2_zero_grec;
79b75f7095STaehee Yoo #endif
80b75f7095STaehee Yoo 
amt_skb_cb(struct sk_buff * skb)81cbc21dc1STaehee Yoo static struct amt_skb_cb *amt_skb_cb(struct sk_buff *skb)
82cbc21dc1STaehee Yoo {
83*05f7e346STaehee Yoo 	BUILD_BUG_ON(sizeof(struct amt_skb_cb) + sizeof(struct tc_skb_cb) >
84cbc21dc1STaehee Yoo 		     sizeof_field(struct sk_buff, cb));
85cbc21dc1STaehee Yoo 
86cbc21dc1STaehee Yoo 	return (struct amt_skb_cb *)((void *)skb->cb +
87*05f7e346STaehee Yoo 		sizeof(struct tc_skb_cb));
88cbc21dc1STaehee Yoo }
89cbc21dc1STaehee Yoo 
__amt_source_gc_work(void)90bc54e49cSTaehee Yoo static void __amt_source_gc_work(void)
91bc54e49cSTaehee Yoo {
92bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
93bc54e49cSTaehee Yoo 	struct hlist_head gc_list;
94bc54e49cSTaehee Yoo 	struct hlist_node *t;
95bc54e49cSTaehee Yoo 
96bc54e49cSTaehee Yoo 	spin_lock_bh(&source_gc_lock);
97bc54e49cSTaehee Yoo 	hlist_move_list(&source_gc_list, &gc_list);
98bc54e49cSTaehee Yoo 	spin_unlock_bh(&source_gc_lock);
99bc54e49cSTaehee Yoo 
100bc54e49cSTaehee Yoo 	hlist_for_each_entry_safe(snode, t, &gc_list, node) {
101bc54e49cSTaehee Yoo 		hlist_del_rcu(&snode->node);
102bc54e49cSTaehee Yoo 		kfree_rcu(snode, rcu);
103bc54e49cSTaehee Yoo 	}
104bc54e49cSTaehee Yoo }
105bc54e49cSTaehee Yoo 
amt_source_gc_work(struct work_struct * work)106bc54e49cSTaehee Yoo static void amt_source_gc_work(struct work_struct *work)
107bc54e49cSTaehee Yoo {
108bc54e49cSTaehee Yoo 	__amt_source_gc_work();
109bc54e49cSTaehee Yoo 
110bc54e49cSTaehee Yoo 	spin_lock_bh(&source_gc_lock);
111bc54e49cSTaehee Yoo 	mod_delayed_work(amt_wq, &source_gc_wq,
112bc54e49cSTaehee Yoo 			 msecs_to_jiffies(AMT_GC_INTERVAL));
113bc54e49cSTaehee Yoo 	spin_unlock_bh(&source_gc_lock);
114bc54e49cSTaehee Yoo }
115bc54e49cSTaehee Yoo 
amt_addr_equal(union amt_addr * a,union amt_addr * b)116bc54e49cSTaehee Yoo static bool amt_addr_equal(union amt_addr *a, union amt_addr *b)
117bc54e49cSTaehee Yoo {
118bc54e49cSTaehee Yoo 	return !memcmp(a, b, sizeof(union amt_addr));
119bc54e49cSTaehee Yoo }
120bc54e49cSTaehee Yoo 
amt_source_hash(struct amt_tunnel_list * tunnel,union amt_addr * src)121bc54e49cSTaehee Yoo static u32 amt_source_hash(struct amt_tunnel_list *tunnel, union amt_addr *src)
122bc54e49cSTaehee Yoo {
123bc54e49cSTaehee Yoo 	u32 hash = jhash(src, sizeof(*src), tunnel->amt->hash_seed);
124bc54e49cSTaehee Yoo 
125bc54e49cSTaehee Yoo 	return reciprocal_scale(hash, tunnel->amt->hash_buckets);
126bc54e49cSTaehee Yoo }
127bc54e49cSTaehee Yoo 
amt_status_filter(struct amt_source_node * snode,enum amt_filter filter)128bc54e49cSTaehee Yoo static bool amt_status_filter(struct amt_source_node *snode,
129bc54e49cSTaehee Yoo 			      enum amt_filter filter)
130bc54e49cSTaehee Yoo {
131bc54e49cSTaehee Yoo 	bool rc = false;
132bc54e49cSTaehee Yoo 
133bc54e49cSTaehee Yoo 	switch (filter) {
134bc54e49cSTaehee Yoo 	case AMT_FILTER_FWD:
135bc54e49cSTaehee Yoo 		if (snode->status == AMT_SOURCE_STATUS_FWD &&
136bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_OLD)
137bc54e49cSTaehee Yoo 			rc = true;
138bc54e49cSTaehee Yoo 		break;
139bc54e49cSTaehee Yoo 	case AMT_FILTER_D_FWD:
140bc54e49cSTaehee Yoo 		if (snode->status == AMT_SOURCE_STATUS_D_FWD &&
141bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_OLD)
142bc54e49cSTaehee Yoo 			rc = true;
143bc54e49cSTaehee Yoo 		break;
144bc54e49cSTaehee Yoo 	case AMT_FILTER_FWD_NEW:
145bc54e49cSTaehee Yoo 		if (snode->status == AMT_SOURCE_STATUS_FWD &&
146bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_NEW)
147bc54e49cSTaehee Yoo 			rc = true;
148bc54e49cSTaehee Yoo 		break;
149bc54e49cSTaehee Yoo 	case AMT_FILTER_D_FWD_NEW:
150bc54e49cSTaehee Yoo 		if (snode->status == AMT_SOURCE_STATUS_D_FWD &&
151bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_NEW)
152bc54e49cSTaehee Yoo 			rc = true;
153bc54e49cSTaehee Yoo 		break;
154bc54e49cSTaehee Yoo 	case AMT_FILTER_ALL:
155bc54e49cSTaehee Yoo 		rc = true;
156bc54e49cSTaehee Yoo 		break;
157bc54e49cSTaehee Yoo 	case AMT_FILTER_NONE_NEW:
158bc54e49cSTaehee Yoo 		if (snode->status == AMT_SOURCE_STATUS_NONE &&
159bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_NEW)
160bc54e49cSTaehee Yoo 			rc = true;
161bc54e49cSTaehee Yoo 		break;
162bc54e49cSTaehee Yoo 	case AMT_FILTER_BOTH:
163bc54e49cSTaehee Yoo 		if ((snode->status == AMT_SOURCE_STATUS_D_FWD ||
164bc54e49cSTaehee Yoo 		     snode->status == AMT_SOURCE_STATUS_FWD) &&
165bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_OLD)
166bc54e49cSTaehee Yoo 			rc = true;
167bc54e49cSTaehee Yoo 		break;
168bc54e49cSTaehee Yoo 	case AMT_FILTER_BOTH_NEW:
169bc54e49cSTaehee Yoo 		if ((snode->status == AMT_SOURCE_STATUS_D_FWD ||
170bc54e49cSTaehee Yoo 		     snode->status == AMT_SOURCE_STATUS_FWD) &&
171bc54e49cSTaehee Yoo 		    snode->flags == AMT_SOURCE_NEW)
172bc54e49cSTaehee Yoo 			rc = true;
173bc54e49cSTaehee Yoo 		break;
174bc54e49cSTaehee Yoo 	default:
175bc54e49cSTaehee Yoo 		WARN_ON_ONCE(1);
176bc54e49cSTaehee Yoo 		break;
177bc54e49cSTaehee Yoo 	}
178bc54e49cSTaehee Yoo 
179bc54e49cSTaehee Yoo 	return rc;
180bc54e49cSTaehee Yoo }
181bc54e49cSTaehee Yoo 
amt_lookup_src(struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,enum amt_filter filter,union amt_addr * src)182bc54e49cSTaehee Yoo static struct amt_source_node *amt_lookup_src(struct amt_tunnel_list *tunnel,
183bc54e49cSTaehee Yoo 					      struct amt_group_node *gnode,
184bc54e49cSTaehee Yoo 					      enum amt_filter filter,
185bc54e49cSTaehee Yoo 					      union amt_addr *src)
186bc54e49cSTaehee Yoo {
187bc54e49cSTaehee Yoo 	u32 hash = amt_source_hash(tunnel, src);
188bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
189bc54e49cSTaehee Yoo 
190bc54e49cSTaehee Yoo 	hlist_for_each_entry_rcu(snode, &gnode->sources[hash], node)
191bc54e49cSTaehee Yoo 		if (amt_status_filter(snode, filter) &&
192bc54e49cSTaehee Yoo 		    amt_addr_equal(&snode->source_addr, src))
193bc54e49cSTaehee Yoo 			return snode;
194bc54e49cSTaehee Yoo 
195bc54e49cSTaehee Yoo 	return NULL;
196bc54e49cSTaehee Yoo }
197bc54e49cSTaehee Yoo 
amt_group_hash(struct amt_tunnel_list * tunnel,union amt_addr * group)198bc54e49cSTaehee Yoo static u32 amt_group_hash(struct amt_tunnel_list *tunnel, union amt_addr *group)
199bc54e49cSTaehee Yoo {
200bc54e49cSTaehee Yoo 	u32 hash = jhash(group, sizeof(*group), tunnel->amt->hash_seed);
201bc54e49cSTaehee Yoo 
202bc54e49cSTaehee Yoo 	return reciprocal_scale(hash, tunnel->amt->hash_buckets);
203bc54e49cSTaehee Yoo }
204bc54e49cSTaehee Yoo 
amt_lookup_group(struct amt_tunnel_list * tunnel,union amt_addr * group,union amt_addr * host,bool v6)205bc54e49cSTaehee Yoo static struct amt_group_node *amt_lookup_group(struct amt_tunnel_list *tunnel,
206bc54e49cSTaehee Yoo 					       union amt_addr *group,
207bc54e49cSTaehee Yoo 					       union amt_addr *host,
208bc54e49cSTaehee Yoo 					       bool v6)
209bc54e49cSTaehee Yoo {
210bc54e49cSTaehee Yoo 	u32 hash = amt_group_hash(tunnel, group);
211bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
212bc54e49cSTaehee Yoo 
213bc54e49cSTaehee Yoo 	hlist_for_each_entry_rcu(gnode, &tunnel->groups[hash], node) {
214bc54e49cSTaehee Yoo 		if (amt_addr_equal(&gnode->group_addr, group) &&
215bc54e49cSTaehee Yoo 		    amt_addr_equal(&gnode->host_addr, host) &&
216bc54e49cSTaehee Yoo 		    gnode->v6 == v6)
217bc54e49cSTaehee Yoo 			return gnode;
218bc54e49cSTaehee Yoo 	}
219bc54e49cSTaehee Yoo 
220bc54e49cSTaehee Yoo 	return NULL;
221bc54e49cSTaehee Yoo }
222bc54e49cSTaehee Yoo 
amt_destroy_source(struct amt_source_node * snode)223bc54e49cSTaehee Yoo static void amt_destroy_source(struct amt_source_node *snode)
224bc54e49cSTaehee Yoo {
225bc54e49cSTaehee Yoo 	struct amt_group_node *gnode = snode->gnode;
226bc54e49cSTaehee Yoo 	struct amt_tunnel_list *tunnel;
227bc54e49cSTaehee Yoo 
228bc54e49cSTaehee Yoo 	tunnel = gnode->tunnel_list;
229bc54e49cSTaehee Yoo 
230bc54e49cSTaehee Yoo 	if (!gnode->v6) {
231bc54e49cSTaehee Yoo 		netdev_dbg(snode->gnode->amt->dev,
232bc54e49cSTaehee Yoo 			   "Delete source %pI4 from %pI4\n",
233bc54e49cSTaehee Yoo 			   &snode->source_addr.ip4,
234bc54e49cSTaehee Yoo 			   &gnode->group_addr.ip4);
235b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
236b75f7095STaehee Yoo 	} else {
237b75f7095STaehee Yoo 		netdev_dbg(snode->gnode->amt->dev,
238b75f7095STaehee Yoo 			   "Delete source %pI6 from %pI6\n",
239b75f7095STaehee Yoo 			   &snode->source_addr.ip6,
240b75f7095STaehee Yoo 			   &gnode->group_addr.ip6);
241b75f7095STaehee Yoo #endif
242bc54e49cSTaehee Yoo 	}
243bc54e49cSTaehee Yoo 
244bc54e49cSTaehee Yoo 	cancel_delayed_work(&snode->source_timer);
245bc54e49cSTaehee Yoo 	hlist_del_init_rcu(&snode->node);
246bc54e49cSTaehee Yoo 	tunnel->nr_sources--;
247bc54e49cSTaehee Yoo 	gnode->nr_sources--;
248bc54e49cSTaehee Yoo 	spin_lock_bh(&source_gc_lock);
249bc54e49cSTaehee Yoo 	hlist_add_head_rcu(&snode->node, &source_gc_list);
250bc54e49cSTaehee Yoo 	spin_unlock_bh(&source_gc_lock);
251bc54e49cSTaehee Yoo }
252bc54e49cSTaehee Yoo 
amt_del_group(struct amt_dev * amt,struct amt_group_node * gnode)253bc54e49cSTaehee Yoo static void amt_del_group(struct amt_dev *amt, struct amt_group_node *gnode)
254bc54e49cSTaehee Yoo {
255bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
256bc54e49cSTaehee Yoo 	struct hlist_node *t;
257bc54e49cSTaehee Yoo 	int i;
258bc54e49cSTaehee Yoo 
259bc54e49cSTaehee Yoo 	if (cancel_delayed_work(&gnode->group_timer))
260bc54e49cSTaehee Yoo 		dev_put(amt->dev);
261bc54e49cSTaehee Yoo 	hlist_del_rcu(&gnode->node);
262bc54e49cSTaehee Yoo 	gnode->tunnel_list->nr_groups--;
263bc54e49cSTaehee Yoo 
264bc54e49cSTaehee Yoo 	if (!gnode->v6)
265bc54e49cSTaehee Yoo 		netdev_dbg(amt->dev, "Leave group %pI4\n",
266bc54e49cSTaehee Yoo 			   &gnode->group_addr.ip4);
267b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
268b75f7095STaehee Yoo 	else
269b75f7095STaehee Yoo 		netdev_dbg(amt->dev, "Leave group %pI6\n",
270b75f7095STaehee Yoo 			   &gnode->group_addr.ip6);
271b75f7095STaehee Yoo #endif
272bc54e49cSTaehee Yoo 	for (i = 0; i < amt->hash_buckets; i++)
273bc54e49cSTaehee Yoo 		hlist_for_each_entry_safe(snode, t, &gnode->sources[i], node)
274bc54e49cSTaehee Yoo 			amt_destroy_source(snode);
275bc54e49cSTaehee Yoo 
276bc54e49cSTaehee Yoo 	/* tunnel->lock was acquired outside of amt_del_group()
277bc54e49cSTaehee Yoo 	 * But rcu_read_lock() was acquired too so It's safe.
278bc54e49cSTaehee Yoo 	 */
279bc54e49cSTaehee Yoo 	kfree_rcu(gnode, rcu);
280bc54e49cSTaehee Yoo }
281bc54e49cSTaehee Yoo 
282bc54e49cSTaehee Yoo /* If a source timer expires with a router filter-mode for the group of
283bc54e49cSTaehee Yoo  * INCLUDE, the router concludes that traffic from this particular
284bc54e49cSTaehee Yoo  * source is no longer desired on the attached network, and deletes the
285bc54e49cSTaehee Yoo  * associated source record.
286bc54e49cSTaehee Yoo  */
amt_source_work(struct work_struct * work)287bc54e49cSTaehee Yoo static void amt_source_work(struct work_struct *work)
288bc54e49cSTaehee Yoo {
289bc54e49cSTaehee Yoo 	struct amt_source_node *snode = container_of(to_delayed_work(work),
290bc54e49cSTaehee Yoo 						     struct amt_source_node,
291bc54e49cSTaehee Yoo 						     source_timer);
292bc54e49cSTaehee Yoo 	struct amt_group_node *gnode = snode->gnode;
293bc54e49cSTaehee Yoo 	struct amt_dev *amt = gnode->amt;
294bc54e49cSTaehee Yoo 	struct amt_tunnel_list *tunnel;
295bc54e49cSTaehee Yoo 
296bc54e49cSTaehee Yoo 	tunnel = gnode->tunnel_list;
297bc54e49cSTaehee Yoo 	spin_lock_bh(&tunnel->lock);
298bc54e49cSTaehee Yoo 	rcu_read_lock();
299bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
300bc54e49cSTaehee Yoo 		amt_destroy_source(snode);
301bc54e49cSTaehee Yoo 		if (!gnode->nr_sources)
302bc54e49cSTaehee Yoo 			amt_del_group(amt, gnode);
303bc54e49cSTaehee Yoo 	} else {
304bc54e49cSTaehee Yoo 		/* When a router filter-mode for a group is EXCLUDE,
305bc54e49cSTaehee Yoo 		 * source records are only deleted when the group timer expires
306bc54e49cSTaehee Yoo 		 */
307bc54e49cSTaehee Yoo 		snode->status = AMT_SOURCE_STATUS_D_FWD;
308bc54e49cSTaehee Yoo 	}
309bc54e49cSTaehee Yoo 	rcu_read_unlock();
310bc54e49cSTaehee Yoo 	spin_unlock_bh(&tunnel->lock);
311bc54e49cSTaehee Yoo }
312bc54e49cSTaehee Yoo 
amt_act_src(struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,struct amt_source_node * snode,enum amt_act act)313bc54e49cSTaehee Yoo static void amt_act_src(struct amt_tunnel_list *tunnel,
314bc54e49cSTaehee Yoo 			struct amt_group_node *gnode,
315bc54e49cSTaehee Yoo 			struct amt_source_node *snode,
316bc54e49cSTaehee Yoo 			enum amt_act act)
317bc54e49cSTaehee Yoo {
318bc54e49cSTaehee Yoo 	struct amt_dev *amt = tunnel->amt;
319bc54e49cSTaehee Yoo 
320bc54e49cSTaehee Yoo 	switch (act) {
321bc54e49cSTaehee Yoo 	case AMT_ACT_GMI:
322bc54e49cSTaehee Yoo 		mod_delayed_work(amt_wq, &snode->source_timer,
323bc54e49cSTaehee Yoo 				 msecs_to_jiffies(amt_gmi(amt)));
324bc54e49cSTaehee Yoo 		break;
325bc54e49cSTaehee Yoo 	case AMT_ACT_GMI_ZERO:
326bc54e49cSTaehee Yoo 		cancel_delayed_work(&snode->source_timer);
327bc54e49cSTaehee Yoo 		break;
328bc54e49cSTaehee Yoo 	case AMT_ACT_GT:
329bc54e49cSTaehee Yoo 		mod_delayed_work(amt_wq, &snode->source_timer,
330bc54e49cSTaehee Yoo 				 gnode->group_timer.timer.expires);
331bc54e49cSTaehee Yoo 		break;
332bc54e49cSTaehee Yoo 	case AMT_ACT_STATUS_FWD_NEW:
333bc54e49cSTaehee Yoo 		snode->status = AMT_SOURCE_STATUS_FWD;
334bc54e49cSTaehee Yoo 		snode->flags = AMT_SOURCE_NEW;
335bc54e49cSTaehee Yoo 		break;
336bc54e49cSTaehee Yoo 	case AMT_ACT_STATUS_D_FWD_NEW:
337bc54e49cSTaehee Yoo 		snode->status = AMT_SOURCE_STATUS_D_FWD;
338bc54e49cSTaehee Yoo 		snode->flags = AMT_SOURCE_NEW;
339bc54e49cSTaehee Yoo 		break;
340bc54e49cSTaehee Yoo 	case AMT_ACT_STATUS_NONE_NEW:
341bc54e49cSTaehee Yoo 		cancel_delayed_work(&snode->source_timer);
342bc54e49cSTaehee Yoo 		snode->status = AMT_SOURCE_STATUS_NONE;
343bc54e49cSTaehee Yoo 		snode->flags = AMT_SOURCE_NEW;
344bc54e49cSTaehee Yoo 		break;
345bc54e49cSTaehee Yoo 	default:
346bc54e49cSTaehee Yoo 		WARN_ON_ONCE(1);
347bc54e49cSTaehee Yoo 		return;
348bc54e49cSTaehee Yoo 	}
349bc54e49cSTaehee Yoo 
350bc54e49cSTaehee Yoo 	if (!gnode->v6)
351bc54e49cSTaehee Yoo 		netdev_dbg(amt->dev, "Source %pI4 from %pI4 Acted %s\n",
352bc54e49cSTaehee Yoo 			   &snode->source_addr.ip4,
353bc54e49cSTaehee Yoo 			   &gnode->group_addr.ip4,
354bc54e49cSTaehee Yoo 			   action_str[act]);
355b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
356b75f7095STaehee Yoo 	else
357b75f7095STaehee Yoo 		netdev_dbg(amt->dev, "Source %pI6 from %pI6 Acted %s\n",
358b75f7095STaehee Yoo 			   &snode->source_addr.ip6,
359b75f7095STaehee Yoo 			   &gnode->group_addr.ip6,
360b75f7095STaehee Yoo 			   action_str[act]);
361b75f7095STaehee Yoo #endif
362bc54e49cSTaehee Yoo }
363bc54e49cSTaehee Yoo 
amt_alloc_snode(struct amt_group_node * gnode,union amt_addr * src)364bc54e49cSTaehee Yoo static struct amt_source_node *amt_alloc_snode(struct amt_group_node *gnode,
365bc54e49cSTaehee Yoo 					       union amt_addr *src)
366bc54e49cSTaehee Yoo {
367bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
368bc54e49cSTaehee Yoo 
369bc54e49cSTaehee Yoo 	snode = kzalloc(sizeof(*snode), GFP_ATOMIC);
370bc54e49cSTaehee Yoo 	if (!snode)
371bc54e49cSTaehee Yoo 		return NULL;
372bc54e49cSTaehee Yoo 
373bc54e49cSTaehee Yoo 	memcpy(&snode->source_addr, src, sizeof(union amt_addr));
374bc54e49cSTaehee Yoo 	snode->gnode = gnode;
375bc54e49cSTaehee Yoo 	snode->status = AMT_SOURCE_STATUS_NONE;
376bc54e49cSTaehee Yoo 	snode->flags = AMT_SOURCE_NEW;
377bc54e49cSTaehee Yoo 	INIT_HLIST_NODE(&snode->node);
378bc54e49cSTaehee Yoo 	INIT_DELAYED_WORK(&snode->source_timer, amt_source_work);
379bc54e49cSTaehee Yoo 
380bc54e49cSTaehee Yoo 	return snode;
381bc54e49cSTaehee Yoo }
382bc54e49cSTaehee Yoo 
383bc54e49cSTaehee Yoo /* RFC 3810 - 7.2.2.  Definition of Filter Timers
384bc54e49cSTaehee Yoo  *
385bc54e49cSTaehee Yoo  *  Router Mode          Filter Timer         Actions/Comments
386bc54e49cSTaehee Yoo  *  -----------       -----------------       ----------------
387bc54e49cSTaehee Yoo  *
388bc54e49cSTaehee Yoo  *    INCLUDE             Not Used            All listeners in
389bc54e49cSTaehee Yoo  *                                            INCLUDE mode.
390bc54e49cSTaehee Yoo  *
391bc54e49cSTaehee Yoo  *    EXCLUDE             Timer > 0           At least one listener
392bc54e49cSTaehee Yoo  *                                            in EXCLUDE mode.
393bc54e49cSTaehee Yoo  *
394bc54e49cSTaehee Yoo  *    EXCLUDE             Timer == 0          No more listeners in
395bc54e49cSTaehee Yoo  *                                            EXCLUDE mode for the
396bc54e49cSTaehee Yoo  *                                            multicast address.
397bc54e49cSTaehee Yoo  *                                            If the Requested List
398bc54e49cSTaehee Yoo  *                                            is empty, delete
399bc54e49cSTaehee Yoo  *                                            Multicast Address
400bc54e49cSTaehee Yoo  *                                            Record.  If not, switch
401bc54e49cSTaehee Yoo  *                                            to INCLUDE filter mode;
402bc54e49cSTaehee Yoo  *                                            the sources in the
403bc54e49cSTaehee Yoo  *                                            Requested List are
404bc54e49cSTaehee Yoo  *                                            moved to the Include
405bc54e49cSTaehee Yoo  *                                            List, and the Exclude
406bc54e49cSTaehee Yoo  *                                            List is deleted.
407bc54e49cSTaehee Yoo  */
amt_group_work(struct work_struct * work)408bc54e49cSTaehee Yoo static void amt_group_work(struct work_struct *work)
409bc54e49cSTaehee Yoo {
410bc54e49cSTaehee Yoo 	struct amt_group_node *gnode = container_of(to_delayed_work(work),
411bc54e49cSTaehee Yoo 						    struct amt_group_node,
412bc54e49cSTaehee Yoo 						    group_timer);
413bc54e49cSTaehee Yoo 	struct amt_tunnel_list *tunnel = gnode->tunnel_list;
414bc54e49cSTaehee Yoo 	struct amt_dev *amt = gnode->amt;
415bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
416bc54e49cSTaehee Yoo 	bool delete_group = true;
417bc54e49cSTaehee Yoo 	struct hlist_node *t;
418bc54e49cSTaehee Yoo 	int i, buckets;
419bc54e49cSTaehee Yoo 
420bc54e49cSTaehee Yoo 	buckets = amt->hash_buckets;
421bc54e49cSTaehee Yoo 
422bc54e49cSTaehee Yoo 	spin_lock_bh(&tunnel->lock);
423bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
424bc54e49cSTaehee Yoo 		/* Not Used */
425bc54e49cSTaehee Yoo 		spin_unlock_bh(&tunnel->lock);
426bc54e49cSTaehee Yoo 		goto out;
427bc54e49cSTaehee Yoo 	}
428bc54e49cSTaehee Yoo 
429bc54e49cSTaehee Yoo 	rcu_read_lock();
430bc54e49cSTaehee Yoo 	for (i = 0; i < buckets; i++) {
431bc54e49cSTaehee Yoo 		hlist_for_each_entry_safe(snode, t,
432bc54e49cSTaehee Yoo 					  &gnode->sources[i], node) {
433bc54e49cSTaehee Yoo 			if (!delayed_work_pending(&snode->source_timer) ||
434bc54e49cSTaehee Yoo 			    snode->status == AMT_SOURCE_STATUS_D_FWD) {
435bc54e49cSTaehee Yoo 				amt_destroy_source(snode);
436bc54e49cSTaehee Yoo 			} else {
437bc54e49cSTaehee Yoo 				delete_group = false;
438bc54e49cSTaehee Yoo 				snode->status = AMT_SOURCE_STATUS_FWD;
439bc54e49cSTaehee Yoo 			}
440bc54e49cSTaehee Yoo 		}
441bc54e49cSTaehee Yoo 	}
442bc54e49cSTaehee Yoo 	if (delete_group)
443bc54e49cSTaehee Yoo 		amt_del_group(amt, gnode);
444bc54e49cSTaehee Yoo 	else
445bc54e49cSTaehee Yoo 		gnode->filter_mode = MCAST_INCLUDE;
446bc54e49cSTaehee Yoo 	rcu_read_unlock();
447bc54e49cSTaehee Yoo 	spin_unlock_bh(&tunnel->lock);
448bc54e49cSTaehee Yoo out:
449bc54e49cSTaehee Yoo 	dev_put(amt->dev);
450bc54e49cSTaehee Yoo }
451bc54e49cSTaehee Yoo 
45239befe3aSRuffalo Lavoisier /* Non-existent group is created as INCLUDE {empty}:
453bc54e49cSTaehee Yoo  *
454bc54e49cSTaehee Yoo  * RFC 3376 - 5.1. Action on Change of Interface State
455bc54e49cSTaehee Yoo  *
456bc54e49cSTaehee Yoo  * If no interface state existed for that multicast address before
457bc54e49cSTaehee Yoo  * the change (i.e., the change consisted of creating a new
458bc54e49cSTaehee Yoo  * per-interface record), or if no state exists after the change
459bc54e49cSTaehee Yoo  * (i.e., the change consisted of deleting a per-interface record),
460bc54e49cSTaehee Yoo  * then the "non-existent" state is considered to have a filter mode
461bc54e49cSTaehee Yoo  * of INCLUDE and an empty source list.
462bc54e49cSTaehee Yoo  */
amt_add_group(struct amt_dev * amt,struct amt_tunnel_list * tunnel,union amt_addr * group,union amt_addr * host,bool v6)463bc54e49cSTaehee Yoo static struct amt_group_node *amt_add_group(struct amt_dev *amt,
464bc54e49cSTaehee Yoo 					    struct amt_tunnel_list *tunnel,
465bc54e49cSTaehee Yoo 					    union amt_addr *group,
466bc54e49cSTaehee Yoo 					    union amt_addr *host,
467bc54e49cSTaehee Yoo 					    bool v6)
468bc54e49cSTaehee Yoo {
469bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
470bc54e49cSTaehee Yoo 	u32 hash;
471bc54e49cSTaehee Yoo 	int i;
472bc54e49cSTaehee Yoo 
473bc54e49cSTaehee Yoo 	if (tunnel->nr_groups >= amt->max_groups)
474bc54e49cSTaehee Yoo 		return ERR_PTR(-ENOSPC);
475bc54e49cSTaehee Yoo 
476bc54e49cSTaehee Yoo 	gnode = kzalloc(sizeof(*gnode) +
477bc54e49cSTaehee Yoo 			(sizeof(struct hlist_head) * amt->hash_buckets),
478bc54e49cSTaehee Yoo 			GFP_ATOMIC);
479bc54e49cSTaehee Yoo 	if (unlikely(!gnode))
480bc54e49cSTaehee Yoo 		return ERR_PTR(-ENOMEM);
481bc54e49cSTaehee Yoo 
482bc54e49cSTaehee Yoo 	gnode->amt = amt;
483bc54e49cSTaehee Yoo 	gnode->group_addr = *group;
484bc54e49cSTaehee Yoo 	gnode->host_addr = *host;
485bc54e49cSTaehee Yoo 	gnode->v6 = v6;
486bc54e49cSTaehee Yoo 	gnode->tunnel_list = tunnel;
487bc54e49cSTaehee Yoo 	gnode->filter_mode = MCAST_INCLUDE;
488bc54e49cSTaehee Yoo 	INIT_HLIST_NODE(&gnode->node);
489bc54e49cSTaehee Yoo 	INIT_DELAYED_WORK(&gnode->group_timer, amt_group_work);
490bc54e49cSTaehee Yoo 	for (i = 0; i < amt->hash_buckets; i++)
491bc54e49cSTaehee Yoo 		INIT_HLIST_HEAD(&gnode->sources[i]);
492bc54e49cSTaehee Yoo 
493bc54e49cSTaehee Yoo 	hash = amt_group_hash(tunnel, group);
494bc54e49cSTaehee Yoo 	hlist_add_head_rcu(&gnode->node, &tunnel->groups[hash]);
495bc54e49cSTaehee Yoo 	tunnel->nr_groups++;
496bc54e49cSTaehee Yoo 
497bc54e49cSTaehee Yoo 	if (!gnode->v6)
498bc54e49cSTaehee Yoo 		netdev_dbg(amt->dev, "Join group %pI4\n",
499bc54e49cSTaehee Yoo 			   &gnode->group_addr.ip4);
500b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
501b75f7095STaehee Yoo 	else
502b75f7095STaehee Yoo 		netdev_dbg(amt->dev, "Join group %pI6\n",
503b75f7095STaehee Yoo 			   &gnode->group_addr.ip6);
504b75f7095STaehee Yoo #endif
505b75f7095STaehee Yoo 
506bc54e49cSTaehee Yoo 	return gnode;
507bc54e49cSTaehee Yoo }
508bc54e49cSTaehee Yoo 
amt_build_igmp_gq(struct amt_dev * amt)509cbc21dc1STaehee Yoo static struct sk_buff *amt_build_igmp_gq(struct amt_dev *amt)
510cbc21dc1STaehee Yoo {
511cbc21dc1STaehee Yoo 	u8 ra[AMT_IPHDR_OPTS] = { IPOPT_RA, 4, 0, 0 };
512cbc21dc1STaehee Yoo 	int hlen = LL_RESERVED_SPACE(amt->dev);
513cbc21dc1STaehee Yoo 	int tlen = amt->dev->needed_tailroom;
514cbc21dc1STaehee Yoo 	struct igmpv3_query *ihv3;
515cbc21dc1STaehee Yoo 	void *csum_start = NULL;
516cbc21dc1STaehee Yoo 	__sum16 *csum = NULL;
517cbc21dc1STaehee Yoo 	struct sk_buff *skb;
518cbc21dc1STaehee Yoo 	struct ethhdr *eth;
519cbc21dc1STaehee Yoo 	struct iphdr *iph;
520cbc21dc1STaehee Yoo 	unsigned int len;
521cbc21dc1STaehee Yoo 	int offset;
522cbc21dc1STaehee Yoo 
523cbc21dc1STaehee Yoo 	len = hlen + tlen + sizeof(*iph) + AMT_IPHDR_OPTS + sizeof(*ihv3);
524cbc21dc1STaehee Yoo 	skb = netdev_alloc_skb_ip_align(amt->dev, len);
525cbc21dc1STaehee Yoo 	if (!skb)
526cbc21dc1STaehee Yoo 		return NULL;
527cbc21dc1STaehee Yoo 
528cbc21dc1STaehee Yoo 	skb_reserve(skb, hlen);
529cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*eth));
530cbc21dc1STaehee Yoo 	skb->protocol = htons(ETH_P_IP);
531cbc21dc1STaehee Yoo 	skb_reset_mac_header(skb);
532cbc21dc1STaehee Yoo 	skb->priority = TC_PRIO_CONTROL;
533cbc21dc1STaehee Yoo 	skb_put(skb, sizeof(*iph));
534cbc21dc1STaehee Yoo 	skb_put_data(skb, ra, sizeof(ra));
535cbc21dc1STaehee Yoo 	skb_put(skb, sizeof(*ihv3));
536cbc21dc1STaehee Yoo 	skb_pull(skb, sizeof(*eth));
537cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
538cbc21dc1STaehee Yoo 
539cbc21dc1STaehee Yoo 	iph		= ip_hdr(skb);
540cbc21dc1STaehee Yoo 	iph->version	= 4;
541cbc21dc1STaehee Yoo 	iph->ihl	= (sizeof(struct iphdr) + AMT_IPHDR_OPTS) >> 2;
542cbc21dc1STaehee Yoo 	iph->tos	= AMT_TOS;
543cbc21dc1STaehee Yoo 	iph->tot_len	= htons(sizeof(*iph) + AMT_IPHDR_OPTS + sizeof(*ihv3));
544cbc21dc1STaehee Yoo 	iph->frag_off	= htons(IP_DF);
545cbc21dc1STaehee Yoo 	iph->ttl	= 1;
546cbc21dc1STaehee Yoo 	iph->id		= 0;
547cbc21dc1STaehee Yoo 	iph->protocol	= IPPROTO_IGMP;
548cbc21dc1STaehee Yoo 	iph->daddr	= htonl(INADDR_ALLHOSTS_GROUP);
549cbc21dc1STaehee Yoo 	iph->saddr	= htonl(INADDR_ANY);
550cbc21dc1STaehee Yoo 	ip_send_check(iph);
551cbc21dc1STaehee Yoo 
552cbc21dc1STaehee Yoo 	eth = eth_hdr(skb);
553cbc21dc1STaehee Yoo 	ether_addr_copy(eth->h_source, amt->dev->dev_addr);
554cbc21dc1STaehee Yoo 	ip_eth_mc_map(htonl(INADDR_ALLHOSTS_GROUP), eth->h_dest);
555cbc21dc1STaehee Yoo 	eth->h_proto = htons(ETH_P_IP);
556cbc21dc1STaehee Yoo 
557cbc21dc1STaehee Yoo 	ihv3		= skb_pull(skb, sizeof(*iph) + AMT_IPHDR_OPTS);
558cbc21dc1STaehee Yoo 	skb_reset_transport_header(skb);
559cbc21dc1STaehee Yoo 	ihv3->type	= IGMP_HOST_MEMBERSHIP_QUERY;
560cbc21dc1STaehee Yoo 	ihv3->code	= 1;
561cbc21dc1STaehee Yoo 	ihv3->group	= 0;
562cbc21dc1STaehee Yoo 	ihv3->qqic	= amt->qi;
563cbc21dc1STaehee Yoo 	ihv3->nsrcs	= 0;
564cbc21dc1STaehee Yoo 	ihv3->resv	= 0;
565cbc21dc1STaehee Yoo 	ihv3->suppress	= false;
5668ebcc62cSKuniyuki Iwashima 	ihv3->qrv	= READ_ONCE(amt->net->ipv4.sysctl_igmp_qrv);
567cbc21dc1STaehee Yoo 	ihv3->csum	= 0;
568cbc21dc1STaehee Yoo 	csum		= &ihv3->csum;
569cbc21dc1STaehee Yoo 	csum_start	= (void *)ihv3;
570cbc21dc1STaehee Yoo 	*csum		= ip_compute_csum(csum_start, sizeof(*ihv3));
571cbc21dc1STaehee Yoo 	offset		= skb_transport_offset(skb);
572cbc21dc1STaehee Yoo 	skb->csum	= skb_checksum(skb, offset, skb->len - offset, 0);
573cbc21dc1STaehee Yoo 	skb->ip_summed	= CHECKSUM_NONE;
574cbc21dc1STaehee Yoo 
575cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*eth) + sizeof(*iph) + AMT_IPHDR_OPTS);
576cbc21dc1STaehee Yoo 
577cbc21dc1STaehee Yoo 	return skb;
578cbc21dc1STaehee Yoo }
579cbc21dc1STaehee Yoo 
amt_update_gw_status(struct amt_dev * amt,enum amt_status status,bool validate)5809c343ea6STaehee Yoo static void amt_update_gw_status(struct amt_dev *amt, enum amt_status status,
581cbc21dc1STaehee Yoo 				 bool validate)
582cbc21dc1STaehee Yoo {
583cbc21dc1STaehee Yoo 	if (validate && amt->status >= status)
584cbc21dc1STaehee Yoo 		return;
585cbc21dc1STaehee Yoo 	netdev_dbg(amt->dev, "Update GW status %s -> %s",
586cbc21dc1STaehee Yoo 		   status_str[amt->status], status_str[status]);
587928f353cSTaehee Yoo 	WRITE_ONCE(amt->status, status);
588cbc21dc1STaehee Yoo }
589cbc21dc1STaehee Yoo 
__amt_update_relay_status(struct amt_tunnel_list * tunnel,enum amt_status status,bool validate)590cbc21dc1STaehee Yoo static void __amt_update_relay_status(struct amt_tunnel_list *tunnel,
591cbc21dc1STaehee Yoo 				      enum amt_status status,
592cbc21dc1STaehee Yoo 				      bool validate)
593cbc21dc1STaehee Yoo {
594cbc21dc1STaehee Yoo 	if (validate && tunnel->status >= status)
595cbc21dc1STaehee Yoo 		return;
596cbc21dc1STaehee Yoo 	netdev_dbg(tunnel->amt->dev,
597cbc21dc1STaehee Yoo 		   "Update Tunnel(IP = %pI4, PORT = %u) status %s -> %s",
598cbc21dc1STaehee Yoo 		   &tunnel->ip4, ntohs(tunnel->source_port),
599cbc21dc1STaehee Yoo 		   status_str[tunnel->status], status_str[status]);
600cbc21dc1STaehee Yoo 	tunnel->status = status;
601cbc21dc1STaehee Yoo }
602cbc21dc1STaehee Yoo 
amt_update_relay_status(struct amt_tunnel_list * tunnel,enum amt_status status,bool validate)603cbc21dc1STaehee Yoo static void amt_update_relay_status(struct amt_tunnel_list *tunnel,
604cbc21dc1STaehee Yoo 				    enum amt_status status, bool validate)
605cbc21dc1STaehee Yoo {
606cbc21dc1STaehee Yoo 	spin_lock_bh(&tunnel->lock);
607cbc21dc1STaehee Yoo 	__amt_update_relay_status(tunnel, status, validate);
608cbc21dc1STaehee Yoo 	spin_unlock_bh(&tunnel->lock);
609cbc21dc1STaehee Yoo }
610cbc21dc1STaehee Yoo 
amt_send_discovery(struct amt_dev * amt)611cbc21dc1STaehee Yoo static void amt_send_discovery(struct amt_dev *amt)
612cbc21dc1STaehee Yoo {
613cbc21dc1STaehee Yoo 	struct amt_header_discovery *amtd;
614cbc21dc1STaehee Yoo 	int hlen, tlen, offset;
615cbc21dc1STaehee Yoo 	struct socket *sock;
616cbc21dc1STaehee Yoo 	struct udphdr *udph;
617cbc21dc1STaehee Yoo 	struct sk_buff *skb;
618cbc21dc1STaehee Yoo 	struct iphdr *iph;
619cbc21dc1STaehee Yoo 	struct rtable *rt;
620cbc21dc1STaehee Yoo 	struct flowi4 fl4;
621cbc21dc1STaehee Yoo 	u32 len;
622cbc21dc1STaehee Yoo 	int err;
623cbc21dc1STaehee Yoo 
624cbc21dc1STaehee Yoo 	rcu_read_lock();
625cbc21dc1STaehee Yoo 	sock = rcu_dereference(amt->sock);
626cbc21dc1STaehee Yoo 	if (!sock)
627cbc21dc1STaehee Yoo 		goto out;
628cbc21dc1STaehee Yoo 
629cbc21dc1STaehee Yoo 	if (!netif_running(amt->stream_dev) || !netif_running(amt->dev))
630cbc21dc1STaehee Yoo 		goto out;
631cbc21dc1STaehee Yoo 
632cbc21dc1STaehee Yoo 	rt = ip_route_output_ports(amt->net, &fl4, sock->sk,
633cbc21dc1STaehee Yoo 				   amt->discovery_ip, amt->local_ip,
634cbc21dc1STaehee Yoo 				   amt->gw_port, amt->relay_port,
635cbc21dc1STaehee Yoo 				   IPPROTO_UDP, 0,
636cbc21dc1STaehee Yoo 				   amt->stream_dev->ifindex);
637cbc21dc1STaehee Yoo 	if (IS_ERR(rt)) {
638cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
639cbc21dc1STaehee Yoo 		goto out;
640cbc21dc1STaehee Yoo 	}
641cbc21dc1STaehee Yoo 
642cbc21dc1STaehee Yoo 	hlen = LL_RESERVED_SPACE(amt->dev);
643cbc21dc1STaehee Yoo 	tlen = amt->dev->needed_tailroom;
644cbc21dc1STaehee Yoo 	len = hlen + tlen + sizeof(*iph) + sizeof(*udph) + sizeof(*amtd);
645cbc21dc1STaehee Yoo 	skb = netdev_alloc_skb_ip_align(amt->dev, len);
646cbc21dc1STaehee Yoo 	if (!skb) {
647cbc21dc1STaehee Yoo 		ip_rt_put(rt);
648cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
649cbc21dc1STaehee Yoo 		goto out;
650cbc21dc1STaehee Yoo 	}
651cbc21dc1STaehee Yoo 
652cbc21dc1STaehee Yoo 	skb->priority = TC_PRIO_CONTROL;
653cbc21dc1STaehee Yoo 	skb_dst_set(skb, &rt->dst);
654cbc21dc1STaehee Yoo 
655cbc21dc1STaehee Yoo 	len = sizeof(*iph) + sizeof(*udph) + sizeof(*amtd);
656cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
657cbc21dc1STaehee Yoo 	skb_put(skb, len);
658cbc21dc1STaehee Yoo 	amtd = skb_pull(skb, sizeof(*iph) + sizeof(*udph));
659cbc21dc1STaehee Yoo 	amtd->version	= 0;
660cbc21dc1STaehee Yoo 	amtd->type	= AMT_MSG_DISCOVERY;
661cbc21dc1STaehee Yoo 	amtd->reserved	= 0;
662cbc21dc1STaehee Yoo 	amtd->nonce	= amt->nonce;
663cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*udph));
664cbc21dc1STaehee Yoo 	skb_reset_transport_header(skb);
665cbc21dc1STaehee Yoo 	udph		= udp_hdr(skb);
666cbc21dc1STaehee Yoo 	udph->source	= amt->gw_port;
667cbc21dc1STaehee Yoo 	udph->dest	= amt->relay_port;
668cbc21dc1STaehee Yoo 	udph->len	= htons(sizeof(*udph) + sizeof(*amtd));
669cbc21dc1STaehee Yoo 	udph->check	= 0;
670cbc21dc1STaehee Yoo 	offset = skb_transport_offset(skb);
671cbc21dc1STaehee Yoo 	skb->csum = skb_checksum(skb, offset, skb->len - offset, 0);
672cbc21dc1STaehee Yoo 	udph->check = csum_tcpudp_magic(amt->local_ip, amt->discovery_ip,
673cbc21dc1STaehee Yoo 					sizeof(*udph) + sizeof(*amtd),
674cbc21dc1STaehee Yoo 					IPPROTO_UDP, skb->csum);
675cbc21dc1STaehee Yoo 
676cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*iph));
677cbc21dc1STaehee Yoo 	iph		= ip_hdr(skb);
678cbc21dc1STaehee Yoo 	iph->version	= 4;
679cbc21dc1STaehee Yoo 	iph->ihl	= (sizeof(struct iphdr)) >> 2;
680cbc21dc1STaehee Yoo 	iph->tos	= AMT_TOS;
681cbc21dc1STaehee Yoo 	iph->frag_off	= 0;
682cbc21dc1STaehee Yoo 	iph->ttl	= ip4_dst_hoplimit(&rt->dst);
683cbc21dc1STaehee Yoo 	iph->daddr	= amt->discovery_ip;
684cbc21dc1STaehee Yoo 	iph->saddr	= amt->local_ip;
685cbc21dc1STaehee Yoo 	iph->protocol	= IPPROTO_UDP;
686cbc21dc1STaehee Yoo 	iph->tot_len	= htons(len);
687cbc21dc1STaehee Yoo 
688cbc21dc1STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
689cbc21dc1STaehee Yoo 	ip_select_ident(amt->net, skb, NULL);
690cbc21dc1STaehee Yoo 	ip_send_check(iph);
691cbc21dc1STaehee Yoo 	err = ip_local_out(amt->net, sock->sk, skb);
692cbc21dc1STaehee Yoo 	if (unlikely(net_xmit_eval(err)))
693cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
694cbc21dc1STaehee Yoo 
6959c343ea6STaehee Yoo 	amt_update_gw_status(amt, AMT_STATUS_SENT_DISCOVERY, true);
696cbc21dc1STaehee Yoo out:
697cbc21dc1STaehee Yoo 	rcu_read_unlock();
698cbc21dc1STaehee Yoo }
699cbc21dc1STaehee Yoo 
amt_send_request(struct amt_dev * amt,bool v6)700cbc21dc1STaehee Yoo static void amt_send_request(struct amt_dev *amt, bool v6)
701cbc21dc1STaehee Yoo {
702cbc21dc1STaehee Yoo 	struct amt_header_request *amtrh;
703cbc21dc1STaehee Yoo 	int hlen, tlen, offset;
704cbc21dc1STaehee Yoo 	struct socket *sock;
705cbc21dc1STaehee Yoo 	struct udphdr *udph;
706cbc21dc1STaehee Yoo 	struct sk_buff *skb;
707cbc21dc1STaehee Yoo 	struct iphdr *iph;
708cbc21dc1STaehee Yoo 	struct rtable *rt;
709cbc21dc1STaehee Yoo 	struct flowi4 fl4;
710cbc21dc1STaehee Yoo 	u32 len;
711cbc21dc1STaehee Yoo 	int err;
712cbc21dc1STaehee Yoo 
713cbc21dc1STaehee Yoo 	rcu_read_lock();
714cbc21dc1STaehee Yoo 	sock = rcu_dereference(amt->sock);
715cbc21dc1STaehee Yoo 	if (!sock)
716cbc21dc1STaehee Yoo 		goto out;
717cbc21dc1STaehee Yoo 
718cbc21dc1STaehee Yoo 	if (!netif_running(amt->stream_dev) || !netif_running(amt->dev))
719cbc21dc1STaehee Yoo 		goto out;
720cbc21dc1STaehee Yoo 
721cbc21dc1STaehee Yoo 	rt = ip_route_output_ports(amt->net, &fl4, sock->sk,
722cbc21dc1STaehee Yoo 				   amt->remote_ip, amt->local_ip,
723cbc21dc1STaehee Yoo 				   amt->gw_port, amt->relay_port,
724cbc21dc1STaehee Yoo 				   IPPROTO_UDP, 0,
725cbc21dc1STaehee Yoo 				   amt->stream_dev->ifindex);
726cbc21dc1STaehee Yoo 	if (IS_ERR(rt)) {
727cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
728cbc21dc1STaehee Yoo 		goto out;
729cbc21dc1STaehee Yoo 	}
730cbc21dc1STaehee Yoo 
731cbc21dc1STaehee Yoo 	hlen = LL_RESERVED_SPACE(amt->dev);
732cbc21dc1STaehee Yoo 	tlen = amt->dev->needed_tailroom;
733cbc21dc1STaehee Yoo 	len = hlen + tlen + sizeof(*iph) + sizeof(*udph) + sizeof(*amtrh);
734cbc21dc1STaehee Yoo 	skb = netdev_alloc_skb_ip_align(amt->dev, len);
735cbc21dc1STaehee Yoo 	if (!skb) {
736cbc21dc1STaehee Yoo 		ip_rt_put(rt);
737cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
738cbc21dc1STaehee Yoo 		goto out;
739cbc21dc1STaehee Yoo 	}
740cbc21dc1STaehee Yoo 
741cbc21dc1STaehee Yoo 	skb->priority = TC_PRIO_CONTROL;
742cbc21dc1STaehee Yoo 	skb_dst_set(skb, &rt->dst);
743cbc21dc1STaehee Yoo 
744cbc21dc1STaehee Yoo 	len = sizeof(*iph) + sizeof(*udph) + sizeof(*amtrh);
745cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
746cbc21dc1STaehee Yoo 	skb_put(skb, len);
747cbc21dc1STaehee Yoo 	amtrh = skb_pull(skb, sizeof(*iph) + sizeof(*udph));
748cbc21dc1STaehee Yoo 	amtrh->version	 = 0;
749cbc21dc1STaehee Yoo 	amtrh->type	 = AMT_MSG_REQUEST;
750cbc21dc1STaehee Yoo 	amtrh->reserved1 = 0;
751cbc21dc1STaehee Yoo 	amtrh->p	 = v6;
752cbc21dc1STaehee Yoo 	amtrh->reserved2 = 0;
753cbc21dc1STaehee Yoo 	amtrh->nonce	 = amt->nonce;
754cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*udph));
755cbc21dc1STaehee Yoo 	skb_reset_transport_header(skb);
756cbc21dc1STaehee Yoo 	udph		= udp_hdr(skb);
757cbc21dc1STaehee Yoo 	udph->source	= amt->gw_port;
758cbc21dc1STaehee Yoo 	udph->dest	= amt->relay_port;
759cbc21dc1STaehee Yoo 	udph->len	= htons(sizeof(*amtrh) + sizeof(*udph));
760cbc21dc1STaehee Yoo 	udph->check	= 0;
761cbc21dc1STaehee Yoo 	offset = skb_transport_offset(skb);
762cbc21dc1STaehee Yoo 	skb->csum = skb_checksum(skb, offset, skb->len - offset, 0);
763cbc21dc1STaehee Yoo 	udph->check = csum_tcpudp_magic(amt->local_ip, amt->remote_ip,
764cbc21dc1STaehee Yoo 					sizeof(*udph) + sizeof(*amtrh),
765cbc21dc1STaehee Yoo 					IPPROTO_UDP, skb->csum);
766cbc21dc1STaehee Yoo 
767cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*iph));
768cbc21dc1STaehee Yoo 	iph		= ip_hdr(skb);
769cbc21dc1STaehee Yoo 	iph->version	= 4;
770cbc21dc1STaehee Yoo 	iph->ihl	= (sizeof(struct iphdr)) >> 2;
771cbc21dc1STaehee Yoo 	iph->tos	= AMT_TOS;
772cbc21dc1STaehee Yoo 	iph->frag_off	= 0;
773cbc21dc1STaehee Yoo 	iph->ttl	= ip4_dst_hoplimit(&rt->dst);
774cbc21dc1STaehee Yoo 	iph->daddr	= amt->remote_ip;
775cbc21dc1STaehee Yoo 	iph->saddr	= amt->local_ip;
776cbc21dc1STaehee Yoo 	iph->protocol	= IPPROTO_UDP;
777cbc21dc1STaehee Yoo 	iph->tot_len	= htons(len);
778cbc21dc1STaehee Yoo 
779cbc21dc1STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
780cbc21dc1STaehee Yoo 	ip_select_ident(amt->net, skb, NULL);
781cbc21dc1STaehee Yoo 	ip_send_check(iph);
782cbc21dc1STaehee Yoo 	err = ip_local_out(amt->net, sock->sk, skb);
783cbc21dc1STaehee Yoo 	if (unlikely(net_xmit_eval(err)))
784cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
785cbc21dc1STaehee Yoo 
786cbc21dc1STaehee Yoo out:
787cbc21dc1STaehee Yoo 	rcu_read_unlock();
788cbc21dc1STaehee Yoo }
789cbc21dc1STaehee Yoo 
amt_send_igmp_gq(struct amt_dev * amt,struct amt_tunnel_list * tunnel)790cbc21dc1STaehee Yoo static void amt_send_igmp_gq(struct amt_dev *amt,
791cbc21dc1STaehee Yoo 			     struct amt_tunnel_list *tunnel)
792cbc21dc1STaehee Yoo {
793cbc21dc1STaehee Yoo 	struct sk_buff *skb;
794cbc21dc1STaehee Yoo 
795cbc21dc1STaehee Yoo 	skb = amt_build_igmp_gq(amt);
796cbc21dc1STaehee Yoo 	if (!skb)
797cbc21dc1STaehee Yoo 		return;
798cbc21dc1STaehee Yoo 
799cbc21dc1STaehee Yoo 	amt_skb_cb(skb)->tunnel = tunnel;
800cbc21dc1STaehee Yoo 	dev_queue_xmit(skb);
801cbc21dc1STaehee Yoo }
802cbc21dc1STaehee Yoo 
803b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
amt_build_mld_gq(struct amt_dev * amt)804b75f7095STaehee Yoo static struct sk_buff *amt_build_mld_gq(struct amt_dev *amt)
805b75f7095STaehee Yoo {
806b75f7095STaehee Yoo 	u8 ra[AMT_IP6HDR_OPTS] = { IPPROTO_ICMPV6, 0, IPV6_TLV_ROUTERALERT,
807b75f7095STaehee Yoo 				   2, 0, 0, IPV6_TLV_PAD1, IPV6_TLV_PAD1 };
808b75f7095STaehee Yoo 	int hlen = LL_RESERVED_SPACE(amt->dev);
809b75f7095STaehee Yoo 	int tlen = amt->dev->needed_tailroom;
810b75f7095STaehee Yoo 	struct mld2_query *mld2q;
811b75f7095STaehee Yoo 	void *csum_start = NULL;
812b75f7095STaehee Yoo 	struct ipv6hdr *ip6h;
813b75f7095STaehee Yoo 	struct sk_buff *skb;
814b75f7095STaehee Yoo 	struct ethhdr *eth;
815b75f7095STaehee Yoo 	u32 len;
816b75f7095STaehee Yoo 
817b75f7095STaehee Yoo 	len = hlen + tlen + sizeof(*ip6h) + sizeof(ra) + sizeof(*mld2q);
818b75f7095STaehee Yoo 	skb = netdev_alloc_skb_ip_align(amt->dev, len);
819b75f7095STaehee Yoo 	if (!skb)
820b75f7095STaehee Yoo 		return NULL;
821b75f7095STaehee Yoo 
822b75f7095STaehee Yoo 	skb_reserve(skb, hlen);
823b75f7095STaehee Yoo 	skb_push(skb, sizeof(*eth));
824b75f7095STaehee Yoo 	skb_reset_mac_header(skb);
825b75f7095STaehee Yoo 	eth = eth_hdr(skb);
826b75f7095STaehee Yoo 	skb->priority = TC_PRIO_CONTROL;
827b75f7095STaehee Yoo 	skb->protocol = htons(ETH_P_IPV6);
828b75f7095STaehee Yoo 	skb_put_zero(skb, sizeof(*ip6h));
829b75f7095STaehee Yoo 	skb_put_data(skb, ra, sizeof(ra));
830b75f7095STaehee Yoo 	skb_put_zero(skb, sizeof(*mld2q));
831b75f7095STaehee Yoo 	skb_pull(skb, sizeof(*eth));
832b75f7095STaehee Yoo 	skb_reset_network_header(skb);
833b75f7095STaehee Yoo 	ip6h			= ipv6_hdr(skb);
834b75f7095STaehee Yoo 	ip6h->payload_len	= htons(sizeof(ra) + sizeof(*mld2q));
835b75f7095STaehee Yoo 	ip6h->nexthdr		= NEXTHDR_HOP;
836b75f7095STaehee Yoo 	ip6h->hop_limit		= 1;
837b75f7095STaehee Yoo 	ip6h->daddr		= mld2_all_node;
838b75f7095STaehee Yoo 	ip6_flow_hdr(ip6h, 0, 0);
839b75f7095STaehee Yoo 
840b75f7095STaehee Yoo 	if (ipv6_dev_get_saddr(amt->net, amt->dev, &ip6h->daddr, 0,
841b75f7095STaehee Yoo 			       &ip6h->saddr)) {
842b75f7095STaehee Yoo 		amt->dev->stats.tx_errors++;
843b75f7095STaehee Yoo 		kfree_skb(skb);
844b75f7095STaehee Yoo 		return NULL;
845b75f7095STaehee Yoo 	}
846b75f7095STaehee Yoo 
847b75f7095STaehee Yoo 	eth->h_proto = htons(ETH_P_IPV6);
848b75f7095STaehee Yoo 	ether_addr_copy(eth->h_source, amt->dev->dev_addr);
849b75f7095STaehee Yoo 	ipv6_eth_mc_map(&mld2_all_node, eth->h_dest);
850b75f7095STaehee Yoo 
851b75f7095STaehee Yoo 	skb_pull(skb, sizeof(*ip6h) + sizeof(ra));
852b75f7095STaehee Yoo 	skb_reset_transport_header(skb);
853b75f7095STaehee Yoo 	mld2q			= (struct mld2_query *)icmp6_hdr(skb);
854b75f7095STaehee Yoo 	mld2q->mld2q_mrc	= htons(1);
855b75f7095STaehee Yoo 	mld2q->mld2q_type	= ICMPV6_MGM_QUERY;
856b75f7095STaehee Yoo 	mld2q->mld2q_code	= 0;
857b75f7095STaehee Yoo 	mld2q->mld2q_cksum	= 0;
858b75f7095STaehee Yoo 	mld2q->mld2q_resv1	= 0;
859b75f7095STaehee Yoo 	mld2q->mld2q_resv2	= 0;
860b75f7095STaehee Yoo 	mld2q->mld2q_suppress	= 0;
861b75f7095STaehee Yoo 	mld2q->mld2q_qrv	= amt->qrv;
862b75f7095STaehee Yoo 	mld2q->mld2q_nsrcs	= 0;
863b75f7095STaehee Yoo 	mld2q->mld2q_qqic	= amt->qi;
864b75f7095STaehee Yoo 	csum_start		= (void *)mld2q;
865b75f7095STaehee Yoo 	mld2q->mld2q_cksum = csum_ipv6_magic(&ip6h->saddr, &ip6h->daddr,
866b75f7095STaehee Yoo 					     sizeof(*mld2q),
867b75f7095STaehee Yoo 					     IPPROTO_ICMPV6,
868b75f7095STaehee Yoo 					     csum_partial(csum_start,
869b75f7095STaehee Yoo 							  sizeof(*mld2q), 0));
870b75f7095STaehee Yoo 
871b75f7095STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
872b75f7095STaehee Yoo 	skb_push(skb, sizeof(*eth) + sizeof(*ip6h) + sizeof(ra));
873b75f7095STaehee Yoo 	return skb;
874b75f7095STaehee Yoo }
875b75f7095STaehee Yoo 
amt_send_mld_gq(struct amt_dev * amt,struct amt_tunnel_list * tunnel)876b75f7095STaehee Yoo static void amt_send_mld_gq(struct amt_dev *amt, struct amt_tunnel_list *tunnel)
877b75f7095STaehee Yoo {
878b75f7095STaehee Yoo 	struct sk_buff *skb;
879b75f7095STaehee Yoo 
880b75f7095STaehee Yoo 	skb = amt_build_mld_gq(amt);
881b75f7095STaehee Yoo 	if (!skb)
882b75f7095STaehee Yoo 		return;
883b75f7095STaehee Yoo 
884b75f7095STaehee Yoo 	amt_skb_cb(skb)->tunnel = tunnel;
885b75f7095STaehee Yoo 	dev_queue_xmit(skb);
886b75f7095STaehee Yoo }
887b75f7095STaehee Yoo #else
amt_send_mld_gq(struct amt_dev * amt,struct amt_tunnel_list * tunnel)888b75f7095STaehee Yoo static void amt_send_mld_gq(struct amt_dev *amt, struct amt_tunnel_list *tunnel)
889b75f7095STaehee Yoo {
890b75f7095STaehee Yoo }
891b75f7095STaehee Yoo #endif
892b75f7095STaehee Yoo 
amt_queue_event(struct amt_dev * amt,enum amt_event event,struct sk_buff * skb)89330e22a6eSTaehee Yoo static bool amt_queue_event(struct amt_dev *amt, enum amt_event event,
89430e22a6eSTaehee Yoo 			    struct sk_buff *skb)
89530e22a6eSTaehee Yoo {
89630e22a6eSTaehee Yoo 	int index;
89730e22a6eSTaehee Yoo 
89830e22a6eSTaehee Yoo 	spin_lock_bh(&amt->lock);
89930e22a6eSTaehee Yoo 	if (amt->nr_events >= AMT_MAX_EVENTS) {
90030e22a6eSTaehee Yoo 		spin_unlock_bh(&amt->lock);
90130e22a6eSTaehee Yoo 		return 1;
90230e22a6eSTaehee Yoo 	}
90330e22a6eSTaehee Yoo 
90430e22a6eSTaehee Yoo 	index = (amt->event_idx + amt->nr_events) % AMT_MAX_EVENTS;
90530e22a6eSTaehee Yoo 	amt->events[index].event = event;
90630e22a6eSTaehee Yoo 	amt->events[index].skb = skb;
90730e22a6eSTaehee Yoo 	amt->nr_events++;
90830e22a6eSTaehee Yoo 	amt->event_idx %= AMT_MAX_EVENTS;
90930e22a6eSTaehee Yoo 	queue_work(amt_wq, &amt->event_wq);
91030e22a6eSTaehee Yoo 	spin_unlock_bh(&amt->lock);
91130e22a6eSTaehee Yoo 
91230e22a6eSTaehee Yoo 	return 0;
91330e22a6eSTaehee Yoo }
91430e22a6eSTaehee Yoo 
amt_secret_work(struct work_struct * work)915cbc21dc1STaehee Yoo static void amt_secret_work(struct work_struct *work)
916cbc21dc1STaehee Yoo {
917cbc21dc1STaehee Yoo 	struct amt_dev *amt = container_of(to_delayed_work(work),
918cbc21dc1STaehee Yoo 					   struct amt_dev,
919cbc21dc1STaehee Yoo 					   secret_wq);
920cbc21dc1STaehee Yoo 
921cbc21dc1STaehee Yoo 	spin_lock_bh(&amt->lock);
922cbc21dc1STaehee Yoo 	get_random_bytes(&amt->key, sizeof(siphash_key_t));
923cbc21dc1STaehee Yoo 	spin_unlock_bh(&amt->lock);
924cbc21dc1STaehee Yoo 	mod_delayed_work(amt_wq, &amt->secret_wq,
925cbc21dc1STaehee Yoo 			 msecs_to_jiffies(AMT_SECRET_TIMEOUT));
926cbc21dc1STaehee Yoo }
927cbc21dc1STaehee Yoo 
amt_event_send_discovery(struct amt_dev * amt)92830e22a6eSTaehee Yoo static void amt_event_send_discovery(struct amt_dev *amt)
929cbc21dc1STaehee Yoo {
930cbc21dc1STaehee Yoo 	if (amt->status > AMT_STATUS_SENT_DISCOVERY)
931cbc21dc1STaehee Yoo 		goto out;
932cbc21dc1STaehee Yoo 	get_random_bytes(&amt->nonce, sizeof(__be32));
933cbc21dc1STaehee Yoo 
934cbc21dc1STaehee Yoo 	amt_send_discovery(amt);
935cbc21dc1STaehee Yoo out:
936cbc21dc1STaehee Yoo 	mod_delayed_work(amt_wq, &amt->discovery_wq,
937cbc21dc1STaehee Yoo 			 msecs_to_jiffies(AMT_DISCOVERY_TIMEOUT));
938cbc21dc1STaehee Yoo }
939cbc21dc1STaehee Yoo 
amt_discovery_work(struct work_struct * work)940cbc21dc1STaehee Yoo static void amt_discovery_work(struct work_struct *work)
941cbc21dc1STaehee Yoo {
942cbc21dc1STaehee Yoo 	struct amt_dev *amt = container_of(to_delayed_work(work),
943cbc21dc1STaehee Yoo 					   struct amt_dev,
944cbc21dc1STaehee Yoo 					   discovery_wq);
945cbc21dc1STaehee Yoo 
94630e22a6eSTaehee Yoo 	if (amt_queue_event(amt, AMT_EVENT_SEND_DISCOVERY, NULL))
947cbc21dc1STaehee Yoo 		mod_delayed_work(amt_wq, &amt->discovery_wq,
948cbc21dc1STaehee Yoo 				 msecs_to_jiffies(AMT_DISCOVERY_TIMEOUT));
94930e22a6eSTaehee Yoo }
95030e22a6eSTaehee Yoo 
amt_event_send_request(struct amt_dev * amt)95130e22a6eSTaehee Yoo static void amt_event_send_request(struct amt_dev *amt)
95230e22a6eSTaehee Yoo {
953cbc21dc1STaehee Yoo 	u32 exp;
954cbc21dc1STaehee Yoo 
955cbc21dc1STaehee Yoo 	if (amt->status < AMT_STATUS_RECEIVED_ADVERTISEMENT)
956cbc21dc1STaehee Yoo 		goto out;
957cbc21dc1STaehee Yoo 
958937956baSTaehee Yoo 	if (amt->req_cnt > AMT_MAX_REQ_COUNT) {
959cbc21dc1STaehee Yoo 		netdev_dbg(amt->dev, "Gateway is not ready");
960cbc21dc1STaehee Yoo 		amt->qi = AMT_INIT_REQ_TIMEOUT;
961928f353cSTaehee Yoo 		WRITE_ONCE(amt->ready4, false);
962928f353cSTaehee Yoo 		WRITE_ONCE(amt->ready6, false);
963cbc21dc1STaehee Yoo 		amt->remote_ip = 0;
9649c343ea6STaehee Yoo 		amt_update_gw_status(amt, AMT_STATUS_INIT, false);
965cbc21dc1STaehee Yoo 		amt->req_cnt = 0;
966627f1693STaehee Yoo 		amt->nonce = 0;
967937956baSTaehee Yoo 		goto out;
968cbc21dc1STaehee Yoo 	}
969cbc21dc1STaehee Yoo 
970239d8866STaehee Yoo 	if (!amt->req_cnt) {
971239d8866STaehee Yoo 		WRITE_ONCE(amt->ready4, false);
972239d8866STaehee Yoo 		WRITE_ONCE(amt->ready6, false);
973627f1693STaehee Yoo 		get_random_bytes(&amt->nonce, sizeof(__be32));
974239d8866STaehee Yoo 	}
975627f1693STaehee Yoo 
976cbc21dc1STaehee Yoo 	amt_send_request(amt, false);
977cbc21dc1STaehee Yoo 	amt_send_request(amt, true);
9789c343ea6STaehee Yoo 	amt_update_gw_status(amt, AMT_STATUS_SENT_REQUEST, true);
979937956baSTaehee Yoo 	amt->req_cnt++;
980cbc21dc1STaehee Yoo out:
981cbc21dc1STaehee Yoo 	exp = min_t(u32, (1 * (1 << amt->req_cnt)), AMT_MAX_REQ_TIMEOUT);
982cbc21dc1STaehee Yoo 	mod_delayed_work(amt_wq, &amt->req_wq, msecs_to_jiffies(exp * 1000));
983cbc21dc1STaehee Yoo }
984cbc21dc1STaehee Yoo 
amt_req_work(struct work_struct * work)985cbc21dc1STaehee Yoo static void amt_req_work(struct work_struct *work)
986cbc21dc1STaehee Yoo {
987cbc21dc1STaehee Yoo 	struct amt_dev *amt = container_of(to_delayed_work(work),
988cbc21dc1STaehee Yoo 					   struct amt_dev,
989cbc21dc1STaehee Yoo 					   req_wq);
990cbc21dc1STaehee Yoo 
99130e22a6eSTaehee Yoo 	if (amt_queue_event(amt, AMT_EVENT_SEND_REQUEST, NULL))
99230e22a6eSTaehee Yoo 		mod_delayed_work(amt_wq, &amt->req_wq,
99330e22a6eSTaehee Yoo 				 msecs_to_jiffies(100));
994cbc21dc1STaehee Yoo }
995cbc21dc1STaehee Yoo 
amt_send_membership_update(struct amt_dev * amt,struct sk_buff * skb,bool v6)996cbc21dc1STaehee Yoo static bool amt_send_membership_update(struct amt_dev *amt,
997cbc21dc1STaehee Yoo 				       struct sk_buff *skb,
998cbc21dc1STaehee Yoo 				       bool v6)
999cbc21dc1STaehee Yoo {
1000cbc21dc1STaehee Yoo 	struct amt_header_membership_update *amtmu;
1001cbc21dc1STaehee Yoo 	struct socket *sock;
1002cbc21dc1STaehee Yoo 	struct iphdr *iph;
1003cbc21dc1STaehee Yoo 	struct flowi4 fl4;
1004cbc21dc1STaehee Yoo 	struct rtable *rt;
1005cbc21dc1STaehee Yoo 	int err;
1006cbc21dc1STaehee Yoo 
1007cbc21dc1STaehee Yoo 	sock = rcu_dereference_bh(amt->sock);
1008cbc21dc1STaehee Yoo 	if (!sock)
1009cbc21dc1STaehee Yoo 		return true;
1010cbc21dc1STaehee Yoo 
1011cbc21dc1STaehee Yoo 	err = skb_cow_head(skb, LL_RESERVED_SPACE(amt->dev) + sizeof(*amtmu) +
1012cbc21dc1STaehee Yoo 			   sizeof(*iph) + sizeof(struct udphdr));
1013cbc21dc1STaehee Yoo 	if (err)
1014cbc21dc1STaehee Yoo 		return true;
1015cbc21dc1STaehee Yoo 
1016cbc21dc1STaehee Yoo 	skb_reset_inner_headers(skb);
1017cbc21dc1STaehee Yoo 	memset(&fl4, 0, sizeof(struct flowi4));
1018cbc21dc1STaehee Yoo 	fl4.flowi4_oif         = amt->stream_dev->ifindex;
1019cbc21dc1STaehee Yoo 	fl4.daddr              = amt->remote_ip;
1020cbc21dc1STaehee Yoo 	fl4.saddr              = amt->local_ip;
1021cbc21dc1STaehee Yoo 	fl4.flowi4_tos         = AMT_TOS;
1022cbc21dc1STaehee Yoo 	fl4.flowi4_proto       = IPPROTO_UDP;
1023cbc21dc1STaehee Yoo 	rt = ip_route_output_key(amt->net, &fl4);
1024cbc21dc1STaehee Yoo 	if (IS_ERR(rt)) {
1025cbc21dc1STaehee Yoo 		netdev_dbg(amt->dev, "no route to %pI4\n", &amt->remote_ip);
1026cbc21dc1STaehee Yoo 		return true;
1027cbc21dc1STaehee Yoo 	}
1028cbc21dc1STaehee Yoo 
1029cbc21dc1STaehee Yoo 	amtmu			= skb_push(skb, sizeof(*amtmu));
1030cbc21dc1STaehee Yoo 	amtmu->version		= 0;
1031cbc21dc1STaehee Yoo 	amtmu->type		= AMT_MSG_MEMBERSHIP_UPDATE;
1032cbc21dc1STaehee Yoo 	amtmu->reserved		= 0;
1033cbc21dc1STaehee Yoo 	amtmu->nonce		= amt->nonce;
1034cbc21dc1STaehee Yoo 	amtmu->response_mac	= amt->mac;
1035cbc21dc1STaehee Yoo 
1036cbc21dc1STaehee Yoo 	if (!v6)
1037cbc21dc1STaehee Yoo 		skb_set_inner_protocol(skb, htons(ETH_P_IP));
1038cbc21dc1STaehee Yoo 	else
1039cbc21dc1STaehee Yoo 		skb_set_inner_protocol(skb, htons(ETH_P_IPV6));
1040cbc21dc1STaehee Yoo 	udp_tunnel_xmit_skb(rt, sock->sk, skb,
1041cbc21dc1STaehee Yoo 			    fl4.saddr,
1042cbc21dc1STaehee Yoo 			    fl4.daddr,
1043cbc21dc1STaehee Yoo 			    AMT_TOS,
1044cbc21dc1STaehee Yoo 			    ip4_dst_hoplimit(&rt->dst),
1045cbc21dc1STaehee Yoo 			    0,
1046cbc21dc1STaehee Yoo 			    amt->gw_port,
1047cbc21dc1STaehee Yoo 			    amt->relay_port,
1048cbc21dc1STaehee Yoo 			    false,
1049cbc21dc1STaehee Yoo 			    false);
1050cbc21dc1STaehee Yoo 	amt_update_gw_status(amt, AMT_STATUS_SENT_UPDATE, true);
1051cbc21dc1STaehee Yoo 	return false;
1052cbc21dc1STaehee Yoo }
1053cbc21dc1STaehee Yoo 
amt_send_multicast_data(struct amt_dev * amt,const struct sk_buff * oskb,struct amt_tunnel_list * tunnel,bool v6)1054cbc21dc1STaehee Yoo static void amt_send_multicast_data(struct amt_dev *amt,
1055cbc21dc1STaehee Yoo 				    const struct sk_buff *oskb,
1056cbc21dc1STaehee Yoo 				    struct amt_tunnel_list *tunnel,
1057cbc21dc1STaehee Yoo 				    bool v6)
1058cbc21dc1STaehee Yoo {
1059cbc21dc1STaehee Yoo 	struct amt_header_mcast_data *amtmd;
1060cbc21dc1STaehee Yoo 	struct socket *sock;
1061cbc21dc1STaehee Yoo 	struct sk_buff *skb;
1062cbc21dc1STaehee Yoo 	struct iphdr *iph;
1063cbc21dc1STaehee Yoo 	struct flowi4 fl4;
1064cbc21dc1STaehee Yoo 	struct rtable *rt;
1065cbc21dc1STaehee Yoo 
1066cbc21dc1STaehee Yoo 	sock = rcu_dereference_bh(amt->sock);
1067cbc21dc1STaehee Yoo 	if (!sock)
1068cbc21dc1STaehee Yoo 		return;
1069cbc21dc1STaehee Yoo 
1070cbc21dc1STaehee Yoo 	skb = skb_copy_expand(oskb, sizeof(*amtmd) + sizeof(*iph) +
1071cbc21dc1STaehee Yoo 			      sizeof(struct udphdr), 0, GFP_ATOMIC);
1072cbc21dc1STaehee Yoo 	if (!skb)
1073cbc21dc1STaehee Yoo 		return;
1074cbc21dc1STaehee Yoo 
1075cbc21dc1STaehee Yoo 	skb_reset_inner_headers(skb);
1076cbc21dc1STaehee Yoo 	memset(&fl4, 0, sizeof(struct flowi4));
1077cbc21dc1STaehee Yoo 	fl4.flowi4_oif         = amt->stream_dev->ifindex;
1078cbc21dc1STaehee Yoo 	fl4.daddr              = tunnel->ip4;
1079cbc21dc1STaehee Yoo 	fl4.saddr              = amt->local_ip;
1080cbc21dc1STaehee Yoo 	fl4.flowi4_proto       = IPPROTO_UDP;
1081cbc21dc1STaehee Yoo 	rt = ip_route_output_key(amt->net, &fl4);
1082cbc21dc1STaehee Yoo 	if (IS_ERR(rt)) {
1083cbc21dc1STaehee Yoo 		netdev_dbg(amt->dev, "no route to %pI4\n", &tunnel->ip4);
1084cbc21dc1STaehee Yoo 		kfree_skb(skb);
1085cbc21dc1STaehee Yoo 		return;
1086cbc21dc1STaehee Yoo 	}
1087cbc21dc1STaehee Yoo 
1088cbc21dc1STaehee Yoo 	amtmd = skb_push(skb, sizeof(*amtmd));
1089cbc21dc1STaehee Yoo 	amtmd->version = 0;
1090cbc21dc1STaehee Yoo 	amtmd->reserved = 0;
1091cbc21dc1STaehee Yoo 	amtmd->type = AMT_MSG_MULTICAST_DATA;
1092cbc21dc1STaehee Yoo 
1093cbc21dc1STaehee Yoo 	if (!v6)
1094cbc21dc1STaehee Yoo 		skb_set_inner_protocol(skb, htons(ETH_P_IP));
1095cbc21dc1STaehee Yoo 	else
1096cbc21dc1STaehee Yoo 		skb_set_inner_protocol(skb, htons(ETH_P_IPV6));
1097cbc21dc1STaehee Yoo 	udp_tunnel_xmit_skb(rt, sock->sk, skb,
1098cbc21dc1STaehee Yoo 			    fl4.saddr,
1099cbc21dc1STaehee Yoo 			    fl4.daddr,
1100cbc21dc1STaehee Yoo 			    AMT_TOS,
1101cbc21dc1STaehee Yoo 			    ip4_dst_hoplimit(&rt->dst),
1102cbc21dc1STaehee Yoo 			    0,
1103cbc21dc1STaehee Yoo 			    amt->relay_port,
1104cbc21dc1STaehee Yoo 			    tunnel->source_port,
1105cbc21dc1STaehee Yoo 			    false,
1106cbc21dc1STaehee Yoo 			    false);
1107cbc21dc1STaehee Yoo }
1108cbc21dc1STaehee Yoo 
amt_send_membership_query(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel,bool v6)1109cbc21dc1STaehee Yoo static bool amt_send_membership_query(struct amt_dev *amt,
1110cbc21dc1STaehee Yoo 				      struct sk_buff *skb,
1111cbc21dc1STaehee Yoo 				      struct amt_tunnel_list *tunnel,
1112cbc21dc1STaehee Yoo 				      bool v6)
1113cbc21dc1STaehee Yoo {
1114cbc21dc1STaehee Yoo 	struct amt_header_membership_query *amtmq;
1115cbc21dc1STaehee Yoo 	struct socket *sock;
1116cbc21dc1STaehee Yoo 	struct rtable *rt;
1117cbc21dc1STaehee Yoo 	struct flowi4 fl4;
1118cbc21dc1STaehee Yoo 	int err;
1119cbc21dc1STaehee Yoo 
1120cbc21dc1STaehee Yoo 	sock = rcu_dereference_bh(amt->sock);
1121cbc21dc1STaehee Yoo 	if (!sock)
1122cbc21dc1STaehee Yoo 		return true;
1123cbc21dc1STaehee Yoo 
1124cbc21dc1STaehee Yoo 	err = skb_cow_head(skb, LL_RESERVED_SPACE(amt->dev) + sizeof(*amtmq) +
1125cbc21dc1STaehee Yoo 			   sizeof(struct iphdr) + sizeof(struct udphdr));
1126cbc21dc1STaehee Yoo 	if (err)
1127cbc21dc1STaehee Yoo 		return true;
1128cbc21dc1STaehee Yoo 
1129cbc21dc1STaehee Yoo 	skb_reset_inner_headers(skb);
1130cbc21dc1STaehee Yoo 	memset(&fl4, 0, sizeof(struct flowi4));
1131cbc21dc1STaehee Yoo 	fl4.flowi4_oif         = amt->stream_dev->ifindex;
1132cbc21dc1STaehee Yoo 	fl4.daddr              = tunnel->ip4;
1133cbc21dc1STaehee Yoo 	fl4.saddr              = amt->local_ip;
1134cbc21dc1STaehee Yoo 	fl4.flowi4_tos         = AMT_TOS;
1135cbc21dc1STaehee Yoo 	fl4.flowi4_proto       = IPPROTO_UDP;
1136cbc21dc1STaehee Yoo 	rt = ip_route_output_key(amt->net, &fl4);
1137cbc21dc1STaehee Yoo 	if (IS_ERR(rt)) {
1138cbc21dc1STaehee Yoo 		netdev_dbg(amt->dev, "no route to %pI4\n", &tunnel->ip4);
1139dd3ca4c5STaehee Yoo 		return true;
1140cbc21dc1STaehee Yoo 	}
1141cbc21dc1STaehee Yoo 
1142cbc21dc1STaehee Yoo 	amtmq		= skb_push(skb, sizeof(*amtmq));
1143cbc21dc1STaehee Yoo 	amtmq->version	= 0;
1144cbc21dc1STaehee Yoo 	amtmq->type	= AMT_MSG_MEMBERSHIP_QUERY;
1145cbc21dc1STaehee Yoo 	amtmq->reserved = 0;
1146cbc21dc1STaehee Yoo 	amtmq->l	= 0;
1147cbc21dc1STaehee Yoo 	amtmq->g	= 0;
1148cbc21dc1STaehee Yoo 	amtmq->nonce	= tunnel->nonce;
1149cbc21dc1STaehee Yoo 	amtmq->response_mac = tunnel->mac;
1150cbc21dc1STaehee Yoo 
1151cbc21dc1STaehee Yoo 	if (!v6)
1152cbc21dc1STaehee Yoo 		skb_set_inner_protocol(skb, htons(ETH_P_IP));
1153cbc21dc1STaehee Yoo 	else
1154cbc21dc1STaehee Yoo 		skb_set_inner_protocol(skb, htons(ETH_P_IPV6));
1155cbc21dc1STaehee Yoo 	udp_tunnel_xmit_skb(rt, sock->sk, skb,
1156cbc21dc1STaehee Yoo 			    fl4.saddr,
1157cbc21dc1STaehee Yoo 			    fl4.daddr,
1158cbc21dc1STaehee Yoo 			    AMT_TOS,
1159cbc21dc1STaehee Yoo 			    ip4_dst_hoplimit(&rt->dst),
1160cbc21dc1STaehee Yoo 			    0,
1161cbc21dc1STaehee Yoo 			    amt->relay_port,
1162cbc21dc1STaehee Yoo 			    tunnel->source_port,
1163cbc21dc1STaehee Yoo 			    false,
1164cbc21dc1STaehee Yoo 			    false);
1165cbc21dc1STaehee Yoo 	amt_update_relay_status(tunnel, AMT_STATUS_SENT_QUERY, true);
1166cbc21dc1STaehee Yoo 	return false;
1167cbc21dc1STaehee Yoo }
1168cbc21dc1STaehee Yoo 
amt_dev_xmit(struct sk_buff * skb,struct net_device * dev)1169cbc21dc1STaehee Yoo static netdev_tx_t amt_dev_xmit(struct sk_buff *skb, struct net_device *dev)
1170cbc21dc1STaehee Yoo {
1171cbc21dc1STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
1172cbc21dc1STaehee Yoo 	struct amt_tunnel_list *tunnel;
1173bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
1174bc54e49cSTaehee Yoo 	union amt_addr group = {0,};
1175b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1176b75f7095STaehee Yoo 	struct ipv6hdr *ip6h;
1177b75f7095STaehee Yoo 	struct mld_msg *mld;
1178b75f7095STaehee Yoo #endif
1179cbc21dc1STaehee Yoo 	bool report = false;
1180cbc21dc1STaehee Yoo 	struct igmphdr *ih;
1181cbc21dc1STaehee Yoo 	bool query = false;
1182cbc21dc1STaehee Yoo 	struct iphdr *iph;
1183cbc21dc1STaehee Yoo 	bool data = false;
1184cbc21dc1STaehee Yoo 	bool v6 = false;
1185bc54e49cSTaehee Yoo 	u32 hash;
1186cbc21dc1STaehee Yoo 
1187cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
1188cbc21dc1STaehee Yoo 	if (iph->version == 4) {
1189cbc21dc1STaehee Yoo 		if (!ipv4_is_multicast(iph->daddr))
1190cbc21dc1STaehee Yoo 			goto free;
1191cbc21dc1STaehee Yoo 
1192cbc21dc1STaehee Yoo 		if (!ip_mc_check_igmp(skb)) {
1193cbc21dc1STaehee Yoo 			ih = igmp_hdr(skb);
1194cbc21dc1STaehee Yoo 			switch (ih->type) {
1195cbc21dc1STaehee Yoo 			case IGMPV3_HOST_MEMBERSHIP_REPORT:
1196cbc21dc1STaehee Yoo 			case IGMP_HOST_MEMBERSHIP_REPORT:
1197cbc21dc1STaehee Yoo 				report = true;
1198cbc21dc1STaehee Yoo 				break;
1199cbc21dc1STaehee Yoo 			case IGMP_HOST_MEMBERSHIP_QUERY:
1200cbc21dc1STaehee Yoo 				query = true;
1201cbc21dc1STaehee Yoo 				break;
1202cbc21dc1STaehee Yoo 			default:
1203cbc21dc1STaehee Yoo 				goto free;
1204cbc21dc1STaehee Yoo 			}
1205cbc21dc1STaehee Yoo 		} else {
1206cbc21dc1STaehee Yoo 			data = true;
1207cbc21dc1STaehee Yoo 		}
1208cbc21dc1STaehee Yoo 		v6 = false;
1209bc54e49cSTaehee Yoo 		group.ip4 = iph->daddr;
1210b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1211b75f7095STaehee Yoo 	} else if (iph->version == 6) {
1212b75f7095STaehee Yoo 		ip6h = ipv6_hdr(skb);
1213b75f7095STaehee Yoo 		if (!ipv6_addr_is_multicast(&ip6h->daddr))
1214b75f7095STaehee Yoo 			goto free;
1215b75f7095STaehee Yoo 
1216b75f7095STaehee Yoo 		if (!ipv6_mc_check_mld(skb)) {
1217b75f7095STaehee Yoo 			mld = (struct mld_msg *)skb_transport_header(skb);
1218b75f7095STaehee Yoo 			switch (mld->mld_type) {
1219b75f7095STaehee Yoo 			case ICMPV6_MGM_REPORT:
1220b75f7095STaehee Yoo 			case ICMPV6_MLD2_REPORT:
1221b75f7095STaehee Yoo 				report = true;
1222b75f7095STaehee Yoo 				break;
1223b75f7095STaehee Yoo 			case ICMPV6_MGM_QUERY:
1224b75f7095STaehee Yoo 				query = true;
1225b75f7095STaehee Yoo 				break;
1226b75f7095STaehee Yoo 			default:
1227b75f7095STaehee Yoo 				goto free;
1228b75f7095STaehee Yoo 			}
1229b75f7095STaehee Yoo 		} else {
1230b75f7095STaehee Yoo 			data = true;
1231b75f7095STaehee Yoo 		}
1232b75f7095STaehee Yoo 		v6 = true;
1233b75f7095STaehee Yoo 		group.ip6 = ip6h->daddr;
1234b75f7095STaehee Yoo #endif
1235cbc21dc1STaehee Yoo 	} else {
1236cbc21dc1STaehee Yoo 		dev->stats.tx_errors++;
1237cbc21dc1STaehee Yoo 		goto free;
1238cbc21dc1STaehee Yoo 	}
1239cbc21dc1STaehee Yoo 
1240cbc21dc1STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(struct ethhdr)))
1241cbc21dc1STaehee Yoo 		goto free;
1242cbc21dc1STaehee Yoo 
1243cbc21dc1STaehee Yoo 	skb_pull(skb, sizeof(struct ethhdr));
1244cbc21dc1STaehee Yoo 
1245cbc21dc1STaehee Yoo 	if (amt->mode == AMT_MODE_GATEWAY) {
1246cbc21dc1STaehee Yoo 		/* Gateway only passes IGMP/MLD packets */
1247cbc21dc1STaehee Yoo 		if (!report)
1248cbc21dc1STaehee Yoo 			goto free;
1249928f353cSTaehee Yoo 		if ((!v6 && !READ_ONCE(amt->ready4)) ||
1250928f353cSTaehee Yoo 		    (v6 && !READ_ONCE(amt->ready6)))
1251cbc21dc1STaehee Yoo 			goto free;
1252cbc21dc1STaehee Yoo 		if (amt_send_membership_update(amt, skb,  v6))
1253cbc21dc1STaehee Yoo 			goto free;
1254cbc21dc1STaehee Yoo 		goto unlock;
1255cbc21dc1STaehee Yoo 	} else if (amt->mode == AMT_MODE_RELAY) {
1256cbc21dc1STaehee Yoo 		if (query) {
1257cbc21dc1STaehee Yoo 			tunnel = amt_skb_cb(skb)->tunnel;
1258cbc21dc1STaehee Yoo 			if (!tunnel) {
1259cbc21dc1STaehee Yoo 				WARN_ON(1);
1260cbc21dc1STaehee Yoo 				goto free;
1261cbc21dc1STaehee Yoo 			}
1262cbc21dc1STaehee Yoo 
1263cbc21dc1STaehee Yoo 			/* Do not forward unexpected query */
1264cbc21dc1STaehee Yoo 			if (amt_send_membership_query(amt, skb, tunnel, v6))
1265cbc21dc1STaehee Yoo 				goto free;
1266cbc21dc1STaehee Yoo 			goto unlock;
1267cbc21dc1STaehee Yoo 		}
1268cbc21dc1STaehee Yoo 
1269cbc21dc1STaehee Yoo 		if (!data)
1270cbc21dc1STaehee Yoo 			goto free;
1271bc54e49cSTaehee Yoo 		list_for_each_entry_rcu(tunnel, &amt->tunnel_list, list) {
1272bc54e49cSTaehee Yoo 			hash = amt_group_hash(tunnel, &group);
1273bc54e49cSTaehee Yoo 			hlist_for_each_entry_rcu(gnode, &tunnel->groups[hash],
1274bc54e49cSTaehee Yoo 						 node) {
1275b75f7095STaehee Yoo 				if (!v6) {
1276bc54e49cSTaehee Yoo 					if (gnode->group_addr.ip4 == iph->daddr)
1277bc54e49cSTaehee Yoo 						goto found;
1278b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1279b75f7095STaehee Yoo 				} else {
1280b75f7095STaehee Yoo 					if (ipv6_addr_equal(&gnode->group_addr.ip6,
1281b75f7095STaehee Yoo 							    &ip6h->daddr))
1282b75f7095STaehee Yoo 						goto found;
1283b75f7095STaehee Yoo #endif
1284b75f7095STaehee Yoo 				}
1285bc54e49cSTaehee Yoo 			}
1286bc54e49cSTaehee Yoo 			continue;
1287b75f7095STaehee Yoo found:
1288cbc21dc1STaehee Yoo 			amt_send_multicast_data(amt, skb, tunnel, v6);
1289cbc21dc1STaehee Yoo 		}
1290bc54e49cSTaehee Yoo 	}
1291cbc21dc1STaehee Yoo 
1292cbc21dc1STaehee Yoo 	dev_kfree_skb(skb);
1293cbc21dc1STaehee Yoo 	return NETDEV_TX_OK;
1294cbc21dc1STaehee Yoo free:
1295cbc21dc1STaehee Yoo 	dev_kfree_skb(skb);
1296cbc21dc1STaehee Yoo unlock:
1297cbc21dc1STaehee Yoo 	dev->stats.tx_dropped++;
1298cbc21dc1STaehee Yoo 	return NETDEV_TX_OK;
1299cbc21dc1STaehee Yoo }
1300cbc21dc1STaehee Yoo 
amt_parse_type(struct sk_buff * skb)1301cbc21dc1STaehee Yoo static int amt_parse_type(struct sk_buff *skb)
1302cbc21dc1STaehee Yoo {
1303cbc21dc1STaehee Yoo 	struct amt_header *amth;
1304cbc21dc1STaehee Yoo 
1305cbc21dc1STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(struct udphdr) +
1306cbc21dc1STaehee Yoo 			   sizeof(struct amt_header)))
1307cbc21dc1STaehee Yoo 		return -1;
1308cbc21dc1STaehee Yoo 
1309cbc21dc1STaehee Yoo 	amth = (struct amt_header *)(udp_hdr(skb) + 1);
1310cbc21dc1STaehee Yoo 
1311cbc21dc1STaehee Yoo 	if (amth->version != 0)
1312cbc21dc1STaehee Yoo 		return -1;
1313cbc21dc1STaehee Yoo 
1314cbc21dc1STaehee Yoo 	if (amth->type >= __AMT_MSG_MAX || !amth->type)
1315cbc21dc1STaehee Yoo 		return -1;
1316cbc21dc1STaehee Yoo 	return amth->type;
1317cbc21dc1STaehee Yoo }
1318cbc21dc1STaehee Yoo 
amt_clear_groups(struct amt_tunnel_list * tunnel)1319bc54e49cSTaehee Yoo static void amt_clear_groups(struct amt_tunnel_list *tunnel)
1320bc54e49cSTaehee Yoo {
1321bc54e49cSTaehee Yoo 	struct amt_dev *amt = tunnel->amt;
1322bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
1323bc54e49cSTaehee Yoo 	struct hlist_node *t;
1324bc54e49cSTaehee Yoo 	int i;
1325bc54e49cSTaehee Yoo 
1326bc54e49cSTaehee Yoo 	spin_lock_bh(&tunnel->lock);
1327bc54e49cSTaehee Yoo 	rcu_read_lock();
1328bc54e49cSTaehee Yoo 	for (i = 0; i < amt->hash_buckets; i++)
1329bc54e49cSTaehee Yoo 		hlist_for_each_entry_safe(gnode, t, &tunnel->groups[i], node)
1330bc54e49cSTaehee Yoo 			amt_del_group(amt, gnode);
1331bc54e49cSTaehee Yoo 	rcu_read_unlock();
1332bc54e49cSTaehee Yoo 	spin_unlock_bh(&tunnel->lock);
1333bc54e49cSTaehee Yoo }
1334bc54e49cSTaehee Yoo 
amt_tunnel_expire(struct work_struct * work)1335cbc21dc1STaehee Yoo static void amt_tunnel_expire(struct work_struct *work)
1336cbc21dc1STaehee Yoo {
1337cbc21dc1STaehee Yoo 	struct amt_tunnel_list *tunnel = container_of(to_delayed_work(work),
1338cbc21dc1STaehee Yoo 						      struct amt_tunnel_list,
1339cbc21dc1STaehee Yoo 						      gc_wq);
1340cbc21dc1STaehee Yoo 	struct amt_dev *amt = tunnel->amt;
1341cbc21dc1STaehee Yoo 
1342cbc21dc1STaehee Yoo 	spin_lock_bh(&amt->lock);
1343cbc21dc1STaehee Yoo 	rcu_read_lock();
1344cbc21dc1STaehee Yoo 	list_del_rcu(&tunnel->list);
1345cbc21dc1STaehee Yoo 	amt->nr_tunnels--;
1346bc54e49cSTaehee Yoo 	amt_clear_groups(tunnel);
1347cbc21dc1STaehee Yoo 	rcu_read_unlock();
1348cbc21dc1STaehee Yoo 	spin_unlock_bh(&amt->lock);
1349cbc21dc1STaehee Yoo 	kfree_rcu(tunnel, rcu);
1350cbc21dc1STaehee Yoo }
1351cbc21dc1STaehee Yoo 
amt_cleanup_srcs(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode)1352bc54e49cSTaehee Yoo static void amt_cleanup_srcs(struct amt_dev *amt,
1353bc54e49cSTaehee Yoo 			     struct amt_tunnel_list *tunnel,
1354bc54e49cSTaehee Yoo 			     struct amt_group_node *gnode)
1355bc54e49cSTaehee Yoo {
1356bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
1357bc54e49cSTaehee Yoo 	struct hlist_node *t;
1358bc54e49cSTaehee Yoo 	int i;
1359bc54e49cSTaehee Yoo 
1360bc54e49cSTaehee Yoo 	/* Delete old sources */
1361bc54e49cSTaehee Yoo 	for (i = 0; i < amt->hash_buckets; i++) {
1362bc54e49cSTaehee Yoo 		hlist_for_each_entry_safe(snode, t, &gnode->sources[i], node) {
1363bc54e49cSTaehee Yoo 			if (snode->flags == AMT_SOURCE_OLD)
1364bc54e49cSTaehee Yoo 				amt_destroy_source(snode);
1365bc54e49cSTaehee Yoo 		}
1366bc54e49cSTaehee Yoo 	}
1367bc54e49cSTaehee Yoo 
1368bc54e49cSTaehee Yoo 	/* switch from new to old */
1369bc54e49cSTaehee Yoo 	for (i = 0; i < amt->hash_buckets; i++)  {
1370bc54e49cSTaehee Yoo 		hlist_for_each_entry_rcu(snode, &gnode->sources[i], node) {
1371bc54e49cSTaehee Yoo 			snode->flags = AMT_SOURCE_OLD;
1372bc54e49cSTaehee Yoo 			if (!gnode->v6)
1373bc54e49cSTaehee Yoo 				netdev_dbg(snode->gnode->amt->dev,
1374bc54e49cSTaehee Yoo 					   "Add source as OLD %pI4 from %pI4\n",
1375bc54e49cSTaehee Yoo 					   &snode->source_addr.ip4,
1376bc54e49cSTaehee Yoo 					   &gnode->group_addr.ip4);
1377b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1378b75f7095STaehee Yoo 			else
1379b75f7095STaehee Yoo 				netdev_dbg(snode->gnode->amt->dev,
1380b75f7095STaehee Yoo 					   "Add source as OLD %pI6 from %pI6\n",
1381b75f7095STaehee Yoo 					   &snode->source_addr.ip6,
1382b75f7095STaehee Yoo 					   &gnode->group_addr.ip6);
1383b75f7095STaehee Yoo #endif
1384bc54e49cSTaehee Yoo 		}
1385bc54e49cSTaehee Yoo 	}
1386bc54e49cSTaehee Yoo }
1387bc54e49cSTaehee Yoo 
amt_add_srcs(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,bool v6)1388bc54e49cSTaehee Yoo static void amt_add_srcs(struct amt_dev *amt, struct amt_tunnel_list *tunnel,
1389bc54e49cSTaehee Yoo 			 struct amt_group_node *gnode, void *grec,
1390bc54e49cSTaehee Yoo 			 bool v6)
1391bc54e49cSTaehee Yoo {
1392bc54e49cSTaehee Yoo 	struct igmpv3_grec *igmp_grec;
1393bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
1394b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1395b75f7095STaehee Yoo 	struct mld2_grec *mld_grec;
1396b75f7095STaehee Yoo #endif
1397bc54e49cSTaehee Yoo 	union amt_addr src = {0,};
1398bc54e49cSTaehee Yoo 	u16 nsrcs;
1399bc54e49cSTaehee Yoo 	u32 hash;
1400bc54e49cSTaehee Yoo 	int i;
1401bc54e49cSTaehee Yoo 
1402bc54e49cSTaehee Yoo 	if (!v6) {
1403d13a3205SYu Zhe 		igmp_grec = grec;
1404bc54e49cSTaehee Yoo 		nsrcs = ntohs(igmp_grec->grec_nsrcs);
1405bc54e49cSTaehee Yoo 	} else {
1406b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1407d13a3205SYu Zhe 		mld_grec = grec;
1408b75f7095STaehee Yoo 		nsrcs = ntohs(mld_grec->grec_nsrcs);
1409b75f7095STaehee Yoo #else
1410bc54e49cSTaehee Yoo 	return;
1411b75f7095STaehee Yoo #endif
1412bc54e49cSTaehee Yoo 	}
1413bc54e49cSTaehee Yoo 	for (i = 0; i < nsrcs; i++) {
1414bc54e49cSTaehee Yoo 		if (tunnel->nr_sources >= amt->max_sources)
1415bc54e49cSTaehee Yoo 			return;
1416bc54e49cSTaehee Yoo 		if (!v6)
1417bc54e49cSTaehee Yoo 			src.ip4 = igmp_grec->grec_src[i];
1418b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1419b75f7095STaehee Yoo 		else
1420b75f7095STaehee Yoo 			memcpy(&src.ip6, &mld_grec->grec_src[i],
1421b75f7095STaehee Yoo 			       sizeof(struct in6_addr));
1422b75f7095STaehee Yoo #endif
1423bc54e49cSTaehee Yoo 		if (amt_lookup_src(tunnel, gnode, AMT_FILTER_ALL, &src))
1424bc54e49cSTaehee Yoo 			continue;
1425bc54e49cSTaehee Yoo 
1426bc54e49cSTaehee Yoo 		snode = amt_alloc_snode(gnode, &src);
1427bc54e49cSTaehee Yoo 		if (snode) {
1428bc54e49cSTaehee Yoo 			hash = amt_source_hash(tunnel, &snode->source_addr);
1429bc54e49cSTaehee Yoo 			hlist_add_head_rcu(&snode->node, &gnode->sources[hash]);
1430bc54e49cSTaehee Yoo 			tunnel->nr_sources++;
1431bc54e49cSTaehee Yoo 			gnode->nr_sources++;
1432bc54e49cSTaehee Yoo 
1433bc54e49cSTaehee Yoo 			if (!gnode->v6)
1434bc54e49cSTaehee Yoo 				netdev_dbg(snode->gnode->amt->dev,
1435bc54e49cSTaehee Yoo 					   "Add source as NEW %pI4 from %pI4\n",
1436bc54e49cSTaehee Yoo 					   &snode->source_addr.ip4,
1437bc54e49cSTaehee Yoo 					   &gnode->group_addr.ip4);
1438b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1439b75f7095STaehee Yoo 			else
1440b75f7095STaehee Yoo 				netdev_dbg(snode->gnode->amt->dev,
1441b75f7095STaehee Yoo 					   "Add source as NEW %pI6 from %pI6\n",
1442b75f7095STaehee Yoo 					   &snode->source_addr.ip6,
1443b75f7095STaehee Yoo 					   &gnode->group_addr.ip6);
1444b75f7095STaehee Yoo #endif
1445bc54e49cSTaehee Yoo 		}
1446bc54e49cSTaehee Yoo 	}
1447bc54e49cSTaehee Yoo }
1448bc54e49cSTaehee Yoo 
1449bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State
1450bc54e49cSTaehee Yoo  * ------------   ------------ ----------------
1451bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  IS_IN (A)    EXCLUDE (X+A,Y-A)
1452bc54e49cSTaehee Yoo  *
1453bc54e49cSTaehee Yoo  * -----------+-----------+-----------+
1454bc54e49cSTaehee Yoo  *            |    OLD    |    NEW    |
1455bc54e49cSTaehee Yoo  * -----------+-----------+-----------+
1456bc54e49cSTaehee Yoo  *    FWD     |     X     |    X+A    |
1457bc54e49cSTaehee Yoo  * -----------+-----------+-----------+
1458bc54e49cSTaehee Yoo  *    D_FWD   |     Y     |    Y-A    |
1459bc54e49cSTaehee Yoo  * -----------+-----------+-----------+
1460bc54e49cSTaehee Yoo  *    NONE    |           |     A     |
1461bc54e49cSTaehee Yoo  * -----------+-----------+-----------+
1462bc54e49cSTaehee Yoo  *
1463bc54e49cSTaehee Yoo  * a) Received sources are NONE/NEW
1464bc54e49cSTaehee Yoo  * b) All NONE will be deleted by amt_cleanup_srcs().
1465bc54e49cSTaehee Yoo  * c) All OLD will be deleted by amt_cleanup_srcs().
1466bc54e49cSTaehee Yoo  * d) After delete, NEW source will be switched to OLD.
1467bc54e49cSTaehee Yoo  */
amt_lookup_act_srcs(struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,enum amt_ops ops,enum amt_filter filter,enum amt_act act,bool v6)1468bc54e49cSTaehee Yoo static void amt_lookup_act_srcs(struct amt_tunnel_list *tunnel,
1469bc54e49cSTaehee Yoo 				struct amt_group_node *gnode,
1470bc54e49cSTaehee Yoo 				void *grec,
1471bc54e49cSTaehee Yoo 				enum amt_ops ops,
1472bc54e49cSTaehee Yoo 				enum amt_filter filter,
1473bc54e49cSTaehee Yoo 				enum amt_act act,
1474bc54e49cSTaehee Yoo 				bool v6)
1475bc54e49cSTaehee Yoo {
1476bc54e49cSTaehee Yoo 	struct amt_dev *amt = tunnel->amt;
1477bc54e49cSTaehee Yoo 	struct amt_source_node *snode;
1478bc54e49cSTaehee Yoo 	struct igmpv3_grec *igmp_grec;
1479b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1480b75f7095STaehee Yoo 	struct mld2_grec *mld_grec;
1481b75f7095STaehee Yoo #endif
1482bc54e49cSTaehee Yoo 	union amt_addr src = {0,};
1483bc54e49cSTaehee Yoo 	struct hlist_node *t;
1484bc54e49cSTaehee Yoo 	u16 nsrcs;
1485bc54e49cSTaehee Yoo 	int i, j;
1486bc54e49cSTaehee Yoo 
1487bc54e49cSTaehee Yoo 	if (!v6) {
1488d13a3205SYu Zhe 		igmp_grec = grec;
1489bc54e49cSTaehee Yoo 		nsrcs = ntohs(igmp_grec->grec_nsrcs);
1490bc54e49cSTaehee Yoo 	} else {
1491b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1492d13a3205SYu Zhe 		mld_grec = grec;
1493b75f7095STaehee Yoo 		nsrcs = ntohs(mld_grec->grec_nsrcs);
1494b75f7095STaehee Yoo #else
1495bc54e49cSTaehee Yoo 	return;
1496b75f7095STaehee Yoo #endif
1497bc54e49cSTaehee Yoo 	}
1498bc54e49cSTaehee Yoo 
1499bc54e49cSTaehee Yoo 	memset(&src, 0, sizeof(union amt_addr));
1500bc54e49cSTaehee Yoo 	switch (ops) {
1501bc54e49cSTaehee Yoo 	case AMT_OPS_INT:
1502bc54e49cSTaehee Yoo 		/* A*B */
1503bc54e49cSTaehee Yoo 		for (i = 0; i < nsrcs; i++) {
1504bc54e49cSTaehee Yoo 			if (!v6)
1505bc54e49cSTaehee Yoo 				src.ip4 = igmp_grec->grec_src[i];
1506b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1507b75f7095STaehee Yoo 			else
1508b75f7095STaehee Yoo 				memcpy(&src.ip6, &mld_grec->grec_src[i],
1509b75f7095STaehee Yoo 				       sizeof(struct in6_addr));
1510b75f7095STaehee Yoo #endif
1511bc54e49cSTaehee Yoo 			snode = amt_lookup_src(tunnel, gnode, filter, &src);
1512bc54e49cSTaehee Yoo 			if (!snode)
1513bc54e49cSTaehee Yoo 				continue;
1514bc54e49cSTaehee Yoo 			amt_act_src(tunnel, gnode, snode, act);
1515bc54e49cSTaehee Yoo 		}
1516bc54e49cSTaehee Yoo 		break;
1517bc54e49cSTaehee Yoo 	case AMT_OPS_UNI:
1518bc54e49cSTaehee Yoo 		/* A+B */
1519bc54e49cSTaehee Yoo 		for (i = 0; i < amt->hash_buckets; i++) {
1520bc54e49cSTaehee Yoo 			hlist_for_each_entry_safe(snode, t, &gnode->sources[i],
1521bc54e49cSTaehee Yoo 						  node) {
1522bc54e49cSTaehee Yoo 				if (amt_status_filter(snode, filter))
1523bc54e49cSTaehee Yoo 					amt_act_src(tunnel, gnode, snode, act);
1524bc54e49cSTaehee Yoo 			}
1525bc54e49cSTaehee Yoo 		}
1526bc54e49cSTaehee Yoo 		for (i = 0; i < nsrcs; i++) {
1527bc54e49cSTaehee Yoo 			if (!v6)
1528bc54e49cSTaehee Yoo 				src.ip4 = igmp_grec->grec_src[i];
1529b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1530b75f7095STaehee Yoo 			else
1531b75f7095STaehee Yoo 				memcpy(&src.ip6, &mld_grec->grec_src[i],
1532b75f7095STaehee Yoo 				       sizeof(struct in6_addr));
1533b75f7095STaehee Yoo #endif
1534bc54e49cSTaehee Yoo 			snode = amt_lookup_src(tunnel, gnode, filter, &src);
1535bc54e49cSTaehee Yoo 			if (!snode)
1536bc54e49cSTaehee Yoo 				continue;
1537bc54e49cSTaehee Yoo 			amt_act_src(tunnel, gnode, snode, act);
1538bc54e49cSTaehee Yoo 		}
1539bc54e49cSTaehee Yoo 		break;
1540bc54e49cSTaehee Yoo 	case AMT_OPS_SUB:
1541bc54e49cSTaehee Yoo 		/* A-B */
1542bc54e49cSTaehee Yoo 		for (i = 0; i < amt->hash_buckets; i++) {
1543bc54e49cSTaehee Yoo 			hlist_for_each_entry_safe(snode, t, &gnode->sources[i],
1544bc54e49cSTaehee Yoo 						  node) {
1545bc54e49cSTaehee Yoo 				if (!amt_status_filter(snode, filter))
1546bc54e49cSTaehee Yoo 					continue;
1547bc54e49cSTaehee Yoo 				for (j = 0; j < nsrcs; j++) {
1548bc54e49cSTaehee Yoo 					if (!v6)
1549bc54e49cSTaehee Yoo 						src.ip4 = igmp_grec->grec_src[j];
1550b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1551b75f7095STaehee Yoo 					else
1552b75f7095STaehee Yoo 						memcpy(&src.ip6,
1553b75f7095STaehee Yoo 						       &mld_grec->grec_src[j],
1554b75f7095STaehee Yoo 						       sizeof(struct in6_addr));
1555b75f7095STaehee Yoo #endif
1556bc54e49cSTaehee Yoo 					if (amt_addr_equal(&snode->source_addr,
1557bc54e49cSTaehee Yoo 							   &src))
1558bc54e49cSTaehee Yoo 						goto out_sub;
1559bc54e49cSTaehee Yoo 				}
1560bc54e49cSTaehee Yoo 				amt_act_src(tunnel, gnode, snode, act);
1561bc54e49cSTaehee Yoo 				continue;
1562bc54e49cSTaehee Yoo out_sub:;
1563bc54e49cSTaehee Yoo 			}
1564bc54e49cSTaehee Yoo 		}
1565bc54e49cSTaehee Yoo 		break;
1566bc54e49cSTaehee Yoo 	case AMT_OPS_SUB_REV:
1567bc54e49cSTaehee Yoo 		/* B-A */
1568bc54e49cSTaehee Yoo 		for (i = 0; i < nsrcs; i++) {
1569bc54e49cSTaehee Yoo 			if (!v6)
1570bc54e49cSTaehee Yoo 				src.ip4 = igmp_grec->grec_src[i];
1571b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
1572b75f7095STaehee Yoo 			else
1573b75f7095STaehee Yoo 				memcpy(&src.ip6, &mld_grec->grec_src[i],
1574b75f7095STaehee Yoo 				       sizeof(struct in6_addr));
1575b75f7095STaehee Yoo #endif
1576bc54e49cSTaehee Yoo 			snode = amt_lookup_src(tunnel, gnode, AMT_FILTER_ALL,
1577bc54e49cSTaehee Yoo 					       &src);
1578bc54e49cSTaehee Yoo 			if (!snode) {
1579bc54e49cSTaehee Yoo 				snode = amt_lookup_src(tunnel, gnode,
1580bc54e49cSTaehee Yoo 						       filter, &src);
1581bc54e49cSTaehee Yoo 				if (snode)
1582bc54e49cSTaehee Yoo 					amt_act_src(tunnel, gnode, snode, act);
1583bc54e49cSTaehee Yoo 			}
1584bc54e49cSTaehee Yoo 		}
1585bc54e49cSTaehee Yoo 		break;
1586bc54e49cSTaehee Yoo 	default:
1587bc54e49cSTaehee Yoo 		netdev_dbg(amt->dev, "Invalid type\n");
1588bc54e49cSTaehee Yoo 		return;
1589bc54e49cSTaehee Yoo 	}
1590bc54e49cSTaehee Yoo }
1591bc54e49cSTaehee Yoo 
amt_mcast_is_in_handler(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,void * zero_grec,bool v6)1592bc54e49cSTaehee Yoo static void amt_mcast_is_in_handler(struct amt_dev *amt,
1593bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel,
1594bc54e49cSTaehee Yoo 				    struct amt_group_node *gnode,
1595bc54e49cSTaehee Yoo 				    void *grec, void *zero_grec, bool v6)
1596bc54e49cSTaehee Yoo {
1597bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
1598bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1599bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1600bc54e49cSTaehee Yoo  * INCLUDE (A)    IS_IN (B)    INCLUDE (A+B)           (B)=GMI
1601bc54e49cSTaehee Yoo  */
1602bc54e49cSTaehee Yoo 		/* Update IS_IN (B) as FWD/NEW */
1603bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1604bc54e49cSTaehee Yoo 				    AMT_FILTER_NONE_NEW,
1605bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1606bc54e49cSTaehee Yoo 				    v6);
1607bc54e49cSTaehee Yoo 		/* Update INCLUDE (A) as NEW */
1608bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1609bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1610bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1611bc54e49cSTaehee Yoo 				    v6);
1612bc54e49cSTaehee Yoo 		/* (B)=GMI */
1613bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1614bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD_NEW,
1615bc54e49cSTaehee Yoo 				    AMT_ACT_GMI,
1616bc54e49cSTaehee Yoo 				    v6);
1617bc54e49cSTaehee Yoo 	} else {
1618bc54e49cSTaehee Yoo /* State        Actions
1619bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1620bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  IS_IN (A)    EXCLUDE (X+A,Y-A)       (A)=GMI
1621bc54e49cSTaehee Yoo  */
1622bc54e49cSTaehee Yoo 		/* Update (A) in (X, Y) as NONE/NEW */
1623bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1624bc54e49cSTaehee Yoo 				    AMT_FILTER_BOTH,
1625bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_NONE_NEW,
1626bc54e49cSTaehee Yoo 				    v6);
1627bc54e49cSTaehee Yoo 		/* Update FWD/OLD as FWD/NEW */
1628bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, zero_grec, AMT_OPS_UNI,
1629bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1630bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1631bc54e49cSTaehee Yoo 				    v6);
1632bc54e49cSTaehee Yoo 		/* Update IS_IN (A) as FWD/NEW */
1633bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1634bc54e49cSTaehee Yoo 				    AMT_FILTER_NONE_NEW,
1635bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1636bc54e49cSTaehee Yoo 				    v6);
1637bc54e49cSTaehee Yoo 		/* Update EXCLUDE (, Y-A) as D_FWD_NEW */
1638bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB,
1639bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1640bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1641bc54e49cSTaehee Yoo 				    v6);
1642bc54e49cSTaehee Yoo 	}
1643bc54e49cSTaehee Yoo }
1644bc54e49cSTaehee Yoo 
amt_mcast_is_ex_handler(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,void * zero_grec,bool v6)1645bc54e49cSTaehee Yoo static void amt_mcast_is_ex_handler(struct amt_dev *amt,
1646bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel,
1647bc54e49cSTaehee Yoo 				    struct amt_group_node *gnode,
1648bc54e49cSTaehee Yoo 				    void *grec, void *zero_grec, bool v6)
1649bc54e49cSTaehee Yoo {
1650bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
1651bc54e49cSTaehee Yoo /* Router State   Report Rec'd  New Router State         Actions
1652bc54e49cSTaehee Yoo  * ------------   ------------  ----------------         -------
1653bc54e49cSTaehee Yoo  * INCLUDE (A)    IS_EX (B)     EXCLUDE (A*B,B-A)        (B-A)=0
1654bc54e49cSTaehee Yoo  *                                                       Delete (A-B)
1655bc54e49cSTaehee Yoo  *                                                       Group Timer=GMI
1656bc54e49cSTaehee Yoo  */
1657bc54e49cSTaehee Yoo 		/* EXCLUDE(A*B, ) */
1658bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1659bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1660bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1661bc54e49cSTaehee Yoo 				    v6);
1662bc54e49cSTaehee Yoo 		/* EXCLUDE(, B-A) */
1663bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1664bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1665bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1666bc54e49cSTaehee Yoo 				    v6);
1667bc54e49cSTaehee Yoo 		/* (B-A)=0 */
1668bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, zero_grec, AMT_OPS_UNI,
1669bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD_NEW,
1670bc54e49cSTaehee Yoo 				    AMT_ACT_GMI_ZERO,
1671bc54e49cSTaehee Yoo 				    v6);
1672bc54e49cSTaehee Yoo 		/* Group Timer=GMI */
1673bc54e49cSTaehee Yoo 		if (!mod_delayed_work(amt_wq, &gnode->group_timer,
1674bc54e49cSTaehee Yoo 				      msecs_to_jiffies(amt_gmi(amt))))
1675bc54e49cSTaehee Yoo 			dev_hold(amt->dev);
1676bc54e49cSTaehee Yoo 		gnode->filter_mode = MCAST_EXCLUDE;
1677bc54e49cSTaehee Yoo 		/* Delete (A-B) will be worked by amt_cleanup_srcs(). */
1678bc54e49cSTaehee Yoo 	} else {
1679bc54e49cSTaehee Yoo /* Router State   Report Rec'd  New Router State	Actions
1680bc54e49cSTaehee Yoo  * ------------   ------------  ----------------	-------
1681bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  IS_EX (A)     EXCLUDE (A-Y,Y*A)	(A-X-Y)=GMI
1682bc54e49cSTaehee Yoo  *							Delete (X-A)
1683bc54e49cSTaehee Yoo  *							Delete (Y-A)
1684bc54e49cSTaehee Yoo  *							Group Timer=GMI
1685bc54e49cSTaehee Yoo  */
1686bc54e49cSTaehee Yoo 		/* EXCLUDE (A-Y, ) */
1687bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1688bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1689bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1690bc54e49cSTaehee Yoo 				    v6);
1691bc54e49cSTaehee Yoo 		/* EXCLUDE (, Y*A ) */
1692bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1693bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1694bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1695bc54e49cSTaehee Yoo 				    v6);
1696bc54e49cSTaehee Yoo 		/* (A-X-Y)=GMI */
1697bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1698bc54e49cSTaehee Yoo 				    AMT_FILTER_BOTH_NEW,
1699bc54e49cSTaehee Yoo 				    AMT_ACT_GMI,
1700bc54e49cSTaehee Yoo 				    v6);
1701bc54e49cSTaehee Yoo 		/* Group Timer=GMI */
1702bc54e49cSTaehee Yoo 		if (!mod_delayed_work(amt_wq, &gnode->group_timer,
1703bc54e49cSTaehee Yoo 				      msecs_to_jiffies(amt_gmi(amt))))
1704bc54e49cSTaehee Yoo 			dev_hold(amt->dev);
1705bc54e49cSTaehee Yoo 		/* Delete (X-A), (Y-A) will be worked by amt_cleanup_srcs(). */
1706bc54e49cSTaehee Yoo 	}
1707bc54e49cSTaehee Yoo }
1708bc54e49cSTaehee Yoo 
amt_mcast_to_in_handler(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,void * zero_grec,bool v6)1709bc54e49cSTaehee Yoo static void amt_mcast_to_in_handler(struct amt_dev *amt,
1710bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel,
1711bc54e49cSTaehee Yoo 				    struct amt_group_node *gnode,
1712bc54e49cSTaehee Yoo 				    void *grec, void *zero_grec, bool v6)
1713bc54e49cSTaehee Yoo {
1714bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
1715bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1716bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1717bc54e49cSTaehee Yoo  * INCLUDE (A)    TO_IN (B)    INCLUDE (A+B)           (B)=GMI
1718bc54e49cSTaehee Yoo  *						       Send Q(G,A-B)
1719bc54e49cSTaehee Yoo  */
1720bc54e49cSTaehee Yoo 		/* Update TO_IN (B) sources as FWD/NEW */
1721bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1722bc54e49cSTaehee Yoo 				    AMT_FILTER_NONE_NEW,
1723bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1724bc54e49cSTaehee Yoo 				    v6);
1725bc54e49cSTaehee Yoo 		/* Update INCLUDE (A) sources as NEW */
1726bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1727bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1728bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1729bc54e49cSTaehee Yoo 				    v6);
1730bc54e49cSTaehee Yoo 		/* (B)=GMI */
1731bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1732bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD_NEW,
1733bc54e49cSTaehee Yoo 				    AMT_ACT_GMI,
1734bc54e49cSTaehee Yoo 				    v6);
1735bc54e49cSTaehee Yoo 	} else {
1736bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1737bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1738bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  TO_IN (A)    EXCLUDE (X+A,Y-A)       (A)=GMI
1739bc54e49cSTaehee Yoo  *						       Send Q(G,X-A)
1740bc54e49cSTaehee Yoo  *						       Send Q(G)
1741bc54e49cSTaehee Yoo  */
1742bc54e49cSTaehee Yoo 		/* Update TO_IN (A) sources as FWD/NEW */
1743bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1744bc54e49cSTaehee Yoo 				    AMT_FILTER_NONE_NEW,
1745bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1746bc54e49cSTaehee Yoo 				    v6);
1747bc54e49cSTaehee Yoo 		/* Update EXCLUDE(X,) sources as FWD/NEW */
1748bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1749bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1750bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1751bc54e49cSTaehee Yoo 				    v6);
1752bc54e49cSTaehee Yoo 		/* EXCLUDE (, Y-A)
1753bc54e49cSTaehee Yoo 		 * (A) are already switched to FWD_NEW.
1754bc54e49cSTaehee Yoo 		 * So, D_FWD/OLD -> D_FWD/NEW is okay.
1755bc54e49cSTaehee Yoo 		 */
1756bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, zero_grec, AMT_OPS_UNI,
1757bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1758bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1759bc54e49cSTaehee Yoo 				    v6);
1760bc54e49cSTaehee Yoo 		/* (A)=GMI
1761bc54e49cSTaehee Yoo 		 * Only FWD_NEW will have (A) sources.
1762bc54e49cSTaehee Yoo 		 */
1763bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1764bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD_NEW,
1765bc54e49cSTaehee Yoo 				    AMT_ACT_GMI,
1766bc54e49cSTaehee Yoo 				    v6);
1767bc54e49cSTaehee Yoo 	}
1768bc54e49cSTaehee Yoo }
1769bc54e49cSTaehee Yoo 
amt_mcast_to_ex_handler(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,void * zero_grec,bool v6)1770bc54e49cSTaehee Yoo static void amt_mcast_to_ex_handler(struct amt_dev *amt,
1771bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel,
1772bc54e49cSTaehee Yoo 				    struct amt_group_node *gnode,
1773bc54e49cSTaehee Yoo 				    void *grec, void *zero_grec, bool v6)
1774bc54e49cSTaehee Yoo {
1775bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
1776bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1777bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1778bc54e49cSTaehee Yoo  * INCLUDE (A)    TO_EX (B)    EXCLUDE (A*B,B-A)       (B-A)=0
1779bc54e49cSTaehee Yoo  *						       Delete (A-B)
1780bc54e49cSTaehee Yoo  *						       Send Q(G,A*B)
1781bc54e49cSTaehee Yoo  *						       Group Timer=GMI
1782bc54e49cSTaehee Yoo  */
1783bc54e49cSTaehee Yoo 		/* EXCLUDE (A*B, ) */
1784bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1785bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1786bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1787bc54e49cSTaehee Yoo 				    v6);
1788bc54e49cSTaehee Yoo 		/* EXCLUDE (, B-A) */
1789bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1790bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1791bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1792bc54e49cSTaehee Yoo 				    v6);
1793bc54e49cSTaehee Yoo 		/* (B-A)=0 */
1794bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, zero_grec, AMT_OPS_UNI,
1795bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD_NEW,
1796bc54e49cSTaehee Yoo 				    AMT_ACT_GMI_ZERO,
1797bc54e49cSTaehee Yoo 				    v6);
1798bc54e49cSTaehee Yoo 		/* Group Timer=GMI */
1799bc54e49cSTaehee Yoo 		if (!mod_delayed_work(amt_wq, &gnode->group_timer,
1800bc54e49cSTaehee Yoo 				      msecs_to_jiffies(amt_gmi(amt))))
1801bc54e49cSTaehee Yoo 			dev_hold(amt->dev);
1802bc54e49cSTaehee Yoo 		gnode->filter_mode = MCAST_EXCLUDE;
1803bc54e49cSTaehee Yoo 		/* Delete (A-B) will be worked by amt_cleanup_srcs(). */
1804bc54e49cSTaehee Yoo 	} else {
1805bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1806bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1807bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  TO_EX (A)    EXCLUDE (A-Y,Y*A)       (A-X-Y)=Group Timer
1808bc54e49cSTaehee Yoo  *						       Delete (X-A)
1809bc54e49cSTaehee Yoo  *						       Delete (Y-A)
1810bc54e49cSTaehee Yoo  *						       Send Q(G,A-Y)
1811bc54e49cSTaehee Yoo  *						       Group Timer=GMI
1812bc54e49cSTaehee Yoo  */
1813bc54e49cSTaehee Yoo 		/* Update (A-X-Y) as NONE/OLD */
1814bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1815bc54e49cSTaehee Yoo 				    AMT_FILTER_BOTH,
1816bc54e49cSTaehee Yoo 				    AMT_ACT_GT,
1817bc54e49cSTaehee Yoo 				    v6);
1818bc54e49cSTaehee Yoo 		/* EXCLUDE (A-Y, ) */
1819bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1820bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1821bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1822bc54e49cSTaehee Yoo 				    v6);
1823bc54e49cSTaehee Yoo 		/* EXCLUDE (, Y*A) */
1824bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1825bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1826bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1827bc54e49cSTaehee Yoo 				    v6);
1828bc54e49cSTaehee Yoo 		/* Group Timer=GMI */
1829bc54e49cSTaehee Yoo 		if (!mod_delayed_work(amt_wq, &gnode->group_timer,
1830bc54e49cSTaehee Yoo 				      msecs_to_jiffies(amt_gmi(amt))))
1831bc54e49cSTaehee Yoo 			dev_hold(amt->dev);
1832bc54e49cSTaehee Yoo 		/* Delete (X-A), (Y-A) will be worked by amt_cleanup_srcs(). */
1833bc54e49cSTaehee Yoo 	}
1834bc54e49cSTaehee Yoo }
1835bc54e49cSTaehee Yoo 
amt_mcast_allow_handler(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,void * zero_grec,bool v6)1836bc54e49cSTaehee Yoo static void amt_mcast_allow_handler(struct amt_dev *amt,
1837bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel,
1838bc54e49cSTaehee Yoo 				    struct amt_group_node *gnode,
1839bc54e49cSTaehee Yoo 				    void *grec, void *zero_grec, bool v6)
1840bc54e49cSTaehee Yoo {
1841bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
1842bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1843bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1844bc54e49cSTaehee Yoo  * INCLUDE (A)    ALLOW (B)    INCLUDE (A+B)	       (B)=GMI
1845bc54e49cSTaehee Yoo  */
1846bc54e49cSTaehee Yoo 		/* INCLUDE (A+B) */
1847bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1848bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1849bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1850bc54e49cSTaehee Yoo 				    v6);
1851bc54e49cSTaehee Yoo 		/* (B)=GMI */
1852bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1853bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD_NEW,
1854bc54e49cSTaehee Yoo 				    AMT_ACT_GMI,
1855bc54e49cSTaehee Yoo 				    v6);
1856bc54e49cSTaehee Yoo 	} else {
1857bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1858bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1859bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  ALLOW (A)    EXCLUDE (X+A,Y-A)       (A)=GMI
1860bc54e49cSTaehee Yoo  */
1861bc54e49cSTaehee Yoo 		/* EXCLUDE (X+A, ) */
1862bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1863bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1864bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1865bc54e49cSTaehee Yoo 				    v6);
1866bc54e49cSTaehee Yoo 		/* EXCLUDE (, Y-A) */
1867bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB,
1868bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1869bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1870bc54e49cSTaehee Yoo 				    v6);
1871bc54e49cSTaehee Yoo 		/* (A)=GMI
1872bc54e49cSTaehee Yoo 		 * All (A) source are now FWD/NEW status.
1873bc54e49cSTaehee Yoo 		 */
1874bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_INT,
1875bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD_NEW,
1876bc54e49cSTaehee Yoo 				    AMT_ACT_GMI,
1877bc54e49cSTaehee Yoo 				    v6);
1878bc54e49cSTaehee Yoo 	}
1879bc54e49cSTaehee Yoo }
1880bc54e49cSTaehee Yoo 
amt_mcast_block_handler(struct amt_dev * amt,struct amt_tunnel_list * tunnel,struct amt_group_node * gnode,void * grec,void * zero_grec,bool v6)1881bc54e49cSTaehee Yoo static void amt_mcast_block_handler(struct amt_dev *amt,
1882bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel,
1883bc54e49cSTaehee Yoo 				    struct amt_group_node *gnode,
1884bc54e49cSTaehee Yoo 				    void *grec, void *zero_grec, bool v6)
1885bc54e49cSTaehee Yoo {
1886bc54e49cSTaehee Yoo 	if (gnode->filter_mode == MCAST_INCLUDE) {
1887bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1888bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1889bc54e49cSTaehee Yoo  * INCLUDE (A)    BLOCK (B)    INCLUDE (A)             Send Q(G,A*B)
1890bc54e49cSTaehee Yoo  */
1891bc54e49cSTaehee Yoo 		/* INCLUDE (A) */
1892bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, zero_grec, AMT_OPS_UNI,
1893bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1894bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1895bc54e49cSTaehee Yoo 				    v6);
1896bc54e49cSTaehee Yoo 	} else {
1897bc54e49cSTaehee Yoo /* Router State   Report Rec'd New Router State        Actions
1898bc54e49cSTaehee Yoo  * ------------   ------------ ----------------        -------
1899bc54e49cSTaehee Yoo  * EXCLUDE (X,Y)  BLOCK (A)    EXCLUDE (X+(A-Y),Y)     (A-X-Y)=Group Timer
1900bc54e49cSTaehee Yoo  *						       Send Q(G,A-Y)
1901bc54e49cSTaehee Yoo  */
1902bc54e49cSTaehee Yoo 		/* (A-X-Y)=Group Timer */
1903bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1904bc54e49cSTaehee Yoo 				    AMT_FILTER_BOTH,
1905bc54e49cSTaehee Yoo 				    AMT_ACT_GT,
1906bc54e49cSTaehee Yoo 				    v6);
1907bc54e49cSTaehee Yoo 		/* EXCLUDE (X, ) */
1908bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1909bc54e49cSTaehee Yoo 				    AMT_FILTER_FWD,
1910bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1911bc54e49cSTaehee Yoo 				    v6);
1912bc54e49cSTaehee Yoo 		/* EXCLUDE (X+(A-Y) */
1913bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_SUB_REV,
1914bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1915bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_FWD_NEW,
1916bc54e49cSTaehee Yoo 				    v6);
1917bc54e49cSTaehee Yoo 		/* EXCLUDE (, Y) */
1918bc54e49cSTaehee Yoo 		amt_lookup_act_srcs(tunnel, gnode, grec, AMT_OPS_UNI,
1919bc54e49cSTaehee Yoo 				    AMT_FILTER_D_FWD,
1920bc54e49cSTaehee Yoo 				    AMT_ACT_STATUS_D_FWD_NEW,
1921bc54e49cSTaehee Yoo 				    v6);
1922bc54e49cSTaehee Yoo 	}
1923bc54e49cSTaehee Yoo }
1924bc54e49cSTaehee Yoo 
1925bc54e49cSTaehee Yoo /* RFC 3376
1926bc54e49cSTaehee Yoo  * 7.3.2. In the Presence of Older Version Group Members
1927bc54e49cSTaehee Yoo  *
1928bc54e49cSTaehee Yoo  * When Group Compatibility Mode is IGMPv2, a router internally
1929bc54e49cSTaehee Yoo  * translates the following IGMPv2 messages for that group to their
1930bc54e49cSTaehee Yoo  * IGMPv3 equivalents:
1931bc54e49cSTaehee Yoo  *
1932bc54e49cSTaehee Yoo  * IGMPv2 Message                IGMPv3 Equivalent
1933bc54e49cSTaehee Yoo  * --------------                -----------------
1934bc54e49cSTaehee Yoo  * Report                        IS_EX( {} )
1935bc54e49cSTaehee Yoo  * Leave                         TO_IN( {} )
1936bc54e49cSTaehee Yoo  */
amt_igmpv2_report_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)1937bc54e49cSTaehee Yoo static void amt_igmpv2_report_handler(struct amt_dev *amt, struct sk_buff *skb,
1938bc54e49cSTaehee Yoo 				      struct amt_tunnel_list *tunnel)
1939bc54e49cSTaehee Yoo {
1940bc54e49cSTaehee Yoo 	struct igmphdr *ih = igmp_hdr(skb);
1941bc54e49cSTaehee Yoo 	struct iphdr *iph = ip_hdr(skb);
1942bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
1943bc54e49cSTaehee Yoo 	union amt_addr group, host;
1944bc54e49cSTaehee Yoo 
1945bc54e49cSTaehee Yoo 	memset(&group, 0, sizeof(union amt_addr));
1946bc54e49cSTaehee Yoo 	group.ip4 = ih->group;
1947bc54e49cSTaehee Yoo 	memset(&host, 0, sizeof(union amt_addr));
1948bc54e49cSTaehee Yoo 	host.ip4 = iph->saddr;
1949bc54e49cSTaehee Yoo 
1950bc54e49cSTaehee Yoo 	gnode = amt_lookup_group(tunnel, &group, &host, false);
1951bc54e49cSTaehee Yoo 	if (!gnode) {
1952bc54e49cSTaehee Yoo 		gnode = amt_add_group(amt, tunnel, &group, &host, false);
1953bc54e49cSTaehee Yoo 		if (!IS_ERR(gnode)) {
1954bc54e49cSTaehee Yoo 			gnode->filter_mode = MCAST_EXCLUDE;
1955bc54e49cSTaehee Yoo 			if (!mod_delayed_work(amt_wq, &gnode->group_timer,
1956bc54e49cSTaehee Yoo 					      msecs_to_jiffies(amt_gmi(amt))))
1957bc54e49cSTaehee Yoo 				dev_hold(amt->dev);
1958bc54e49cSTaehee Yoo 		}
1959bc54e49cSTaehee Yoo 	}
1960bc54e49cSTaehee Yoo }
1961bc54e49cSTaehee Yoo 
1962bc54e49cSTaehee Yoo /* RFC 3376
1963bc54e49cSTaehee Yoo  * 7.3.2. In the Presence of Older Version Group Members
1964bc54e49cSTaehee Yoo  *
1965bc54e49cSTaehee Yoo  * When Group Compatibility Mode is IGMPv2, a router internally
1966bc54e49cSTaehee Yoo  * translates the following IGMPv2 messages for that group to their
1967bc54e49cSTaehee Yoo  * IGMPv3 equivalents:
1968bc54e49cSTaehee Yoo  *
1969bc54e49cSTaehee Yoo  * IGMPv2 Message                IGMPv3 Equivalent
1970bc54e49cSTaehee Yoo  * --------------                -----------------
1971bc54e49cSTaehee Yoo  * Report                        IS_EX( {} )
1972bc54e49cSTaehee Yoo  * Leave                         TO_IN( {} )
1973bc54e49cSTaehee Yoo  */
amt_igmpv2_leave_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)1974bc54e49cSTaehee Yoo static void amt_igmpv2_leave_handler(struct amt_dev *amt, struct sk_buff *skb,
1975bc54e49cSTaehee Yoo 				     struct amt_tunnel_list *tunnel)
1976bc54e49cSTaehee Yoo {
1977bc54e49cSTaehee Yoo 	struct igmphdr *ih = igmp_hdr(skb);
1978bc54e49cSTaehee Yoo 	struct iphdr *iph = ip_hdr(skb);
1979bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
1980bc54e49cSTaehee Yoo 	union amt_addr group, host;
1981bc54e49cSTaehee Yoo 
1982bc54e49cSTaehee Yoo 	memset(&group, 0, sizeof(union amt_addr));
1983bc54e49cSTaehee Yoo 	group.ip4 = ih->group;
1984bc54e49cSTaehee Yoo 	memset(&host, 0, sizeof(union amt_addr));
1985bc54e49cSTaehee Yoo 	host.ip4 = iph->saddr;
1986bc54e49cSTaehee Yoo 
1987bc54e49cSTaehee Yoo 	gnode = amt_lookup_group(tunnel, &group, &host, false);
1988bc54e49cSTaehee Yoo 	if (gnode)
1989bc54e49cSTaehee Yoo 		amt_del_group(amt, gnode);
1990bc54e49cSTaehee Yoo }
1991bc54e49cSTaehee Yoo 
amt_igmpv3_report_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)1992bc54e49cSTaehee Yoo static void amt_igmpv3_report_handler(struct amt_dev *amt, struct sk_buff *skb,
1993bc54e49cSTaehee Yoo 				      struct amt_tunnel_list *tunnel)
1994bc54e49cSTaehee Yoo {
1995bc54e49cSTaehee Yoo 	struct igmpv3_report *ihrv3 = igmpv3_report_hdr(skb);
1996bc54e49cSTaehee Yoo 	int len = skb_transport_offset(skb) + sizeof(*ihrv3);
1997bc54e49cSTaehee Yoo 	void *zero_grec = (void *)&igmpv3_zero_grec;
1998bc54e49cSTaehee Yoo 	struct iphdr *iph = ip_hdr(skb);
1999bc54e49cSTaehee Yoo 	struct amt_group_node *gnode;
2000bc54e49cSTaehee Yoo 	union amt_addr group, host;
2001bc54e49cSTaehee Yoo 	struct igmpv3_grec *grec;
2002bc54e49cSTaehee Yoo 	u16 nsrcs;
2003bc54e49cSTaehee Yoo 	int i;
2004bc54e49cSTaehee Yoo 
2005bc54e49cSTaehee Yoo 	for (i = 0; i < ntohs(ihrv3->ngrec); i++) {
2006bc54e49cSTaehee Yoo 		len += sizeof(*grec);
2007bc54e49cSTaehee Yoo 		if (!ip_mc_may_pull(skb, len))
2008bc54e49cSTaehee Yoo 			break;
2009bc54e49cSTaehee Yoo 
2010bc54e49cSTaehee Yoo 		grec = (void *)(skb->data + len - sizeof(*grec));
2011bc54e49cSTaehee Yoo 		nsrcs = ntohs(grec->grec_nsrcs);
2012bc54e49cSTaehee Yoo 
2013bc54e49cSTaehee Yoo 		len += nsrcs * sizeof(__be32);
2014bc54e49cSTaehee Yoo 		if (!ip_mc_may_pull(skb, len))
2015bc54e49cSTaehee Yoo 			break;
2016bc54e49cSTaehee Yoo 
2017bc54e49cSTaehee Yoo 		memset(&group, 0, sizeof(union amt_addr));
2018bc54e49cSTaehee Yoo 		group.ip4 = grec->grec_mca;
2019bc54e49cSTaehee Yoo 		memset(&host, 0, sizeof(union amt_addr));
2020bc54e49cSTaehee Yoo 		host.ip4 = iph->saddr;
2021bc54e49cSTaehee Yoo 		gnode = amt_lookup_group(tunnel, &group, &host, false);
2022bc54e49cSTaehee Yoo 		if (!gnode) {
2023bc54e49cSTaehee Yoo 			gnode = amt_add_group(amt, tunnel, &group, &host,
2024bc54e49cSTaehee Yoo 					      false);
2025bc54e49cSTaehee Yoo 			if (IS_ERR(gnode))
2026bc54e49cSTaehee Yoo 				continue;
2027bc54e49cSTaehee Yoo 		}
2028bc54e49cSTaehee Yoo 
2029bc54e49cSTaehee Yoo 		amt_add_srcs(amt, tunnel, gnode, grec, false);
2030bc54e49cSTaehee Yoo 		switch (grec->grec_type) {
2031bc54e49cSTaehee Yoo 		case IGMPV3_MODE_IS_INCLUDE:
2032bc54e49cSTaehee Yoo 			amt_mcast_is_in_handler(amt, tunnel, gnode, grec,
2033bc54e49cSTaehee Yoo 						zero_grec, false);
2034bc54e49cSTaehee Yoo 			break;
2035bc54e49cSTaehee Yoo 		case IGMPV3_MODE_IS_EXCLUDE:
2036bc54e49cSTaehee Yoo 			amt_mcast_is_ex_handler(amt, tunnel, gnode, grec,
2037bc54e49cSTaehee Yoo 						zero_grec, false);
2038bc54e49cSTaehee Yoo 			break;
2039bc54e49cSTaehee Yoo 		case IGMPV3_CHANGE_TO_INCLUDE:
2040bc54e49cSTaehee Yoo 			amt_mcast_to_in_handler(amt, tunnel, gnode, grec,
2041bc54e49cSTaehee Yoo 						zero_grec, false);
2042bc54e49cSTaehee Yoo 			break;
2043bc54e49cSTaehee Yoo 		case IGMPV3_CHANGE_TO_EXCLUDE:
2044bc54e49cSTaehee Yoo 			amt_mcast_to_ex_handler(amt, tunnel, gnode, grec,
2045bc54e49cSTaehee Yoo 						zero_grec, false);
2046bc54e49cSTaehee Yoo 			break;
2047bc54e49cSTaehee Yoo 		case IGMPV3_ALLOW_NEW_SOURCES:
2048bc54e49cSTaehee Yoo 			amt_mcast_allow_handler(amt, tunnel, gnode, grec,
2049bc54e49cSTaehee Yoo 						zero_grec, false);
2050bc54e49cSTaehee Yoo 			break;
2051bc54e49cSTaehee Yoo 		case IGMPV3_BLOCK_OLD_SOURCES:
2052bc54e49cSTaehee Yoo 			amt_mcast_block_handler(amt, tunnel, gnode, grec,
2053bc54e49cSTaehee Yoo 						zero_grec, false);
2054bc54e49cSTaehee Yoo 			break;
2055bc54e49cSTaehee Yoo 		default:
2056bc54e49cSTaehee Yoo 			break;
2057bc54e49cSTaehee Yoo 		}
2058bc54e49cSTaehee Yoo 		amt_cleanup_srcs(amt, tunnel, gnode);
2059bc54e49cSTaehee Yoo 	}
2060bc54e49cSTaehee Yoo }
2061bc54e49cSTaehee Yoo 
2062bc54e49cSTaehee Yoo /* caller held tunnel->lock */
amt_igmp_report_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)2063bc54e49cSTaehee Yoo static void amt_igmp_report_handler(struct amt_dev *amt, struct sk_buff *skb,
2064bc54e49cSTaehee Yoo 				    struct amt_tunnel_list *tunnel)
2065bc54e49cSTaehee Yoo {
2066bc54e49cSTaehee Yoo 	struct igmphdr *ih = igmp_hdr(skb);
2067bc54e49cSTaehee Yoo 
2068bc54e49cSTaehee Yoo 	switch (ih->type) {
2069bc54e49cSTaehee Yoo 	case IGMPV3_HOST_MEMBERSHIP_REPORT:
2070bc54e49cSTaehee Yoo 		amt_igmpv3_report_handler(amt, skb, tunnel);
2071bc54e49cSTaehee Yoo 		break;
2072bc54e49cSTaehee Yoo 	case IGMPV2_HOST_MEMBERSHIP_REPORT:
2073bc54e49cSTaehee Yoo 		amt_igmpv2_report_handler(amt, skb, tunnel);
2074bc54e49cSTaehee Yoo 		break;
2075bc54e49cSTaehee Yoo 	case IGMP_HOST_LEAVE_MESSAGE:
2076bc54e49cSTaehee Yoo 		amt_igmpv2_leave_handler(amt, skb, tunnel);
2077bc54e49cSTaehee Yoo 		break;
2078bc54e49cSTaehee Yoo 	default:
2079bc54e49cSTaehee Yoo 		break;
2080bc54e49cSTaehee Yoo 	}
2081bc54e49cSTaehee Yoo }
2082bc54e49cSTaehee Yoo 
2083b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
2084b75f7095STaehee Yoo /* RFC 3810
2085b75f7095STaehee Yoo  * 8.3.2. In the Presence of MLDv1 Multicast Address Listeners
2086b75f7095STaehee Yoo  *
2087b75f7095STaehee Yoo  * When Multicast Address Compatibility Mode is MLDv2, a router acts
2088b75f7095STaehee Yoo  * using the MLDv2 protocol for that multicast address.  When Multicast
2089b75f7095STaehee Yoo  * Address Compatibility Mode is MLDv1, a router internally translates
2090b75f7095STaehee Yoo  * the following MLDv1 messages for that multicast address to their
2091b75f7095STaehee Yoo  * MLDv2 equivalents:
2092b75f7095STaehee Yoo  *
2093b75f7095STaehee Yoo  * MLDv1 Message                 MLDv2 Equivalent
2094b75f7095STaehee Yoo  * --------------                -----------------
2095b75f7095STaehee Yoo  * Report                        IS_EX( {} )
2096b75f7095STaehee Yoo  * Done                          TO_IN( {} )
2097b75f7095STaehee Yoo  */
amt_mldv1_report_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)2098b75f7095STaehee Yoo static void amt_mldv1_report_handler(struct amt_dev *amt, struct sk_buff *skb,
2099b75f7095STaehee Yoo 				     struct amt_tunnel_list *tunnel)
2100b75f7095STaehee Yoo {
2101b75f7095STaehee Yoo 	struct mld_msg *mld = (struct mld_msg *)icmp6_hdr(skb);
2102b75f7095STaehee Yoo 	struct ipv6hdr *ip6h = ipv6_hdr(skb);
2103b75f7095STaehee Yoo 	struct amt_group_node *gnode;
2104b75f7095STaehee Yoo 	union amt_addr group, host;
2105b75f7095STaehee Yoo 
2106b75f7095STaehee Yoo 	memcpy(&group.ip6, &mld->mld_mca, sizeof(struct in6_addr));
2107b75f7095STaehee Yoo 	memcpy(&host.ip6, &ip6h->saddr, sizeof(struct in6_addr));
2108b75f7095STaehee Yoo 
2109b75f7095STaehee Yoo 	gnode = amt_lookup_group(tunnel, &group, &host, true);
2110b75f7095STaehee Yoo 	if (!gnode) {
2111b75f7095STaehee Yoo 		gnode = amt_add_group(amt, tunnel, &group, &host, true);
2112b75f7095STaehee Yoo 		if (!IS_ERR(gnode)) {
2113b75f7095STaehee Yoo 			gnode->filter_mode = MCAST_EXCLUDE;
2114b75f7095STaehee Yoo 			if (!mod_delayed_work(amt_wq, &gnode->group_timer,
2115b75f7095STaehee Yoo 					      msecs_to_jiffies(amt_gmi(amt))))
2116b75f7095STaehee Yoo 				dev_hold(amt->dev);
2117b75f7095STaehee Yoo 		}
2118b75f7095STaehee Yoo 	}
2119b75f7095STaehee Yoo }
2120b75f7095STaehee Yoo 
2121b75f7095STaehee Yoo /* RFC 3810
2122b75f7095STaehee Yoo  * 8.3.2. In the Presence of MLDv1 Multicast Address Listeners
2123b75f7095STaehee Yoo  *
2124b75f7095STaehee Yoo  * When Multicast Address Compatibility Mode is MLDv2, a router acts
2125b75f7095STaehee Yoo  * using the MLDv2 protocol for that multicast address.  When Multicast
2126b75f7095STaehee Yoo  * Address Compatibility Mode is MLDv1, a router internally translates
2127b75f7095STaehee Yoo  * the following MLDv1 messages for that multicast address to their
2128b75f7095STaehee Yoo  * MLDv2 equivalents:
2129b75f7095STaehee Yoo  *
2130b75f7095STaehee Yoo  * MLDv1 Message                 MLDv2 Equivalent
2131b75f7095STaehee Yoo  * --------------                -----------------
2132b75f7095STaehee Yoo  * Report                        IS_EX( {} )
2133b75f7095STaehee Yoo  * Done                          TO_IN( {} )
2134b75f7095STaehee Yoo  */
amt_mldv1_leave_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)2135b75f7095STaehee Yoo static void amt_mldv1_leave_handler(struct amt_dev *amt, struct sk_buff *skb,
2136b75f7095STaehee Yoo 				    struct amt_tunnel_list *tunnel)
2137b75f7095STaehee Yoo {
2138b75f7095STaehee Yoo 	struct mld_msg *mld = (struct mld_msg *)icmp6_hdr(skb);
2139b75f7095STaehee Yoo 	struct iphdr *iph = ip_hdr(skb);
2140b75f7095STaehee Yoo 	struct amt_group_node *gnode;
2141b75f7095STaehee Yoo 	union amt_addr group, host;
2142b75f7095STaehee Yoo 
2143b75f7095STaehee Yoo 	memcpy(&group.ip6, &mld->mld_mca, sizeof(struct in6_addr));
2144b75f7095STaehee Yoo 	memset(&host, 0, sizeof(union amt_addr));
2145b75f7095STaehee Yoo 	host.ip4 = iph->saddr;
2146b75f7095STaehee Yoo 
2147b75f7095STaehee Yoo 	gnode = amt_lookup_group(tunnel, &group, &host, true);
2148b75f7095STaehee Yoo 	if (gnode) {
2149b75f7095STaehee Yoo 		amt_del_group(amt, gnode);
2150b75f7095STaehee Yoo 		return;
2151b75f7095STaehee Yoo 	}
2152b75f7095STaehee Yoo }
2153b75f7095STaehee Yoo 
amt_mldv2_report_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)2154b75f7095STaehee Yoo static void amt_mldv2_report_handler(struct amt_dev *amt, struct sk_buff *skb,
2155b75f7095STaehee Yoo 				     struct amt_tunnel_list *tunnel)
2156b75f7095STaehee Yoo {
2157b75f7095STaehee Yoo 	struct mld2_report *mld2r = (struct mld2_report *)icmp6_hdr(skb);
2158b75f7095STaehee Yoo 	int len = skb_transport_offset(skb) + sizeof(*mld2r);
2159b75f7095STaehee Yoo 	void *zero_grec = (void *)&mldv2_zero_grec;
2160b75f7095STaehee Yoo 	struct ipv6hdr *ip6h = ipv6_hdr(skb);
2161b75f7095STaehee Yoo 	struct amt_group_node *gnode;
2162b75f7095STaehee Yoo 	union amt_addr group, host;
2163b75f7095STaehee Yoo 	struct mld2_grec *grec;
2164b75f7095STaehee Yoo 	u16 nsrcs;
2165b75f7095STaehee Yoo 	int i;
2166b75f7095STaehee Yoo 
2167b75f7095STaehee Yoo 	for (i = 0; i < ntohs(mld2r->mld2r_ngrec); i++) {
2168b75f7095STaehee Yoo 		len += sizeof(*grec);
2169b75f7095STaehee Yoo 		if (!ipv6_mc_may_pull(skb, len))
2170b75f7095STaehee Yoo 			break;
2171b75f7095STaehee Yoo 
2172b75f7095STaehee Yoo 		grec = (void *)(skb->data + len - sizeof(*grec));
2173b75f7095STaehee Yoo 		nsrcs = ntohs(grec->grec_nsrcs);
2174b75f7095STaehee Yoo 
2175b75f7095STaehee Yoo 		len += nsrcs * sizeof(struct in6_addr);
2176b75f7095STaehee Yoo 		if (!ipv6_mc_may_pull(skb, len))
2177b75f7095STaehee Yoo 			break;
2178b75f7095STaehee Yoo 
2179b75f7095STaehee Yoo 		memset(&group, 0, sizeof(union amt_addr));
2180b75f7095STaehee Yoo 		group.ip6 = grec->grec_mca;
2181b75f7095STaehee Yoo 		memset(&host, 0, sizeof(union amt_addr));
2182b75f7095STaehee Yoo 		host.ip6 = ip6h->saddr;
2183b75f7095STaehee Yoo 		gnode = amt_lookup_group(tunnel, &group, &host, true);
2184b75f7095STaehee Yoo 		if (!gnode) {
2185b75f7095STaehee Yoo 			gnode = amt_add_group(amt, tunnel, &group, &host,
2186b75f7095STaehee Yoo 					      ETH_P_IPV6);
2187b75f7095STaehee Yoo 			if (IS_ERR(gnode))
2188b75f7095STaehee Yoo 				continue;
2189b75f7095STaehee Yoo 		}
2190b75f7095STaehee Yoo 
2191b75f7095STaehee Yoo 		amt_add_srcs(amt, tunnel, gnode, grec, true);
2192b75f7095STaehee Yoo 		switch (grec->grec_type) {
2193b75f7095STaehee Yoo 		case MLD2_MODE_IS_INCLUDE:
2194b75f7095STaehee Yoo 			amt_mcast_is_in_handler(amt, tunnel, gnode, grec,
2195b75f7095STaehee Yoo 						zero_grec, true);
2196b75f7095STaehee Yoo 			break;
2197b75f7095STaehee Yoo 		case MLD2_MODE_IS_EXCLUDE:
2198b75f7095STaehee Yoo 			amt_mcast_is_ex_handler(amt, tunnel, gnode, grec,
2199b75f7095STaehee Yoo 						zero_grec, true);
2200b75f7095STaehee Yoo 			break;
2201b75f7095STaehee Yoo 		case MLD2_CHANGE_TO_INCLUDE:
2202b75f7095STaehee Yoo 			amt_mcast_to_in_handler(amt, tunnel, gnode, grec,
2203b75f7095STaehee Yoo 						zero_grec, true);
2204b75f7095STaehee Yoo 			break;
2205b75f7095STaehee Yoo 		case MLD2_CHANGE_TO_EXCLUDE:
2206b75f7095STaehee Yoo 			amt_mcast_to_ex_handler(amt, tunnel, gnode, grec,
2207b75f7095STaehee Yoo 						zero_grec, true);
2208b75f7095STaehee Yoo 			break;
2209b75f7095STaehee Yoo 		case MLD2_ALLOW_NEW_SOURCES:
2210b75f7095STaehee Yoo 			amt_mcast_allow_handler(amt, tunnel, gnode, grec,
2211b75f7095STaehee Yoo 						zero_grec, true);
2212b75f7095STaehee Yoo 			break;
2213b75f7095STaehee Yoo 		case MLD2_BLOCK_OLD_SOURCES:
2214b75f7095STaehee Yoo 			amt_mcast_block_handler(amt, tunnel, gnode, grec,
2215b75f7095STaehee Yoo 						zero_grec, true);
2216b75f7095STaehee Yoo 			break;
2217b75f7095STaehee Yoo 		default:
2218b75f7095STaehee Yoo 			break;
2219b75f7095STaehee Yoo 		}
2220b75f7095STaehee Yoo 		amt_cleanup_srcs(amt, tunnel, gnode);
2221b75f7095STaehee Yoo 	}
2222b75f7095STaehee Yoo }
2223b75f7095STaehee Yoo 
2224b75f7095STaehee Yoo /* caller held tunnel->lock */
amt_mld_report_handler(struct amt_dev * amt,struct sk_buff * skb,struct amt_tunnel_list * tunnel)2225b75f7095STaehee Yoo static void amt_mld_report_handler(struct amt_dev *amt, struct sk_buff *skb,
2226b75f7095STaehee Yoo 				   struct amt_tunnel_list *tunnel)
2227b75f7095STaehee Yoo {
2228b75f7095STaehee Yoo 	struct mld_msg *mld = (struct mld_msg *)icmp6_hdr(skb);
2229b75f7095STaehee Yoo 
2230b75f7095STaehee Yoo 	switch (mld->mld_type) {
2231b75f7095STaehee Yoo 	case ICMPV6_MGM_REPORT:
2232b75f7095STaehee Yoo 		amt_mldv1_report_handler(amt, skb, tunnel);
2233b75f7095STaehee Yoo 		break;
2234b75f7095STaehee Yoo 	case ICMPV6_MLD2_REPORT:
2235b75f7095STaehee Yoo 		amt_mldv2_report_handler(amt, skb, tunnel);
2236b75f7095STaehee Yoo 		break;
2237b75f7095STaehee Yoo 	case ICMPV6_MGM_REDUCTION:
2238b75f7095STaehee Yoo 		amt_mldv1_leave_handler(amt, skb, tunnel);
2239b75f7095STaehee Yoo 		break;
2240b75f7095STaehee Yoo 	default:
2241b75f7095STaehee Yoo 		break;
2242b75f7095STaehee Yoo 	}
2243b75f7095STaehee Yoo }
2244b75f7095STaehee Yoo #endif
2245b75f7095STaehee Yoo 
amt_advertisement_handler(struct amt_dev * amt,struct sk_buff * skb)2246cbc21dc1STaehee Yoo static bool amt_advertisement_handler(struct amt_dev *amt, struct sk_buff *skb)
2247cbc21dc1STaehee Yoo {
2248cbc21dc1STaehee Yoo 	struct amt_header_advertisement *amta;
2249cbc21dc1STaehee Yoo 	int hdr_size;
2250cbc21dc1STaehee Yoo 
2251f55a0707STaehee Yoo 	hdr_size = sizeof(*amta) + sizeof(struct udphdr);
2252cbc21dc1STaehee Yoo 	if (!pskb_may_pull(skb, hdr_size))
2253cbc21dc1STaehee Yoo 		return true;
2254cbc21dc1STaehee Yoo 
2255cbc21dc1STaehee Yoo 	amta = (struct amt_header_advertisement *)(udp_hdr(skb) + 1);
2256cbc21dc1STaehee Yoo 	if (!amta->ip4)
2257cbc21dc1STaehee Yoo 		return true;
2258cbc21dc1STaehee Yoo 
2259cbc21dc1STaehee Yoo 	if (amta->reserved || amta->version)
2260cbc21dc1STaehee Yoo 		return true;
2261cbc21dc1STaehee Yoo 
2262cbc21dc1STaehee Yoo 	if (ipv4_is_loopback(amta->ip4) || ipv4_is_multicast(amta->ip4) ||
2263cbc21dc1STaehee Yoo 	    ipv4_is_zeronet(amta->ip4))
2264cbc21dc1STaehee Yoo 		return true;
2265cbc21dc1STaehee Yoo 
226640185f35STaehee Yoo 	if (amt->status != AMT_STATUS_SENT_DISCOVERY ||
226740185f35STaehee Yoo 	    amt->nonce != amta->nonce)
226840185f35STaehee Yoo 		return true;
226940185f35STaehee Yoo 
2270cbc21dc1STaehee Yoo 	amt->remote_ip = amta->ip4;
2271cbc21dc1STaehee Yoo 	netdev_dbg(amt->dev, "advertised remote ip = %pI4\n", &amt->remote_ip);
2272cbc21dc1STaehee Yoo 	mod_delayed_work(amt_wq, &amt->req_wq, 0);
2273cbc21dc1STaehee Yoo 
2274cbc21dc1STaehee Yoo 	amt_update_gw_status(amt, AMT_STATUS_RECEIVED_ADVERTISEMENT, true);
2275cbc21dc1STaehee Yoo 	return false;
2276cbc21dc1STaehee Yoo }
2277cbc21dc1STaehee Yoo 
amt_multicast_data_handler(struct amt_dev * amt,struct sk_buff * skb)2278cbc21dc1STaehee Yoo static bool amt_multicast_data_handler(struct amt_dev *amt, struct sk_buff *skb)
2279cbc21dc1STaehee Yoo {
2280cbc21dc1STaehee Yoo 	struct amt_header_mcast_data *amtmd;
2281cbc21dc1STaehee Yoo 	int hdr_size, len, err;
2282cbc21dc1STaehee Yoo 	struct ethhdr *eth;
2283cbc21dc1STaehee Yoo 	struct iphdr *iph;
2284cbc21dc1STaehee Yoo 
2285e882827dSTaehee Yoo 	if (READ_ONCE(amt->status) != AMT_STATUS_SENT_UPDATE)
2286e882827dSTaehee Yoo 		return true;
2287e882827dSTaehee Yoo 
2288f55a0707STaehee Yoo 	hdr_size = sizeof(*amtmd) + sizeof(struct udphdr);
2289f55a0707STaehee Yoo 	if (!pskb_may_pull(skb, hdr_size))
2290f55a0707STaehee Yoo 		return true;
2291f55a0707STaehee Yoo 
2292cbc21dc1STaehee Yoo 	amtmd = (struct amt_header_mcast_data *)(udp_hdr(skb) + 1);
2293cbc21dc1STaehee Yoo 	if (amtmd->reserved || amtmd->version)
2294cbc21dc1STaehee Yoo 		return true;
2295cbc21dc1STaehee Yoo 
2296cbc21dc1STaehee Yoo 	if (iptunnel_pull_header(skb, hdr_size, htons(ETH_P_IP), false))
2297cbc21dc1STaehee Yoo 		return true;
2298f55a0707STaehee Yoo 
2299cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
2300cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*eth));
2301cbc21dc1STaehee Yoo 	skb_reset_mac_header(skb);
2302cbc21dc1STaehee Yoo 	skb_pull(skb, sizeof(*eth));
2303cbc21dc1STaehee Yoo 	eth = eth_hdr(skb);
2304f55a0707STaehee Yoo 
2305f55a0707STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(*iph)))
2306f55a0707STaehee Yoo 		return true;
2307cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2308f55a0707STaehee Yoo 
2309cbc21dc1STaehee Yoo 	if (iph->version == 4) {
2310cbc21dc1STaehee Yoo 		if (!ipv4_is_multicast(iph->daddr))
2311cbc21dc1STaehee Yoo 			return true;
2312cbc21dc1STaehee Yoo 		skb->protocol = htons(ETH_P_IP);
2313cbc21dc1STaehee Yoo 		eth->h_proto = htons(ETH_P_IP);
2314cbc21dc1STaehee Yoo 		ip_eth_mc_map(iph->daddr, eth->h_dest);
2315b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
2316b75f7095STaehee Yoo 	} else if (iph->version == 6) {
2317b75f7095STaehee Yoo 		struct ipv6hdr *ip6h;
2318b75f7095STaehee Yoo 
2319f55a0707STaehee Yoo 		if (!pskb_may_pull(skb, sizeof(*ip6h)))
2320f55a0707STaehee Yoo 			return true;
2321f55a0707STaehee Yoo 
2322b75f7095STaehee Yoo 		ip6h = ipv6_hdr(skb);
2323b75f7095STaehee Yoo 		if (!ipv6_addr_is_multicast(&ip6h->daddr))
2324b75f7095STaehee Yoo 			return true;
2325b75f7095STaehee Yoo 		skb->protocol = htons(ETH_P_IPV6);
2326b75f7095STaehee Yoo 		eth->h_proto = htons(ETH_P_IPV6);
2327b75f7095STaehee Yoo 		ipv6_eth_mc_map(&ip6h->daddr, eth->h_dest);
2328b75f7095STaehee Yoo #endif
2329cbc21dc1STaehee Yoo 	} else {
2330cbc21dc1STaehee Yoo 		return true;
2331cbc21dc1STaehee Yoo 	}
2332cbc21dc1STaehee Yoo 
2333cbc21dc1STaehee Yoo 	skb->pkt_type = PACKET_MULTICAST;
2334cbc21dc1STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
2335cbc21dc1STaehee Yoo 	len = skb->len;
2336cbc21dc1STaehee Yoo 	err = gro_cells_receive(&amt->gro_cells, skb);
2337cbc21dc1STaehee Yoo 	if (likely(err == NET_RX_SUCCESS))
2338cbc21dc1STaehee Yoo 		dev_sw_netstats_rx_add(amt->dev, len);
2339cbc21dc1STaehee Yoo 	else
2340cbc21dc1STaehee Yoo 		amt->dev->stats.rx_dropped++;
2341cbc21dc1STaehee Yoo 
2342cbc21dc1STaehee Yoo 	return false;
2343cbc21dc1STaehee Yoo }
2344cbc21dc1STaehee Yoo 
amt_membership_query_handler(struct amt_dev * amt,struct sk_buff * skb)2345cbc21dc1STaehee Yoo static bool amt_membership_query_handler(struct amt_dev *amt,
2346cbc21dc1STaehee Yoo 					 struct sk_buff *skb)
2347cbc21dc1STaehee Yoo {
2348cbc21dc1STaehee Yoo 	struct amt_header_membership_query *amtmq;
2349cbc21dc1STaehee Yoo 	struct igmpv3_query *ihv3;
2350cbc21dc1STaehee Yoo 	struct ethhdr *eth, *oeth;
2351cbc21dc1STaehee Yoo 	struct iphdr *iph;
2352cbc21dc1STaehee Yoo 	int hdr_size, len;
2353cbc21dc1STaehee Yoo 
2354f55a0707STaehee Yoo 	hdr_size = sizeof(*amtmq) + sizeof(struct udphdr);
2355cbc21dc1STaehee Yoo 	if (!pskb_may_pull(skb, hdr_size))
2356cbc21dc1STaehee Yoo 		return true;
2357cbc21dc1STaehee Yoo 
2358cbc21dc1STaehee Yoo 	amtmq = (struct amt_header_membership_query *)(udp_hdr(skb) + 1);
2359cbc21dc1STaehee Yoo 	if (amtmq->reserved || amtmq->version)
2360cbc21dc1STaehee Yoo 		return true;
2361cbc21dc1STaehee Yoo 
2362239d8866STaehee Yoo 	if (amtmq->nonce != amt->nonce)
2363239d8866STaehee Yoo 		return true;
2364239d8866STaehee Yoo 
2365f55a0707STaehee Yoo 	hdr_size -= sizeof(*eth);
2366cbc21dc1STaehee Yoo 	if (iptunnel_pull_header(skb, hdr_size, htons(ETH_P_TEB), false))
2367cbc21dc1STaehee Yoo 		return true;
2368f55a0707STaehee Yoo 
2369cbc21dc1STaehee Yoo 	oeth = eth_hdr(skb);
2370cbc21dc1STaehee Yoo 	skb_reset_mac_header(skb);
2371cbc21dc1STaehee Yoo 	skb_pull(skb, sizeof(*eth));
2372cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
2373cbc21dc1STaehee Yoo 	eth = eth_hdr(skb);
2374f55a0707STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(*iph)))
2375f55a0707STaehee Yoo 		return true;
2376f55a0707STaehee Yoo 
2377cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2378cbc21dc1STaehee Yoo 	if (iph->version == 4) {
2379239d8866STaehee Yoo 		if (READ_ONCE(amt->ready4))
2380239d8866STaehee Yoo 			return true;
2381239d8866STaehee Yoo 
2382cbc21dc1STaehee Yoo 		if (!pskb_may_pull(skb, sizeof(*iph) + AMT_IPHDR_OPTS +
2383cbc21dc1STaehee Yoo 				   sizeof(*ihv3)))
2384cbc21dc1STaehee Yoo 			return true;
2385cbc21dc1STaehee Yoo 
2386f55a0707STaehee Yoo 		if (!ipv4_is_multicast(iph->daddr))
2387f55a0707STaehee Yoo 			return true;
2388f55a0707STaehee Yoo 
2389cbc21dc1STaehee Yoo 		ihv3 = skb_pull(skb, sizeof(*iph) + AMT_IPHDR_OPTS);
2390cbc21dc1STaehee Yoo 		skb_reset_transport_header(skb);
2391cbc21dc1STaehee Yoo 		skb_push(skb, sizeof(*iph) + AMT_IPHDR_OPTS);
2392928f353cSTaehee Yoo 		WRITE_ONCE(amt->ready4, true);
2393cbc21dc1STaehee Yoo 		amt->mac = amtmq->response_mac;
2394cbc21dc1STaehee Yoo 		amt->req_cnt = 0;
2395cbc21dc1STaehee Yoo 		amt->qi = ihv3->qqic;
2396cbc21dc1STaehee Yoo 		skb->protocol = htons(ETH_P_IP);
2397cbc21dc1STaehee Yoo 		eth->h_proto = htons(ETH_P_IP);
2398cbc21dc1STaehee Yoo 		ip_eth_mc_map(iph->daddr, eth->h_dest);
2399b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
2400b75f7095STaehee Yoo 	} else if (iph->version == 6) {
2401b75f7095STaehee Yoo 		struct mld2_query *mld2q;
2402f55a0707STaehee Yoo 		struct ipv6hdr *ip6h;
2403b75f7095STaehee Yoo 
2404239d8866STaehee Yoo 		if (READ_ONCE(amt->ready6))
2405239d8866STaehee Yoo 			return true;
2406239d8866STaehee Yoo 
2407b75f7095STaehee Yoo 		if (!pskb_may_pull(skb, sizeof(*ip6h) + AMT_IP6HDR_OPTS +
2408b75f7095STaehee Yoo 				   sizeof(*mld2q)))
2409b75f7095STaehee Yoo 			return true;
2410b75f7095STaehee Yoo 
2411f55a0707STaehee Yoo 		ip6h = ipv6_hdr(skb);
2412f55a0707STaehee Yoo 		if (!ipv6_addr_is_multicast(&ip6h->daddr))
2413f55a0707STaehee Yoo 			return true;
2414f55a0707STaehee Yoo 
2415b75f7095STaehee Yoo 		mld2q = skb_pull(skb, sizeof(*ip6h) + AMT_IP6HDR_OPTS);
2416b75f7095STaehee Yoo 		skb_reset_transport_header(skb);
2417b75f7095STaehee Yoo 		skb_push(skb, sizeof(*ip6h) + AMT_IP6HDR_OPTS);
2418928f353cSTaehee Yoo 		WRITE_ONCE(amt->ready6, true);
2419b75f7095STaehee Yoo 		amt->mac = amtmq->response_mac;
2420b75f7095STaehee Yoo 		amt->req_cnt = 0;
2421b75f7095STaehee Yoo 		amt->qi = mld2q->mld2q_qqic;
2422b75f7095STaehee Yoo 		skb->protocol = htons(ETH_P_IPV6);
2423b75f7095STaehee Yoo 		eth->h_proto = htons(ETH_P_IPV6);
2424b75f7095STaehee Yoo 		ipv6_eth_mc_map(&ip6h->daddr, eth->h_dest);
2425b75f7095STaehee Yoo #endif
2426cbc21dc1STaehee Yoo 	} else {
2427cbc21dc1STaehee Yoo 		return true;
2428cbc21dc1STaehee Yoo 	}
2429cbc21dc1STaehee Yoo 
2430cbc21dc1STaehee Yoo 	ether_addr_copy(eth->h_source, oeth->h_source);
2431cbc21dc1STaehee Yoo 	skb->pkt_type = PACKET_MULTICAST;
2432cbc21dc1STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
2433cbc21dc1STaehee Yoo 	len = skb->len;
243430e22a6eSTaehee Yoo 	local_bh_disable();
2435baebdf48SSebastian Andrzej Siewior 	if (__netif_rx(skb) == NET_RX_SUCCESS) {
2436cbc21dc1STaehee Yoo 		amt_update_gw_status(amt, AMT_STATUS_RECEIVED_QUERY, true);
2437cbc21dc1STaehee Yoo 		dev_sw_netstats_rx_add(amt->dev, len);
2438cbc21dc1STaehee Yoo 	} else {
2439cbc21dc1STaehee Yoo 		amt->dev->stats.rx_dropped++;
2440cbc21dc1STaehee Yoo 	}
244130e22a6eSTaehee Yoo 	local_bh_enable();
2442cbc21dc1STaehee Yoo 
2443cbc21dc1STaehee Yoo 	return false;
2444cbc21dc1STaehee Yoo }
2445cbc21dc1STaehee Yoo 
amt_update_handler(struct amt_dev * amt,struct sk_buff * skb)2446cbc21dc1STaehee Yoo static bool amt_update_handler(struct amt_dev *amt, struct sk_buff *skb)
2447cbc21dc1STaehee Yoo {
2448cbc21dc1STaehee Yoo 	struct amt_header_membership_update *amtmu;
2449cbc21dc1STaehee Yoo 	struct amt_tunnel_list *tunnel;
2450cbc21dc1STaehee Yoo 	struct ethhdr *eth;
2451cbc21dc1STaehee Yoo 	struct iphdr *iph;
2452f55a0707STaehee Yoo 	int len, hdr_size;
2453cbc21dc1STaehee Yoo 
2454cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2455cbc21dc1STaehee Yoo 
2456f55a0707STaehee Yoo 	hdr_size = sizeof(*amtmu) + sizeof(struct udphdr);
2457f55a0707STaehee Yoo 	if (!pskb_may_pull(skb, hdr_size))
2458cbc21dc1STaehee Yoo 		return true;
2459cbc21dc1STaehee Yoo 
2460f55a0707STaehee Yoo 	amtmu = (struct amt_header_membership_update *)(udp_hdr(skb) + 1);
2461cbc21dc1STaehee Yoo 	if (amtmu->reserved || amtmu->version)
2462cbc21dc1STaehee Yoo 		return true;
2463cbc21dc1STaehee Yoo 
2464f55a0707STaehee Yoo 	if (iptunnel_pull_header(skb, hdr_size, skb->protocol, false))
2465f55a0707STaehee Yoo 		return true;
2466f55a0707STaehee Yoo 
2467cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
2468cbc21dc1STaehee Yoo 
2469cbc21dc1STaehee Yoo 	list_for_each_entry_rcu(tunnel, &amt->tunnel_list, list) {
2470cbc21dc1STaehee Yoo 		if (tunnel->ip4 == iph->saddr) {
2471cbc21dc1STaehee Yoo 			if ((amtmu->nonce == tunnel->nonce &&
2472cbc21dc1STaehee Yoo 			     amtmu->response_mac == tunnel->mac)) {
2473cbc21dc1STaehee Yoo 				mod_delayed_work(amt_wq, &tunnel->gc_wq,
2474cbc21dc1STaehee Yoo 						 msecs_to_jiffies(amt_gmi(amt))
2475cbc21dc1STaehee Yoo 								  * 3);
2476cbc21dc1STaehee Yoo 				goto report;
2477cbc21dc1STaehee Yoo 			} else {
2478cbc21dc1STaehee Yoo 				netdev_dbg(amt->dev, "Invalid MAC\n");
2479cbc21dc1STaehee Yoo 				return true;
2480cbc21dc1STaehee Yoo 			}
2481cbc21dc1STaehee Yoo 		}
2482cbc21dc1STaehee Yoo 	}
2483cbc21dc1STaehee Yoo 
2484ac1dbf55STaehee Yoo 	return true;
2485cbc21dc1STaehee Yoo 
2486cbc21dc1STaehee Yoo report:
2487f55a0707STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(*iph)))
2488f55a0707STaehee Yoo 		return true;
2489f55a0707STaehee Yoo 
2490cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2491cbc21dc1STaehee Yoo 	if (iph->version == 4) {
2492cbc21dc1STaehee Yoo 		if (ip_mc_check_igmp(skb)) {
2493cbc21dc1STaehee Yoo 			netdev_dbg(amt->dev, "Invalid IGMP\n");
2494cbc21dc1STaehee Yoo 			return true;
2495cbc21dc1STaehee Yoo 		}
2496cbc21dc1STaehee Yoo 
2497bc54e49cSTaehee Yoo 		spin_lock_bh(&tunnel->lock);
2498bc54e49cSTaehee Yoo 		amt_igmp_report_handler(amt, skb, tunnel);
2499bc54e49cSTaehee Yoo 		spin_unlock_bh(&tunnel->lock);
2500bc54e49cSTaehee Yoo 
2501cbc21dc1STaehee Yoo 		skb_push(skb, sizeof(struct ethhdr));
2502cbc21dc1STaehee Yoo 		skb_reset_mac_header(skb);
2503cbc21dc1STaehee Yoo 		eth = eth_hdr(skb);
2504cbc21dc1STaehee Yoo 		skb->protocol = htons(ETH_P_IP);
2505cbc21dc1STaehee Yoo 		eth->h_proto = htons(ETH_P_IP);
2506cbc21dc1STaehee Yoo 		ip_eth_mc_map(iph->daddr, eth->h_dest);
2507b75f7095STaehee Yoo #if IS_ENABLED(CONFIG_IPV6)
2508b75f7095STaehee Yoo 	} else if (iph->version == 6) {
2509b75f7095STaehee Yoo 		struct ipv6hdr *ip6h = ipv6_hdr(skb);
2510b75f7095STaehee Yoo 
2511b75f7095STaehee Yoo 		if (ipv6_mc_check_mld(skb)) {
2512b75f7095STaehee Yoo 			netdev_dbg(amt->dev, "Invalid MLD\n");
2513b75f7095STaehee Yoo 			return true;
2514b75f7095STaehee Yoo 		}
2515b75f7095STaehee Yoo 
2516b75f7095STaehee Yoo 		spin_lock_bh(&tunnel->lock);
2517b75f7095STaehee Yoo 		amt_mld_report_handler(amt, skb, tunnel);
2518b75f7095STaehee Yoo 		spin_unlock_bh(&tunnel->lock);
2519b75f7095STaehee Yoo 
2520b75f7095STaehee Yoo 		skb_push(skb, sizeof(struct ethhdr));
2521b75f7095STaehee Yoo 		skb_reset_mac_header(skb);
2522b75f7095STaehee Yoo 		eth = eth_hdr(skb);
2523b75f7095STaehee Yoo 		skb->protocol = htons(ETH_P_IPV6);
2524b75f7095STaehee Yoo 		eth->h_proto = htons(ETH_P_IPV6);
2525b75f7095STaehee Yoo 		ipv6_eth_mc_map(&ip6h->daddr, eth->h_dest);
2526b75f7095STaehee Yoo #endif
2527cbc21dc1STaehee Yoo 	} else {
2528cbc21dc1STaehee Yoo 		netdev_dbg(amt->dev, "Unsupported Protocol\n");
2529cbc21dc1STaehee Yoo 		return true;
2530cbc21dc1STaehee Yoo 	}
2531cbc21dc1STaehee Yoo 
2532cbc21dc1STaehee Yoo 	skb_pull(skb, sizeof(struct ethhdr));
2533cbc21dc1STaehee Yoo 	skb->pkt_type = PACKET_MULTICAST;
2534cbc21dc1STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
2535cbc21dc1STaehee Yoo 	len = skb->len;
2536baebdf48SSebastian Andrzej Siewior 	if (__netif_rx(skb) == NET_RX_SUCCESS) {
2537cbc21dc1STaehee Yoo 		amt_update_relay_status(tunnel, AMT_STATUS_RECEIVED_UPDATE,
2538cbc21dc1STaehee Yoo 					true);
2539cbc21dc1STaehee Yoo 		dev_sw_netstats_rx_add(amt->dev, len);
2540cbc21dc1STaehee Yoo 	} else {
2541cbc21dc1STaehee Yoo 		amt->dev->stats.rx_dropped++;
2542cbc21dc1STaehee Yoo 	}
2543cbc21dc1STaehee Yoo 
2544cbc21dc1STaehee Yoo 	return false;
2545cbc21dc1STaehee Yoo }
2546cbc21dc1STaehee Yoo 
amt_send_advertisement(struct amt_dev * amt,__be32 nonce,__be32 daddr,__be16 dport)2547cbc21dc1STaehee Yoo static void amt_send_advertisement(struct amt_dev *amt, __be32 nonce,
2548cbc21dc1STaehee Yoo 				   __be32 daddr, __be16 dport)
2549cbc21dc1STaehee Yoo {
2550cbc21dc1STaehee Yoo 	struct amt_header_advertisement *amta;
2551cbc21dc1STaehee Yoo 	int hlen, tlen, offset;
2552cbc21dc1STaehee Yoo 	struct socket *sock;
2553cbc21dc1STaehee Yoo 	struct udphdr *udph;
2554cbc21dc1STaehee Yoo 	struct sk_buff *skb;
2555cbc21dc1STaehee Yoo 	struct iphdr *iph;
2556cbc21dc1STaehee Yoo 	struct rtable *rt;
2557cbc21dc1STaehee Yoo 	struct flowi4 fl4;
2558cbc21dc1STaehee Yoo 	u32 len;
2559cbc21dc1STaehee Yoo 	int err;
2560cbc21dc1STaehee Yoo 
2561cbc21dc1STaehee Yoo 	rcu_read_lock();
2562cbc21dc1STaehee Yoo 	sock = rcu_dereference(amt->sock);
2563cbc21dc1STaehee Yoo 	if (!sock)
2564cbc21dc1STaehee Yoo 		goto out;
2565cbc21dc1STaehee Yoo 
2566cbc21dc1STaehee Yoo 	if (!netif_running(amt->stream_dev) || !netif_running(amt->dev))
2567cbc21dc1STaehee Yoo 		goto out;
2568cbc21dc1STaehee Yoo 
2569cbc21dc1STaehee Yoo 	rt = ip_route_output_ports(amt->net, &fl4, sock->sk,
2570cbc21dc1STaehee Yoo 				   daddr, amt->local_ip,
2571cbc21dc1STaehee Yoo 				   dport, amt->relay_port,
2572cbc21dc1STaehee Yoo 				   IPPROTO_UDP, 0,
2573cbc21dc1STaehee Yoo 				   amt->stream_dev->ifindex);
2574cbc21dc1STaehee Yoo 	if (IS_ERR(rt)) {
2575cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
2576cbc21dc1STaehee Yoo 		goto out;
2577cbc21dc1STaehee Yoo 	}
2578cbc21dc1STaehee Yoo 
2579cbc21dc1STaehee Yoo 	hlen = LL_RESERVED_SPACE(amt->dev);
2580cbc21dc1STaehee Yoo 	tlen = amt->dev->needed_tailroom;
2581cbc21dc1STaehee Yoo 	len = hlen + tlen + sizeof(*iph) + sizeof(*udph) + sizeof(*amta);
2582cbc21dc1STaehee Yoo 	skb = netdev_alloc_skb_ip_align(amt->dev, len);
2583cbc21dc1STaehee Yoo 	if (!skb) {
2584cbc21dc1STaehee Yoo 		ip_rt_put(rt);
2585cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
2586cbc21dc1STaehee Yoo 		goto out;
2587cbc21dc1STaehee Yoo 	}
2588cbc21dc1STaehee Yoo 
2589cbc21dc1STaehee Yoo 	skb->priority = TC_PRIO_CONTROL;
2590cbc21dc1STaehee Yoo 	skb_dst_set(skb, &rt->dst);
2591cbc21dc1STaehee Yoo 
2592cbc21dc1STaehee Yoo 	len = sizeof(*iph) + sizeof(*udph) + sizeof(*amta);
2593cbc21dc1STaehee Yoo 	skb_reset_network_header(skb);
2594cbc21dc1STaehee Yoo 	skb_put(skb, len);
2595cbc21dc1STaehee Yoo 	amta = skb_pull(skb, sizeof(*iph) + sizeof(*udph));
2596cbc21dc1STaehee Yoo 	amta->version	= 0;
2597cbc21dc1STaehee Yoo 	amta->type	= AMT_MSG_ADVERTISEMENT;
2598cbc21dc1STaehee Yoo 	amta->reserved	= 0;
2599cbc21dc1STaehee Yoo 	amta->nonce	= nonce;
2600cbc21dc1STaehee Yoo 	amta->ip4	= amt->local_ip;
2601cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*udph));
2602cbc21dc1STaehee Yoo 	skb_reset_transport_header(skb);
2603cbc21dc1STaehee Yoo 	udph		= udp_hdr(skb);
2604cbc21dc1STaehee Yoo 	udph->source	= amt->relay_port;
2605cbc21dc1STaehee Yoo 	udph->dest	= dport;
2606cbc21dc1STaehee Yoo 	udph->len	= htons(sizeof(*amta) + sizeof(*udph));
2607cbc21dc1STaehee Yoo 	udph->check	= 0;
2608cbc21dc1STaehee Yoo 	offset = skb_transport_offset(skb);
2609cbc21dc1STaehee Yoo 	skb->csum = skb_checksum(skb, offset, skb->len - offset, 0);
2610cbc21dc1STaehee Yoo 	udph->check = csum_tcpudp_magic(amt->local_ip, daddr,
2611cbc21dc1STaehee Yoo 					sizeof(*udph) + sizeof(*amta),
2612cbc21dc1STaehee Yoo 					IPPROTO_UDP, skb->csum);
2613cbc21dc1STaehee Yoo 
2614cbc21dc1STaehee Yoo 	skb_push(skb, sizeof(*iph));
2615cbc21dc1STaehee Yoo 	iph		= ip_hdr(skb);
2616cbc21dc1STaehee Yoo 	iph->version	= 4;
2617cbc21dc1STaehee Yoo 	iph->ihl	= (sizeof(struct iphdr)) >> 2;
2618cbc21dc1STaehee Yoo 	iph->tos	= AMT_TOS;
2619cbc21dc1STaehee Yoo 	iph->frag_off	= 0;
2620cbc21dc1STaehee Yoo 	iph->ttl	= ip4_dst_hoplimit(&rt->dst);
2621cbc21dc1STaehee Yoo 	iph->daddr	= daddr;
2622cbc21dc1STaehee Yoo 	iph->saddr	= amt->local_ip;
2623cbc21dc1STaehee Yoo 	iph->protocol	= IPPROTO_UDP;
2624cbc21dc1STaehee Yoo 	iph->tot_len	= htons(len);
2625cbc21dc1STaehee Yoo 
2626cbc21dc1STaehee Yoo 	skb->ip_summed = CHECKSUM_NONE;
2627cbc21dc1STaehee Yoo 	ip_select_ident(amt->net, skb, NULL);
2628cbc21dc1STaehee Yoo 	ip_send_check(iph);
2629cbc21dc1STaehee Yoo 	err = ip_local_out(amt->net, sock->sk, skb);
2630cbc21dc1STaehee Yoo 	if (unlikely(net_xmit_eval(err)))
2631cbc21dc1STaehee Yoo 		amt->dev->stats.tx_errors++;
2632cbc21dc1STaehee Yoo 
2633cbc21dc1STaehee Yoo out:
2634cbc21dc1STaehee Yoo 	rcu_read_unlock();
2635cbc21dc1STaehee Yoo }
2636cbc21dc1STaehee Yoo 
amt_discovery_handler(struct amt_dev * amt,struct sk_buff * skb)2637cbc21dc1STaehee Yoo static bool amt_discovery_handler(struct amt_dev *amt, struct sk_buff *skb)
2638cbc21dc1STaehee Yoo {
2639cbc21dc1STaehee Yoo 	struct amt_header_discovery *amtd;
2640cbc21dc1STaehee Yoo 	struct udphdr *udph;
2641cbc21dc1STaehee Yoo 	struct iphdr *iph;
2642cbc21dc1STaehee Yoo 
2643cbc21dc1STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(*udph) + sizeof(*amtd)))
2644cbc21dc1STaehee Yoo 		return true;
2645cbc21dc1STaehee Yoo 
2646cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2647cbc21dc1STaehee Yoo 	udph = udp_hdr(skb);
2648cbc21dc1STaehee Yoo 	amtd = (struct amt_header_discovery *)(udp_hdr(skb) + 1);
2649cbc21dc1STaehee Yoo 
2650cbc21dc1STaehee Yoo 	if (amtd->reserved || amtd->version)
2651cbc21dc1STaehee Yoo 		return true;
2652cbc21dc1STaehee Yoo 
2653cbc21dc1STaehee Yoo 	amt_send_advertisement(amt, amtd->nonce, iph->saddr, udph->source);
2654cbc21dc1STaehee Yoo 
2655cbc21dc1STaehee Yoo 	return false;
2656cbc21dc1STaehee Yoo }
2657cbc21dc1STaehee Yoo 
amt_request_handler(struct amt_dev * amt,struct sk_buff * skb)2658cbc21dc1STaehee Yoo static bool amt_request_handler(struct amt_dev *amt, struct sk_buff *skb)
2659cbc21dc1STaehee Yoo {
2660cbc21dc1STaehee Yoo 	struct amt_header_request *amtrh;
2661cbc21dc1STaehee Yoo 	struct amt_tunnel_list *tunnel;
2662cbc21dc1STaehee Yoo 	unsigned long long key;
2663cbc21dc1STaehee Yoo 	struct udphdr *udph;
2664cbc21dc1STaehee Yoo 	struct iphdr *iph;
2665cbc21dc1STaehee Yoo 	u64 mac;
2666cbc21dc1STaehee Yoo 	int i;
2667cbc21dc1STaehee Yoo 
2668cbc21dc1STaehee Yoo 	if (!pskb_may_pull(skb, sizeof(*udph) + sizeof(*amtrh)))
2669cbc21dc1STaehee Yoo 		return true;
2670cbc21dc1STaehee Yoo 
2671cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2672cbc21dc1STaehee Yoo 	udph = udp_hdr(skb);
2673cbc21dc1STaehee Yoo 	amtrh = (struct amt_header_request *)(udp_hdr(skb) + 1);
2674cbc21dc1STaehee Yoo 
2675cbc21dc1STaehee Yoo 	if (amtrh->reserved1 || amtrh->reserved2 || amtrh->version)
2676cbc21dc1STaehee Yoo 		return true;
2677cbc21dc1STaehee Yoo 
2678cbc21dc1STaehee Yoo 	list_for_each_entry_rcu(tunnel, &amt->tunnel_list, list)
2679cbc21dc1STaehee Yoo 		if (tunnel->ip4 == iph->saddr)
2680cbc21dc1STaehee Yoo 			goto send;
2681cbc21dc1STaehee Yoo 
268298991848STaehee Yoo 	spin_lock_bh(&amt->lock);
2683cbc21dc1STaehee Yoo 	if (amt->nr_tunnels >= amt->max_tunnels) {
268498991848STaehee Yoo 		spin_unlock_bh(&amt->lock);
2685cbc21dc1STaehee Yoo 		icmp_ndo_send(skb, ICMP_DEST_UNREACH, ICMP_HOST_UNREACH, 0);
2686cbc21dc1STaehee Yoo 		return true;
2687cbc21dc1STaehee Yoo 	}
2688cbc21dc1STaehee Yoo 
2689cbc21dc1STaehee Yoo 	tunnel = kzalloc(sizeof(*tunnel) +
2690cbc21dc1STaehee Yoo 			 (sizeof(struct hlist_head) * amt->hash_buckets),
2691cbc21dc1STaehee Yoo 			 GFP_ATOMIC);
269298991848STaehee Yoo 	if (!tunnel) {
269398991848STaehee Yoo 		spin_unlock_bh(&amt->lock);
2694cbc21dc1STaehee Yoo 		return true;
269598991848STaehee Yoo 	}
2696cbc21dc1STaehee Yoo 
2697cbc21dc1STaehee Yoo 	tunnel->source_port = udph->source;
2698cbc21dc1STaehee Yoo 	tunnel->ip4 = iph->saddr;
2699cbc21dc1STaehee Yoo 
2700cbc21dc1STaehee Yoo 	memcpy(&key, &tunnel->key, sizeof(unsigned long long));
2701cbc21dc1STaehee Yoo 	tunnel->amt = amt;
2702cbc21dc1STaehee Yoo 	spin_lock_init(&tunnel->lock);
2703cbc21dc1STaehee Yoo 	for (i = 0; i < amt->hash_buckets; i++)
2704cbc21dc1STaehee Yoo 		INIT_HLIST_HEAD(&tunnel->groups[i]);
2705cbc21dc1STaehee Yoo 
2706cbc21dc1STaehee Yoo 	INIT_DELAYED_WORK(&tunnel->gc_wq, amt_tunnel_expire);
2707cbc21dc1STaehee Yoo 
2708cbc21dc1STaehee Yoo 	list_add_tail_rcu(&tunnel->list, &amt->tunnel_list);
2709cbc21dc1STaehee Yoo 	tunnel->key = amt->key;
271098991848STaehee Yoo 	__amt_update_relay_status(tunnel, AMT_STATUS_RECEIVED_REQUEST, true);
2711cbc21dc1STaehee Yoo 	amt->nr_tunnels++;
2712cbc21dc1STaehee Yoo 	mod_delayed_work(amt_wq, &tunnel->gc_wq,
2713cbc21dc1STaehee Yoo 			 msecs_to_jiffies(amt_gmi(amt)));
2714cbc21dc1STaehee Yoo 	spin_unlock_bh(&amt->lock);
2715cbc21dc1STaehee Yoo 
2716cbc21dc1STaehee Yoo send:
2717cbc21dc1STaehee Yoo 	tunnel->nonce = amtrh->nonce;
2718cbc21dc1STaehee Yoo 	mac = siphash_3u32((__force u32)tunnel->ip4,
2719cbc21dc1STaehee Yoo 			   (__force u32)tunnel->source_port,
2720cbc21dc1STaehee Yoo 			   (__force u32)tunnel->nonce,
2721cbc21dc1STaehee Yoo 			   &tunnel->key);
2722cbc21dc1STaehee Yoo 	tunnel->mac = mac >> 16;
2723cbc21dc1STaehee Yoo 
2724cbc21dc1STaehee Yoo 	if (!netif_running(amt->dev) || !netif_running(amt->stream_dev))
2725cbc21dc1STaehee Yoo 		return true;
2726cbc21dc1STaehee Yoo 
2727cbc21dc1STaehee Yoo 	if (!amtrh->p)
2728cbc21dc1STaehee Yoo 		amt_send_igmp_gq(amt, tunnel);
2729b75f7095STaehee Yoo 	else
2730b75f7095STaehee Yoo 		amt_send_mld_gq(amt, tunnel);
2731cbc21dc1STaehee Yoo 
2732cbc21dc1STaehee Yoo 	return false;
2733cbc21dc1STaehee Yoo }
2734cbc21dc1STaehee Yoo 
amt_gw_rcv(struct amt_dev * amt,struct sk_buff * skb)273530e22a6eSTaehee Yoo static void amt_gw_rcv(struct amt_dev *amt, struct sk_buff *skb)
273630e22a6eSTaehee Yoo {
273730e22a6eSTaehee Yoo 	int type = amt_parse_type(skb);
273830e22a6eSTaehee Yoo 	int err = 1;
273930e22a6eSTaehee Yoo 
274030e22a6eSTaehee Yoo 	if (type == -1)
274130e22a6eSTaehee Yoo 		goto drop;
274230e22a6eSTaehee Yoo 
274330e22a6eSTaehee Yoo 	if (amt->mode == AMT_MODE_GATEWAY) {
274430e22a6eSTaehee Yoo 		switch (type) {
274530e22a6eSTaehee Yoo 		case AMT_MSG_ADVERTISEMENT:
274630e22a6eSTaehee Yoo 			err = amt_advertisement_handler(amt, skb);
274730e22a6eSTaehee Yoo 			break;
274830e22a6eSTaehee Yoo 		case AMT_MSG_MEMBERSHIP_QUERY:
274930e22a6eSTaehee Yoo 			err = amt_membership_query_handler(amt, skb);
275030e22a6eSTaehee Yoo 			if (!err)
275130e22a6eSTaehee Yoo 				return;
275230e22a6eSTaehee Yoo 			break;
275330e22a6eSTaehee Yoo 		default:
275430e22a6eSTaehee Yoo 			netdev_dbg(amt->dev, "Invalid type of Gateway\n");
275530e22a6eSTaehee Yoo 			break;
275630e22a6eSTaehee Yoo 		}
275730e22a6eSTaehee Yoo 	}
275830e22a6eSTaehee Yoo drop:
275930e22a6eSTaehee Yoo 	if (err) {
276030e22a6eSTaehee Yoo 		amt->dev->stats.rx_dropped++;
276130e22a6eSTaehee Yoo 		kfree_skb(skb);
276230e22a6eSTaehee Yoo 	} else {
276330e22a6eSTaehee Yoo 		consume_skb(skb);
276430e22a6eSTaehee Yoo 	}
276530e22a6eSTaehee Yoo }
276630e22a6eSTaehee Yoo 
amt_rcv(struct sock * sk,struct sk_buff * skb)2767cbc21dc1STaehee Yoo static int amt_rcv(struct sock *sk, struct sk_buff *skb)
2768cbc21dc1STaehee Yoo {
2769cbc21dc1STaehee Yoo 	struct amt_dev *amt;
2770cbc21dc1STaehee Yoo 	struct iphdr *iph;
2771cbc21dc1STaehee Yoo 	int type;
2772cbc21dc1STaehee Yoo 	bool err;
2773cbc21dc1STaehee Yoo 
2774cbc21dc1STaehee Yoo 	rcu_read_lock_bh();
2775cbc21dc1STaehee Yoo 	amt = rcu_dereference_sk_user_data(sk);
2776cbc21dc1STaehee Yoo 	if (!amt) {
2777cbc21dc1STaehee Yoo 		err = true;
2778d16207f9STaehee Yoo 		kfree_skb(skb);
2779d16207f9STaehee Yoo 		goto out;
2780cbc21dc1STaehee Yoo 	}
2781cbc21dc1STaehee Yoo 
2782cbc21dc1STaehee Yoo 	skb->dev = amt->dev;
2783cbc21dc1STaehee Yoo 	iph = ip_hdr(skb);
2784cbc21dc1STaehee Yoo 	type = amt_parse_type(skb);
2785cbc21dc1STaehee Yoo 	if (type == -1) {
2786cbc21dc1STaehee Yoo 		err = true;
2787cbc21dc1STaehee Yoo 		goto drop;
2788cbc21dc1STaehee Yoo 	}
2789cbc21dc1STaehee Yoo 
2790cbc21dc1STaehee Yoo 	if (amt->mode == AMT_MODE_GATEWAY) {
2791cbc21dc1STaehee Yoo 		switch (type) {
2792cbc21dc1STaehee Yoo 		case AMT_MSG_ADVERTISEMENT:
2793cbc21dc1STaehee Yoo 			if (iph->saddr != amt->discovery_ip) {
2794cbc21dc1STaehee Yoo 				netdev_dbg(amt->dev, "Invalid Relay IP\n");
2795cbc21dc1STaehee Yoo 				err = true;
2796cbc21dc1STaehee Yoo 				goto drop;
2797cbc21dc1STaehee Yoo 			}
279830e22a6eSTaehee Yoo 			if (amt_queue_event(amt, AMT_EVENT_RECEIVE, skb)) {
279930e22a6eSTaehee Yoo 				netdev_dbg(amt->dev, "AMT Event queue full\n");
280030e22a6eSTaehee Yoo 				err = true;
280130e22a6eSTaehee Yoo 				goto drop;
280230e22a6eSTaehee Yoo 			}
280330e22a6eSTaehee Yoo 			goto out;
2804cbc21dc1STaehee Yoo 		case AMT_MSG_MULTICAST_DATA:
2805cbc21dc1STaehee Yoo 			if (iph->saddr != amt->remote_ip) {
2806cbc21dc1STaehee Yoo 				netdev_dbg(amt->dev, "Invalid Relay IP\n");
2807cbc21dc1STaehee Yoo 				err = true;
2808cbc21dc1STaehee Yoo 				goto drop;
2809cbc21dc1STaehee Yoo 			}
2810cbc21dc1STaehee Yoo 			err = amt_multicast_data_handler(amt, skb);
2811cbc21dc1STaehee Yoo 			if (err)
2812cbc21dc1STaehee Yoo 				goto drop;
2813cbc21dc1STaehee Yoo 			else
2814cbc21dc1STaehee Yoo 				goto out;
2815cbc21dc1STaehee Yoo 		case AMT_MSG_MEMBERSHIP_QUERY:
2816cbc21dc1STaehee Yoo 			if (iph->saddr != amt->remote_ip) {
2817cbc21dc1STaehee Yoo 				netdev_dbg(amt->dev, "Invalid Relay IP\n");
2818cbc21dc1STaehee Yoo 				err = true;
2819cbc21dc1STaehee Yoo 				goto drop;
2820cbc21dc1STaehee Yoo 			}
282130e22a6eSTaehee Yoo 			if (amt_queue_event(amt, AMT_EVENT_RECEIVE, skb)) {
282230e22a6eSTaehee Yoo 				netdev_dbg(amt->dev, "AMT Event queue full\n");
282330e22a6eSTaehee Yoo 				err = true;
2824cbc21dc1STaehee Yoo 				goto drop;
282530e22a6eSTaehee Yoo 			}
2826cbc21dc1STaehee Yoo 			goto out;
2827cbc21dc1STaehee Yoo 		default:
2828cbc21dc1STaehee Yoo 			err = true;
2829cbc21dc1STaehee Yoo 			netdev_dbg(amt->dev, "Invalid type of Gateway\n");
2830cbc21dc1STaehee Yoo 			break;
2831cbc21dc1STaehee Yoo 		}
2832cbc21dc1STaehee Yoo 	} else {
2833cbc21dc1STaehee Yoo 		switch (type) {
2834cbc21dc1STaehee Yoo 		case AMT_MSG_DISCOVERY:
2835cbc21dc1STaehee Yoo 			err = amt_discovery_handler(amt, skb);
2836cbc21dc1STaehee Yoo 			break;
2837cbc21dc1STaehee Yoo 		case AMT_MSG_REQUEST:
2838cbc21dc1STaehee Yoo 			err = amt_request_handler(amt, skb);
2839cbc21dc1STaehee Yoo 			break;
2840cbc21dc1STaehee Yoo 		case AMT_MSG_MEMBERSHIP_UPDATE:
2841cbc21dc1STaehee Yoo 			err = amt_update_handler(amt, skb);
2842cbc21dc1STaehee Yoo 			if (err)
2843cbc21dc1STaehee Yoo 				goto drop;
2844cbc21dc1STaehee Yoo 			else
2845cbc21dc1STaehee Yoo 				goto out;
2846cbc21dc1STaehee Yoo 		default:
2847cbc21dc1STaehee Yoo 			err = true;
2848cbc21dc1STaehee Yoo 			netdev_dbg(amt->dev, "Invalid type of relay\n");
2849cbc21dc1STaehee Yoo 			break;
2850cbc21dc1STaehee Yoo 		}
2851cbc21dc1STaehee Yoo 	}
2852cbc21dc1STaehee Yoo drop:
2853cbc21dc1STaehee Yoo 	if (err) {
2854cbc21dc1STaehee Yoo 		amt->dev->stats.rx_dropped++;
2855cbc21dc1STaehee Yoo 		kfree_skb(skb);
2856cbc21dc1STaehee Yoo 	} else {
2857cbc21dc1STaehee Yoo 		consume_skb(skb);
2858cbc21dc1STaehee Yoo 	}
2859cbc21dc1STaehee Yoo out:
2860cbc21dc1STaehee Yoo 	rcu_read_unlock_bh();
2861cbc21dc1STaehee Yoo 	return 0;
2862cbc21dc1STaehee Yoo }
2863cbc21dc1STaehee Yoo 
amt_event_work(struct work_struct * work)286430e22a6eSTaehee Yoo static void amt_event_work(struct work_struct *work)
286530e22a6eSTaehee Yoo {
286630e22a6eSTaehee Yoo 	struct amt_dev *amt = container_of(work, struct amt_dev, event_wq);
286730e22a6eSTaehee Yoo 	struct sk_buff *skb;
286830e22a6eSTaehee Yoo 	u8 event;
286930e22a6eSTaehee Yoo 	int i;
287030e22a6eSTaehee Yoo 
287130e22a6eSTaehee Yoo 	for (i = 0; i < AMT_MAX_EVENTS; i++) {
287230e22a6eSTaehee Yoo 		spin_lock_bh(&amt->lock);
287330e22a6eSTaehee Yoo 		if (amt->nr_events == 0) {
287430e22a6eSTaehee Yoo 			spin_unlock_bh(&amt->lock);
287530e22a6eSTaehee Yoo 			return;
287630e22a6eSTaehee Yoo 		}
287730e22a6eSTaehee Yoo 		event = amt->events[amt->event_idx].event;
287830e22a6eSTaehee Yoo 		skb = amt->events[amt->event_idx].skb;
287930e22a6eSTaehee Yoo 		amt->events[amt->event_idx].event = AMT_EVENT_NONE;
288030e22a6eSTaehee Yoo 		amt->events[amt->event_idx].skb = NULL;
288130e22a6eSTaehee Yoo 		amt->nr_events--;
288230e22a6eSTaehee Yoo 		amt->event_idx++;
288330e22a6eSTaehee Yoo 		amt->event_idx %= AMT_MAX_EVENTS;
288430e22a6eSTaehee Yoo 		spin_unlock_bh(&amt->lock);
288530e22a6eSTaehee Yoo 
288630e22a6eSTaehee Yoo 		switch (event) {
288730e22a6eSTaehee Yoo 		case AMT_EVENT_RECEIVE:
288830e22a6eSTaehee Yoo 			amt_gw_rcv(amt, skb);
288930e22a6eSTaehee Yoo 			break;
289030e22a6eSTaehee Yoo 		case AMT_EVENT_SEND_DISCOVERY:
289130e22a6eSTaehee Yoo 			amt_event_send_discovery(amt);
289230e22a6eSTaehee Yoo 			break;
289330e22a6eSTaehee Yoo 		case AMT_EVENT_SEND_REQUEST:
289430e22a6eSTaehee Yoo 			amt_event_send_request(amt);
289530e22a6eSTaehee Yoo 			break;
289630e22a6eSTaehee Yoo 		default:
289730e22a6eSTaehee Yoo 			kfree_skb(skb);
289830e22a6eSTaehee Yoo 			break;
289930e22a6eSTaehee Yoo 		}
290030e22a6eSTaehee Yoo 	}
290130e22a6eSTaehee Yoo }
290230e22a6eSTaehee Yoo 
amt_err_lookup(struct sock * sk,struct sk_buff * skb)2903cbc21dc1STaehee Yoo static int amt_err_lookup(struct sock *sk, struct sk_buff *skb)
2904cbc21dc1STaehee Yoo {
2905cbc21dc1STaehee Yoo 	struct amt_dev *amt;
2906cbc21dc1STaehee Yoo 	int type;
2907cbc21dc1STaehee Yoo 
2908cbc21dc1STaehee Yoo 	rcu_read_lock_bh();
2909cbc21dc1STaehee Yoo 	amt = rcu_dereference_sk_user_data(sk);
2910cbc21dc1STaehee Yoo 	if (!amt)
29113f81c579SYang Li 		goto out;
2912cbc21dc1STaehee Yoo 
2913cbc21dc1STaehee Yoo 	if (amt->mode != AMT_MODE_GATEWAY)
2914cbc21dc1STaehee Yoo 		goto drop;
2915cbc21dc1STaehee Yoo 
2916cbc21dc1STaehee Yoo 	type = amt_parse_type(skb);
2917cbc21dc1STaehee Yoo 	if (type == -1)
2918cbc21dc1STaehee Yoo 		goto drop;
2919cbc21dc1STaehee Yoo 
2920cbc21dc1STaehee Yoo 	netdev_dbg(amt->dev, "Received IGMP Unreachable of %s\n",
2921cbc21dc1STaehee Yoo 		   type_str[type]);
2922cbc21dc1STaehee Yoo 	switch (type) {
2923cbc21dc1STaehee Yoo 	case AMT_MSG_DISCOVERY:
2924cbc21dc1STaehee Yoo 		break;
2925cbc21dc1STaehee Yoo 	case AMT_MSG_REQUEST:
2926cbc21dc1STaehee Yoo 	case AMT_MSG_MEMBERSHIP_UPDATE:
2927928f353cSTaehee Yoo 		if (READ_ONCE(amt->status) >= AMT_STATUS_RECEIVED_ADVERTISEMENT)
2928cbc21dc1STaehee Yoo 			mod_delayed_work(amt_wq, &amt->req_wq, 0);
2929cbc21dc1STaehee Yoo 		break;
2930cbc21dc1STaehee Yoo 	default:
2931cbc21dc1STaehee Yoo 		goto drop;
2932cbc21dc1STaehee Yoo 	}
29333f81c579SYang Li out:
2934cbc21dc1STaehee Yoo 	rcu_read_unlock_bh();
2935cbc21dc1STaehee Yoo 	return 0;
2936cbc21dc1STaehee Yoo drop:
2937cbc21dc1STaehee Yoo 	rcu_read_unlock_bh();
2938cbc21dc1STaehee Yoo 	amt->dev->stats.rx_dropped++;
2939cbc21dc1STaehee Yoo 	return 0;
2940cbc21dc1STaehee Yoo }
2941cbc21dc1STaehee Yoo 
amt_create_sock(struct net * net,__be16 port)2942b9022b53STaehee Yoo static struct socket *amt_create_sock(struct net *net, __be16 port)
2943b9022b53STaehee Yoo {
2944b9022b53STaehee Yoo 	struct udp_port_cfg udp_conf;
2945b9022b53STaehee Yoo 	struct socket *sock;
2946b9022b53STaehee Yoo 	int err;
2947b9022b53STaehee Yoo 
2948b9022b53STaehee Yoo 	memset(&udp_conf, 0, sizeof(udp_conf));
2949b9022b53STaehee Yoo 	udp_conf.family = AF_INET;
2950b9022b53STaehee Yoo 	udp_conf.local_ip.s_addr = htonl(INADDR_ANY);
2951b9022b53STaehee Yoo 
2952b9022b53STaehee Yoo 	udp_conf.local_udp_port = port;
2953b9022b53STaehee Yoo 
2954b9022b53STaehee Yoo 	err = udp_sock_create(net, &udp_conf, &sock);
2955b9022b53STaehee Yoo 	if (err < 0)
2956b9022b53STaehee Yoo 		return ERR_PTR(err);
2957b9022b53STaehee Yoo 
2958b9022b53STaehee Yoo 	return sock;
2959b9022b53STaehee Yoo }
2960b9022b53STaehee Yoo 
amt_socket_create(struct amt_dev * amt)2961b9022b53STaehee Yoo static int amt_socket_create(struct amt_dev *amt)
2962b9022b53STaehee Yoo {
2963b9022b53STaehee Yoo 	struct udp_tunnel_sock_cfg tunnel_cfg;
2964b9022b53STaehee Yoo 	struct socket *sock;
2965b9022b53STaehee Yoo 
2966b9022b53STaehee Yoo 	sock = amt_create_sock(amt->net, amt->relay_port);
2967b9022b53STaehee Yoo 	if (IS_ERR(sock))
2968b9022b53STaehee Yoo 		return PTR_ERR(sock);
2969b9022b53STaehee Yoo 
2970b9022b53STaehee Yoo 	/* Mark socket as an encapsulation socket */
2971b9022b53STaehee Yoo 	memset(&tunnel_cfg, 0, sizeof(tunnel_cfg));
2972b9022b53STaehee Yoo 	tunnel_cfg.sk_user_data = amt;
2973b9022b53STaehee Yoo 	tunnel_cfg.encap_type = 1;
2974cbc21dc1STaehee Yoo 	tunnel_cfg.encap_rcv = amt_rcv;
2975cbc21dc1STaehee Yoo 	tunnel_cfg.encap_err_lookup = amt_err_lookup;
2976b9022b53STaehee Yoo 	tunnel_cfg.encap_destroy = NULL;
2977b9022b53STaehee Yoo 	setup_udp_tunnel_sock(amt->net, sock, &tunnel_cfg);
2978b9022b53STaehee Yoo 
2979b9022b53STaehee Yoo 	rcu_assign_pointer(amt->sock, sock);
2980b9022b53STaehee Yoo 	return 0;
2981b9022b53STaehee Yoo }
2982b9022b53STaehee Yoo 
amt_dev_open(struct net_device * dev)2983b9022b53STaehee Yoo static int amt_dev_open(struct net_device *dev)
2984b9022b53STaehee Yoo {
2985b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
2986b9022b53STaehee Yoo 	int err;
2987b9022b53STaehee Yoo 
2988b9022b53STaehee Yoo 	amt->ready4 = false;
2989b9022b53STaehee Yoo 	amt->ready6 = false;
299030e22a6eSTaehee Yoo 	amt->event_idx = 0;
299130e22a6eSTaehee Yoo 	amt->nr_events = 0;
2992b9022b53STaehee Yoo 
2993b9022b53STaehee Yoo 	err = amt_socket_create(amt);
2994b9022b53STaehee Yoo 	if (err)
2995b9022b53STaehee Yoo 		return err;
2996b9022b53STaehee Yoo 
2997b9022b53STaehee Yoo 	amt->req_cnt = 0;
2998b9022b53STaehee Yoo 	amt->remote_ip = 0;
299940185f35STaehee Yoo 	amt->nonce = 0;
3000b9022b53STaehee Yoo 	get_random_bytes(&amt->key, sizeof(siphash_key_t));
3001b9022b53STaehee Yoo 
3002b9022b53STaehee Yoo 	amt->status = AMT_STATUS_INIT;
3003cbc21dc1STaehee Yoo 	if (amt->mode == AMT_MODE_GATEWAY) {
3004cbc21dc1STaehee Yoo 		mod_delayed_work(amt_wq, &amt->discovery_wq, 0);
3005cbc21dc1STaehee Yoo 		mod_delayed_work(amt_wq, &amt->req_wq, 0);
3006cbc21dc1STaehee Yoo 	} else if (amt->mode == AMT_MODE_RELAY) {
3007cbc21dc1STaehee Yoo 		mod_delayed_work(amt_wq, &amt->secret_wq,
3008cbc21dc1STaehee Yoo 				 msecs_to_jiffies(AMT_SECRET_TIMEOUT));
3009cbc21dc1STaehee Yoo 	}
3010b9022b53STaehee Yoo 	return err;
3011b9022b53STaehee Yoo }
3012b9022b53STaehee Yoo 
amt_dev_stop(struct net_device * dev)3013b9022b53STaehee Yoo static int amt_dev_stop(struct net_device *dev)
3014b9022b53STaehee Yoo {
3015b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
3016cbc21dc1STaehee Yoo 	struct amt_tunnel_list *tunnel, *tmp;
3017b9022b53STaehee Yoo 	struct socket *sock;
301830e22a6eSTaehee Yoo 	struct sk_buff *skb;
301930e22a6eSTaehee Yoo 	int i;
3020b9022b53STaehee Yoo 
3021cbc21dc1STaehee Yoo 	cancel_delayed_work_sync(&amt->req_wq);
3022cbc21dc1STaehee Yoo 	cancel_delayed_work_sync(&amt->discovery_wq);
3023cbc21dc1STaehee Yoo 	cancel_delayed_work_sync(&amt->secret_wq);
3024cbc21dc1STaehee Yoo 
3025b9022b53STaehee Yoo 	/* shutdown */
3026b9022b53STaehee Yoo 	sock = rtnl_dereference(amt->sock);
3027b9022b53STaehee Yoo 	RCU_INIT_POINTER(amt->sock, NULL);
3028b9022b53STaehee Yoo 	synchronize_net();
3029b9022b53STaehee Yoo 	if (sock)
3030b9022b53STaehee Yoo 		udp_tunnel_sock_release(sock);
3031b9022b53STaehee Yoo 
303230e22a6eSTaehee Yoo 	cancel_work_sync(&amt->event_wq);
303330e22a6eSTaehee Yoo 	for (i = 0; i < AMT_MAX_EVENTS; i++) {
303430e22a6eSTaehee Yoo 		skb = amt->events[i].skb;
303530e22a6eSTaehee Yoo 		kfree_skb(skb);
303630e22a6eSTaehee Yoo 		amt->events[i].event = AMT_EVENT_NONE;
303730e22a6eSTaehee Yoo 		amt->events[i].skb = NULL;
303830e22a6eSTaehee Yoo 	}
303930e22a6eSTaehee Yoo 
3040b9022b53STaehee Yoo 	amt->ready4 = false;
3041b9022b53STaehee Yoo 	amt->ready6 = false;
3042b9022b53STaehee Yoo 	amt->req_cnt = 0;
3043b9022b53STaehee Yoo 	amt->remote_ip = 0;
3044b9022b53STaehee Yoo 
3045cbc21dc1STaehee Yoo 	list_for_each_entry_safe(tunnel, tmp, &amt->tunnel_list, list) {
3046cbc21dc1STaehee Yoo 		list_del_rcu(&tunnel->list);
3047cbc21dc1STaehee Yoo 		amt->nr_tunnels--;
3048cbc21dc1STaehee Yoo 		cancel_delayed_work_sync(&tunnel->gc_wq);
3049bc54e49cSTaehee Yoo 		amt_clear_groups(tunnel);
3050cbc21dc1STaehee Yoo 		kfree_rcu(tunnel, rcu);
3051cbc21dc1STaehee Yoo 	}
3052cbc21dc1STaehee Yoo 
3053b9022b53STaehee Yoo 	return 0;
3054b9022b53STaehee Yoo }
3055b9022b53STaehee Yoo 
3056b9022b53STaehee Yoo static const struct device_type amt_type = {
3057b9022b53STaehee Yoo 	.name = "amt",
3058b9022b53STaehee Yoo };
3059b9022b53STaehee Yoo 
amt_dev_init(struct net_device * dev)3060b9022b53STaehee Yoo static int amt_dev_init(struct net_device *dev)
3061b9022b53STaehee Yoo {
3062b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
3063b9022b53STaehee Yoo 	int err;
3064b9022b53STaehee Yoo 
3065b9022b53STaehee Yoo 	amt->dev = dev;
3066b9022b53STaehee Yoo 	dev->tstats = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats);
3067b9022b53STaehee Yoo 	if (!dev->tstats)
3068b9022b53STaehee Yoo 		return -ENOMEM;
3069b9022b53STaehee Yoo 
3070b9022b53STaehee Yoo 	err = gro_cells_init(&amt->gro_cells, dev);
3071b9022b53STaehee Yoo 	if (err) {
3072b9022b53STaehee Yoo 		free_percpu(dev->tstats);
3073b9022b53STaehee Yoo 		return err;
3074b9022b53STaehee Yoo 	}
3075b9022b53STaehee Yoo 
3076b9022b53STaehee Yoo 	return 0;
3077b9022b53STaehee Yoo }
3078b9022b53STaehee Yoo 
amt_dev_uninit(struct net_device * dev)3079b9022b53STaehee Yoo static void amt_dev_uninit(struct net_device *dev)
3080b9022b53STaehee Yoo {
3081b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
3082b9022b53STaehee Yoo 
3083b9022b53STaehee Yoo 	gro_cells_destroy(&amt->gro_cells);
3084b9022b53STaehee Yoo 	free_percpu(dev->tstats);
3085b9022b53STaehee Yoo }
3086b9022b53STaehee Yoo 
3087b9022b53STaehee Yoo static const struct net_device_ops amt_netdev_ops = {
3088b9022b53STaehee Yoo 	.ndo_init               = amt_dev_init,
3089b9022b53STaehee Yoo 	.ndo_uninit             = amt_dev_uninit,
3090b9022b53STaehee Yoo 	.ndo_open		= amt_dev_open,
3091b9022b53STaehee Yoo 	.ndo_stop		= amt_dev_stop,
3092cbc21dc1STaehee Yoo 	.ndo_start_xmit         = amt_dev_xmit,
3093b9022b53STaehee Yoo 	.ndo_get_stats64        = dev_get_tstats64,
3094b9022b53STaehee Yoo };
3095b9022b53STaehee Yoo 
amt_link_setup(struct net_device * dev)3096b9022b53STaehee Yoo static void amt_link_setup(struct net_device *dev)
3097b9022b53STaehee Yoo {
3098b9022b53STaehee Yoo 	dev->netdev_ops         = &amt_netdev_ops;
3099b9022b53STaehee Yoo 	dev->needs_free_netdev  = true;
3100b9022b53STaehee Yoo 	SET_NETDEV_DEVTYPE(dev, &amt_type);
3101b9022b53STaehee Yoo 	dev->min_mtu		= ETH_MIN_MTU;
3102b9022b53STaehee Yoo 	dev->max_mtu		= ETH_MAX_MTU;
3103b9022b53STaehee Yoo 	dev->type		= ARPHRD_NONE;
3104b9022b53STaehee Yoo 	dev->flags		= IFF_POINTOPOINT | IFF_NOARP | IFF_MULTICAST;
3105b9022b53STaehee Yoo 	dev->hard_header_len	= 0;
3106b9022b53STaehee Yoo 	dev->addr_len		= 0;
3107b9022b53STaehee Yoo 	dev->priv_flags		|= IFF_NO_QUEUE;
3108b9022b53STaehee Yoo 	dev->features		|= NETIF_F_LLTX;
3109b9022b53STaehee Yoo 	dev->features		|= NETIF_F_GSO_SOFTWARE;
3110b9022b53STaehee Yoo 	dev->features		|= NETIF_F_NETNS_LOCAL;
3111b9022b53STaehee Yoo 	dev->hw_features	|= NETIF_F_SG | NETIF_F_HW_CSUM;
3112b9022b53STaehee Yoo 	dev->hw_features	|= NETIF_F_FRAGLIST | NETIF_F_RXCSUM;
3113b9022b53STaehee Yoo 	dev->hw_features	|= NETIF_F_GSO_SOFTWARE;
3114b9022b53STaehee Yoo 	eth_hw_addr_random(dev);
3115b9022b53STaehee Yoo 	eth_zero_addr(dev->broadcast);
3116b9022b53STaehee Yoo 	ether_setup(dev);
3117b9022b53STaehee Yoo }
3118b9022b53STaehee Yoo 
3119b9022b53STaehee Yoo static const struct nla_policy amt_policy[IFLA_AMT_MAX + 1] = {
3120b9022b53STaehee Yoo 	[IFLA_AMT_MODE]		= { .type = NLA_U32 },
3121b9022b53STaehee Yoo 	[IFLA_AMT_RELAY_PORT]	= { .type = NLA_U16 },
3122b9022b53STaehee Yoo 	[IFLA_AMT_GATEWAY_PORT]	= { .type = NLA_U16 },
3123b9022b53STaehee Yoo 	[IFLA_AMT_LINK]		= { .type = NLA_U32 },
3124b9022b53STaehee Yoo 	[IFLA_AMT_LOCAL_IP]	= { .len = sizeof_field(struct iphdr, daddr) },
3125b9022b53STaehee Yoo 	[IFLA_AMT_REMOTE_IP]	= { .len = sizeof_field(struct iphdr, daddr) },
3126b9022b53STaehee Yoo 	[IFLA_AMT_DISCOVERY_IP]	= { .len = sizeof_field(struct iphdr, daddr) },
3127b9022b53STaehee Yoo 	[IFLA_AMT_MAX_TUNNELS]	= { .type = NLA_U32 },
3128b9022b53STaehee Yoo };
3129b9022b53STaehee Yoo 
amt_validate(struct nlattr * tb[],struct nlattr * data[],struct netlink_ext_ack * extack)3130b9022b53STaehee Yoo static int amt_validate(struct nlattr *tb[], struct nlattr *data[],
3131b9022b53STaehee Yoo 			struct netlink_ext_ack *extack)
3132b9022b53STaehee Yoo {
3133b9022b53STaehee Yoo 	if (!data)
3134b9022b53STaehee Yoo 		return -EINVAL;
3135b9022b53STaehee Yoo 
3136b9022b53STaehee Yoo 	if (!data[IFLA_AMT_LINK]) {
3137b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, data[IFLA_AMT_LINK],
3138b9022b53STaehee Yoo 				    "Link attribute is required");
3139b9022b53STaehee Yoo 		return -EINVAL;
3140b9022b53STaehee Yoo 	}
3141b9022b53STaehee Yoo 
3142b9022b53STaehee Yoo 	if (!data[IFLA_AMT_MODE]) {
3143b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, data[IFLA_AMT_MODE],
3144b9022b53STaehee Yoo 				    "Mode attribute is required");
3145b9022b53STaehee Yoo 		return -EINVAL;
3146b9022b53STaehee Yoo 	}
3147b9022b53STaehee Yoo 
3148b9022b53STaehee Yoo 	if (nla_get_u32(data[IFLA_AMT_MODE]) > AMT_MODE_MAX) {
3149b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, data[IFLA_AMT_MODE],
3150b9022b53STaehee Yoo 				    "Mode attribute is not valid");
3151b9022b53STaehee Yoo 		return -EINVAL;
3152b9022b53STaehee Yoo 	}
3153b9022b53STaehee Yoo 
3154b9022b53STaehee Yoo 	if (!data[IFLA_AMT_LOCAL_IP]) {
3155b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, data[IFLA_AMT_DISCOVERY_IP],
3156b9022b53STaehee Yoo 				    "Local attribute is required");
3157b9022b53STaehee Yoo 		return -EINVAL;
3158b9022b53STaehee Yoo 	}
3159b9022b53STaehee Yoo 
3160b9022b53STaehee Yoo 	if (!data[IFLA_AMT_DISCOVERY_IP] &&
3161b9022b53STaehee Yoo 	    nla_get_u32(data[IFLA_AMT_MODE]) == AMT_MODE_GATEWAY) {
3162b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, data[IFLA_AMT_LOCAL_IP],
3163b9022b53STaehee Yoo 				    "Discovery attribute is required");
3164b9022b53STaehee Yoo 		return -EINVAL;
3165b9022b53STaehee Yoo 	}
3166b9022b53STaehee Yoo 
3167b9022b53STaehee Yoo 	return 0;
3168b9022b53STaehee Yoo }
3169b9022b53STaehee Yoo 
amt_newlink(struct net * net,struct net_device * dev,struct nlattr * tb[],struct nlattr * data[],struct netlink_ext_ack * extack)3170b9022b53STaehee Yoo static int amt_newlink(struct net *net, struct net_device *dev,
3171b9022b53STaehee Yoo 		       struct nlattr *tb[], struct nlattr *data[],
3172b9022b53STaehee Yoo 		       struct netlink_ext_ack *extack)
3173b9022b53STaehee Yoo {
3174b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
3175b9022b53STaehee Yoo 	int err = -EINVAL;
3176b9022b53STaehee Yoo 
3177b9022b53STaehee Yoo 	amt->net = net;
3178b9022b53STaehee Yoo 	amt->mode = nla_get_u32(data[IFLA_AMT_MODE]);
3179b9022b53STaehee Yoo 
3180cbc21dc1STaehee Yoo 	if (data[IFLA_AMT_MAX_TUNNELS] &&
3181cbc21dc1STaehee Yoo 	    nla_get_u32(data[IFLA_AMT_MAX_TUNNELS]))
3182b9022b53STaehee Yoo 		amt->max_tunnels = nla_get_u32(data[IFLA_AMT_MAX_TUNNELS]);
3183b9022b53STaehee Yoo 	else
3184b9022b53STaehee Yoo 		amt->max_tunnels = AMT_MAX_TUNNELS;
3185b9022b53STaehee Yoo 
3186b9022b53STaehee Yoo 	spin_lock_init(&amt->lock);
3187b9022b53STaehee Yoo 	amt->max_groups = AMT_MAX_GROUP;
3188b9022b53STaehee Yoo 	amt->max_sources = AMT_MAX_SOURCE;
3189b9022b53STaehee Yoo 	amt->hash_buckets = AMT_HSIZE;
3190b9022b53STaehee Yoo 	amt->nr_tunnels = 0;
3191b9022b53STaehee Yoo 	get_random_bytes(&amt->hash_seed, sizeof(amt->hash_seed));
3192b9022b53STaehee Yoo 	amt->stream_dev = dev_get_by_index(net,
3193b9022b53STaehee Yoo 					   nla_get_u32(data[IFLA_AMT_LINK]));
3194b9022b53STaehee Yoo 	if (!amt->stream_dev) {
3195b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_LINK],
3196b9022b53STaehee Yoo 				    "Can't find stream device");
3197b9022b53STaehee Yoo 		return -ENODEV;
3198b9022b53STaehee Yoo 	}
3199b9022b53STaehee Yoo 
3200b9022b53STaehee Yoo 	if (amt->stream_dev->type != ARPHRD_ETHER) {
3201b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_LINK],
3202b9022b53STaehee Yoo 				    "Invalid stream device type");
3203b9022b53STaehee Yoo 		goto err;
3204b9022b53STaehee Yoo 	}
3205b9022b53STaehee Yoo 
3206b9022b53STaehee Yoo 	amt->local_ip = nla_get_in_addr(data[IFLA_AMT_LOCAL_IP]);
3207b9022b53STaehee Yoo 	if (ipv4_is_loopback(amt->local_ip) ||
3208b9022b53STaehee Yoo 	    ipv4_is_zeronet(amt->local_ip) ||
3209b9022b53STaehee Yoo 	    ipv4_is_multicast(amt->local_ip)) {
3210b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_LOCAL_IP],
3211b9022b53STaehee Yoo 				    "Invalid Local address");
3212b9022b53STaehee Yoo 		goto err;
3213b9022b53STaehee Yoo 	}
3214b9022b53STaehee Yoo 
3215b9022b53STaehee Yoo 	if (data[IFLA_AMT_RELAY_PORT])
3216b9022b53STaehee Yoo 		amt->relay_port = nla_get_be16(data[IFLA_AMT_RELAY_PORT]);
3217b9022b53STaehee Yoo 	else
3218b9022b53STaehee Yoo 		amt->relay_port = htons(IANA_AMT_UDP_PORT);
3219b9022b53STaehee Yoo 
3220b9022b53STaehee Yoo 	if (data[IFLA_AMT_GATEWAY_PORT])
3221b9022b53STaehee Yoo 		amt->gw_port = nla_get_be16(data[IFLA_AMT_GATEWAY_PORT]);
3222b9022b53STaehee Yoo 	else
3223b9022b53STaehee Yoo 		amt->gw_port = htons(IANA_AMT_UDP_PORT);
3224b9022b53STaehee Yoo 
3225b9022b53STaehee Yoo 	if (!amt->relay_port) {
3226b9022b53STaehee Yoo 		NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_DISCOVERY_IP],
3227b9022b53STaehee Yoo 				    "relay port must not be 0");
3228b9022b53STaehee Yoo 		goto err;
3229b9022b53STaehee Yoo 	}
3230b9022b53STaehee Yoo 	if (amt->mode == AMT_MODE_RELAY) {
32318ebcc62cSKuniyuki Iwashima 		amt->qrv = READ_ONCE(amt->net->ipv4.sysctl_igmp_qrv);
3232b9022b53STaehee Yoo 		amt->qri = 10;
3233b9022b53STaehee Yoo 		dev->needed_headroom = amt->stream_dev->needed_headroom +
3234b9022b53STaehee Yoo 				       AMT_RELAY_HLEN;
3235b9022b53STaehee Yoo 		dev->mtu = amt->stream_dev->mtu - AMT_RELAY_HLEN;
3236b9022b53STaehee Yoo 		dev->max_mtu = dev->mtu;
3237b9022b53STaehee Yoo 		dev->min_mtu = ETH_MIN_MTU + AMT_RELAY_HLEN;
3238b9022b53STaehee Yoo 	} else {
3239b9022b53STaehee Yoo 		if (!data[IFLA_AMT_DISCOVERY_IP]) {
3240b9022b53STaehee Yoo 			NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_DISCOVERY_IP],
3241b9022b53STaehee Yoo 					    "discovery must be set in gateway mode");
3242b9022b53STaehee Yoo 			goto err;
3243b9022b53STaehee Yoo 		}
3244b9022b53STaehee Yoo 		if (!amt->gw_port) {
3245b9022b53STaehee Yoo 			NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_DISCOVERY_IP],
3246b9022b53STaehee Yoo 					    "gateway port must not be 0");
3247b9022b53STaehee Yoo 			goto err;
3248b9022b53STaehee Yoo 		}
3249b9022b53STaehee Yoo 		amt->remote_ip = 0;
3250b9022b53STaehee Yoo 		amt->discovery_ip = nla_get_in_addr(data[IFLA_AMT_DISCOVERY_IP]);
3251b9022b53STaehee Yoo 		if (ipv4_is_loopback(amt->discovery_ip) ||
3252b9022b53STaehee Yoo 		    ipv4_is_zeronet(amt->discovery_ip) ||
3253b9022b53STaehee Yoo 		    ipv4_is_multicast(amt->discovery_ip)) {
3254b9022b53STaehee Yoo 			NL_SET_ERR_MSG_ATTR(extack, tb[IFLA_AMT_DISCOVERY_IP],
3255b9022b53STaehee Yoo 					    "discovery must be unicast");
3256b9022b53STaehee Yoo 			goto err;
3257b9022b53STaehee Yoo 		}
3258b9022b53STaehee Yoo 
3259b9022b53STaehee Yoo 		dev->needed_headroom = amt->stream_dev->needed_headroom +
3260b9022b53STaehee Yoo 				       AMT_GW_HLEN;
3261b9022b53STaehee Yoo 		dev->mtu = amt->stream_dev->mtu - AMT_GW_HLEN;
3262b9022b53STaehee Yoo 		dev->max_mtu = dev->mtu;
3263b9022b53STaehee Yoo 		dev->min_mtu = ETH_MIN_MTU + AMT_GW_HLEN;
3264b9022b53STaehee Yoo 	}
3265b9022b53STaehee Yoo 	amt->qi = AMT_INIT_QUERY_INTERVAL;
3266b9022b53STaehee Yoo 
3267b9022b53STaehee Yoo 	err = register_netdevice(dev);
3268b9022b53STaehee Yoo 	if (err < 0) {
3269b9022b53STaehee Yoo 		netdev_dbg(dev, "failed to register new netdev %d\n", err);
3270b9022b53STaehee Yoo 		goto err;
3271b9022b53STaehee Yoo 	}
3272b9022b53STaehee Yoo 
3273b9022b53STaehee Yoo 	err = netdev_upper_dev_link(amt->stream_dev, dev, extack);
3274b9022b53STaehee Yoo 	if (err < 0) {
3275b9022b53STaehee Yoo 		unregister_netdevice(dev);
3276b9022b53STaehee Yoo 		goto err;
3277b9022b53STaehee Yoo 	}
3278b9022b53STaehee Yoo 
3279cbc21dc1STaehee Yoo 	INIT_DELAYED_WORK(&amt->discovery_wq, amt_discovery_work);
3280cbc21dc1STaehee Yoo 	INIT_DELAYED_WORK(&amt->req_wq, amt_req_work);
3281cbc21dc1STaehee Yoo 	INIT_DELAYED_WORK(&amt->secret_wq, amt_secret_work);
328230e22a6eSTaehee Yoo 	INIT_WORK(&amt->event_wq, amt_event_work);
3283cbc21dc1STaehee Yoo 	INIT_LIST_HEAD(&amt->tunnel_list);
3284b9022b53STaehee Yoo 	return 0;
3285b9022b53STaehee Yoo err:
3286b9022b53STaehee Yoo 	dev_put(amt->stream_dev);
3287b9022b53STaehee Yoo 	return err;
3288b9022b53STaehee Yoo }
3289b9022b53STaehee Yoo 
amt_dellink(struct net_device * dev,struct list_head * head)3290b9022b53STaehee Yoo static void amt_dellink(struct net_device *dev, struct list_head *head)
3291b9022b53STaehee Yoo {
3292b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
3293b9022b53STaehee Yoo 
3294b9022b53STaehee Yoo 	unregister_netdevice_queue(dev, head);
3295b9022b53STaehee Yoo 	netdev_upper_dev_unlink(amt->stream_dev, dev);
3296b9022b53STaehee Yoo 	dev_put(amt->stream_dev);
3297b9022b53STaehee Yoo }
3298b9022b53STaehee Yoo 
amt_get_size(const struct net_device * dev)3299b9022b53STaehee Yoo static size_t amt_get_size(const struct net_device *dev)
3300b9022b53STaehee Yoo {
3301b9022b53STaehee Yoo 	return nla_total_size(sizeof(__u32)) + /* IFLA_AMT_MODE */
3302b9022b53STaehee Yoo 	       nla_total_size(sizeof(__u16)) + /* IFLA_AMT_RELAY_PORT */
3303b9022b53STaehee Yoo 	       nla_total_size(sizeof(__u16)) + /* IFLA_AMT_GATEWAY_PORT */
3304b9022b53STaehee Yoo 	       nla_total_size(sizeof(__u32)) + /* IFLA_AMT_LINK */
3305b9022b53STaehee Yoo 	       nla_total_size(sizeof(__u32)) + /* IFLA_MAX_TUNNELS */
3306b9022b53STaehee Yoo 	       nla_total_size(sizeof(struct iphdr)) + /* IFLA_AMT_DISCOVERY_IP */
3307b9022b53STaehee Yoo 	       nla_total_size(sizeof(struct iphdr)) + /* IFLA_AMT_REMOTE_IP */
3308b9022b53STaehee Yoo 	       nla_total_size(sizeof(struct iphdr)); /* IFLA_AMT_LOCAL_IP */
3309b9022b53STaehee Yoo }
3310b9022b53STaehee Yoo 
amt_fill_info(struct sk_buff * skb,const struct net_device * dev)3311b9022b53STaehee Yoo static int amt_fill_info(struct sk_buff *skb, const struct net_device *dev)
3312b9022b53STaehee Yoo {
3313b9022b53STaehee Yoo 	struct amt_dev *amt = netdev_priv(dev);
3314b9022b53STaehee Yoo 
3315b9022b53STaehee Yoo 	if (nla_put_u32(skb, IFLA_AMT_MODE, amt->mode))
3316b9022b53STaehee Yoo 		goto nla_put_failure;
3317b9022b53STaehee Yoo 	if (nla_put_be16(skb, IFLA_AMT_RELAY_PORT, amt->relay_port))
3318b9022b53STaehee Yoo 		goto nla_put_failure;
3319b9022b53STaehee Yoo 	if (nla_put_be16(skb, IFLA_AMT_GATEWAY_PORT, amt->gw_port))
3320b9022b53STaehee Yoo 		goto nla_put_failure;
3321b9022b53STaehee Yoo 	if (nla_put_u32(skb, IFLA_AMT_LINK, amt->stream_dev->ifindex))
3322b9022b53STaehee Yoo 		goto nla_put_failure;
3323b9022b53STaehee Yoo 	if (nla_put_in_addr(skb, IFLA_AMT_LOCAL_IP, amt->local_ip))
3324b9022b53STaehee Yoo 		goto nla_put_failure;
3325b9022b53STaehee Yoo 	if (nla_put_in_addr(skb, IFLA_AMT_DISCOVERY_IP, amt->discovery_ip))
3326b9022b53STaehee Yoo 		goto nla_put_failure;
3327b9022b53STaehee Yoo 	if (amt->remote_ip)
3328b9022b53STaehee Yoo 		if (nla_put_in_addr(skb, IFLA_AMT_REMOTE_IP, amt->remote_ip))
3329b9022b53STaehee Yoo 			goto nla_put_failure;
3330b9022b53STaehee Yoo 	if (nla_put_u32(skb, IFLA_AMT_MAX_TUNNELS, amt->max_tunnels))
3331b9022b53STaehee Yoo 		goto nla_put_failure;
3332b9022b53STaehee Yoo 
3333b9022b53STaehee Yoo 	return 0;
3334b9022b53STaehee Yoo 
3335b9022b53STaehee Yoo nla_put_failure:
3336b9022b53STaehee Yoo 	return -EMSGSIZE;
3337b9022b53STaehee Yoo }
3338b9022b53STaehee Yoo 
3339b9022b53STaehee Yoo static struct rtnl_link_ops amt_link_ops __read_mostly = {
3340b9022b53STaehee Yoo 	.kind		= "amt",
3341b9022b53STaehee Yoo 	.maxtype	= IFLA_AMT_MAX,
3342b9022b53STaehee Yoo 	.policy		= amt_policy,
3343b9022b53STaehee Yoo 	.priv_size	= sizeof(struct amt_dev),
3344b9022b53STaehee Yoo 	.setup		= amt_link_setup,
3345b9022b53STaehee Yoo 	.validate	= amt_validate,
3346b9022b53STaehee Yoo 	.newlink	= amt_newlink,
3347b9022b53STaehee Yoo 	.dellink	= amt_dellink,
3348b9022b53STaehee Yoo 	.get_size       = amt_get_size,
3349b9022b53STaehee Yoo 	.fill_info      = amt_fill_info,
3350b9022b53STaehee Yoo };
3351b9022b53STaehee Yoo 
amt_lookup_upper_dev(struct net_device * dev)3352b9022b53STaehee Yoo static struct net_device *amt_lookup_upper_dev(struct net_device *dev)
3353b9022b53STaehee Yoo {
3354b9022b53STaehee Yoo 	struct net_device *upper_dev;
3355b9022b53STaehee Yoo 	struct amt_dev *amt;
3356b9022b53STaehee Yoo 
3357b9022b53STaehee Yoo 	for_each_netdev(dev_net(dev), upper_dev) {
3358b9022b53STaehee Yoo 		if (netif_is_amt(upper_dev)) {
3359b9022b53STaehee Yoo 			amt = netdev_priv(upper_dev);
3360b9022b53STaehee Yoo 			if (amt->stream_dev == dev)
3361b9022b53STaehee Yoo 				return upper_dev;
3362b9022b53STaehee Yoo 		}
3363b9022b53STaehee Yoo 	}
3364b9022b53STaehee Yoo 
3365b9022b53STaehee Yoo 	return NULL;
3366b9022b53STaehee Yoo }
3367b9022b53STaehee Yoo 
amt_device_event(struct notifier_block * unused,unsigned long event,void * ptr)3368b9022b53STaehee Yoo static int amt_device_event(struct notifier_block *unused,
3369b9022b53STaehee Yoo 			    unsigned long event, void *ptr)
3370b9022b53STaehee Yoo {
3371b9022b53STaehee Yoo 	struct net_device *dev = netdev_notifier_info_to_dev(ptr);
3372b9022b53STaehee Yoo 	struct net_device *upper_dev;
3373b9022b53STaehee Yoo 	struct amt_dev *amt;
3374b9022b53STaehee Yoo 	LIST_HEAD(list);
3375b9022b53STaehee Yoo 	int new_mtu;
3376b9022b53STaehee Yoo 
3377b9022b53STaehee Yoo 	upper_dev = amt_lookup_upper_dev(dev);
3378b9022b53STaehee Yoo 	if (!upper_dev)
3379b9022b53STaehee Yoo 		return NOTIFY_DONE;
3380b9022b53STaehee Yoo 	amt = netdev_priv(upper_dev);
3381b9022b53STaehee Yoo 
3382b9022b53STaehee Yoo 	switch (event) {
3383b9022b53STaehee Yoo 	case NETDEV_UNREGISTER:
3384b9022b53STaehee Yoo 		amt_dellink(amt->dev, &list);
3385b9022b53STaehee Yoo 		unregister_netdevice_many(&list);
3386b9022b53STaehee Yoo 		break;
3387b9022b53STaehee Yoo 	case NETDEV_CHANGEMTU:
3388b9022b53STaehee Yoo 		if (amt->mode == AMT_MODE_RELAY)
3389b9022b53STaehee Yoo 			new_mtu = dev->mtu - AMT_RELAY_HLEN;
3390b9022b53STaehee Yoo 		else
3391b9022b53STaehee Yoo 			new_mtu = dev->mtu - AMT_GW_HLEN;
3392b9022b53STaehee Yoo 
3393b9022b53STaehee Yoo 		dev_set_mtu(amt->dev, new_mtu);
3394b9022b53STaehee Yoo 		break;
3395b9022b53STaehee Yoo 	}
3396b9022b53STaehee Yoo 
3397b9022b53STaehee Yoo 	return NOTIFY_DONE;
3398b9022b53STaehee Yoo }
3399b9022b53STaehee Yoo 
3400b9022b53STaehee Yoo static struct notifier_block amt_notifier_block __read_mostly = {
3401b9022b53STaehee Yoo 	.notifier_call = amt_device_event,
3402b9022b53STaehee Yoo };
3403b9022b53STaehee Yoo 
amt_init(void)3404b9022b53STaehee Yoo static int __init amt_init(void)
3405b9022b53STaehee Yoo {
3406b9022b53STaehee Yoo 	int err;
3407b9022b53STaehee Yoo 
3408b9022b53STaehee Yoo 	err = register_netdevice_notifier(&amt_notifier_block);
3409b9022b53STaehee Yoo 	if (err < 0)
3410b9022b53STaehee Yoo 		goto err;
3411b9022b53STaehee Yoo 
3412b9022b53STaehee Yoo 	err = rtnl_link_register(&amt_link_ops);
3413b9022b53STaehee Yoo 	if (err < 0)
3414b9022b53STaehee Yoo 		goto unregister_notifier;
3415b9022b53STaehee Yoo 
341630e22a6eSTaehee Yoo 	amt_wq = alloc_workqueue("amt", WQ_UNBOUND, 0);
3417db243434SYang Yingliang 	if (!amt_wq) {
3418db243434SYang Yingliang 		err = -ENOMEM;
3419b9022b53STaehee Yoo 		goto rtnl_unregister;
3420db243434SYang Yingliang 	}
3421b9022b53STaehee Yoo 
3422bc54e49cSTaehee Yoo 	spin_lock_init(&source_gc_lock);
3423bc54e49cSTaehee Yoo 	spin_lock_bh(&source_gc_lock);
3424bc54e49cSTaehee Yoo 	INIT_DELAYED_WORK(&source_gc_wq, amt_source_gc_work);
3425bc54e49cSTaehee Yoo 	mod_delayed_work(amt_wq, &source_gc_wq,
3426bc54e49cSTaehee Yoo 			 msecs_to_jiffies(AMT_GC_INTERVAL));
3427bc54e49cSTaehee Yoo 	spin_unlock_bh(&source_gc_lock);
3428bc54e49cSTaehee Yoo 
3429b9022b53STaehee Yoo 	return 0;
3430b9022b53STaehee Yoo 
3431b9022b53STaehee Yoo rtnl_unregister:
3432b9022b53STaehee Yoo 	rtnl_link_unregister(&amt_link_ops);
3433b9022b53STaehee Yoo unregister_notifier:
3434b9022b53STaehee Yoo 	unregister_netdevice_notifier(&amt_notifier_block);
3435b9022b53STaehee Yoo err:
3436b9022b53STaehee Yoo 	pr_err("error loading AMT module loaded\n");
3437b9022b53STaehee Yoo 	return err;
3438b9022b53STaehee Yoo }
3439b9022b53STaehee Yoo late_initcall(amt_init);
3440b9022b53STaehee Yoo 
amt_fini(void)3441b9022b53STaehee Yoo static void __exit amt_fini(void)
3442b9022b53STaehee Yoo {
3443b9022b53STaehee Yoo 	rtnl_link_unregister(&amt_link_ops);
3444b9022b53STaehee Yoo 	unregister_netdevice_notifier(&amt_notifier_block);
3445b0024a04STaehee Yoo 	cancel_delayed_work_sync(&source_gc_wq);
3446bc54e49cSTaehee Yoo 	__amt_source_gc_work();
3447b9022b53STaehee Yoo 	destroy_workqueue(amt_wq);
3448b9022b53STaehee Yoo }
3449b9022b53STaehee Yoo module_exit(amt_fini);
3450b9022b53STaehee Yoo 
3451b9022b53STaehee Yoo MODULE_LICENSE("GPL");
3452b9022b53STaehee Yoo MODULE_AUTHOR("Taehee Yoo <ap420073@gmail.com>");
3453b9022b53STaehee Yoo MODULE_ALIAS_RTNL_LINK("amt");
3454