2008-02-23 17:17:09 +03:00
/*
2009-11-10 02:46:58 +03:00
* Copyright ( c ) 2008 , 2009 open80211s Ltd .
2008-02-23 17:17:09 +03:00
* Authors : Luis Carlos Cobo < luisca @ cozybit . com >
* Javier Cardona < javier @ cozybit . com >
*
* 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 .
*/
2008-04-23 23:15:29 +04:00
# include <asm/unaligned.h>
2008-02-23 17:17:09 +03:00
# include "ieee80211_i.h"
# include "mesh.h"
2008-09-11 02:01:49 +04:00
# define IEEE80211_MESH_PEER_INACTIVITY_LIMIT (1800 * HZ)
# define IEEE80211_MESH_HOUSEKEEPING_INTERVAL (60 * HZ)
2009-11-10 02:46:56 +03:00
# define IEEE80211_MESH_RANN_INTERVAL (1 * HZ)
2008-09-11 02:01:49 +04:00
2009-11-10 02:46:39 +03:00
# define MESHCONF_CAPAB_ACCEPT_PLINKS 0x01
# define MESHCONF_CAPAB_FORWARDING 0x08
2008-02-23 17:17:09 +03:00
2009-05-17 13:40:42 +04:00
# define TMR_RUNNING_HK 0
# define TMR_RUNNING_MP 1
2009-11-10 02:46:56 +03:00
# define TMR_RUNNING_MPR 2
2009-05-17 13:40:42 +04:00
2008-02-23 17:17:09 +03:00
int mesh_allocated ;
static struct kmem_cache * rm_cache ;
void ieee80211s_init ( void )
{
mesh_pathtbl_init ( ) ;
mesh_allocated = 1 ;
rm_cache = kmem_cache_create ( " mesh_rmc " , sizeof ( struct rmc_entry ) ,
0 , 0 , NULL ) ;
}
void ieee80211s_stop ( void )
{
mesh_pathtbl_unregister ( ) ;
kmem_cache_destroy ( rm_cache ) ;
}
2008-09-11 02:01:49 +04:00
static void ieee80211_mesh_housekeeping_timer ( unsigned long data )
{
struct ieee80211_sub_if_data * sdata = ( void * ) data ;
struct ieee80211_local * local = sdata - > local ;
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
2009-10-21 00:21:48 +04:00
set_bit ( MESH_WORK_HOUSEKEEPING , & ifmsh - > wrkq_flags ) ;
2009-05-17 13:40:42 +04:00
if ( local - > quiescing ) {
set_bit ( TMR_RUNNING_HK , & ifmsh - > timers_running ) ;
return ;
}
2009-08-05 02:06:26 +04:00
ieee80211_queue_work ( & local - > hw , & ifmsh - > work ) ;
2008-09-11 02:01:49 +04:00
}
2008-02-23 17:17:09 +03:00
/**
* mesh_matches_local - check if the config of a mesh point matches ours
*
* @ ie : information elements of a management frame from the mesh peer
2008-08-03 04:04:37 +04:00
* @ sdata : local mesh subif
2008-02-23 17:17:09 +03:00
*
* This function checks if the mesh configuration of a mesh point matches the
* local mesh configuration , i . e . if both nodes belong to the same mesh network .
*/
2008-08-03 04:04:37 +04:00
bool mesh_matches_local ( struct ieee802_11_elems * ie , struct ieee80211_sub_if_data * sdata )
2008-02-23 17:17:09 +03:00
{
2008-09-11 02:01:49 +04:00
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
2008-02-23 17:17:09 +03:00
/*
* As support for each feature is added , check for matching
* - On mesh config capabilities
* - Power Save Support En
* - Sync support enabled
* - Sync support active
* - Sync support required from peer
* - MDA enabled
* - Power management control on fc
*/
2008-09-11 02:01:49 +04:00
if ( ifmsh - > mesh_id_len = = ie - > mesh_id_len & &
memcmp ( ifmsh - > mesh_id , ie - > mesh_id , ie - > mesh_id_len ) = = 0 & &
2009-11-18 21:40:00 +03:00
( ifmsh - > mesh_pp_id = = ie - > mesh_config - > meshconf_psel ) & &
( ifmsh - > mesh_pm_id = = ie - > mesh_config - > meshconf_pmetric ) & &
( ifmsh - > mesh_cc_id = = ie - > mesh_config - > meshconf_congest ) & &
( ifmsh - > mesh_sp_id = = ie - > mesh_config - > meshconf_synch ) & &
( ifmsh - > mesh_auth_id = = ie - > mesh_config - > meshconf_auth ) )
2008-02-23 17:17:09 +03:00
return true ;
return false ;
}
/**
* mesh_peer_accepts_plinks - check if an mp is willing to establish peer links
*
* @ ie : information elements of a management frame from the mesh peer
*/
2008-08-03 04:04:37 +04:00
bool mesh_peer_accepts_plinks ( struct ieee802_11_elems * ie )
2008-02-23 17:17:09 +03:00
{
2009-11-18 21:40:00 +03:00
return ( ie - > mesh_config - > meshconf_cap &
2009-11-10 02:46:39 +03:00
MESHCONF_CAPAB_ACCEPT_PLINKS ) ! = 0 ;
2008-02-23 17:17:09 +03:00
}
/**
* mesh_accept_plinks_update : update accepting_plink in local mesh beacons
*
2008-02-25 18:27:46 +03:00
* @ sdata : mesh interface in which mesh beacons are going to be updated
2008-02-23 17:17:09 +03:00
*/
2008-02-25 18:27:46 +03:00
void mesh_accept_plinks_update ( struct ieee80211_sub_if_data * sdata )
2008-02-23 17:17:09 +03:00
{
bool free_plinks ;
/* In case mesh_plink_free_count > 0 and mesh_plinktbl_capacity == 0,
* the mesh interface might be able to establish plinks with peers that
2008-03-01 02:46:08 +03:00
* are already on the table but are not on PLINK_ESTAB state . However ,
* in general the mesh interface is not accepting peer link requests
* from new peers , and that must be reflected in the beacon
2008-02-23 17:17:09 +03:00
*/
free_plinks = mesh_plink_availables ( sdata ) ;
2008-09-11 02:01:49 +04:00
if ( free_plinks ! = sdata - > u . mesh . accepting_plinks )
ieee80211_mesh_housekeeping_timer ( ( unsigned long ) sdata ) ;
2008-02-23 17:17:09 +03:00
}
2008-09-11 02:01:49 +04:00
void mesh_ids_set_default ( struct ieee80211_if_mesh * sta )
2008-02-23 17:17:09 +03:00
{
2009-11-10 02:46:39 +03:00
sta - > mesh_pp_id = 0 ; /* HWMP */
sta - > mesh_pm_id = 0 ; /* Airtime */
sta - > mesh_cc_id = 0 ; /* Disabled */
sta - > mesh_sp_id = 0 ; /* Neighbor Offset */
sta - > mesh_auth_id = 0 ; /* Disabled */
2008-02-23 17:17:09 +03:00
}
2008-08-03 04:04:37 +04:00
int mesh_rmc_init ( struct ieee80211_sub_if_data * sdata )
2008-02-23 17:17:09 +03:00
{
int i ;
2008-09-11 02:01:49 +04:00
sdata - > u . mesh . rmc = kmalloc ( sizeof ( struct mesh_rmc ) , GFP_KERNEL ) ;
if ( ! sdata - > u . mesh . rmc )
2008-02-23 17:17:09 +03:00
return - ENOMEM ;
2008-09-11 02:01:49 +04:00
sdata - > u . mesh . rmc - > idx_mask = RMC_BUCKETS - 1 ;
2008-02-23 17:17:09 +03:00
for ( i = 0 ; i < RMC_BUCKETS ; i + + )
2008-09-11 02:01:49 +04:00
INIT_LIST_HEAD ( & sdata - > u . mesh . rmc - > bucket [ i ] . list ) ;
2008-02-23 17:17:09 +03:00
return 0 ;
}
2008-08-03 04:04:37 +04:00
void mesh_rmc_free ( struct ieee80211_sub_if_data * sdata )
2008-02-23 17:17:09 +03:00
{
2008-09-11 02:01:49 +04:00
struct mesh_rmc * rmc = sdata - > u . mesh . rmc ;
2008-02-23 17:17:09 +03:00
struct rmc_entry * p , * n ;
int i ;
2008-09-11 02:01:49 +04:00
if ( ! sdata - > u . mesh . rmc )
2008-02-23 17:17:09 +03:00
return ;
for ( i = 0 ; i < RMC_BUCKETS ; i + + )
list_for_each_entry_safe ( p , n , & rmc - > bucket [ i ] . list , list ) {
list_del ( & p - > list ) ;
kmem_cache_free ( rm_cache , p ) ;
}
kfree ( rmc ) ;
2008-09-11 02:01:49 +04:00
sdata - > u . mesh . rmc = NULL ;
2008-02-23 17:17:09 +03:00
}
/**
* mesh_rmc_check - Check frame in recent multicast cache and add if absent .
*
* @ sa : source address
* @ mesh_hdr : mesh_header
*
* Returns : 0 if the frame is not in the cache , nonzero otherwise .
*
* Checks using the source address and the mesh sequence number if we have
* received this frame lately . If the frame is not in the cache , it is added to
* it .
*/
int mesh_rmc_check ( u8 * sa , struct ieee80211s_hdr * mesh_hdr ,
2008-08-03 04:04:37 +04:00
struct ieee80211_sub_if_data * sdata )
2008-02-23 17:17:09 +03:00
{
2008-09-11 02:01:49 +04:00
struct mesh_rmc * rmc = sdata - > u . mesh . rmc ;
2008-02-23 17:17:09 +03:00
u32 seqnum = 0 ;
int entries = 0 ;
u8 idx ;
struct rmc_entry * p , * n ;
/* Don't care about endianness since only match matters */
2008-04-23 23:15:29 +04:00
memcpy ( & seqnum , & mesh_hdr - > seqnum , sizeof ( mesh_hdr - > seqnum ) ) ;
idx = le32_to_cpu ( mesh_hdr - > seqnum ) & rmc - > idx_mask ;
2008-02-23 17:17:09 +03:00
list_for_each_entry_safe ( p , n , & rmc - > bucket [ idx ] . list , list ) {
+ + entries ;
if ( time_after ( jiffies , p - > exp_time ) | |
( entries = = RMC_QUEUE_MAX_LEN ) ) {
list_del ( & p - > list ) ;
kmem_cache_free ( rm_cache , p ) ;
- - entries ;
} else if ( ( seqnum = = p - > seqnum )
& & ( memcmp ( sa , p - > sa , ETH_ALEN ) = = 0 ) )
return - 1 ;
}
p = kmem_cache_alloc ( rm_cache , GFP_ATOMIC ) ;
if ( ! p ) {
printk ( KERN_DEBUG " o11s: could not allocate RMC entry \n " ) ;
return 0 ;
}
p - > seqnum = seqnum ;
p - > exp_time = jiffies + RMC_TIMEOUT ;
memcpy ( p - > sa , sa , ETH_ALEN ) ;
list_add ( & p - > list , & rmc - > bucket [ idx ] . list ) ;
return 0 ;
}
2008-08-03 04:04:37 +04:00
void mesh_mgmt_ies_add ( struct sk_buff * skb , struct ieee80211_sub_if_data * sdata )
2008-02-23 17:17:09 +03:00
{
2008-08-03 04:04:37 +04:00
struct ieee80211_local * local = sdata - > local ;
2008-02-23 17:17:09 +03:00
struct ieee80211_supported_band * sband ;
u8 * pos ;
int len , i , rate ;
2009-11-10 02:46:41 +03:00
u8 neighbors ;
2008-02-23 17:17:09 +03:00
sband = local - > hw . wiphy - > bands [ local - > hw . conf . channel - > band ] ;
len = sband - > n_bitrates ;
if ( len > 8 )
len = 8 ;
pos = skb_put ( skb , len + 2 ) ;
* pos + + = WLAN_EID_SUPP_RATES ;
* pos + + = len ;
for ( i = 0 ; i < len ; i + + ) {
rate = sband - > bitrates [ i ] . bitrate ;
* pos + + = ( u8 ) ( rate / 5 ) ;
}
if ( sband - > n_bitrates > len ) {
pos = skb_put ( skb , sband - > n_bitrates - len + 2 ) ;
* pos + + = WLAN_EID_EXT_SUPP_RATES ;
* pos + + = sband - > n_bitrates - len ;
for ( i = len ; i < sband - > n_bitrates ; i + + ) {
rate = sband - > bitrates [ i ] . bitrate ;
* pos + + = ( u8 ) ( rate / 5 ) ;
}
}
2009-11-10 02:46:54 +03:00
if ( sband - > band = = IEEE80211_BAND_2GHZ ) {
pos = skb_put ( skb , 2 + 1 ) ;
* pos + + = WLAN_EID_DS_PARAMS ;
* pos + + = 1 ;
* pos + + = ieee80211_frequency_to_channel ( local - > hw . conf . channel - > center_freq ) ;
}
2008-09-11 02:01:49 +04:00
pos = skb_put ( skb , 2 + sdata - > u . mesh . mesh_id_len ) ;
2008-02-23 17:17:09 +03:00
* pos + + = WLAN_EID_MESH_ID ;
2008-09-11 02:01:49 +04:00
* pos + + = sdata - > u . mesh . mesh_id_len ;
if ( sdata - > u . mesh . mesh_id_len )
memcpy ( pos , sdata - > u . mesh . mesh_id , sdata - > u . mesh . mesh_id_len ) ;
2008-02-23 17:17:09 +03:00
2009-11-18 21:40:00 +03:00
pos = skb_put ( skb , 2 + sizeof ( struct ieee80211_meshconf_ie ) ) ;
2008-02-23 17:17:09 +03:00
* pos + + = WLAN_EID_MESH_CONFIG ;
2009-11-18 21:40:00 +03:00
* pos + + = sizeof ( struct ieee80211_meshconf_ie ) ;
2008-02-23 17:17:09 +03:00
/* Active path selection protocol ID */
2009-11-10 02:46:39 +03:00
* pos + + = sdata - > u . mesh . mesh_pp_id ;
2008-02-23 17:17:09 +03:00
/* Active path selection metric ID */
2009-11-10 02:46:39 +03:00
* pos + + = sdata - > u . mesh . mesh_pm_id ;
2008-02-23 17:17:09 +03:00
/* Congestion control mode identifier */
2009-11-10 02:46:39 +03:00
* pos + + = sdata - > u . mesh . mesh_cc_id ;
2008-02-23 17:17:09 +03:00
2009-08-20 20:21:45 +04:00
/* Synchronization protocol identifier */
2009-11-10 02:46:39 +03:00
* pos + + = sdata - > u . mesh . mesh_sp_id ;
2008-02-23 17:17:09 +03:00
2009-08-20 20:21:45 +04:00
/* Authentication Protocol identifier */
2009-11-10 02:46:39 +03:00
* pos + + = sdata - > u . mesh . mesh_auth_id ;
2009-08-20 20:21:45 +04:00
2009-11-10 02:46:41 +03:00
/* Mesh Formation Info - number of neighbors */
neighbors = atomic_read ( & sdata - > u . mesh . mshstats . estab_plinks ) ;
/* Number of neighbor mesh STAs or 15 whichever is smaller */
neighbors = ( neighbors > 15 ) ? 15 : neighbors ;
* pos + + = neighbors < < 1 ;
2009-08-20 20:21:45 +04:00
2008-02-23 17:17:09 +03:00
/* Mesh capability */
2008-09-11 02:01:49 +04:00
sdata - > u . mesh . accepting_plinks = mesh_plink_availables ( sdata ) ;
2009-11-10 02:46:39 +03:00
* pos = MESHCONF_CAPAB_FORWARDING ;
* pos + + | = sdata - > u . mesh . accepting_plinks ?
MESHCONF_CAPAB_ACCEPT_PLINKS : 0x00 ;
2008-02-23 17:17:09 +03:00
* pos + + = 0x00 ;
return ;
}
2008-08-03 04:04:37 +04:00
u32 mesh_table_hash ( u8 * addr , struct ieee80211_sub_if_data * sdata , struct mesh_table * tbl )
2008-02-23 17:17:09 +03:00
{
/* Use last four bytes of hw addr and interface index as hash index */
2008-08-03 04:04:37 +04:00
return jhash_2words ( * ( u32 * ) ( addr + 2 ) , sdata - > dev - > ifindex , tbl - > hash_rnd )
2008-02-23 17:17:09 +03:00
& tbl - > hash_mask ;
}
struct mesh_table * mesh_table_alloc ( int size_order )
{
int i ;
struct mesh_table * newtbl ;
newtbl = kmalloc ( sizeof ( struct mesh_table ) , GFP_KERNEL ) ;
if ( ! newtbl )
return NULL ;
newtbl - > hash_buckets = kzalloc ( sizeof ( struct hlist_head ) *
( 1 < < size_order ) , GFP_KERNEL ) ;
if ( ! newtbl - > hash_buckets ) {
kfree ( newtbl ) ;
return NULL ;
}
newtbl - > hashwlock = kmalloc ( sizeof ( spinlock_t ) *
( 1 < < size_order ) , GFP_KERNEL ) ;
if ( ! newtbl - > hashwlock ) {
kfree ( newtbl - > hash_buckets ) ;
kfree ( newtbl ) ;
return NULL ;
}
newtbl - > size_order = size_order ;
newtbl - > hash_mask = ( 1 < < size_order ) - 1 ;
atomic_set ( & newtbl - > entries , 0 ) ;
get_random_bytes ( & newtbl - > hash_rnd ,
sizeof ( newtbl - > hash_rnd ) ) ;
for ( i = 0 ; i < = newtbl - > hash_mask ; i + + )
spin_lock_init ( & newtbl - > hashwlock [ i ] ) ;
return newtbl ;
}
static void ieee80211_mesh_path_timer ( unsigned long data )
{
struct ieee80211_sub_if_data * sdata =
( struct ieee80211_sub_if_data * ) data ;
2008-09-11 02:01:49 +04:00
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
2008-09-16 16:18:59 +04:00
struct ieee80211_local * local = sdata - > local ;
2008-02-23 17:17:09 +03:00
2009-05-17 13:40:42 +04:00
if ( local - > quiescing ) {
set_bit ( TMR_RUNNING_MP , & ifmsh - > timers_running ) ;
return ;
}
2009-08-05 02:06:26 +04:00
ieee80211_queue_work ( & local - > hw , & ifmsh - > work ) ;
2008-02-23 17:17:09 +03:00
}
2009-11-10 02:46:56 +03:00
static void ieee80211_mesh_path_root_timer ( unsigned long data )
{
struct ieee80211_sub_if_data * sdata =
( struct ieee80211_sub_if_data * ) data ;
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
struct ieee80211_local * local = sdata - > local ;
set_bit ( MESH_WORK_ROOT , & ifmsh - > wrkq_flags ) ;
if ( local - > quiescing ) {
set_bit ( TMR_RUNNING_MPR , & ifmsh - > timers_running ) ;
return ;
}
ieee80211_queue_work ( & local - > hw , & ifmsh - > work ) ;
}
2009-11-10 02:46:57 +03:00
void ieee80211_mesh_root_setup ( struct ieee80211_if_mesh * ifmsh )
{
if ( ifmsh - > mshcfg . dot11MeshHWMPRootMode )
set_bit ( MESH_WORK_ROOT , & ifmsh - > wrkq_flags ) ;
else {
clear_bit ( MESH_WORK_ROOT , & ifmsh - > wrkq_flags ) ;
/* stop running timer */
del_timer_sync ( & ifmsh - > mesh_path_root_timer ) ;
}
}
2009-08-10 23:15:48 +04:00
/**
* ieee80211_fill_mesh_addresses - fill addresses of a locally originated mesh frame
* @ hdr : 802.11 frame header
* @ fc : frame control field
* @ meshda : destination address in the mesh
* @ meshsa : source address address in the mesh . Same as TA , as frame is
* locally originated .
*
* Return the length of the 802.11 ( does not include a mesh control header )
*/
2009-11-17 15:34:04 +03:00
int ieee80211_fill_mesh_addresses ( struct ieee80211_hdr * hdr , __le16 * fc ,
const u8 * meshda , const u8 * meshsa )
{
2009-08-10 23:15:48 +04:00
if ( is_multicast_ether_addr ( meshda ) ) {
* fc | = cpu_to_le16 ( IEEE80211_FCTL_FROMDS ) ;
/* DA TA SA */
memcpy ( hdr - > addr1 , meshda , ETH_ALEN ) ;
memcpy ( hdr - > addr2 , meshsa , ETH_ALEN ) ;
memcpy ( hdr - > addr3 , meshsa , ETH_ALEN ) ;
return 24 ;
} else {
* fc | = cpu_to_le16 ( IEEE80211_FCTL_FROMDS |
IEEE80211_FCTL_TODS ) ;
/* RA TA DA SA */
memset ( hdr - > addr1 , 0 , ETH_ALEN ) ; /* RA is resolved later */
memcpy ( hdr - > addr2 , meshsa , ETH_ALEN ) ;
memcpy ( hdr - > addr3 , meshda , ETH_ALEN ) ;
memcpy ( hdr - > addr4 , meshsa , ETH_ALEN ) ;
return 30 ;
}
}
2008-02-23 17:17:19 +03:00
/**
* ieee80211_new_mesh_header - create a new mesh header
* @ meshhdr : uninitialized mesh header
* @ sdata : mesh interface to be used
2009-08-10 23:15:48 +04:00
* @ addr4 : addr4 of the mesh frame ( 1 st in ae header )
* may be NULL
* @ addr5 : addr5 of the mesh frame ( 1 st or 2 nd in ae header )
* may be NULL unless addr6 is present
* @ addr6 : addr6 of the mesh frame ( 2 nd or 3 rd in ae header )
* may be NULL unless addr5 is present
2008-02-23 17:17:19 +03:00
*
* Return the header length .
*/
int ieee80211_new_mesh_header ( struct ieee80211s_hdr * meshhdr ,
2009-08-10 23:15:48 +04:00
struct ieee80211_sub_if_data * sdata , char * addr4 ,
char * addr5 , char * addr6 )
2008-02-23 17:17:19 +03:00
{
2009-08-10 23:15:48 +04:00
int aelen = 0 ;
memset ( meshhdr , 0 , sizeof ( meshhdr ) ) ;
2008-09-11 02:01:49 +04:00
meshhdr - > ttl = sdata - > u . mesh . mshcfg . dot11MeshTTL ;
put_unaligned ( cpu_to_le32 ( sdata - > u . mesh . mesh_seqnum ) , & meshhdr - > seqnum ) ;
sdata - > u . mesh . mesh_seqnum + + ;
2009-08-10 23:15:48 +04:00
if ( addr4 ) {
meshhdr - > flags | = MESH_FLAGS_AE_A4 ;
aelen + = ETH_ALEN ;
memcpy ( meshhdr - > eaddr1 , addr4 , ETH_ALEN ) ;
}
if ( addr5 & & addr6 ) {
meshhdr - > flags | = MESH_FLAGS_AE_A5_A6 ;
aelen + = 2 * ETH_ALEN ;
if ( ! addr4 ) {
memcpy ( meshhdr - > eaddr1 , addr5 , ETH_ALEN ) ;
memcpy ( meshhdr - > eaddr2 , addr6 , ETH_ALEN ) ;
} else {
memcpy ( meshhdr - > eaddr2 , addr5 , ETH_ALEN ) ;
memcpy ( meshhdr - > eaddr3 , addr6 , ETH_ALEN ) ;
}
}
return 6 + aelen ;
2008-02-23 17:17:19 +03:00
}
2008-09-11 02:01:49 +04:00
static void ieee80211_mesh_housekeeping ( struct ieee80211_sub_if_data * sdata ,
struct ieee80211_if_mesh * ifmsh )
{
bool free_plinks ;
# ifdef CONFIG_MAC80211_VERBOSE_DEBUG
printk ( KERN_DEBUG " %s: running mesh housekeeping \n " ,
sdata - > dev - > name ) ;
# endif
ieee80211_sta_expire ( sdata , IEEE80211_MESH_PEER_INACTIVITY_LIMIT ) ;
mesh_path_expire ( sdata ) ;
free_plinks = mesh_plink_availables ( sdata ) ;
if ( free_plinks ! = sdata - > u . mesh . accepting_plinks )
2009-04-23 18:13:26 +04:00
ieee80211_bss_info_change_notify ( sdata , BSS_CHANGED_BEACON ) ;
2008-09-11 02:01:49 +04:00
mod_timer ( & ifmsh - > housekeeping_timer ,
round_jiffies ( jiffies + IEEE80211_MESH_HOUSEKEEPING_INTERVAL ) ) ;
}
2009-11-10 02:46:56 +03:00
static void ieee80211_mesh_rootpath ( struct ieee80211_sub_if_data * sdata )
{
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
mesh_path_tx_root_frame ( sdata ) ;
mod_timer ( & ifmsh - > mesh_path_root_timer ,
round_jiffies ( jiffies + IEEE80211_MESH_RANN_INTERVAL ) ) ;
}
2009-05-17 13:40:42 +04:00
# ifdef CONFIG_PM
void ieee80211_mesh_quiesce ( struct ieee80211_sub_if_data * sdata )
{
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
/* might restart the timer but that doesn't matter */
cancel_work_sync ( & ifmsh - > work ) ;
/* use atomic bitops in case both timers fire at the same time */
if ( del_timer_sync ( & ifmsh - > housekeeping_timer ) )
set_bit ( TMR_RUNNING_HK , & ifmsh - > timers_running ) ;
if ( del_timer_sync ( & ifmsh - > mesh_path_timer ) )
set_bit ( TMR_RUNNING_MP , & ifmsh - > timers_running ) ;
2009-11-10 02:46:56 +03:00
if ( del_timer_sync ( & ifmsh - > mesh_path_root_timer ) )
set_bit ( TMR_RUNNING_MPR , & ifmsh - > timers_running ) ;
2009-05-17 13:40:42 +04:00
}
void ieee80211_mesh_restart ( struct ieee80211_sub_if_data * sdata )
{
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
if ( test_and_clear_bit ( TMR_RUNNING_HK , & ifmsh - > timers_running ) )
add_timer ( & ifmsh - > housekeeping_timer ) ;
if ( test_and_clear_bit ( TMR_RUNNING_MP , & ifmsh - > timers_running ) )
add_timer ( & ifmsh - > mesh_path_timer ) ;
2009-11-10 02:46:56 +03:00
if ( test_and_clear_bit ( TMR_RUNNING_MPR , & ifmsh - > timers_running ) )
add_timer ( & ifmsh - > mesh_path_root_timer ) ;
2009-11-10 02:46:57 +03:00
ieee80211_mesh_root_setup ( ifmsh ) ;
2009-05-17 13:40:42 +04:00
}
# endif
2008-09-11 02:01:49 +04:00
void ieee80211_start_mesh ( struct ieee80211_sub_if_data * sdata )
{
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
struct ieee80211_local * local = sdata - > local ;
2009-10-21 00:21:48 +04:00
set_bit ( MESH_WORK_HOUSEKEEPING , & ifmsh - > wrkq_flags ) ;
2009-11-10 02:46:57 +03:00
ieee80211_mesh_root_setup ( ifmsh ) ;
2009-08-10 23:15:52 +04:00
ieee80211_queue_work ( & local - > hw , & ifmsh - > work ) ;
2009-08-10 23:15:51 +04:00
sdata - > vif . bss_conf . beacon_int = MESH_DEFAULT_BEACON_INTERVAL ;
2009-04-23 18:13:26 +04:00
ieee80211_bss_info_change_notify ( sdata , BSS_CHANGED_BEACON |
2009-08-10 23:15:51 +04:00
BSS_CHANGED_BEACON_ENABLED |
BSS_CHANGED_BEACON_INT ) ;
2008-09-11 02:01:49 +04:00
}
void ieee80211_stop_mesh ( struct ieee80211_sub_if_data * sdata )
{
del_timer_sync ( & sdata - > u . mesh . housekeeping_timer ) ;
2009-11-10 02:46:56 +03:00
del_timer_sync ( & sdata - > u . mesh . mesh_path_root_timer ) ;
2008-09-11 02:01:50 +04:00
/*
* If the timer fired while we waited for it , it will have
* requeued the work . Now the work will be running again
* but will not rearm the timer again because it checks
* whether the interface is running , which , at this point ,
* it no longer is .
*/
cancel_work_sync ( & sdata - > u . mesh . work ) ;
2008-09-11 02:01:49 +04:00
/*
* When we get here , the interface is marked down .
* Call synchronize_rcu ( ) to wait for the RX path
* should it be using the interface and enqueuing
* frames at this very time on another CPU .
*/
2009-06-26 14:45:53 +04:00
rcu_barrier ( ) ; /* Wait for RX path and call_rcu()'s */
2008-09-11 02:01:49 +04:00
skb_queue_purge ( & sdata - > u . mesh . skb_queue ) ;
}
static void ieee80211_mesh_rx_bcn_presp ( struct ieee80211_sub_if_data * sdata ,
u16 stype ,
struct ieee80211_mgmt * mgmt ,
size_t len ,
struct ieee80211_rx_status * rx_status )
{
2008-10-07 14:04:32 +04:00
struct ieee80211_local * local = sdata - > local ;
2008-09-11 02:01:49 +04:00
struct ieee802_11_elems elems ;
struct ieee80211_channel * channel ;
2009-01-21 17:13:48 +03:00
u32 supp_rates = 0 ;
2008-09-11 02:01:49 +04:00
size_t baselen ;
int freq ;
enum ieee80211_band band = rx_status - > band ;
/* ignore ProbeResp to foreign address */
if ( stype = = IEEE80211_STYPE_PROBE_RESP & &
compare_ether_addr ( mgmt - > da , sdata - > dev - > dev_addr ) )
return ;
baselen = ( u8 * ) mgmt - > u . probe_resp . variable - ( u8 * ) mgmt ;
if ( baselen > len )
return ;
ieee802_11_parse_elems ( mgmt - > u . probe_resp . variable , len - baselen ,
& elems ) ;
if ( elems . ds_params & & elems . ds_params_len = = 1 )
freq = ieee80211_channel_to_frequency ( elems . ds_params [ 0 ] ) ;
else
freq = rx_status - > freq ;
channel = ieee80211_get_channel ( local - > hw . wiphy , freq ) ;
if ( ! channel | | channel - > flags & IEEE80211_CHAN_DISABLED )
return ;
if ( elems . mesh_id & & elems . mesh_config & &
mesh_matches_local ( & elems , sdata ) ) {
supp_rates = ieee80211_sta_get_rates ( local , & elems , band ) ;
mesh_neighbour_update ( mgmt - > sa , supp_rates , sdata ,
mesh_peer_accepts_plinks ( & elems ) ) ;
}
}
static void ieee80211_mesh_rx_mgmt_action ( struct ieee80211_sub_if_data * sdata ,
struct ieee80211_mgmt * mgmt ,
size_t len ,
struct ieee80211_rx_status * rx_status )
{
switch ( mgmt - > u . action . category ) {
2009-11-10 02:46:43 +03:00
case MESH_PLINK_CATEGORY :
2008-09-11 02:01:49 +04:00
mesh_rx_plink_frame ( sdata , mgmt , len , rx_status ) ;
break ;
case MESH_PATH_SEL_CATEGORY :
mesh_rx_path_sel_frame ( sdata , mgmt , len ) ;
break ;
}
}
static void ieee80211_mesh_rx_queued_mgmt ( struct ieee80211_sub_if_data * sdata ,
struct sk_buff * skb )
{
struct ieee80211_rx_status * rx_status ;
struct ieee80211_if_mesh * ifmsh ;
struct ieee80211_mgmt * mgmt ;
u16 stype ;
ifmsh = & sdata - > u . mesh ;
2009-06-17 15:13:00 +04:00
rx_status = IEEE80211_SKB_RXCB ( skb ) ;
2008-09-11 02:01:49 +04:00
mgmt = ( struct ieee80211_mgmt * ) skb - > data ;
stype = le16_to_cpu ( mgmt - > frame_control ) & IEEE80211_FCTL_STYPE ;
switch ( stype ) {
case IEEE80211_STYPE_PROBE_RESP :
case IEEE80211_STYPE_BEACON :
ieee80211_mesh_rx_bcn_presp ( sdata , stype , mgmt , skb - > len ,
rx_status ) ;
break ;
case IEEE80211_STYPE_ACTION :
ieee80211_mesh_rx_mgmt_action ( sdata , mgmt , skb - > len , rx_status ) ;
break ;
}
kfree_skb ( skb ) ;
}
static void ieee80211_mesh_work ( struct work_struct * work )
{
struct ieee80211_sub_if_data * sdata =
container_of ( work , struct ieee80211_sub_if_data , u . mesh . work ) ;
struct ieee80211_local * local = sdata - > local ;
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
struct sk_buff * skb ;
if ( ! netif_running ( sdata - > dev ) )
return ;
2009-07-23 14:14:04 +04:00
if ( local - > scanning )
2008-09-11 02:01:49 +04:00
return ;
while ( ( skb = skb_dequeue ( & ifmsh - > skb_queue ) ) )
ieee80211_mesh_rx_queued_mgmt ( sdata , skb ) ;
if ( ifmsh - > preq_queue_len & &
time_after ( jiffies ,
ifmsh - > last_preq + msecs_to_jiffies ( ifmsh - > mshcfg . dot11MeshHWMPpreqMinInterval ) ) )
mesh_path_start_discovery ( sdata ) ;
2009-08-10 23:15:52 +04:00
if ( test_and_clear_bit ( MESH_WORK_GROW_MPATH_TABLE , & ifmsh - > wrkq_flags ) )
mesh_mpath_table_grow ( ) ;
if ( test_and_clear_bit ( MESH_WORK_GROW_MPATH_TABLE , & ifmsh - > wrkq_flags ) )
mesh_mpp_table_grow ( ) ;
if ( test_and_clear_bit ( MESH_WORK_HOUSEKEEPING , & ifmsh - > wrkq_flags ) )
2008-09-11 02:01:49 +04:00
ieee80211_mesh_housekeeping ( sdata , ifmsh ) ;
2009-11-10 02:46:56 +03:00
if ( test_and_clear_bit ( MESH_WORK_ROOT , & ifmsh - > wrkq_flags ) )
ieee80211_mesh_rootpath ( sdata ) ;
2008-09-11 02:01:49 +04:00
}
void ieee80211_mesh_notify_scan_completed ( struct ieee80211_local * local )
{
struct ieee80211_sub_if_data * sdata ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( sdata , & local - > interfaces , list )
if ( ieee80211_vif_is_mesh ( & sdata - > vif ) )
2009-08-05 02:06:26 +04:00
ieee80211_queue_work ( & local - > hw , & sdata - > u . mesh . work ) ;
2008-09-11 02:01:49 +04:00
rcu_read_unlock ( ) ;
}
2008-02-23 17:17:19 +03:00
void ieee80211_mesh_init_sdata ( struct ieee80211_sub_if_data * sdata )
{
2008-09-11 02:01:49 +04:00
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
INIT_WORK ( & ifmsh - > work , ieee80211_mesh_work ) ;
setup_timer ( & ifmsh - > housekeeping_timer ,
ieee80211_mesh_housekeeping_timer ,
( unsigned long ) sdata ) ;
skb_queue_head_init ( & sdata - > u . mesh . skb_queue ) ;
ifmsh - > mshcfg . dot11MeshRetryTimeout = MESH_RET_T ;
ifmsh - > mshcfg . dot11MeshConfirmTimeout = MESH_CONF_T ;
ifmsh - > mshcfg . dot11MeshHoldingTimeout = MESH_HOLD_T ;
ifmsh - > mshcfg . dot11MeshMaxRetries = MESH_MAX_RETR ;
ifmsh - > mshcfg . dot11MeshTTL = MESH_TTL ;
ifmsh - > mshcfg . auto_open_plinks = true ;
ifmsh - > mshcfg . dot11MeshMaxPeerLinks =
2008-02-23 17:17:19 +03:00
MESH_MAX_ESTAB_PLINKS ;
2008-09-11 02:01:49 +04:00
ifmsh - > mshcfg . dot11MeshHWMPactivePathTimeout =
2008-02-23 17:17:19 +03:00
MESH_PATH_TIMEOUT ;
2008-09-11 02:01:49 +04:00
ifmsh - > mshcfg . dot11MeshHWMPpreqMinInterval =
2008-02-23 17:17:19 +03:00
MESH_PREQ_MIN_INT ;
2008-09-11 02:01:49 +04:00
ifmsh - > mshcfg . dot11MeshHWMPnetDiameterTraversalTime =
2008-02-23 17:17:19 +03:00
MESH_DIAM_TRAVERSAL_TIME ;
2008-09-11 02:01:49 +04:00
ifmsh - > mshcfg . dot11MeshHWMPmaxPREQretries =
2008-02-23 17:17:19 +03:00
MESH_MAX_PREQ_RETRIES ;
2008-09-11 02:01:49 +04:00
ifmsh - > mshcfg . path_refresh_time =
2008-02-23 17:17:19 +03:00
MESH_PATH_REFRESH_TIME ;
2008-09-11 02:01:49 +04:00
ifmsh - > mshcfg . min_discovery_timeout =
2008-02-23 17:17:19 +03:00
MESH_MIN_DISCOVERY_TIMEOUT ;
2008-09-11 02:01:49 +04:00
ifmsh - > accepting_plinks = true ;
ifmsh - > preq_id = 0 ;
2009-11-10 02:46:55 +03:00
ifmsh - > sn = 0 ;
2008-09-11 02:01:49 +04:00
atomic_set ( & ifmsh - > mpaths , 0 ) ;
2008-08-03 04:04:37 +04:00
mesh_rmc_init ( sdata ) ;
2008-09-11 02:01:49 +04:00
ifmsh - > last_preq = jiffies ;
2008-02-23 17:17:19 +03:00
/* Allocate all mesh structures when creating the first mesh interface. */
if ( ! mesh_allocated )
ieee80211s_init ( ) ;
2008-09-11 02:01:49 +04:00
mesh_ids_set_default ( ifmsh ) ;
setup_timer ( & ifmsh - > mesh_path_timer ,
2008-02-23 17:17:19 +03:00
ieee80211_mesh_path_timer ,
( unsigned long ) sdata ) ;
2009-11-10 02:46:56 +03:00
setup_timer ( & ifmsh - > mesh_path_root_timer ,
ieee80211_mesh_path_root_timer ,
( unsigned long ) sdata ) ;
2008-09-11 02:01:49 +04:00
INIT_LIST_HEAD ( & ifmsh - > preq_queue . list ) ;
spin_lock_init ( & ifmsh - > mesh_preq_queue_lock ) ;
}
ieee80211_rx_result
2009-06-17 15:13:00 +04:00
ieee80211_mesh_rx_mgmt ( struct ieee80211_sub_if_data * sdata , struct sk_buff * skb )
2008-09-11 02:01:49 +04:00
{
struct ieee80211_local * local = sdata - > local ;
struct ieee80211_if_mesh * ifmsh = & sdata - > u . mesh ;
struct ieee80211_mgmt * mgmt ;
u16 fc ;
if ( skb - > len < 24 )
return RX_DROP_MONITOR ;
mgmt = ( struct ieee80211_mgmt * ) skb - > data ;
fc = le16_to_cpu ( mgmt - > frame_control ) ;
switch ( fc & IEEE80211_FCTL_STYPE ) {
2009-07-10 13:39:26 +04:00
case IEEE80211_STYPE_ACTION :
if ( skb - > len < IEEE80211_MIN_ACTION_SIZE )
return RX_DROP_MONITOR ;
/* fall through */
2008-09-11 02:01:49 +04:00
case IEEE80211_STYPE_PROBE_RESP :
case IEEE80211_STYPE_BEACON :
skb_queue_tail ( & ifmsh - > skb_queue , skb ) ;
2009-08-05 02:06:26 +04:00
ieee80211_queue_work ( & local - > hw , & ifmsh - > work ) ;
2008-09-11 02:01:49 +04:00
return RX_QUEUED ;
}
return RX_CONTINUE ;
2008-02-23 17:17:19 +03:00
}