diff options
Diffstat (limited to 'net/ipv6/mcast.c')
-rw-r--r-- | net/ipv6/mcast.c | 10 |
1 files changed, 9 insertions, 1 deletions
diff --git a/net/ipv6/mcast.c b/net/ipv6/mcast.c index 9db4581d0d79..398c982625f8 100644 --- a/net/ipv6/mcast.c +++ b/net/ipv6/mcast.c | |||
@@ -281,7 +281,7 @@ int ipv6_sock_mc_drop(struct sock *sk, int ifindex, struct in6_addr *addr) | |||
281 | } | 281 | } |
282 | write_unlock_bh(&ipv6_sk_mc_lock); | 282 | write_unlock_bh(&ipv6_sk_mc_lock); |
283 | 283 | ||
284 | return -ENOENT; | 284 | return -EADDRNOTAVAIL; |
285 | } | 285 | } |
286 | 286 | ||
287 | static struct inet6_dev *ip6_mc_find_dev(struct in6_addr *group, int ifindex) | 287 | static struct inet6_dev *ip6_mc_find_dev(struct in6_addr *group, int ifindex) |
@@ -492,6 +492,7 @@ int ip6_mc_msfilter(struct sock *sk, struct group_filter *gsf) | |||
492 | struct inet6_dev *idev; | 492 | struct inet6_dev *idev; |
493 | struct ipv6_pinfo *inet6 = inet6_sk(sk); | 493 | struct ipv6_pinfo *inet6 = inet6_sk(sk); |
494 | struct ip6_sf_socklist *newpsl, *psl; | 494 | struct ip6_sf_socklist *newpsl, *psl; |
495 | int leavegroup = 0; | ||
495 | int i, err; | 496 | int i, err; |
496 | 497 | ||
497 | group = &((struct sockaddr_in6 *)&gsf->gf_group)->sin6_addr; | 498 | group = &((struct sockaddr_in6 *)&gsf->gf_group)->sin6_addr; |
@@ -508,6 +509,11 @@ int ip6_mc_msfilter(struct sock *sk, struct group_filter *gsf) | |||
508 | return -ENODEV; | 509 | return -ENODEV; |
509 | dev = idev->dev; | 510 | dev = idev->dev; |
510 | 511 | ||
512 | if (gsf->gf_fmode == MCAST_INCLUDE && gsf->gf_numsrc == 0) { | ||
513 | leavegroup = 1; | ||
514 | goto done; | ||
515 | } | ||
516 | |||
511 | for (pmc=inet6->ipv6_mc_list; pmc; pmc=pmc->next) { | 517 | for (pmc=inet6->ipv6_mc_list; pmc; pmc=pmc->next) { |
512 | if (pmc->ifindex != gsf->gf_interface) | 518 | if (pmc->ifindex != gsf->gf_interface) |
513 | continue; | 519 | continue; |
@@ -554,6 +560,8 @@ done: | |||
554 | read_unlock_bh(&idev->lock); | 560 | read_unlock_bh(&idev->lock); |
555 | in6_dev_put(idev); | 561 | in6_dev_put(idev); |
556 | dev_put(dev); | 562 | dev_put(dev); |
563 | if (leavegroup) | ||
564 | err = ipv6_sock_mc_drop(sk, gsf->gf_interface, group); | ||
557 | return err; | 565 | return err; |
558 | } | 566 | } |
559 | 567 | ||