Blame view

net/ipv6/mcast.c 70.2 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
  /*
   *	Multicast support for IPv6
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
3
   *	Linux INET6 implementation
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
4
5
   *
   *	Authors:
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
6
   *	Pedro Roque		<roque@di.fc.ul.pt>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
7
   *
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
8
   *	Based on linux/ipv4/igmp.c and linux/ipv4/ip_sockglue.c
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
   *
   *	This program is free software; you can redistribute it and/or
   *      modify it under the terms of the GNU General Public License
   *      as published by the Free Software Foundation; either version
   *      2 of the License, or (at your option) any later version.
   */
  
  /* Changes:
   *
   *	yoshfuji	: fix format of router-alert option
   *	YOSHIFUJI Hideaki @USAGI:
   *		Fixed source address for MLD message based on
   *		<draft-ietf-magma-mld-source-05.txt>.
   *	YOSHIFUJI Hideaki @USAGI:
   *		- Ignore Queries for invalid addresses.
   *		- MLD for link-local addresses.
   *	David L Stevens <dlstevens@us.ibm.com>:
   *		- MLDv2 support
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
  #include <linux/module.h>
  #include <linux/errno.h>
  #include <linux/types.h>
  #include <linux/string.h>
  #include <linux/socket.h>
  #include <linux/sockios.h>
  #include <linux/jiffies.h>
  #include <linux/times.h>
  #include <linux/net.h>
  #include <linux/in.h>
  #include <linux/in6.h>
  #include <linux/netdevice.h>
  #include <linux/if_arp.h>
  #include <linux/route.h>
  #include <linux/init.h>
  #include <linux/proc_fs.h>
  #include <linux/seq_file.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
45
  #include <linux/slab.h>
9d4a03146   Hannes Frederic Sowa   ipv4, ipv6: send ...
46
  #include <linux/pkt_sched.h>
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
47
  #include <net/mld.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
48
49
50
  
  #include <linux/netfilter.h>
  #include <linux/netfilter_ipv6.h>
457c4cbc5   Eric W. Biederman   [NET]: Make /proc...
51
  #include <net/net_namespace.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
52
53
54
55
56
57
58
59
60
  #include <net/sock.h>
  #include <net/snmp.h>
  
  #include <net/ipv6.h>
  #include <net/protocol.h>
  #include <net/if_inet6.h>
  #include <net/ndisc.h>
  #include <net/addrconf.h>
  #include <net/ip6_route.h>
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
61
  #include <net/inet_common.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
62
63
  
  #include <net/ip6_checksum.h>
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
64
65
66
67
68
  /* Ensure that we have struct in6_addr aligned on 32bit word. */
  static void *__mld2_query_bugs[] __attribute__((__unused__)) = {
  	BUILD_BUG_ON_NULL(offsetof(struct mld2_query, mld2q_srcs) % 4),
  	BUILD_BUG_ON_NULL(offsetof(struct mld2_report, mld2r_grec) % 4),
  	BUILD_BUG_ON_NULL(offsetof(struct mld2_grec, grec_mca) % 4)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
69
70
71
  };
  
  static struct in6_addr mld2_all_mcr = MLD2_ALL_MCR_INIT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
72
73
74
75
76
77
78
79
  static void igmp6_join_group(struct ifmcaddr6 *ma);
  static void igmp6_leave_group(struct ifmcaddr6 *ma);
  static void igmp6_timer_handler(unsigned long data);
  
  static void mld_gq_timer_expire(unsigned long data);
  static void mld_ifc_timer_expire(unsigned long data);
  static void mld_ifc_event(struct inet6_dev *idev);
  static void mld_add_delrec(struct inet6_dev *idev, struct ifmcaddr6 *pmc);
