2007-05-05 11:45:53 -07:00
/*
* Copyright 2002 - 2005 , Instant802 Networks , Inc .
* Copyright 2006 - 2007 Jiri Benc < jbenc @ suse . cz >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/netdevice.h>
# include <linux/types.h>
# include <linux/slab.h>
# include <linux/skbuff.h>
# include <linux/if_arp.h>
2007-12-17 15:07:43 +01:00
# include <linux/timer.h>
2008-02-25 16:27:46 +01:00
# include <linux/rtnetlink.h>
2007-05-05 11:45:53 -07:00
# include <net/mac80211.h>
# include "ieee80211_i.h"
2008-04-08 15:14:40 -04:00
# include "rate.h"
2007-05-05 11:45:53 -07:00
# include "sta_info.h"
2007-05-05 11:46:38 -07:00
# include "debugfs_sta.h"
2008-02-23 15:17:11 +01:00
# include "mesh.h"
2007-05-05 11:45:53 -07:00
2008-02-25 16:27:46 +01:00
/**
* DOC : STA information lifetime rules
*
* STA info structures ( & struct sta_info ) are managed in a hash table
* for faster lookup and a list for iteration . They are managed using
* RCU , i . e . access to the list and hash table is protected by RCU .
*
2008-02-27 09:56:40 +01:00
* Upon allocating a STA info structure with sta_info_alloc ( ) , the caller owns
* that structure . It must then either destroy it using sta_info_destroy ( )
* ( which is pretty useless ) or insert it into the hash table using
* sta_info_insert ( ) which demotes the reference from ownership to a regular
* RCU - protected reference ; if the function is called without protection by an
2008-04-01 15:21:00 +02:00
* RCU critical section the reference is instantly invalidated . Note that the
* caller may not do much with the STA info before inserting it , in particular ,
* it may not start any mesh peer link management or add encryption keys .
*
* When the insertion fails ( sta_info_insert ( ) ) returns non - zero ) , the
* structure will have been freed by sta_info_insert ( ) !
2008-02-25 16:27:46 +01:00
*
* Because there are debugfs entries for each station , and adding those
* must be able to sleep , it is also possible to " pin " a station entry ,
* that means it can be removed from the hash table but not be freed .
2008-04-01 15:21:00 +02:00
* See the comment in __sta_info_unlink ( ) for more information , this is
* an internal capability only .
2008-02-25 16:27:46 +01:00
*
* In order to remove a STA info structure , the caller needs to first
2008-02-26 14:34:06 +01:00
* unlink it ( sta_info_unlink ( ) ) from the list and hash tables and
2008-04-08 17:56:52 +02:00
* then destroy it ; sta_info_destroy ( ) will wait for an RCU grace period
* to elapse before actually freeing it . Due to the pinning and the
* possibility of multiple callers trying to remove the same STA info at
* the same time , sta_info_unlink ( ) can clear the STA info pointer it is
* passed to indicate that the STA info is owned by somebody else now .
2008-02-25 16:27:46 +01:00
*
2008-02-26 14:34:06 +01:00
* If sta_info_unlink ( ) did not clear the pointer then the caller owns
2008-02-25 16:27:46 +01:00
* the STA info structure now and is responsible of destroying it with
2008-04-08 17:56:52 +02:00
* a call to sta_info_destroy ( ) .
2008-02-25 16:27:46 +01:00
*
* In all other cases , there is no concept of ownership on a STA entry ,
* each structure is owned by the global hash table / list until it is
* removed . All users of the structure need to be RCU protected so that
* the structure won ' t be freed before they are done using it .
*/
2007-05-05 11:45:53 -07:00
/* Caller must hold local->sta_lock */
2007-07-27 15:43:23 +02:00
static int sta_info_hash_del ( struct ieee80211_local * local ,
struct sta_info * sta )
2007-05-05 11:45:53 -07:00
{
struct sta_info * s ;
s = local - > sta_hash [ STA_HASH ( sta - > addr ) ] ;
if ( ! s )
2007-07-27 15:43:23 +02:00
return - ENOENT ;
if ( s = = sta ) {
2008-02-25 16:27:46 +01:00
rcu_assign_pointer ( local - > sta_hash [ STA_HASH ( sta - > addr ) ] ,
s - > hnext ) ;
2007-07-27 15:43:23 +02:00
return 0 ;
2007-05-05 11:45:53 -07:00
}
2007-07-27 15:43:23 +02:00
while ( s - > hnext & & s - > hnext ! = sta )
2007-05-05 11:45:53 -07:00
s = s - > hnext ;
2007-07-27 15:43:23 +02:00
if ( s - > hnext ) {
2008-02-25 16:27:46 +01:00
rcu_assign_pointer ( s - > hnext , sta - > hnext ) ;
2007-07-27 15:43:23 +02:00
return 0 ;
}
2007-05-05 11:45:53 -07:00
2007-07-27 15:43:23 +02:00
return - ENOENT ;
2007-05-05 11:45:53 -07:00
}
2008-02-25 16:27:46 +01:00
/* protected by RCU */
2008-02-21 14:09:30 +01:00
static struct sta_info * __sta_info_find ( struct ieee80211_local * local ,
u8 * addr )
2007-05-05 11:45:53 -07:00
{
struct sta_info * sta ;
2008-02-25 16:27:46 +01:00
sta = rcu_dereference ( local - > sta_hash [ STA_HASH ( addr ) ] ) ;
2007-05-05 11:45:53 -07:00
while ( sta ) {
2008-02-21 14:09:30 +01:00
if ( compare_ether_addr ( sta - > addr , addr ) = = 0 )
2007-05-05 11:45:53 -07:00
break ;
2008-02-25 16:27:46 +01:00
sta = rcu_dereference ( sta - > hnext ) ;
2007-05-05 11:45:53 -07:00
}
2008-02-21 14:09:30 +01:00
return sta ;
}
struct sta_info * sta_info_get ( struct ieee80211_local * local , u8 * addr )
{
2008-02-25 16:27:46 +01:00
return __sta_info_find ( local , addr ) ;
2007-05-05 11:45:53 -07:00
}
EXPORT_SYMBOL ( sta_info_get ) ;
2008-02-23 15:17:11 +01:00
struct sta_info * sta_info_get_by_idx ( struct ieee80211_local * local , int idx ,
struct net_device * dev )
{
struct sta_info * sta ;
int i = 0 ;
2008-02-25 16:27:46 +01:00
list_for_each_entry_rcu ( sta , & local - > sta_list , list ) {
2008-02-29 17:51:25 -08:00
if ( dev & & dev ! = sta - > sdata - > dev )
continue ;
2008-02-23 15:17:11 +01:00
if ( i < idx ) {
+ + i ;
continue ;
}
2008-02-29 17:51:25 -08:00
return sta ;
2008-02-23 15:17:11 +01:00
}
return NULL ;
}
2007-05-05 11:45:53 -07:00
2008-04-01 15:21:00 +02:00
/**
* __sta_info_free - internal STA free helper
*
* @ sta : STA info to free
*
* This function must undo everything done by sta_info_alloc ( )
* that may happen before sta_info_insert ( ) .
*/
static void __sta_info_free ( struct ieee80211_local * local ,
struct sta_info * sta )
{
DECLARE_MAC_BUF ( mbuf ) ;
rate_control_free_sta ( sta - > rate_ctrl , sta - > rate_ctrl_priv ) ;
rate_control_put ( sta - > rate_ctrl ) ;
# ifdef CONFIG_MAC80211_VERBOSE_DEBUG
printk ( KERN_DEBUG " %s: Destroyed STA %s \n " ,
wiphy_name ( local - > hw . wiphy ) , print_mac ( mbuf , sta - > addr ) ) ;
# endif /* CONFIG_MAC80211_VERBOSE_DEBUG */
kfree ( sta ) ;
}
2008-02-25 16:27:46 +01:00
void sta_info_destroy ( struct sta_info * sta )
2007-05-05 11:45:53 -07:00
{
2008-03-31 19:23:00 +02:00
struct ieee80211_local * local ;
2007-05-05 11:45:53 -07:00
struct sk_buff * skb ;
2007-12-25 17:00:33 +02:00
int i ;
2008-02-25 16:27:47 +01:00
2008-03-31 19:23:00 +02:00
might_sleep ( ) ;
2008-02-25 16:27:47 +01:00
if ( ! sta )
return ;
2007-05-05 11:45:53 -07:00
2008-03-31 19:23:00 +02:00
local = sta - > local ;
2008-02-25 16:27:46 +01:00
rate_control_remove_sta_debugfs ( sta ) ;
ieee80211_sta_debugfs_remove ( sta ) ;
# ifdef CONFIG_MAC80211_MESH
if ( ieee80211_vif_is_mesh ( & sta - > sdata - > vif ) )
mesh_plink_deactivate ( sta ) ;
# endif
2008-04-08 17:56:52 +02:00
/*
* We have only unlinked the key , and actually destroying it
* may mean it is removed from hardware which requires that
* the key - > sta pointer is still valid , so flush the key todo
* list here .
*
* ieee80211_key_todo ( ) will synchronize_rcu ( ) so after this
* nothing can reference this sta struct any more .
*/
ieee80211_key_todo ( ) ;
2008-02-25 16:27:46 +01:00
# ifdef CONFIG_MAC80211_MESH
if ( ieee80211_vif_is_mesh ( & sta - > sdata - > vif ) )
del_timer_sync ( & sta - > plink_timer ) ;
# endif
2007-05-05 11:45:53 -07:00
while ( ( skb = skb_dequeue ( & sta - > ps_tx_buf ) ) ! = NULL ) {
local - > total_ps_buffered - - ;
dev_kfree_skb_any ( skb ) ;
}
2008-02-25 16:27:46 +01:00
while ( ( skb = skb_dequeue ( & sta - > tx_filtered ) ) ! = NULL )
2007-05-05 11:45:53 -07:00
dev_kfree_skb_any ( skb ) ;
2008-02-25 16:27:46 +01:00
2008-01-28 14:07:19 +02:00
for ( i = 0 ; i < STA_TID_NUM ; i + + ) {
2008-05-03 01:02:02 +02:00
spin_lock_bh ( & sta - > lock ) ;
2008-03-26 20:36:03 +02:00
if ( sta - > ampdu_mlme . tid_rx [ i ] )
del_timer_sync ( & sta - > ampdu_mlme . tid_rx [ i ] - > session_timer ) ;
if ( sta - > ampdu_mlme . tid_tx [ i ] )
del_timer_sync ( & sta - > ampdu_mlme . tid_tx [ i ] - > addba_resp_timer ) ;
2008-05-03 01:02:02 +02:00
spin_unlock_bh ( & sta - > lock ) ;
2008-01-28 14:07:19 +02:00
}
2008-03-26 20:36:03 +02:00
2008-04-01 15:21:00 +02:00
__sta_info_free ( local , sta ) ;
2007-05-05 11:45:53 -07:00
}
2008-02-25 16:27:46 +01:00
/* Caller must hold local->sta_lock */
static void sta_info_hash_add ( struct ieee80211_local * local ,
struct sta_info * sta )
2007-05-05 11:45:53 -07:00
{
2008-02-25 16:27:46 +01:00
sta - > hnext = local - > sta_hash [ STA_HASH ( sta - > addr ) ] ;
rcu_assign_pointer ( local - > sta_hash [ STA_HASH ( sta - > addr ) ] , sta ) ;
2007-05-05 11:45:53 -07:00
}
2008-02-25 16:27:47 +01:00
struct sta_info * sta_info_alloc ( struct ieee80211_sub_if_data * sdata ,
u8 * addr , gfp_t gfp )
2007-05-05 11:45:53 -07:00
{
2008-02-25 16:27:46 +01:00
struct ieee80211_local * local = sdata - > local ;
2007-05-05 11:45:53 -07:00
struct sta_info * sta ;
2007-12-25 17:00:34 +02:00
int i ;
2008-02-25 16:27:47 +01:00
DECLARE_MAC_BUF ( mbuf ) ;
2007-05-05 11:45:53 -07:00
2008-02-25 16:27:47 +01:00
sta = kzalloc ( sizeof ( * sta ) , gfp ) ;
2007-05-05 11:45:53 -07:00
if ( ! sta )
2008-02-25 16:27:47 +01:00
return NULL ;
2007-05-05 11:45:53 -07:00
2008-05-03 01:02:02 +02:00
spin_lock_init ( & sta - > lock ) ;
2008-02-25 16:27:46 +01:00
memcpy ( sta - > addr , addr , ETH_ALEN ) ;
sta - > local = local ;
sta - > sdata = sdata ;
2007-05-05 11:45:53 -07:00
sta - > rate_ctrl = rate_control_get ( local - > rate_ctrl ) ;
2008-02-25 16:27:46 +01:00
sta - > rate_ctrl_priv = rate_control_alloc_sta ( sta - > rate_ctrl ,
2008-02-25 16:27:47 +01:00
gfp ) ;
2007-05-05 11:45:53 -07:00
if ( ! sta - > rate_ctrl_priv ) {
rate_control_put ( sta - > rate_ctrl ) ;
kfree ( sta ) ;
2008-02-25 16:27:47 +01:00
return NULL ;
2007-05-05 11:45:53 -07:00
}
2007-12-25 17:00:34 +02:00
for ( i = 0 ; i < STA_TID_NUM ; i + + ) {
/* timer_to_tid must be initialized with identity mapping to
* enable session_timer ' s data differentiation . refer to
* sta_rx_agg_session_timer_expired for useage */
sta - > timer_to_tid [ i ] = i ;
2008-01-28 14:07:19 +02:00
/* tid to tx queue: initialize according to HW (0 is valid) */
2008-04-30 18:51:21 +02:00
sta - > tid_to_tx_q [ i ] = local - > hw . queues + local - > hw . ampdu_queues ;
2008-03-26 20:36:03 +02:00
/* rx */
sta - > ampdu_mlme . tid_state_rx [ i ] = HT_AGG_STATE_IDLE ;
sta - > ampdu_mlme . tid_rx [ i ] = NULL ;
/* tx */
sta - > ampdu_mlme . tid_state_tx [ i ] = HT_AGG_STATE_IDLE ;
sta - > ampdu_mlme . tid_tx [ i ] = NULL ;
sta - > ampdu_mlme . addba_req_num [ i ] = 0 ;
2007-12-25 17:00:34 +02:00
}
2007-05-05 11:45:53 -07:00
skb_queue_head_init ( & sta - > ps_tx_buf ) ;
skb_queue_head_init ( & sta - > tx_filtered ) ;
2008-02-25 16:27:47 +01:00
# ifdef CONFIG_MAC80211_VERBOSE_DEBUG
printk ( KERN_DEBUG " %s: Allocated STA %s \n " ,
wiphy_name ( local - > hw . wiphy ) , print_mac ( mbuf , sta - > addr ) ) ;
# endif /* CONFIG_MAC80211_VERBOSE_DEBUG */
2008-02-27 09:56:40 +01:00
# ifdef CONFIG_MAC80211_MESH
2008-02-29 15:46:08 -08:00
sta - > plink_state = PLINK_LISTEN ;
2008-02-27 09:56:40 +01:00
init_timer ( & sta - > plink_timer ) ;
# endif
2008-02-25 16:27:47 +01:00
return sta ;
}
int sta_info_insert ( struct sta_info * sta )
{
struct ieee80211_local * local = sta - > local ;
struct ieee80211_sub_if_data * sdata = sta - > sdata ;
unsigned long flags ;
2008-04-01 15:21:00 +02:00
int err = 0 ;
2008-02-25 16:27:47 +01:00
DECLARE_MAC_BUF ( mac ) ;
2008-02-27 09:56:40 +01:00
/*
* Can ' t be a WARN_ON because it can be triggered through a race :
* something inserts a STA ( on one CPU ) without holding the RTNL
* and another CPU turns off the net device .
*/
2008-04-01 15:21:00 +02:00
if ( unlikely ( ! netif_running ( sdata - > dev ) ) ) {
err = - ENETDOWN ;
goto out_free ;
}
2008-02-27 09:56:40 +01:00
2008-04-01 15:21:00 +02:00
if ( WARN_ON ( compare_ether_addr ( sta - > addr , sdata - > dev - > dev_addr ) = = 0 | |
is_multicast_ether_addr ( sta - > addr ) ) ) {
err = - EINVAL ;
goto out_free ;
}
2008-02-25 16:27:49 +01:00
2008-02-25 16:27:46 +01:00
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
2008-02-21 14:09:30 +01:00
/* check if STA exists already */
2008-02-25 16:27:47 +01:00
if ( __sta_info_find ( local , sta - > addr ) ) {
2008-02-25 16:27:46 +01:00
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
2008-04-01 15:21:00 +02:00
err = - EEXIST ;
goto out_free ;
2008-02-21 14:09:30 +01:00
}
2007-05-05 11:45:53 -07:00
list_add ( & sta - > list , & local - > sta_list ) ;
local - > num_sta + + ;
sta_info_hash_add ( local , sta ) ;
2007-12-19 01:31:26 +01:00
2008-02-25 16:27:46 +01:00
/* notify driver */
if ( local - > ops - > sta_notify ) {
2007-12-19 01:31:27 +01:00
if ( sdata - > vif . type = = IEEE80211_IF_TYPE_VLAN )
2007-12-19 01:31:26 +01:00
sdata = sdata - > u . vlan . ap ;
local - > ops - > sta_notify ( local_to_hw ( local ) , & sdata - > vif ,
2008-02-25 16:27:47 +01:00
STA_NOTIFY_ADD , sta - > addr ) ;
2007-12-19 01:31:26 +01:00
}
2008-02-25 16:27:46 +01:00
2007-05-05 11:45:53 -07:00
# ifdef CONFIG_MAC80211_VERBOSE_DEBUG
2008-02-25 16:27:47 +01:00
printk ( KERN_DEBUG " %s: Inserted STA %s \n " ,
wiphy_name ( local - > hw . wiphy ) , print_mac ( mac , sta - > addr ) ) ;
2007-05-05 11:45:53 -07:00
# endif /* CONFIG_MAC80211_VERBOSE_DEBUG */
2008-02-25 16:27:47 +01:00
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
2007-05-05 11:46:38 -07:00
# ifdef CONFIG_MAC80211_DEBUGFS
2008-04-01 15:21:00 +02:00
/*
* Debugfs entry adding might sleep , so schedule process
2007-07-27 15:43:23 +02:00
* context task for adding entry for STAs that do not yet
2008-04-01 15:21:00 +02:00
* have one .
* NOTE : due to auto - freeing semantics this may only be done
* if the insertion is successful !
*/
2008-04-03 14:31:05 +02:00
schedule_work ( & local - > sta_debugfs_add ) ;
2007-05-05 11:46:38 -07:00
# endif
2008-02-25 16:27:47 +01:00
if ( ieee80211_vif_is_mesh ( & sdata - > vif ) )
mesh_accept_plinks_update ( sdata ) ;
return 0 ;
2008-04-01 15:21:00 +02:00
out_free :
BUG_ON ( ! err ) ;
__sta_info_free ( local , sta ) ;
return err ;
2007-05-05 11:45:53 -07:00
}
2008-02-20 11:21:35 +01:00
static inline void __bss_tim_set ( struct ieee80211_if_ap * bss , u16 aid )
{
/*
* This format has been mandated by the IEEE specifications ,
* so this line may not be changed to use the __set_bit ( ) format .
*/
bss - > tim [ aid / 8 ] | = ( 1 < < ( aid % 8 ) ) ;
}
static inline void __bss_tim_clear ( struct ieee80211_if_ap * bss , u16 aid )
{
/*
* This format has been mandated by the IEEE specifications ,
* so this line may not be changed to use the __clear_bit ( ) format .
*/
bss - > tim [ aid / 8 ] & = ~ ( 1 < < ( aid % 8 ) ) ;
}
static void __sta_info_set_tim_bit ( struct ieee80211_if_ap * bss ,
struct sta_info * sta )
{
if ( bss )
__bss_tim_set ( bss , sta - > aid ) ;
2008-02-25 16:27:46 +01:00
if ( sta - > local - > ops - > set_tim ) {
sta - > local - > tim_in_locked_section = true ;
2008-02-20 11:21:35 +01:00
sta - > local - > ops - > set_tim ( local_to_hw ( sta - > local ) , sta - > aid , 1 ) ;
2008-02-25 16:27:46 +01:00
sta - > local - > tim_in_locked_section = false ;
}
2008-02-20 11:21:35 +01:00
}
void sta_info_set_tim_bit ( struct sta_info * sta )
{
2008-02-25 16:27:46 +01:00
unsigned long flags ;
2008-02-20 11:21:35 +01:00
2008-02-25 16:27:46 +01:00
spin_lock_irqsave ( & sta - > local - > sta_lock , flags ) ;
__sta_info_set_tim_bit ( sta - > sdata - > bss , sta ) ;
spin_unlock_irqrestore ( & sta - > local - > sta_lock , flags ) ;
2008-02-20 11:21:35 +01:00
}
static void __sta_info_clear_tim_bit ( struct ieee80211_if_ap * bss ,
struct sta_info * sta )
{
if ( bss )
__bss_tim_clear ( bss , sta - > aid ) ;
2008-02-25 16:27:46 +01:00
if ( sta - > local - > ops - > set_tim ) {
sta - > local - > tim_in_locked_section = true ;
2008-02-20 11:21:35 +01:00
sta - > local - > ops - > set_tim ( local_to_hw ( sta - > local ) , sta - > aid , 0 ) ;
2008-02-25 16:27:46 +01:00
sta - > local - > tim_in_locked_section = false ;
}
2008-02-20 11:21:35 +01:00
}
void sta_info_clear_tim_bit ( struct sta_info * sta )
{
2008-02-25 16:27:46 +01:00
unsigned long flags ;
2008-02-20 11:21:35 +01:00
2008-02-25 16:27:46 +01:00
spin_lock_irqsave ( & sta - > local - > sta_lock , flags ) ;
__sta_info_clear_tim_bit ( sta - > sdata - > bss , sta ) ;
spin_unlock_irqrestore ( & sta - > local - > sta_lock , flags ) ;
2008-02-20 11:21:35 +01:00
}
2008-03-31 15:21:23 -07:00
void __sta_info_unlink ( struct sta_info * * sta )
2007-05-05 11:45:53 -07:00
{
2008-02-25 16:27:46 +01:00
struct ieee80211_local * local = ( * sta ) - > local ;
struct ieee80211_sub_if_data * sdata = ( * sta ) - > sdata ;
# ifdef CONFIG_MAC80211_VERBOSE_DEBUG
DECLARE_MAC_BUF ( mbuf ) ;
# endif
/*
* pull caller ' s reference if we ' re already gone .
*/
if ( sta_info_hash_del ( local , * sta ) ) {
* sta = NULL ;
return ;
}
2007-07-27 15:43:23 +02:00
2008-04-08 17:56:52 +02:00
if ( ( * sta ) - > key ) {
ieee80211_key_free ( ( * sta ) - > key ) ;
WARN_ON ( ( * sta ) - > key ) ;
}
2008-04-08 13:08:20 +02:00
list_del ( & ( * sta ) - > list ) ;
2008-05-03 01:02:02 +02:00
if ( test_and_clear_sta_flags ( * sta , WLAN_STA_PS ) ) {
2008-04-08 13:08:20 +02:00
if ( sdata - > bss )
atomic_dec ( & sdata - > bss - > num_sta_ps ) ;
__sta_info_clear_tim_bit ( sdata - > bss , * sta ) ;
}
local - > num_sta - - ;
if ( local - > ops - > sta_notify ) {
if ( sdata - > vif . type = = IEEE80211_IF_TYPE_VLAN )
sdata = sdata - > u . vlan . ap ;
local - > ops - > sta_notify ( local_to_hw ( local ) , & sdata - > vif ,
STA_NOTIFY_REMOVE , ( * sta ) - > addr ) ;
}
if ( ieee80211_vif_is_mesh ( & sdata - > vif ) ) {
mesh_accept_plinks_update ( sdata ) ;
# ifdef CONFIG_MAC80211_MESH
del_timer ( & ( * sta ) - > plink_timer ) ;
# endif
}
# ifdef CONFIG_MAC80211_VERBOSE_DEBUG
printk ( KERN_DEBUG " %s: Removed STA %s \n " ,
wiphy_name ( local - > hw . wiphy ) , print_mac ( mbuf , ( * sta ) - > addr ) ) ;
# endif /* CONFIG_MAC80211_VERBOSE_DEBUG */
2008-02-25 16:27:46 +01:00
/*
2008-04-08 13:08:20 +02:00
* Finally , pull caller ' s reference if the STA is pinned by the
2008-02-25 16:27:46 +01:00
* task that is adding the debugfs entries . In that case , we
* leave the STA " to be freed " .
*
* The rules are not trivial , but not too complex either :
* ( 1 ) pin_status is only modified under the sta_lock
2008-04-03 14:31:05 +02:00
* ( 2 ) STAs may only be pinned under the RTNL so that
* sta_info_flush ( ) is guaranteed to actually destroy
* all STAs that are active for a given interface , this
* is required for correctness because otherwise we
* could notify a driver that an interface is going
* away and only after that ( ! ) notify it about a STA
* on that interface going away .
* ( 3 ) sta_info_debugfs_add_work ( ) will set the status
2008-02-25 16:27:46 +01:00
* to PINNED when it found an item that needs a new
* debugfs directory created . In that case , that item
* must not be freed although all * RCU * users are done
* with it . Hence , we tell the caller of _unlink ( )
* that the item is already gone ( as can happen when
* two tasks try to unlink / destroy at the same time )
2008-04-03 14:31:05 +02:00
* ( 4 ) We set the pin_status to DESTROY here when we
2008-02-25 16:27:46 +01:00
* find such an item .
2008-04-03 14:31:05 +02:00
* ( 5 ) sta_info_debugfs_add_work ( ) will reset the pin_status
2008-02-25 16:27:46 +01:00
* from PINNED to NORMAL when it is done with the item ,
* but will check for DESTROY before resetting it in
* which case it will free the item .
*/
if ( ( * sta ) - > pin_status = = STA_INFO_PIN_STAT_PINNED ) {
( * sta ) - > pin_status = STA_INFO_PIN_STAT_DESTROY ;
* sta = NULL ;
return ;
}
2007-05-05 11:45:53 -07:00
}
2008-02-25 16:27:46 +01:00
void sta_info_unlink ( struct sta_info * * sta )
{
struct ieee80211_local * local = ( * sta ) - > local ;
unsigned long flags ;
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
__sta_info_unlink ( sta ) ;
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
}
2007-05-05 11:45:53 -07:00
static inline int sta_info_buffer_expired ( struct ieee80211_local * local ,
struct sta_info * sta ,
struct sk_buff * skb )
{
2008-05-15 12:55:29 +02:00
struct ieee80211_tx_info * info ;
2007-05-05 11:45:53 -07:00
int timeout ;
if ( ! skb )
return 0 ;
2008-05-15 12:55:29 +02:00
info = IEEE80211_SKB_CB ( skb ) ;
2007-05-05 11:45:53 -07:00
/* Timeout: (2 * listen_interval * beacon_int * 1024 / 1000000) sec */
timeout = ( sta - > listen_interval * local - > hw . conf . beacon_int * 32 /
15625 ) * HZ ;
if ( timeout < STA_TX_BUFFER_EXPIRE )
timeout = STA_TX_BUFFER_EXPIRE ;
2008-05-15 12:55:29 +02:00
return time_after ( jiffies , info - > control . jiffies + timeout ) ;
2007-05-05 11:45:53 -07:00
}
static void sta_info_cleanup_expire_buffered ( struct ieee80211_local * local ,
struct sta_info * sta )
{
unsigned long flags ;
struct sk_buff * skb ;
2008-02-20 02:07:21 +01:00
struct ieee80211_sub_if_data * sdata ;
2007-10-03 17:59:30 -07:00
DECLARE_MAC_BUF ( mac ) ;
2007-05-05 11:45:53 -07:00
if ( skb_queue_empty ( & sta - > ps_tx_buf ) )
return ;
for ( ; ; ) {
spin_lock_irqsave ( & sta - > ps_tx_buf . lock , flags ) ;
skb = skb_peek ( & sta - > ps_tx_buf ) ;
2008-02-20 02:07:21 +01:00
if ( sta_info_buffer_expired ( local , sta , skb ) )
2007-05-05 11:45:53 -07:00
skb = __skb_dequeue ( & sta - > ps_tx_buf ) ;
2008-02-20 02:07:21 +01:00
else
2007-05-05 11:45:53 -07:00
skb = NULL ;
spin_unlock_irqrestore ( & sta - > ps_tx_buf . lock , flags ) ;
2008-02-20 02:07:21 +01:00
if ( ! skb )
2007-05-05 11:45:53 -07:00
break ;
2008-02-20 02:07:21 +01:00
2008-02-25 16:27:46 +01:00
sdata = sta - > sdata ;
2008-02-20 02:07:21 +01:00
local - > total_ps_buffered - - ;
printk ( KERN_DEBUG " Buffered frame expired (STA "
" %s) \n " , print_mac ( mac , sta - > addr ) ) ;
dev_kfree_skb ( skb ) ;
2008-02-20 11:21:35 +01:00
if ( skb_queue_empty ( & sta - > ps_tx_buf ) )
sta_info_clear_tim_bit ( sta ) ;
2007-05-05 11:45:53 -07:00
}
}
static void sta_info_cleanup ( unsigned long data )
{
struct ieee80211_local * local = ( struct ieee80211_local * ) data ;
struct sta_info * sta ;
2008-02-25 16:27:46 +01:00
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( sta , & local - > sta_list , list )
2007-05-05 11:45:53 -07:00
sta_info_cleanup_expire_buffered ( local , sta ) ;
2008-02-25 16:27:46 +01:00
rcu_read_unlock ( ) ;
2007-05-05 11:45:53 -07:00
2007-12-17 15:07:43 +01:00
local - > sta_cleanup . expires =
round_jiffies ( jiffies + STA_INFO_CLEANUP_INTERVAL ) ;
2007-05-05 11:45:53 -07:00
add_timer ( & local - > sta_cleanup ) ;
}
2007-05-05 11:46:38 -07:00
# ifdef CONFIG_MAC80211_DEBUGFS
2008-04-07 21:53:49 +02:00
/*
* See comment in __sta_info_unlink ,
* caller must hold local - > sta_lock .
*/
static void __sta_info_pin ( struct sta_info * sta )
{
WARN_ON ( sta - > pin_status ! = STA_INFO_PIN_STAT_NORMAL ) ;
sta - > pin_status = STA_INFO_PIN_STAT_PINNED ;
}
/*
* See comment in __sta_info_unlink , returns sta if it
* needs to be destroyed .
*/
static struct sta_info * __sta_info_unpin ( struct sta_info * sta )
{
struct sta_info * ret = NULL ;
unsigned long flags ;
spin_lock_irqsave ( & sta - > local - > sta_lock , flags ) ;
WARN_ON ( sta - > pin_status ! = STA_INFO_PIN_STAT_DESTROY & &
sta - > pin_status ! = STA_INFO_PIN_STAT_PINNED ) ;
if ( sta - > pin_status = = STA_INFO_PIN_STAT_DESTROY )
ret = sta ;
sta - > pin_status = STA_INFO_PIN_STAT_NORMAL ;
spin_unlock_irqrestore ( & sta - > local - > sta_lock , flags ) ;
return ret ;
}
2008-02-25 16:27:46 +01:00
static void sta_info_debugfs_add_work ( struct work_struct * work )
2007-05-05 11:46:38 -07:00
{
struct ieee80211_local * local =
container_of ( work , struct ieee80211_local , sta_debugfs_add ) ;
struct sta_info * sta , * tmp ;
2008-02-25 16:27:46 +01:00
unsigned long flags ;
2007-05-05 11:46:38 -07:00
2008-04-03 14:31:05 +02:00
/* We need to keep the RTNL across the whole pinned status. */
rtnl_lock ( ) ;
2007-05-05 11:46:38 -07:00
while ( 1 ) {
sta = NULL ;
2008-02-25 16:27:46 +01:00
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
2007-05-05 11:46:38 -07:00
list_for_each_entry ( tmp , & local - > sta_list , list ) {
2007-07-27 15:43:23 +02:00
if ( ! tmp - > debugfs . dir ) {
2007-05-05 11:46:38 -07:00
sta = tmp ;
2008-02-25 16:27:46 +01:00
__sta_info_pin ( sta ) ;
2007-05-05 11:46:38 -07:00
break ;
}
}
2008-02-25 16:27:46 +01:00
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
2007-05-05 11:46:38 -07:00
if ( ! sta )
break ;
ieee80211_sta_debugfs_add ( sta ) ;
rate_control_add_sta_debugfs ( sta ) ;
2008-02-25 16:27:46 +01:00
sta = __sta_info_unpin ( sta ) ;
2008-03-31 19:23:02 +02:00
sta_info_destroy ( sta ) ;
2007-05-05 11:46:38 -07:00
}
2008-04-03 14:31:05 +02:00
rtnl_unlock ( ) ;
2007-05-05 11:46:38 -07:00
}
# endif
2008-04-08 17:56:52 +02:00
static void __ieee80211_run_pending_flush ( struct ieee80211_local * local )
2008-03-31 19:23:03 +02:00
{
struct sta_info * sta ;
unsigned long flags ;
ASSERT_RTNL ( ) ;
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
while ( ! list_empty ( & local - > sta_flush_list ) ) {
sta = list_first_entry ( & local - > sta_flush_list ,
struct sta_info , list ) ;
list_del ( & sta - > list ) ;
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
sta_info_destroy ( sta ) ;
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
}
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
}
static void ieee80211_sta_flush_work ( struct work_struct * work )
{
struct ieee80211_local * local =
container_of ( work , struct ieee80211_local , sta_flush_work ) ;
rtnl_lock ( ) ;
__ieee80211_run_pending_flush ( local ) ;
rtnl_unlock ( ) ;
}
2007-05-05 11:45:53 -07:00
void sta_info_init ( struct ieee80211_local * local )
{
2008-02-25 16:27:46 +01:00
spin_lock_init ( & local - > sta_lock ) ;
2007-05-05 11:45:53 -07:00
INIT_LIST_HEAD ( & local - > sta_list ) ;
2008-03-31 19:23:03 +02:00
INIT_LIST_HEAD ( & local - > sta_flush_list ) ;
INIT_WORK ( & local - > sta_flush_work , ieee80211_sta_flush_work ) ;
2007-05-05 11:45:53 -07:00
2008-01-23 21:20:07 -08:00
setup_timer ( & local - > sta_cleanup , sta_info_cleanup ,
( unsigned long ) local ) ;
2007-12-17 15:07:43 +01:00
local - > sta_cleanup . expires =
round_jiffies ( jiffies + STA_INFO_CLEANUP_INTERVAL ) ;
2007-05-05 11:46:38 -07:00
# ifdef CONFIG_MAC80211_DEBUGFS
2008-02-25 16:27:46 +01:00
INIT_WORK ( & local - > sta_debugfs_add , sta_info_debugfs_add_work ) ;
2007-05-05 11:46:38 -07:00
# endif
2007-05-05 11:45:53 -07:00
}
int sta_info_start ( struct ieee80211_local * local )
{
add_timer ( & local - > sta_cleanup ) ;
return 0 ;
}
void sta_info_stop ( struct ieee80211_local * local )
{
del_timer ( & local - > sta_cleanup ) ;
2008-03-31 19:23:03 +02:00
cancel_work_sync ( & local - > sta_flush_work ) ;
2008-04-03 14:31:05 +02:00
# ifdef CONFIG_MAC80211_DEBUGFS
/*
* Make sure the debugfs adding work isn ' t pending after this
* because we ' re about to be destroyed . It doesn ' t matter
* whether it ran or not since we ' re going to flush all STAs
* anyway .
*/
cancel_work_sync ( & local - > sta_debugfs_add ) ;
# endif
2008-03-31 19:23:03 +02:00
rtnl_lock ( ) ;
2007-07-27 15:43:23 +02:00
sta_info_flush ( local , NULL ) ;
2008-03-31 19:23:03 +02:00
__ieee80211_run_pending_flush ( local ) ;
rtnl_unlock ( ) ;
2007-05-05 11:45:53 -07:00
}
/**
* sta_info_flush - flush matching STA entries from the STA table
2008-02-25 16:27:49 +01:00
*
* Returns the number of removed STA entries .
*
2007-05-05 11:45:53 -07:00
* @ local : local interface data
2008-02-25 16:27:46 +01:00
* @ sdata : matching rule for the net device ( sta - > dev ) or % NULL to match all STAs
2007-05-05 11:45:53 -07:00
*/
2008-02-25 16:27:49 +01:00
int sta_info_flush ( struct ieee80211_local * local ,
2008-02-25 16:27:46 +01:00
struct ieee80211_sub_if_data * sdata )
2007-05-05 11:45:53 -07:00
{
struct sta_info * sta , * tmp ;
2007-07-27 15:43:23 +02:00
LIST_HEAD ( tmp_list ) ;
2008-02-25 16:27:49 +01:00
int ret = 0 ;
2008-02-25 16:27:46 +01:00
unsigned long flags ;
2007-05-05 11:45:53 -07:00
2008-02-25 16:27:46 +01:00
might_sleep ( ) ;
2008-03-31 19:23:03 +02:00
ASSERT_RTNL ( ) ;
2007-07-27 15:43:23 +02:00
2008-02-25 16:27:46 +01:00
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
list_for_each_entry_safe ( sta , tmp , & local - > sta_list , list ) {
if ( ! sdata | | sdata = = sta - > sdata ) {
__sta_info_unlink ( & sta ) ;
2008-02-25 16:27:49 +01:00
if ( sta ) {
2008-02-25 16:27:46 +01:00
list_add_tail ( & sta - > list , & tmp_list ) ;
2008-02-25 16:27:49 +01:00
ret + + ;
}
2008-02-25 16:27:46 +01:00
}
2007-07-27 15:43:23 +02:00
}
2008-02-25 16:27:46 +01:00
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
list_for_each_entry_safe ( sta , tmp , & tmp_list , list )
sta_info_destroy ( sta ) ;
2008-02-25 16:27:49 +01:00
return ret ;
2007-05-05 11:45:53 -07:00
}
2008-03-31 19:23:03 +02:00
/**
* sta_info_flush_delayed - flush matching STA entries from the STA table
*
* This function unlinks all stations for a given interface and queues
* them for freeing . Note that the workqueue function scheduled here has
* to run before any new keys can be added to the system to avoid set_key ( )
* callback ordering issues .
*
* @ sdata : the interface
*/
void sta_info_flush_delayed ( struct ieee80211_sub_if_data * sdata )
{
struct ieee80211_local * local = sdata - > local ;
struct sta_info * sta , * tmp ;
unsigned long flags ;
bool work = false ;
spin_lock_irqsave ( & local - > sta_lock , flags ) ;
list_for_each_entry_safe ( sta , tmp , & local - > sta_list , list ) {
if ( sdata = = sta - > sdata ) {
__sta_info_unlink ( & sta ) ;
if ( sta ) {
list_add_tail ( & sta - > list ,
& local - > sta_flush_list ) ;
work = true ;
}
}
}
if ( work )
schedule_work ( & local - > sta_flush_work ) ;
spin_unlock_irqrestore ( & local - > sta_lock , flags ) ;
}