2005-04-17 02:20:36 +04:00
/*
* Linux network device link state notification
*
* Author :
* Stefan Rompf < sux @ loplof . de >
*
* 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/config.h>
# include <linux/module.h>
# include <linux/netdevice.h>
# include <linux/if.h>
# include <net/sock.h>
2005-05-04 03:18:52 +04:00
# include <net/pkt_sched.h>
2005-04-17 02:20:36 +04:00
# include <linux/rtnetlink.h>
# include <linux/jiffies.h>
# include <linux/spinlock.h>
# include <linux/list.h>
# include <linux/slab.h>
# include <linux/workqueue.h>
# include <linux/bitops.h>
# include <asm/types.h>
enum lw_bits {
LW_RUNNING = 0 ,
LW_SE_USED
} ;
static unsigned long linkwatch_flags ;
static unsigned long linkwatch_nextevent ;
static void linkwatch_event ( void * dummy ) ;
static DECLARE_WORK ( linkwatch_work , linkwatch_event , NULL ) ;
static LIST_HEAD ( lweventlist ) ;
static DEFINE_SPINLOCK ( lweventlist_lock ) ;
struct lw_event {
struct list_head list ;
struct net_device * dev ;
} ;
/* Avoid kmalloc() for most systems */
static struct lw_event singleevent ;
2006-03-21 04:09:11 +03:00
static unsigned char default_operstate ( const struct net_device * dev )
{
if ( ! netif_carrier_ok ( dev ) )
return ( dev - > ifindex ! = dev - > iflink ?
IF_OPER_LOWERLAYERDOWN : IF_OPER_DOWN ) ;
if ( netif_dormant ( dev ) )
return IF_OPER_DORMANT ;
return IF_OPER_UP ;
}
static void rfc2863_policy ( struct net_device * dev )
{
unsigned char operstate = default_operstate ( dev ) ;
if ( operstate = = dev - > operstate )
return ;
write_lock_bh ( & dev_base_lock ) ;
switch ( dev - > link_mode ) {
case IF_LINK_MODE_DORMANT :
if ( operstate = = IF_OPER_UP )
operstate = IF_OPER_DORMANT ;
break ;
case IF_LINK_MODE_DEFAULT :
default :
break ;
} ;
dev - > operstate = operstate ;
write_unlock_bh ( & dev_base_lock ) ;
}
2005-04-17 02:20:36 +04:00
/* Must be called with the rtnl semaphore held */
void linkwatch_run_queue ( void )
{
2006-06-23 13:05:55 +04:00
struct list_head head , * n , * next ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & lweventlist_lock ) ;
2006-06-23 13:05:55 +04:00
list_replace_init ( & lweventlist , & head ) ;
2005-04-17 02:20:36 +04:00
spin_unlock_irq ( & lweventlist_lock ) ;
list_for_each_safe ( n , next , & head ) {
struct lw_event * event = list_entry ( n , struct lw_event , list ) ;
struct net_device * dev = event - > dev ;
if ( event = = & singleevent ) {
clear_bit ( LW_SE_USED , & linkwatch_flags ) ;
} else {
kfree ( event ) ;
}
/* We are about to handle this device,
* so new events can be accepted
*/
clear_bit ( __LINK_STATE_LINKWATCH_PENDING , & dev - > state ) ;
2006-03-21 04:09:11 +03:00
rfc2863_policy ( dev ) ;
2005-04-17 02:20:36 +04:00
if ( dev - > flags & IFF_UP ) {
2005-05-04 03:18:52 +04:00
if ( netif_carrier_ok ( dev ) ) {
WARN_ON ( dev - > qdisc_sleeping = = & noop_qdisc ) ;
dev_activate ( dev ) ;
} else
dev_deactivate ( dev ) ;
2005-04-17 02:20:36 +04:00
netdev_state_change ( dev ) ;
}
dev_put ( dev ) ;
}
}
static void linkwatch_event ( void * dummy )
{
/* Limit the number of linkwatch events to one
* per second so that a runaway driver does not
* cause a storm of messages on the netlink
* socket
*/
linkwatch_nextevent = jiffies + HZ ;
clear_bit ( LW_RUNNING , & linkwatch_flags ) ;
2006-03-21 09:23:58 +03:00
rtnl_lock ( ) ;
2005-04-17 02:20:36 +04:00
linkwatch_run_queue ( ) ;
2006-03-21 09:23:58 +03:00
rtnl_unlock ( ) ;
2005-04-17 02:20:36 +04:00
}
void linkwatch_fire_event ( struct net_device * dev )
{
if ( ! test_and_set_bit ( __LINK_STATE_LINKWATCH_PENDING , & dev - > state ) ) {
unsigned long flags ;
struct lw_event * event ;
if ( test_and_set_bit ( LW_SE_USED , & linkwatch_flags ) ) {
event = kmalloc ( sizeof ( struct lw_event ) , GFP_ATOMIC ) ;
if ( unlikely ( event = = NULL ) ) {
clear_bit ( __LINK_STATE_LINKWATCH_PENDING , & dev - > state ) ;
return ;
}
} else {
event = & singleevent ;
}
dev_hold ( dev ) ;
event - > dev = dev ;
spin_lock_irqsave ( & lweventlist_lock , flags ) ;
list_add_tail ( & event - > list , & lweventlist ) ;
spin_unlock_irqrestore ( & lweventlist_lock , flags ) ;
if ( ! test_and_set_bit ( LW_RUNNING , & linkwatch_flags ) ) {
2006-05-10 02:27:54 +04:00
unsigned long delay = linkwatch_nextevent - jiffies ;
2005-04-17 02:20:36 +04:00
2006-05-10 02:27:54 +04:00
/* If we wrap around we'll delay it by at most HZ. */
if ( ! delay | | delay > HZ )
2005-04-17 02:20:36 +04:00
schedule_work ( & linkwatch_work ) ;
2006-05-10 02:27:54 +04:00
else
schedule_delayed_work ( & linkwatch_work , delay ) ;
2005-04-17 02:20:36 +04:00
}
}
}
EXPORT_SYMBOL ( linkwatch_fire_event ) ;