Blame view

net/ipv6/mcast.c 62.4 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>
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
46
  #include <net/mld.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
47
48
49
  
  #include <linux/netfilter.h>
  #include <linux/netfilter_ipv6.h>
457c4cbc5   Eric W. Biederman   [NET]: Make /proc...
50
  #include <net/net_namespace.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
51
52
53
54
55
56
57
58
59
  #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 ...
60
  #include <net/inet_common.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
61
62
63
64
65
66
67
68
69
70
71
  
  #include <net/ip6_checksum.h>
  
  /* Set to 3 to get tracing... */
  #define MCAST_DEBUG 2
  
  #if MCAST_DEBUG >= 3
  #define MDBG(x) printk x
  #else
  #define MDBG(x)
  #endif
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
72
73
74
75
76
  /* 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
77
78
79
80
81
  };
  
  static struct in6_addr mld2_all_mcr = MLD2_ALL_MCR_INIT;
  
  /* Big mc list lock for all the sockets */
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
82
  static DEFINE_SPINLOCK(ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
83

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
84
85
86
87
88
89
90
91
  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...
92
  static void mld_del_delrec(struct inet6_dev *idev, const struct in6_addr *addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
93
94
95
96
  static void mld_clear_delrec(struct inet6_dev *idev);
  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...
97
98
  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
99
  			  int delta);
b71d1d426   Eric Dumazet   inet: constify ip...
100
101
  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
102
103
104
105
106
107
108
  			  int delta);
  static int ip6_mc_leave_src(struct sock *sk, struct ipv6_mc_socklist *iml,
  			    struct inet6_dev *idev);
  
  
  #define IGMP6_UNSOLICITED_IVAL	(10*HZ)
  #define MLD_QRV_DEFAULT		2
53b7997fd   YOSHIFUJI Hideaki   ipv6 netns: Make ...
109
  #define MLD_V1_SEEN(idev) (dev_net((idev)->dev)->ipv6.devconf_all->force_mld_version == 1 || \
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
110
111
112
  		(idev)->cnf.force_mld_version == 1 || \
  		((idev)->mc_v1_seen && \
  		time_before(jiffies, (idev)->mc_v1_seen)))
6f4353d89   David L Stevens   [IPV6]: Increase ...
113
  #define IPV6_MLD_MAX_MSF	64
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
114

ab32ea5d8   Brian Haley   [NET/IPV4/IPV6]: ...
115
  int sysctl_mld_max_msf __read_mostly = IPV6_MLD_MAX_MSF;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
