forked from luck/tmp_suning_uos_patched
ipv6: Add sockopt IPV6_MULTICAST_ALL analogue to IP_MULTICAST_ALL
The socket option will be enabled by default to ensure current behaviour is not changed. This is the same for the IPv4 version. A socket bound to in6addr_any and a specific port will receive all traffic on that port. Analogue to IP_MULTICAST_ALL, disable this behaviour, if one or more multicast groups were joined (using said socket) and only pass on multicast traffic from groups, which were explicitly joined via this socket. Without this option disabled a socket (system even) joined to multiple multicast groups is very hard to get right. Filtering by destination address has to take place in user space to avoid receiving multicast traffic from other multicast groups, which might have traffic on the same port. The extension of the IP_MULTICAST_ALL socketoption to just apply to ipv6, too, is not done to avoid changing the behaviour of current applications. Signed-off-by: Andre Naujoks <nautsch2@gmail.com> Acked-By: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
d03790f55a
commit
15033f0457
|
@ -274,7 +274,8 @@ struct ipv6_pinfo {
|
|||
*/
|
||||
dontfrag:1,
|
||||
autoflowlabel:1,
|
||||
autoflowlabel_set:1;
|
||||
autoflowlabel_set:1,
|
||||
mc_all:1;
|
||||
__u8 min_hopcount;
|
||||
__u8 tclass;
|
||||
__be32 rcv_flowinfo;
|
||||
|
|
|
@ -177,6 +177,7 @@ struct in6_flowlabel_req {
|
|||
#define IPV6_V6ONLY 26
|
||||
#define IPV6_JOIN_ANYCAST 27
|
||||
#define IPV6_LEAVE_ANYCAST 28
|
||||
#define IPV6_MULTICAST_ALL 29
|
||||
|
||||
/* IPV6_MTU_DISCOVER values */
|
||||
#define IPV6_PMTUDISC_DONT 0
|
||||
|
|
|
@ -209,6 +209,7 @@ static int inet6_create(struct net *net, struct socket *sock, int protocol,
|
|||
np->hop_limit = -1;
|
||||
np->mcast_hops = IPV6_DEFAULT_MCASTHOPS;
|
||||
np->mc_loop = 1;
|
||||
np->mc_all = 1;
|
||||
np->pmtudisc = IPV6_PMTUDISC_WANT;
|
||||
np->repflow = net->ipv6.sysctl.flowlabel_reflect;
|
||||
sk->sk_ipv6only = net->ipv6.sysctl.bindv6only;
|
||||
|
|
|
@ -674,6 +674,13 @@ static int do_ipv6_setsockopt(struct sock *sk, int level, int optname,
|
|||
retv = ipv6_sock_ac_drop(sk, mreq.ipv6mr_ifindex, &mreq.ipv6mr_acaddr);
|
||||
break;
|
||||
}
|
||||
case IPV6_MULTICAST_ALL:
|
||||
if (optlen < sizeof(int))
|
||||
goto e_inval;
|
||||
np->mc_all = valbool;
|
||||
retv = 0;
|
||||
break;
|
||||
|
||||
case MCAST_JOIN_GROUP:
|
||||
case MCAST_LEAVE_GROUP:
|
||||
{
|
||||
|
@ -1266,6 +1273,10 @@ static int do_ipv6_getsockopt(struct sock *sk, int level, int optname,
|
|||
val = np->mcast_oif;
|
||||
break;
|
||||
|
||||
case IPV6_MULTICAST_ALL:
|
||||
val = np->mc_all;
|
||||
break;
|
||||
|
||||
case IPV6_UNICAST_IF:
|
||||
val = (__force int)htonl((__u32) np->ucast_oif);
|
||||
break;
|
||||
|
|
|
@ -636,7 +636,7 @@ bool inet6_mc_check(struct sock *sk, const struct in6_addr *mc_addr,
|
|||
}
|
||||
if (!mc) {
|
||||
rcu_read_unlock();
|
||||
return true;
|
||||
return np->mc_all;
|
||||
}
|
||||
read_lock(&mc->sflock);
|
||||
psl = mc->sflist;
|
||||
|
|
Loading…
Reference in New Issue
Block a user