2005-04-16 15:20:36 -07:00
/*
* Device event handling
* Linux ethernet bridge
*
* Authors :
* Lennert Buytenhek < buytenh @ gnu . org >
*
* 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/kernel.h>
2006-05-25 16:00:12 -07:00
# include <linux/rtnetlink.h>
2007-09-12 13:02:17 +02:00
# include <net/net_namespace.h>
2005-04-16 15:20:36 -07:00
# include "br_private.h"
static int br_device_event ( struct notifier_block * unused , unsigned long event , void * ptr ) ;
struct notifier_block br_device_notifier = {
. notifier_call = br_device_event
} ;
/*
* Handle changes in state of network devices enslaved to a bridge .
2007-02-09 23:24:35 +09:00
*
2005-04-16 15:20:36 -07:00
* Note : don ' t care about up / down if bridge itself is down , because
* port state is checked when bridge is brought up .
*/
static int br_device_event ( struct notifier_block * unused , unsigned long event , void * ptr )
{
2013-05-28 01:30:21 +00:00
struct net_device * dev = netdev_notifier_info_to_dev ( ptr ) ;
2010-11-15 06:38:13 +00:00
struct net_bridge_port * p ;
2005-04-16 15:20:36 -07:00
struct net_bridge * br ;
2011-08-05 11:04:10 +00:00
bool changed_addr ;
2010-05-10 09:31:11 +00:00
int err ;
2005-04-16 15:20:36 -07:00
2011-04-04 14:03:32 +00:00
/* register of bridge completed, add sysfs entries */
2011-04-17 17:52:51 -07:00
if ( ( dev - > priv_flags & IFF_EBRIDGE ) & & event = = NETDEV_REGISTER ) {
2011-04-04 14:03:32 +00:00
br_sysfs_addbr ( dev ) ;
return NOTIFY_DONE ;
}
2005-04-16 15:20:36 -07:00
/* not a port of a bridge */
2010-11-15 06:38:14 +00:00
p = br_port_get_rtnl ( dev ) ;
if ( ! p )
2005-04-16 15:20:36 -07:00
return NOTIFY_DONE ;
br = p - > br ;
switch ( event ) {
case NETDEV_CHANGEMTU :
dev_set_mtu ( br - > dev , br_min_mtu ( br ) ) ;
break ;
case NETDEV_CHANGEADDR :
2007-02-22 01:10:18 -08:00
spin_lock_bh ( & br - > lock ) ;
2005-04-16 15:20:36 -07:00
br_fdb_changeaddr ( p , dev - > dev_addr ) ;
2011-08-05 11:04:10 +00:00
changed_addr = br_stp_recalculate_bridge_id ( br ) ;
2007-02-22 01:10:18 -08:00
spin_unlock_bh ( & br - > lock ) ;
2011-08-05 11:04:10 +00:00
if ( changed_addr )
call_netdevice_notifiers ( NETDEV_CHANGEADDR , br - > dev ) ;
2005-04-16 15:20:36 -07:00
break ;
2005-12-20 15:19:51 -08:00
case NETDEV_CHANGE :
2007-02-22 01:10:18 -08:00
br_port_carrier_check ( p ) ;
2005-04-16 15:20:36 -07:00
break ;
2005-05-29 14:15:17 -07:00
case NETDEV_FEAT_CHANGE :
2011-04-22 06:31:16 +00:00
netdev_update_features ( br - > dev ) ;
2005-05-29 14:15:17 -07:00
break ;
2005-04-16 15:20:36 -07:00
case NETDEV_DOWN :
2007-02-22 01:10:18 -08:00
spin_lock_bh ( & br - > lock ) ;
2005-04-16 15:20:36 -07:00
if ( br - > dev - > flags & IFF_UP )
br_stp_disable_port ( p ) ;
2007-02-22 01:10:18 -08:00
spin_unlock_bh ( & br - > lock ) ;
2005-04-16 15:20:36 -07:00
break ;
case NETDEV_UP :
2012-12-28 18:15:22 +00:00
if ( netif_running ( br - > dev ) & & netif_oper_up ( dev ) ) {
2007-03-22 14:08:46 -07:00
spin_lock_bh ( & br - > lock ) ;
2005-04-16 15:20:36 -07:00
br_stp_enable_port ( p ) ;
2007-03-22 14:08:46 -07:00
spin_unlock_bh ( & br - > lock ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case NETDEV_UNREGISTER :
br_del_if ( br , dev ) ;
2007-02-22 01:10:18 -08:00
break ;
2010-03-10 10:30:19 +00:00
2010-05-10 09:31:11 +00:00
case NETDEV_CHANGENAME :
err = br_sysfs_renameif ( p ) ;
if ( err )
return notifier_from_errno ( err ) ;
break ;
2010-03-10 10:30:19 +00:00
case NETDEV_PRE_TYPE_CHANGE :
/* Forbid underlaying device to change its type. */
return NOTIFY_BAD ;
2013-07-20 12:13:53 +02:00
case NETDEV_RESEND_IGMP :
/* Propagate to master device */
call_netdevice_notifiers ( event , br - > dev ) ;
break ;
2007-02-09 23:24:35 +09:00
}
2005-04-16 15:20:36 -07:00
2007-03-22 14:08:46 -07:00
/* Events that may cause spanning tree to refresh */
if ( event = = NETDEV_CHANGEADDR | | event = = NETDEV_UP | |
event = = NETDEV_CHANGE | | event = = NETDEV_DOWN )
br_ifinfo_notify ( RTM_NEWLINK , p ) ;
2005-04-16 15:20:36 -07:00
return NOTIFY_DONE ;
}