116
117
118
119
  
  /*
   *	socket join on multicast group
   */
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
120
121
122
123
  #define for_each_pmc_rcu(np, pmc)				\
  	for (pmc = rcu_dereference(np->ipv6_mc_list);		\
  	     pmc != NULL;					\
  	     pmc = rcu_dereference(pmc->next))
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
124
  int ipv6_sock_mc_join(struct sock *sk, int ifindex, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
125
126
127
128
  {
  	struct net_device *dev = NULL;
  	struct ipv6_mc_socklist *mc_lst;
  	struct ipv6_pinfo *np = inet6_sk(sk);
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
129
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
130
131
132
133
  	int err;
  
  	if (!ipv6_addr_is_multicast(addr))
  		return -EINVAL;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
134
135
  	rcu_read_lock();
  	for_each_pmc_rcu(np, mc_lst) {
c9e3e8b69   David L Stevens   [IPV6]: multicast...
136
137
  		if ((ifindex == 0 || mc_lst->ifindex == ifindex) &&
  		    ipv6_addr_equal(&mc_lst->addr, addr)) {
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
138
  			rcu_read_unlock();
c9e3e8b69   David L Stevens   [IPV6]: multicast...
139
140
141
  			return -EADDRINUSE;
  		}
  	}
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
142
  	rcu_read_unlock();
c9e3e8b69   David L Stevens   [IPV6]: multicast...
143

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
144
145
146
147
148
149
  	mc_lst = sock_kmalloc(sk, sizeof(struct ipv6_mc_socklist), GFP_KERNEL);
  
  	if (mc_lst == NULL)
  		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

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
152
  	rcu_read_lock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
153
154
  	if (ifindex == 0) {
  		struct rt6_info *rt;
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
155
  		rt = rt6_lookup(net, addr, NULL, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
156
  		if (rt) {
d19185428   David S. Miller   ipv6: Kill rt6i_d...
157
  			dev = rt->dst.dev;
d8d1f30b9   Changli Gao   net-next: remove ...
158
  			dst_release(&rt->dst);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
159
160
  		}
  	} else
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
161
  		dev = dev_get_by_index_rcu(net, ifindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
162
163
  
  	if (dev == NULL) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
164
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
165
166
167
168
169
170
  		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...
171
  	rwlock_init(&mc_lst->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
172
173
174
175
176
177
178
179
180
  	mc_lst->sflist = NULL;
  
  	/*
  	 *	now add/increase the group membership on the device
  	 */
  
  	err = ipv6_dev_mc_inc(dev, addr);
  
  	if (err) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
181
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
182
  		sock_kfree_s(sk, mc_lst, sizeof(*mc_lst));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
183
184
  		return err;
  	}
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
185
  	spin_lock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
186
  	mc_lst->next = np->ipv6_mc_list;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
187
188
  	rcu_assign_pointer(np->ipv6_mc_list, mc_lst);
  	spin_unlock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
189

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
190
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
191
192
193
194
195
196
197
  
  	return 0;
  }
  
  /*
   *	socket leave on multicast group
   */
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
198
  int ipv6_sock_mc_drop(struct sock *sk, int ifindex, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
199
200
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
201
202
  	struct ipv6_mc_socklist *mc_lst;
  	struct ipv6_mc_socklist __rcu **lnk;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
203
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
204

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
205
206
207
208
209
  	spin_lock(&ipv6_sk_mc_lock);
  	for (lnk = &np->ipv6_mc_list;
  	     (mc_lst = rcu_dereference_protected(*lnk,
  			lockdep_is_held(&ipv6_sk_mc_lock))) !=NULL ;
  	      lnk = &mc_lst->next) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
210
211
212
213
214
  		if ((ifindex == 0 || mc_lst->ifindex == ifindex) &&
  		    ipv6_addr_equal(&mc_lst->addr, addr)) {
  			struct net_device *dev;
  
  			*lnk = mc_lst->next;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
215
  			spin_unlock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
216

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
217
218
  			rcu_read_lock();
  			dev = dev_get_by_index_rcu(net, mc_lst->ifindex);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
219
  			if (dev != NULL) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
220
  				struct inet6_dev *idev = __in6_dev_get(dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
221

acd6e00b8   David L Stevens   [MCAST]: Fix filt...
222
  				(void) ip6_mc_leave_src(sk, mc_lst, idev);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
223
  				if (idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
224
  					__ipv6_dev_mc_dec(idev, &mc_lst->addr);
acd6e00b8   David L Stevens   [MCAST]: Fix filt...
225
226
  			} else
  				(void) ip6_mc_leave_src(sk, mc_lst, NULL);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
227
  			rcu_read_unlock();
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
228
  			atomic_sub(sizeof(*mc_lst), &sk->sk_omem_alloc);
e3cbf28fa   Lai Jiangshan   net,rcu: convert ...
229
  			kfree_rcu(mc_lst, rcu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
230
231
232
  			return 0;
  		}
  	}
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
233
  	spin_unlock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
234

9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
235
  	return -EADDRNOTAVAIL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
236
  }
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
237
238
  /* called with rcu_read_lock() */
  static struct inet6_dev *ip6_mc_find_dev_rcu(struct net *net,
b71d1d426   Eric Dumazet   inet: constify ip...
239
  					     const struct in6_addr *group,
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
240
  					     int ifindex)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
241
242
243
244
245
  {
  	struct net_device *dev = NULL;
  	struct inet6_dev *idev = NULL;
  
  	if (ifindex == 0) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
246
  		struct rt6_info *rt = rt6_lookup(net, group, NULL, 0, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
247

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
248
  		if (rt) {
d19185428   David S. Miller   ipv6: Kill rt6i_d...
249
  			dev = rt->dst.dev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
250
  			dev_hold(dev);
d8d1f30b9   Changli Gao   net-next: remove ...
251
  			dst_release(&rt->dst);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
252
253
  		}
  	} else
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
254
  		dev = dev_get_by_index_rcu(net, ifindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
255
256
  
  	if (!dev)
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
257
258
  		return NULL;
  	idev = __in6_dev_get(dev);
448eb71f4   Ilpo Järvinen   ipv6/mcast: join ...
259
  	if (!idev)
8a22c99a8   Joe Perches   net/ipv6/mcast.c:...
260
  		return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
261
  	read_lock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
262
263
264
265
  	if (idev->dead) {
  		read_unlock_bh(&idev->lock);
  		return NULL;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
266
267
268
269
270
271
272
  	return idev;
  }
  
  void ipv6_sock_mc_close(struct sock *sk)
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
  	struct ipv6_mc_socklist *mc_lst;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
273
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
274

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
275
276
277
  	spin_lock(&ipv6_sk_mc_lock);
  	while ((mc_lst = rcu_dereference_protected(np->ipv6_mc_list,
  				lockdep_is_held(&ipv6_sk_mc_lock))) != NULL) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
278
279
280
  		struct net_device *dev;
  
  		np->ipv6_mc_list = mc_lst->next;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
281
  		spin_unlock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
282

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
283
284
  		rcu_read_lock();
  		dev = dev_get_by_index_rcu(net, mc_lst->ifindex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
285
  		if (dev) {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
286
  			struct inet6_dev *idev = __in6_dev_get(dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
287

acd6e00b8   David L Stevens   [MCAST]: Fix filt...
288
  			(void) ip6_mc_leave_src(sk, mc_lst, idev);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
289
  			if (idev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
290
  				__ipv6_dev_mc_dec(idev, &mc_lst->addr);
acd6e00b8   David L Stevens   [MCAST]: Fix filt...
291
292
  		} else
  			(void) ip6_mc_leave_src(sk, mc_lst, NULL);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
293
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
294

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
295
  		atomic_sub(sizeof(*mc_lst), &sk->sk_omem_alloc);
e3cbf28fa   Lai Jiangshan   net,rcu: convert ...
296
  		kfree_rcu(mc_lst, rcu);
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
297
298
  
  		spin_lock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
299
  	}
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
300
  	spin_unlock(&ipv6_sk_mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
301
302
303
304
305
306
307
  }
  
  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
308
309
310
  	struct inet6_dev *idev;
  	struct ipv6_pinfo *inet6 = inet6_sk(sk);
  	struct ip6_sf_socklist *psl;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
311
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
312
  	int i, j, rv;
c9e3e8b69   David L Stevens   [IPV6]: multicast...
313
  	int leavegroup = 0;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
314
  	int pmclocked = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
315
  	int err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
316
317
318
319
320
  	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 ...
321
322
323
324
  	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
325
  		return -ENODEV;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
326
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
327
328
  
  	err = -EADDRNOTAVAIL;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
329
  	for_each_pmc_rcu(inet6, pmc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
330
331
332
333
334
  		if (pgsr->gsr_interface && pmc->ifindex != pgsr->gsr_interface)
  			continue;
  		if (ipv6_addr_equal(&pmc->addr, group))
  			break;
  	}
917f2f105   David L Stevens   [IPV4]: multicast...
335
336
  	if (!pmc) {		/* must have a prior join */
  		err = -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
337
  		goto done;
917f2f105   David L Stevens   [IPV4]: multicast...
338
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
339
340
  	/* if a source filter was set, must be the same mode as before */
  	if (pmc->sflist) {
917f2f105   David L Stevens   [IPV4]: multicast...
341
342
  		if (pmc->sfmode != omode) {
  			err = -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
343
  			goto done;
917f2f105   David L Stevens   [IPV4]: multicast...
344
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
345
346
347
348
349
350
  	} 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 ...
351
  	write_lock(&pmc->sflock);
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
352
  	pmclocked = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
353
354
355
  	psl = pmc->sflist;
  	if (!add) {
  		if (!psl)
917f2f105   David L Stevens   [IPV4]: multicast...
356
  			goto done;	/* err = -EADDRNOTAVAIL */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
357
358
359
360
361
362
363
364
  		rv = !0;
  		for (i=0; i<psl->sl_count; i++) {
  			rv = memcmp(&psl->sl_addr[i], source,
  				sizeof(struct in6_addr));
  			if (rv == 0)
  				break;
  		}
  		if (rv)		/* source not found */
917f2f105   David L Stevens   [IPV4]: multicast...
365
  			goto done;	/* err = -EADDRNOTAVAIL */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
366

c9e3e8b69   David L Stevens   [IPV6]: multicast...
367
368
369
370
371
  		/* 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
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
  		/* update the interface filter */
  		ip6_mc_del_src(idev, group, omode, 1, source, 1);
  
  		for (j=i+1; j<psl->sl_count; j++)
  			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...
393
  		newpsl = sock_kmalloc(sk, IP6_SFLSIZE(count), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
  		if (!newpsl) {
  			err = -ENOBUFS;
  			goto done;
  		}
  		newpsl->sl_max = count;
  		newpsl->sl_count = count - IP6_SFBLOCK;
  		if (psl) {
  			for (i=0; i<psl->sl_count; i++)
  				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 */
  	for (i=0; i<psl->sl_count; i++) {
  		rv = memcmp(&psl->sl_addr[i], source, sizeof(struct in6_addr));
  		if (rv == 0)
  			break;
  	}
  	if (rv == 0)		/* address already there is an error */
  		goto done;
  	for (j=psl->sl_count-1; j>=i; j--)
  		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...
423
  	if (pmclocked)
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
424
  		write_unlock(&pmc->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
425
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
426
  	rcu_read_unlock();
c9e3e8b69   David L Stevens   [IPV6]: multicast...
427
428
  	if (leavegroup)
  		return ipv6_sock_mc_drop(sk, pgsr->gsr_interface, group);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
429
430
431
432
433
  	return err;
  }
  
  int ip6_mc_msfilter(struct sock *sk, struct group_filter *gsf)
  {
b71d1d426   Eric Dumazet   inet: constify ip...
434
  	const struct in6_addr *group;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
435
  	struct ipv6_mc_socklist *pmc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
436
437
438
  	struct inet6_dev *idev;
  	struct ipv6_pinfo *inet6 = inet6_sk(sk);
  	struct ip6_sf_socklist *newpsl, *psl;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
439
  	struct net *net = sock_net(sk);
9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
440
  	int leavegroup = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
441
442
443
444
445
446
447
448
449
  	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 ...
450
451
  	rcu_read_lock();
  	idev = ip6_mc_find_dev_rcu(net, group, gsf->gf_interface);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
452

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
453
454
  	if (!idev) {
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
455
  		return -ENODEV;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
456
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
457

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

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

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
499
  	write_lock(&pmc->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
500
501
502
503
504
505
506
507
508
  	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 ...
509
  	write_unlock(&pmc->sflock);
917f2f105   David L Stevens   [IPV4]: multicast...
510
  	err = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
511
512
  done:
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
513
  	rcu_read_unlock();
9951f036f   David L Stevens   [IPV4]: (INCLUDE,...
514
515
  	if (leavegroup)
  		err = ipv6_sock_mc_drop(sk, gsf->gf_interface, group);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
516
517
518
519
520
521
522
  	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...
523
  	const struct in6_addr *group;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
524
525
  	struct ipv6_mc_socklist *pmc;
  	struct inet6_dev *idev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
526
527
  	struct ipv6_pinfo *inet6 = inet6_sk(sk);
  	struct ip6_sf_socklist *psl;
3b1e0a655   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
528
  	struct net *net = sock_net(sk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
529
530
531
532
533
  
  	group = &((struct sockaddr_in6 *)&gsf->gf_group)->sin6_addr;
  
  	if (!ipv6_addr_is_multicast(group))
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
534
535
  	rcu_read_lock();
  	idev = ip6_mc_find_dev_rcu(net, group, gsf->gf_interface);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
536

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
537
538
  	if (!idev) {
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
539
  		return -ENODEV;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
540
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
541
542
  
  	err = -EADDRNOTAVAIL;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
543
544
545
546
547
  	/*
  	 * changes to the ipv6_mc_list require the socket lock and
  	 * a read lock on ip6_sk_mc_lock. We have the socket lock,
  	 * so reading the list is safe.
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
548

456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
549
  	for_each_pmc_rcu(inet6, pmc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
550
551
552
553
554
555
556
557
558
559
560
  		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 ...
561
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
562
563
564
565
566
567
568
  
  	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;
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
569
570
571
572
  	/* changes to psl require the socket lock, a read lock on
  	 * on ipv6_sk_mc_lock and a write lock on pmc->sflock. We
  	 * have the socket lock, so reading here is safe.
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
573
574
575
576
577
578
579
580
  	for (i=0; i<copycount; i++) {
  		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...
581
  		if (copy_to_user(&optval->gf_slist[i], &ss, sizeof(ss)))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
582
583
584
585
586
  			return -EFAULT;
  	}
  	return 0;
  done:
  	read_unlock_bh(&idev->lock);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
587
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
588
589
  	return err;
  }
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
590
591
  int 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
592
593
594
595
596
  {
  	struct ipv6_pinfo *np = inet6_sk(sk);
  	struct ipv6_mc_socklist *mc;
  	struct ip6_sf_socklist *psl;
  	int rv = 1;
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
597
598
  	rcu_read_lock();
  	for_each_pmc_rcu(np, mc) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
599
600
601
602
  		if (ipv6_addr_equal(&mc->addr, mc_addr))
  			break;
  	}
  	if (!mc) {
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
603
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
604
605
  		return 1;
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
606
  	read_lock(&mc->sflock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
  	psl = mc->sflist;
  	if (!psl) {
  		rv = mc->sfmode == MCAST_EXCLUDE;
  	} else {
  		int i;
  
  		for (i=0; i<psl->sl_count; i++) {
  			if (ipv6_addr_equal(&psl->sl_addr[i], src_addr))
  				break;
  		}
  		if (mc->sfmode == MCAST_INCLUDE && i >= psl->sl_count)
  			rv = 0;
  		if (mc->sfmode == MCAST_EXCLUDE && i < psl->sl_count)
  			rv = 0;
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
622
  	read_unlock(&mc->sflock);
456b61bca   Eric Dumazet   ipv6: mcast: RCU ...
623
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
  
  	return rv;
  }
  
  static void ma_put(struct ifmcaddr6 *mc)
  {
  	if (atomic_dec_and_test(&mc->mca_refcnt)) {
  		in6_dev_put(mc->idev);
  		kfree(mc);
  	}
  }
  
  static void igmp6_group_added(struct ifmcaddr6 *mc)
  {
  	struct net_device *dev = mc->idev->dev;
  	char buf[MAX_ADDR_LEN];
  
  	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...
645
  			dev_mc_add(dev, buf);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
  	}
  	spin_unlock_bh(&mc->mca_lock);
  
  	if (!(dev->flags & IFF_UP) || (mc->mca_flags & MAF_NOREPORT))
  		return;
  
  	if (MLD_V1_SEEN(mc->idev)) {
  		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];
  
  	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...
671
  			dev_mc_del(dev, buf);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
  	}
  
  	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...
702
  	pmc = kzalloc(sizeof(*pmc), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
703
704
  	if (!pmc)
  		return;
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
705

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
  	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;
  		for (psf=pmc->mca_sources; psf; psf=psf->sf_next)
  			psf->sf_crcount = pmc->mca_crcount;
  	}
  	spin_unlock_bh(&im->mca_lock);
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
723
  	spin_lock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
724
725
  	pmc->next = idev->mc_tomb;
  	idev->mc_tomb = pmc;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
726
  	spin_unlock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
727
  }
b71d1d426   Eric Dumazet   inet: constify ip...
728
  static void mld_del_delrec(struct inet6_dev *idev, const struct in6_addr *pmca)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
729
730
731
  {
  	struct ifmcaddr6 *pmc, *pmc_prev;
  	struct ip6_sf_list *psf, *psf_next;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
732
  	spin_lock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
733
734
735
736
737
738
739
740
741
742
743
744
  	pmc_prev = NULL;
  	for (pmc=idev->mc_tomb; pmc; pmc=pmc->next) {
  		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_...
745
  	spin_unlock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
746
747
748
749
750
751
752
753
754
755
756
757
758
  	if (pmc) {
  		for (psf=pmc->mca_tomb; psf; psf=psf_next) {
  			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_...
759
  	spin_lock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
760
761
  	pmc = idev->mc_tomb;
  	idev->mc_tomb = NULL;
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
762
  	spin_unlock_bh(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
  
  	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);
  	for (pmc=idev->mc_list; pmc; pmc=pmc->next) {
  		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);
  		for (; psf; psf=psf_next) {
  			psf_next = psf->sf_next;
  			kfree(psf);
  		}
  	}
  	read_unlock_bh(&idev->lock);
  }
  
  
  /*
   *	device multicast group inc (add if not found)
   */
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
792
  int ipv6_dev_mc_inc(struct net_device *dev, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
793
794
795
  {
  	struct ifmcaddr6 *mc;
  	struct inet6_dev *idev;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
796
  	/* we need to take a reference on idev */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
  	idev = in6_dev_get(dev);
  
  	if (idev == NULL)
  		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;
  		}
  	}
  
  	/*
  	 *	not found: create a new one.
  	 */
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
823
  	mc = kzalloc(sizeof(struct ifmcaddr6), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
824
825
826
827
828
829
  
  	if (mc == NULL) {
  		write_unlock_bh(&idev->lock);
  		in6_dev_put(idev);
  		return -ENOMEM;
  	}
b24b8a247   Pavel Emelyanov   [NET]: Convert in...
830
  	setup_timer(&mc->mca_timer, igmp6_timer_handler, (unsigned long)mc);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
831

4e3fd7a06   Alexey Dobriyan   net: remove ipv6_...
832
  	mc->mca_addr = *addr;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
833
  	mc->idev = idev; /* (reference taken) */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
  	mc->mca_users = 1;
  	/* mca_stamp should be updated upon changes */
  	mc->mca_cstamp = mc->mca_tstamp = jiffies;
  	atomic_set(&mc->mca_refcnt, 2);
  	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;
  
  	mc->next = idev->mc_list;
  	idev->mc_list = mc;
  	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...
861
  int __ipv6_dev_mc_dec(struct inet6_dev *idev, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
  {
  	struct ifmcaddr6 *ma, **map;
  
  	write_lock_bh(&idev->lock);
  	for (map = &idev->mc_list; (ma=*map) != NULL; map = &ma->next) {
  		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...
885
  int ipv6_dev_mc_dec(struct net_device *dev, const struct in6_addr *addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
886
  {
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
887
  	struct inet6_dev *idev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
888
  	int err;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
889
  	rcu_read_lock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
890

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
891
892
893
894
895
  	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
896

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
897
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
  	return err;
  }
  
  /*
   * identify MLD packets for MLD filter exceptions
   */
  int ipv6_is_mld(struct sk_buff *skb, int nexthdr)
  {
  	struct icmp6hdr *pic;
  
  	if (nexthdr != IPPROTO_ICMPV6)
  		return 0;
  
  	if (!pskb_may_pull(skb, sizeof(struct icmp6hdr)))
  		return 0;
cc70ab261   Arnaldo Carvalho de Melo   [ICMP6]: Introduc...
913
  	pic = icmp6_hdr(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
  
  	switch (pic->icmp6_type) {
  	case ICMPV6_MGM_QUERY:
  	case ICMPV6_MGM_REPORT:
  	case ICMPV6_MGM_REDUCTION:
  	case ICMPV6_MLD2_REPORT:
  		return 1;
  	default:
  		break;
  	}
  	return 0;
  }
  
  /*
   *	check if the interface/address pair is valid
   */
9acd9f3ae   YOSHIFUJI Hideaki   [IPV6]: Make addr...
930
931
  int 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
932
933
934
935
  {
  	struct inet6_dev *idev;
  	struct ifmcaddr6 *mc;
  	int rv = 0;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
936
937
  	rcu_read_lock();
  	idev = __in6_dev_get(dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
  	if (idev) {
  		read_lock_bh(&idev->lock);
  		for (mc = idev->mc_list; mc; mc=mc->next) {
  			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);
  				for (psf=mc->mca_sources;psf;psf=psf->sf_next) {
  					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
  					rv = mc->mca_sfcount[MCAST_EXCLUDE] !=0;
  				spin_unlock_bh(&mc->mca_lock);
  			} else
  				rv = 1; /* don't filter unspecified source */
  		}
  		read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
964
  	}
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
965
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
  	return rv;
  }
  
  static void mld_gq_start_timer(struct inet6_dev *idev)
  {
  	int tv = net_random() % idev->mc_maxdelay;
  
  	idev->mc_gq_running = 1;
  	if (!mod_timer(&idev->mc_gq_timer, jiffies+tv+2))
  		in6_dev_hold(idev);
  }
  
  static void mld_ifc_start_timer(struct inet6_dev *idev, int delay)
  {
  	int tv = net_random() % delay;
  
  	if (!mod_timer(&idev->mc_ifc_timer, jiffies+tv+2))
  		in6_dev_hold(idev);
  }
  
  /*
   *	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;
  	}
  
  	if (delay >= resptime) {
  		if (resptime)
  			delay = net_random() % resptime;
  		else
  			delay = 1;
  	}
  	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
1016
  /* mark EXCLUDE-mode sources */
  static int mld_xmarksources(struct ifmcaddr6 *pmc, int nsrcs,
b71d1d426   Eric Dumazet   inet: constify ip...
1017
  	const struct in6_addr *srcs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1018
1019
1020
1021
1022
1023
1024
1025
  {
  	struct ip6_sf_list *psf;
  	int i, scount;
  
  	scount = 0;
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next) {
  		if (scount == nsrcs)
  			break;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1026
1027
  		for (i=0; i<nsrcs; i++) {
  			/* 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
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
  			    pmc->mca_sfcount[MCAST_EXCLUDE] !=
  			    psf->sf_count[MCAST_EXCLUDE])
  				continue;
  			if (ipv6_addr_equal(&srcs[i], &psf->sf_addr)) {
  				scount++;
  				break;
  			}
  		}
  	}
  	pmc->mca_flags &= ~MAF_GSQUERY;
  	if (scount == nsrcs)	/* all sources excluded */
  		return 0;
  	return 1;
  }
  
  static int mld_marksources(struct ifmcaddr6 *pmc, int nsrcs,
b71d1d426   Eric Dumazet   inet: constify ip...
1045
  	const struct in6_addr *srcs)
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
  {
  	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;
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next) {
  		if (scount == nsrcs)
  			break;
  		for (i=0; i<nsrcs; i++) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1060
1061
1062
1063
1064
  			if (ipv6_addr_equal(&srcs[i], &psf->sf_addr)) {
  				psf->sf_gsresp = 1;
  				scount++;
  				break;
  			}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1065
1066
1067
1068
1069
  		}
  	}
  	if (!scount) {
  		pmc->mca_flags &= ~MAF_GSQUERY;
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1070
  	}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1071
1072
  	pmc->mca_flags |= MAF_GSQUERY;
  	return 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1073
  }
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1074
  /* called with rcu_read_lock() */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1075
1076
  int igmp6_event_query(struct sk_buff *skb)
  {
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1077
  	struct mld2_query *mlh2 = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1078
  	struct ifmcaddr6 *ma;
b71d1d426   Eric Dumazet   inet: constify ip...
1079
  	const struct in6_addr *group;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1080
1081
  	unsigned long max_delay;
  	struct inet6_dev *idev;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1082
  	struct mld_msg *mld;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1083
1084
1085
1086
1087
1088
1089
1090
  	int group_type;
  	int mark = 0;
  	int len;
  
  	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...
1091
  	len = ntohs(ipv6_hdr(skb)->payload_len) + sizeof(struct ipv6hdr);
cfe1fc775   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1092
  	len -= skb_network_header_len(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1093
1094
  
  	/* Drop queries with not link local source */
0660e03f6   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1095
  	if (!(ipv6_addr_type(&ipv6_hdr(skb)->saddr) & IPV6_ADDR_LINKLOCAL))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1096
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1097
  	idev = __in6_dev_get(skb->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1098
1099
1100
  
  	if (idev == NULL)
  		return 0;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1101
1102
  	mld = (struct mld_msg *)icmp6_hdr(skb);
  	group = &mld->mld_mca;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1103
1104
1105
  	group_type = ipv6_addr_type(group);
  
  	if (group_type != IPV6_ADDR_ANY &&
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1106
  	    !(group_type&IPV6_ADDR_MULTICAST))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1107
  		return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1108
1109
1110
1111
1112
1113
  
  	if (len == 24) {
  		int switchback;
  		/* MLDv1 router present */
  
  		/* Translate milliseconds to jiffies */
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1114
  		max_delay = (ntohs(mld->mld_maxdelay)*HZ)/1000;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
  
  		switchback = (idev->mc_qrv + 1) * max_delay;
  		idev->mc_v1_seen = jiffies + switchback;
  
  		/* cancel the interface change timer */
  		idev->mc_ifc_count = 0;
  		if (del_timer(&idev->mc_ifc_timer))
  			__in6_dev_put(idev);
  		/* clear deleted report items */
  		mld_clear_delrec(idev);
  	} else if (len >= 28) {
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1126
  		int srcs_offset = sizeof(struct mld2_query) -
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1127
  				  sizeof(struct icmp6hdr);
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1128
  		if (!pskb_may_pull(skb, srcs_offset))
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1129
  			return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1130

9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1131
  		mlh2 = (struct mld2_query *)skb_transport_header(skb);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1132
  		max_delay = (MLDV2_MRC(ntohs(mlh2->mld2q_mrc))*HZ)/1000;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1133
1134
1135
  		if (!max_delay)
  			max_delay = 1;
  		idev->mc_maxdelay = max_delay;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1136
1137
  		if (mlh2->mld2q_qrv)
  			idev->mc_qrv = mlh2->mld2q_qrv;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1138
  		if (group_type == IPV6_ADDR_ANY) { /* general query */
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1139
  			if (mlh2->mld2q_nsrcs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1140
  				return -EINVAL; /* no sources allowed */
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1141

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1142
  			mld_gq_start_timer(idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1143
1144
1145
  			return 0;
  		}
  		/* mark sources to include, if group & source-specific */
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1146
  		if (mlh2->mld2q_nsrcs != 0) {
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1147
  			if (!pskb_may_pull(skb, srcs_offset +
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1148
  			    ntohs(mlh2->mld2q_nsrcs) * sizeof(struct in6_addr)))
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1149
  				return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1150

9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1151
  			mlh2 = (struct mld2_query *)skb_transport_header(skb);
97300b5fd   Yan Zheng   [MCAST] IPv6: Che...
1152
1153
  			mark = 1;
  		}
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1154
  	} else
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1155
  		return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
  
  	read_lock_bh(&idev->lock);
  	if (group_type == IPV6_ADDR_ANY) {
  		for (ma = idev->mc_list; ma; ma=ma->next) {
  			spin_lock_bh(&ma->mca_lock);
  			igmp6_group_queried(ma, max_delay);
  			spin_unlock_bh(&ma->mca_lock);
  		}
  	} else {
  		for (ma = idev->mc_list; ma; ma=ma->next) {
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1166
  			if (!ipv6_addr_equal(group, &ma->mca_addr))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
  				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...
1180
  			if (!(ma->mca_flags & MAF_GSQUERY) ||
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1181
  			    mld_marksources(ma, ntohs(mlh2->mld2q_nsrcs), mlh2->mld2q_srcs))
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1182
  				igmp6_group_queried(ma, max_delay);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1183
  			spin_unlock_bh(&ma->mca_lock);
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1184
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1185
1186
1187
  		}
  	}
  	read_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1188
1189
1190
  
  	return 0;
  }
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1191
  /* called with rcu_read_lock() */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1192
1193
1194
  int igmp6_event_report(struct sk_buff *skb)
  {
  	struct ifmcaddr6 *ma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1195
  	struct inet6_dev *idev;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1196
  	struct mld_msg *mld;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1197
1198
1199
1200
1201
  	int addr_type;
  
  	/* Our own report looped back. Ignore it. */
  	if (skb->pkt_type == PACKET_LOOPBACK)
  		return 0;
24c692750   David Stevens   [IGMP]: workaroun...
1202
1203
1204
1205
  	/* 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...
1206
  	if (!pskb_may_pull(skb, sizeof(*mld) - sizeof(struct icmp6hdr)))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1207
  		return -EINVAL;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1208
  	mld = (struct mld_msg *)icmp6_hdr(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1209
1210
  
  	/* Drop reports with not link local source */
0660e03f6   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1211
  	addr_type = ipv6_addr_type(&ipv6_hdr(skb)->saddr);
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1212
  	if (addr_type != IPV6_ADDR_ANY &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1213
1214
  	    !(addr_type&IPV6_ADDR_LINKLOCAL))
  		return -EINVAL;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1215
  	idev = __in6_dev_get(skb->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1216
1217
1218
1219
1220
1221
1222
1223
1224
  	if (idev == NULL)
  		return -ENODEV;
  
  	/*
  	 *	Cancel the timer for this group
  	 */
  
  	read_lock_bh(&idev->lock);
  	for (ma = idev->mc_list; ma; ma=ma->next) {
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1225
  		if (ipv6_addr_equal(&ma->mca_addr, &mld->mld_mca)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1226
1227
1228
1229
1230
1231
1232
1233
1234
  			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
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
  	return 0;
  }
  
  static int is_in(struct ifmcaddr6 *pmc, struct ip6_sf_list *psf, int type,
  	int gdeleted, int sdeleted)
  {
  	switch (type) {
  	case MLD2_MODE_IS_INCLUDE:
  	case MLD2_MODE_IS_EXCLUDE:
  		if (gdeleted || sdeleted)
  			return 0;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1246
1247
1248
1249
1250
1251
1252
  		if (!((pmc->mca_flags & MAF_GSQUERY) && !psf->sf_gsresp)) {
  			if (pmc->mca_sfmode == MCAST_INCLUDE)
  				return 1;
  			/* don't include if this source is excluded
  			 * in all filters
  			 */
  			if (psf->sf_count[MCAST_INCLUDE])
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1253
  				return type == MLD2_MODE_IS_INCLUDE;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1254
1255
1256
1257
  			return pmc->mca_sfcount[MCAST_EXCLUDE] ==
  				psf->sf_count[MCAST_EXCLUDE];
  		}
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
  	case MLD2_CHANGE_TO_INCLUDE:
  		if (gdeleted || sdeleted)
  			return 0;
  		return psf->sf_count[MCAST_INCLUDE] != 0;
  	case MLD2_CHANGE_TO_EXCLUDE:
  		if (gdeleted || sdeleted)
  			return 0;
  		if (pmc->mca_sfcount[MCAST_EXCLUDE] == 0 ||
  		    psf->sf_count[MCAST_INCLUDE])
  			return 0;
  		return pmc->mca_sfcount[MCAST_EXCLUDE] ==
  			psf->sf_count[MCAST_EXCLUDE];
  	case MLD2_ALLOW_NEW_SOURCES:
  		if (gdeleted || !psf->sf_crcount)
  			return 0;
  		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;
  	}
  	return 0;
  }
  
  static int
  mld_scount(struct ifmcaddr6 *pmc, int type, int gdeleted, int sdeleted)
  {
  	struct ip6_sf_list *psf;
  	int scount = 0;
  
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next) {
  		if (!is_in(pmc, psf, type, gdeleted, sdeleted))
  			continue;
  		scount++;
  	}
  	return scount;
  }
  
  static struct sk_buff *mld_newpack(struct net_device *dev, int size)
  {
c346dca10   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
1298
  	struct net *net = dev_net(dev);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
1299
  	struct sock *sk = net->ipv6.igmp_sk;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1300
1301
1302
  	struct sk_buff *skb;
  	struct mld2_report *pmr;
  	struct in6_addr addr_buf;
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1303
  	const struct in6_addr *saddr;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1304
1305
  	int hlen = LL_RESERVED_SPACE(dev);
  	int tlen = dev->needed_tailroom;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1306
1307
1308
1309
1310
1311
  	int err;
  	u8 ra[8] = { IPPROTO_ICMPV6, 0,
  		     IPV6_TLV_ROUTERALERT, 2, 0, 0,
  		     IPV6_TLV_PADN, 0 };
  
  	/* we assume size > sizeof(ra) here */
a7ae19922   Herbert Xu   ipv6: Remove all ...
1312
  	size += hlen + tlen;
72e09ad10   Eric Dumazet   ipv6: avoid high ...
1313
1314
1315
  	/* 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
1316

cfcabdcc2   Stephen Hemminger   [NET]: sparse war...
1317
  	if (!skb)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1318
  		return NULL;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1319
  	skb_reserve(skb, hlen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1320

95c385b4d   Neil Horman   [IPV6] ADDRCONF: ...
1321
  	if (ipv6_get_lladdr(dev, &addr_buf, IFA_F_TENTATIVE)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1322
  		/* <draft-ietf-magma-mld-source-05.txt>:
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1323
  		 * use unspecified address as the source address
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1324
1325
  		 * when a valid link-local address is not available.
  		 */
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1326
1327
1328
  		saddr = &in6addr_any;
  	} else
  		saddr = &addr_buf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1329

d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1330
  	ip6_nd_hdr(sk, skb, dev, saddr, &mld2_all_mcr, NEXTHDR_HOP, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1331
1332
  
  	memcpy(skb_put(skb, sizeof(ra)), ra, sizeof(ra));
27a884dc3   Arnaldo Carvalho de Melo   [SK_BUFF]: Conver...
1333
  	skb_set_transport_header(skb, skb_tail_pointer(skb) - skb->data);
d10ba34b0   Arnaldo Carvalho de Melo   [SK_BUFF]: More s...
1334
1335
  	skb_put(skb, sizeof(*pmr));
  	pmr = (struct mld2_report *)skb_transport_header(skb);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1336
1337
1338
1339
1340
  	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
1341
1342
1343
1344
1345
  	return skb;
  }
  
  static void mld_sendpack(struct sk_buff *skb)
  {
0660e03f6   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1346
  	struct ipv6hdr *pip6 = ipv6_hdr(skb);
9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1347
1348
  	struct mld2_report *pmr =
  			      (struct mld2_report *)skb_transport_header(skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1349
  	int payload_len, mldlen;
96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1350
  	struct inet6_dev *idev;
c346dca10   YOSHIFUJI Hideaki   [NET] NETNS: Omit...
1351
  	struct net *net = dev_net(skb->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1352
  	int err;
4c9483b2f   David S. Miller   ipv6: Convert to ...
1353
  	struct flowi6 fl6;
adf30907d   Eric Dumazet   net: skb->dst acc...
1354
  	struct dst_entry *dst;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1355

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1356
1357
  	rcu_read_lock();
  	idev = __in6_dev_get(skb->dev);
edf391ff1   Neil Horman   snmp: add missing...
1358
  	IP6_UPD_PO_STATS(net, idev, IPSTATS_MIB_OUT, skb->len);
27a884dc3   Arnaldo Carvalho de Melo   [SK_BUFF]: Conver...
1359
1360
  	payload_len = (skb->tail - skb->network_header) - sizeof(*pip6);
  	mldlen = skb->tail - skb->transport_header;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1361
  	pip6->payload_len = htons(payload_len);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1362
1363
1364
1365
  	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...
1366

4c9483b2f   David S. Miller   ipv6: Convert to ...
1367
  	icmpv6_flow_init(net->ipv6.igmp_sk, &fl6, ICMPV6_MLD2_REPORT,
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1368
1369
  			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr,
  			 skb->dev->ifindex);
87a115783   David S. Miller   ipv6: Move xfrm_l...
1370
  	dst = icmp6_dst_alloc(skb->dev, NULL, &fl6);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1371

452edd598   David S. Miller   xfrm: Return dst ...
1372
1373
1374
1375
1376
  	err = 0;
  	if (IS_ERR(dst)) {
  		err = PTR_ERR(dst);
  		dst = NULL;
  	}
adf30907d   Eric Dumazet   net: skb->dst acc...
1377
  	skb_dst_set(skb, dst);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1378
1379
  	if (err)
  		goto err_out;
edf391ff1   Neil Horman   snmp: add missing...
1380
  	payload_len = skb->len;
b2e0b385d   Jan Engelhardt   netfilter: ipv6: ...
1381
  	err = NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_OUT, skb, NULL, skb->dev,
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1382
1383
  		      dst_output);
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1384
  	if (!err) {
5a57d4c7f   Denis V. Lunev   ipv6: added net a...
1385
  		ICMP6MSGOUT_INC_STATS_BH(net, idev, ICMPV6_MLD2_REPORT);
e41b5368e   Denis V. Lunev   ipv6: added net a...
1386
  		ICMP6_INC_STATS_BH(net, idev, ICMP6_MIB_OUTMSGS);
edf391ff1   Neil Horman   snmp: add missing...
1387
  		IP6_UPD_PO_STATS_BH(net, idev, IPSTATS_MIB_OUTMCAST, payload_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1388
  	} else
483a47d2f   Denis V. Lunev   ipv6: added net a...
1389
  		IP6_INC_STATS_BH(net, idev, IPSTATS_MIB_OUTDISCARDS);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1390

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1391
  	rcu_read_unlock();
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1392
1393
1394
1395
1396
  	return;
  
  err_out:
  	kfree_skb(skb);
  	goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1397
1398
1399
1400
  }
  
  static int grec_size(struct ifmcaddr6 *pmc, int type, int gdel, int sdel)
  {
fab10fe37   Yan Zheng   [MCAST] ipv6: Fix...
1401
  	return sizeof(struct mld2_grec) + 16 * mld_scount(pmc,type,gdel,sdel);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
  }
  
  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)
  		skb = mld_newpack(dev, dev->mtu);
  	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...
1420
  	pmr = (struct mld2_report *)skb_transport_header(skb);
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1421
  	pmr->mld2r_ngrec = htons(ntohs(pmr->mld2r_ngrec)+1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
  	*ppgr = pgr;
  	return skb;
  }
  
  #define AVAILABLE(skb) ((skb) ? ((skb)->dev ? (skb)->dev->mtu - (skb)->len : \
  	skb_tailroom(skb)) : 0)
  
  static struct sk_buff *add_grec(struct sk_buff *skb, struct ifmcaddr6 *pmc,
  	int type, int gdeleted, int sdeleted)
  {
  	struct net_device *dev = pmc->idev->dev;
  	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...
1436
  	int scount, stotal, first, isquery, truncate;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1437
1438
1439
1440
1441
1442
1443
1444
  
  	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...
1445
  	stotal = scount = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1446
  	psf_list = sdeleted ? &pmc->mca_tomb : &pmc->mca_sources;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1447
1448
  	if (!*psf_list)
  		goto empty_source;
9c70220b7   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
1449
  	pmr = skb ? (struct mld2_report *)skb_transport_header(skb) : NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1450
1451
1452
  
  	/* EX and TO_EX get a fresh packet, if needed */
  	if (truncate) {
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1453
  		if (pmr && pmr->mld2r_ngrec &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1454
1455
1456
1457
1458
1459
1460
  		    AVAILABLE(skb) < grec_size(pmc, type, gdeleted, sdeleted)) {
  			if (skb)
  				mld_sendpack(skb);
  			skb = mld_newpack(dev, dev->mtu);
  		}
  	}
  	first = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
  	psf_prev = NULL;
  	for (psf=*psf_list; psf; psf=psf_next) {
  		struct in6_addr *psrc;
  
  		psf_next = psf->sf_next;
  
  		if (!is_in(pmc, psf, type, gdeleted, sdeleted)) {
  			psf_prev = psf;
  			continue;
  		}
  
  		/* 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);
  			skb = mld_newpack(dev, dev->mtu);
  			first = 1;
  			scount = 0;
  		}
  		if (first) {
  			skb = add_grhead(skb, pmc, type, &pgr);
  			first = 0;
  		}
cc63f70b8   Alexey Dobriyan   [IPV4/IPV6] multi...
1492
1493
  		if (!skb)
  			return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1494
1495
  		psrc = (struct in6_addr *)skb_put(skb, sizeof(*psrc));
  		*psrc = psf->sf_addr;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1496
  		scount++; stotal++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
  		if ((type == MLD2_ALLOW_NEW_SOURCES ||
  		     type == MLD2_BLOCK_OLD_SOURCES) && psf->sf_crcount) {
  			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...
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
  
  empty_source:
  	if (!stotal) {
  		if (type == MLD2_ALLOW_NEW_SOURCES ||
  		    type == MLD2_BLOCK_OLD_SOURCES)
  			return skb;
  		if (pmc->mca_crcount || isquery) {
  			/* 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
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
1579
1580
1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
  	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;
  
  	if (!pmc) {
  		read_lock_bh(&idev->lock);
  		for (pmc=idev->mc_list; pmc; pmc=pmc->next) {
  			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;
  			skb = add_grec(skb, pmc, type, 0, 0);
  			spin_unlock_bh(&pmc->mca_lock);
  		}
  		read_unlock_bh(&idev->lock);
  	} else {
  		spin_lock_bh(&pmc->mca_lock);
  		if (pmc->mca_sfcount[MCAST_EXCLUDE])
  			type = MLD2_MODE_IS_EXCLUDE;
  		else
  			type = MLD2_MODE_IS_INCLUDE;
  		skb = add_grec(skb, pmc, type, 0, 0);
  		spin_unlock_bh(&pmc->mca_lock);
  	}
  	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;
  	for (psf=*ppsf; psf; psf = psf_next) {
  		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_...
1594
  	spin_lock(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
  
  	/* deleted MCA's */
  	pmc_prev = NULL;
  	for (pmc=idev->mc_tomb; pmc; pmc=pmc_next) {
  		pmc_next = pmc->next;
  		if (pmc->mca_sfmode == MCAST_INCLUDE) {
  			type = MLD2_BLOCK_OLD_SOURCES;
  			dtype = MLD2_BLOCK_OLD_SOURCES;
  			skb = add_grec(skb, pmc, type, 1, 0);
  			skb = add_grec(skb, pmc, dtype, 1, 1);
  		}
  		if (pmc->mca_crcount) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1607
1608
1609
1610
  			if (pmc->mca_sfmode == MCAST_EXCLUDE) {
  				type = MLD2_CHANGE_TO_INCLUDE;
  				skb = add_grec(skb, pmc, type, 1, 0);
  			}
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1611
  			pmc->mca_crcount--;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1612
1613
1614
1615
1616
1617
1618
1619
1620
1621
1622
1623
1624
1625
1626
1627
  			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_...
1628
  	spin_unlock(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
  
  	/* change recs */
  	for (pmc=idev->mc_list; pmc; pmc=pmc->next) {
  		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;
  		}
  		skb = add_grec(skb, pmc, type, 0, 0);
  		skb = add_grec(skb, pmc, dtype, 0, 1);	/* deleted sources */
  
  		/* filter mode changes */
  		if (pmc->mca_crcount) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1645
1646
1647
1648
1649
  			if (pmc->mca_sfmode == MCAST_EXCLUDE)
  				type = MLD2_CHANGE_TO_EXCLUDE;
  			else
  				type = MLD2_CHANGE_TO_INCLUDE;
  			skb = add_grec(skb, pmc, type, 0, 0);
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
1650
  			pmc->mca_crcount--;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
  		}
  		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...
1662
  	struct net *net = dev_net(dev);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
1663
  	struct sock *sk = net->ipv6.igmp_sk;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1664
  	struct inet6_dev *idev;
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1665
  	struct sk_buff *skb;
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1666
  	struct mld_msg *hdr;
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1667
  	const struct in6_addr *snd_addr, *saddr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1668
  	struct in6_addr addr_buf;
a7ae19922   Herbert Xu   ipv6: Remove all ...
1669
1670
  	int hlen = LL_RESERVED_SPACE(dev);
  	int tlen = dev->needed_tailroom;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1671
1672
1673
1674
  	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 ...
1675
  	struct flowi6 fl6;
adf30907d   Eric Dumazet   net: skb->dst acc...
1676
  	struct dst_entry *dst;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1677

f3ee4010e   YOSHIFUJI Hideaki   [IPV6]: Define co...
1678
1679
1680
1681
  	if (type == ICMPV6_MGM_REDUCTION)
  		snd_addr = &in6addr_linklocal_allrouters;
  	else
  		snd_addr = addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1682
1683
1684
1685
  
  	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...
1686
1687
1688
1689
  	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 ...
1690
  	skb = sock_alloc_send_skb(sk, hlen + tlen + full_len, 1, &err);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1691
1692
  
  	if (skb == NULL) {
a11d206d0   YOSHIFUJI Hideaki   [IPV6]: Per-inter...
1693
  		rcu_read_lock();
3bd653c84   Denis V. Lunev   netns: add net pa...
1694
  		IP6_INC_STATS(net, __in6_dev_get(dev),
a11d206d0   YOSHIFUJI Hideaki   [IPV6]: Per-inter...
1695
1696
  			      IPSTATS_MIB_OUTDISCARDS);
  		rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1697
1698
  		return;
  	}
a7ae19922   Herbert Xu   ipv6: Remove all ...
1699
  	skb_reserve(skb, hlen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1700

95c385b4d   Neil Horman   [IPV6] ADDRCONF: ...
1701
  	if (ipv6_get_lladdr(dev, &addr_buf, IFA_F_TENTATIVE)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1702
  		/* <draft-ietf-magma-mld-source-05.txt>:
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
1703
  		 * use unspecified address as the source address
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1704
1705
  		 * when a valid link-local address is not available.
  		 */
d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1706
1707
1708
  		saddr = &in6addr_any;
  	} else
  		saddr = &addr_buf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1709

d7aabf22e   YOSHIFUJI Hideaki   [IPV6]: Use in6ad...
1710
  	ip6_nd_hdr(sk, skb, dev, saddr, snd_addr, NEXTHDR_HOP, payload_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1711
1712
  
  	memcpy(skb_put(skb, sizeof(ra)), ra, sizeof(ra));
6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1713
1714
1715
  	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_...
1716
  	hdr->mld_mca = *addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1717

6e7cb8370   YOSHIFUJI Hideaki   ipv6 mcast: Intro...
1718
1719
1720
  	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
1721

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

4c9483b2f   David S. Miller   ipv6: Convert to ...
1725
  	icmpv6_flow_init(sk, &fl6, type,
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1726
1727
  			 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr,
  			 skb->dev->ifindex);
87a115783   David S. Miller   ipv6: Move xfrm_l...
1728
  	dst = icmp6_dst_alloc(skb->dev, NULL, &fl6);
452edd598   David S. Miller   xfrm: Return dst ...
1729
1730
  	if (IS_ERR(dst)) {
  		err = PTR_ERR(dst);
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1731
  		goto err_out;
452edd598   David S. Miller   xfrm: Return dst ...
1732
  	}
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1733

adf30907d   Eric Dumazet   net: skb->dst acc...
1734
  	skb_dst_set(skb, dst);
b2e0b385d   Jan Engelhardt   netfilter: ipv6: ...
1735
  	err = NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_OUT, skb, NULL, skb->dev,
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1736
1737
  		      dst_output);
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1738
  	if (!err) {
5c5d244bd   Denis V. Lunev   ipv6: added net a...
1739
  		ICMP6MSGOUT_INC_STATS(net, idev, type);
a862f6a6d   Denis V. Lunev   ipv6: added net a...
1740
  		ICMP6_INC_STATS(net, idev, ICMP6_MIB_OUTMSGS);
edf391ff1   Neil Horman   snmp: add missing...
1741
  		IP6_UPD_PO_STATS(net, idev, IPSTATS_MIB_OUTMCAST, full_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1742
  	} else
3bd653c84   Denis V. Lunev   netns: add net pa...
1743
  		IP6_INC_STATS(net, idev, IPSTATS_MIB_OUTDISCARDS);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1744

96b52e61b   Eric Dumazet   ipv6: mcast: RCU ...
1745
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1746
  	return;
419271780   YOSHIFUJI Hideaki   [IPV6] MCAST: Use...
1747
1748
1749
1750
  
  err_out:
  	kfree_skb(skb);
  	goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1751
1752
1753
  }
  
  static int ip6_mc_del1_src(struct ifmcaddr6 *pmc, int sfmode,
b71d1d426   Eric Dumazet   inet: constify ip...
1754
  	const struct in6_addr *psfsrc)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
1782
1783
1784
1785
1786
1787
1788
  {
  	struct ip6_sf_list *psf, *psf_prev;
  	int rv = 0;
  
  	psf_prev = NULL;
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next) {
  		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) &&
  		    !MLD_V1_SEEN(idev)) {
  			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...
1789
1790
  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
1791
1792
1793
1794
1795
1796
1797
1798
1799
1800
1801
1802
1803
1804
1805
1806
1807
1808
1809
1810
1811
1812
1813
1814
1815
1816
1817
1818
1819
1820
1821
1822
1823
1824
1825
1826
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
1837
1838
1839
1840
1841
1842
1843
1844
1845
1846
1847
1848
1849
  			  int delta)
  {
  	struct ifmcaddr6 *pmc;
  	int	changerec = 0;
  	int	i, err;
  
  	if (!idev)
  		return -ENODEV;
  	read_lock_bh(&idev->lock);
  	for (pmc=idev->mc_list; pmc; pmc=pmc->next) {
  		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;
  	for (i=0; i<sfcount; i++) {
  		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;
  		for (psf=pmc->mca_sources; psf; psf = psf->sf_next)
  			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...
1850
  	const struct in6_addr *psfsrc)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1851
1852
1853
1854
1855
1856
1857
1858
1859
1860
  {
  	struct ip6_sf_list *psf, *psf_prev;
  
  	psf_prev = NULL;
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next) {
  		if (ipv6_addr_equal(&psf->sf_addr, psfsrc))
  			break;
  		psf_prev = psf;
  	}
  	if (!psf) {
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
1861
  		psf = kzalloc(sizeof(*psf), GFP_ATOMIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1862
1863
  		if (!psf)
  			return -ENOBUFS;
0c600eda4   Ingo Oeser   [IPV6]: Nearly co...
1864

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1865
1866
1867
1868
1869
1870
1871
1872
1873
1874
1875
1876
1877
1878
1879
1880
1881
1882
1883
1884
1885
1886
1887
1888
1889
1890
  		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];
  
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next)
  		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...
1891
  	struct ip6_sf_list *psf, *dpsf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1892
1893
1894
1895
1896
1897
1898
1899
1900
1901
1902
  	int mca_xcount = pmc->mca_sfcount[MCAST_EXCLUDE];
  	int qrv = pmc->idev->mc_qrv;
  	int new_in, rv;
  
  	rv = 0;
  	for (psf=pmc->mca_sources; psf; psf=psf->sf_next) {
  		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...
1903
1904
  		if (new_in) {
  			if (!psf->sf_oldin) {
e80e28b6b   Al Viro   [PATCH] net/ipv6/...
1905
  				struct ip6_sf_list *prev = NULL;
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1906
1907
1908
1909
1910
1911
1912
1913
1914
1915
1916
1917
1918
1919
1920
1921
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931
1932
1933
1934
  
  				for (dpsf=pmc->mca_tomb; dpsf;
  				     dpsf=dpsf->sf_next) {
  					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
  			 */
  			for (dpsf=pmc->mca_tomb; dpsf; dpsf=dpsf->sf_next)
  				if (ipv6_addr_equal(&dpsf->sf_addr,
  				    &psf->sf_addr))
  					break;
  			if (!dpsf) {
5d55354f1   Joe Perches   net/ipv6/mcast.c:...
1935
  				dpsf = kmalloc(sizeof(*dpsf), GFP_ATOMIC);
7add2a439   David L Stevens   [IPV6] MLDv2: fix...
1936
1937
1938
1939
1940
1941
1942
1943
  				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
1944
1945
1946
1947
1948
1949
1950
1951
1952
  			rv++;
  		}
  	}
  	return rv;
  }
  
  /*
   * Add multicast source filter list to the interface list
   */
b71d1d426   Eric Dumazet   inet: constify ip...
1953
1954
  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
1955
1956
1957
1958
1959
1960
1961
1962
1963
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
  			  int delta)
  {
  	struct ifmcaddr6 *pmc;
  	int	isexclude;
  	int	i, err;
  
  	if (!idev)
  		return -ENODEV;
  	read_lock_bh(&idev->lock);
  	for (pmc=idev->mc_list; pmc; pmc=pmc->next) {
  		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;
  	for (i=0; i<sfcount; i++) {
99d2f47aa   Jun Zhao   ipv6 : mcast : De...
1981
  		err = ip6_mc_add1_src(pmc, sfmode, &psfsrc[i]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
  		if (err)
  			break;
  	}
  	if (err) {
  		int j;
  
  		if (!delta)
  			pmc->mca_sfcount[sfmode]--;
  		for (j=0; j<i; j++)
  			(void) ip6_mc_del1_src(pmc, sfmode, &psfsrc[i]);
  	} else if (isexclude != (pmc->mca_sfcount[MCAST_EXCLUDE] != 0)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
2027
2028
  		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;
  		for (psf=pmc->mca_sources; psf; psf = psf->sf_next)
  			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;
  
  	for (psf=pmc->mca_tomb; psf; psf=nextpsf) {
  		nextpsf = psf->sf_next;
  		kfree(psf);
  	}
  	pmc->mca_tomb = NULL;
  	for (psf=pmc->mca_sources; psf; psf=nextpsf) {
  		nextpsf = psf->sf_next;
  		kfree(psf);
  	}
  	pmc->mca_sources = NULL;
  	pmc->mca_sfmode = MCAST_EXCLUDE;
de9daad90   Denis Lukianov   [MCAST]: Fix MCAS...
2029
  	pmc->mca_sfcount[MCAST_INCLUDE] = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2030
2031
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
2042
2043
2044
2045
2046
2047
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
  	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);
  
  	delay = net_random() % IGMP6_UNSOLICITED_IVAL;
  
  	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;
5ab4a6c81   David L Stevens   [IPV6] mcast: Fix...
2061
2062
2063
  	/* callers have the socket lock and a write lock on ipv6_sk_mc_lock,
  	 * so no other readers or writers of iml or its sflist
  	 */
cfcabdcc2   Stephen Hemminger   [NET]: sparse war...
2064
  	if (!iml->sflist) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2065
2066
2067
2068
2069
2070
2071
2072
2073
2074
2075
2076
2077
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
2091
2092
2093
2094
2095
2096
2097
2098
2099
2100
2101
2102
2103
2104
2105
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
2119
2120
2121
2122
2123
2124
2125
2126
2127
2128
2129
2130
2131
2132
  		/* 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)
  {
  	if (MLD_V1_SEEN(ma->idev)) {
  		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);
  	__in6_dev_put(idev);
  }
  
  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);
  	}
  	__in6_dev_put(idev);
  }
  
  static void mld_ifc_event(struct inet6_dev *idev)
  {
  	if (MLD_V1_SEEN(idev))
  		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;
  
  	if (MLD_V1_SEEN(ma->idev))
  		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...
2133
2134
2135
2136
2137
2138
2139
2140
2141
2142
2143
2144
2145
2146
2147
2148
2149
2150
  /* 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
2151
2152
2153
2154
2155
2156
2157
2158
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
2172
2173
2174
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
2190
2191
2192
  /* Device going down */
  
  void ipv6_mc_down(struct inet6_dev *idev)
  {
  	struct ifmcaddr6 *i;
  
  	/* Withdraw multicast list */
  
  	read_lock_bh(&idev->lock);
  	idev->mc_ifc_count = 0;
  	if (del_timer(&idev->mc_ifc_timer))
  		__in6_dev_put(idev);
  	idev->mc_gq_running = 0;
  	if (del_timer(&idev->mc_gq_timer))
  		__in6_dev_put(idev);
  
  	for (i = idev->mc_list; i; i=i->next)
  		igmp6_group_dropped(i);
  	read_unlock_bh(&idev->lock);
  
  	mld_clear_delrec(idev);
  }
  
  
  /* 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);
  	for (i = idev->mc_list; i; i=i->next)
  		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
2193
  	write_lock_bh(&idev->lock);
6457d26bd   Stephen Hemminger   IPv6: convert mc_...
2194
  	spin_lock_init(&idev->mc_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2195
  	idev->mc_gq_running = 0;
b24b8a247   Pavel Emelyanov   [NET]: Convert in...
2196
2197
  	setup_timer(&idev->mc_gq_timer, mld_gq_timer_expire,
  			(unsigned long)idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2198
2199
  	idev->mc_tomb = NULL;
  	idev->mc_ifc_count = 0;
b24b8a247   Pavel Emelyanov   [NET]: Convert in...
2200
2201
  	setup_timer(&idev->mc_ifc_timer, mld_ifc_timer_expire,
  			(unsigned long)idev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2202
2203
2204
2205
  	idev->mc_qrv = MLD_QRV_DEFAULT;
  	idev->mc_maxdelay = IGMP6_UNSOLICITED_IVAL;
  	idev->mc_v1_seen = 0;
  	write_unlock_bh(&idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2206
2207
2208
2209
2210
2211
2212
2213
2214
  }
  
  /*
   *	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
2215
2216
2217
2218
2219
  
  	/* Deactivate timers */
  	ipv6_mc_down(idev);
  
  	/* Delete all-nodes address. */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2220
2221
2222
2223
  	/* 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...
2224
  	__ipv6_dev_mc_dec(idev, &in6addr_linklocal_allnodes);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2225

f3ee4010e   YOSHIFUJI Hideaki   [IPV6]: Define co...
2226
2227
  	if (idev->cnf.forwarding)
  		__ipv6_dev_mc_dec(idev, &in6addr_linklocal_allrouters);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2228
2229
2230
2231
2232
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
  
  	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...
2244
  	struct seq_net_private p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
  	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...
2255
  	struct net *net = seq_file_net(seq);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2256

7562f876c   Pavel Emelianov   [NET]: Rework dev...
2257
  	state->idev = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2258
  	for_each_netdev_rcu(net, state->dev) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2259
  		struct inet6_dev *idev;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2260
  		idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2261
2262
2263
2264
2265
2266
2267
2268
2269
  		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
2270
2271
2272
2273
2274
2275
2276
2277
2278
2279
  	}
  	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) {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2280
  		if (likely(state->idev != NULL))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2281
  			read_unlock_bh(&state->idev->lock);
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2282
2283
  
  		state->dev = next_net_device_rcu(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2284
2285
2286
2287
  		if (!state->dev) {
  			state->idev = NULL;
  			break;
  		}
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2288
  		state->idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2289
2290
2291
2292
2293
2294
2295
2296
2297
2298
2299
2300
2301
2302
2303
2304
2305
2306
  		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 ...
2307
  	__acquires(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2308
  {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2309
  	rcu_read_lock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2310
2311
2312
2313
2314
  	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 ...
2315
  	struct ifmcaddr6 *im = igmp6_mc_get_next(seq, v);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2316
2317
2318
2319
2320
  	++*pos;
  	return im;
  }
  
  static void igmp6_mc_seq_stop(struct seq_file *seq, void *v)
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2321
  	__releases(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2322
2323
  {
  	struct igmp6_mc_iter_state *state = igmp6_mc_seq_private(seq);
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2324

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2325
2326
  	if (likely(state->idev != NULL)) {
  		read_unlock_bh(&state->idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2327
2328
2329
  		state->idev = NULL;
  	}
  	state->dev = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2330
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2331
2332
2333
2334
2335
2336
2337
2338
  }
  
  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...
2339
2340
  		   "%-4d %-15s %pi6 %5d %08X %ld
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2341
  		   state->dev->ifindex, state->dev->name,
b071195de   Harvey Harrison   net: replace all ...
2342
  		   &im->mca_addr,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2343
2344
2345
2346
2347
  		   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...
2348
  static const struct seq_operations igmp6_mc_seq_ops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2349
2350
2351
2352
2353
2354
2355
2356
  	.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...
2357
2358
  	return seq_open_net(inode, file, &igmp6_mc_seq_ops,
  			    sizeof(struct igmp6_mc_iter_state));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2359
  }
9a32144e9   Arjan van de Ven   [PATCH] mark stru...
2360
  static const struct file_operations igmp6_mc_seq_fops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2361
2362
2363
2364
  	.owner		=	THIS_MODULE,
  	.open		=	igmp6_mc_seq_open,
  	.read		=	seq_read,
  	.llseek		=	seq_lseek,
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2365
  	.release	=	seq_release_net,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2366
2367
2368
  };
  
  struct igmp6_mcf_iter_state {
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2369
  	struct seq_net_private p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2370
2371
2372
2373
2374
2375
2376
2377
2378
2379
2380
2381
  	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...
2382
  	struct net *net = seq_file_net(seq);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2383

7562f876c   Pavel Emelianov   [NET]: Rework dev...
2384
2385
  	state->idev = NULL;
  	state->im = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2386
  	for_each_netdev_rcu(net, state->dev) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2387
  		struct inet6_dev *idev;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2388
  		idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2389
2390
2391
2392
2393
2394
2395
2396
2397
2398
2399
2400
2401
2402
2403
  		if (unlikely(idev == NULL))
  			continue;
  		read_lock_bh(&idev->lock);
  		im = idev->mc_list;
  		if (likely(im != NULL)) {
  			spin_lock_bh(&im->mca_lock);
  			psf = im->mca_sources;
  			if (likely(psf != NULL)) {
  				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
2404
2405
2406
2407
2408
2409
2410
2411
2412
2413
2414
2415
2416
  	}
  	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) {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2417
  			if (likely(state->idev != NULL))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2418
  				read_unlock_bh(&state->idev->lock);
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2419
2420
  
  			state->dev = next_net_device_rcu(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2421
2422
2423
2424
  			if (!state->dev) {
  				state->idev = NULL;
  				goto out;
  			}
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2425
  			state->idev = __in6_dev_get(state->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2426
2427
2428
2429
2430
2431
2432
2433
2434
2435
2436
2437
2438
2439
2440
2441
2442
2443
2444
2445
2446
2447
2448
2449
  			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 ...
2450
  	__acquires(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2451
  {
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2452
  	rcu_read_lock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2453
2454
2455
2456
2457
2458
2459
2460
2461
2462
2463
2464
2465
2466
2467
  	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 ...
2468
  	__releases(RCU)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2469
2470
2471
2472
2473
2474
2475
2476
  {
  	struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq);
  	if (likely(state->im != NULL)) {
  		spin_unlock_bh(&state->im->mca_lock);
  		state->im = NULL;
  	}
  	if (likely(state->idev != NULL)) {
  		read_unlock_bh(&state->idev->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2477
2478
2479
  		state->idev = NULL;
  	}
  	state->dev = NULL;
ce81b76a3   Eric Dumazet   ipv6: use RCU to ...
2480
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2481
2482
2483
2484
2485
2486
2487
2488
  }
  
  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) {
1ab1457c4   YOSHIFUJI Hideaki   [NET] IPV6: Fix w...
2489
  		seq_printf(seq,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2490
  			   "%3s %6s "
9343e79a7   YOSHIFUJI Hideaki   [IPV6]: Preserve ...
2491
2492
  			   "%32s %32s %6s %6s
  ", "Idx",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2493
2494
2495
2496
  			   "Device", "Multicast Address",
  			   "Source Address", "INC", "EXC");
  	} else {
  		seq_printf(seq,
4b7a4274c   Harvey Harrison   net: replace %#p6...
2497
2498
  			   "%3d %6.6s %pi6 %pi6 %6lu %6lu
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2499
  			   state->dev->ifindex, state->dev->name,
b071195de   Harvey Harrison   net: replace all ...
2500
2501
  			   &state->im->mca_addr,
  			   &psf->sf_addr,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2502
2503
2504
2505
2506
  			   psf->sf_count[MCAST_INCLUDE],
  			   psf->sf_count[MCAST_EXCLUDE]);
  	}
  	return 0;
  }
56b3d975b   Philippe De Muyter   [NET]: Make all i...
2507
  static const struct seq_operations igmp6_mcf_seq_ops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2508
2509
2510
2511
2512
2513
2514
2515
  	.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...
2516
2517
  	return seq_open_net(inode, file, &igmp6_mcf_seq_ops,
  			    sizeof(struct igmp6_mcf_iter_state));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2518
  }
9a32144e9   Arjan van de Ven   [PATCH] mark stru...
2519
  static const struct file_operations igmp6_mcf_seq_fops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2520
2521
2522
2523
  	.owner		=	THIS_MODULE,
  	.open		=	igmp6_mcf_seq_open,
  	.read		=	seq_read,
  	.llseek		=	seq_lseek,
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2524
  	.release	=	seq_release_net,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2525
  };
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2526

2c8c1e729   Alexey Dobriyan   net: spread __net...
2527
  static int __net_init igmp6_proc_init(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2528
2529
2530
2531
2532
2533
2534
2535
2536
2537
2538
2539
2540
2541
2542
2543
2544
2545
  {
  	int err;
  
  	err = -ENOMEM;
  	if (!proc_net_fops_create(net, "igmp6", S_IRUGO, &igmp6_mc_seq_fops))
  		goto out;
  	if (!proc_net_fops_create(net, "mcfilter6", S_IRUGO,
  				  &igmp6_mcf_seq_fops))
  		goto out_proc_net_igmp6;
  
  	err = 0;
  out:
  	return err;
  
  out_proc_net_igmp6:
  	proc_net_remove(net, "igmp6");
  	goto out;
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
2546
  static void __net_exit igmp6_proc_exit(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2547
2548
2549
2550
2551
  {
  	proc_net_remove(net, "mcfilter6");
  	proc_net_remove(net, "igmp6");
  }
  #else
2c8c1e729   Alexey Dobriyan   net: spread __net...
2552
  static inline int igmp6_proc_init(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2553
2554
2555
  {
  	return 0;
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
2556
  static inline void igmp6_proc_exit(struct net *net)
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2557
  {
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2558
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2559
  #endif
2c8c1e729   Alexey Dobriyan   net: spread __net...
2560
  static int __net_init igmp6_net_init(struct net *net)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2561
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2562
  	int err;
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2563
2564
  	err = inet_ctl_sock_create(&net->ipv6.igmp_sk, PF_INET6,
  				   SOCK_RAW, IPPROTO_ICMPV6, net);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2565
2566
2567
2568
2569
  	if (err < 0) {
  		printk(KERN_ERR
  		       "Failed to initialize the IGMP6 control socket (err %d).
  ",
  		       err);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2570
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2571
  	}
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2572
  	inet6_sk(net->ipv6.igmp_sk)->hop_limit = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2573

ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2574
2575
  	err = igmp6_proc_init(net);
  	if (err)
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2576
  		goto out_sock_create;
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2577
2578
2579
2580
  out:
  	return err;
  
  out_sock_create:
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2581
  	inet_ctl_sock_destroy(net->ipv6.igmp_sk);
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2582
  	goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2583
  }
2c8c1e729   Alexey Dobriyan   net: spread __net...
2584
  static void __net_exit igmp6_net_exit(struct net *net)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2585
  {
1ed8516f0   Denis V. Lunev   [IPV6]: Simplify ...
2586
  	inet_ctl_sock_destroy(net->ipv6.igmp_sk);
ea82edf70   Daniel Lezcano   [NETNS][IPV6] mca...
2587
  	igmp6_proc_exit(net);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2588
  }
b8ad0cbc5   Daniel Lezcano   [NETNS][IPV6] mca...
2589
2590
2591
2592
2593
2594
2595
2596
2597
2598
2599
2600
2601
2602
2603
  
  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);
  }