2019-05-27 09:55:01 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
* Spanning tree protocol ; timer - related code
* Linux ethernet bridge
*
* Authors :
* Lennert Buytenhek < buytenh @ gnu . org >
*/
# include <linux/kernel.h>
# include <linux/times.h>
# include "br_private.h"
# include "br_private_stp.h"
/* called under bridge lock */
static int br_is_designated_for_some_port ( const struct net_bridge * br )
{
struct net_bridge_port * p ;
list_for_each_entry ( p , & br - > port_list , list ) {
if ( p - > state ! = BR_STATE_DISABLED & &
2007-02-09 17:24:35 +03:00
! memcmp ( & p - > designated_bridge , & br - > bridge_id , 8 ) )
2005-04-17 02:20:36 +04:00
return 1 ;
}
return 0 ;
}
2017-11-03 09:21:11 +03:00
static void br_hello_timer_expired ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-11-03 09:21:11 +03:00
struct net_bridge * br = from_timer ( br , t , hello_timer ) ;
2007-02-09 17:24:35 +03:00
2010-05-10 13:31:09 +04:00
br_debug ( br , " hello timer expired \n " ) ;
2006-03-21 09:56:25 +03:00
spin_lock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
if ( br - > dev - > flags & IFF_UP ) {
br_config_bpdu_generation ( br ) ;
2017-05-19 17:20:29 +03:00
if ( br - > stp_enabled = = BR_KERNEL_STP )
2015-07-23 21:01:05 +03:00
mod_timer ( & br - > hello_timer ,
round_jiffies ( jiffies + br - > hello_time ) ) ;
2005-04-17 02:20:36 +04:00
}
2006-03-21 09:56:25 +03:00
spin_unlock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
}
2017-11-03 09:21:11 +03:00
static void br_message_age_timer_expired ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-11-03 09:21:11 +03:00
struct net_bridge_port * p = from_timer ( p , t , message_age_timer ) ;
2005-04-17 02:20:36 +04:00
struct net_bridge * br = p - > br ;
const bridge_id * id = & p - > designated_bridge ;
int was_root ;
if ( p - > state = = BR_STATE_DISABLED )
return ;
2010-05-10 13:31:09 +04:00
br_info ( br , " port %u(%s) neighbor %.2x%.2x.%pM lost \n " ,
2012-04-15 09:58:06 +04:00
( unsigned int ) p - > port_no , p - > dev - > name ,
2010-05-10 13:31:09 +04:00
id - > prio [ 0 ] , id - > prio [ 1 ] , & id - > addr ) ;
2005-04-17 02:20:36 +04:00
/*
* According to the spec , the message age timer cannot be
* running when we are the root bridge . So . . this was_root
* check is redundant . I ' m leaving it in for now , though .
*/
2006-03-21 09:56:25 +03:00
spin_lock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
if ( p - > state = = BR_STATE_DISABLED )
goto unlock ;
was_root = br_is_root_bridge ( br ) ;
br_become_designated_port ( p ) ;
br_configuration_update ( br ) ;
br_port_state_selection ( br ) ;
if ( br_is_root_bridge ( br ) & & ! was_root )
br_become_root_bridge ( br ) ;
unlock :
2006-03-21 09:56:25 +03:00
spin_unlock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
}
2017-11-03 09:21:11 +03:00
static void br_forward_delay_timer_expired ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-11-03 09:21:11 +03:00
struct net_bridge_port * p = from_timer ( p , t , forward_delay_timer ) ;
2005-04-17 02:20:36 +04:00
struct net_bridge * br = p - > br ;
2010-05-10 13:31:09 +04:00
br_debug ( br , " port %u(%s) forward delay timer \n " ,
2012-04-15 09:58:06 +04:00
( unsigned int ) p - > port_no , p - > dev - > name ) ;
2006-03-21 09:56:25 +03:00
spin_lock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
if ( p - > state = = BR_STATE_LISTENING ) {
2014-10-01 03:13:19 +04:00
br_set_state ( p , BR_STATE_LEARNING ) ;
2005-04-17 02:20:36 +04:00
mod_timer ( & p - > forward_delay_timer ,
jiffies + br - > forward_delay ) ;
} else if ( p - > state = = BR_STATE_LEARNING ) {
2014-10-01 03:13:19 +04:00
br_set_state ( p , BR_STATE_FORWARDING ) ;
2005-04-17 02:20:36 +04:00
if ( br_is_designated_for_some_port ( br ) )
br_topology_change_detection ( br ) ;
2011-03-07 11:34:06 +03:00
netif_carrier_on ( br - > dev ) ;
2005-04-17 02:20:36 +04:00
}
2015-05-21 23:28:29 +03:00
rcu_read_lock ( ) ;
2017-11-01 13:18:13 +03:00
br_ifinfo_notify ( RTM_NEWLINK , NULL , p ) ;
2015-05-21 23:28:29 +03:00
rcu_read_unlock ( ) ;
2006-03-21 09:56:25 +03:00
spin_unlock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
}
2017-11-03 09:21:11 +03:00
static void br_tcn_timer_expired ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-11-03 09:21:11 +03:00
struct net_bridge * br = from_timer ( br , t , tcn_timer ) ;
2005-04-17 02:20:36 +04:00
2010-05-10 13:31:09 +04:00
br_debug ( br , " tcn timer expired \n " ) ;
2006-03-21 09:56:25 +03:00
spin_lock ( & br - > lock ) ;
2013-05-02 18:23:28 +04:00
if ( ! br_is_root_bridge ( br ) & & ( br - > dev - > flags & IFF_UP ) ) {
2005-04-17 02:20:36 +04:00
br_transmit_tcn ( br ) ;
2007-02-09 17:24:35 +03:00
2013-12-19 09:28:12 +04:00
mod_timer ( & br - > tcn_timer , jiffies + br - > bridge_hello_time ) ;
2005-04-17 02:20:36 +04:00
}
2006-03-21 09:56:25 +03:00
spin_unlock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
}
2017-11-03 09:21:11 +03:00
static void br_topology_change_timer_expired ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-11-03 09:21:11 +03:00
struct net_bridge * br = from_timer ( br , t , topology_change_timer ) ;
2005-04-17 02:20:36 +04:00
2010-05-10 13:31:09 +04:00
br_debug ( br , " topo change timer expired \n " ) ;
2006-03-21 09:56:25 +03:00
spin_lock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
br - > topology_change_detected = 0 ;
2016-12-10 21:44:28 +03:00
__br_set_topology_change ( br , 0 ) ;
2006-03-21 09:56:25 +03:00
spin_unlock ( & br - > lock ) ;
2005-04-17 02:20:36 +04:00
}
2017-11-03 09:21:11 +03:00
static void br_hold_timer_expired ( struct timer_list * t )
2005-04-17 02:20:36 +04:00
{
2017-11-03 09:21:11 +03:00
struct net_bridge_port * p = from_timer ( p , t , hold_timer ) ;
2005-04-17 02:20:36 +04:00
2010-05-10 13:31:09 +04:00
br_debug ( p - > br , " port %u(%s) hold timer expired \n " ,
2012-04-15 09:58:06 +04:00
( unsigned int ) p - > port_no , p - > dev - > name ) ;
2005-04-17 02:20:36 +04:00
2006-03-21 09:56:25 +03:00
spin_lock ( & p - > br - > lock ) ;
2005-04-17 02:20:36 +04:00
if ( p - > config_pending )
br_transmit_config ( p ) ;
2006-03-21 09:56:25 +03:00
spin_unlock ( & p - > br - > lock ) ;
2005-04-17 02:20:36 +04:00
}
void br_stp_timer_init ( struct net_bridge * br )
{
2017-11-03 09:21:11 +03:00
timer_setup ( & br - > hello_timer , br_hello_timer_expired , 0 ) ;
timer_setup ( & br - > tcn_timer , br_tcn_timer_expired , 0 ) ;
timer_setup ( & br - > topology_change_timer ,
br_topology_change_timer_expired , 0 ) ;
2005-04-17 02:20:36 +04:00
}
void br_stp_port_timer_init ( struct net_bridge_port * p )
{
2017-11-03 09:21:11 +03:00
timer_setup ( & p - > message_age_timer , br_message_age_timer_expired , 0 ) ;
timer_setup ( & p - > forward_delay_timer , br_forward_delay_timer_expired , 0 ) ;
timer_setup ( & p - > hold_timer , br_hold_timer_expired , 0 ) ;
2007-02-09 17:24:35 +03:00
}
2005-04-17 02:20:36 +04:00
/* Report ticks left (in USER_HZ) used for API */
unsigned long br_timer_value ( const struct timer_list * timer )
{
return timer_pending ( timer )
2012-08-09 01:13:53 +04:00
? jiffies_delta_to_clock_t ( timer - > expires - jiffies ) : 0 ;
2005-04-17 02:20:36 +04:00
}