2010-02-27 22:41:45 +03:00
/*
* Bridge multicast support .
*
* Copyright ( c ) 2010 Herbert Xu < herbert @ gondor . apana . org . au >
*
* 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 .
*
*/
# include <linux/err.h>
# include <linux/if_ether.h>
# include <linux/igmp.h>
# include <linux/jhash.h>
# include <linux/kernel.h>
2010-02-27 22:41:51 +03:00
# include <linux/log2.h>
2010-02-27 22:41:45 +03:00
# include <linux/netdevice.h>
# include <linux/netfilter_bridge.h>
# include <linux/random.h>
# include <linux/rculist.h>
# include <linux/skbuff.h>
# include <linux/slab.h>
# include <linux/timer.h>
# include <net/ip.h>
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
# include <net/ipv6.h>
# include <net/mld.h>
2010-04-27 21:16:54 +04:00
# include <net/ip6_checksum.h>
2010-04-22 20:54:22 +04:00
# endif
2010-02-27 22:41:45 +03:00
# include "br_private.h"
2012-04-13 06:37:42 +04:00
static void br_multicast_start_querier ( struct net_bridge * br ) ;
2012-12-10 06:15:35 +04:00
unsigned int br_mdb_rehash_seq ;
2012-04-13 06:37:42 +04:00
2010-04-18 07:42:07 +04:00
static inline int br_ip_equal ( const struct br_ip * a , const struct br_ip * b )
{
if ( a - > proto ! = b - > proto )
return 0 ;
switch ( a - > proto ) {
case htons ( ETH_P_IP ) :
return a - > u . ip4 = = b - > u . ip4 ;
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
case htons ( ETH_P_IPV6 ) :
return ipv6_addr_equal ( & a - > u . ip6 , & b - > u . ip6 ) ;
# endif
2010-04-18 07:42:07 +04:00
}
return 0 ;
}
static inline int __br_ip4_hash ( struct net_bridge_mdb_htable * mdb , __be32 ip )
2010-02-27 22:41:45 +03:00
{
2010-04-21 06:06:52 +04:00
return jhash_1word ( mdb - > secret , ( __force u32 ) ip ) & ( mdb - > max - 1 ) ;
2010-02-27 22:41:45 +03:00
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static inline int __br_ip6_hash ( struct net_bridge_mdb_htable * mdb ,
const struct in6_addr * ip )
{
return jhash2 ( ( __force u32 * ) ip - > s6_addr32 , 4 , mdb - > secret ) & ( mdb - > max - 1 ) ;
}
# endif
2010-04-18 07:42:07 +04:00
static inline int br_ip_hash ( struct net_bridge_mdb_htable * mdb ,
struct br_ip * ip )
{
switch ( ip - > proto ) {
case htons ( ETH_P_IP ) :
return __br_ip4_hash ( mdb , ip - > u . ip4 ) ;
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
case htons ( ETH_P_IPV6 ) :
return __br_ip6_hash ( mdb , & ip - > u . ip6 ) ;
# endif
2010-04-18 07:42:07 +04:00
}
return 0 ;
2010-02-27 22:41:45 +03:00
}
static struct net_bridge_mdb_entry * __br_mdb_ip_get (
2010-04-18 07:42:07 +04:00
struct net_bridge_mdb_htable * mdb , struct br_ip * dst , int hash )
2010-02-27 22:41:45 +03:00
{
struct net_bridge_mdb_entry * mp ;
struct hlist_node * p ;
2010-03-06 00:07:39 +03:00
hlist_for_each_entry_rcu ( mp , p , & mdb - > mhash [ hash ] , hlist [ mdb - > ver ] ) {
2010-04-18 07:42:07 +04:00
if ( br_ip_equal ( & mp - > addr , dst ) )
2010-02-27 22:41:45 +03:00
return mp ;
}
return NULL ;
}
2012-12-12 02:23:08 +04:00
struct net_bridge_mdb_entry * br_mdb_ip_get ( struct net_bridge_mdb_htable * mdb ,
struct br_ip * dst )
2010-07-05 18:50:08 +04:00
{
if ( ! mdb )
return NULL ;
return __br_mdb_ip_get ( mdb , dst , br_ip_hash ( mdb , dst ) ) ;
}
2010-04-18 07:42:07 +04:00
static struct net_bridge_mdb_entry * br_mdb_ip4_get (
2010-02-27 22:41:45 +03:00
struct net_bridge_mdb_htable * mdb , __be32 dst )
{
2010-04-18 07:42:07 +04:00
struct br_ip br_dst ;
br_dst . u . ip4 = dst ;
br_dst . proto = htons ( ETH_P_IP ) ;
2010-03-16 06:38:25 +03:00
2010-07-05 18:50:08 +04:00
return br_mdb_ip_get ( mdb , & br_dst ) ;
2010-04-18 07:42:07 +04:00
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static struct net_bridge_mdb_entry * br_mdb_ip6_get (
struct net_bridge_mdb_htable * mdb , const struct in6_addr * dst )
{
struct br_ip br_dst ;
2010-03-16 06:38:25 +03:00
2011-11-21 07:39:03 +04:00
br_dst . u . ip6 = * dst ;
2010-04-22 20:54:22 +04:00
br_dst . proto = htons ( ETH_P_IPV6 ) ;
2010-07-05 18:50:08 +04:00
return br_mdb_ip_get ( mdb , & br_dst ) ;
2010-04-22 20:54:22 +04:00
}
# endif
2010-02-27 22:41:45 +03:00
struct net_bridge_mdb_entry * br_mdb_get ( struct net_bridge * br ,
struct sk_buff * skb )
{
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * mdb = rcu_dereference ( br - > mdb ) ;
2010-04-18 07:42:07 +04:00
struct br_ip ip ;
2010-07-05 18:50:08 +04:00
if ( br - > multicast_disabled )
2010-02-27 22:41:45 +03:00
return NULL ;
2010-04-18 07:42:07 +04:00
if ( BR_INPUT_SKB_CB ( skb ) - > igmp )
2010-02-27 22:41:45 +03:00
return NULL ;
2010-04-18 07:42:07 +04:00
ip . proto = skb - > protocol ;
2010-02-27 22:41:45 +03:00
switch ( skb - > protocol ) {
case htons ( ETH_P_IP ) :
2010-04-18 07:42:07 +04:00
ip . u . ip4 = ip_hdr ( skb ) - > daddr ;
break ;
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
case htons ( ETH_P_IPV6 ) :
2011-11-21 07:39:03 +04:00
ip . u . ip6 = ipv6_hdr ( skb ) - > daddr ;
2010-04-22 20:54:22 +04:00
break ;
# endif
2010-04-18 07:42:07 +04:00
default :
return NULL ;
2010-02-27 22:41:45 +03:00
}
2010-04-18 07:42:07 +04:00
return br_mdb_ip_get ( mdb , & ip ) ;
2010-02-27 22:41:45 +03:00
}
static void br_mdb_free ( struct rcu_head * head )
{
struct net_bridge_mdb_htable * mdb =
container_of ( head , struct net_bridge_mdb_htable , rcu ) ;
struct net_bridge_mdb_htable * old = mdb - > old ;
mdb - > old = NULL ;
kfree ( old - > mhash ) ;
kfree ( old ) ;
}
static int br_mdb_copy ( struct net_bridge_mdb_htable * new ,
struct net_bridge_mdb_htable * old ,
int elasticity )
{
struct net_bridge_mdb_entry * mp ;
struct hlist_node * p ;
int maxlen ;
int len ;
int i ;
for ( i = 0 ; i < old - > max ; i + + )
hlist_for_each_entry ( mp , p , & old - > mhash [ i ] , hlist [ old - > ver ] )
hlist_add_head ( & mp - > hlist [ new - > ver ] ,
2010-04-18 07:42:07 +04:00
& new - > mhash [ br_ip_hash ( new , & mp - > addr ) ] ) ;
2010-02-27 22:41:45 +03:00
if ( ! elasticity )
return 0 ;
maxlen = 0 ;
for ( i = 0 ; i < new - > max ; i + + ) {
len = 0 ;
hlist_for_each_entry ( mp , p , & new - > mhash [ i ] , hlist [ new - > ver ] )
len + + ;
if ( len > maxlen )
maxlen = len ;
}
return maxlen > elasticity ? - EINVAL : 0 ;
}
2012-12-12 02:23:08 +04:00
void br_multicast_free_pg ( struct rcu_head * head )
2010-02-27 22:41:45 +03:00
{
struct net_bridge_port_group * p =
container_of ( head , struct net_bridge_port_group , rcu ) ;
kfree ( p ) ;
}
static void br_multicast_free_group ( struct rcu_head * head )
{
struct net_bridge_mdb_entry * mp =
container_of ( head , struct net_bridge_mdb_entry , rcu ) ;
kfree ( mp ) ;
}
static void br_multicast_group_expired ( unsigned long data )
{
struct net_bridge_mdb_entry * mp = ( void * ) data ;
struct net_bridge * br = mp - > br ;
struct net_bridge_mdb_htable * mdb ;
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | | timer_pending ( & mp - > timer ) )
goto out ;
2011-02-12 12:05:42 +03:00
mp - > mglist = false ;
2010-02-27 22:41:45 +03:00
if ( mp - > ports )
goto out ;
2010-11-15 09:38:10 +03:00
mdb = mlock_dereference ( br - > mdb , br ) ;
2010-02-27 22:41:45 +03:00
hlist_del_rcu ( & mp - > hlist [ mdb - > ver ] ) ;
mdb - > size - - ;
call_rcu_bh ( & mp - > rcu , br_multicast_free_group ) ;
out :
spin_unlock ( & br - > multicast_lock ) ;
}
static void br_multicast_del_pg ( struct net_bridge * br ,
struct net_bridge_port_group * pg )
{
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * mdb ;
2010-02-27 22:41:45 +03:00
struct net_bridge_mdb_entry * mp ;
struct net_bridge_port_group * p ;
2010-11-15 09:38:10 +03:00
struct net_bridge_port_group __rcu * * pp ;
mdb = mlock_dereference ( br - > mdb , br ) ;
2010-02-27 22:41:45 +03:00
2010-04-18 07:42:07 +04:00
mp = br_mdb_ip_get ( mdb , & pg - > addr ) ;
2010-02-27 22:41:45 +03:00
if ( WARN_ON ( ! mp ) )
return ;
2010-11-15 09:38:10 +03:00
for ( pp = & mp - > ports ;
( p = mlock_dereference ( * pp , br ) ) ! = NULL ;
pp = & p - > next ) {
2010-02-27 22:41:45 +03:00
if ( p ! = pg )
continue ;
2010-04-27 19:01:06 +04:00
rcu_assign_pointer ( * pp , p - > next ) ;
2010-02-27 22:41:45 +03:00
hlist_del_init ( & p - > mglist ) ;
del_timer ( & p - > timer ) ;
call_rcu_bh ( & p - > rcu , br_multicast_free_pg ) ;
2011-02-12 12:05:42 +03:00
if ( ! mp - > ports & & ! mp - > mglist & &
2010-02-27 22:41:45 +03:00
netif_running ( br - > dev ) )
mod_timer ( & mp - > timer , jiffies ) ;
return ;
}
WARN_ON ( 1 ) ;
}
static void br_multicast_port_group_expired ( unsigned long data )
{
struct net_bridge_port_group * pg = ( void * ) data ;
struct net_bridge * br = pg - > port - > br ;
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | | timer_pending ( & pg - > timer ) | |
2012-12-15 02:09:51 +04:00
hlist_unhashed ( & pg - > mglist ) | | pg - > state & MDB_PERMANENT )
2010-02-27 22:41:45 +03:00
goto out ;
br_multicast_del_pg ( br , pg ) ;
out :
spin_unlock ( & br - > multicast_lock ) ;
}
2010-11-15 09:38:10 +03:00
static int br_mdb_rehash ( struct net_bridge_mdb_htable __rcu * * mdbp , int max ,
2010-02-27 22:41:45 +03:00
int elasticity )
{
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * old = rcu_dereference_protected ( * mdbp , 1 ) ;
2010-02-27 22:41:45 +03:00
struct net_bridge_mdb_htable * mdb ;
int err ;
mdb = kmalloc ( sizeof ( * mdb ) , GFP_ATOMIC ) ;
if ( ! mdb )
return - ENOMEM ;
mdb - > max = max ;
mdb - > old = old ;
mdb - > mhash = kzalloc ( max * sizeof ( * mdb - > mhash ) , GFP_ATOMIC ) ;
if ( ! mdb - > mhash ) {
kfree ( mdb ) ;
return - ENOMEM ;
}
mdb - > size = old ? old - > size : 0 ;
mdb - > ver = old ? old - > ver ^ 1 : 0 ;
if ( ! old | | elasticity )
get_random_bytes ( & mdb - > secret , sizeof ( mdb - > secret ) ) ;
else
mdb - > secret = old - > secret ;
if ( ! old )
goto out ;
err = br_mdb_copy ( mdb , old , elasticity ) ;
if ( err ) {
kfree ( mdb - > mhash ) ;
kfree ( mdb ) ;
return err ;
}
2012-12-10 06:15:35 +04:00
br_mdb_rehash_seq + + ;
2010-02-27 22:41:45 +03:00
call_rcu_bh ( & mdb - > rcu , br_mdb_free ) ;
out :
rcu_assign_pointer ( * mdbp , mdb ) ;
return 0 ;
}
2010-04-18 07:42:07 +04:00
static struct sk_buff * br_ip4_multicast_alloc_query ( struct net_bridge * br ,
__be32 group )
2010-02-27 22:41:45 +03:00
{
struct sk_buff * skb ;
struct igmphdr * ih ;
struct ethhdr * eth ;
struct iphdr * iph ;
skb = netdev_alloc_skb_ip_align ( br - > dev , sizeof ( * eth ) + sizeof ( * iph ) +
sizeof ( * ih ) + 4 ) ;
if ( ! skb )
goto out ;
skb - > protocol = htons ( ETH_P_IP ) ;
skb_reset_mac_header ( skb ) ;
eth = eth_hdr ( skb ) ;
memcpy ( eth - > h_source , br - > dev - > dev_addr , 6 ) ;
eth - > h_dest [ 0 ] = 1 ;
eth - > h_dest [ 1 ] = 0 ;
eth - > h_dest [ 2 ] = 0x5e ;
eth - > h_dest [ 3 ] = 0 ;
eth - > h_dest [ 4 ] = 0 ;
eth - > h_dest [ 5 ] = 1 ;
eth - > h_proto = htons ( ETH_P_IP ) ;
skb_put ( skb , sizeof ( * eth ) ) ;
skb_set_network_header ( skb , skb - > len ) ;
iph = ip_hdr ( skb ) ;
iph - > version = 4 ;
iph - > ihl = 6 ;
iph - > tos = 0xc0 ;
iph - > tot_len = htons ( sizeof ( * iph ) + sizeof ( * ih ) + 4 ) ;
iph - > id = 0 ;
iph - > frag_off = htons ( IP_DF ) ;
iph - > ttl = 1 ;
iph - > protocol = IPPROTO_IGMP ;
iph - > saddr = 0 ;
iph - > daddr = htonl ( INADDR_ALLHOSTS_GROUP ) ;
( ( u8 * ) & iph [ 1 ] ) [ 0 ] = IPOPT_RA ;
( ( u8 * ) & iph [ 1 ] ) [ 1 ] = 4 ;
( ( u8 * ) & iph [ 1 ] ) [ 2 ] = 0 ;
( ( u8 * ) & iph [ 1 ] ) [ 3 ] = 0 ;
ip_send_check ( iph ) ;
skb_put ( skb , 24 ) ;
skb_set_transport_header ( skb , skb - > len ) ;
ih = igmp_hdr ( skb ) ;
ih - > type = IGMP_HOST_MEMBERSHIP_QUERY ;
ih - > code = ( group ? br - > multicast_last_member_interval :
br - > multicast_query_response_interval ) /
( HZ / IGMP_TIMER_SCALE ) ;
ih - > group = group ;
ih - > csum = 0 ;
ih - > csum = ip_compute_csum ( ( void * ) ih , sizeof ( struct igmphdr ) ) ;
skb_put ( skb , sizeof ( * ih ) ) ;
__skb_pull ( skb , sizeof ( * eth ) ) ;
out :
return skb ;
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static struct sk_buff * br_ip6_multicast_alloc_query ( struct net_bridge * br ,
2011-04-22 08:53:02 +04:00
const struct in6_addr * group )
2010-04-22 20:54:22 +04:00
{
struct sk_buff * skb ;
struct ipv6hdr * ip6h ;
struct mld_msg * mldq ;
struct ethhdr * eth ;
u8 * hopopt ;
unsigned long interval ;
skb = netdev_alloc_skb_ip_align ( br - > dev , sizeof ( * eth ) + sizeof ( * ip6h ) +
8 + sizeof ( * mldq ) ) ;
if ( ! skb )
goto out ;
skb - > protocol = htons ( ETH_P_IPV6 ) ;
/* Ethernet header */
skb_reset_mac_header ( skb ) ;
eth = eth_hdr ( skb ) ;
memcpy ( eth - > h_source , br - > dev - > dev_addr , 6 ) ;
eth - > h_proto = htons ( ETH_P_IPV6 ) ;
skb_put ( skb , sizeof ( * eth ) ) ;
/* IPv6 header + HbH option */
skb_set_network_header ( skb , skb - > len ) ;
ip6h = ipv6_hdr ( skb ) ;
* ( __force __be32 * ) ip6h = htonl ( 0x60000000 ) ;
2010-12-14 11:42:16 +03:00
ip6h - > payload_len = htons ( 8 + sizeof ( * mldq ) ) ;
2010-04-22 20:54:22 +04:00
ip6h - > nexthdr = IPPROTO_HOPOPTS ;
ip6h - > hop_limit = 1 ;
2011-03-22 14:40:32 +03:00
ipv6_addr_set ( & ip6h - > daddr , htonl ( 0xff020000 ) , 0 , 0 , htonl ( 1 ) ) ;
2012-03-05 08:52:44 +04:00
if ( ipv6_dev_get_saddr ( dev_net ( br - > dev ) , br - > dev , & ip6h - > daddr , 0 ,
& ip6h - > saddr ) ) {
kfree_skb ( skb ) ;
return NULL ;
}
2011-02-17 11:17:51 +03:00
ipv6_eth_mc_map ( & ip6h - > daddr , eth - > h_dest ) ;
2010-04-22 20:54:22 +04:00
hopopt = ( u8 * ) ( ip6h + 1 ) ;
hopopt [ 0 ] = IPPROTO_ICMPV6 ; /* next hdr */
hopopt [ 1 ] = 0 ; /* length of HbH */
hopopt [ 2 ] = IPV6_TLV_ROUTERALERT ; /* Router Alert */
hopopt [ 3 ] = 2 ; /* Length of RA Option */
hopopt [ 4 ] = 0 ; /* Type = 0x0000 (MLD) */
hopopt [ 5 ] = 0 ;
2012-05-17 10:00:25 +04:00
hopopt [ 6 ] = IPV6_TLV_PAD1 ; /* Pad1 */
hopopt [ 7 ] = IPV6_TLV_PAD1 ; /* Pad1 */
2010-04-22 20:54:22 +04:00
skb_put ( skb , sizeof ( * ip6h ) + 8 ) ;
/* ICMPv6 */
skb_set_transport_header ( skb , skb - > len ) ;
mldq = ( struct mld_msg * ) icmp6_hdr ( skb ) ;
interval = ipv6_addr_any ( group ) ? br - > multicast_last_member_interval :
br - > multicast_query_response_interval ;
mldq - > mld_type = ICMPV6_MGM_QUERY ;
mldq - > mld_code = 0 ;
mldq - > mld_cksum = 0 ;
mldq - > mld_maxdelay = htons ( ( u16 ) jiffies_to_msecs ( interval ) ) ;
mldq - > mld_reserved = 0 ;
2011-11-21 07:39:03 +04:00
mldq - > mld_mca = * group ;
2010-04-22 20:54:22 +04:00
/* checksum */
mldq - > mld_cksum = csum_ipv6_magic ( & ip6h - > saddr , & ip6h - > daddr ,
sizeof ( * mldq ) , IPPROTO_ICMPV6 ,
csum_partial ( mldq ,
sizeof ( * mldq ) , 0 ) ) ;
skb_put ( skb , sizeof ( * mldq ) ) ;
__skb_pull ( skb , sizeof ( * eth ) ) ;
out :
return skb ;
}
# endif
2010-04-18 07:42:07 +04:00
static struct sk_buff * br_multicast_alloc_query ( struct net_bridge * br ,
struct br_ip * addr )
{
switch ( addr - > proto ) {
case htons ( ETH_P_IP ) :
return br_ip4_multicast_alloc_query ( br , addr - > u . ip4 ) ;
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
case htons ( ETH_P_IPV6 ) :
return br_ip6_multicast_alloc_query ( br , & addr - > u . ip6 ) ;
# endif
2010-04-18 07:42:07 +04:00
}
return NULL ;
}
2010-02-27 22:41:45 +03:00
static struct net_bridge_mdb_entry * br_multicast_get_group (
2010-04-18 07:42:07 +04:00
struct net_bridge * br , struct net_bridge_port * port ,
struct br_ip * group , int hash )
2010-02-27 22:41:45 +03:00
{
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * mdb ;
2010-02-27 22:41:45 +03:00
struct net_bridge_mdb_entry * mp ;
struct hlist_node * p ;
2012-04-15 09:58:06 +04:00
unsigned int count = 0 ;
unsigned int max ;
2010-02-27 22:41:45 +03:00
int elasticity ;
int err ;
2010-11-15 09:38:10 +03:00
mdb = rcu_dereference_protected ( br - > mdb , 1 ) ;
2010-02-27 22:41:45 +03:00
hlist_for_each_entry ( mp , p , & mdb - > mhash [ hash ] , hlist [ mdb - > ver ] ) {
count + + ;
2010-04-18 07:42:07 +04:00
if ( unlikely ( br_ip_equal ( group , & mp - > addr ) ) )
2010-02-27 22:41:45 +03:00
return mp ;
}
elasticity = 0 ;
max = mdb - > max ;
if ( unlikely ( count > br - > hash_elasticity & & count ) ) {
if ( net_ratelimit ( ) )
2010-05-10 13:31:09 +04:00
br_info ( br , " Multicast hash table "
" chain limit reached: %s \n " ,
port ? port - > dev - > name : br - > dev - > name ) ;
2010-02-27 22:41:45 +03:00
elasticity = br - > hash_elasticity ;
}
if ( mdb - > size > = max ) {
max * = 2 ;
2012-07-11 02:29:19 +04:00
if ( unlikely ( max > br - > hash_max ) ) {
br_warn ( br , " Multicast hash table maximum of %d "
" reached, disabling snooping: %s \n " ,
br - > hash_max ,
port ? port - > dev - > name : br - > dev - > name ) ;
2010-02-27 22:41:45 +03:00
err = - E2BIG ;
disable :
br - > multicast_disabled = 1 ;
goto err ;
}
}
if ( max > mdb - > max | | elasticity ) {
if ( mdb - > old ) {
if ( net_ratelimit ( ) )
2010-05-10 13:31:09 +04:00
br_info ( br , " Multicast hash table "
" on fire: %s \n " ,
port ? port - > dev - > name : br - > dev - > name ) ;
2010-02-27 22:41:45 +03:00
err = - EEXIST ;
goto err ;
}
err = br_mdb_rehash ( & br - > mdb , max , elasticity ) ;
if ( err ) {
2010-05-10 13:31:09 +04:00
br_warn ( br , " Cannot rehash multicast "
" hash table, disabling snooping: %s, %d, %d \n " ,
port ? port - > dev - > name : br - > dev - > name ,
mdb - > size , err ) ;
2010-02-27 22:41:45 +03:00
goto disable ;
}
err = - EAGAIN ;
goto err ;
}
return NULL ;
err :
mp = ERR_PTR ( err ) ;
return mp ;
}
2012-12-12 02:23:08 +04:00
struct net_bridge_mdb_entry * br_multicast_new_group ( struct net_bridge * br ,
struct net_bridge_port * port , struct br_ip * group )
2010-02-27 22:41:45 +03:00
{
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * mdb ;
2010-02-27 22:41:45 +03:00
struct net_bridge_mdb_entry * mp ;
int hash ;
2010-12-10 06:18:04 +03:00
int err ;
2010-02-27 22:41:45 +03:00
2010-11-15 09:38:10 +03:00
mdb = rcu_dereference_protected ( br - > mdb , 1 ) ;
2010-02-27 22:41:45 +03:00
if ( ! mdb ) {
2010-12-10 06:18:04 +03:00
err = br_mdb_rehash ( & br - > mdb , BR_HASH_SIZE , 0 ) ;
if ( err )
return ERR_PTR ( err ) ;
2010-02-27 22:41:45 +03:00
goto rehash ;
}
hash = br_ip_hash ( mdb , group ) ;
mp = br_multicast_get_group ( br , port , group , hash ) ;
switch ( PTR_ERR ( mp ) ) {
case 0 :
break ;
case - EAGAIN :
rehash :
2010-11-15 09:38:10 +03:00
mdb = rcu_dereference_protected ( br - > mdb , 1 ) ;
2010-02-27 22:41:45 +03:00
hash = br_ip_hash ( mdb , group ) ;
break ;
default :
goto out ;
}
mp = kzalloc ( sizeof ( * mp ) , GFP_ATOMIC ) ;
if ( unlikely ( ! mp ) )
2010-12-10 06:18:04 +03:00
return ERR_PTR ( - ENOMEM ) ;
2010-02-27 22:41:45 +03:00
mp - > br = br ;
2010-04-18 07:42:07 +04:00
mp - > addr = * group ;
2010-02-27 22:41:45 +03:00
setup_timer ( & mp - > timer , br_multicast_group_expired ,
( unsigned long ) mp ) ;
hlist_add_head_rcu ( & mp - > hlist [ mdb - > ver ] , & mdb - > mhash [ hash ] ) ;
mdb - > size + + ;
out :
return mp ;
}
2012-12-12 02:23:08 +04:00
struct net_bridge_port_group * br_multicast_new_port_group (
struct net_bridge_port * port ,
struct br_ip * group ,
2012-12-15 02:09:51 +04:00
struct net_bridge_port_group __rcu * next ,
unsigned char state )
2012-12-12 02:23:08 +04:00
{
struct net_bridge_port_group * p ;
p = kzalloc ( sizeof ( * p ) , GFP_ATOMIC ) ;
if ( unlikely ( ! p ) )
return NULL ;
p - > addr = * group ;
p - > port = port ;
2012-12-15 02:09:51 +04:00
p - > state = state ;
2012-12-13 10:51:28 +04:00
rcu_assign_pointer ( p - > next , next ) ;
2012-12-12 02:23:08 +04:00
hlist_add_head ( & p - > mglist , & port - > mglist ) ;
setup_timer ( & p - > timer , br_multicast_port_group_expired ,
( unsigned long ) p ) ;
return p ;
}
2010-02-27 22:41:45 +03:00
static int br_multicast_add_group ( struct net_bridge * br ,
2010-04-18 07:42:07 +04:00
struct net_bridge_port * port ,
struct br_ip * group )
2010-02-27 22:41:45 +03:00
{
struct net_bridge_mdb_entry * mp ;
struct net_bridge_port_group * p ;
2010-11-15 09:38:10 +03:00
struct net_bridge_port_group __rcu * * pp ;
2010-02-27 22:41:45 +03:00
unsigned long now = jiffies ;
int err ;
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | |
( port & & port - > state = = BR_STATE_DISABLED ) )
goto out ;
mp = br_multicast_new_group ( br , port , group ) ;
err = PTR_ERR ( mp ) ;
2010-12-10 06:18:04 +03:00
if ( IS_ERR ( mp ) )
2010-02-27 22:41:45 +03:00
goto err ;
if ( ! port ) {
2011-02-12 12:05:42 +03:00
mp - > mglist = true ;
2010-02-27 22:41:45 +03:00
mod_timer ( & mp - > timer , now + br - > multicast_membership_interval ) ;
goto out ;
}
2010-11-15 09:38:10 +03:00
for ( pp = & mp - > ports ;
( p = mlock_dereference ( * pp , br ) ) ! = NULL ;
pp = & p - > next ) {
2010-02-27 22:41:45 +03:00
if ( p - > port = = port )
goto found ;
if ( ( unsigned long ) p - > port < ( unsigned long ) port )
break ;
}
2012-12-15 02:09:51 +04:00
p = br_multicast_new_port_group ( port , group , * pp , MDB_TEMPORARY ) ;
2010-02-27 22:41:45 +03:00
if ( unlikely ( ! p ) )
goto err ;
rcu_assign_pointer ( * pp , p ) ;
2012-12-12 02:23:07 +04:00
br_mdb_notify ( br - > dev , port , group , RTM_NEWMDB ) ;
2010-02-27 22:41:45 +03:00
found :
mod_timer ( & p - > timer , now + br - > multicast_membership_interval ) ;
out :
err = 0 ;
err :
spin_unlock ( & br - > multicast_lock ) ;
return err ;
}
2010-04-18 07:42:07 +04:00
static int br_ip4_multicast_add_group ( struct net_bridge * br ,
struct net_bridge_port * port ,
__be32 group )
{
struct br_ip br_group ;
if ( ipv4_is_local_multicast ( group ) )
return 0 ;
br_group . u . ip4 = group ;
br_group . proto = htons ( ETH_P_IP ) ;
return br_multicast_add_group ( br , port , & br_group ) ;
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static int br_ip6_multicast_add_group ( struct net_bridge * br ,
struct net_bridge_port * port ,
const struct in6_addr * group )
{
struct br_ip br_group ;
2011-02-15 16:19:21 +03:00
if ( ! ipv6_is_transient_multicast ( group ) )
2010-04-22 20:54:22 +04:00
return 0 ;
2011-11-21 07:39:03 +04:00
br_group . u . ip6 = * group ;
2011-02-15 16:19:17 +03:00
br_group . proto = htons ( ETH_P_IPV6 ) ;
2010-04-22 20:54:22 +04:00
return br_multicast_add_group ( br , port , & br_group ) ;
}
# endif
2010-02-27 22:41:45 +03:00
static void br_multicast_router_expired ( unsigned long data )
{
struct net_bridge_port * port = ( void * ) data ;
struct net_bridge * br = port - > br ;
spin_lock ( & br - > multicast_lock ) ;
if ( port - > multicast_router ! = 1 | |
timer_pending ( & port - > multicast_router_timer ) | |
hlist_unhashed ( & port - > rlist ) )
goto out ;
hlist_del_init_rcu ( & port - > rlist ) ;
out :
spin_unlock ( & br - > multicast_lock ) ;
}
static void br_multicast_local_router_expired ( unsigned long data )
{
}
2012-04-13 06:37:42 +04:00
static void br_multicast_querier_expired ( unsigned long data )
{
2012-04-30 04:22:56 +04:00
struct net_bridge * br = ( void * ) data ;
2012-04-13 06:37:42 +04:00
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | | br - > multicast_disabled )
goto out ;
br_multicast_start_querier ( br ) ;
out :
spin_unlock ( & br - > multicast_lock ) ;
}
2010-04-18 07:42:07 +04:00
static void __br_multicast_send_query ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct br_ip * ip )
2010-02-27 22:41:45 +03:00
{
struct sk_buff * skb ;
2010-04-18 07:42:07 +04:00
skb = br_multicast_alloc_query ( br , ip ) ;
2010-02-27 22:41:45 +03:00
if ( ! skb )
2010-04-18 07:42:07 +04:00
return ;
2010-02-27 22:41:45 +03:00
if ( port ) {
__skb_push ( skb , sizeof ( struct ethhdr ) ) ;
skb - > dev = port - > dev ;
2010-03-23 06:07:21 +03:00
NF_HOOK ( NFPROTO_BRIDGE , NF_BR_LOCAL_OUT , skb , NULL , skb - > dev ,
2010-02-27 22:41:45 +03:00
dev_queue_xmit ) ;
} else
netif_rx ( skb ) ;
2010-04-18 07:42:07 +04:00
}
static void br_multicast_send_query ( struct net_bridge * br ,
struct net_bridge_port * port , u32 sent )
{
unsigned long time ;
struct br_ip br_group ;
if ( ! netif_running ( br - > dev ) | | br - > multicast_disabled | |
2012-04-13 06:37:42 +04:00
! br - > multicast_querier | |
2010-04-18 07:42:07 +04:00
timer_pending ( & br - > multicast_querier_timer ) )
return ;
2010-04-22 20:54:22 +04:00
memset ( & br_group . u , 0 , sizeof ( br_group . u ) ) ;
2010-04-18 07:42:07 +04:00
br_group . proto = htons ( ETH_P_IP ) ;
2010-04-22 20:54:22 +04:00
__br_multicast_send_query ( br , port , & br_group ) ;
2010-04-18 07:42:07 +04:00
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
br_group . proto = htons ( ETH_P_IPV6 ) ;
2010-04-18 07:42:07 +04:00
__br_multicast_send_query ( br , port , & br_group ) ;
2010-04-22 20:54:22 +04:00
# endif
2010-02-27 22:41:45 +03:00
time = jiffies ;
time + = sent < br - > multicast_startup_query_count ?
br - > multicast_startup_query_interval :
br - > multicast_query_interval ;
mod_timer ( port ? & port - > multicast_query_timer :
& br - > multicast_query_timer , time ) ;
}
static void br_multicast_port_query_expired ( unsigned long data )
{
struct net_bridge_port * port = ( void * ) data ;
struct net_bridge * br = port - > br ;
spin_lock ( & br - > multicast_lock ) ;
2010-03-06 04:14:09 +03:00
if ( port - > state = = BR_STATE_DISABLED | |
port - > state = = BR_STATE_BLOCKING )
2010-02-27 22:41:45 +03:00
goto out ;
if ( port - > multicast_startup_queries_sent <
br - > multicast_startup_query_count )
port - > multicast_startup_queries_sent + + ;
br_multicast_send_query ( port - > br , port ,
port - > multicast_startup_queries_sent ) ;
out :
spin_unlock ( & br - > multicast_lock ) ;
}
void br_multicast_add_port ( struct net_bridge_port * port )
{
port - > multicast_router = 1 ;
setup_timer ( & port - > multicast_router_timer , br_multicast_router_expired ,
( unsigned long ) port ) ;
setup_timer ( & port - > multicast_query_timer ,
br_multicast_port_query_expired , ( unsigned long ) port ) ;
}
void br_multicast_del_port ( struct net_bridge_port * port )
{
del_timer_sync ( & port - > multicast_router_timer ) ;
}
2010-02-27 22:41:50 +03:00
static void __br_multicast_enable_port ( struct net_bridge_port * port )
{
port - > multicast_startup_queries_sent = 0 ;
if ( try_to_del_timer_sync ( & port - > multicast_query_timer ) > = 0 | |
del_timer ( & port - > multicast_query_timer ) )
mod_timer ( & port - > multicast_query_timer , jiffies ) ;
}
2010-02-27 22:41:45 +03:00
void br_multicast_enable_port ( struct net_bridge_port * port )
{
struct net_bridge * br = port - > br ;
spin_lock ( & br - > multicast_lock ) ;
if ( br - > multicast_disabled | | ! netif_running ( br - > dev ) )
goto out ;
2010-02-27 22:41:50 +03:00
__br_multicast_enable_port ( port ) ;
2010-02-27 22:41:45 +03:00
out :
spin_unlock ( & br - > multicast_lock ) ;
}
void br_multicast_disable_port ( struct net_bridge_port * port )
{
struct net_bridge * br = port - > br ;
struct net_bridge_port_group * pg ;
struct hlist_node * p , * n ;
spin_lock ( & br - > multicast_lock ) ;
hlist_for_each_entry_safe ( pg , p , n , & port - > mglist , mglist )
br_multicast_del_pg ( br , pg ) ;
if ( ! hlist_unhashed ( & port - > rlist ) )
hlist_del_init_rcu ( & port - > rlist ) ;
del_timer ( & port - > multicast_router_timer ) ;
del_timer ( & port - > multicast_query_timer ) ;
spin_unlock ( & br - > multicast_lock ) ;
}
2010-04-18 07:42:07 +04:00
static int br_ip4_multicast_igmp3_report ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
2010-02-27 22:41:45 +03:00
{
struct igmpv3_report * ih ;
struct igmpv3_grec * grec ;
int i ;
int len ;
int num ;
int type ;
int err = 0 ;
__be32 group ;
if ( ! pskb_may_pull ( skb , sizeof ( * ih ) ) )
return - EINVAL ;
ih = igmpv3_report_hdr ( skb ) ;
num = ntohs ( ih - > ngrec ) ;
len = sizeof ( * ih ) ;
for ( i = 0 ; i < num ; i + + ) {
len + = sizeof ( * grec ) ;
if ( ! pskb_may_pull ( skb , len ) )
return - EINVAL ;
2010-04-08 08:20:47 +04:00
grec = ( void * ) ( skb - > data + len - sizeof ( * grec ) ) ;
2010-02-27 22:41:45 +03:00
group = grec - > grec_mca ;
type = grec - > grec_type ;
2010-04-20 07:20:05 +04:00
len + = ntohs ( grec - > grec_nsrcs ) * 4 ;
2010-02-27 22:41:45 +03:00
if ( ! pskb_may_pull ( skb , len ) )
return - EINVAL ;
/* We treat this as an IGMPv2 report for now. */
switch ( type ) {
case IGMPV3_MODE_IS_INCLUDE :
case IGMPV3_MODE_IS_EXCLUDE :
case IGMPV3_CHANGE_TO_INCLUDE :
case IGMPV3_CHANGE_TO_EXCLUDE :
case IGMPV3_ALLOW_NEW_SOURCES :
case IGMPV3_BLOCK_OLD_SOURCES :
break ;
default :
continue ;
}
2010-04-18 07:42:07 +04:00
err = br_ip4_multicast_add_group ( br , port , group ) ;
2010-02-27 22:41:45 +03:00
if ( err )
break ;
}
return err ;
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static int br_ip6_multicast_mld2_report ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
{
struct icmp6hdr * icmp6h ;
struct mld2_grec * grec ;
int i ;
int len ;
int num ;
int err = 0 ;
if ( ! pskb_may_pull ( skb , sizeof ( * icmp6h ) ) )
return - EINVAL ;
icmp6h = icmp6_hdr ( skb ) ;
num = ntohs ( icmp6h - > icmp6_dataun . un_data16 [ 1 ] ) ;
len = sizeof ( * icmp6h ) ;
for ( i = 0 ; i < num ; i + + ) {
__be16 * nsrcs , _nsrcs ;
nsrcs = skb_header_pointer ( skb ,
len + offsetof ( struct mld2_grec ,
2011-02-15 16:19:18 +03:00
grec_nsrcs ) ,
2010-04-22 20:54:22 +04:00
sizeof ( _nsrcs ) , & _nsrcs ) ;
if ( ! nsrcs )
return - EINVAL ;
if ( ! pskb_may_pull ( skb ,
len + sizeof ( * grec ) +
2011-02-15 16:19:19 +03:00
sizeof ( struct in6_addr ) * ntohs ( * nsrcs ) ) )
2010-04-22 20:54:22 +04:00
return - EINVAL ;
grec = ( struct mld2_grec * ) ( skb - > data + len ) ;
2011-02-15 16:19:19 +03:00
len + = sizeof ( * grec ) +
sizeof ( struct in6_addr ) * ntohs ( * nsrcs ) ;
2010-04-22 20:54:22 +04:00
/* We treat these as MLDv1 reports for now. */
switch ( grec - > grec_type ) {
case MLD2_MODE_IS_INCLUDE :
case MLD2_MODE_IS_EXCLUDE :
case MLD2_CHANGE_TO_INCLUDE :
case MLD2_CHANGE_TO_EXCLUDE :
case MLD2_ALLOW_NEW_SOURCES :
case MLD2_BLOCK_OLD_SOURCES :
break ;
default :
continue ;
}
err = br_ip6_multicast_add_group ( br , port , & grec - > grec_mca ) ;
if ( ! err )
break ;
}
return err ;
}
# endif
2010-04-27 19:01:04 +04:00
/*
* Add port to rotuer_list
* list is maintained ordered by pointer value
* and locked by br - > multicast_lock and RCU
*/
2010-02-27 22:41:49 +03:00
static void br_multicast_add_router ( struct net_bridge * br ,
struct net_bridge_port * port )
{
2010-04-27 11:13:11 +04:00
struct net_bridge_port * p ;
2010-04-27 19:01:04 +04:00
struct hlist_node * n , * slot = NULL ;
2010-04-27 11:13:11 +04:00
2010-04-28 03:49:58 +04:00
hlist_for_each_entry ( p , n , & br - > router_list , rlist ) {
2010-04-27 19:01:04 +04:00
if ( ( unsigned long ) port > = ( unsigned long ) p )
break ;
slot = n ;
2010-04-27 11:13:11 +04:00
}
2010-04-27 19:01:04 +04:00
if ( slot )
hlist_add_after_rcu ( slot , & port - > rlist ) ;
2010-04-27 11:13:11 +04:00
else
hlist_add_head_rcu ( & port - > rlist , & br - > router_list ) ;
2010-02-27 22:41:49 +03:00
}
2010-02-27 22:41:45 +03:00
static void br_multicast_mark_router ( struct net_bridge * br ,
struct net_bridge_port * port )
{
unsigned long now = jiffies ;
if ( ! port ) {
if ( br - > multicast_router = = 1 )
mod_timer ( & br - > multicast_router_timer ,
now + br - > multicast_querier_interval ) ;
return ;
}
if ( port - > multicast_router ! = 1 )
return ;
if ( ! hlist_unhashed ( & port - > rlist ) )
goto timer ;
2010-02-27 22:41:49 +03:00
br_multicast_add_router ( br , port ) ;
2010-02-27 22:41:45 +03:00
timer :
mod_timer ( & port - > multicast_router_timer ,
now + br - > multicast_querier_interval ) ;
}
static void br_multicast_query_received ( struct net_bridge * br ,
struct net_bridge_port * port ,
2010-04-18 07:42:07 +04:00
int saddr )
2010-02-27 22:41:45 +03:00
{
if ( saddr )
mod_timer ( & br - > multicast_querier_timer ,
jiffies + br - > multicast_querier_interval ) ;
else if ( timer_pending ( & br - > multicast_querier_timer ) )
return ;
br_multicast_mark_router ( br , port ) ;
}
2010-04-18 07:42:07 +04:00
static int br_ip4_multicast_query ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
2010-02-27 22:41:45 +03:00
{
2011-04-22 08:53:02 +04:00
const struct iphdr * iph = ip_hdr ( skb ) ;
2010-02-27 22:41:45 +03:00
struct igmphdr * ih = igmp_hdr ( skb ) ;
struct net_bridge_mdb_entry * mp ;
struct igmpv3_query * ih3 ;
struct net_bridge_port_group * p ;
2010-11-15 09:38:10 +03:00
struct net_bridge_port_group __rcu * * pp ;
2010-02-27 22:41:45 +03:00
unsigned long max_delay ;
unsigned long now = jiffies ;
__be32 group ;
2010-03-13 23:27:21 +03:00
int err = 0 ;
2010-02-27 22:41:45 +03:00
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | |
( port & & port - > state = = BR_STATE_DISABLED ) )
goto out ;
2010-04-18 07:42:07 +04:00
br_multicast_query_received ( br , port , ! ! iph - > saddr ) ;
2010-02-27 22:41:45 +03:00
group = ih - > group ;
if ( skb - > len = = sizeof ( * ih ) ) {
max_delay = ih - > code * ( HZ / IGMP_TIMER_SCALE ) ;
if ( ! max_delay ) {
max_delay = 10 * HZ ;
group = 0 ;
}
} else {
2010-03-13 23:27:21 +03:00
if ( ! pskb_may_pull ( skb , sizeof ( struct igmpv3_query ) ) ) {
err = - EINVAL ;
goto out ;
}
2010-02-27 22:41:45 +03:00
ih3 = igmpv3_query_hdr ( skb ) ;
if ( ih3 - > nsrcs )
2010-03-13 23:27:21 +03:00
goto out ;
2010-02-27 22:41:45 +03:00
2010-03-15 22:27:00 +03:00
max_delay = ih3 - > code ?
IGMPV3_MRC ( ih3 - > code ) * ( HZ / IGMP_TIMER_SCALE ) : 1 ;
2010-02-27 22:41:45 +03:00
}
if ( ! group )
goto out ;
2010-11-15 09:38:10 +03:00
mp = br_mdb_ip4_get ( mlock_dereference ( br - > mdb , br ) , group ) ;
2010-02-27 22:41:45 +03:00
if ( ! mp )
goto out ;
max_delay * = br - > multicast_last_member_count ;
2011-02-12 12:05:42 +03:00
if ( mp - > mglist & &
2010-02-27 22:41:45 +03:00
( timer_pending ( & mp - > timer ) ?
time_after ( mp - > timer . expires , now + max_delay ) :
try_to_del_timer_sync ( & mp - > timer ) > = 0 ) )
mod_timer ( & mp - > timer , now + max_delay ) ;
2010-11-15 09:38:10 +03:00
for ( pp = & mp - > ports ;
( p = mlock_dereference ( * pp , br ) ) ! = NULL ;
pp = & p - > next ) {
2010-02-27 22:41:45 +03:00
if ( timer_pending ( & p - > timer ) ?
time_after ( p - > timer . expires , now + max_delay ) :
try_to_del_timer_sync ( & p - > timer ) > = 0 )
2011-02-11 15:42:07 +03:00
mod_timer ( & p - > timer , now + max_delay ) ;
2010-02-27 22:41:45 +03:00
}
out :
spin_unlock ( & br - > multicast_lock ) ;
2010-03-13 23:27:21 +03:00
return err ;
2010-02-27 22:41:45 +03:00
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static int br_ip6_multicast_query ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
{
2011-04-22 08:53:02 +04:00
const struct ipv6hdr * ip6h = ipv6_hdr ( skb ) ;
2012-12-13 10:51:28 +04:00
struct mld_msg * mld ;
2010-04-22 20:54:22 +04:00
struct net_bridge_mdb_entry * mp ;
struct mld2_query * mld2q ;
2010-11-15 09:38:10 +03:00
struct net_bridge_port_group * p ;
struct net_bridge_port_group __rcu * * pp ;
2010-04-22 20:54:22 +04:00
unsigned long max_delay ;
unsigned long now = jiffies ;
2011-04-22 08:53:02 +04:00
const struct in6_addr * group = NULL ;
2010-04-22 20:54:22 +04:00
int err = 0 ;
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | |
( port & & port - > state = = BR_STATE_DISABLED ) )
goto out ;
br_multicast_query_received ( br , port , ! ipv6_addr_any ( & ip6h - > saddr ) ) ;
if ( skb - > len = = sizeof ( * mld ) ) {
if ( ! pskb_may_pull ( skb , sizeof ( * mld ) ) ) {
err = - EINVAL ;
goto out ;
}
mld = ( struct mld_msg * ) icmp6_hdr ( skb ) ;
2012-07-10 03:56:12 +04:00
max_delay = msecs_to_jiffies ( ntohs ( mld - > mld_maxdelay ) ) ;
2010-04-22 20:54:22 +04:00
if ( max_delay )
group = & mld - > mld_mca ;
} else if ( skb - > len > = sizeof ( * mld2q ) ) {
if ( ! pskb_may_pull ( skb , sizeof ( * mld2q ) ) ) {
err = - EINVAL ;
goto out ;
}
mld2q = ( struct mld2_query * ) icmp6_hdr ( skb ) ;
if ( ! mld2q - > mld2q_nsrcs )
group = & mld2q - > mld2q_mca ;
2012-12-14 03:08:39 +04:00
max_delay = mld2q - > mld2q_mrc ? MLDV2_MRC ( ntohs ( mld2q - > mld2q_mrc ) ) : 1 ;
2010-04-22 20:54:22 +04:00
}
if ( ! group )
goto out ;
2010-11-15 09:38:10 +03:00
mp = br_mdb_ip6_get ( mlock_dereference ( br - > mdb , br ) , group ) ;
2010-04-22 20:54:22 +04:00
if ( ! mp )
goto out ;
max_delay * = br - > multicast_last_member_count ;
2011-02-12 12:05:42 +03:00
if ( mp - > mglist & &
2010-04-22 20:54:22 +04:00
( timer_pending ( & mp - > timer ) ?
time_after ( mp - > timer . expires , now + max_delay ) :
try_to_del_timer_sync ( & mp - > timer ) > = 0 ) )
mod_timer ( & mp - > timer , now + max_delay ) ;
2010-11-15 09:38:10 +03:00
for ( pp = & mp - > ports ;
( p = mlock_dereference ( * pp , br ) ) ! = NULL ;
pp = & p - > next ) {
2010-04-22 20:54:22 +04:00
if ( timer_pending ( & p - > timer ) ?
time_after ( p - > timer . expires , now + max_delay ) :
try_to_del_timer_sync ( & p - > timer ) > = 0 )
2011-02-11 15:42:07 +03:00
mod_timer ( & p - > timer , now + max_delay ) ;
2010-04-22 20:54:22 +04:00
}
out :
spin_unlock ( & br - > multicast_lock ) ;
return err ;
}
# endif
2010-02-27 22:41:45 +03:00
static void br_multicast_leave_group ( struct net_bridge * br ,
struct net_bridge_port * port ,
2010-04-18 07:42:07 +04:00
struct br_ip * group )
2010-02-27 22:41:45 +03:00
{
struct net_bridge_mdb_htable * mdb ;
struct net_bridge_mdb_entry * mp ;
struct net_bridge_port_group * p ;
unsigned long now ;
unsigned long time ;
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | |
( port & & port - > state = = BR_STATE_DISABLED ) | |
timer_pending ( & br - > multicast_querier_timer ) )
goto out ;
2010-11-15 09:38:10 +03:00
mdb = mlock_dereference ( br - > mdb , br ) ;
2010-02-27 22:41:45 +03:00
mp = br_mdb_ip_get ( mdb , group ) ;
if ( ! mp )
goto out ;
2012-12-06 01:24:45 +04:00
if ( port & & ( port - > flags & BR_MULTICAST_FAST_LEAVE ) ) {
2012-12-04 03:56:40 +04:00
struct net_bridge_port_group __rcu * * pp ;
for ( pp = & mp - > ports ;
( p = mlock_dereference ( * pp , br ) ) ! = NULL ;
pp = & p - > next ) {
if ( p - > port ! = port )
continue ;
rcu_assign_pointer ( * pp , p - > next ) ;
hlist_del_init ( & p - > mglist ) ;
del_timer ( & p - > timer ) ;
call_rcu_bh ( & p - > rcu , br_multicast_free_pg ) ;
2012-12-12 02:23:07 +04:00
br_mdb_notify ( br - > dev , port , group , RTM_DELMDB ) ;
2012-12-04 03:56:40 +04:00
if ( ! mp - > ports & & ! mp - > mglist & &
netif_running ( br - > dev ) )
mod_timer ( & mp - > timer , jiffies ) ;
}
goto out ;
}
2010-02-27 22:41:45 +03:00
now = jiffies ;
time = now + br - > multicast_last_member_count *
br - > multicast_last_member_interval ;
if ( ! port ) {
2011-02-12 12:05:42 +03:00
if ( mp - > mglist & &
2010-02-27 22:41:45 +03:00
( timer_pending ( & mp - > timer ) ?
time_after ( mp - > timer . expires , time ) :
try_to_del_timer_sync ( & mp - > timer ) > = 0 ) ) {
mod_timer ( & mp - > timer , time ) ;
}
goto out ;
}
2010-11-15 09:38:10 +03:00
for ( p = mlock_dereference ( mp - > ports , br ) ;
p ! = NULL ;
p = mlock_dereference ( p - > next , br ) ) {
2010-02-27 22:41:45 +03:00
if ( p - > port ! = port )
continue ;
if ( ! hlist_unhashed ( & p - > mglist ) & &
( timer_pending ( & p - > timer ) ?
time_after ( p - > timer . expires , time ) :
try_to_del_timer_sync ( & p - > timer ) > = 0 ) ) {
mod_timer ( & p - > timer , time ) ;
}
break ;
}
out :
spin_unlock ( & br - > multicast_lock ) ;
}
2010-04-18 07:42:07 +04:00
static void br_ip4_multicast_leave_group ( struct net_bridge * br ,
struct net_bridge_port * port ,
__be32 group )
{
struct br_ip br_group ;
if ( ipv4_is_local_multicast ( group ) )
return ;
br_group . u . ip4 = group ;
br_group . proto = htons ( ETH_P_IP ) ;
br_multicast_leave_group ( br , port , & br_group ) ;
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static void br_ip6_multicast_leave_group ( struct net_bridge * br ,
struct net_bridge_port * port ,
const struct in6_addr * group )
{
struct br_ip br_group ;
2011-02-15 16:19:21 +03:00
if ( ! ipv6_is_transient_multicast ( group ) )
2010-04-22 20:54:22 +04:00
return ;
2011-11-21 07:39:03 +04:00
br_group . u . ip6 = * group ;
2010-04-22 20:54:22 +04:00
br_group . proto = htons ( ETH_P_IPV6 ) ;
br_multicast_leave_group ( br , port , & br_group ) ;
}
# endif
2010-04-18 07:42:07 +04:00
2010-02-27 22:41:45 +03:00
static int br_multicast_ipv4_rcv ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
{
struct sk_buff * skb2 = skb ;
2011-04-22 08:53:02 +04:00
const struct iphdr * iph ;
2010-02-27 22:41:45 +03:00
struct igmphdr * ih ;
2012-04-15 09:58:06 +04:00
unsigned int len ;
unsigned int offset ;
2010-02-27 22:41:45 +03:00
int err ;
/* We treat OOM as packet loss for now. */
if ( ! pskb_may_pull ( skb , sizeof ( * iph ) ) )
return - EINVAL ;
iph = ip_hdr ( skb ) ;
if ( iph - > ihl < 5 | | iph - > version ! = 4 )
return - EINVAL ;
if ( ! pskb_may_pull ( skb , ip_hdrlen ( skb ) ) )
return - EINVAL ;
iph = ip_hdr ( skb ) ;
if ( unlikely ( ip_fast_csum ( ( u8 * ) iph , iph - > ihl ) ) )
return - EINVAL ;
2011-06-23 06:39:12 +04:00
if ( iph - > protocol ! = IPPROTO_IGMP ) {
if ( ( iph - > daddr & IGMP_LOCAL_GROUP_MASK ) ! = IGMP_LOCAL_GROUP )
BR_INPUT_SKB_CB ( skb ) - > mrouters_only = 1 ;
2010-02-27 22:41:45 +03:00
return 0 ;
2011-06-23 06:39:12 +04:00
}
2010-02-27 22:41:45 +03:00
len = ntohs ( iph - > tot_len ) ;
if ( skb - > len < len | | len < ip_hdrlen ( skb ) )
return - EINVAL ;
if ( skb - > len > len ) {
skb2 = skb_clone ( skb , GFP_ATOMIC ) ;
if ( ! skb2 )
return - ENOMEM ;
err = pskb_trim_rcsum ( skb2 , len ) ;
if ( err )
2010-03-15 22:26:56 +03:00
goto err_out ;
2010-02-27 22:41:45 +03:00
}
len - = ip_hdrlen ( skb2 ) ;
offset = skb_network_offset ( skb2 ) + ip_hdrlen ( skb2 ) ;
__skb_pull ( skb2 , offset ) ;
skb_reset_transport_header ( skb2 ) ;
err = - EINVAL ;
if ( ! pskb_may_pull ( skb2 , sizeof ( * ih ) ) )
goto out ;
switch ( skb2 - > ip_summed ) {
case CHECKSUM_COMPLETE :
if ( ! csum_fold ( skb2 - > csum ) )
break ;
/* fall through */
case CHECKSUM_NONE :
skb2 - > csum = 0 ;
if ( skb_checksum_complete ( skb2 ) )
2010-03-15 22:26:56 +03:00
goto out ;
2010-02-27 22:41:45 +03:00
}
err = 0 ;
BR_INPUT_SKB_CB ( skb ) - > igmp = 1 ;
ih = igmp_hdr ( skb2 ) ;
switch ( ih - > type ) {
case IGMP_HOST_MEMBERSHIP_REPORT :
case IGMPV2_HOST_MEMBERSHIP_REPORT :
2011-06-13 19:04:43 +04:00
BR_INPUT_SKB_CB ( skb ) - > mrouters_only = 1 ;
2010-04-18 07:42:07 +04:00
err = br_ip4_multicast_add_group ( br , port , ih - > group ) ;
2010-02-27 22:41:45 +03:00
break ;
case IGMPV3_HOST_MEMBERSHIP_REPORT :
2010-04-18 07:42:07 +04:00
err = br_ip4_multicast_igmp3_report ( br , port , skb2 ) ;
2010-02-27 22:41:45 +03:00
break ;
case IGMP_HOST_MEMBERSHIP_QUERY :
2010-04-18 07:42:07 +04:00
err = br_ip4_multicast_query ( br , port , skb2 ) ;
2010-02-27 22:41:45 +03:00
break ;
case IGMP_HOST_LEAVE_MESSAGE :
2010-04-18 07:42:07 +04:00
br_ip4_multicast_leave_group ( br , port , ih - > group ) ;
2010-02-27 22:41:45 +03:00
break ;
}
out :
__skb_push ( skb2 , offset ) ;
2010-03-15 22:26:56 +03:00
err_out :
2010-02-27 22:41:45 +03:00
if ( skb2 ! = skb )
kfree_skb ( skb2 ) ;
return err ;
}
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
static int br_multicast_ipv6_rcv ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
{
2011-01-03 22:26:08 +03:00
struct sk_buff * skb2 ;
2011-04-22 08:53:02 +04:00
const struct ipv6hdr * ip6h ;
2011-08-23 23:57:05 +04:00
u8 icmp6_type ;
2010-04-22 20:54:22 +04:00
u8 nexthdr ;
2011-12-01 05:05:51 +04:00
__be16 frag_off ;
2012-04-15 09:58:06 +04:00
unsigned int len ;
2010-07-15 12:47:33 +04:00
int offset ;
2010-04-22 20:54:22 +04:00
int err ;
if ( ! pskb_may_pull ( skb , sizeof ( * ip6h ) ) )
return - EINVAL ;
ip6h = ipv6_hdr ( skb ) ;
/*
* We ' re interested in MLD messages only .
* - Version is 6
* - MLD has always Router Alert hop - by - hop option
* - But we do not support jumbrograms .
*/
if ( ip6h - > version ! = 6 | |
ip6h - > nexthdr ! = IPPROTO_HOPOPTS | |
ip6h - > payload_len = = 0 )
return 0 ;
2011-03-26 23:27:24 +03:00
len = ntohs ( ip6h - > payload_len ) + sizeof ( * ip6h ) ;
2010-04-22 20:54:22 +04:00
if ( skb - > len < len )
return - EINVAL ;
nexthdr = ip6h - > nexthdr ;
2011-12-01 05:05:51 +04:00
offset = ipv6_skip_exthdr ( skb , sizeof ( * ip6h ) , & nexthdr , & frag_off ) ;
2010-04-22 20:54:22 +04:00
if ( offset < 0 | | nexthdr ! = IPPROTO_ICMPV6 )
return 0 ;
/* Okay, we found ICMPv6 header */
skb2 = skb_clone ( skb , GFP_ATOMIC ) ;
if ( ! skb2 )
return - ENOMEM ;
2011-01-03 22:26:08 +03:00
err = - EINVAL ;
if ( ! pskb_may_pull ( skb2 , offset + sizeof ( struct icmp6hdr ) ) )
goto out ;
2010-04-22 20:54:22 +04:00
len - = offset - skb_network_offset ( skb2 ) ;
__skb_pull ( skb2 , offset ) ;
skb_reset_transport_header ( skb2 ) ;
2011-11-15 12:09:14 +04:00
skb_postpull_rcsum ( skb2 , skb_network_header ( skb2 ) ,
skb_network_header_len ( skb2 ) ) ;
2010-04-22 20:54:22 +04:00
2011-08-23 23:57:05 +04:00
icmp6_type = icmp6_hdr ( skb2 ) - > icmp6_type ;
2010-04-22 20:54:22 +04:00
2011-08-23 23:57:05 +04:00
switch ( icmp6_type ) {
2010-04-22 20:54:22 +04:00
case ICMPV6_MGM_QUERY :
case ICMPV6_MGM_REPORT :
case ICMPV6_MGM_REDUCTION :
case ICMPV6_MLD2_REPORT :
break ;
default :
err = 0 ;
goto out ;
}
/* Okay, we found MLD message. Check further. */
if ( skb2 - > len > len ) {
err = pskb_trim_rcsum ( skb2 , len ) ;
if ( err )
goto out ;
2011-08-24 02:54:33 +04:00
err = - EINVAL ;
2010-04-22 20:54:22 +04:00
}
2011-08-24 02:54:33 +04:00
ip6h = ipv6_hdr ( skb2 ) ;
2010-04-22 20:54:22 +04:00
switch ( skb2 - > ip_summed ) {
case CHECKSUM_COMPLETE :
2011-08-24 02:54:33 +04:00
if ( ! csum_ipv6_magic ( & ip6h - > saddr , & ip6h - > daddr , skb2 - > len ,
IPPROTO_ICMPV6 , skb2 - > csum ) )
2010-04-22 20:54:22 +04:00
break ;
/*FALLTHROUGH*/
case CHECKSUM_NONE :
2011-08-24 02:54:33 +04:00
skb2 - > csum = ~ csum_unfold ( csum_ipv6_magic ( & ip6h - > saddr ,
& ip6h - > daddr ,
skb2 - > len ,
IPPROTO_ICMPV6 , 0 ) ) ;
if ( __skb_checksum_complete ( skb2 ) )
2010-04-22 20:54:22 +04:00
goto out ;
}
err = 0 ;
BR_INPUT_SKB_CB ( skb ) - > igmp = 1 ;
2011-08-23 23:57:05 +04:00
switch ( icmp6_type ) {
2010-04-22 20:54:22 +04:00
case ICMPV6_MGM_REPORT :
{
2011-01-03 22:26:08 +03:00
struct mld_msg * mld ;
if ( ! pskb_may_pull ( skb2 , sizeof ( * mld ) ) ) {
err = - EINVAL ;
goto out ;
}
mld = ( struct mld_msg * ) skb_transport_header ( skb2 ) ;
2011-06-13 19:06:58 +04:00
BR_INPUT_SKB_CB ( skb ) - > mrouters_only = 1 ;
2010-04-22 20:54:22 +04:00
err = br_ip6_multicast_add_group ( br , port , & mld - > mld_mca ) ;
break ;
}
case ICMPV6_MLD2_REPORT :
err = br_ip6_multicast_mld2_report ( br , port , skb2 ) ;
break ;
case ICMPV6_MGM_QUERY :
err = br_ip6_multicast_query ( br , port , skb2 ) ;
break ;
case ICMPV6_MGM_REDUCTION :
{
2011-01-03 22:26:08 +03:00
struct mld_msg * mld ;
if ( ! pskb_may_pull ( skb2 , sizeof ( * mld ) ) ) {
err = - EINVAL ;
goto out ;
}
mld = ( struct mld_msg * ) skb_transport_header ( skb2 ) ;
2010-04-22 20:54:22 +04:00
br_ip6_multicast_leave_group ( br , port , & mld - > mld_mca ) ;
}
}
out :
2011-01-03 22:26:08 +03:00
kfree_skb ( skb2 ) ;
2010-04-22 20:54:22 +04:00
return err ;
}
# endif
2010-02-27 22:41:45 +03:00
int br_multicast_rcv ( struct net_bridge * br , struct net_bridge_port * port ,
struct sk_buff * skb )
{
2010-04-25 12:06:40 +04:00
BR_INPUT_SKB_CB ( skb ) - > igmp = 0 ;
BR_INPUT_SKB_CB ( skb ) - > mrouters_only = 0 ;
2010-02-27 22:41:45 +03:00
if ( br - > multicast_disabled )
return 0 ;
switch ( skb - > protocol ) {
case htons ( ETH_P_IP ) :
return br_multicast_ipv4_rcv ( br , port , skb ) ;
2011-12-10 13:48:31 +04:00
# if IS_ENABLED(CONFIG_IPV6)
2010-04-22 20:54:22 +04:00
case htons ( ETH_P_IPV6 ) :
return br_multicast_ipv6_rcv ( br , port , skb ) ;
# endif
2010-02-27 22:41:45 +03:00
}
return 0 ;
}
static void br_multicast_query_expired ( unsigned long data )
{
struct net_bridge * br = ( void * ) data ;
spin_lock ( & br - > multicast_lock ) ;
if ( br - > multicast_startup_queries_sent <
br - > multicast_startup_query_count )
br - > multicast_startup_queries_sent + + ;
br_multicast_send_query ( br , NULL , br - > multicast_startup_queries_sent ) ;
spin_unlock ( & br - > multicast_lock ) ;
}
void br_multicast_init ( struct net_bridge * br )
{
br - > hash_elasticity = 4 ;
br - > hash_max = 512 ;
br - > multicast_router = 1 ;
2012-04-13 06:37:42 +04:00
br - > multicast_querier = 0 ;
2010-02-27 22:41:45 +03:00
br - > multicast_last_member_count = 2 ;
br - > multicast_startup_query_count = 2 ;
br - > multicast_last_member_interval = HZ ;
br - > multicast_query_response_interval = 10 * HZ ;
br - > multicast_startup_query_interval = 125 * HZ / 4 ;
br - > multicast_query_interval = 125 * HZ ;
br - > multicast_querier_interval = 255 * HZ ;
br - > multicast_membership_interval = 260 * HZ ;
spin_lock_init ( & br - > multicast_lock ) ;
setup_timer ( & br - > multicast_router_timer ,
br_multicast_local_router_expired , 0 ) ;
setup_timer ( & br - > multicast_querier_timer ,
2012-04-30 04:22:56 +04:00
br_multicast_querier_expired , ( unsigned long ) br ) ;
2010-02-27 22:41:45 +03:00
setup_timer ( & br - > multicast_query_timer , br_multicast_query_expired ,
( unsigned long ) br ) ;
}
void br_multicast_open ( struct net_bridge * br )
{
br - > multicast_startup_queries_sent = 0 ;
if ( br - > multicast_disabled )
return ;
mod_timer ( & br - > multicast_query_timer , jiffies ) ;
}
void br_multicast_stop ( struct net_bridge * br )
{
struct net_bridge_mdb_htable * mdb ;
struct net_bridge_mdb_entry * mp ;
struct hlist_node * p , * n ;
u32 ver ;
int i ;
del_timer_sync ( & br - > multicast_router_timer ) ;
del_timer_sync ( & br - > multicast_querier_timer ) ;
del_timer_sync ( & br - > multicast_query_timer ) ;
spin_lock_bh ( & br - > multicast_lock ) ;
2010-11-15 09:38:10 +03:00
mdb = mlock_dereference ( br - > mdb , br ) ;
2010-02-27 22:41:45 +03:00
if ( ! mdb )
goto out ;
br - > mdb = NULL ;
ver = mdb - > ver ;
for ( i = 0 ; i < mdb - > max ; i + + ) {
hlist_for_each_entry_safe ( mp , p , n , & mdb - > mhash [ i ] ,
hlist [ ver ] ) {
del_timer ( & mp - > timer ) ;
call_rcu_bh ( & mp - > rcu , br_multicast_free_group ) ;
}
}
if ( mdb - > old ) {
spin_unlock_bh ( & br - > multicast_lock ) ;
2010-03-06 00:03:35 +03:00
rcu_barrier_bh ( ) ;
2010-02-27 22:41:45 +03:00
spin_lock_bh ( & br - > multicast_lock ) ;
WARN_ON ( mdb - > old ) ;
}
mdb - > old = mdb ;
call_rcu_bh ( & mdb - > rcu , br_mdb_free ) ;
out :
spin_unlock_bh ( & br - > multicast_lock ) ;
}
2010-02-27 22:41:49 +03:00
int br_multicast_set_router ( struct net_bridge * br , unsigned long val )
{
int err = - ENOENT ;
spin_lock_bh ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) )
goto unlock ;
switch ( val ) {
case 0 :
case 2 :
del_timer ( & br - > multicast_router_timer ) ;
/* fall through */
case 1 :
br - > multicast_router = val ;
err = 0 ;
break ;
default :
err = - EINVAL ;
break ;
}
unlock :
spin_unlock_bh ( & br - > multicast_lock ) ;
return err ;
}
int br_multicast_set_port_router ( struct net_bridge_port * p , unsigned long val )
{
struct net_bridge * br = p - > br ;
int err = - ENOENT ;
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) | | p - > state = = BR_STATE_DISABLED )
goto unlock ;
switch ( val ) {
case 0 :
case 1 :
case 2 :
p - > multicast_router = val ;
err = 0 ;
if ( val < 2 & & ! hlist_unhashed ( & p - > rlist ) )
hlist_del_init_rcu ( & p - > rlist ) ;
if ( val = = 1 )
break ;
del_timer ( & p - > multicast_router_timer ) ;
if ( val = = 0 )
break ;
br_multicast_add_router ( br , p ) ;
break ;
default :
err = - EINVAL ;
break ;
}
unlock :
spin_unlock ( & br - > multicast_lock ) ;
return err ;
}
2010-02-27 22:41:50 +03:00
2012-04-13 06:37:42 +04:00
static void br_multicast_start_querier ( struct net_bridge * br )
2010-02-27 22:41:50 +03:00
{
struct net_bridge_port * port ;
2012-04-13 06:37:42 +04:00
br_multicast_open ( br ) ;
list_for_each_entry ( port , & br - > port_list , list ) {
if ( port - > state = = BR_STATE_DISABLED | |
port - > state = = BR_STATE_BLOCKING )
continue ;
__br_multicast_enable_port ( port ) ;
}
}
int br_multicast_toggle ( struct net_bridge * br , unsigned long val )
{
2010-07-29 04:45:30 +04:00
int err = 0 ;
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * mdb ;
2010-02-27 22:41:50 +03:00
2011-11-10 09:48:03 +04:00
spin_lock_bh ( & br - > multicast_lock ) ;
2010-02-27 22:41:50 +03:00
if ( br - > multicast_disabled = = ! val )
goto unlock ;
br - > multicast_disabled = ! val ;
if ( br - > multicast_disabled )
goto unlock ;
2010-07-29 04:45:30 +04:00
if ( ! netif_running ( br - > dev ) )
goto unlock ;
2010-11-15 09:38:10 +03:00
mdb = mlock_dereference ( br - > mdb , br ) ;
if ( mdb ) {
if ( mdb - > old ) {
2010-02-27 22:41:50 +03:00
err = - EEXIST ;
rollback :
br - > multicast_disabled = ! ! val ;
goto unlock ;
}
2010-11-15 09:38:10 +03:00
err = br_mdb_rehash ( & br - > mdb , mdb - > max ,
2010-02-27 22:41:50 +03:00
br - > hash_elasticity ) ;
if ( err )
goto rollback ;
}
2012-04-13 06:37:42 +04:00
br_multicast_start_querier ( br ) ;
2010-02-27 22:41:50 +03:00
unlock :
2011-11-10 09:48:03 +04:00
spin_unlock_bh ( & br - > multicast_lock ) ;
2010-02-27 22:41:50 +03:00
return err ;
}
2010-02-27 22:41:51 +03:00
2012-04-13 06:37:42 +04:00
int br_multicast_set_querier ( struct net_bridge * br , unsigned long val )
{
val = ! ! val ;
spin_lock_bh ( & br - > multicast_lock ) ;
if ( br - > multicast_querier = = val )
goto unlock ;
br - > multicast_querier = val ;
if ( val )
br_multicast_start_querier ( br ) ;
unlock :
spin_unlock_bh ( & br - > multicast_lock ) ;
return 0 ;
}
2010-02-27 22:41:51 +03:00
int br_multicast_set_hash_max ( struct net_bridge * br , unsigned long val )
{
int err = - ENOENT ;
u32 old ;
2010-11-15 09:38:10 +03:00
struct net_bridge_mdb_htable * mdb ;
2010-02-27 22:41:51 +03:00
spin_lock ( & br - > multicast_lock ) ;
if ( ! netif_running ( br - > dev ) )
goto unlock ;
err = - EINVAL ;
if ( ! is_power_of_2 ( val ) )
goto unlock ;
2010-11-15 09:38:10 +03:00
mdb = mlock_dereference ( br - > mdb , br ) ;
if ( mdb & & val < mdb - > size )
2010-02-27 22:41:51 +03:00
goto unlock ;
err = 0 ;
old = br - > hash_max ;
br - > hash_max = val ;
2010-11-15 09:38:10 +03:00
if ( mdb ) {
if ( mdb - > old ) {
2010-02-27 22:41:51 +03:00
err = - EEXIST ;
rollback :
br - > hash_max = old ;
goto unlock ;
}
err = br_mdb_rehash ( & br - > mdb , br - > hash_max ,
br - > hash_elasticity ) ;
if ( err )
goto rollback ;
}
unlock :
spin_unlock ( & br - > multicast_lock ) ;
return err ;
}