b71d1d426   Eric Dumazet   inet: constify ip...
80
  static void mld_del_delrec(struct inet6_dev *idev, const struct in6_addr *addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
81
  static void mld_clear_delrec(struct inet6_dev *idev);
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
82
  static bool mld_in_v1_mode(const struct inet6_dev *idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
83
84
85
  static int sf_setstate(struct ifmcaddr6 *pmc);
  static void sf_markstate(struct ifmcaddr6 *pmc);
  static void ip6_mc_clear_src(struct ifmcaddr6 *pmc);
b71d1d426   Eric Dumazet   inet: constify ip...
86
87
  static int ip6_mc_del_src(struct inet6_dev *idev, const struct in6_addr *pmca,
  			  int sfmode, int sfcount, const struct in6_addr *psfsrc,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
88
  			  int delta);
b71d1d426   Eric Dumazet   inet: constify ip...
89
90
  static int ip6_mc_add_src(struct inet6_dev *idev, const struct in6_addr *pmca,
  			  int sfmode, int sfcount, const struct in6_addr *psfsrc,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
91
92
93
  			  int delta);
  static int ip6_mc_leave_src(struct sock *sk, struct ipv6_mc_socklist *iml,
  			    struct inet6_dev *idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
94
  #define MLD_QRV_DEFAULT		2
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
95
96
97
98
  /* RFC3810, 9.2. Query Interval */
  #define MLD_QI_DEFAULT		(125 * HZ)
  /* RFC3810, 9.3. Query Response Interval */
  #define MLD_QRI_DEFAULT		(10 * HZ)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
99

9fd078416   Daniel Borkmann   net: ipv6: mcast:...
100
101
102
  /* RFC3810, 8.1 Query Version Distinctions */
  #define MLD_V1_QUERY_LEN	24
  #define MLD_V2_QUERY_LEN_MIN	28
6f4353d89   David L Stevens   [IPV6]: Increase ...
103
  #define IPV6_MLD_MAX_MSF	64
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
104

ab32ea5d8   Brian Haley   [NET/IPV4/IPV6]: ...
105
  int sysctl_mld_max_msf __read_mostly = IPV6_MLD_MAX_MSF;
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
106
  int sysctl_mld_qrv __read_mostly = MLD_QRV_DEFAULT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
107
108
109
110
  
  /*
   *	socket join on multicast group
   */
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
111
112
113
114
  #define for_each_pmc_rcu(np, pmc)				\
  	for (pmc = rcu_dereference(np->ipv6_mc_list);		\
  	     pmc != NULL;					\
  	     pmc = rcu_dereference(pmc->next))
fc4eba58b   Hannes Frederic Sowa   ipv6: make unsoli...
115
116
117
  static int unsolicited_report_interval(struct inet6_dev *idev)
  {
  	int iv;
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
118
  	if (mld_in_v1_mode(idev))
fc4eba58b   Hannes Frederic Sowa   ipv6: make unsoli...
119
120
121
122
123
124
  		iv = idev->cnf.mldv1_unsolicited_report_interval;
  	else
  		iv = idev->cnf.mldv2_unsolicited_report_interval;
  
  	return iv > 0 ? iv : 1;
  }
54ff9ef36   Marcelo Ricardo Leitner   ipv4, ipv6: kill ...
125
  int ipv6_sock_mc_join(struct sock *sk, int ifindex, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
127
128
129
  {
  	struct net_device *dev = NULL;
  	struct ipv6_mc_socklist *mc_lst;
  	struct ipv6_pinfo *np = inet6_sk(sk);
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
130
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
131
  	int err;
46a4dee07   Madhu Challa   igmp v6: add __ip...
132
  	ASSERT_RTNL();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
133
134
  	if (!ipv6_addr_is_multicast(addr))
  		return -EINVAL;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
135
136
  	rcu_read_lock();
  	for_each_pmc_rcu(np, mc_lst) {
c9e3e8b69   David L Stevens   [IPV6]: multicast...
137
138
  		if ((ifindex == 0 || mc_lst->ifindex == ifindex) &&
  		    ipv6_addr_equal(&mc_lst->addr, addr)) {
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
139
  			rcu_read_unlock();
c9e3e8b69   David L Stevens   [IPV6]: multicast...
140
141
142
  			return -EADDRINUSE;
  		}
  	}
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
143
  	rcu_read_unlock();
c9e3e8b69   David L Stevens   [IPV6]: multicast...
144

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
145
  	mc_lst = sock_kmalloc(sk, sizeof(struct ipv6_mc_socklist), GFP_KERNEL);
63159f29b   Ian Morris   ipv6: coding styl...
146
  	if (!mc_lst)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
147
148
149
  		return -ENOMEM;
  
  	mc_lst->next = NULL;
4e3fd7a06   Alexey Dobriyan   net: remove ipv6_...
150
  	mc_lst->addr = *addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
151
152
153
  
  	if (ifindex == 0) {
  		struct rt6_info *rt;
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
154
  		rt = rt6_lookup(net, addr, NULL, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
155
  		if (rt) {
d19185428   David S. Miller   ipv6: Kill rt6i_d...
156
  			dev = rt->dst.dev;
94e187c01   Amerigo Wang   ipv6: introduce i...
157
  			ip6_rt_put(rt);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
158
159
  		}
  	} else
414b6c943   WANG Cong   ipv6: drop some r...
160
  		dev = __dev_get_by_index(net, ifindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
161

63159f29b   Ian Morris   ipv6: coding styl...
162
  	if (!dev) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
163
164
165
166
167
168
  		sock_kfree_s(sk, mc_lst, sizeof(*mc_lst));
  		return -ENODEV;
  	}
  
  	mc_lst->ifindex = dev->ifindex;
  	mc_lst->sfmode = MCAST_EXCLUDE;
196433c5b   YOSHIFUJI Hideaki   [IPV6]: Use macro...
169
  	rwlock_init(&mc_lst->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
170
171
172
173
174
175
176
177
178
179
  	mc_lst->sflist = NULL;
  
  	/*
  	 *	now add/increase the group membership on the device
  	 */
  
  	err = ipv6_dev_mc_inc(dev, addr);
  
  	if (err) {
  		sock_kfree_s(sk, mc_lst, sizeof(*mc_lst));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
180
181
  		return err;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
182
  	mc_lst->next = np->ipv6_mc_list;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
183
  	rcu_assign_pointer(np->ipv6_mc_list, mc_lst);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
184

46a4dee07   Madhu Challa   igmp v6: add __ip...
185
186
  	return 0;
  }
46a4dee07   Madhu Challa   igmp v6: add __ip...
187
  EXPORT_SYMBOL(ipv6_sock_mc_join);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
188
189
190
191
  
  /*
   *	socket leave on multicast group
   */
54ff9ef36   Marcelo Ricardo Leitner   ipv4, ipv6: kill ...
192
  int ipv6_sock_mc_drop(struct sock *sk, int ifindex, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
193
194
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
195
196
  	struct ipv6_mc_socklist *mc_lst;
  	struct ipv6_mc_socklist __rcu **lnk;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
197
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
198

46a4dee07   Madhu Challa   igmp v6: add __ip...
199
  	ASSERT_RTNL();
a858d64b7   Li Wei   ipv6: fix unappro...
200
201
  	if (!ipv6_addr_is_multicast(addr))
  		return -EINVAL;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
202
  	for (lnk = &np->ipv6_mc_list;
b5350916b   WANG Cong   ipv6: drop ipv6_s...
203
  	     (mc_lst = rtnl_dereference(*lnk)) != NULL;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
204
  	      lnk = &mc_lst->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
205
206
207
208
209
  		if ((ifindex == 0 || mc_lst->ifindex == ifindex) &&
  		    ipv6_addr_equal(&mc_lst->addr, addr)) {
  			struct net_device *dev;
  
  			*lnk = mc_lst->next;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
210

414b6c943   WANG Cong   ipv6: drop some r...
211
  			dev = __dev_get_by_index(net, mc_lst->ifindex);
53b24b8f9   Ian Morris   ipv6: coding styl...
212
  			if (dev) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
213
  				struct inet6_dev *idev = __in6_dev_get(dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
214

acd6e00b8   David L Stevens   [MCAST]: Fix filt...
215
  				(void) ip6_mc_leave_src(sk, mc_lst, idev);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
216
  				if (idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
217
  					__ipv6_dev_mc_dec(idev, &mc_lst->addr);
acd6e00b8   David L Stevens   [MCAST]: Fix filt...
218
219
  			} else
  				(void) ip6_mc_leave_src(sk, mc_lst, NULL);
a9ed4a298   Sabrina Dubroca   ipv6: fix rtnl lo...
220

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
221
  			atomic_sub(sizeof(*mc_lst), &sk->sk_omem_alloc);
e3cbf28fa   Lai Jiangshan   net,rcu: convert ...
222
  			kfree_rcu(mc_lst, rcu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
223
224
225
  			return 0;
  		}
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
226

9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
227
  	return -EADDRNOTAVAIL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
228
  }
46a4dee07   Madhu Challa   igmp v6: add __ip...
229
  EXPORT_SYMBOL(ipv6_sock_mc_drop);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
230

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
231
232
  /* called with rcu_read_lock() */
  static struct inet6_dev *ip6_mc_find_dev_rcu(struct net *net,
b71d1d426   Eric Dumazet   inet: constify ip...
233
  					     const struct in6_addr *group,
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
234
  					     int ifindex)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
235
236
237
238
239
  {
  	struct net_device *dev = NULL;
  	struct inet6_dev *idev = NULL;
  
  	if (ifindex == 0) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
240
  		struct rt6_info *rt = rt6_lookup(net, group, NULL, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
241

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
242
  		if (rt) {
d19185428   David S. Miller   ipv6: Kill rt6i_d...
243
  			dev = rt->dst.dev;
94e187c01   Amerigo Wang   ipv6: introduce i...
244
  			ip6_rt_put(rt);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
245
246
  		}
  	} else
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
247
  		dev = dev_get_by_index_rcu(net, ifindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
248
249
  
  	if (!dev)
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
250
251
  		return NULL;
  	idev = __in6_dev_get(dev);
448eb71f4   Ilpo Järvinen   ipv6/mcast: join ...
252
  	if (!idev)
8a22c99a8   Joe Perches   net/ipv6/mcast.c:...
253
  		return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254
  	read_lock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
255
256
257
258
  	if (idev->dead) {
  		read_unlock_bh(&idev->lock);
  		return NULL;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
259
260
  	return idev;
  }
8651be8f1   WANG Cong   ipv6: fix a poten...
261
  void __ipv6_sock_mc_close(struct sock *sk)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
262
263
264
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
  	struct ipv6_mc_socklist *mc_lst;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
265
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
266

8651be8f1   WANG Cong   ipv6: fix a poten...
267
  	ASSERT_RTNL();
0e1efe9d5   Eric Dumazet   ipv6: avoid takin...
268

b5350916b   WANG Cong   ipv6: drop ipv6_s...
269
  	while ((mc_lst = rtnl_dereference(np->ipv6_mc_list)) != NULL) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
270
271
272
  		struct net_device *dev;
  
  		np->ipv6_mc_list = mc_lst->next;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
273

414b6c943   WANG Cong   ipv6: drop some r...
274
  		dev = __dev_get_by_index(net, mc_lst->ifindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
275
  		if (dev) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
276
  			struct inet6_dev *idev = __in6_dev_get(dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
277

acd6e00b8   David L Stevens   [MCAST]: Fix filt...
278
  			(void) ip6_mc_leave_src(sk, mc_lst, idev);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
279
  			if (idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
280
  				__ipv6_dev_mc_dec(idev, &mc_lst->addr);
acd6e00b8   David L Stevens   [MCAST]: Fix filt...
281
282
  		} else
  			(void) ip6_mc_leave_src(sk, mc_lst, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
283

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
284
  		atomic_sub(sizeof(*mc_lst), &sk->sk_omem_alloc);
e3cbf28fa   Lai Jiangshan   net,rcu: convert ...
285
  		kfree_rcu(mc_lst, rcu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
286
  	}
8651be8f1   WANG Cong   ipv6: fix a poten...
287
288
289
290
291
292
293
294
295
296
  }
  
  void ipv6_sock_mc_close(struct sock *sk)
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
  
  	if (!rcu_access_pointer(np->ipv6_mc_list))
  		return;
  	rtnl_lock();
  	__ipv6_sock_mc_close(sk);
a9ed4a298   Sabrina Dubroca   ipv6: fix rtnl lo...
297
  	rtnl_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
298
299
300
301
302
303
304
  }
  
  int ip6_mc_source(int add, int omode, struct sock *sk,
  	struct group_source_req *pgsr)
  {
  	struct in6_addr *source, *group;
  	struct ipv6_mc_socklist *pmc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
305
306
307
  	struct inet6_dev *idev;
  	struct ipv6_pinfo *inet6 = inet6_sk(sk);
  	struct ip6_sf_socklist *psl;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
308
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
309
  	int i, j, rv;
c9e3e8b69   David L Stevens   [IPV6]: multicast...
310
  	int leavegroup = 0;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
311
  	int pmclocked = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
312
  	int err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
313
314
315
316
317
  	source = &((struct sockaddr_in6 *)&pgsr->gsr_source)->sin6_addr;
  	group = &((struct sockaddr_in6 *)&pgsr->gsr_group)->sin6_addr;
  
  	if (!ipv6_addr_is_multicast(group))
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
318
319
320
321
  	rcu_read_lock();
  	idev = ip6_mc_find_dev_rcu(net, group, pgsr->gsr_interface);
  	if (!idev) {
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
322
  		return -ENODEV;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
323
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
324
325
  
  	err = -EADDRNOTAVAIL;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
326
  	for_each_pmc_rcu(inet6, pmc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
327
328
329
330
331
  		if (pgsr->gsr_interface && pmc->ifindex != pgsr->gsr_interface)
  			continue;
  		if (ipv6_addr_equal(&pmc->addr, group))
  			break;
  	}
917f2f105   David L Stevens   [IPV4]: multicast...
332
333
  	if (!pmc) {		/* must have a prior join */
  		err = -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
334
  		goto done;
917f2f105   David L Stevens   [IPV4]: multicast...
335
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
336
337
  	/* if a source filter was set, must be the same mode as before */
  	if (pmc->sflist) {
917f2f105   David L Stevens   [IPV4]: multicast...
338
339
  		if (pmc->sfmode != omode) {
  			err = -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
340
  			goto done;
917f2f105   David L Stevens   [IPV4]: multicast...
341
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
342
343
344
345
346
347
  	} else if (pmc->sfmode != omode) {
  		/* allow mode switches for empty-set filters */
  		ip6_mc_add_src(idev, group, omode, 0, NULL, 0);
  		ip6_mc_del_src(idev, group, pmc->sfmode, 0, NULL, 0);
  		pmc->sfmode = omode;
  	}
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
348
  	write_lock(&pmc->sflock);
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
349
  	pmclocked = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
350
351
352
  	psl = pmc->sflist;
  	if (!add) {
  		if (!psl)
917f2f105   David L Stevens   [IPV4]: multicast...
353
  			goto done;	/* err = -EADDRNOTAVAIL */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
354
  		rv = !0;
67ba4152e   Ian Morris   ipv6: White-space...
355
  		for (i = 0; i < psl->sl_count; i++) {
07c2fecc3   YOSHIFUJI Hideaki / 吉藤英明   ipv6 mcast: Use i...
356
  			rv = !ipv6_addr_equal(&psl->sl_addr[i], source);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
357
358
359
360
  			if (rv == 0)
  				break;
  		}
  		if (rv)		/* source not found */
917f2f105   David L Stevens   [IPV4]: multicast...
361
  			goto done;	/* err = -EADDRNOTAVAIL */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
362

c9e3e8b69   David L Stevens   [IPV6]: multicast...
363
364
365
366
367
  		/* special case - (INCLUDE, empty) == LEAVE_GROUP */
  		if (psl->sl_count == 1 && omode == MCAST_INCLUDE) {
  			leavegroup = 1;
  			goto done;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
368
369
  		/* update the interface filter */
  		ip6_mc_del_src(idev, group, omode, 1, source, 1);
67ba4152e   Ian Morris   ipv6: White-space...
370
  		for (j = i+1; j < psl->sl_count; j++)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
  			psl->sl_addr[j-1] = psl->sl_addr[j];
  		psl->sl_count--;
  		err = 0;
  		goto done;
  	}
  	/* else, add a new source to the filter */
  
  	if (psl && psl->sl_count >= sysctl_mld_max_msf) {
  		err = -ENOBUFS;
  		goto done;
  	}
  	if (!psl || psl->sl_count == psl->sl_max) {
  		struct ip6_sf_socklist *newpsl;
  		int count = IP6_SFBLOCK;
  
  		if (psl)
  			count += psl->sl_max;
8b3a70058   Kris Katterjohn   [NET]: Remove mor...
388
  		newpsl = sock_kmalloc(sk, IP6_SFLSIZE(count), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
389
390
391
392
393
394
395
  		if (!newpsl) {
  			err = -ENOBUFS;
  			goto done;
  		}
  		newpsl->sl_max = count;
  		newpsl->sl_count = count - IP6_SFBLOCK;
  		if (psl) {
67ba4152e   Ian Morris   ipv6: White-space...
396
  			for (i = 0; i < psl->sl_count; i++)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
397
398
399
400
401
402
  				newpsl->sl_addr[i] = psl->sl_addr[i];
  			sock_kfree_s(sk, psl, IP6_SFLSIZE(psl->sl_max));
  		}
  		pmc->sflist = psl = newpsl;
  	}
  	rv = 1;	/* > 0 for insert logic below if sl_count is 0 */
67ba4152e   Ian Morris   ipv6: White-space...
403
  	for (i = 0; i < psl->sl_count; i++) {
07c2fecc3   YOSHIFUJI Hideaki / 吉藤英明   ipv6 mcast: Use i...
404
  		rv = !ipv6_addr_equal(&psl->sl_addr[i], source);
56db1c5f4   Jean Sacren   mcast: do not che...
405
406
  		if (rv == 0) /* There is an error in the address. */
  			goto done;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
407
  	}
67ba4152e   Ian Morris   ipv6: White-space...
408
  	for (j = psl->sl_count-1; j >= i; j--)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
409
410
411
412
413
414
415
  		psl->sl_addr[j+1] = psl->sl_addr[j];
  	psl->sl_addr[i] = *source;
  	psl->sl_count++;
  	err = 0;
  	/* update the interface list */
  	ip6_mc_add_src(idev, group, omode, 1, source, 1);
  done:
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
416
  	if (pmclocked)
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
417
  		write_unlock(&pmc->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
418
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
419
  	rcu_read_unlock();
c9e3e8b69   David L Stevens   [IPV6]: multicast...
420
  	if (leavegroup)
54ff9ef36   Marcelo Ricardo Leitner   ipv4, ipv6: kill ...
421
  		err = ipv6_sock_mc_drop(sk, pgsr->gsr_interface, group);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
422
423
424
425
426
  	return err;
  }
  
  int ip6_mc_msfilter(struct sock *sk, struct group_filter *gsf)
  {
b71d1d426   Eric Dumazet   inet: constify ip...
427
  	const struct in6_addr *group;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
428
  	struct ipv6_mc_socklist *pmc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
429
430
431
  	struct inet6_dev *idev;
  	struct ipv6_pinfo *inet6 = inet6_sk(sk);
  	struct ip6_sf_socklist *newpsl, *psl;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
432
  	struct net *net = sock_net(sk);
9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
433
  	int leavegroup = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
434
435
436
437
438
439
440
441
442
  	int i, err;
  
  	group = &((struct sockaddr_in6 *)&gsf->gf_group)->sin6_addr;
  
  	if (!ipv6_addr_is_multicast(group))
  		return -EINVAL;
  	if (gsf->gf_fmode != MCAST_INCLUDE &&
  	    gsf->gf_fmode != MCAST_EXCLUDE)
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
443
444
  	rcu_read_lock();
  	idev = ip6_mc_find_dev_rcu(net, group, gsf->gf_interface);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
445

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
446
447
  	if (!idev) {
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
448
  		return -ENODEV;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
449
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
450

9c05989bb   David S. Miller   [IPV6]: Fix warni...
451
  	err = 0;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
452

9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
453
454
455
456
  	if (gsf->gf_fmode == MCAST_INCLUDE && gsf->gf_numsrc == 0) {
  		leavegroup = 1;
  		goto done;
  	}
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
457
  	for_each_pmc_rcu(inet6, pmc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
458
459
460
461
462
  		if (pmc->ifindex != gsf->gf_interface)
  			continue;
  		if (ipv6_addr_equal(&pmc->addr, group))
  			break;
  	}
917f2f105   David L Stevens   [IPV4]: multicast...
463
464
  	if (!pmc) {		/* must have a prior join */
  		err = -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
465
  		goto done;
917f2f105   David L Stevens   [IPV4]: multicast...
466
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
467
  	if (gsf->gf_numsrc) {
8b3a70058   Kris Katterjohn   [NET]: Remove mor...
468
469
  		newpsl = sock_kmalloc(sk, IP6_SFLSIZE(gsf->gf_numsrc),
  							  GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
470
471
472
473
474
  		if (!newpsl) {
  			err = -ENOBUFS;
  			goto done;
  		}
  		newpsl->sl_max = newpsl->sl_count = gsf->gf_numsrc;
67ba4152e   Ian Morris   ipv6: White-space...
475
  		for (i = 0; i < newpsl->sl_count; ++i) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
476
477
478
479
480
481
482
483
484
485
486
  			struct sockaddr_in6 *psin6;
  
  			psin6 = (struct sockaddr_in6 *)&gsf->gf_slist[i];
  			newpsl->sl_addr[i] = psin6->sin6_addr;
  		}
  		err = ip6_mc_add_src(idev, group, gsf->gf_fmode,
  			newpsl->sl_count, newpsl->sl_addr, 0);
  		if (err) {
  			sock_kfree_s(sk, newpsl, IP6_SFLSIZE(newpsl->sl_max));
  			goto done;
  		}
8713dbf05   Yan Zheng   [MCAST]: ip[6]_mc...
487
  	} else {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
488
  		newpsl = NULL;
8713dbf05   Yan Zheng   [MCAST]: ip[6]_mc...
489
490
  		(void) ip6_mc_add_src(idev, group, gsf->gf_fmode, 0, NULL, 0);
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
491

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
492
  	write_lock(&pmc->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
493
494
495
496
497
498
499
500
501
  	psl = pmc->sflist;
  	if (psl) {
  		(void) ip6_mc_del_src(idev, group, pmc->sfmode,
  			psl->sl_count, psl->sl_addr, 0);
  		sock_kfree_s(sk, psl, IP6_SFLSIZE(psl->sl_max));
  	} else
  		(void) ip6_mc_del_src(idev, group, pmc->sfmode, 0, NULL, 0);
  	pmc->sflist = newpsl;
  	pmc->sfmode = gsf->gf_fmode;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
502
  	write_unlock(&pmc->sflock);
917f2f105   David L Stevens   [IPV4]: multicast...
503
  	err = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
504
505
  done:
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
506
  	rcu_read_unlock();
9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
507
508
  	if (leavegroup)
  		err = ipv6_sock_mc_drop(sk, gsf->gf_interface, group);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
509
510
511
512
513
514
515
  	return err;
  }
  
  int ip6_mc_msfget(struct sock *sk, struct group_filter *gsf,
  	struct group_filter __user *optval, int __user *optlen)
  {
  	int err, i, count, copycount;
b71d1d426   Eric Dumazet   inet: constify ip...
516
  	const struct in6_addr *group;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
517
518
  	struct ipv6_mc_socklist *pmc;
  	struct inet6_dev *idev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
519
520
  	struct ipv6_pinfo *inet6 = inet6_sk(sk);
  	struct ip6_sf_socklist *psl;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
521
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
522
523
524
525
526
  
  	group = &((struct sockaddr_in6 *)&gsf->gf_group)->sin6_addr;
  
  	if (!ipv6_addr_is_multicast(group))
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
527
528
  	rcu_read_lock();
  	idev = ip6_mc_find_dev_rcu(net, group, gsf->gf_interface);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
529

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
530
531
  	if (!idev) {
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
532
  		return -ENODEV;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
533
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
534
535
  
  	err = -EADDRNOTAVAIL;
f7ed925c1   WANG Cong   ipv6: update the ...
536
537
  	/* changes to the ipv6_mc_list require the socket lock and
  	 * rtnl lock. We have the socket lock and rcu read lock,
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
538
539
  	 * so reading the list is safe.
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
540

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
541
  	for_each_pmc_rcu(inet6, pmc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
542
543
544
545
546
547
548
549
550
551
552
  		if (pmc->ifindex != gsf->gf_interface)
  			continue;
  		if (ipv6_addr_equal(group, &pmc->addr))
  			break;
  	}
  	if (!pmc)		/* must have a prior join */
  		goto done;
  	gsf->gf_fmode = pmc->sfmode;
  	psl = pmc->sflist;
  	count = psl ? psl->sl_count : 0;
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
553
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
554
555
556
557
558
559
560
  
  	copycount = count < gsf->gf_numsrc ? count : gsf->gf_numsrc;
  	gsf->gf_numsrc = count;
  	if (put_user(GROUP_FILTER_SIZE(copycount), optlen) ||
  	    copy_to_user(optval, gsf, GROUP_FILTER_SIZE(0))) {
  		return -EFAULT;
  	}
f7ed925c1   WANG Cong   ipv6: update the ...
561
562
  	/* changes to psl require the socket lock, and a write lock
  	 * on pmc->sflock. We have the socket lock so reading here is safe.
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
563
  	 */
67ba4152e   Ian Morris   ipv6: White-space...
564
  	for (i = 0; i < copycount; i++) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
565
566
567
568
569
570
571
  		struct sockaddr_in6 *psin6;
  		struct sockaddr_storage ss;
  
  		psin6 = (struct sockaddr_in6 *)&ss;
  		memset(&ss, 0, sizeof(ss));
  		psin6->sin6_family = AF_INET6;
  		psin6->sin6_addr = psl->sl_addr[i];
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
572
  		if (copy_to_user(&optval->gf_slist[i], &ss, sizeof(ss)))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
573
574
575
576
577
  			return -EFAULT;
  	}
  	return 0;
  done:
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
578
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
579
580
  	return err;
  }
a50feda54   Eric Dumazet   ipv6: bool/const ...
581
582
  bool inet6_mc_check(struct sock *sk, const struct in6_addr *mc_addr,
  		    const struct in6_addr *src_addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
583
584
585
586
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
  	struct ipv6_mc_socklist *mc;
  	struct ip6_sf_socklist *psl;
a50feda54   Eric Dumazet   ipv6: bool/const ...
587
  	bool rv = true;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
588

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
589
590
  	rcu_read_lock();
  	for_each_pmc_rcu(np, mc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
591
592
593
594
  		if (ipv6_addr_equal(&mc->addr, mc_addr))
  			break;
  	}
  	if (!mc) {
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
595
  		rcu_read_unlock();
a50feda54   Eric Dumazet   ipv6: bool/const ...
596
  		return true;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
597
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
598
  	read_lock(&mc->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
599
600
601
602
603
  	psl = mc->sflist;
  	if (!psl) {
  		rv = mc->sfmode == MCAST_EXCLUDE;
  	} else {
  		int i;
67ba4152e   Ian Morris   ipv6: White-space...
604
  		for (i = 0; i < psl->sl_count; i++) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
605
606
607
608
  			if (ipv6_addr_equal(&psl->sl_addr[i], src_addr))
  				break;
  		}
  		if (mc->sfmode == MCAST_INCLUDE && i >= psl->sl_count)
a50feda54   Eric Dumazet   ipv6: bool/const ...
609
  			rv = false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
610
  		if (mc->sfmode == MCAST_EXCLUDE && i < psl->sl_count)
a50feda54   Eric Dumazet   ipv6: bool/const ...
611
  			rv = false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
612
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
613
  	read_unlock(&mc->sflock);
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
614
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
615
616
617
  
  	return rv;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
618
619
620
621
  static void igmp6_group_added(struct ifmcaddr6 *mc)
  {
  	struct net_device *dev = mc->idev->dev;
  	char buf[MAX_ADDR_LEN];
ec16ef222   YOSHIFUJI Hideaki / 吉藤英明   ipv6 mcast: Do no...
622
623
624
  	if (IPV6_ADDR_MC_SCOPE(&mc->mca_addr) <
  	    IPV6_ADDR_SCOPE_LINKLOCAL)
  		return;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
625
626
627
628
  	spin_lock_bh(&mc->mca_lock);
  	if (!(mc->mca_flags&MAF_LOADED)) {
  		mc->mca_flags |= MAF_LOADED;
  		if (ndisc_mc_map(&mc->mca_addr, buf, dev, 0) == 0)
22bedad3c   Jiri Pirko   net: convert mult...
629
  			dev_mc_add(dev, buf);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
630
631
632
633
634
  	}
  	spin_unlock_bh(&mc->mca_lock);
  
  	if (!(dev->flags & IFF_UP) || (mc->mca_flags & MAF_NOREPORT))
  		return;
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
635
  	if (mld_in_v1_mode(mc->idev)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
636
637
638
639
640
641
642
643
644
645
646
647
648
  		igmp6_join_group(mc);
  		return;
  	}
  	/* else v2 */
  
  	mc->mca_crcount = mc->idev->mc_qrv;
  	mld_ifc_event(mc->idev);
  }
  
  static void igmp6_group_dropped(struct ifmcaddr6 *mc)
  {
  	struct net_device *dev = mc->idev->dev;
  	char buf[MAX_ADDR_LEN];
ec16ef222   YOSHIFUJI Hideaki / 吉藤英明   ipv6 mcast: Do no...
649
650
651
  	if (IPV6_ADDR_MC_SCOPE(&mc->mca_addr) <
  	    IPV6_ADDR_SCOPE_LINKLOCAL)
  		return;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
652
653
654
655
  	spin_lock_bh(&mc->mca_lock);
  	if (mc->mca_flags&MAF_LOADED) {
  		mc->mca_flags &= ~MAF_LOADED;
  		if (ndisc_mc_map(&mc->mca_addr, buf, dev, 0) == 0)
22bedad3c   Jiri Pirko   net: convert mult...
656
  			dev_mc_del(dev, buf);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
  	}
  
  	if (mc->mca_flags & MAF_NOREPORT)
  		goto done;
  	spin_unlock_bh(&mc->mca_lock);
  
  	if (!mc->idev->dead)
  		igmp6_leave_group(mc);
  
  	spin_lock_bh(&mc->mca_lock);
  	if (del_timer(&mc->mca_timer))
  		atomic_dec(&mc->mca_refcnt);
  done:
  	ip6_mc_clear_src(mc);
  	spin_unlock_bh(&mc->mca_lock);
  }
  
  /*
   * deleted ifmcaddr6 manipulation
   */
  static void mld_add_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im)
  {
  	struct ifmcaddr6 *pmc;
  
  	/* this is an "ifmcaddr6" for convenience; only the fields below
  	 * are actually used. In particular, the refcnt and users are not
  	 * used for management of the delete list. Using the same structure
  	 * for deleted items allows change reports to use common code with
  	 * non-deleted or query-response MCA's.
  	 */
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
687
  	pmc = kzalloc(sizeof(*pmc), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
688
689
  	if (!pmc)
  		return;
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
690

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
691
692
693
694
695
696
697
698
699
700
701
702
703
  	spin_lock_bh(&im->mca_lock);
  	spin_lock_init(&pmc->mca_lock);
  	pmc->idev = im->idev;
  	in6_dev_hold(idev);
  	pmc->mca_addr = im->mca_addr;
  	pmc->mca_crcount = idev->mc_qrv;
  	pmc->mca_sfmode = im->mca_sfmode;
  	if (pmc->mca_sfmode == MCAST_INCLUDE) {
  		struct ip6_sf_list *psf;
  
  		pmc->mca_tomb = im->mca_tomb;
  		pmc->mca_sources = im->mca_sources;
  		im->mca_tomb = im->mca_sources = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
704
  		for (psf = pmc->mca_sources; psf; psf = psf->sf_next)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
705
706
707
  			psf->sf_crcount = pmc->mca_crcount;
  	}
  	spin_unlock_bh(&im->mca_lock);
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
708
  	spin_lock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
709
710
  	pmc->next = idev->mc_tomb;
  	idev->mc_tomb = pmc;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
711
  	spin_unlock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
712
  }
b71d1d426   Eric Dumazet   inet: constify ip...
713
  static void mld_del_delrec(struct inet6_dev *idev, const struct in6_addr *pmca)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
714
715
716
  {
  	struct ifmcaddr6 *pmc, *pmc_prev;
  	struct ip6_sf_list *psf, *psf_next;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
717
  	spin_lock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
718
  	pmc_prev = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
719
  	for (pmc = idev->mc_tomb; pmc; pmc = pmc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
720
721
722
723
724
725
726
727
728
729
  		if (ipv6_addr_equal(&pmc->mca_addr, pmca))
  			break;
  		pmc_prev = pmc;
  	}
  	if (pmc) {
  		if (pmc_prev)
  			pmc_prev->next = pmc->next;
  		else
  			idev->mc_tomb = pmc->next;
  	}
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
730
  	spin_unlock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
731
  	if (pmc) {
67ba4152e   Ian Morris   ipv6: White-space...
732
  		for (psf = pmc->mca_tomb; psf; psf = psf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
733
734
735
736
737
738
739
740
741
742
743
  			psf_next = psf->sf_next;
  			kfree(psf);
  		}
  		in6_dev_put(pmc->idev);
  		kfree(pmc);
  	}
  }
  
  static void mld_clear_delrec(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *pmc, *nextpmc;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
744
  	spin_lock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
745
746
  	pmc = idev->mc_tomb;
  	idev->mc_tomb = NULL;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
747
  	spin_unlock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
748
749
750
751
752
753
754
755
756
757
  
  	for (; pmc; pmc = nextpmc) {
  		nextpmc = pmc->next;
  		ip6_mc_clear_src(pmc);
  		in6_dev_put(pmc->idev);
  		kfree(pmc);
  	}
  
  	/* clear dead sources, too */
  	read_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
758
  	for (pmc = idev->mc_list; pmc; pmc = pmc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
759
760
761
762
763
764
  		struct ip6_sf_list *psf, *psf_next;
  
  		spin_lock_bh(&pmc->mca_lock);
  		psf = pmc->mca_tomb;
  		pmc->mca_tomb = NULL;
  		spin_unlock_bh(&pmc->mca_lock);
67ba4152e   Ian Morris   ipv6: White-space...
765
  		for (; psf; psf = psf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
766
767
768
769
770
771
  			psf_next = psf->sf_next;
  			kfree(psf);
  		}
  	}
  	read_unlock_bh(&idev->lock);
  }
1691c63ea   WANG Cong   ipv6: refactor ip...
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
  static void mca_get(struct ifmcaddr6 *mc)
  {
  	atomic_inc(&mc->mca_refcnt);
  }
  
  static void ma_put(struct ifmcaddr6 *mc)
  {
  	if (atomic_dec_and_test(&mc->mca_refcnt)) {
  		in6_dev_put(mc->idev);
  		kfree(mc);
  	}
  }
  
  static struct ifmcaddr6 *mca_alloc(struct inet6_dev *idev,
  				   const struct in6_addr *addr)
  {
  	struct ifmcaddr6 *mc;
  
  	mc = kzalloc(sizeof(*mc), GFP_ATOMIC);
63159f29b   Ian Morris   ipv6: coding styl...
791
  	if (!mc)
1691c63ea   WANG Cong   ipv6: refactor ip...
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
  		return NULL;
  
  	setup_timer(&mc->mca_timer, igmp6_timer_handler, (unsigned long)mc);
  
  	mc->mca_addr = *addr;
  	mc->idev = idev; /* reference taken by caller */
  	mc->mca_users = 1;
  	/* mca_stamp should be updated upon changes */
  	mc->mca_cstamp = mc->mca_tstamp = jiffies;
  	atomic_set(&mc->mca_refcnt, 1);
  	spin_lock_init(&mc->mca_lock);
  
  	/* initial mode is (EX, empty) */
  	mc->mca_sfmode = MCAST_EXCLUDE;
  	mc->mca_sfcount[MCAST_EXCLUDE] = 1;
  
  	if (ipv6_addr_is_ll_all_nodes(&mc->mca_addr) ||
  	    IPV6_ADDR_MC_SCOPE(&mc->mca_addr) < IPV6_ADDR_SCOPE_LINKLOCAL)
  		mc->mca_flags |= MAF_NOREPORT;
  
  	return mc;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
814
815
816
817
  
  /*
   *	device multicast group inc (add if not found)
   */
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
818
  int ipv6_dev_mc_inc(struct net_device *dev, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
819
820
821
  {
  	struct ifmcaddr6 *mc;
  	struct inet6_dev *idev;
a9ed4a298   Sabrina Dubroca   ipv6: fix rtnl lo...
822
  	ASSERT_RTNL();
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
823
  	/* we need to take a reference on idev */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
824
  	idev = in6_dev_get(dev);
63159f29b   Ian Morris   ipv6: coding styl...
825
  	if (!idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
  		return -EINVAL;
  
  	write_lock_bh(&idev->lock);
  	if (idev->dead) {
  		write_unlock_bh(&idev->lock);
  		in6_dev_put(idev);
  		return -ENODEV;
  	}
  
  	for (mc = idev->mc_list; mc; mc = mc->next) {
  		if (ipv6_addr_equal(&mc->mca_addr, addr)) {
  			mc->mca_users++;
  			write_unlock_bh(&idev->lock);
  			ip6_mc_add_src(idev, &mc->mca_addr, MCAST_EXCLUDE, 0,
  				NULL, 0);
  			in6_dev_put(idev);
  			return 0;
  		}
  	}
1691c63ea   WANG Cong   ipv6: refactor ip...
845
846
  	mc = mca_alloc(idev, addr);
  	if (!mc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
847
848
849
850
  		write_unlock_bh(&idev->lock);
  		in6_dev_put(idev);
  		return -ENOMEM;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
851
852
  	mc->next = idev->mc_list;
  	idev->mc_list = mc;
1691c63ea   WANG Cong   ipv6: refactor ip...
853
854
855
856
857
  
  	/* Hold this for the code below before we unlock,
  	 * it is already exposed via idev->mc_list.
  	 */
  	mca_get(mc);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
858
859
860
861
862
863
864
865
866
867
868
  	write_unlock_bh(&idev->lock);
  
  	mld_del_delrec(idev, &mc->mca_addr);
  	igmp6_group_added(mc);
  	ma_put(mc);
  	return 0;
  }
  
  /*
   *	device multicast group del
   */
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
869
  int __ipv6_dev_mc_dec(struct inet6_dev *idev, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
870
871
  {
  	struct ifmcaddr6 *ma, **map;
a9ed4a298   Sabrina Dubroca   ipv6: fix rtnl lo...
872
  	ASSERT_RTNL();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
873
  	write_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
874
  	for (map = &idev->mc_list; (ma = *map) != NULL; map = &ma->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
  		if (ipv6_addr_equal(&ma->mca_addr, addr)) {
  			if (--ma->mca_users == 0) {
  				*map = ma->next;
  				write_unlock_bh(&idev->lock);
  
  				igmp6_group_dropped(ma);
  
  				ma_put(ma);
  				return 0;
  			}
  			write_unlock_bh(&idev->lock);
  			return 0;
  		}
  	}
  	write_unlock_bh(&idev->lock);
  
  	return -ENOENT;
  }
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
893
  int ipv6_dev_mc_dec(struct net_device *dev, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
894
  {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
895
  	struct inet6_dev *idev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
896
  	int err;
414b6c943   WANG Cong   ipv6: drop some r...
897
  	ASSERT_RTNL();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
898

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
899
900
901
902
903
  	idev = __in6_dev_get(dev);
  	if (!idev)
  		err = -ENODEV;
  	else
  		err = __ipv6_dev_mc_dec(idev, addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
904
905
906
907
908
  
  	return err;
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
909
910
   *	check if the interface/address pair is valid
   */
a50feda54   Eric Dumazet   ipv6: bool/const ...
911
912
  bool ipv6_chk_mcast_addr(struct net_device *dev, const struct in6_addr *group,
  			 const struct in6_addr *src_addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
913
914
915
  {
  	struct inet6_dev *idev;
  	struct ifmcaddr6 *mc;
a50feda54   Eric Dumazet   ipv6: bool/const ...
916
  	bool rv = false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
917

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
918
919
  	rcu_read_lock();
  	idev = __in6_dev_get(dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
920
921
  	if (idev) {
  		read_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
922
  		for (mc = idev->mc_list; mc; mc = mc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
923
924
925
926
927
928
929
930
  			if (ipv6_addr_equal(&mc->mca_addr, group))
  				break;
  		}
  		if (mc) {
  			if (src_addr && !ipv6_addr_any(src_addr)) {
  				struct ip6_sf_list *psf;
  
  				spin_lock_bh(&mc->mca_lock);
67ba4152e   Ian Morris   ipv6: White-space...
931
  				for (psf = mc->mca_sources; psf; psf = psf->sf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
932
933
934
935
936
937
938
939
  					if (ipv6_addr_equal(&psf->sf_addr, src_addr))
  						break;
  				}
  				if (psf)
  					rv = psf->sf_count[MCAST_INCLUDE] ||
  						psf->sf_count[MCAST_EXCLUDE] !=
  						mc->mca_sfcount[MCAST_EXCLUDE];
  				else
67ba4152e   Ian Morris   ipv6: White-space...
940
  					rv = mc->mca_sfcount[MCAST_EXCLUDE] != 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
941
942
  				spin_unlock_bh(&mc->mca_lock);
  			} else
a50feda54   Eric Dumazet   ipv6: bool/const ...
943
  				rv = true; /* don't filter unspecified source */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
944
945
  		}
  		read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
946
  	}
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
947
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
948
949
950
951
952
  	return rv;
  }
  
  static void mld_gq_start_timer(struct inet6_dev *idev)
  {
63862b5be   Aruna-Hewapathirane   net: replace macr...
953
  	unsigned long tv = prandom_u32() % idev->mc_maxdelay;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
954
955
956
957
958
  
  	idev->mc_gq_running = 1;
  	if (!mod_timer(&idev->mc_gq_timer, jiffies+tv+2))
  		in6_dev_hold(idev);
  }
b4af8def5   Daniel Borkmann   net: ipv6: mld: i...
959
960
961
962
963
964
  static void mld_gq_stop_timer(struct inet6_dev *idev)
  {
  	idev->mc_gq_running = 0;
  	if (del_timer(&idev->mc_gq_timer))
  		__in6_dev_put(idev);
  }
c2cef4e88   Daniel Borkmann   net: ipv6: minor:...
965
  static void mld_ifc_start_timer(struct inet6_dev *idev, unsigned long delay)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
966
  {
63862b5be   Aruna-Hewapathirane   net: replace macr...
967
  	unsigned long tv = prandom_u32() % delay;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
968
969
970
971
  
  	if (!mod_timer(&idev->mc_ifc_timer, jiffies+tv+2))
  		in6_dev_hold(idev);
  }
b4af8def5   Daniel Borkmann   net: ipv6: mld: i...
972
973
974
975
976
977
  static void mld_ifc_stop_timer(struct inet6_dev *idev)
  {
  	idev->mc_ifc_count = 0;
  	if (del_timer(&idev->mc_ifc_timer))
  		__in6_dev_put(idev);
  }
c2cef4e88   Daniel Borkmann   net: ipv6: minor:...
978
  static void mld_dad_start_timer(struct inet6_dev *idev, unsigned long delay)
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
979
  {
63862b5be   Aruna-Hewapathirane   net: replace macr...
980
  	unsigned long tv = prandom_u32() % delay;
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
981
982
983
984
  
  	if (!mod_timer(&idev->mc_dad_timer, jiffies+tv+2))
  		in6_dev_hold(idev);
  }
b4af8def5   Daniel Borkmann   net: ipv6: mld: i...
985
986
987
988
989
  static void mld_dad_stop_timer(struct inet6_dev *idev)
  {
  	if (del_timer(&idev->mc_dad_timer))
  		__in6_dev_put(idev);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
  /*
   *	IGMP handling (alias multicast ICMPv6 messages)
   */
  
  static void igmp6_group_queried(struct ifmcaddr6 *ma, unsigned long resptime)
  {
  	unsigned long delay = resptime;
  
  	/* Do not start timer for these addresses */
  	if (ipv6_addr_is_ll_all_nodes(&ma->mca_addr) ||
  	    IPV6_ADDR_MC_SCOPE(&ma->mca_addr) < IPV6_ADDR_SCOPE_LINKLOCAL)
  		return;
  
  	if (del_timer(&ma->mca_timer)) {
  		atomic_dec(&ma->mca_refcnt);
  		delay = ma->mca_timer.expires - jiffies;
  	}
cc7f7ab75   Daniel Borkmann   net: ipv6: mld: s...
1007
  	if (delay >= resptime)
63862b5be   Aruna-Hewapathirane   net: replace macr...
1008
  		delay = prandom_u32() % resptime;
cc7f7ab75   Daniel Borkmann   net: ipv6: mld: s...
1009

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1010
1011
1012
1013
1014
  	ma->mca_timer.expires = jiffies + delay;
  	if (!mod_timer(&ma->mca_timer, jiffies + delay))
  		atomic_inc(&ma->mca_refcnt);
  	ma->mca_flags |= MAF_TIMER_RUNNING;
  }
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1015
  /* mark EXCLUDE-mode sources */
a50feda54   Eric Dumazet   ipv6: bool/const ...
1016
1017
  static bool mld_xmarksources(struct ifmcaddr6 *pmc, int nsrcs,
  			     const struct in6_addr *srcs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1018
1019
1020
1021
1022
  {
  	struct ip6_sf_list *psf;
  	int i, scount;
  
  	scount = 0;
67ba4152e   Ian Morris   ipv6: White-space...
1023
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1024
1025
  		if (scount == nsrcs)
  			break;
67ba4152e   Ian Morris   ipv6: White-space...
1026
  		for (i = 0; i < nsrcs; i++) {
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1027
  			/* skip inactive filters */
e05c4ad3e   Yan, Zheng   mcast: Fix source...
1028
  			if (psf->sf_count[MCAST_INCLUDE] ||
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1029
1030
  			    pmc->mca_sfcount[MCAST_EXCLUDE] !=
  			    psf->sf_count[MCAST_EXCLUDE])
ce713ee5a   RongQing.Li   net: replace cont...
1031
  				break;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1032
1033
1034
1035
1036
1037
1038
1039
  			if (ipv6_addr_equal(&srcs[i], &psf->sf_addr)) {
  				scount++;
  				break;
  			}
  		}
  	}
  	pmc->mca_flags &= ~MAF_GSQUERY;
  	if (scount == nsrcs)	/* all sources excluded */
a50feda54   Eric Dumazet   ipv6: bool/const ...
1040
1041
  		return false;
  	return true;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1042
  }
a50feda54   Eric Dumazet   ipv6: bool/const ...
1043
1044
  static bool mld_marksources(struct ifmcaddr6 *pmc, int nsrcs,
  			    const struct in6_addr *srcs)
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
  {
  	struct ip6_sf_list *psf;
  	int i, scount;
  
  	if (pmc->mca_sfmode == MCAST_EXCLUDE)
  		return mld_xmarksources(pmc, nsrcs, srcs);
  
  	/* mark INCLUDE-mode sources */
  
  	scount = 0;
67ba4152e   Ian Morris   ipv6: White-space...
1055
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next) {
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1056
1057
  		if (scount == nsrcs)
  			break;
67ba4152e   Ian Morris   ipv6: White-space...
1058
  		for (i = 0; i < nsrcs; i++) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1059
1060
1061
1062
1063
  			if (ipv6_addr_equal(&srcs[i], &psf->sf_addr)) {
  				psf->sf_gsresp = 1;
  				scount++;
  				break;
  			}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1064
1065
1066
1067
  		}
  	}
  	if (!scount) {
  		pmc->mca_flags &= ~MAF_GSQUERY;
a50feda54   Eric Dumazet   ipv6: bool/const ...
1068
  		return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1069
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1070
  	pmc->mca_flags |= MAF_GSQUERY;
a50feda54   Eric Dumazet   ipv6: bool/const ...
1071
  	return true;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1072
  }
58c0ecfd8   Daniel Borkmann   net: ipv6: mld: i...
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
  static int mld_force_mld_version(const struct inet6_dev *idev)
  {
  	/* Normally, both are 0 here. If enforcement to a particular is
  	 * being used, individual device enforcement will have a lower
  	 * precedence over 'all' device (.../conf/all/force_mld_version).
  	 */
  
  	if (dev_net(idev->dev)->ipv6.devconf_all->force_mld_version != 0)
  		return dev_net(idev->dev)->ipv6.devconf_all->force_mld_version;
  	else
  		return idev->cnf.force_mld_version;
  }
  
  static bool mld_in_v2_mode_only(const struct inet6_dev *idev)
  {
  	return mld_force_mld_version(idev) == 2;
  }
  
  static bool mld_in_v1_mode_only(const struct inet6_dev *idev)
  {
  	return mld_force_mld_version(idev) == 1;
  }
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
1095
1096
  static bool mld_in_v1_mode(const struct inet6_dev *idev)
  {
58c0ecfd8   Daniel Borkmann   net: ipv6: mld: i...
1097
1098
1099
  	if (mld_in_v2_mode_only(idev))
  		return false;
  	if (mld_in_v1_mode_only(idev))
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
1100
1101
1102
1103
1104
1105
  		return true;
  	if (idev->mc_v1_seen && time_before(jiffies, idev->mc_v1_seen))
  		return true;
  
  	return false;
  }
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
  static void mld_set_v1_mode(struct inet6_dev *idev)
  {
  	/* RFC3810, relevant sections:
  	 *  - 9.1. Robustness Variable
  	 *  - 9.2. Query Interval
  	 *  - 9.3. Query Response Interval
  	 *  - 9.12. Older Version Querier Present Timeout
  	 */
  	unsigned long switchback;
  
  	switchback = (idev->mc_qrv * idev->mc_qi) + idev->mc_qri;
  
  	idev->mc_v1_seen = jiffies + switchback;
  }
  
  static void mld_update_qrv(struct inet6_dev *idev,
  			   const struct mld2_query *mlh2)
  {
  	/* RFC3810, relevant sections:
  	 *  - 5.1.8. QRV (Querier's Robustness Variable)
  	 *  - 9.1. Robustness Variable
  	 */
  
  	/* The value of the Robustness Variable MUST NOT be zero,
  	 * and SHOULD NOT be one. Catch this here if we ever run
  	 * into such a case in future.
  	 */
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
1133
  	const int min_qrv = min(MLD_QRV_DEFAULT, sysctl_mld_qrv);
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1134
1135
1136
1137
  	WARN_ON(idev->mc_qrv == 0);
  
  	if (mlh2->mld2q_qrv > 0)
  		idev->mc_qrv = mlh2->mld2q_qrv;
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
1138
  	if (unlikely(idev->mc_qrv < min_qrv)) {
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1139
1140
  		net_warn_ratelimited("IPv6: MLD: clamping QRV from %u to %u!
  ",
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
1141
1142
  				     idev->mc_qrv, min_qrv);
  		idev->mc_qrv = min_qrv;
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
  	}
  }
  
  static void mld_update_qi(struct inet6_dev *idev,
  			  const struct mld2_query *mlh2)
  {
  	/* RFC3810, relevant sections:
  	 *  - 5.1.9. QQIC (Querier's Query Interval Code)
  	 *  - 9.2. Query Interval
  	 *  - 9.12. Older Version Querier Present Timeout
  	 *    (the [Query Interval] in the last Query received)
  	 */
  	unsigned long mc_qqi;
  
  	if (mlh2->mld2q_qqic < 128) {
  		mc_qqi = mlh2->mld2q_qqic;
  	} else {
  		unsigned long mc_man, mc_exp;
  
  		mc_exp = MLDV2_QQIC_EXP(mlh2->mld2q_qqic);
  		mc_man = MLDV2_QQIC_MAN(mlh2->mld2q_qqic);
  
  		mc_qqi = (mc_man | 0x10) << (mc_exp + 3);
  	}
  
  	idev->mc_qi = mc_qqi * HZ;
  }
  
  static void mld_update_qri(struct inet6_dev *idev,
  			   const struct mld2_query *mlh2)
  {
  	/* RFC3810, relevant sections:
  	 *  - 5.1.3. Maximum Response Code
  	 *  - 9.3. Query Response Interval
  	 */
e3f5b1704   Daniel Borkmann   net: ipv6: mld: g...
1178
  	idev->mc_qri = msecs_to_jiffies(mldv2_mrc(mlh2));
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1179
  }
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1180
  static int mld_process_v1(struct inet6_dev *idev, struct mld_msg *mld,
35f7aa530   Daniel Borkmann   ipv6: mld: answer...
1181
  			  unsigned long *max_delay, bool v1_query)
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1182
1183
1184
1185
1186
1187
  {
  	unsigned long mldv1_md;
  
  	/* Ignore v1 queries */
  	if (mld_in_v2_mode_only(idev))
  		return -EINVAL;
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1188
  	mldv1_md = ntohs(mld->mld_maxdelay);
35f7aa530   Daniel Borkmann   ipv6: mld: answer...
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
  
  	/* When in MLDv1 fallback and a MLDv2 router start-up being
  	 * unaware of current MLDv1 operation, the MRC == MRD mapping
  	 * only works when the exponential algorithm is not being
  	 * used (as MLDv1 is unaware of such things).
  	 *
  	 * According to the RFC author, the MLDv2 implementations
  	 * he's aware of all use a MRC < 32768 on start up queries.
  	 *
  	 * Thus, should we *ever* encounter something else larger
  	 * than that, just assume the maximum possible within our
  	 * reach.
  	 */
  	if (!v1_query)
  		mldv1_md = min(mldv1_md, MLDV1_MRD_MAX_COMPAT);
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1204
  	*max_delay = max(msecs_to_jiffies(mldv1_md), 1UL);
35f7aa530   Daniel Borkmann   ipv6: mld: answer...
1205
1206
1207
1208
1209
1210
1211
  	/* MLDv1 router present: we need to go into v1 mode *only*
  	 * when an MLDv1 query is received as per section 9.12. of
  	 * RFC3810! And we know from RFC2710 section 3.7 that MLDv1
  	 * queries MUST be of exactly 24 octets.
  	 */
  	if (v1_query)
  		mld_set_v1_mode(idev);
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1212
1213
  
  	/* cancel MLDv2 report timer */
b4af8def5   Daniel Borkmann   net: ipv6: mld: i...
1214
  	mld_gq_stop_timer(idev);
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1215
  	/* cancel the interface change timer */
b4af8def5   Daniel Borkmann   net: ipv6: mld: i...
1216
  	mld_ifc_stop_timer(idev);
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1217
1218
1219
1220
1221
1222
1223
1224
1225
  	/* clear deleted report items */
  	mld_clear_delrec(idev);
  
  	return 0;
  }
  
  static int mld_process_v2(struct inet6_dev *idev, struct mld2_query *mld,
  			  unsigned long *max_delay)
  {
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
  	*max_delay = max(msecs_to_jiffies(mldv2_mrc(mld)), 1UL);
  
  	mld_update_qrv(idev, mld);
  	mld_update_qi(idev, mld);
  	mld_update_qri(idev, mld);
  
  	idev->mc_maxdelay = *max_delay;
  
  	return 0;
  }
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1236
  /* called with rcu_read_lock() */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1237
1238
  int igmp6_event_query(struct sk_buff *skb)
  {
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1239
  	struct mld2_query *mlh2 = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1240
  	struct ifmcaddr6 *ma;
b71d1d426   Eric Dumazet   inet: constify ip...
1241
  	const struct in6_addr *group;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1242
1243
  	unsigned long max_delay;
  	struct inet6_dev *idev;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1244
  	struct mld_msg *mld;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1245
1246
  	int group_type;
  	int mark = 0;
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1247
  	int len, err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1248
1249
1250
1251
1252
  
  	if (!pskb_may_pull(skb, sizeof(struct in6_addr)))
  		return -EINVAL;
  
  	/* compute payload length excluding extension headers */
0660e03f6   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1253
  	len = ntohs(ipv6_hdr(skb)->payload_len) + sizeof(struct ipv6hdr);
cfe1fc775   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1254
  	len -= skb_network_header_len(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1255

e940f5d6b   Hangbin Liu   ipv6: Fix MLD Que...
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
  	/* RFC3810 6.2
  	 * Upon reception of an MLD message that contains a Query, the node
  	 * checks if the source address of the message is a valid link-local
  	 * address, if the Hop Limit is set to 1, and if the Router Alert
  	 * option is present in the Hop-By-Hop Options header of the IPv6
  	 * packet.  If any of these checks fails, the packet is dropped.
  	 */
  	if (!(ipv6_addr_type(&ipv6_hdr(skb)->saddr) & IPV6_ADDR_LINKLOCAL) ||
  	    ipv6_hdr(skb)->hop_limit != 1 ||
  	    !(IP6CB(skb)->flags & IP6SKB_ROUTERALERT) ||
  	    IP6CB(skb)->ra != htons(IPV6_OPT_ROUTERALERT_MLD))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1267
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1268
  	idev = __in6_dev_get(skb->dev);
63159f29b   Ian Morris   ipv6: coding styl...
1269
  	if (!idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1270
  		return 0;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1271
1272
  	mld = (struct mld_msg *)icmp6_hdr(skb);
  	group = &mld->mld_mca;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1273
1274
1275
  	group_type = ipv6_addr_type(group);
  
  	if (group_type != IPV6_ADDR_ANY &&
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1276
  	    !(group_type&IPV6_ADDR_MULTICAST))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1277
  		return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1278

35f7aa530   Daniel Borkmann   ipv6: mld: answer...
1279
1280
1281
1282
1283
  	if (len < MLD_V1_QUERY_LEN) {
  		return -EINVAL;
  	} else if (len == MLD_V1_QUERY_LEN || mld_in_v1_mode(idev)) {
  		err = mld_process_v1(idev, mld, &max_delay,
  				     len == MLD_V1_QUERY_LEN);
2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1284
1285
  		if (err < 0)
  			return err;
9fd078416   Daniel Borkmann   net: ipv6: mcast:...
1286
  	} else if (len >= MLD_V2_QUERY_LEN_MIN) {
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1287
  		int srcs_offset = sizeof(struct mld2_query) -
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1288
  				  sizeof(struct icmp6hdr);
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1289

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1290
  		if (!pskb_may_pull(skb, srcs_offset))
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1291
  			return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1292

9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1293
  		mlh2 = (struct mld2_query *)skb_transport_header(skb);
846989635   Daniel Borkmann   net: ipv6: igmp6_...
1294

2b7c121f8   Daniel Borkmann   net: ipv6: mld: r...
1295
1296
1297
  		err = mld_process_v2(idev, mlh2, &max_delay);
  		if (err < 0)
  			return err;
89225d1ce   Daniel Borkmann   net: ipv6: mld: f...
1298

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1299
  		if (group_type == IPV6_ADDR_ANY) { /* general query */
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1300
  			if (mlh2->mld2q_nsrcs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1301
  				return -EINVAL; /* no sources allowed */
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1302

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1303
  			mld_gq_start_timer(idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1304
1305
1306
  			return 0;
  		}
  		/* mark sources to include, if group & source-specific */
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1307
  		if (mlh2->mld2q_nsrcs != 0) {
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1308
  			if (!pskb_may_pull(skb, srcs_offset +
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1309
  			    ntohs(mlh2->mld2q_nsrcs) * sizeof(struct in6_addr)))
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1310
  				return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1311

9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1312
  			mlh2 = (struct mld2_query *)skb_transport_header(skb);
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1313
1314
  			mark = 1;
  		}
35f7aa530   Daniel Borkmann   ipv6: mld: answer...
1315
  	} else {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1316
  		return -EINVAL;
35f7aa530   Daniel Borkmann   ipv6: mld: answer...
1317
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1318
1319
1320
  
  	read_lock_bh(&idev->lock);
  	if (group_type == IPV6_ADDR_ANY) {
67ba4152e   Ian Morris   ipv6: White-space...
1321
  		for (ma = idev->mc_list; ma; ma = ma->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1322
1323
1324
1325
1326
  			spin_lock_bh(&ma->mca_lock);
  			igmp6_group_queried(ma, max_delay);
  			spin_unlock_bh(&ma->mca_lock);
  		}
  	} else {
67ba4152e   Ian Morris   ipv6: White-space...
1327
  		for (ma = idev->mc_list; ma; ma = ma->next) {
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1328
  			if (!ipv6_addr_equal(group, &ma->mca_addr))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
  				continue;
  			spin_lock_bh(&ma->mca_lock);
  			if (ma->mca_flags & MAF_TIMER_RUNNING) {
  				/* gsquery <- gsquery && mark */
  				if (!mark)
  					ma->mca_flags &= ~MAF_GSQUERY;
  			} else {
  				/* gsquery <- mark */
  				if (mark)
  					ma->mca_flags |= MAF_GSQUERY;
  				else
  					ma->mca_flags &= ~MAF_GSQUERY;
  			}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1342
  			if (!(ma->mca_flags & MAF_GSQUERY) ||
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1343
  			    mld_marksources(ma, ntohs(mlh2->mld2q_nsrcs), mlh2->mld2q_srcs))
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1344
  				igmp6_group_queried(ma, max_delay);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1345
  			spin_unlock_bh(&ma->mca_lock);
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1346
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1347
1348
1349
  		}
  	}
  	read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1350
1351
1352
  
  	return 0;
  }
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1353
  /* called with rcu_read_lock() */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1354
1355
1356
  int igmp6_event_report(struct sk_buff *skb)
  {
  	struct ifmcaddr6 *ma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1357
  	struct inet6_dev *idev;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1358
  	struct mld_msg *mld;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1359
1360
1361
1362
1363
  	int addr_type;
  
  	/* Our own report looped back. Ignore it. */
  	if (skb->pkt_type == PACKET_LOOPBACK)
  		return 0;
24c692750   David Stevens   [IGMP]: workaroun...
1364
1365
1366
1367
  	/* send our report if the MC router may not have heard this report */
  	if (skb->pkt_type != PACKET_MULTICAST &&
  	    skb->pkt_type != PACKET_BROADCAST)
  		return 0;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1368
  	if (!pskb_may_pull(skb, sizeof(*mld) - sizeof(struct icmp6hdr)))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1369
  		return -EINVAL;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1370
  	mld = (struct mld_msg *)icmp6_hdr(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1371
1372
  
  	/* Drop reports with not link local source */
0660e03f6   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1373
  	addr_type = ipv6_addr_type(&ipv6_hdr(skb)->saddr);
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1374
  	if (addr_type != IPV6_ADDR_ANY &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1375
1376
  	    !(addr_type&IPV6_ADDR_LINKLOCAL))
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1377
  	idev = __in6_dev_get(skb->dev);
63159f29b   Ian Morris   ipv6: coding styl...
1378
  	if (!idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1379
1380
1381
1382
1383
1384
1385
  		return -ENODEV;
  
  	/*
  	 *	Cancel the timer for this group
  	 */
  
  	read_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
1386
  	for (ma = idev->mc_list; ma; ma = ma->next) {
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1387
  		if (ipv6_addr_equal(&ma->mca_addr, &mld->mld_mca)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1388
1389
1390
1391
1392
1393
1394
1395
1396
  			spin_lock(&ma->mca_lock);
  			if (del_timer(&ma->mca_timer))
  				atomic_dec(&ma->mca_refcnt);
  			ma->mca_flags &= ~(MAF_LAST_REPORTER|MAF_TIMER_RUNNING);
  			spin_unlock(&ma->mca_lock);
  			break;
  		}
  	}
  	read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1397
1398
  	return 0;
  }
a50feda54   Eric Dumazet   ipv6: bool/const ...
1399
1400
  static bool is_in(struct ifmcaddr6 *pmc, struct ip6_sf_list *psf, int type,
  		  int gdeleted, int sdeleted)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1401
1402
1403
1404
1405
  {
  	switch (type) {
  	case MLD2_MODE_IS_INCLUDE:
  	case MLD2_MODE_IS_EXCLUDE:
  		if (gdeleted || sdeleted)
a50feda54   Eric Dumazet   ipv6: bool/const ...
1406
  			return false;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1407
1408
  		if (!((pmc->mca_flags & MAF_GSQUERY) && !psf->sf_gsresp)) {
  			if (pmc->mca_sfmode == MCAST_INCLUDE)
a50feda54   Eric Dumazet   ipv6: bool/const ...
1409
  				return true;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1410
1411
1412
1413
  			/* don't include if this source is excluded
  			 * in all filters
  			 */
  			if (psf->sf_count[MCAST_INCLUDE])
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1414
  				return type == MLD2_MODE_IS_INCLUDE;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1415
1416
1417
  			return pmc->mca_sfcount[MCAST_EXCLUDE] ==
  				psf->sf_count[MCAST_EXCLUDE];
  		}
a50feda54   Eric Dumazet   ipv6: bool/const ...
1418
  		return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1419
1420
  	case MLD2_CHANGE_TO_INCLUDE:
  		if (gdeleted || sdeleted)
a50feda54   Eric Dumazet   ipv6: bool/const ...
1421
  			return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1422
1423
1424
  		return psf->sf_count[MCAST_INCLUDE] != 0;
  	case MLD2_CHANGE_TO_EXCLUDE:
  		if (gdeleted || sdeleted)
a50feda54   Eric Dumazet   ipv6: bool/const ...
1425
  			return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1426
1427
  		if (pmc->mca_sfcount[MCAST_EXCLUDE] == 0 ||
  		    psf->sf_count[MCAST_INCLUDE])
a50feda54   Eric Dumazet   ipv6: bool/const ...
1428
  			return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1429
1430
1431
1432
  		return pmc->mca_sfcount[MCAST_EXCLUDE] ==
  			psf->sf_count[MCAST_EXCLUDE];
  	case MLD2_ALLOW_NEW_SOURCES:
  		if (gdeleted || !psf->sf_crcount)
a50feda54   Eric Dumazet   ipv6: bool/const ...
1433
  			return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1434
1435
1436
1437
1438
1439
  		return (pmc->mca_sfmode == MCAST_INCLUDE) ^ sdeleted;
  	case MLD2_BLOCK_OLD_SOURCES:
  		if (pmc->mca_sfmode == MCAST_INCLUDE)
  			return gdeleted || (psf->sf_crcount && sdeleted);
  		return psf->sf_crcount && !gdeleted && !sdeleted;
  	}
a50feda54   Eric Dumazet   ipv6: bool/const ...
1440
  	return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1441
1442
1443
1444
1445
1446
1447
  }
  
  static int
  mld_scount(struct ifmcaddr6 *pmc, int type, int gdeleted, int sdeleted)
  {
  	struct ip6_sf_list *psf;
  	int scount = 0;
67ba4152e   Ian Morris   ipv6: White-space...
1448
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1449
1450
1451
1452
1453
1454
  		if (!is_in(pmc, psf, type, gdeleted, sdeleted))
  			continue;
  		scount++;
  	}
  	return scount;
  }
2576f17df   YOSHIFUJI Hideaki / 吉藤英明   ipv6: Unshare ip6...
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
  static void ip6_mc_hdr(struct sock *sk, struct sk_buff *skb,
  		       struct net_device *dev,
  		       const struct in6_addr *saddr,
  		       const struct in6_addr *daddr,
  		       int proto, int len)
  {
  	struct ipv6hdr *hdr;
  
  	skb->protocol = htons(ETH_P_IPV6);
  	skb->dev = dev;
  
  	skb_reset_network_header(skb);
  	skb_put(skb, sizeof(struct ipv6hdr));
  	hdr = ipv6_hdr(skb);
  
  	ip6_flow_hdr(hdr, 0, 0);
  
  	hdr->payload_len = htons(len);
  	hdr->nexthdr = proto;
  	hdr->hop_limit = inet6_sk(sk)->hop_limit;
  
  	hdr->saddr = *saddr;
  	hdr->daddr = *daddr;
  }
4c672e4b4   Daniel Borkmann   ipv6: mld: fix ad...
1479
  static struct sk_buff *mld_newpack(struct inet6_dev *idev, unsigned int mtu)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1480
  {
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1481
  	struct net_device *dev = idev->dev;
c346dca10   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
1482
  	struct net *net = dev_net(dev);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
1483
  	struct sock *sk = net->ipv6.igmp_sk;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1484
1485
1486
  	struct sk_buff *skb;
  	struct mld2_report *pmr;
  	struct in6_addr addr_buf;
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1487
  	const struct in6_addr *saddr;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1488
1489
  	int hlen = LL_RESERVED_SPACE(dev);
  	int tlen = dev->needed_tailroom;
4c672e4b4   Daniel Borkmann   ipv6: mld: fix ad...
1490
  	unsigned int size = mtu + hlen + tlen;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1491
1492
1493
1494
1495
1496
  	int err;
  	u8 ra[8] = { IPPROTO_ICMPV6, 0,
  		     IPV6_TLV_ROUTERALERT, 2, 0, 0,
  		     IPV6_TLV_PADN, 0 };
  
  	/* we assume size > sizeof(ra) here */
72e09ad10   Eric Dumazet   ipv6: avoid high ...
1497
1498
1499
  	/* limit our allocations to order-0 page */
  	size = min_t(int, size, SKB_MAX_ORDER(0, 0));
  	skb = sock_alloc_send_skb(sk, size, 1, &err);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1500

cfcabdcc2   Stephen Hemminger   [NET]: sparse war...
1501
  	if (!skb)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1502
  		return NULL;
9d4a03146   Hannes Frederic Sowa   ipv4, ipv6: send ...
1503
  	skb->priority = TC_PRIO_CONTROL;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1504
  	skb_reserve(skb, hlen);
1837b2e2b   Benjamin Poirier   mld, igmp: Fix re...
1505
  	skb_tailroom_reserve(skb, mtu, tlen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1506

8965779d2   Amerigo Wang   ipv6,mcast: alway...
1507
  	if (__ipv6_get_lladdr(idev, &addr_buf, IFA_F_TENTATIVE)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1508
  		/* <draft-ietf-magma-mld-source-05.txt>:
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1509
  		 * use unspecified address as the source address
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1510
1511
  		 * when a valid link-local address is not available.
  		 */
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1512
1513
1514
  		saddr = &in6addr_any;
  	} else
  		saddr = &addr_buf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1515

2576f17df   YOSHIFUJI Hideaki / 吉藤英明   ipv6: Unshare ip6...
1516
  	ip6_mc_hdr(sk, skb, dev, saddr, &mld2_all_mcr, NEXTHDR_HOP, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1517
1518
  
  	memcpy(skb_put(skb, sizeof(ra)), ra, sizeof(ra));
27a884dc3   Arnaldo Carvalho de Melo   [SK_BUFF]: Conver...
1519
  	skb_set_transport_header(skb, skb_tail_pointer(skb) - skb->data);
d10ba34b0   Arnaldo Carvalho de Melo   [SK_BUFF]: More s...
1520
1521
  	skb_put(skb, sizeof(*pmr));
  	pmr = (struct mld2_report *)skb_transport_header(skb);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1522
1523
1524
1525
1526
  	pmr->mld2r_type = ICMPV6_MLD2_REPORT;
  	pmr->mld2r_resv1 = 0;
  	pmr->mld2r_cksum = 0;
  	pmr->mld2r_resv2 = 0;
  	pmr->mld2r_ngrec = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1527
1528
1529
1530
1531
  	return skb;
  }
  
  static void mld_sendpack(struct sk_buff *skb)
  {
0660e03f6   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1532
  	struct ipv6hdr *pip6 = ipv6_hdr(skb);
9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1533
1534
  	struct mld2_report *pmr =
  			      (struct mld2_report *)skb_transport_header(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1535
  	int payload_len, mldlen;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1536
  	struct inet6_dev *idev;
c346dca10   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
1537
  	struct net *net = dev_net(skb->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1538
  	int err;
4c9483b2f   David S. Miller   ipv6: Convert to ...
1539
  	struct flowi6 fl6;
adf30907d   Eric Dumazet   net: skb->dst acc...
1540
  	struct dst_entry *dst;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1541

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1542
1543
  	rcu_read_lock();
  	idev = __in6_dev_get(skb->dev);
edf391ff1   Neil Horman   snmp: add missing...
1544
  	IP6_UPD_PO_STATS(net, idev, IPSTATS_MIB_OUT, skb->len);
29a3cad5c   Simon Horman   ipv6: Correct com...
1545
1546
1547
  	payload_len = (skb_tail_pointer(skb) - skb_network_header(skb)) -
  		sizeof(*pip6);
  	mldlen = skb_tail_pointer(skb) - skb_transport_header(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1548
  	pip6->payload_len = htons(payload_len);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1549
1550
1551
1552
  	pmr->mld2r_cksum = csum_ipv6_magic(&pip6->saddr, &pip6->daddr, mldlen,
  					   IPPROTO_ICMPV6,
  					   csum_partial(skb_transport_header(skb),
  							mldlen, 0));
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1553

4c9483b2f   David S. Miller   ipv6: Convert to ...
1554
  	icmpv6_flow_init(net->ipv6.igmp_sk, &fl6, ICMPV6_MLD2_REPORT,
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1555
1556
  			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr,
  			 skb->dev->ifindex);
12fd84f43   YOSHIFUJI Hideaki / 吉藤英明   ipv6: Remove unus...
1557
  	dst = icmp6_dst_alloc(skb->dev, &fl6);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1558

452edd598   David S. Miller   xfrm: Return dst ...
1559
1560
1561
1562
1563
  	err = 0;
  	if (IS_ERR(dst)) {
  		err = PTR_ERR(dst);
  		dst = NULL;
  	}
adf30907d   Eric Dumazet   net: skb->dst acc...
1564
  	skb_dst_set(skb, dst);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1565
1566
  	if (err)
  		goto err_out;
edf391ff1   Neil Horman   snmp: add missing...
1567
  	payload_len = skb->len;
7026b1ddb   David Miller   netfilter: Pass s...
1568
  	err = NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_OUT,
29a26a568   Eric W. Biederman   netfilter: Pass s...
1569
  		      net, net->ipv6.igmp_sk, skb, NULL, skb->dev,
13206b6bf   Eric W. Biederman   net: Pass net int...
1570
  		      dst_output);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1571
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1572
  	if (!err) {
43a43b604   Hannes Frederic Sowa   ipv6: some ipv6 s...
1573
1574
  		ICMP6MSGOUT_INC_STATS(net, idev, ICMPV6_MLD2_REPORT);
  		ICMP6_INC_STATS(net, idev, ICMP6_MIB_OUTMSGS);
43a43b604   Hannes Frederic Sowa   ipv6: some ipv6 s...
1575
1576
1577
  	} else {
  		IP6_INC_STATS(net, idev, IPSTATS_MIB_OUTDISCARDS);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1578

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1579
  	rcu_read_unlock();
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1580
1581
1582
1583
1584
  	return;
  
  err_out:
  	kfree_skb(skb);
  	goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1585
1586
1587
1588
  }
  
  static int grec_size(struct ifmcaddr6 *pmc, int type, int gdel, int sdel)
  {
fab10fe37   Yan Zheng   [MCAST] ipv6: Fix...
1589
  	return sizeof(struct mld2_grec) + 16 * mld_scount(pmc,type,gdel,sdel);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
  }
  
  static struct sk_buff *add_grhead(struct sk_buff *skb, struct ifmcaddr6 *pmc,
  	int type, struct mld2_grec **ppgr)
  {
  	struct net_device *dev = pmc->idev->dev;
  	struct mld2_report *pmr;
  	struct mld2_grec *pgr;
  
  	if (!skb)
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1600
  		skb = mld_newpack(pmc->idev, dev->mtu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1601
1602
1603
1604
1605
1606
1607
  	if (!skb)
  		return NULL;
  	pgr = (struct mld2_grec *)skb_put(skb, sizeof(struct mld2_grec));
  	pgr->grec_type = type;
  	pgr->grec_auxwords = 0;
  	pgr->grec_nsrcs = 0;
  	pgr->grec_mca = pmc->mca_addr;	/* structure copy */
9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1608
  	pmr = (struct mld2_report *)skb_transport_header(skb);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1609
  	pmr->mld2r_ngrec = htons(ntohs(pmr->mld2r_ngrec)+1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1610
1611
1612
  	*ppgr = pgr;
  	return skb;
  }
4c672e4b4   Daniel Borkmann   ipv6: mld: fix ad...
1613
  #define AVAILABLE(skb)	((skb) ? skb_availroom(skb) : 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1614
1615
  
  static struct sk_buff *add_grec(struct sk_buff *skb, struct ifmcaddr6 *pmc,
6a7cc4187   Flavio Leitner   ipv6: send Change...
1616
  	int type, int gdeleted, int sdeleted, int crsend)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1617
  {
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1618
1619
  	struct inet6_dev *idev = pmc->idev;
  	struct net_device *dev = idev->dev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1620
1621
1622
  	struct mld2_report *pmr;
  	struct mld2_grec *pgr = NULL;
  	struct ip6_sf_list *psf, *psf_next, *psf_prev, **psf_list;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1623
  	int scount, stotal, first, isquery, truncate;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1624
1625
1626
1627
1628
1629
1630
1631
  
  	if (pmc->mca_flags & MAF_NOREPORT)
  		return skb;
  
  	isquery = type == MLD2_MODE_IS_INCLUDE ||
  		  type == MLD2_MODE_IS_EXCLUDE;
  	truncate = type == MLD2_MODE_IS_EXCLUDE ||
  		    type == MLD2_CHANGE_TO_EXCLUDE;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1632
  	stotal = scount = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1633
  	psf_list = sdeleted ? &pmc->mca_tomb : &pmc->mca_sources;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1634
1635
  	if (!*psf_list)
  		goto empty_source;
9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1636
  	pmr = skb ? (struct mld2_report *)skb_transport_header(skb) : NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1637
1638
1639
  
  	/* EX and TO_EX get a fresh packet, if needed */
  	if (truncate) {
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1640
  		if (pmr && pmr->mld2r_ngrec &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1641
1642
1643
  		    AVAILABLE(skb) < grec_size(pmc, type, gdeleted, sdeleted)) {
  			if (skb)
  				mld_sendpack(skb);
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1644
  			skb = mld_newpack(idev, dev->mtu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1645
1646
1647
  		}
  	}
  	first = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1648
  	psf_prev = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
1649
  	for (psf = *psf_list; psf; psf = psf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1650
1651
1652
1653
1654
1655
1656
1657
  		struct in6_addr *psrc;
  
  		psf_next = psf->sf_next;
  
  		if (!is_in(pmc, psf, type, gdeleted, sdeleted)) {
  			psf_prev = psf;
  			continue;
  		}
a052517a8   Hangbin Liu   net/multicast: sh...
1658
1659
1660
1661
1662
1663
1664
1665
  		/* Based on RFC3810 6.1. Should not send source-list change
  		 * records when there is a filter mode change.
  		 */
  		if (((gdeleted && pmc->mca_sfmode == MCAST_EXCLUDE) ||
  		     (!gdeleted && pmc->mca_crcount)) &&
  		    (type == MLD2_ALLOW_NEW_SOURCES ||
  		     type == MLD2_BLOCK_OLD_SOURCES) && psf->sf_crcount)
  			goto decrease_sf_crcount;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
  		/* clear marks on query responses */
  		if (isquery)
  			psf->sf_gsresp = 0;
  
  		if (AVAILABLE(skb) < sizeof(*psrc) +
  		    first*sizeof(struct mld2_grec)) {
  			if (truncate && !first)
  				break;	 /* truncate these */
  			if (pgr)
  				pgr->grec_nsrcs = htons(scount);
  			if (skb)
  				mld_sendpack(skb);
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1678
  			skb = mld_newpack(idev, dev->mtu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1679
1680
1681
1682
1683
1684
1685
  			first = 1;
  			scount = 0;
  		}
  		if (first) {
  			skb = add_grhead(skb, pmc, type, &pgr);
  			first = 0;
  		}
cc63f70b8   Alexey Dobriyan   [IPV4/IPV6] multi...
1686
1687
  		if (!skb)
  			return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1688
1689
  		psrc = (struct in6_addr *)skb_put(skb, sizeof(*psrc));
  		*psrc = psf->sf_addr;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1690
  		scount++; stotal++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1691
1692
  		if ((type == MLD2_ALLOW_NEW_SOURCES ||
  		     type == MLD2_BLOCK_OLD_SOURCES) && psf->sf_crcount) {
a052517a8   Hangbin Liu   net/multicast: sh...
1693
  decrease_sf_crcount:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
  			psf->sf_crcount--;
  			if ((sdeleted || gdeleted) && psf->sf_crcount == 0) {
  				if (psf_prev)
  					psf_prev->sf_next = psf->sf_next;
  				else
  					*psf_list = psf->sf_next;
  				kfree(psf);
  				continue;
  			}
  		}
  		psf_prev = psf;
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1706
1707
1708
1709
1710
1711
  
  empty_source:
  	if (!stotal) {
  		if (type == MLD2_ALLOW_NEW_SOURCES ||
  		    type == MLD2_BLOCK_OLD_SOURCES)
  			return skb;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1712
  		if (pmc->mca_crcount || isquery || crsend) {
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1713
1714
1715
1716
1717
1718
1719
1720
  			/* make sure we have room for group header */
  			if (skb && AVAILABLE(skb) < sizeof(struct mld2_grec)) {
  				mld_sendpack(skb);
  				skb = NULL; /* add_grhead will get a new one */
  			}
  			skb = add_grhead(skb, pmc, type, &pgr);
  		}
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1721
1722
1723
1724
1725
1726
1727
1728
1729
1730
1731
1732
  	if (pgr)
  		pgr->grec_nsrcs = htons(scount);
  
  	if (isquery)
  		pmc->mca_flags &= ~MAF_GSQUERY;	/* clear query state */
  	return skb;
  }
  
  static void mld_send_report(struct inet6_dev *idev, struct ifmcaddr6 *pmc)
  {
  	struct sk_buff *skb = NULL;
  	int type;
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1733
  	read_lock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1734
  	if (!pmc) {
67ba4152e   Ian Morris   ipv6: White-space...
1735
  		for (pmc = idev->mc_list; pmc; pmc = pmc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1736
1737
1738
1739
1740
1741
1742
  			if (pmc->mca_flags & MAF_NOREPORT)
  				continue;
  			spin_lock_bh(&pmc->mca_lock);
  			if (pmc->mca_sfcount[MCAST_EXCLUDE])
  				type = MLD2_MODE_IS_EXCLUDE;
  			else
  				type = MLD2_MODE_IS_INCLUDE;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1743
  			skb = add_grec(skb, pmc, type, 0, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1744
1745
  			spin_unlock_bh(&pmc->mca_lock);
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1746
1747
1748
1749
1750
1751
  	} else {
  		spin_lock_bh(&pmc->mca_lock);
  		if (pmc->mca_sfcount[MCAST_EXCLUDE])
  			type = MLD2_MODE_IS_EXCLUDE;
  		else
  			type = MLD2_MODE_IS_INCLUDE;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1752
  		skb = add_grec(skb, pmc, type, 0, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1753
1754
  		spin_unlock_bh(&pmc->mca_lock);
  	}
8965779d2   Amerigo Wang   ipv6,mcast: alway...
1755
  	read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
  	if (skb)
  		mld_sendpack(skb);
  }
  
  /*
   * remove zero-count source records from a source filter list
   */
  static void mld_clear_zeros(struct ip6_sf_list **ppsf)
  {
  	struct ip6_sf_list *psf_prev, *psf_next, *psf;
  
  	psf_prev = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
1768
  	for (psf = *ppsf; psf; psf = psf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
1786
1787
  		psf_next = psf->sf_next;
  		if (psf->sf_crcount == 0) {
  			if (psf_prev)
  				psf_prev->sf_next = psf->sf_next;
  			else
  				*ppsf = psf->sf_next;
  			kfree(psf);
  		} else
  			psf_prev = psf;
  	}
  }
  
  static void mld_send_cr(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *pmc, *pmc_prev, *pmc_next;
  	struct sk_buff *skb = NULL;
  	int type, dtype;
  
  	read_lock_bh(&idev->lock);
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
1788
  	spin_lock(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1789
1790
1791
  
  	/* deleted MCA's */
  	pmc_prev = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
1792
  	for (pmc = idev->mc_tomb; pmc; pmc = pmc_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1793
1794
1795
1796
  		pmc_next = pmc->next;
  		if (pmc->mca_sfmode == MCAST_INCLUDE) {
  			type = MLD2_BLOCK_OLD_SOURCES;
  			dtype = MLD2_BLOCK_OLD_SOURCES;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1797
1798
  			skb = add_grec(skb, pmc, type, 1, 0, 0);
  			skb = add_grec(skb, pmc, dtype, 1, 1, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1799
1800
  		}
  		if (pmc->mca_crcount) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1801
1802
  			if (pmc->mca_sfmode == MCAST_EXCLUDE) {
  				type = MLD2_CHANGE_TO_INCLUDE;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1803
  				skb = add_grec(skb, pmc, type, 1, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1804
  			}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1805
  			pmc->mca_crcount--;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1806
1807
1808
1809
1810
1811
1812
1813
1814
1815
1816
1817
1818
1819
1820
1821
  			if (pmc->mca_crcount == 0) {
  				mld_clear_zeros(&pmc->mca_tomb);
  				mld_clear_zeros(&pmc->mca_sources);
  			}
  		}
  		if (pmc->mca_crcount == 0 && !pmc->mca_tomb &&
  		    !pmc->mca_sources) {
  			if (pmc_prev)
  				pmc_prev->next = pmc_next;
  			else
  				idev->mc_tomb = pmc_next;
  			in6_dev_put(pmc->idev);
  			kfree(pmc);
  		} else
  			pmc_prev = pmc;
  	}
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
1822
  	spin_unlock(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1823
1824
  
  	/* change recs */
67ba4152e   Ian Morris   ipv6: White-space...
1825
  	for (pmc = idev->mc_list; pmc; pmc = pmc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1826
1827
1828
1829
1830
1831
1832
1833
  		spin_lock_bh(&pmc->mca_lock);
  		if (pmc->mca_sfcount[MCAST_EXCLUDE]) {
  			type = MLD2_BLOCK_OLD_SOURCES;
  			dtype = MLD2_ALLOW_NEW_SOURCES;
  		} else {
  			type = MLD2_ALLOW_NEW_SOURCES;
  			dtype = MLD2_BLOCK_OLD_SOURCES;
  		}
6a7cc4187   Flavio Leitner   ipv6: send Change...
1834
1835
  		skb = add_grec(skb, pmc, type, 0, 0, 0);
  		skb = add_grec(skb, pmc, dtype, 0, 1, 0);	/* deleted sources */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1836
1837
1838
  
  		/* filter mode changes */
  		if (pmc->mca_crcount) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1839
1840
1841
1842
  			if (pmc->mca_sfmode == MCAST_EXCLUDE)
  				type = MLD2_CHANGE_TO_EXCLUDE;
  			else
  				type = MLD2_CHANGE_TO_INCLUDE;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1843
  			skb = add_grec(skb, pmc, type, 0, 0, 0);
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1844
  			pmc->mca_crcount--;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1845
1846
1847
1848
1849
1850
1851
1852
1853
1854
1855
  		}
  		spin_unlock_bh(&pmc->mca_lock);
  	}
  	read_unlock_bh(&idev->lock);
  	if (!skb)
  		return;
  	(void) mld_sendpack(skb);
  }
  
  static void igmp6_send(struct in6_addr *addr, struct net_device *dev, int type)
  {
c346dca10   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
1856
  	struct net *net = dev_net(dev);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
1857
  	struct sock *sk = net->ipv6.igmp_sk;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1858
  	struct inet6_dev *idev;
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1859
  	struct sk_buff *skb;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1860
  	struct mld_msg *hdr;
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1861
  	const struct in6_addr *snd_addr, *saddr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1862
  	struct in6_addr addr_buf;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1863
1864
  	int hlen = LL_RESERVED_SPACE(dev);
  	int tlen = dev->needed_tailroom;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1865
1866
1867
1868
  	int err, len, payload_len, full_len;
  	u8 ra[8] = { IPPROTO_ICMPV6, 0,
  		     IPV6_TLV_ROUTERALERT, 2, 0, 0,
  		     IPV6_TLV_PADN, 0 };
4c9483b2f   David S. Miller   ipv6: Convert to ...
1869
  	struct flowi6 fl6;
adf30907d   Eric Dumazet   net: skb->dst acc...
1870
  	struct dst_entry *dst;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1871

f3ee4010e   YOSHIFUJI Hideaki   [IPV6]: Define co...
1872
1873
1874
1875
  	if (type == ICMPV6_MGM_REDUCTION)
  		snd_addr = &in6addr_linklocal_allrouters;
  	else
  		snd_addr = addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1876
1877
1878
1879
  
  	len = sizeof(struct icmp6hdr) + sizeof(struct in6_addr);
  	payload_len = len + sizeof(ra);
  	full_len = sizeof(struct ipv6hdr) + payload_len;
edf391ff1   Neil Horman   snmp: add missing...
1880
1881
1882
1883
  	rcu_read_lock();
  	IP6_UPD_PO_STATS(net, __in6_dev_get(dev),
  		      IPSTATS_MIB_OUT, full_len);
  	rcu_read_unlock();
a7ae19922   Herbert Xu   ipv6: Remove all ...
1884
  	skb = sock_alloc_send_skb(sk, hlen + tlen + full_len, 1, &err);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1885

63159f29b   Ian Morris   ipv6: coding styl...
1886
  	if (!skb) {
a11d206d0   YOSHIFUJI Hideaki   [IPV6]: Per-inter...
1887
  		rcu_read_lock();
3bd653c84   Denis V. Lunev   netns: add net pa...
1888
  		IP6_INC_STATS(net, __in6_dev_get(dev),
a11d206d0   YOSHIFUJI Hideaki   [IPV6]: Per-inter...
1889
1890
  			      IPSTATS_MIB_OUTDISCARDS);
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1891
1892
  		return;
  	}
9d4a03146   Hannes Frederic Sowa   ipv4, ipv6: send ...
1893
  	skb->priority = TC_PRIO_CONTROL;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1894
  	skb_reserve(skb, hlen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1895

95c385b4d   Neil Horman   [IPV6] ADDRCONF: ...
1896
  	if (ipv6_get_lladdr(dev, &addr_buf, IFA_F_TENTATIVE)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1897
  		/* <draft-ietf-magma-mld-source-05.txt>:
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1898
  		 * use unspecified address as the source address
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1899
1900
  		 * when a valid link-local address is not available.
  		 */
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1901
1902
1903
  		saddr = &in6addr_any;
  	} else
  		saddr = &addr_buf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1904

2576f17df   YOSHIFUJI Hideaki / 吉藤英明   ipv6: Unshare ip6...
1905
  	ip6_mc_hdr(sk, skb, dev, saddr, snd_addr, NEXTHDR_HOP, payload_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1906
1907
  
  	memcpy(skb_put(skb, sizeof(ra)), ra, sizeof(ra));
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1908
1909
1910
  	hdr = (struct mld_msg *) skb_put(skb, sizeof(struct mld_msg));
  	memset(hdr, 0, sizeof(struct mld_msg));
  	hdr->mld_type = type;
4e3fd7a06   Alexey Dobriyan   net: remove ipv6_...
1911
  	hdr->mld_mca = *addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1912

6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1913
1914
1915
  	hdr->mld_cksum = csum_ipv6_magic(saddr, snd_addr, len,
  					 IPPROTO_ICMPV6,
  					 csum_partial(hdr, len, 0));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1916

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1917
1918
  	rcu_read_lock();
  	idev = __in6_dev_get(skb->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1919

4c9483b2f   David S. Miller   ipv6: Convert to ...
1920
  	icmpv6_flow_init(sk, &fl6, type,
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1921
1922
  			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr,
  			 skb->dev->ifindex);
12fd84f43   YOSHIFUJI Hideaki / 吉藤英明   ipv6: Remove unus...
1923
  	dst = icmp6_dst_alloc(skb->dev, &fl6);
452edd598   David S. Miller   xfrm: Return dst ...
1924
1925
  	if (IS_ERR(dst)) {
  		err = PTR_ERR(dst);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1926
  		goto err_out;
452edd598   David S. Miller   xfrm: Return dst ...
1927
  	}
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1928

adf30907d   Eric Dumazet   net: skb->dst acc...
1929
  	skb_dst_set(skb, dst);
29a26a568   Eric W. Biederman   netfilter: Pass s...
1930
1931
  	err = NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_OUT,
  		      net, sk, skb, NULL, skb->dev,
13206b6bf   Eric W. Biederman   net: Pass net int...
1932
  		      dst_output);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1933
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1934
  	if (!err) {
5c5d244bd   Denis V. Lunev   ipv6: added net a...
1935
  		ICMP6MSGOUT_INC_STATS(net, idev, type);
a862f6a6d   Denis V. Lunev   ipv6: added net a...
1936
  		ICMP6_INC_STATS(net, idev, ICMP6_MIB_OUTMSGS);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1937
  	} else
3bd653c84   Denis V. Lunev   netns: add net pa...
1938
  		IP6_INC_STATS(net, idev, IPSTATS_MIB_OUTDISCARDS);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1939

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1940
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1941
  	return;
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1942
1943
1944
1945
  
  err_out:
  	kfree_skb(skb);
  	goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1946
  }
6a7cc4187   Flavio Leitner   ipv6: send Change...
1947
  static void mld_send_initial_cr(struct inet6_dev *idev)
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
1948
  {
6a7cc4187   Flavio Leitner   ipv6: send Change...
1949
1950
1951
1952
1953
1954
1955
1956
1957
  	struct sk_buff *skb;
  	struct ifmcaddr6 *pmc;
  	int type;
  
  	if (mld_in_v1_mode(idev))
  		return;
  
  	skb = NULL;
  	read_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
1958
  	for (pmc = idev->mc_list; pmc; pmc = pmc->next) {
6a7cc4187   Flavio Leitner   ipv6: send Change...
1959
1960
1961
1962
1963
1964
1965
  		spin_lock_bh(&pmc->mca_lock);
  		if (pmc->mca_sfcount[MCAST_EXCLUDE])
  			type = MLD2_CHANGE_TO_EXCLUDE;
  		else
  			type = MLD2_CHANGE_TO_INCLUDE;
  		skb = add_grec(skb, pmc, type, 0, 0, 1);
  		spin_unlock_bh(&pmc->mca_lock);
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
1966
  	}
6a7cc4187   Flavio Leitner   ipv6: send Change...
1967
1968
1969
  	read_unlock_bh(&idev->lock);
  	if (skb)
  		mld_sendpack(skb);
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
1970
1971
1972
1973
1974
1975
  }
  
  void ipv6_mc_dad_complete(struct inet6_dev *idev)
  {
  	idev->mc_dad_count = idev->mc_qrv;
  	if (idev->mc_dad_count) {
6a7cc4187   Flavio Leitner   ipv6: send Change...
1976
  		mld_send_initial_cr(idev);
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
1977
1978
1979
1980
1981
1982
1983
1984
1985
  		idev->mc_dad_count--;
  		if (idev->mc_dad_count)
  			mld_dad_start_timer(idev, idev->mc_maxdelay);
  	}
  }
  
  static void mld_dad_timer_expire(unsigned long data)
  {
  	struct inet6_dev *idev = (struct inet6_dev *)data;
6a7cc4187   Flavio Leitner   ipv6: send Change...
1986
  	mld_send_initial_cr(idev);
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
1987
1988
1989
1990
1991
  	if (idev->mc_dad_count) {
  		idev->mc_dad_count--;
  		if (idev->mc_dad_count)
  			mld_dad_start_timer(idev, idev->mc_maxdelay);
  	}
9260d3e10   Salam Noureddine   ipv6 mcast: use i...
1992
  	in6_dev_put(idev);
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
1993
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1994
  static int ip6_mc_del1_src(struct ifmcaddr6 *pmc, int sfmode,
b71d1d426   Eric Dumazet   inet: constify ip...
1995
  	const struct in6_addr *psfsrc)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1996
1997
1998
1999
2000
  {
  	struct ip6_sf_list *psf, *psf_prev;
  	int rv = 0;
  
  	psf_prev = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
2001
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2002
2003
2004
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
  		if (ipv6_addr_equal(&psf->sf_addr, psfsrc))
  			break;
  		psf_prev = psf;
  	}
  	if (!psf || psf->sf_count[sfmode] == 0) {
  		/* source filter not found, or count wrong =>  bug */
  		return -ESRCH;
  	}
  	psf->sf_count[sfmode]--;
  	if (!psf->sf_count[MCAST_INCLUDE] && !psf->sf_count[MCAST_EXCLUDE]) {
  		struct inet6_dev *idev = pmc->idev;
  
  		/* no more filters for this source */
  		if (psf_prev)
  			psf_prev->sf_next = psf->sf_next;
  		else
  			pmc->mca_sources = psf->sf_next;
  		if (psf->sf_oldin && !(pmc->mca_flags & MAF_NOREPORT) &&
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
2020
  		    !mld_in_v1_mode(idev)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2021
2022
2023
2024
2025
2026
2027
2028
2029
  			psf->sf_crcount = idev->mc_qrv;
  			psf->sf_next = pmc->mca_tomb;
  			pmc->mca_tomb = psf;
  			rv = 1;
  		} else
  			kfree(psf);
  	}
  	return rv;
  }
b71d1d426   Eric Dumazet   inet: constify ip...
2030
2031
  static int ip6_mc_del_src(struct inet6_dev *idev, const struct in6_addr *pmca,
  			  int sfmode, int sfcount, const struct in6_addr *psfsrc,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2032
2033
2034
2035
2036
2037
2038
2039
2040
  			  int delta)
  {
  	struct ifmcaddr6 *pmc;
  	int	changerec = 0;
  	int	i, err;
  
  	if (!idev)
  		return -ENODEV;
  	read_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
2041
  	for (pmc = idev->mc_list; pmc; pmc = pmc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2042
2043
2044
2045
2046
2047
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
  		if (ipv6_addr_equal(pmca, &pmc->mca_addr))
  			break;
  	}
  	if (!pmc) {
  		/* MCA not found?? bug */
  		read_unlock_bh(&idev->lock);
  		return -ESRCH;
  	}
  	spin_lock_bh(&pmc->mca_lock);
  	sf_markstate(pmc);
  	if (!delta) {
  		if (!pmc->mca_sfcount[sfmode]) {
  			spin_unlock_bh(&pmc->mca_lock);
  			read_unlock_bh(&idev->lock);
  			return -EINVAL;
  		}
  		pmc->mca_sfcount[sfmode]--;
  	}
  	err = 0;
67ba4152e   Ian Morris   ipv6: White-space...
2061
  	for (i = 0; i < sfcount; i++) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2062
2063
2064
2065
2066
2067
2068
2069
2070
2071
2072
2073
2074
2075
2076
  		int rv = ip6_mc_del1_src(pmc, sfmode, &psfsrc[i]);
  
  		changerec |= rv > 0;
  		if (!err && rv < 0)
  			err = rv;
  	}
  	if (pmc->mca_sfmode == MCAST_EXCLUDE &&
  	    pmc->mca_sfcount[MCAST_EXCLUDE] == 0 &&
  	    pmc->mca_sfcount[MCAST_INCLUDE]) {
  		struct ip6_sf_list *psf;
  
  		/* filter mode change */
  		pmc->mca_sfmode = MCAST_INCLUDE;
  		pmc->mca_crcount = idev->mc_qrv;
  		idev->mc_ifc_count = pmc->mca_crcount;
67ba4152e   Ian Morris   ipv6: White-space...
2077
  		for (psf = pmc->mca_sources; psf; psf = psf->sf_next)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
  			psf->sf_crcount = 0;
  		mld_ifc_event(pmc->idev);
  	} else if (sf_setstate(pmc) || changerec)
  		mld_ifc_event(pmc->idev);
  	spin_unlock_bh(&pmc->mca_lock);
  	read_unlock_bh(&idev->lock);
  	return err;
  }
  
  /*
   * Add multicast single-source filter to the interface list
   */
  static int ip6_mc_add1_src(struct ifmcaddr6 *pmc, int sfmode,
99d2f47aa   Jun Zhao   ipv6 : mcast : De...
2091
  	const struct in6_addr *psfsrc)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2092
2093
2094
2095
  {
  	struct ip6_sf_list *psf, *psf_prev;
  
  	psf_prev = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
2096
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2097
2098
2099
2100
2101
  		if (ipv6_addr_equal(&psf->sf_addr, psfsrc))
  			break;
  		psf_prev = psf;
  	}
  	if (!psf) {
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
2102
  		psf = kzalloc(sizeof(*psf), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2103
2104
  		if (!psf)
  			return -ENOBUFS;
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
2105

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
2119
  		psf->sf_addr = *psfsrc;
  		if (psf_prev) {
  			psf_prev->sf_next = psf;
  		} else
  			pmc->mca_sources = psf;
  	}
  	psf->sf_count[sfmode]++;
  	return 0;
  }
  
  static void sf_markstate(struct ifmcaddr6 *pmc)
  {
  	struct ip6_sf_list *psf;
  	int mca_xcount = pmc->mca_sfcount[MCAST_EXCLUDE];
67ba4152e   Ian Morris   ipv6: White-space...
2120
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2121
2122
2123
2124
2125
2126
2127
2128
2129
2130
  		if (pmc->mca_sfcount[MCAST_EXCLUDE]) {
  			psf->sf_oldin = mca_xcount ==
  				psf->sf_count[MCAST_EXCLUDE] &&
  				!psf->sf_count[MCAST_INCLUDE];
  		} else
  			psf->sf_oldin = psf->sf_count[MCAST_INCLUDE] != 0;
  }
  
  static int sf_setstate(struct ifmcaddr6 *pmc)
  {
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
2131
  	struct ip6_sf_list *psf, *dpsf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2132
2133
2134
2135
2136
  	int mca_xcount = pmc->mca_sfcount[MCAST_EXCLUDE];
  	int qrv = pmc->idev->mc_qrv;
  	int new_in, rv;
  
  	rv = 0;
67ba4152e   Ian Morris   ipv6: White-space...
2137
  	for (psf = pmc->mca_sources; psf; psf = psf->sf_next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2138
2139
2140
2141
2142
  		if (pmc->mca_sfcount[MCAST_EXCLUDE]) {
  			new_in = mca_xcount == psf->sf_count[MCAST_EXCLUDE] &&
  				!psf->sf_count[MCAST_INCLUDE];
  		} else
  			new_in = psf->sf_count[MCAST_INCLUDE] != 0;
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
2143
2144
  		if (new_in) {
  			if (!psf->sf_oldin) {
e80e28b6b   Al Viro   [PATCH] net/ipv6/...
2145
  				struct ip6_sf_list *prev = NULL;
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
2146

67ba4152e   Ian Morris   ipv6: White-space...
2147
2148
  				for (dpsf = pmc->mca_tomb; dpsf;
  				     dpsf = dpsf->sf_next) {
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
2149
2150
2151
2152
2153
2154
2155
2156
2157
2158
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
  					if (ipv6_addr_equal(&dpsf->sf_addr,
  					    &psf->sf_addr))
  						break;
  					prev = dpsf;
  				}
  				if (dpsf) {
  					if (prev)
  						prev->sf_next = dpsf->sf_next;
  					else
  						pmc->mca_tomb = dpsf->sf_next;
  					kfree(dpsf);
  				}
  				psf->sf_crcount = qrv;
  				rv++;
  			}
  		} else if (psf->sf_oldin) {
  			psf->sf_crcount = 0;
  			/*
  			 * add or update "delete" records if an active filter
  			 * is now inactive
  			 */
67ba4152e   Ian Morris   ipv6: White-space...
2170
  			for (dpsf = pmc->mca_tomb; dpsf; dpsf = dpsf->sf_next)
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
2171
2172
2173
2174
  				if (ipv6_addr_equal(&dpsf->sf_addr,
  				    &psf->sf_addr))
  					break;
  			if (!dpsf) {
5d55354f1   Joe Perches   net/ipv6/mcast.c:...
2175
  				dpsf = kmalloc(sizeof(*dpsf), GFP_ATOMIC);
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
2176
2177
2178
2179
2180
2181
2182
2183
  				if (!dpsf)
  					continue;
  				*dpsf = *psf;
  				/* pmc->mca_lock held by callers */
  				dpsf->sf_next = pmc->mca_tomb;
  				pmc->mca_tomb = dpsf;
  			}
  			dpsf->sf_crcount = qrv;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2184
2185
2186
2187
2188
2189
2190
2191
2192
  			rv++;
  		}
  	}
  	return rv;
  }
  
  /*
   * Add multicast source filter list to the interface list
   */
b71d1d426   Eric Dumazet   inet: constify ip...
2193
2194
  static int ip6_mc_add_src(struct inet6_dev *idev, const struct in6_addr *pmca,
  			  int sfmode, int sfcount, const struct in6_addr *psfsrc,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2195
2196
2197
2198
2199
2200
2201
2202
2203
  			  int delta)
  {
  	struct ifmcaddr6 *pmc;
  	int	isexclude;
  	int	i, err;
  
  	if (!idev)
  		return -ENODEV;
  	read_lock_bh(&idev->lock);
67ba4152e   Ian Morris   ipv6: White-space...
2204
  	for (pmc = idev->mc_list; pmc; pmc = pmc->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2205
2206
2207
2208
2209
2210
2211
2212
2213
2214
2215
2216
2217
2218
2219
  		if (ipv6_addr_equal(pmca, &pmc->mca_addr))
  			break;
  	}
  	if (!pmc) {
  		/* MCA not found?? bug */
  		read_unlock_bh(&idev->lock);
  		return -ESRCH;
  	}
  	spin_lock_bh(&pmc->mca_lock);
  
  	sf_markstate(pmc);
  	isexclude = pmc->mca_sfmode == MCAST_EXCLUDE;
  	if (!delta)
  		pmc->mca_sfcount[sfmode]++;
  	err = 0;
67ba4152e   Ian Morris   ipv6: White-space...
2220
  	for (i = 0; i < sfcount; i++) {
99d2f47aa   Jun Zhao   ipv6 : mcast : De...
2221
  		err = ip6_mc_add1_src(pmc, sfmode, &psfsrc[i]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2222
2223
2224
2225
2226
2227
2228
2229
  		if (err)
  			break;
  	}
  	if (err) {
  		int j;
  
  		if (!delta)
  			pmc->mca_sfcount[sfmode]--;
67ba4152e   Ian Morris   ipv6: White-space...
2230
  		for (j = 0; j < i; j++)
78d50217b   RongQing.Li   ipv6: fix array i...
2231
  			ip6_mc_del1_src(pmc, sfmode, &psfsrc[j]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2232
  	} else if (isexclude != (pmc->mca_sfcount[MCAST_EXCLUDE] != 0)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
  		struct ip6_sf_list *psf;
  
  		/* filter mode change */
  		if (pmc->mca_sfcount[MCAST_EXCLUDE])
  			pmc->mca_sfmode = MCAST_EXCLUDE;
  		else if (pmc->mca_sfcount[MCAST_INCLUDE])
  			pmc->mca_sfmode = MCAST_INCLUDE;
  		/* else no filters; keep old mode for reports */
  
  		pmc->mca_crcount = idev->mc_qrv;
  		idev->mc_ifc_count = pmc->mca_crcount;
67ba4152e   Ian Morris   ipv6: White-space...
2244
  		for (psf = pmc->mca_sources; psf; psf = psf->sf_next)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
2255
2256
  			psf->sf_crcount = 0;
  		mld_ifc_event(idev);
  	} else if (sf_setstate(pmc))
  		mld_ifc_event(idev);
  	spin_unlock_bh(&pmc->mca_lock);
  	read_unlock_bh(&idev->lock);
  	return err;
  }
  
  static void ip6_mc_clear_src(struct ifmcaddr6 *pmc)
  {
  	struct ip6_sf_list *psf, *nextpsf;
67ba4152e   Ian Morris   ipv6: White-space...
2257
  	for (psf = pmc->mca_tomb; psf; psf = nextpsf) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2258
2259
2260
2261
  		nextpsf = psf->sf_next;
  		kfree(psf);
  	}
  	pmc->mca_tomb = NULL;
67ba4152e   Ian Morris   ipv6: White-space...
2262
  	for (psf = pmc->mca_sources; psf; psf = nextpsf) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2263
2264
2265
2266
2267
  		nextpsf = psf->sf_next;
  		kfree(psf);
  	}
  	pmc->mca_sources = NULL;
  	pmc->mca_sfmode = MCAST_EXCLUDE;
de9daad90   Denis Lukianov   [MCAST]: Fix MCAS...
2268
  	pmc->mca_sfcount[MCAST_INCLUDE] = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2269
2270
2271
2272
2273
2274
2275
2276
2277
2278
2279
2280
  	pmc->mca_sfcount[MCAST_EXCLUDE] = 1;
  }
  
  
  static void igmp6_join_group(struct ifmcaddr6 *ma)
  {
  	unsigned long delay;
  
  	if (ma->mca_flags & MAF_NOREPORT)
  		return;
  
  	igmp6_send(&ma->mca_addr, ma->idev->dev, ICMPV6_MGM_REPORT);
63862b5be   Aruna-Hewapathirane   net: replace macr...
2281
  	delay = prandom_u32() % unsolicited_report_interval(ma->idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
2292
2293
2294
2295
2296
2297
2298
  
  	spin_lock_bh(&ma->mca_lock);
  	if (del_timer(&ma->mca_timer)) {
  		atomic_dec(&ma->mca_refcnt);
  		delay = ma->mca_timer.expires - jiffies;
  	}
  
  	if (!mod_timer(&ma->mca_timer, jiffies + delay))
  		atomic_inc(&ma->mca_refcnt);
  	ma->mca_flags |= MAF_TIMER_RUNNING | MAF_LAST_REPORTER;
  	spin_unlock_bh(&ma->mca_lock);
  }
  
  static int ip6_mc_leave_src(struct sock *sk, struct ipv6_mc_socklist *iml,
  			    struct inet6_dev *idev)
  {
  	int err;
f7ed925c1   WANG Cong   ipv6: update the ...
2299
  	/* callers have the socket lock and rtnl lock
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
2300
2301
  	 * so no other readers or writers of iml or its sflist
  	 */
cfcabdcc2   Stephen Hemminger   [NET]: sparse war...
2302
  	if (!iml->sflist) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2303
2304
2305
2306
2307
2308
2309
2310
2311
2312
2313
2314
  		/* any-source empty exclude case */
  		return ip6_mc_del_src(idev, &iml->addr, iml->sfmode, 0, NULL, 0);
  	}
  	err = ip6_mc_del_src(idev, &iml->addr, iml->sfmode,
  		iml->sflist->sl_count, iml->sflist->sl_addr, 0);
  	sock_kfree_s(sk, iml->sflist, IP6_SFLSIZE(iml->sflist->sl_max));
  	iml->sflist = NULL;
  	return err;
  }
  
  static void igmp6_leave_group(struct ifmcaddr6 *ma)
  {
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
2315
  	if (mld_in_v1_mode(ma->idev)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2316
2317
2318
2319
2320
2321
2322
2323
2324
2325
2326
2327
2328
2329
2330
  		if (ma->mca_flags & MAF_LAST_REPORTER)
  			igmp6_send(&ma->mca_addr, ma->idev->dev,
  				ICMPV6_MGM_REDUCTION);
  	} else {
  		mld_add_delrec(ma->idev, ma);
  		mld_ifc_event(ma->idev);
  	}
  }
  
  static void mld_gq_timer_expire(unsigned long data)
  {
  	struct inet6_dev *idev = (struct inet6_dev *)data;
  
  	idev->mc_gq_running = 0;
  	mld_send_report(idev, NULL);
9260d3e10   Salam Noureddine   ipv6 mcast: use i...
2331
  	in6_dev_put(idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2332
2333
2334
2335
2336
2337
2338
2339
2340
2341
2342
2343
  }
  
  static void mld_ifc_timer_expire(unsigned long data)
  {
  	struct inet6_dev *idev = (struct inet6_dev *)data;
  
  	mld_send_cr(idev);
  	if (idev->mc_ifc_count) {
  		idev->mc_ifc_count--;
  		if (idev->mc_ifc_count)
  			mld_ifc_start_timer(idev, idev->mc_maxdelay);
  	}
9260d3e10   Salam Noureddine   ipv6 mcast: use i...
2344
  	in6_dev_put(idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2345
2346
2347
2348
  }
  
  static void mld_ifc_event(struct inet6_dev *idev)
  {
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
2349
  	if (mld_in_v1_mode(idev))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2350
2351
2352
2353
2354
2355
2356
2357
2358
  		return;
  	idev->mc_ifc_count = idev->mc_qrv;
  	mld_ifc_start_timer(idev, 1);
  }
  
  
  static void igmp6_timer_handler(unsigned long data)
  {
  	struct ifmcaddr6 *ma = (struct ifmcaddr6 *) data;
6c567b78c   Daniel Borkmann   net: ipv6: mld: c...
2359
  	if (mld_in_v1_mode(ma->idev))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2360
2361
2362
2363
2364
2365
2366
2367
2368
2369
  		igmp6_send(&ma->mca_addr, ma->idev->dev, ICMPV6_MGM_REPORT);
  	else
  		mld_send_report(ma->idev, ma);
  
  	spin_lock(&ma->mca_lock);
  	ma->mca_flags |=  MAF_LAST_REPORTER;
  	ma->mca_flags &= ~MAF_TIMER_RUNNING;
  	spin_unlock(&ma->mca_lock);
  	ma_put(ma);
  }
75c78500d   Moni Shoua   bonding: remap mu...
2370
2371
2372
2373
2374
2375
2376
2377
2378
2379
2380
2381
2382
2383
2384
2385
2386
2387
  /* Device changing type */
  
  void ipv6_mc_unmap(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *i;
  
  	/* Install multicast list, except for all-nodes (already installed) */
  
  	read_lock_bh(&idev->lock);
  	for (i = idev->mc_list; i; i = i->next)
  		igmp6_group_dropped(i);
  	read_unlock_bh(&idev->lock);
  }
  
  void ipv6_mc_remap(struct inet6_dev *idev)
  {
  	ipv6_mc_up(idev);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2388
2389
2390
2391
2392
2393
2394
2395
2396
  /* Device going down */
  
  void ipv6_mc_down(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *i;
  
  	/* Withdraw multicast list */
  
  	read_lock_bh(&idev->lock);
b4af8def5   Daniel Borkmann   net: ipv6: mld: i...
2397
2398
2399
  	mld_ifc_stop_timer(idev);
  	mld_gq_stop_timer(idev);
  	mld_dad_stop_timer(idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2400

67ba4152e   Ian Morris   ipv6: White-space...
2401
  	for (i = idev->mc_list; i; i = i->next)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2402
2403
2404
2405
2406
  		igmp6_group_dropped(i);
  	read_unlock_bh(&idev->lock);
  
  	mld_clear_delrec(idev);
  }
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
2407
2408
2409
2410
2411
2412
2413
2414
  static void ipv6_mc_reset(struct inet6_dev *idev)
  {
  	idev->mc_qrv = sysctl_mld_qrv;
  	idev->mc_qi = MLD_QI_DEFAULT;
  	idev->mc_qri = MLD_QRI_DEFAULT;
  	idev->mc_v1_seen = 0;
  	idev->mc_maxdelay = unsolicited_report_interval(idev);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2415
2416
2417
2418
2419
2420
2421
2422
2423
2424
  
  /* Device going up */
  
  void ipv6_mc_up(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *i;
  
  	/* Install multicast list, except for all-nodes (already installed) */
  
  	read_lock_bh(&idev->lock);
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
2425
  	ipv6_mc_reset(idev);
67ba4152e   Ian Morris   ipv6: White-space...
2426
  	for (i = idev->mc_list; i; i = i->next)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2427
2428
2429
2430
2431
2432
2433
2434
  		igmp6_group_added(i);
  	read_unlock_bh(&idev->lock);
  }
  
  /* IPv6 device initialization. */
  
  void ipv6_mc_init_dev(struct inet6_dev *idev)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2435
  	write_lock_bh(&idev->lock);
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
2436
  	spin_lock_init(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2437
  	idev->mc_gq_running = 0;
b24b8a247   Pavel Emelyanov   [NET]: Convert in...
2438
2439
  	setup_timer(&idev->mc_gq_timer, mld_gq_timer_expire,
  			(unsigned long)idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2440
2441
  	idev->mc_tomb = NULL;
  	idev->mc_ifc_count = 0;
b24b8a247   Pavel Emelyanov   [NET]: Convert in...
2442
2443
  	setup_timer(&idev->mc_ifc_timer, mld_ifc_timer_expire,
  			(unsigned long)idev);
b173ee488   Hannes Frederic Sowa   ipv6: resend MLD ...
2444
2445
  	setup_timer(&idev->mc_dad_timer, mld_dad_timer_expire,
  		    (unsigned long)idev);
2f711939d   Hannes Frederic Sowa   ipv6: add sysctl_...
2446
  	ipv6_mc_reset(idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2447
  	write_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2448
2449
2450
2451
2452
2453
2454
2455
2456
  }
  
  /*
   *	Device is about to be destroyed: clean up.
   */
  
  void ipv6_mc_destroy_dev(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *i;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2457
2458
2459
2460
2461
  
  	/* Deactivate timers */
  	ipv6_mc_down(idev);
  
  	/* Delete all-nodes address. */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2462
2463
2464
2465
  	/* We cannot call ipv6_dev_mc_dec() directly, our caller in
  	 * addrconf.c has NULL'd out dev->ip6_ptr so in6_dev_get() will
  	 * fail.
  	 */
f3ee4010e   YOSHIFUJI Hideaki   [IPV6]: Define co...
2466
  	__ipv6_dev_mc_dec(idev, &in6addr_linklocal_allnodes);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2467

f3ee4010e   YOSHIFUJI Hideaki   [IPV6]: Define co...
2468
2469
  	if (idev->cnf.forwarding)
  		__ipv6_dev_mc_dec(idev, &in6addr_linklocal_allrouters);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2470
2471
2472
2473
2474
2475
2476
2477
2478
2479
2480
2481
2482
2483
2484
2485
  
  	write_lock_bh(&idev->lock);
  	while ((i = idev->mc_list) != NULL) {
  		idev->mc_list = i->next;
  		write_unlock_bh(&idev->lock);
  
  		igmp6_group_dropped(i);
  		ma_put(i);
  
  		write_lock_bh(&idev->lock);
  	}
  	write_unlock_bh(&idev->lock);
  }
  
  #ifdef CONFIG_PROC_FS
  struct igmp6_mc_iter_state {
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2486
  	struct seq_net_private p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2487
2488
2489
2490
2491
2492
2493
2494
2495
2496
  	struct net_device *dev;
  	struct inet6_dev *idev;
  };
  
  #define igmp6_mc_seq_private(seq)	((struct igmp6_mc_iter_state *)(seq)->private)
  
  static inline struct ifmcaddr6 *igmp6_mc_get_first(struct seq_file *seq)
  {
  	struct ifmcaddr6 *im = NULL;
  	struct igmp6_mc_iter_state *state = igmp6_mc_seq_private(seq);
1218854af   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
2497
  	struct net *net = seq_file_net(seq);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2498

7562f876c   Pavel Emelianov   [NET]: Rework dev...
2499
  	state->idev = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2500
  	for_each_netdev_rcu(net, state->dev) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2501
  		struct inet6_dev *idev;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2502
  		idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2503
2504
2505
2506
2507
2508
2509
2510
2511
  		if (!idev)
  			continue;
  		read_lock_bh(&idev->lock);
  		im = idev->mc_list;
  		if (im) {
  			state->idev = idev;
  			break;
  		}
  		read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2512
2513
2514
2515
2516
2517
2518
2519
2520
2521
  	}
  	return im;
  }
  
  static struct ifmcaddr6 *igmp6_mc_get_next(struct seq_file *seq, struct ifmcaddr6 *im)
  {
  	struct igmp6_mc_iter_state *state = igmp6_mc_seq_private(seq);
  
  	im = im->next;
  	while (!im) {
53b24b8f9   Ian Morris   ipv6: coding styl...
2522
  		if (likely(state->idev))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2523
  			read_unlock_bh(&state->idev->lock);
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2524
2525
  
  		state->dev = next_net_device_rcu(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2526
2527
2528
2529
  		if (!state->dev) {
  			state->idev = NULL;
  			break;
  		}
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2530
  		state->idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2531
2532
2533
2534
2535
2536
2537
2538
2539
2540
2541
2542
2543
2544
2545
2546
2547
2548
  		if (!state->idev)
  			continue;
  		read_lock_bh(&state->idev->lock);
  		im = state->idev->mc_list;
  	}
  	return im;
  }
  
  static struct ifmcaddr6 *igmp6_mc_get_idx(struct seq_file *seq, loff_t pos)
  {
  	struct ifmcaddr6 *im = igmp6_mc_get_first(seq);
  	if (im)
  		while (pos && (im = igmp6_mc_get_next(seq, im)) != NULL)
  			--pos;
  	return pos ? NULL : im;
  }
  
  static void *igmp6_mc_seq_start(struct seq_file *seq, loff_t *pos)
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2549
  	__acquires(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2550
  {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2551
  	rcu_read_lock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2552
2553
2554
2555
2556
  	return igmp6_mc_get_idx(seq, *pos);
  }
  
  static void *igmp6_mc_seq_next(struct seq_file *seq, void *v, loff_t *pos)
  {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2557
  	struct ifmcaddr6 *im = igmp6_mc_get_next(seq, v);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2558
2559
2560
2561
2562
  	++*pos;
  	return im;
  }
  
  static void igmp6_mc_seq_stop(struct seq_file *seq, void *v)
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2563
  	__releases(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2564
2565
  {
  	struct igmp6_mc_iter_state *state = igmp6_mc_seq_private(seq);
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2566

53b24b8f9   Ian Morris   ipv6: coding styl...
2567
  	if (likely(state->idev)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2568
  		read_unlock_bh(&state->idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2569
2570
2571
  		state->idev = NULL;
  	}
  	state->dev = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2572
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2573
2574
2575
2576
2577
2578
2579
2580
  }
  
  static int igmp6_mc_seq_show(struct seq_file *seq, void *v)
  {
  	struct ifmcaddr6 *im = (struct ifmcaddr6 *)v;
  	struct igmp6_mc_iter_state *state = igmp6_mc_seq_private(seq);
  
  	seq_printf(seq,
4b7a4274c   Harvey Harrison   net: replace %#p6...
2581
2582
  		   "%-4d %-15s %pi6 %5d %08X %ld
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2583
  		   state->dev->ifindex, state->dev->name,
b071195de   Harvey Harrison   net: replace all ...
2584
  		   &im->mca_addr,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2585
2586
2587
2588
2589
  		   im->mca_users, im->mca_flags,
  		   (im->mca_flags&MAF_TIMER_RUNNING) ?
  		   jiffies_to_clock_t(im->mca_timer.expires-jiffies) : 0);
  	return 0;
  }
56b3d975b   Philippe De Muyter   [NET]: Make all i...
2590
  static const struct seq_operations igmp6_mc_seq_ops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2591
2592
2593
2594
2595
2596
2597
2598
  	.start	=	igmp6_mc_seq_start,
  	.next	=	igmp6_mc_seq_next,
  	.stop	=	igmp6_mc_seq_stop,
  	.show	=	igmp6_mc_seq_show,
  };
  
  static int igmp6_mc_seq_open(struct inode *inode, struct file *file)
  {
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2599
2600
  	return seq_open_net(inode, file, &igmp6_mc_seq_ops,
  			    sizeof(struct igmp6_mc_iter_state));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2601
  }
9a32144e9   Arjan van de Ven   [PATCH] mark stru...
2602
  static const struct file_operations igmp6_mc_seq_fops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2603
2604
2605
2606
  	.owner		=	THIS_MODULE,
  	.open		=	igmp6_mc_seq_open,
  	.read		=	seq_read,
  	.llseek		=	seq_lseek,
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2607
  	.release	=	seq_release_net,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2608
2609
2610
  };
  
  struct igmp6_mcf_iter_state {
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2611
  	struct seq_net_private p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2612
2613
2614
2615
2616
2617
2618
2619
2620
2621
2622
2623
  	struct net_device *dev;
  	struct inet6_dev *idev;
  	struct ifmcaddr6 *im;
  };
  
  #define igmp6_mcf_seq_private(seq)	((struct igmp6_mcf_iter_state *)(seq)->private)
  
  static inline struct ip6_sf_list *igmp6_mcf_get_first(struct seq_file *seq)
  {
  	struct ip6_sf_list *psf = NULL;
  	struct ifmcaddr6 *im = NULL;
  	struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq);
1218854af   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
2624
  	struct net *net = seq_file_net(seq);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2625

7562f876c   Pavel Emelianov   [NET]: Rework dev...
2626
2627
  	state->idev = NULL;
  	state->im = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2628
  	for_each_netdev_rcu(net, state->dev) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2629
  		struct inet6_dev *idev;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2630
  		idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2631
2632
2633
2634
  		if (unlikely(idev == NULL))
  			continue;
  		read_lock_bh(&idev->lock);
  		im = idev->mc_list;
53b24b8f9   Ian Morris   ipv6: coding styl...
2635
  		if (likely(im)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2636
2637
  			spin_lock_bh(&im->mca_lock);
  			psf = im->mca_sources;
53b24b8f9   Ian Morris   ipv6: coding styl...
2638
  			if (likely(psf)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2639
2640
2641
2642
2643
2644
2645
  				state->im = im;
  				state->idev = idev;
  				break;
  			}
  			spin_unlock_bh(&im->mca_lock);
  		}
  		read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2646
2647
2648
2649
2650
2651
2652
2653
2654
2655
2656
2657
2658
  	}
  	return psf;
  }
  
  static struct ip6_sf_list *igmp6_mcf_get_next(struct seq_file *seq, struct ip6_sf_list *psf)
  {
  	struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq);
  
  	psf = psf->sf_next;
  	while (!psf) {
  		spin_unlock_bh(&state->im->mca_lock);
  		state->im = state->im->next;
  		while (!state->im) {
53b24b8f9   Ian Morris   ipv6: coding styl...
2659
  			if (likely(state->idev))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2660
  				read_unlock_bh(&state->idev->lock);
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2661
2662
  
  			state->dev = next_net_device_rcu(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2663
2664
2665
2666
  			if (!state->dev) {
  				state->idev = NULL;
  				goto out;
  			}
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2667
  			state->idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2668
2669
2670
2671
2672
2673
2674
2675
2676
2677
2678
2679
2680
2681
2682
2683
2684
2685
2686
2687
2688
2689
2690
2691
  			if (!state->idev)
  				continue;
  			read_lock_bh(&state->idev->lock);
  			state->im = state->idev->mc_list;
  		}
  		if (!state->im)
  			break;
  		spin_lock_bh(&state->im->mca_lock);
  		psf = state->im->mca_sources;
  	}
  out:
  	return psf;
  }
  
  static struct ip6_sf_list *igmp6_mcf_get_idx(struct seq_file *seq, loff_t pos)
  {
  	struct ip6_sf_list *psf = igmp6_mcf_get_first(seq);
  	if (psf)
  		while (pos && (psf = igmp6_mcf_get_next(seq, psf)) != NULL)
  			--pos;
  	return pos ? NULL : psf;
  }
  
  static void *igmp6_mcf_seq_start(struct seq_file *seq, loff_t *pos)
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2692
  	__acquires(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2693
  {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2694
  	rcu_read_lock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2695
2696
2697
2698
2699
2700
2701
2702
2703
2704
2705
2706
2707
2708
2709
  	return *pos ? igmp6_mcf_get_idx(seq, *pos - 1) : SEQ_START_TOKEN;
  }
  
  static void *igmp6_mcf_seq_next(struct seq_file *seq, void *v, loff_t *pos)
  {
  	struct ip6_sf_list *psf;
  	if (v == SEQ_START_TOKEN)
  		psf = igmp6_mcf_get_first(seq);
  	else
  		psf = igmp6_mcf_get_next(seq, v);
  	++*pos;
  	return psf;
  }
  
  static void igmp6_mcf_seq_stop(struct seq_file *seq, void *v)
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2710
  	__releases(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2711
2712
  {
  	struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq);
53b24b8f9   Ian Morris   ipv6: coding styl...
2713
  	if (likely(state->im)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2714
2715
2716
  		spin_unlock_bh(&state->im->mca_lock);
  		state->im = NULL;
  	}
53b24b8f9   Ian Morris   ipv6: coding styl...
2717
  	if (likely(state->idev)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2718
  		read_unlock_bh(&state->idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2719
2720
2721
  		state->idev = NULL;
  	}
  	state->dev = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2722
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2723
2724
2725
2726
2727
2728
2729
2730
  }
  
  static int igmp6_mcf_seq_show(struct seq_file *seq, void *v)
  {
  	struct ip6_sf_list *psf = (struct ip6_sf_list *)v;
  	struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq);
  
  	if (v == SEQ_START_TOKEN) {
1744bea1f   Joe Perches   net: Convert SEQ_...
2731
2732
  		seq_puts(seq, "Idx Device                Multicast Address                   Source Address    INC    EXC
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2733
2734
  	} else {
  		seq_printf(seq,
4b7a4274c   Harvey Harrison   net: replace %#p6...
2735
2736
  			   "%3d %6.6s %pi6 %pi6 %6lu %6lu
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2737
  			   state->dev->ifindex, state->dev->name,
b071195de   Harvey Harrison   net: replace all ...
2738
2739
  			   &state->im->mca_addr,
  			   &psf->sf_addr,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2740
2741
2742
2743
2744
  			   psf->sf_count[MCAST_INCLUDE],
  			   psf->sf_count[MCAST_EXCLUDE]);
  	}
  	return 0;
  }
56b3d975b   Philippe De Muyter   [NET]: Make all i...
2745
  static const struct seq_operations igmp6_mcf_seq_ops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2746
2747
2748
2749
2750
2751
2752
2753
  	.start	=	igmp6_mcf_seq_start,
  	.next	=	igmp6_mcf_seq_next,
  	.stop	=	igmp6_mcf_seq_stop,
  	.show	=	igmp6_mcf_seq_show,
  };
  
  static int igmp6_mcf_seq_open(struct inode *inode, struct file *file)
  {
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2754
2755
  	return seq_open_net(inode, file, &igmp6_mcf_seq_ops,
  			    sizeof(struct igmp6_mcf_iter_state));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2756
  }
9a32144e9   Arjan van de Ven   [PATCH] mark stru...
2757
  static const struct file_operations igmp6_mcf_seq_fops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2758
2759
2760
2761
  	.owner		=	THIS_MODULE,
  	.open		=	igmp6_mcf_seq_open,
  	.read		=	seq_read,
  	.llseek		=	seq_lseek,
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2762
  	.release	=	seq_release_net,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2763
  };
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2764

2c8c1e729   Alexey Dobriyan   net: spread __net...
2765
  static int __net_init igmp6_proc_init(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2766
2767
2768
2769
  {
  	int err;
  
  	err = -ENOMEM;
d4beaa66a   Gao feng   net: proc: change...
2770
  	if (!proc_create("igmp6", S_IRUGO, net->proc_net, &igmp6_mc_seq_fops))
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2771
  		goto out;
d4beaa66a   Gao feng   net: proc: change...
2772
2773
  	if (!proc_create("mcfilter6", S_IRUGO, net->proc_net,
  			 &igmp6_mcf_seq_fops))
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2774
2775
2776
2777
2778
2779
2780
  		goto out_proc_net_igmp6;
  
  	err = 0;
  out:
  	return err;
  
  out_proc_net_igmp6:
ece31ffd5   Gao feng   net: proc: change...
2781
  	remove_proc_entry("igmp6", net->proc_net);
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2782
2783
  	goto out;
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
2784
  static void __net_exit igmp6_proc_exit(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2785
  {
ece31ffd5   Gao feng   net: proc: change...
2786
2787
  	remove_proc_entry("mcfilter6", net->proc_net);
  	remove_proc_entry("igmp6", net->proc_net);
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2788
2789
  }
  #else
2c8c1e729   Alexey Dobriyan   net: spread __net...
2790
  static inline int igmp6_proc_init(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2791
2792
2793
  {
  	return 0;
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
2794
  static inline void igmp6_proc_exit(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2795
  {
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2796
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2797
  #endif
2c8c1e729   Alexey Dobriyan   net: spread __net...
2798
  static int __net_init igmp6_net_init(struct net *net)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2799
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2800
  	int err;
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2801
2802
  	err = inet_ctl_sock_create(&net->ipv6.igmp_sk, PF_INET6,
  				   SOCK_RAW, IPPROTO_ICMPV6, net);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2803
  	if (err < 0) {
f32138319   Joe Perches   net: ipv6: Standa...
2804
2805
  		pr_err("Failed to initialize the IGMP6 control socket (err %d)
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2806
  		       err);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2807
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2808
  	}
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2809
  	inet6_sk(net->ipv6.igmp_sk)->hop_limit = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2810

93a714d6b   Madhu Challa   multicast: Extend...
2811
2812
2813
2814
2815
2816
2817
2818
  	err = inet_ctl_sock_create(&net->ipv6.mc_autojoin_sk, PF_INET6,
  				   SOCK_RAW, IPPROTO_ICMPV6, net);
  	if (err < 0) {
  		pr_err("Failed to initialize the IGMP6 autojoin socket (err %d)
  ",
  		       err);
  		goto out_sock_create;
  	}
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2819
2820
  	err = igmp6_proc_init(net);
  	if (err)
93a714d6b   Madhu Challa   multicast: Extend...
2821
2822
2823
  		goto out_sock_create_autojoin;
  
  	return 0;
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2824

93a714d6b   Madhu Challa   multicast: Extend...
2825
2826
  out_sock_create_autojoin:
  	inet_ctl_sock_destroy(net->ipv6.mc_autojoin_sk);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2827
  out_sock_create:
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2828
  	inet_ctl_sock_destroy(net->ipv6.igmp_sk);
93a714d6b   Madhu Challa   multicast: Extend...
2829
2830
  out:
  	return err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2831
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
2832
  static void __net_exit igmp6_net_exit(struct net *net)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2833
  {
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2834
  	inet_ctl_sock_destroy(net->ipv6.igmp_sk);
93a714d6b   Madhu Challa   multicast: Extend...
2835
  	inet_ctl_sock_destroy(net->ipv6.mc_autojoin_sk);
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2836
  	igmp6_proc_exit(net);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2837
  }
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2838
2839
2840
2841
2842
2843
2844
2845
2846
2847
2848
2849
2850
2851
2852
  
  static struct pernet_operations igmp6_net_ops = {
  	.init = igmp6_net_init,
  	.exit = igmp6_net_exit,
  };
  
  int __init igmp6_init(void)
  {
  	return register_pernet_subsys(&igmp6_net_ops);
  }
  
  void igmp6_cleanup(void)
  {
  	unregister_pernet_subsys(&igmp6_net_ops);
  }