Commit f185de28 authored by Taehee Yoo's avatar Taehee Yoo Committed by David S. Miller

mld: add new workqueues for process mld events

When query/report packets are received, mld module processes them.
But they are processed under BH context so it couldn't use sleepable
functions. So, in order to switch context, the two workqueues are
added which processes query and report event.

In the struct inet6_dev, mc_{query | report}_queue are added so it
is per-interface queue.
And mc_{query | report}_work are workqueue structure.

When the query or report event is received, skb is queued to proper
queue and worker function is scheduled immediately.
Workqueues and queues are protected by spinlock, which is
mc_{query | report}_lock, and worker functions are protected by RTNL.
Suggested-by: default avatarCong Wang <xiyou.wangcong@gmail.com>
Signed-off-by: default avatarTaehee Yoo <ap420073@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 88e2ca30
...@@ -125,7 +125,6 @@ struct ifmcaddr6 { ...@@ -125,7 +125,6 @@ struct ifmcaddr6 {
unsigned int mca_flags; unsigned int mca_flags;
int mca_users; int mca_users;
refcount_t mca_refcnt; refcount_t mca_refcnt;
spinlock_t mca_lock;
unsigned long mca_cstamp; unsigned long mca_cstamp;
unsigned long mca_tstamp; unsigned long mca_tstamp;
struct rcu_head rcu; struct rcu_head rcu;
...@@ -183,6 +182,14 @@ struct inet6_dev { ...@@ -183,6 +182,14 @@ struct inet6_dev {
struct delayed_work mc_gq_work; /* general query work */ struct delayed_work mc_gq_work; /* general query work */
struct delayed_work mc_ifc_work; /* interface change work */ struct delayed_work mc_ifc_work; /* interface change work */
struct delayed_work mc_dad_work; /* dad complete mc work */ struct delayed_work mc_dad_work; /* dad complete mc work */
struct delayed_work mc_query_work; /* mld query work */
struct delayed_work mc_report_work; /* mld report work */
struct sk_buff_head mc_query_queue; /* mld query queue */
struct sk_buff_head mc_report_queue; /* mld report queue */
spinlock_t mc_query_lock; /* mld query queue lock */
spinlock_t mc_report_lock; /* mld query report lock */
struct ifacaddr6 *ac_list; struct ifacaddr6 *ac_list;
rwlock_t lock; rwlock_t lock;
......
...@@ -92,6 +92,9 @@ struct mld2_query { ...@@ -92,6 +92,9 @@ struct mld2_query {
#define MLD_EXP_MIN_LIMIT 32768UL #define MLD_EXP_MIN_LIMIT 32768UL
#define MLDV1_MRD_MAX_COMPAT (MLD_EXP_MIN_LIMIT - 1) #define MLDV1_MRD_MAX_COMPAT (MLD_EXP_MIN_LIMIT - 1)
#define MLD_MAX_QUEUE 8
#define MLD_MAX_SKBS 32
static inline unsigned long mldv2_mrc(const struct mld2_query *mlh2) static inline unsigned long mldv2_mrc(const struct mld2_query *mlh2)
{ {
/* RFC3810, 5.1.3. Maximum Response Code */ /* RFC3810, 5.1.3. Maximum Response Code */
......
...@@ -944,11 +944,11 @@ static int icmpv6_rcv(struct sk_buff *skb) ...@@ -944,11 +944,11 @@ static int icmpv6_rcv(struct sk_buff *skb)
case ICMPV6_MGM_QUERY: case ICMPV6_MGM_QUERY:
igmp6_event_query(skb); igmp6_event_query(skb);
break; return 0;
case ICMPV6_MGM_REPORT: case ICMPV6_MGM_REPORT:
igmp6_event_report(skb); igmp6_event_report(skb);
break; return 0;
case ICMPV6_MGM_REDUCTION: case ICMPV6_MGM_REDUCTION:
case ICMPV6_NI_QUERY: case ICMPV6_NI_QUERY:
......
...@@ -439,7 +439,7 @@ int ip6_mc_source(int add, int omode, struct sock *sk, ...@@ -439,7 +439,7 @@ int ip6_mc_source(int add, int omode, struct sock *sk,
if (psl) if (psl)
count += psl->sl_max; count += psl->sl_max;
newpsl = sock_kmalloc(sk, IP6_SFLSIZE(count), GFP_ATOMIC); newpsl = sock_kmalloc(sk, IP6_SFLSIZE(count), GFP_KERNEL);
if (!newpsl) { if (!newpsl) {
err = -ENOBUFS; err = -ENOBUFS;
goto done; goto done;
...@@ -517,7 +517,7 @@ int ip6_mc_msfilter(struct sock *sk, struct group_filter *gsf, ...@@ -517,7 +517,7 @@ int ip6_mc_msfilter(struct sock *sk, struct group_filter *gsf,
} }
if (gsf->gf_numsrc) { if (gsf->gf_numsrc) {
newpsl = sock_kmalloc(sk, IP6_SFLSIZE(gsf->gf_numsrc), newpsl = sock_kmalloc(sk, IP6_SFLSIZE(gsf->gf_numsrc),
GFP_ATOMIC); GFP_KERNEL);
if (!newpsl) { if (!newpsl) {
err = -ENOBUFS; err = -ENOBUFS;
goto done; goto done;
...@@ -659,13 +659,11 @@ static void igmp6_group_added(struct ifmcaddr6 *mc) ...@@ -659,13 +659,11 @@ static void igmp6_group_added(struct ifmcaddr6 *mc)
IPV6_ADDR_SCOPE_LINKLOCAL) IPV6_ADDR_SCOPE_LINKLOCAL)
return; return;
spin_lock_bh(&mc->mca_lock);
if (!(mc->mca_flags&MAF_LOADED)) { if (!(mc->mca_flags&MAF_LOADED)) {
mc->mca_flags |= MAF_LOADED; mc->mca_flags |= MAF_LOADED;
if (ndisc_mc_map(&mc->mca_addr, buf, dev, 0) == 0) if (ndisc_mc_map(&mc->mca_addr, buf, dev, 0) == 0)
dev_mc_add(dev, buf); dev_mc_add(dev, buf);
} }
spin_unlock_bh(&mc->mca_lock);
if (!(dev->flags & IFF_UP) || (mc->mca_flags & MAF_NOREPORT)) if (!(dev->flags & IFF_UP) || (mc->mca_flags & MAF_NOREPORT))
return; return;
...@@ -695,24 +693,20 @@ static void igmp6_group_dropped(struct ifmcaddr6 *mc) ...@@ -695,24 +693,20 @@ static void igmp6_group_dropped(struct ifmcaddr6 *mc)
IPV6_ADDR_SCOPE_LINKLOCAL) IPV6_ADDR_SCOPE_LINKLOCAL)
return; return;
spin_lock_bh(&mc->mca_lock);
if (mc->mca_flags&MAF_LOADED) { if (mc->mca_flags&MAF_LOADED) {
mc->mca_flags &= ~MAF_LOADED; mc->mca_flags &= ~MAF_LOADED;
if (ndisc_mc_map(&mc->mca_addr, buf, dev, 0) == 0) if (ndisc_mc_map(&mc->mca_addr, buf, dev, 0) == 0)
dev_mc_del(dev, buf); dev_mc_del(dev, buf);
} }
spin_unlock_bh(&mc->mca_lock);
if (mc->mca_flags & MAF_NOREPORT) if (mc->mca_flags & MAF_NOREPORT)
return; return;
if (!mc->idev->dead) if (!mc->idev->dead)
igmp6_leave_group(mc); igmp6_leave_group(mc);
spin_lock_bh(&mc->mca_lock);
if (cancel_delayed_work(&mc->mca_work)) if (cancel_delayed_work(&mc->mca_work))
refcount_dec(&mc->mca_refcnt); refcount_dec(&mc->mca_refcnt);
spin_unlock_bh(&mc->mca_lock);
} }
/* /*
...@@ -728,12 +722,10 @@ static void mld_add_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im) ...@@ -728,12 +722,10 @@ static void mld_add_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im)
* for deleted items allows change reports to use common code with * for deleted items allows change reports to use common code with
* non-deleted or query-response MCA's. * non-deleted or query-response MCA's.
*/ */
pmc = kzalloc(sizeof(*pmc), GFP_ATOMIC); pmc = kzalloc(sizeof(*pmc), GFP_KERNEL);
if (!pmc) if (!pmc)
return; return;
spin_lock_bh(&im->mca_lock);
spin_lock_init(&pmc->mca_lock);
pmc->idev = im->idev; pmc->idev = im->idev;
in6_dev_hold(idev); in6_dev_hold(idev);
pmc->mca_addr = im->mca_addr; pmc->mca_addr = im->mca_addr;
...@@ -752,7 +744,6 @@ static void mld_add_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im) ...@@ -752,7 +744,6 @@ static void mld_add_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im)
for_each_psf_rtnl(pmc, psf) for_each_psf_rtnl(pmc, psf)
psf->sf_crcount = pmc->mca_crcount; psf->sf_crcount = pmc->mca_crcount;
} }
spin_unlock_bh(&im->mca_lock);
rcu_assign_pointer(pmc->next, idev->mc_tomb); rcu_assign_pointer(pmc->next, idev->mc_tomb);
rcu_assign_pointer(idev->mc_tomb, pmc); rcu_assign_pointer(idev->mc_tomb, pmc);
...@@ -777,7 +768,6 @@ static void mld_del_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im) ...@@ -777,7 +768,6 @@ static void mld_del_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im)
rcu_assign_pointer(idev->mc_tomb, pmc->next); rcu_assign_pointer(idev->mc_tomb, pmc->next);
} }
spin_lock_bh(&im->mca_lock);
if (pmc) { if (pmc) {
im->idev = pmc->idev; im->idev = pmc->idev;
if (im->mca_sfmode == MCAST_INCLUDE) { if (im->mca_sfmode == MCAST_INCLUDE) {
...@@ -799,7 +789,6 @@ static void mld_del_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im) ...@@ -799,7 +789,6 @@ static void mld_del_delrec(struct inet6_dev *idev, struct ifmcaddr6 *im)
ip6_mc_clear_src(pmc); ip6_mc_clear_src(pmc);
kfree_rcu(pmc, rcu); kfree_rcu(pmc, rcu);
} }
spin_unlock_bh(&im->mca_lock);
} }
static void mld_clear_delrec(struct inet6_dev *idev) static void mld_clear_delrec(struct inet6_dev *idev)
...@@ -820,10 +809,8 @@ static void mld_clear_delrec(struct inet6_dev *idev) ...@@ -820,10 +809,8 @@ static void mld_clear_delrec(struct inet6_dev *idev)
for_each_mc_rtnl(idev, pmc) { for_each_mc_rtnl(idev, pmc) {
struct ip6_sf_list *psf, *psf_next; struct ip6_sf_list *psf, *psf_next;
spin_lock_bh(&pmc->mca_lock);
psf = rtnl_dereference(pmc->mca_tomb); psf = rtnl_dereference(pmc->mca_tomb);
RCU_INIT_POINTER(pmc->mca_tomb, NULL); RCU_INIT_POINTER(pmc->mca_tomb, NULL);
spin_unlock_bh(&pmc->mca_lock);
for (; psf; psf = psf_next) { for (; psf; psf = psf_next) {
psf_next = rtnl_dereference(psf->sf_next); psf_next = rtnl_dereference(psf->sf_next);
kfree_rcu(psf, rcu); kfree_rcu(psf, rcu);
...@@ -831,6 +818,26 @@ static void mld_clear_delrec(struct inet6_dev *idev) ...@@ -831,6 +818,26 @@ static void mld_clear_delrec(struct inet6_dev *idev)
} }
} }
static void mld_clear_query(struct inet6_dev *idev)
{
struct sk_buff *skb;
spin_lock_bh(&idev->mc_query_lock);
while ((skb = __skb_dequeue(&idev->mc_query_queue)))
kfree_skb(skb);
spin_unlock_bh(&idev->mc_query_lock);
}
static void mld_clear_report(struct inet6_dev *idev)
{
struct sk_buff *skb;
spin_lock_bh(&idev->mc_report_lock);
while ((skb = __skb_dequeue(&idev->mc_report_queue)))
kfree_skb(skb);
spin_unlock_bh(&idev->mc_report_lock);
}
static void mca_get(struct ifmcaddr6 *mc) static void mca_get(struct ifmcaddr6 *mc)
{ {
refcount_inc(&mc->mca_refcnt); refcount_inc(&mc->mca_refcnt);
...@@ -850,7 +857,7 @@ static struct ifmcaddr6 *mca_alloc(struct inet6_dev *idev, ...@@ -850,7 +857,7 @@ static struct ifmcaddr6 *mca_alloc(struct inet6_dev *idev,
{ {
struct ifmcaddr6 *mc; struct ifmcaddr6 *mc;
mc = kzalloc(sizeof(*mc), GFP_ATOMIC); mc = kzalloc(sizeof(*mc), GFP_KERNEL);
if (!mc) if (!mc)
return NULL; return NULL;
...@@ -862,7 +869,6 @@ static struct ifmcaddr6 *mca_alloc(struct inet6_dev *idev, ...@@ -862,7 +869,6 @@ static struct ifmcaddr6 *mca_alloc(struct inet6_dev *idev,
/* mca_stamp should be updated upon changes */ /* mca_stamp should be updated upon changes */
mc->mca_cstamp = mc->mca_tstamp = jiffies; mc->mca_cstamp = mc->mca_tstamp = jiffies;
refcount_set(&mc->mca_refcnt, 1); refcount_set(&mc->mca_refcnt, 1);
spin_lock_init(&mc->mca_lock);
mc->mca_sfmode = mode; mc->mca_sfmode = mode;
mc->mca_sfcount[mode] = 1; mc->mca_sfcount[mode] = 1;
...@@ -995,7 +1001,6 @@ bool ipv6_chk_mcast_addr(struct net_device *dev, const struct in6_addr *group, ...@@ -995,7 +1001,6 @@ bool ipv6_chk_mcast_addr(struct net_device *dev, const struct in6_addr *group,
if (src_addr && !ipv6_addr_any(src_addr)) { if (src_addr && !ipv6_addr_any(src_addr)) {
struct ip6_sf_list *psf; struct ip6_sf_list *psf;
spin_lock_bh(&mc->mca_lock);
for_each_psf_rcu(mc, psf) { for_each_psf_rcu(mc, psf) {
if (ipv6_addr_equal(&psf->sf_addr, src_addr)) if (ipv6_addr_equal(&psf->sf_addr, src_addr))
break; break;
...@@ -1006,7 +1011,6 @@ bool ipv6_chk_mcast_addr(struct net_device *dev, const struct in6_addr *group, ...@@ -1006,7 +1011,6 @@ bool ipv6_chk_mcast_addr(struct net_device *dev, const struct in6_addr *group,
mc->mca_sfcount[MCAST_EXCLUDE]; mc->mca_sfcount[MCAST_EXCLUDE];
else else
rv = mc->mca_sfcount[MCAST_EXCLUDE] != 0; rv = mc->mca_sfcount[MCAST_EXCLUDE] != 0;
spin_unlock_bh(&mc->mca_lock);
} else } else
rv = true; /* don't filter unspecified source */ rv = true; /* don't filter unspecified source */
} }
...@@ -1060,6 +1064,20 @@ static void mld_dad_stop_work(struct inet6_dev *idev) ...@@ -1060,6 +1064,20 @@ static void mld_dad_stop_work(struct inet6_dev *idev)
__in6_dev_put(idev); __in6_dev_put(idev);
} }
static void mld_query_stop_work(struct inet6_dev *idev)
{
spin_lock_bh(&idev->mc_query_lock);
if (cancel_delayed_work(&idev->mc_query_work))
__in6_dev_put(idev);
spin_unlock_bh(&idev->mc_query_lock);
}
static void mld_report_stop_work(struct inet6_dev *idev)
{
if (cancel_delayed_work_sync(&idev->mc_report_work))
__in6_dev_put(idev);
}
/* /*
* IGMP handling (alias multicast ICMPv6 messages) * IGMP handling (alias multicast ICMPv6 messages)
*/ */
...@@ -1093,7 +1111,7 @@ static bool mld_xmarksources(struct ifmcaddr6 *pmc, int nsrcs, ...@@ -1093,7 +1111,7 @@ static bool mld_xmarksources(struct ifmcaddr6 *pmc, int nsrcs,
int i, scount; int i, scount;
scount = 0; scount = 0;
for_each_psf_rcu(pmc, psf) { for_each_psf_rtnl(pmc, psf) {
if (scount == nsrcs) if (scount == nsrcs)
break; break;
for (i = 0; i < nsrcs; i++) { for (i = 0; i < nsrcs; i++) {
...@@ -1126,7 +1144,7 @@ static bool mld_marksources(struct ifmcaddr6 *pmc, int nsrcs, ...@@ -1126,7 +1144,7 @@ static bool mld_marksources(struct ifmcaddr6 *pmc, int nsrcs,
/* mark INCLUDE-mode sources */ /* mark INCLUDE-mode sources */
scount = 0; scount = 0;
for_each_psf_rcu(pmc, psf) { for_each_psf_rtnl(pmc, psf) {
if (scount == nsrcs) if (scount == nsrcs)
break; break;
for (i = 0; i < nsrcs; i++) { for (i = 0; i < nsrcs; i++) {
...@@ -1317,19 +1335,42 @@ static int mld_process_v2(struct inet6_dev *idev, struct mld2_query *mld, ...@@ -1317,19 +1335,42 @@ static int mld_process_v2(struct inet6_dev *idev, struct mld2_query *mld,
/* called with rcu_read_lock() */ /* called with rcu_read_lock() */
int igmp6_event_query(struct sk_buff *skb) int igmp6_event_query(struct sk_buff *skb)
{
struct inet6_dev *idev = __in6_dev_get(skb->dev);
if (!idev)
return -EINVAL;
if (idev->dead) {
kfree_skb(skb);
return -ENODEV;
}
spin_lock_bh(&idev->mc_query_lock);
if (skb_queue_len(&idev->mc_query_queue) < MLD_MAX_SKBS) {
__skb_queue_tail(&idev->mc_query_queue, skb);
if (!mod_delayed_work(mld_wq, &idev->mc_query_work, 0))
in6_dev_hold(idev);
}
spin_unlock_bh(&idev->mc_query_lock);
return 0;
}
static void __mld_query_work(struct sk_buff *skb)
{ {
struct mld2_query *mlh2 = NULL; struct mld2_query *mlh2 = NULL;
struct ifmcaddr6 *ma;
const struct in6_addr *group; const struct in6_addr *group;
unsigned long max_delay; unsigned long max_delay;
struct inet6_dev *idev; struct inet6_dev *idev;
struct ifmcaddr6 *ma;
struct mld_msg *mld; struct mld_msg *mld;
int group_type; int group_type;
int mark = 0; int mark = 0;
int len, err; int len, err;
if (!pskb_may_pull(skb, sizeof(struct in6_addr))) if (!pskb_may_pull(skb, sizeof(struct in6_addr)))
return -EINVAL; goto out;
/* compute payload length excluding extension headers */ /* compute payload length excluding extension headers */
len = ntohs(ipv6_hdr(skb)->payload_len) + sizeof(struct ipv6hdr); len = ntohs(ipv6_hdr(skb)->payload_len) + sizeof(struct ipv6hdr);
...@@ -1346,11 +1387,11 @@ int igmp6_event_query(struct sk_buff *skb) ...@@ -1346,11 +1387,11 @@ int igmp6_event_query(struct sk_buff *skb)
ipv6_hdr(skb)->hop_limit != 1 || ipv6_hdr(skb)->hop_limit != 1 ||
!(IP6CB(skb)->flags & IP6SKB_ROUTERALERT) || !(IP6CB(skb)->flags & IP6SKB_ROUTERALERT) ||
IP6CB(skb)->ra != htons(IPV6_OPT_ROUTERALERT_MLD)) IP6CB(skb)->ra != htons(IPV6_OPT_ROUTERALERT_MLD))
return -EINVAL; goto out;
idev = __in6_dev_get(skb->dev); idev = __in6_dev_get(skb->dev);
if (!idev) if (!idev)
return 0; goto out;
mld = (struct mld_msg *)icmp6_hdr(skb); mld = (struct mld_msg *)icmp6_hdr(skb);
group = &mld->mld_mca; group = &mld->mld_mca;
...@@ -1358,59 +1399,56 @@ int igmp6_event_query(struct sk_buff *skb) ...@@ -1358,59 +1399,56 @@ int igmp6_event_query(struct sk_buff *skb)
if (group_type != IPV6_ADDR_ANY && if (group_type != IPV6_ADDR_ANY &&
!(group_type&IPV6_ADDR_MULTICAST)) !(group_type&IPV6_ADDR_MULTICAST))
return -EINVAL; goto out;
if (len < MLD_V1_QUERY_LEN) { if (len < MLD_V1_QUERY_LEN) {
return -EINVAL; goto out;
} else if (len == MLD_V1_QUERY_LEN || mld_in_v1_mode(idev)) { } else if (len == MLD_V1_QUERY_LEN || mld_in_v1_mode(idev)) {
err = mld_process_v1(idev, mld, &max_delay, err = mld_process_v1(idev, mld, &max_delay,
len == MLD_V1_QUERY_LEN); len == MLD_V1_QUERY_LEN);
if (err < 0) if (err < 0)
return err; goto out;
} else if (len >= MLD_V2_QUERY_LEN_MIN) { } else if (len >= MLD_V2_QUERY_LEN_MIN) {
int srcs_offset = sizeof(struct mld2_query) - int srcs_offset = sizeof(struct mld2_query) -
sizeof(struct icmp6hdr); sizeof(struct icmp6hdr);
if (!pskb_may_pull(skb, srcs_offset)) if (!pskb_may_pull(skb, srcs_offset))
return -EINVAL; goto out;
mlh2 = (struct mld2_query *)skb_transport_header(skb); mlh2 = (struct mld2_query *)skb_transport_header(skb);
err = mld_process_v2(idev, mlh2, &max_delay); err = mld_process_v2(idev, mlh2, &max_delay);
if (err < 0) if (err < 0)
return err; goto out;
if (group_type == IPV6_ADDR_ANY) { /* general query */ if (group_type == IPV6_ADDR_ANY) { /* general query */
if (mlh2->mld2q_nsrcs) if (mlh2->mld2q_nsrcs)
return -EINVAL; /* no sources allowed */ goto out; /* no sources allowed */
mld_gq_start_work(idev); mld_gq_start_work(idev);
return 0; goto out;
} }
/* mark sources to include, if group & source-specific */ /* mark sources to include, if group & source-specific */
if (mlh2->mld2q_nsrcs != 0) { if (mlh2->mld2q_nsrcs != 0) {
if (!pskb_may_pull(skb, srcs_offset + if (!pskb_may_pull(skb, srcs_offset +
ntohs(mlh2->mld2q_nsrcs) * sizeof(struct in6_addr))) ntohs(mlh2->mld2q_nsrcs) * sizeof(struct in6_addr)))
return -EINVAL; goto out;
mlh2 = (struct mld2_query *)skb_transport_header(skb); mlh2 = (struct mld2_query *)skb_transport_header(skb);
mark = 1; mark = 1;
} }
} else { } else {
return -EINVAL; goto out;
} }
if (group_type == IPV6_ADDR_ANY) { if (group_type == IPV6_ADDR_ANY) {
for_each_mc_rcu(idev, ma) { for_each_mc_rtnl(idev, ma) {
spin_lock_bh(&ma->mca_lock);
igmp6_group_queried(ma, max_delay); igmp6_group_queried(ma, max_delay);
spin_unlock_bh(&ma->mca_lock);
} }
} else { } else {
for_each_mc_rcu(idev, ma) { for_each_mc_rtnl(idev, ma) {
if (!ipv6_addr_equal(group, &ma->mca_addr)) if (!ipv6_addr_equal(group, &ma->mca_addr))
continue; continue;
spin_lock_bh(&ma->mca_lock);
if (ma->mca_flags & MAF_TIMER_RUNNING) { if (ma->mca_flags & MAF_TIMER_RUNNING) {
/* gsquery <- gsquery && mark */ /* gsquery <- gsquery && mark */
if (!mark) if (!mark)
...@@ -1425,16 +1463,72 @@ int igmp6_event_query(struct sk_buff *skb) ...@@ -1425,16 +1463,72 @@ int igmp6_event_query(struct sk_buff *skb)
if (!(ma->mca_flags & MAF_GSQUERY) || if (!(ma->mca_flags & MAF_GSQUERY) ||
mld_marksources(ma, ntohs(mlh2->mld2q_nsrcs), mlh2->mld2q_srcs)) mld_marksources(ma, ntohs(mlh2->mld2q_nsrcs), mlh2->mld2q_srcs))
igmp6_group_queried(ma, max_delay); igmp6_group_queried(ma, max_delay);
spin_unlock_bh(&ma->mca_lock);
break; break;
} }
} }
return 0; out:
consume_skb(skb);
}
static void mld_query_work(struct work_struct *work)
{
struct inet6_dev *idev = container_of(to_delayed_work(work),
struct inet6_dev,
mc_query_work);
struct sk_buff_head q;
struct sk_buff *skb;
bool rework = false;
int cnt = 0;
skb_queue_head_init(&q);
spin_lock_bh(&idev->mc_query_lock);
while ((skb = __skb_dequeue(&idev->mc_query_queue))) {
__skb_queue_tail(&q, skb);
if (++cnt >= MLD_MAX_QUEUE) {
rework = true;
schedule_delayed_work(&idev->mc_query_work, 0);
break;
}
}
spin_unlock_bh(&idev->mc_query_lock);
rtnl_lock();
while ((skb = __skb_dequeue(&q)))
__mld_query_work(skb);
rtnl_unlock();
if (!rework)
in6_dev_put(idev);
} }
/* called with rcu_read_lock() */ /* called with rcu_read_lock() */
int igmp6_event_report(struct sk_buff *skb) int igmp6_event_report(struct sk_buff *skb)
{
struct inet6_dev *idev = __in6_dev_get(skb->dev);
if (!idev)
return -EINVAL;
if (idev->dead) {
kfree_skb(skb);
return -ENODEV;
}
spin_lock_bh(&idev->mc_report_lock);
if (skb_queue_len(&idev->mc_report_queue) < MLD_MAX_SKBS) {
__skb_queue_tail(&idev->mc_report_queue, skb);
if (!mod_delayed_work(mld_wq, &idev->mc_report_work, 0))
in6_dev_hold(idev);
}
spin_unlock_bh(&idev->mc_report_lock);
return 0;
}
static void __mld_report_work(struct sk_buff *skb)
{ {
struct ifmcaddr6 *ma; struct ifmcaddr6 *ma;
struct inet6_dev *idev; struct inet6_dev *idev;
...@@ -1443,15 +1537,15 @@ int igmp6_event_report(struct sk_buff *skb) ...@@ -1443,15 +1537,15 @@ int igmp6_event_report(struct sk_buff *skb)
/* Our own report looped back. Ignore it. */ /* Our own report looped back. Ignore it. */
if (skb->pkt_type == PACKET_LOOPBACK) if (skb->pkt_type == PACKET_LOOPBACK)
return 0; goto out;
/* send our report if the MC router may not have heard this report */ /* send our report if the MC router may not have heard this report */
if (skb->pkt_type != PACKET_MULTICAST && if (skb->pkt_type != PACKET_MULTICAST &&
skb->pkt_type != PACKET_BROADCAST) skb->pkt_type != PACKET_BROADCAST)
return 0; goto out;
if (!pskb_may_pull(skb, sizeof(*mld) - sizeof(struct icmp6hdr))) if (!pskb_may_pull(skb, sizeof(*mld) - sizeof(struct icmp6hdr)))
return -EINVAL; goto out;
mld = (struct mld_msg *)icmp6_hdr(skb); mld = (struct mld_msg *)icmp6_hdr(skb);
...@@ -1459,28 +1553,60 @@ int igmp6_event_report(struct sk_buff *skb) ...@@ -1459,28 +1553,60 @@ int igmp6_event_report(struct sk_buff *skb)
addr_type = ipv6_addr_type(&ipv6_hdr(skb)->saddr); addr_type = ipv6_addr_type(&ipv6_hdr(skb)->saddr);
if (addr_type != IPV6_ADDR_ANY && if (addr_type != IPV6_ADDR_ANY &&
!(addr_type&IPV6_ADDR_LINKLOCAL)) !(addr_type&IPV6_ADDR_LINKLOCAL))
return -EINVAL; goto out;
idev = __in6_dev_get(skb->dev); idev = __in6_dev_get(skb->dev);
if (!idev) if (!idev)
return -ENODEV; goto out;
/* /*
* Cancel the work for this group * Cancel the work for this group
*/ */
for_each_mc_rcu(idev, ma) { for_each_mc_rtnl(idev, ma) {
if (ipv6_addr_equal(&ma->mca_addr, &mld->mld_mca)) { if (ipv6_addr_equal(&ma->mca_addr, &mld->mld_mca)) {
spin_lock(&ma->mca_lock);
if (cancel_delayed_work(&ma->mca_work)) if (cancel_delayed_work(&ma->mca_work))
refcount_dec(&ma->mca_refcnt); refcount_dec(&ma->mca_refcnt);
ma->mca_flags &= ~(MAF_LAST_REPORTER | ma->mca_flags &= ~(MAF_LAST_REPORTER |
MAF_TIMER_RUNNING); MAF_TIMER_RUNNING);
spin_unlock(&ma->mca_lock);
break; break;
} }
} }
return 0;
out:
consume_skb(skb);
}
static void mld_report_work(struct work_struct *work)
{
struct inet6_dev *idev = container_of(to_delayed_work(work),
struct inet6_dev,
mc_report_work);
struct sk_buff_head q;
struct sk_buff *skb;
bool rework = false;
int cnt = 0;
skb_queue_head_init(&q);
spin_lock_bh(&idev->mc_report_lock);
while ((skb = __skb_dequeue(&idev->mc_report_queue))) {
__skb_queue_tail(&q, skb);
if (++cnt >= MLD_MAX_QUEUE) {
rework = true;
schedule_delayed_work(&idev->mc_report_work, 0);
break;
}
}
spin_unlock_bh(&idev->mc_report_lock);
rtnl_lock();
while ((skb = __skb_dequeue(&q)))
__mld_report_work(skb);
rtnl_unlock();
if (!rework)
in6_dev_put(idev);
} }
static bool is_in(struct ifmcaddr6 *pmc, struct ip6_sf_list *psf, int type, static bool is_in(struct ifmcaddr6 *pmc, struct ip6_sf_list *psf, int type,
...@@ -1847,22 +1973,18 @@ static void mld_send_report(struct inet6_dev *idev, struct ifmcaddr6 *pmc) ...@@ -1847,22 +1973,18 @@ static void mld_send_report(struct inet6_dev *idev, struct ifmcaddr6 *pmc)
for_each_mc_rtnl(idev, pmc) { for_each_mc_rtnl(idev, pmc) {
if (pmc->mca_flags & MAF_NOREPORT) if (pmc->mca_flags & MAF_NOREPORT)
continue; continue;
spin_lock_bh(&pmc->mca_lock);
if (pmc->mca_sfcount[MCAST_EXCLUDE]) if (pmc->mca_sfcount[MCAST_EXCLUDE])
type = MLD2_MODE_IS_EXCLUDE; type = MLD2_MODE_IS_EXCLUDE;
else else
type = MLD2_MODE_IS_INCLUDE; type = MLD2_MODE_IS_INCLUDE;
skb = add_grec(skb, pmc, type, 0, 0, 0); skb = add_grec(skb, pmc, type, 0, 0, 0);
spin_unlock_bh(&pmc->mca_lock);
} }
} else { } else {
spin_lock_bh(&pmc->mca_lock);
if (pmc->mca_sfcount[MCAST_EXCLUDE]) if (pmc->mca_sfcount[MCAST_EXCLUDE])
type = MLD2_MODE_IS_EXCLUDE; type = MLD2_MODE_IS_EXCLUDE;
else else
type = MLD2_MODE_IS_INCLUDE; type = MLD2_MODE_IS_INCLUDE;
skb = add_grec(skb, pmc, type, 0, 0, 0); skb = add_grec(skb, pmc, type, 0, 0, 0);
spin_unlock_bh(&pmc->mca_lock);
} }
if (skb) if (skb)
mld_sendpack(skb); mld_sendpack(skb);
...@@ -1938,7 +2060,6 @@ static void mld_send_cr(struct inet6_dev *idev) ...@@ -1938,7 +2060,6 @@ static void mld_send_cr(struct inet6_dev *idev)
/* change recs */ /* change recs */
for_each_mc_rtnl(idev, pmc) { for_each_mc_rtnl(idev, pmc) {
spin_lock_bh(&pmc->mca_lock);
if (pmc->mca_sfcount[MCAST_EXCLUDE]) { if (pmc->mca_sfcount[MCAST_EXCLUDE]) {
type = MLD2_BLOCK_OLD_SOURCES; type = MLD2_BLOCK_OLD_SOURCES;
dtype = MLD2_ALLOW_NEW_SOURCES; dtype = MLD2_ALLOW_NEW_SOURCES;
...@@ -1958,7 +2079,6 @@ static void mld_send_cr(struct inet6_dev *idev) ...@@ -1958,7 +2079,6 @@ static void mld_send_cr(struct inet6_dev *idev)
skb = add_grec(skb, pmc, type, 0, 0, 0); skb = add_grec(skb, pmc, type, 0, 0, 0);
pmc->mca_crcount--; pmc->mca_crcount--;
} }
spin_unlock_bh(&pmc->mca_lock);
} }
if (!skb) if (!skb)
return; return;
...@@ -2072,13 +2192,11 @@ static void mld_send_initial_cr(struct inet6_dev *idev) ...@@ -2072,13 +2192,11 @@ static void mld_send_initial_cr(struct inet6_dev *idev)
skb = NULL; skb = NULL;
for_each_mc_rtnl(idev, pmc) { for_each_mc_rtnl(idev, pmc) {
spin_lock_bh(&pmc->mca_lock);
if (pmc->mca_sfcount[MCAST_EXCLUDE]) if (pmc->mca_sfcount[MCAST_EXCLUDE])
type = MLD2_CHANGE_TO_EXCLUDE; type = MLD2_CHANGE_TO_EXCLUDE;
else else
type = MLD2_ALLOW_NEW_SOURCES; type = MLD2_ALLOW_NEW_SOURCES;
skb = add_grec(skb, pmc, type, 0, 0, 1); skb = add_grec(skb, pmc, type, 0, 0, 1);
spin_unlock_bh(&pmc->mca_lock);
} }
if (skb) if (skb)
mld_sendpack(skb); mld_sendpack(skb);
...@@ -2104,13 +2222,13 @@ static void mld_dad_work(struct work_struct *work) ...@@ -2104,13 +2222,13 @@ static void mld_dad_work(struct work_struct *work)
rtnl_lock(); rtnl_lock();
mld_send_initial_cr(idev); mld_send_initial_cr(idev);
rtnl_unlock();
if (idev->mc_dad_count) { if (idev->mc_dad_count) {
idev->mc_dad_count--; idev->mc_dad_count--;
if (idev->mc_dad_count) if (idev->mc_dad_count)
mld_dad_start_work(idev, mld_dad_start_work(idev,
unsolicited_report_interval(idev)); unsolicited_report_interval(idev));
} }
rtnl_unlock();
in6_dev_put(idev); in6_dev_put(idev);
} }
...@@ -2173,12 +2291,10 @@ static int ip6_mc_del_src(struct inet6_dev *idev, const struct in6_addr *pmca, ...@@ -2173,12 +2291,10 @@ static int ip6_mc_del_src(struct inet6_dev *idev, const struct in6_addr *pmca,
} }
if (!pmc) if (!pmc)
return -ESRCH; return -ESRCH;
spin_lock_bh(&pmc->mca_lock);
sf_markstate(pmc); sf_markstate(pmc);
if (!delta) { if (!delta) {
if (!pmc->mca_sfcount[sfmode]) { if (!pmc->mca_sfcount[sfmode]) {
spin_unlock_bh(&pmc->mca_lock);
return -EINVAL; return -EINVAL;
} }
...@@ -2206,7 +2322,6 @@ static int ip6_mc_del_src(struct inet6_dev *idev, const struct in6_addr *pmca, ...@@ -2206,7 +2322,6 @@ static int ip6_mc_del_src(struct inet6_dev *idev, const struct in6_addr *pmca,
mld_ifc_event(pmc->idev); mld_ifc_event(pmc->idev);
} else if (sf_setstate(pmc) || changerec) } else if (sf_setstate(pmc) || changerec)
mld_ifc_event(pmc->idev); mld_ifc_event(pmc->idev);
spin_unlock_bh(&pmc->mca_lock);
return err; return err;
} }
...@@ -2225,7 +2340,7 @@ static int ip6_mc_add1_src(struct ifmcaddr6 *pmc, int sfmode, ...@@ -2225,7 +2340,7 @@ static int ip6_mc_add1_src(struct ifmcaddr6 *pmc, int sfmode,
psf_prev = psf; psf_prev = psf;
} }
if (!psf) { if (!psf) {
psf = kzalloc(sizeof(*psf), GFP_ATOMIC); psf = kzalloc(sizeof(*psf), GFP_KERNEL);
if (!psf) if (!psf)
return -ENOBUFS; return -ENOBUFS;
...@@ -2304,7 +2419,7 @@ static int sf_setstate(struct ifmcaddr6 *pmc) ...@@ -2304,7 +2419,7 @@ static int sf_setstate(struct ifmcaddr6 *pmc)
&psf->sf_addr)) &psf->sf_addr))
break; break;
if (!dpsf) { if (!dpsf) {
dpsf = kmalloc(sizeof(*dpsf), GFP_ATOMIC); dpsf = kmalloc(sizeof(*dpsf), GFP_KERNEL);
if (!dpsf) if (!dpsf)
continue; continue;
*dpsf = *psf; *dpsf = *psf;
...@@ -2339,7 +2454,6 @@ static int ip6_mc_add_src(struct inet6_dev *idev, const struct in6_addr *pmca, ...@@ -2339,7 +2454,6 @@ static int ip6_mc_add_src(struct inet6_dev *idev, const struct in6_addr *pmca,
} }
if (!pmc) if (!pmc)
return -ESRCH; return -ESRCH;
spin_lock_bh(&pmc->mca_lock);
sf_markstate(pmc); sf_markstate(pmc);
isexclude = pmc->mca_sfmode == MCAST_EXCLUDE; isexclude = pmc->mca_sfmode == MCAST_EXCLUDE;
...@@ -2376,7 +2490,6 @@ static int ip6_mc_add_src(struct inet6_dev *idev, const struct in6_addr *pmca, ...@@ -2376,7 +2490,6 @@ static int ip6_mc_add_src(struct inet6_dev *idev, const struct in6_addr *pmca,
} else if (sf_setstate(pmc)) { } else if (sf_setstate(pmc)) {
mld_ifc_event(idev); mld_ifc_event(idev);
} }
spin_unlock_bh(&pmc->mca_lock);
return err; return err;
} }
...@@ -2415,7 +2528,6 @@ static void igmp6_join_group(struct ifmcaddr6 *ma) ...@@ -2415,7 +2528,6 @@ static void igmp6_join_group(struct ifmcaddr6 *ma)
delay = prandom_u32() % unsolicited_report_interval(ma->idev); delay = prandom_u32() % unsolicited_report_interval(ma->idev);
spin_lock_bh(&ma->mca_lock);
if (cancel_delayed_work(&ma->mca_work)) { if (cancel_delayed_work(&ma->mca_work)) {
refcount_dec(&ma->mca_refcnt); refcount_dec(&ma->mca_refcnt);
delay = ma->mca_work.timer.expires - jiffies; delay = ma->mca_work.timer.expires - jiffies;
...@@ -2424,7 +2536,6 @@ static void igmp6_join_group(struct ifmcaddr6 *ma) ...@@ -2424,7 +2536,6 @@ static void igmp6_join_group(struct ifmcaddr6 *ma)
if (!mod_delayed_work(mld_wq, &ma->mca_work, delay)) if (!mod_delayed_work(mld_wq, &ma->mca_work, delay))
refcount_inc(&ma->mca_refcnt); refcount_inc(&ma->mca_refcnt);
ma->mca_flags |= MAF_TIMER_RUNNING | MAF_LAST_REPORTER; 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, static int ip6_mc_leave_src(struct sock *sk, struct ipv6_mc_socklist *iml,
...@@ -2469,9 +2580,8 @@ static void mld_gq_work(struct work_struct *work) ...@@ -2469,9 +2580,8 @@ static void mld_gq_work(struct work_struct *work)
rtnl_lock(); rtnl_lock();
mld_send_report(idev, NULL); mld_send_report(idev, NULL);
rtnl_unlock();
idev->mc_gq_running = 0; idev->mc_gq_running = 0;
rtnl_unlock();
in6_dev_put(idev); in6_dev_put(idev);
} }
...@@ -2484,7 +2594,6 @@ static void mld_ifc_work(struct work_struct *work) ...@@ -2484,7 +2594,6 @@ static void mld_ifc_work(struct work_struct *work)
rtnl_lock(); rtnl_lock();
mld_send_cr(idev); mld_send_cr(idev);
rtnl_unlock();
if (idev->mc_ifc_count) { if (idev->mc_ifc_count) {
idev->mc_ifc_count--; idev->mc_ifc_count--;
...@@ -2492,6 +2601,7 @@ static void mld_ifc_work(struct work_struct *work) ...@@ -2492,6 +2601,7 @@ static void mld_ifc_work(struct work_struct *work)
mld_ifc_start_work(idev, mld_ifc_start_work(idev,
unsolicited_report_interval(idev)); unsolicited_report_interval(idev));
} }
rtnl_unlock();
in6_dev_put(idev); in6_dev_put(idev);
} }
...@@ -2514,12 +2624,10 @@ static void mld_mca_work(struct work_struct *work) ...@@ -2514,12 +2624,10 @@ static void mld_mca_work(struct work_struct *work)
igmp6_send(&ma->mca_addr, ma->idev->dev, ICMPV6_MGM_REPORT); igmp6_send(&ma->mca_addr, ma->idev->dev, ICMPV6_MGM_REPORT);
else else
mld_send_report(ma->idev, ma); mld_send_report(ma->idev, ma);
rtnl_unlock();
spin_lock_bh(&ma->mca_lock);
ma->mca_flags |= MAF_LAST_REPORTER; ma->mca_flags |= MAF_LAST_REPORTER;
ma->mca_flags &= ~MAF_TIMER_RUNNING; ma->mca_flags &= ~MAF_TIMER_RUNNING;
spin_unlock_bh(&ma->mca_lock); rtnl_unlock();
ma_put(ma); ma_put(ma);
} }
...@@ -2553,6 +2661,9 @@ void ipv6_mc_down(struct inet6_dev *idev) ...@@ -2553,6 +2661,9 @@ void ipv6_mc_down(struct inet6_dev *idev)
/* Should stop work after group drop. or we will /* Should stop work after group drop. or we will
* start work again in mld_ifc_event() * start work again in mld_ifc_event()
*/ */
synchronize_net();
mld_query_stop_work(idev);
mld_report_stop_work(idev);
mld_ifc_stop_work(idev); mld_ifc_stop_work(idev);
mld_gq_stop_work(idev); mld_gq_stop_work(idev);
mld_dad_stop_work(idev); mld_dad_stop_work(idev);
...@@ -2592,6 +2703,12 @@ void ipv6_mc_init_dev(struct inet6_dev *idev) ...@@ -2592,6 +2703,12 @@ void ipv6_mc_init_dev(struct inet6_dev *idev)
idev->mc_ifc_count = 0; idev->mc_ifc_count = 0;
INIT_DELAYED_WORK(&idev->mc_ifc_work, mld_ifc_work); INIT_DELAYED_WORK(&idev->mc_ifc_work, mld_ifc_work);
INIT_DELAYED_WORK(&idev->mc_dad_work, mld_dad_work); INIT_DELAYED_WORK(&idev->mc_dad_work, mld_dad_work);
INIT_DELAYED_WORK(&idev->mc_query_work, mld_query_work);
INIT_DELAYED_WORK(&idev->mc_report_work, mld_report_work);
skb_queue_head_init(&idev->mc_query_queue);
skb_queue_head_init(&idev->mc_report_queue);
spin_lock_init(&idev->mc_query_lock);
spin_lock_init(&idev->mc_report_lock);
ipv6_mc_reset(idev); ipv6_mc_reset(idev);
} }
...@@ -2606,6 +2723,8 @@ void ipv6_mc_destroy_dev(struct inet6_dev *idev) ...@@ -2606,6 +2723,8 @@ void ipv6_mc_destroy_dev(struct inet6_dev *idev)
/* Deactivate works */ /* Deactivate works */
ipv6_mc_down(idev); ipv6_mc_down(idev);
mld_clear_delrec(idev); mld_clear_delrec(idev);
mld_clear_query(idev);
mld_clear_report(idev);
/* Delete all-nodes address. */ /* Delete all-nodes address. */
/* We cannot call ipv6_dev_mc_dec() directly, our caller in /* We cannot call ipv6_dev_mc_dec() directly, our caller in
...@@ -2795,14 +2914,12 @@ static inline struct ip6_sf_list *igmp6_mcf_get_first(struct seq_file *seq) ...@@ -2795,14 +2914,12 @@ static inline struct ip6_sf_list *igmp6_mcf_get_first(struct seq_file *seq)
im = rcu_dereference(idev->mc_list); im = rcu_dereference(idev->mc_list);
if (likely(im)) { if (likely(im)) {
spin_lock_bh(&im->mca_lock);
psf = rcu_dereference(im->mca_sources); psf = rcu_dereference(im->mca_sources);
if (likely(psf)) { if (likely(psf)) {
state->im = im; state->im = im;
state->idev = idev; state->idev = idev;
break; break;
} }
spin_unlock_bh(&im->mca_lock);
} }
} }
return psf; return psf;
...@@ -2814,7 +2931,6 @@ static struct ip6_sf_list *igmp6_mcf_get_next(struct seq_file *seq, struct ip6_s ...@@ -2814,7 +2931,6 @@ static struct ip6_sf_list *igmp6_mcf_get_next(struct seq_file *seq, struct ip6_s
psf = rcu_dereference(psf->sf_next); psf = rcu_dereference(psf->sf_next);
while (!psf) { while (!psf) {
spin_unlock_bh(&state->im->mca_lock);
state->im = rcu_dereference(state->im->next); state->im = rcu_dereference(state->im->next);
while (!state->im) { while (!state->im) {
state->dev = next_net_device_rcu(state->dev); state->dev = next_net_device_rcu(state->dev);
...@@ -2829,7 +2945,6 @@ static struct ip6_sf_list *igmp6_mcf_get_next(struct seq_file *seq, struct ip6_s ...@@ -2829,7 +2945,6 @@ static struct ip6_sf_list *igmp6_mcf_get_next(struct seq_file *seq, struct ip6_s
} }
if (!state->im) if (!state->im)
break; break;
spin_lock_bh(&state->im->mca_lock);
psf = rcu_dereference(state->im->mca_sources); psf = rcu_dereference(state->im->mca_sources);
} }
out: out:
...@@ -2868,10 +2983,8 @@ static void igmp6_mcf_seq_stop(struct seq_file *seq, void *v) ...@@ -2868,10 +2983,8 @@ static void igmp6_mcf_seq_stop(struct seq_file *seq, void *v)
{ {
struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq); struct igmp6_mcf_iter_state *state = igmp6_mcf_seq_private(seq);
if (likely(state->im)) { if (likely(state->im))
spin_unlock_bh(&state->im->mca_lock);
state->im = NULL; state->im = NULL;
}
if (likely(state->idev)) if (likely(state->idev))
state->idev = NULL; state->idev = NULL;
...@@ -2955,6 +3068,7 @@ static int __net_init igmp6_net_init(struct net *net) ...@@ -2955,6 +3068,7 @@ static int __net_init igmp6_net_init(struct net *net)
} }
inet6_sk(net->ipv6.igmp_sk)->hop_limit = 1; inet6_sk(net->ipv6.igmp_sk)->hop_limit = 1;
net->ipv6.igmp_sk->sk_allocation = GFP_KERNEL;
err = inet_ctl_sock_create(&net->ipv6.mc_autojoin_sk, PF_INET6, err = inet_ctl_sock_create(&net->ipv6.mc_autojoin_sk, PF_INET6,
SOCK_RAW, IPPROTO_ICMPV6, net); SOCK_RAW, IPPROTO_ICMPV6, net);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment