|
@@ -172,6 +172,7 @@ int ipv6_sock_mc_join(struct sock *sk, int ifindex, const struct in6_addr *addr)
|
|
|
mc_lst->next = NULL;
|
|
|
mc_lst->addr = *addr;
|
|
|
|
|
|
+ rtnl_lock();
|
|
|
rcu_read_lock();
|
|
|
if (ifindex == 0) {
|
|
|
struct rt6_info *rt;
|
|
@@ -185,6 +186,7 @@ int ipv6_sock_mc_join(struct sock *sk, int ifindex, const struct in6_addr *addr)
|
|
|
|
|
|
if (dev == NULL) {
|
|
|
rcu_read_unlock();
|
|
|
+ rtnl_unlock();
|
|
|
sock_kfree_s(sk, mc_lst, sizeof(*mc_lst));
|
|
|
return -ENODEV;
|
|
|
}
|
|
@@ -202,6 +204,7 @@ int ipv6_sock_mc_join(struct sock *sk, int ifindex, const struct in6_addr *addr)
|
|
|
|
|
|
if (err) {
|
|
|
rcu_read_unlock();
|
|
|
+ rtnl_unlock();
|
|
|
sock_kfree_s(sk, mc_lst, sizeof(*mc_lst));
|
|
|
return err;
|
|
|
}
|
|
@@ -212,6 +215,7 @@ int ipv6_sock_mc_join(struct sock *sk, int ifindex, const struct in6_addr *addr)
|
|
|
spin_unlock(&ipv6_sk_mc_lock);
|
|
|
|
|
|
rcu_read_unlock();
|
|
|
+ rtnl_unlock();
|
|
|
|
|
|
return 0;
|
|
|
}
|
|
@@ -229,6 +233,7 @@ int ipv6_sock_mc_drop(struct sock *sk, int ifindex, const struct in6_addr *addr)
|
|
|
if (!ipv6_addr_is_multicast(addr))
|
|
|
return -EINVAL;
|
|
|
|
|
|
+ rtnl_lock();
|
|
|
spin_lock(&ipv6_sk_mc_lock);
|
|
|
for (lnk = &np->ipv6_mc_list;
|
|
|
(mc_lst = rcu_dereference_protected(*lnk,
|
|
@@ -252,12 +257,15 @@ int ipv6_sock_mc_drop(struct sock *sk, int ifindex, const struct in6_addr *addr)
|
|
|
} else
|
|
|
(void) ip6_mc_leave_src(sk, mc_lst, NULL);
|
|
|
rcu_read_unlock();
|
|
|
+ rtnl_unlock();
|
|
|
+
|
|
|
atomic_sub(sizeof(*mc_lst), &sk->sk_omem_alloc);
|
|
|
kfree_rcu(mc_lst, rcu);
|
|
|
return 0;
|
|
|
}
|
|
|
}
|
|
|
spin_unlock(&ipv6_sk_mc_lock);
|
|
|
+ rtnl_unlock();
|
|
|
|
|
|
return -EADDRNOTAVAIL;
|
|
|
}
|
|
@@ -302,6 +310,7 @@ void ipv6_sock_mc_close(struct sock *sk)
|
|
|
if (!rcu_access_pointer(np->ipv6_mc_list))
|
|
|
return;
|
|
|
|
|
|
+ rtnl_lock();
|
|
|
spin_lock(&ipv6_sk_mc_lock);
|
|
|
while ((mc_lst = rcu_dereference_protected(np->ipv6_mc_list,
|
|
|
lockdep_is_held(&ipv6_sk_mc_lock))) != NULL) {
|
|
@@ -328,6 +337,7 @@ void ipv6_sock_mc_close(struct sock *sk)
|
|
|
spin_lock(&ipv6_sk_mc_lock);
|
|
|
}
|
|
|
spin_unlock(&ipv6_sk_mc_lock);
|
|
|
+ rtnl_unlock();
|
|
|
}
|
|
|
|
|
|
int ip6_mc_source(int add, int omode, struct sock *sk,
|
|
@@ -845,6 +855,8 @@ int ipv6_dev_mc_inc(struct net_device *dev, const struct in6_addr *addr)
|
|
|
struct ifmcaddr6 *mc;
|
|
|
struct inet6_dev *idev;
|
|
|
|
|
|
+ ASSERT_RTNL();
|
|
|
+
|
|
|
/* we need to take a reference on idev */
|
|
|
idev = in6_dev_get(dev);
|
|
|
|
|
@@ -916,6 +928,8 @@ int __ipv6_dev_mc_dec(struct inet6_dev *idev, const struct in6_addr *addr)
|
|
|
{
|
|
|
struct ifmcaddr6 *ma, **map;
|
|
|
|
|
|
+ ASSERT_RTNL();
|
|
|
+
|
|
|
write_lock_bh(&idev->lock);
|
|
|
for (map = &idev->mc_list; (ma=*map) != NULL; map = &ma->next) {
|
|
|
if (ipv6_addr_equal(&ma->mca_addr, addr)) {
|