2012-05-12 02:09:43 +02:00
/* Copyright (C) 2009-2012 B.A.T.M.A.N. contributors:
2010-12-13 11:19:28 +00:00
*
* Marek Lindner , Simon Wunderlich
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of version 2 of the GNU General Public
* License as published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA
* 02110 - 1301 , USA
*/
# include "main.h"
# include "originator.h"
# include "hash.h"
# include "translation-table.h"
# include "routing.h"
# include "gateway_client.h"
# include "hard-interface.h"
# include "unicast.h"
# include "soft-interface.h"
2012-01-22 20:00:19 +01:00
# include "bridge_loop_avoidance.h"
2010-12-13 11:19:28 +00:00
2012-05-12 18:34:00 +02:00
static void batadv_purge_orig ( struct work_struct * work ) ;
2010-12-13 11:19:28 +00:00
2012-05-12 18:34:00 +02:00
static void batadv_start_purge_timer ( struct bat_priv * bat_priv )
2010-12-13 11:19:28 +00:00
{
2012-05-12 18:34:00 +02:00
INIT_DELAYED_WORK ( & bat_priv - > orig_work , batadv_purge_orig ) ;
2012-05-12 02:09:42 +02:00
queue_delayed_work ( batadv_event_workqueue ,
2012-03-01 15:35:20 +08:00
& bat_priv - > orig_work , msecs_to_jiffies ( 1000 ) ) ;
2010-12-13 11:19:28 +00:00
}
2011-06-15 15:08:59 +02:00
/* returns 1 if they are the same originator */
2012-05-12 18:34:00 +02:00
static int batadv_compare_orig ( const struct hlist_node * node , const void * data2 )
2011-06-15 15:08:59 +02:00
{
const void * data1 = container_of ( node , struct orig_node , hash_entry ) ;
return ( memcmp ( data1 , data2 , ETH_ALEN ) = = 0 ? 1 : 0 ) ;
}
2012-05-12 02:09:34 +02:00
int batadv_originator_init ( struct bat_priv * bat_priv )
2010-12-13 11:19:28 +00:00
{
if ( bat_priv - > orig_hash )
2012-05-05 13:27:28 +02:00
return 0 ;
2010-12-13 11:19:28 +00:00
2012-05-12 02:09:32 +02:00
bat_priv - > orig_hash = batadv_hash_new ( 1024 ) ;
2010-12-13 11:19:28 +00:00
if ( ! bat_priv - > orig_hash )
goto err ;
2012-05-12 18:34:00 +02:00
batadv_start_purge_timer ( bat_priv ) ;
2012-05-05 13:27:28 +02:00
return 0 ;
2010-12-13 11:19:28 +00:00
err :
2012-05-05 13:27:28 +02:00
return - ENOMEM ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 02:09:34 +02:00
void batadv_neigh_node_free_ref ( struct neigh_node * neigh_node )
2011-01-19 20:01:43 +00:00
{
2011-02-10 14:33:53 +00:00
if ( atomic_dec_and_test ( & neigh_node - > refcount ) )
2011-05-01 23:27:50 -07:00
kfree_rcu ( neigh_node , rcu ) ;
2011-01-19 20:01:43 +00:00
}
2011-03-14 22:43:37 +00:00
/* increases the refcounter of a found router */
2012-05-12 02:09:34 +02:00
struct neigh_node * batadv_orig_node_get_router ( struct orig_node * orig_node )
2011-03-14 22:43:37 +00:00
{
struct neigh_node * router ;
rcu_read_lock ( ) ;
router = rcu_dereference ( orig_node - > router ) ;
if ( router & & ! atomic_inc_not_zero ( & router - > refcount ) )
router = NULL ;
rcu_read_unlock ( ) ;
return router ;
}
2012-03-01 15:35:21 +08:00
struct neigh_node * batadv_neigh_node_new ( struct hard_iface * hard_iface ,
const uint8_t * neigh_addr ,
uint32_t seqno )
2010-12-13 11:19:28 +00:00
{
2012-03-01 15:35:21 +08:00
struct bat_priv * bat_priv = netdev_priv ( hard_iface - > soft_iface ) ;
2010-12-13 11:19:28 +00:00
struct neigh_node * neigh_node ;
2011-05-14 23:14:54 +02:00
neigh_node = kzalloc ( sizeof ( * neigh_node ) , GFP_ATOMIC ) ;
2010-12-13 11:19:28 +00:00
if ( ! neigh_node )
2012-03-01 15:35:21 +08:00
goto out ;
2010-12-13 11:19:28 +00:00
2010-12-12 21:57:11 +00:00
INIT_HLIST_NODE ( & neigh_node - > list ) ;
2010-12-13 11:19:28 +00:00
2012-03-01 15:35:21 +08:00
memcpy ( neigh_node - > addr , neigh_addr , ETH_ALEN ) ;
2012-03-17 15:28:32 +08:00
spin_lock_init ( & neigh_node - > lq_update_lock ) ;
2011-02-18 12:28:11 +00:00
/* extra reference for return */
atomic_set ( & neigh_node - > refcount , 2 ) ;
2010-12-13 11:19:28 +00:00
2012-05-12 13:48:58 +02:00
batadv_dbg ( DBG_BATMAN , bat_priv ,
" Creating new neighbor %pM, initial seqno %d \n " ,
neigh_addr , seqno ) ;
2012-03-01 15:35:21 +08:00
out :
2010-12-13 11:19:28 +00:00
return neigh_node ;
}
2012-05-12 18:34:00 +02:00
static void batadv_orig_node_free_rcu ( struct rcu_head * rcu )
2010-12-13 11:19:28 +00:00
{
2010-12-12 21:57:11 +00:00
struct hlist_node * node , * node_tmp ;
2011-01-19 20:01:43 +00:00
struct neigh_node * neigh_node , * tmp_neigh_node ;
2011-01-19 20:01:42 +00:00
struct orig_node * orig_node ;
2011-02-18 12:28:10 +00:00
orig_node = container_of ( rcu , struct orig_node , rcu ) ;
2010-12-13 11:19:28 +00:00
2010-12-12 21:57:12 +00:00
spin_lock_bh ( & orig_node - > neigh_list_lock ) ;
2011-01-19 20:01:43 +00:00
/* for all bonding members ... */
list_for_each_entry_safe ( neigh_node , tmp_neigh_node ,
& orig_node - > bond_list , bonding_list ) {
list_del_rcu ( & neigh_node - > bonding_list ) ;
2012-05-12 02:09:34 +02:00
batadv_neigh_node_free_ref ( neigh_node ) ;
2011-01-19 20:01:43 +00:00
}
2010-12-13 11:19:28 +00:00
/* for all neighbors towards this originator ... */
2010-12-12 21:57:11 +00:00
hlist_for_each_entry_safe ( neigh_node , node , node_tmp ,
& orig_node - > neigh_list , list ) {
2010-12-12 21:57:12 +00:00
hlist_del_rcu ( & neigh_node - > list ) ;
2012-05-12 02:09:34 +02:00
batadv_neigh_node_free_ref ( neigh_node ) ;
2010-12-13 11:19:28 +00:00
}
2010-12-12 21:57:12 +00:00
spin_unlock_bh ( & orig_node - > neigh_list_lock ) ;
2012-05-12 02:09:40 +02:00
batadv_frag_list_free ( & orig_node - > frag_list ) ;
2012-05-12 02:09:39 +02:00
batadv_tt_global_del_orig ( orig_node - > bat_priv , orig_node ,
" originator timed out " ) ;
2010-12-13 11:19:28 +00:00
2011-04-27 14:27:44 +02:00
kfree ( orig_node - > tt_buff ) ;
2010-12-13 11:19:28 +00:00
kfree ( orig_node - > bcast_own ) ;
kfree ( orig_node - > bcast_own_sum ) ;
kfree ( orig_node ) ;
}
2012-05-12 02:09:34 +02:00
void batadv_orig_node_free_ref ( struct orig_node * orig_node )
2011-02-18 12:28:10 +00:00
{
if ( atomic_dec_and_test ( & orig_node - > refcount ) )
2012-05-12 18:34:00 +02:00
call_rcu ( & orig_node - > rcu , batadv_orig_node_free_rcu ) ;
2011-02-18 12:28:10 +00:00
}
2012-05-12 02:09:34 +02:00
void batadv_originator_free ( struct bat_priv * bat_priv )
2010-12-13 11:19:28 +00:00
{
2011-01-19 20:01:42 +00:00
struct hashtable_t * hash = bat_priv - > orig_hash ;
2011-02-18 12:28:09 +00:00
struct hlist_node * node , * node_tmp ;
2011-01-19 20:01:42 +00:00
struct hlist_head * head ;
spinlock_t * list_lock ; /* spinlock to protect write access */
struct orig_node * orig_node ;
2011-10-05 17:05:25 +02:00
uint32_t i ;
2011-01-19 20:01:42 +00:00
if ( ! hash )
2010-12-13 11:19:28 +00:00
return ;
cancel_delayed_work_sync ( & bat_priv - > orig_work ) ;
bat_priv - > orig_hash = NULL ;
2011-01-19 20:01:42 +00:00
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
list_lock = & hash - > list_locks [ i ] ;
spin_lock_bh ( list_lock ) ;
2011-02-18 12:28:09 +00:00
hlist_for_each_entry_safe ( orig_node , node , node_tmp ,
head , hash_entry ) {
2011-01-19 20:01:42 +00:00
2011-02-18 12:28:09 +00:00
hlist_del_rcu ( node ) ;
2012-05-12 02:09:34 +02:00
batadv_orig_node_free_ref ( orig_node ) ;
2011-01-19 20:01:42 +00:00
}
spin_unlock_bh ( list_lock ) ;
}
2012-05-12 02:09:32 +02:00
batadv_hash_destroy ( hash ) ;
2010-12-13 11:19:28 +00:00
}
/* this function finds or creates an originator entry for the given
2012-05-12 02:09:43 +02:00
* address if it does not exits
*/
2012-05-12 02:09:34 +02:00
struct orig_node * batadv_get_orig_node ( struct bat_priv * bat_priv ,
const uint8_t * addr )
2010-12-13 11:19:28 +00:00
{
struct orig_node * orig_node ;
int size ;
int hash_added ;
2012-06-03 22:19:17 +02:00
unsigned long reset_time ;
2010-12-13 11:19:28 +00:00
2012-05-12 13:48:56 +02:00
orig_node = batadv_orig_hash_find ( bat_priv , addr ) ;
2011-02-18 12:28:09 +00:00
if ( orig_node )
2010-12-13 11:19:28 +00:00
return orig_node ;
2012-05-12 13:48:58 +02:00
batadv_dbg ( DBG_BATMAN , bat_priv , " Creating new originator: %pM \n " ,
addr ) ;
2010-12-13 11:19:28 +00:00
2011-05-14 23:14:54 +02:00
orig_node = kzalloc ( sizeof ( * orig_node ) , GFP_ATOMIC ) ;
2010-12-13 11:19:28 +00:00
if ( ! orig_node )
return NULL ;
2010-12-12 21:57:11 +00:00
INIT_HLIST_HEAD ( & orig_node - > neigh_list ) ;
2011-01-19 20:01:43 +00:00
INIT_LIST_HEAD ( & orig_node - > bond_list ) ;
2011-01-19 20:01:42 +00:00
spin_lock_init ( & orig_node - > ogm_cnt_lock ) ;
2011-01-25 21:52:11 +00:00
spin_lock_init ( & orig_node - > bcast_seqno_lock ) ;
2010-12-12 21:57:12 +00:00
spin_lock_init ( & orig_node - > neigh_list_lock ) ;
2011-04-27 14:27:44 +02:00
spin_lock_init ( & orig_node - > tt_buff_lock ) ;
2011-02-18 12:28:10 +00:00
/* extra reference for return */
atomic_set ( & orig_node - > refcount , 2 ) ;
2010-12-13 11:19:28 +00:00
2011-11-07 16:36:40 +01:00
orig_node - > tt_initialised = false ;
2011-04-27 14:27:57 +02:00
orig_node - > tt_poss_change = false ;
2011-01-19 20:01:42 +00:00
orig_node - > bat_priv = bat_priv ;
2010-12-13 11:19:28 +00:00
memcpy ( orig_node - > orig , addr , ETH_ALEN ) ;
orig_node - > router = NULL ;
2011-07-07 01:40:57 +02:00
orig_node - > tt_crc = 0 ;
atomic_set ( & orig_node - > last_ttvn , 0 ) ;
2011-05-05 08:42:45 +02:00
orig_node - > tt_buff = NULL ;
2011-04-27 14:27:44 +02:00
orig_node - > tt_buff_len = 0 ;
atomic_set ( & orig_node - > tt_size , 0 ) ;
2012-06-03 22:19:17 +02:00
reset_time = jiffies - 1 - msecs_to_jiffies ( BATADV_RESET_PROTECTION_MS ) ;
orig_node - > bcast_seqno_reset = reset_time ;
orig_node - > batman_seqno_reset = reset_time ;
2010-12-13 11:19:28 +00:00
2011-01-19 20:01:43 +00:00
atomic_set ( & orig_node - > bond_candidates , 0 ) ;
2012-06-03 22:19:17 +02:00
size = bat_priv - > num_ifaces * sizeof ( unsigned long ) * BATADV_NUM_WORDS ;
2010-12-13 11:19:28 +00:00
orig_node - > bcast_own = kzalloc ( size , GFP_ATOMIC ) ;
if ( ! orig_node - > bcast_own )
goto free_orig_node ;
size = bat_priv - > num_ifaces * sizeof ( uint8_t ) ;
orig_node - > bcast_own_sum = kzalloc ( size , GFP_ATOMIC ) ;
INIT_LIST_HEAD ( & orig_node - > frag_list ) ;
orig_node - > last_frag_packet = 0 ;
if ( ! orig_node - > bcast_own_sum )
goto free_bcast_own ;
2012-05-12 18:34:00 +02:00
hash_added = batadv_hash_add ( bat_priv - > orig_hash , batadv_compare_orig ,
2012-05-12 13:48:56 +02:00
batadv_choose_orig , orig_node ,
2012-05-12 13:48:55 +02:00
& orig_node - > hash_entry ) ;
2011-07-10 00:36:36 +02:00
if ( hash_added ! = 0 )
2010-12-13 11:19:28 +00:00
goto free_bcast_own_sum ;
return orig_node ;
free_bcast_own_sum :
kfree ( orig_node - > bcast_own_sum ) ;
free_bcast_own :
kfree ( orig_node - > bcast_own ) ;
free_orig_node :
kfree ( orig_node ) ;
return NULL ;
}
2012-05-12 18:34:00 +02:00
static bool batadv_purge_orig_neighbors ( struct bat_priv * bat_priv ,
struct orig_node * orig_node ,
struct neigh_node * * best_neigh_node )
2010-12-13 11:19:28 +00:00
{
2010-12-12 21:57:11 +00:00
struct hlist_node * node , * node_tmp ;
2010-12-13 11:19:28 +00:00
struct neigh_node * neigh_node ;
bool neigh_purged = false ;
2012-03-01 15:35:20 +08:00
unsigned long last_seen ;
2012-05-12 13:48:58 +02:00
struct hard_iface * if_incoming ;
2010-12-13 11:19:28 +00:00
* best_neigh_node = NULL ;
2010-12-12 21:57:12 +00:00
spin_lock_bh ( & orig_node - > neigh_list_lock ) ;
2010-12-13 11:19:28 +00:00
/* for all neighbors towards this originator ... */
2010-12-12 21:57:11 +00:00
hlist_for_each_entry_safe ( neigh_node , node , node_tmp ,
& orig_node - > neigh_list , list ) {
2010-12-13 11:19:28 +00:00
2012-05-12 13:48:58 +02:00
last_seen = neigh_node - > last_seen ;
if_incoming = neigh_node - > if_incoming ;
2012-06-03 22:19:17 +02:00
if ( ( batadv_has_timed_out ( last_seen , BATADV_PURGE_TIMEOUT ) ) | |
2012-05-12 13:48:58 +02:00
( if_incoming - > if_status = = IF_INACTIVE ) | |
( if_incoming - > if_status = = IF_NOT_IN_USE ) | |
( if_incoming - > if_status = = IF_TO_BE_REMOVED ) ) {
if ( ( if_incoming - > if_status = = IF_INACTIVE ) | |
( if_incoming - > if_status = = IF_NOT_IN_USE ) | |
( if_incoming - > if_status = = IF_TO_BE_REMOVED ) )
batadv_dbg ( DBG_BATMAN , bat_priv ,
" neighbor purge: originator %pM, neighbor: %pM, iface: %s \n " ,
orig_node - > orig , neigh_node - > addr ,
if_incoming - > net_dev - > name ) ;
2010-12-13 11:19:28 +00:00
else
2012-05-12 13:48:58 +02:00
batadv_dbg ( DBG_BATMAN , bat_priv ,
" neighbor timeout: originator %pM, neighbor: %pM, last_seen: %u \n " ,
orig_node - > orig , neigh_node - > addr ,
jiffies_to_msecs ( last_seen ) ) ;
2010-12-13 11:19:28 +00:00
neigh_purged = true ;
2010-12-12 21:57:11 +00:00
2010-12-12 21:57:12 +00:00
hlist_del_rcu ( & neigh_node - > list ) ;
2012-05-12 02:09:36 +02:00
batadv_bonding_candidate_del ( orig_node , neigh_node ) ;
2012-05-12 02:09:34 +02:00
batadv_neigh_node_free_ref ( neigh_node ) ;
2010-12-13 11:19:28 +00:00
} else {
if ( ( ! * best_neigh_node ) | |
( neigh_node - > tq_avg > ( * best_neigh_node ) - > tq_avg ) )
* best_neigh_node = neigh_node ;
}
}
2010-12-12 21:57:12 +00:00
spin_unlock_bh ( & orig_node - > neigh_list_lock ) ;
2010-12-13 11:19:28 +00:00
return neigh_purged ;
}
2012-05-12 18:34:00 +02:00
static bool batadv_purge_orig_node ( struct bat_priv * bat_priv ,
struct orig_node * orig_node )
2010-12-13 11:19:28 +00:00
{
struct neigh_node * best_neigh_node ;
2012-06-03 22:19:17 +02:00
if ( batadv_has_timed_out ( orig_node - > last_seen ,
2 * BATADV_PURGE_TIMEOUT ) ) {
2012-05-12 13:48:58 +02:00
batadv_dbg ( DBG_BATMAN , bat_priv ,
" Originator timeout: originator %pM, last_seen %u \n " ,
orig_node - > orig ,
jiffies_to_msecs ( orig_node - > last_seen ) ) ;
2010-12-13 11:19:28 +00:00
return true ;
} else {
2012-05-12 18:34:00 +02:00
if ( batadv_purge_orig_neighbors ( bat_priv , orig_node ,
& best_neigh_node ) )
2012-05-12 02:09:36 +02:00
batadv_update_route ( bat_priv , orig_node ,
best_neigh_node ) ;
2010-12-13 11:19:28 +00:00
}
return false ;
}
2012-05-12 18:34:00 +02:00
static void _batadv_purge_orig ( struct bat_priv * bat_priv )
2010-12-13 11:19:28 +00:00
{
struct hashtable_t * hash = bat_priv - > orig_hash ;
2011-02-18 12:28:09 +00:00
struct hlist_node * node , * node_tmp ;
2010-12-13 11:19:28 +00:00
struct hlist_head * head ;
2011-01-19 20:01:40 +00:00
spinlock_t * list_lock ; /* spinlock to protect write access */
2010-12-13 11:19:28 +00:00
struct orig_node * orig_node ;
2011-10-05 17:05:25 +02:00
uint32_t i ;
2010-12-13 11:19:28 +00:00
if ( ! hash )
return ;
/* for all origins... */
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
2011-01-19 20:01:40 +00:00
list_lock = & hash - > list_locks [ i ] ;
2010-12-13 11:19:28 +00:00
2011-01-19 20:01:40 +00:00
spin_lock_bh ( list_lock ) ;
2011-02-18 12:28:09 +00:00
hlist_for_each_entry_safe ( orig_node , node , node_tmp ,
head , hash_entry ) {
2012-05-12 18:34:00 +02:00
if ( batadv_purge_orig_node ( bat_priv , orig_node ) ) {
2010-12-13 11:19:28 +00:00
if ( orig_node - > gw_flags )
2012-05-12 02:09:29 +02:00
batadv_gw_node_delete ( bat_priv ,
orig_node ) ;
2011-02-18 12:28:09 +00:00
hlist_del_rcu ( node ) ;
2012-05-12 02:09:34 +02:00
batadv_orig_node_free_ref ( orig_node ) ;
2011-01-19 20:01:40 +00:00
continue ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 13:48:58 +02:00
if ( batadv_has_timed_out ( orig_node - > last_frag_packet ,
2012-06-03 22:19:15 +02:00
BATADV_FRAG_TIMEOUT ) )
2012-05-12 02:09:40 +02:00
batadv_frag_list_free ( & orig_node - > frag_list ) ;
2010-12-13 11:19:28 +00:00
}
2011-01-19 20:01:40 +00:00
spin_unlock_bh ( list_lock ) ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 02:09:29 +02:00
batadv_gw_node_purge ( bat_priv ) ;
batadv_gw_election ( bat_priv ) ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 18:34:00 +02:00
static void batadv_purge_orig ( struct work_struct * work )
2010-12-13 11:19:28 +00:00
{
struct delayed_work * delayed_work =
container_of ( work , struct delayed_work , work ) ;
struct bat_priv * bat_priv =
container_of ( delayed_work , struct bat_priv , orig_work ) ;
2012-05-12 18:34:00 +02:00
_batadv_purge_orig ( bat_priv ) ;
batadv_start_purge_timer ( bat_priv ) ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 02:09:34 +02:00
void batadv_purge_orig_ref ( struct bat_priv * bat_priv )
2010-12-13 11:19:28 +00:00
{
2012-05-12 18:34:00 +02:00
_batadv_purge_orig ( bat_priv ) ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 02:09:34 +02:00
int batadv_orig_seq_print_text ( struct seq_file * seq , void * offset )
2010-12-13 11:19:28 +00:00
{
struct net_device * net_dev = ( struct net_device * ) seq - > private ;
struct bat_priv * bat_priv = netdev_priv ( net_dev ) ;
struct hashtable_t * hash = bat_priv - > orig_hash ;
2011-02-18 12:28:09 +00:00
struct hlist_node * node , * node_tmp ;
2010-12-13 11:19:28 +00:00
struct hlist_head * head ;
2011-04-20 15:40:58 +02:00
struct hard_iface * primary_if ;
2010-12-13 11:19:28 +00:00
struct orig_node * orig_node ;
2011-03-14 22:43:37 +00:00
struct neigh_node * neigh_node , * neigh_node_tmp ;
2010-12-13 11:19:28 +00:00
int batman_count = 0 ;
int last_seen_secs ;
int last_seen_msecs ;
2011-10-05 17:05:25 +02:00
uint32_t i ;
int ret = 0 ;
2011-04-20 15:40:58 +02:00
2012-05-12 13:48:54 +02:00
primary_if = batadv_primary_if_get_selected ( bat_priv ) ;
2010-12-13 11:19:28 +00:00
2011-04-20 15:40:58 +02:00
if ( ! primary_if ) {
2012-03-07 09:07:45 +01:00
ret = seq_printf ( seq ,
" BATMAN mesh %s disabled - please specify interfaces to enable it \n " ,
2011-04-20 15:40:58 +02:00
net_dev - > name ) ;
goto out ;
}
2010-12-13 11:19:28 +00:00
2011-04-20 15:40:58 +02:00
if ( primary_if - > if_status ! = IF_ACTIVE ) {
2012-03-07 09:07:45 +01:00
ret = seq_printf ( seq ,
" BATMAN mesh %s disabled - primary interface not active \n " ,
2011-04-20 15:40:58 +02:00
net_dev - > name ) ;
goto out ;
2010-12-13 11:19:28 +00:00
}
2011-07-05 10:42:51 +02:00
seq_printf ( seq , " [B.A.T.M.A.N. adv %s, MainIF/MAC: %s/%pM (%s)] \n " ,
2012-06-03 22:19:17 +02:00
BATADV_SOURCE_VERSION , primary_if - > net_dev - > name ,
2011-04-20 15:40:58 +02:00
primary_if - > net_dev - > dev_addr , net_dev - > name ) ;
2010-12-13 11:19:28 +00:00
seq_printf ( seq , " %-15s %s (%s/%i) %17s [%10s]: %20s ... \n " ,
2012-06-03 22:19:17 +02:00
" Originator " , " last-seen " , " # " , BATADV_TQ_MAX_VALUE ,
" Nexthop " , " outgoingIF " , " Potential nexthops " ) ;
2010-12-13 11:19:28 +00:00
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
2011-01-19 20:01:40 +00:00
rcu_read_lock ( ) ;
2011-02-18 12:28:09 +00:00
hlist_for_each_entry_rcu ( orig_node , node , head , hash_entry ) {
2012-05-12 02:09:34 +02:00
neigh_node = batadv_orig_node_get_router ( orig_node ) ;
2011-03-14 22:43:37 +00:00
if ( ! neigh_node )
2010-12-13 11:19:28 +00:00
continue ;
2011-03-14 22:43:37 +00:00
if ( neigh_node - > tq_avg = = 0 )
goto next ;
2010-12-13 11:19:28 +00:00
last_seen_secs = jiffies_to_msecs ( jiffies -
2012-03-01 15:35:19 +08:00
orig_node - > last_seen ) / 1000 ;
2010-12-13 11:19:28 +00:00
last_seen_msecs = jiffies_to_msecs ( jiffies -
2012-03-01 15:35:19 +08:00
orig_node - > last_seen ) % 1000 ;
2010-12-13 11:19:28 +00:00
seq_printf ( seq , " %pM %4i.%03is (%3i) %pM [%10s]: " ,
orig_node - > orig , last_seen_secs ,
last_seen_msecs , neigh_node - > tq_avg ,
neigh_node - > addr ,
neigh_node - > if_incoming - > net_dev - > name ) ;
2011-03-14 22:43:37 +00:00
hlist_for_each_entry_rcu ( neigh_node_tmp , node_tmp ,
2010-12-12 21:57:12 +00:00
& orig_node - > neigh_list , list ) {
2011-03-14 22:43:37 +00:00
seq_printf ( seq , " %pM (%3i) " ,
neigh_node_tmp - > addr ,
neigh_node_tmp - > tq_avg ) ;
2010-12-13 11:19:28 +00:00
}
seq_printf ( seq , " \n " ) ;
batman_count + + ;
2011-03-14 22:43:37 +00:00
next :
2012-05-12 02:09:34 +02:00
batadv_neigh_node_free_ref ( neigh_node ) ;
2010-12-13 11:19:28 +00:00
}
2011-01-19 20:01:40 +00:00
rcu_read_unlock ( ) ;
2010-12-13 11:19:28 +00:00
}
2011-03-14 22:43:37 +00:00
if ( batman_count = = 0 )
2010-12-13 11:19:28 +00:00
seq_printf ( seq , " No batman nodes in range ... \n " ) ;
2011-04-20 15:40:58 +02:00
out :
if ( primary_if )
2012-05-12 13:48:54 +02:00
batadv_hardif_free_ref ( primary_if ) ;
2011-04-20 15:40:58 +02:00
return ret ;
2010-12-13 11:19:28 +00:00
}
2012-05-12 18:34:00 +02:00
static int batadv_orig_node_add_if ( struct orig_node * orig_node , int max_if_num )
2010-12-13 11:19:28 +00:00
{
void * data_ptr ;
2012-06-03 22:19:17 +02:00
size_t data_size , old_size ;
2010-12-13 11:19:28 +00:00
2012-06-03 22:19:17 +02:00
data_size = max_if_num * sizeof ( unsigned long ) * BATADV_NUM_WORDS ;
old_size = ( max_if_num - 1 ) * sizeof ( unsigned long ) * BATADV_NUM_WORDS ;
data_ptr = kmalloc ( data_size , GFP_ATOMIC ) ;
2011-08-29 14:17:24 -07:00
if ( ! data_ptr )
2012-05-05 13:27:28 +02:00
return - ENOMEM ;
2010-12-13 11:19:28 +00:00
2012-06-03 22:19:17 +02:00
memcpy ( data_ptr , orig_node - > bcast_own , old_size ) ;
2010-12-13 11:19:28 +00:00
kfree ( orig_node - > bcast_own ) ;
orig_node - > bcast_own = data_ptr ;
data_ptr = kmalloc ( max_if_num * sizeof ( uint8_t ) , GFP_ATOMIC ) ;
2011-08-29 14:17:24 -07:00
if ( ! data_ptr )
2012-05-05 13:27:28 +02:00
return - ENOMEM ;
2010-12-13 11:19:28 +00:00
memcpy ( data_ptr , orig_node - > bcast_own_sum ,
( max_if_num - 1 ) * sizeof ( uint8_t ) ) ;
kfree ( orig_node - > bcast_own_sum ) ;
orig_node - > bcast_own_sum = data_ptr ;
return 0 ;
}
2012-05-12 02:09:34 +02:00
int batadv_orig_hash_add_if ( struct hard_iface * hard_iface , int max_if_num )
2010-12-13 11:19:28 +00:00
{
2011-02-18 12:33:20 +00:00
struct bat_priv * bat_priv = netdev_priv ( hard_iface - > soft_iface ) ;
2010-12-13 11:19:28 +00:00
struct hashtable_t * hash = bat_priv - > orig_hash ;
2011-02-18 12:28:09 +00:00
struct hlist_node * node ;
2010-12-13 11:19:28 +00:00
struct hlist_head * head ;
struct orig_node * orig_node ;
2011-10-05 17:05:25 +02:00
uint32_t i ;
int ret ;
2010-12-13 11:19:28 +00:00
/* resize all orig nodes because orig_node->bcast_own(_sum) depend on
2012-05-12 02:09:43 +02:00
* if_num
*/
2010-12-13 11:19:28 +00:00
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
2011-01-19 20:01:40 +00:00
rcu_read_lock ( ) ;
2011-02-18 12:28:09 +00:00
hlist_for_each_entry_rcu ( orig_node , node , head , hash_entry ) {
2011-01-19 20:01:42 +00:00
spin_lock_bh ( & orig_node - > ogm_cnt_lock ) ;
2012-05-12 18:34:00 +02:00
ret = batadv_orig_node_add_if ( orig_node , max_if_num ) ;
2011-01-19 20:01:42 +00:00
spin_unlock_bh ( & orig_node - > ogm_cnt_lock ) ;
2012-05-05 13:27:28 +02:00
if ( ret = = - ENOMEM )
2010-12-13 11:19:28 +00:00
goto err ;
}
2011-01-19 20:01:40 +00:00
rcu_read_unlock ( ) ;
2010-12-13 11:19:28 +00:00
}
return 0 ;
err :
2011-01-19 20:01:40 +00:00
rcu_read_unlock ( ) ;
2010-12-13 11:19:28 +00:00
return - ENOMEM ;
}
2012-05-12 18:34:00 +02:00
static int batadv_orig_node_del_if ( struct orig_node * orig_node ,
int max_if_num , int del_if_num )
2010-12-13 11:19:28 +00:00
{
void * data_ptr = NULL ;
int chunk_size ;
/* last interface was removed */
if ( max_if_num = = 0 )
goto free_bcast_own ;
2012-06-03 22:19:17 +02:00
chunk_size = sizeof ( unsigned long ) * BATADV_NUM_WORDS ;
2010-12-13 11:19:28 +00:00
data_ptr = kmalloc ( max_if_num * chunk_size , GFP_ATOMIC ) ;
2011-08-29 14:17:24 -07:00
if ( ! data_ptr )
2012-05-05 13:27:28 +02:00
return - ENOMEM ;
2010-12-13 11:19:28 +00:00
/* copy first part */
memcpy ( data_ptr , orig_node - > bcast_own , del_if_num * chunk_size ) ;
/* copy second part */
2011-05-14 23:14:49 +02:00
memcpy ( ( char * ) data_ptr + del_if_num * chunk_size ,
2010-12-13 11:19:28 +00:00
orig_node - > bcast_own + ( ( del_if_num + 1 ) * chunk_size ) ,
( max_if_num - del_if_num ) * chunk_size ) ;
free_bcast_own :
kfree ( orig_node - > bcast_own ) ;
orig_node - > bcast_own = data_ptr ;
if ( max_if_num = = 0 )
goto free_own_sum ;
data_ptr = kmalloc ( max_if_num * sizeof ( uint8_t ) , GFP_ATOMIC ) ;
2011-08-29 14:17:24 -07:00
if ( ! data_ptr )
2012-05-05 13:27:28 +02:00
return - ENOMEM ;
2010-12-13 11:19:28 +00:00
memcpy ( data_ptr , orig_node - > bcast_own_sum ,
del_if_num * sizeof ( uint8_t ) ) ;
2011-05-14 23:14:49 +02:00
memcpy ( ( char * ) data_ptr + del_if_num * sizeof ( uint8_t ) ,
2010-12-13 11:19:28 +00:00
orig_node - > bcast_own_sum + ( ( del_if_num + 1 ) * sizeof ( uint8_t ) ) ,
( max_if_num - del_if_num ) * sizeof ( uint8_t ) ) ;
free_own_sum :
kfree ( orig_node - > bcast_own_sum ) ;
orig_node - > bcast_own_sum = data_ptr ;
return 0 ;
}
2012-05-12 02:09:34 +02:00
int batadv_orig_hash_del_if ( struct hard_iface * hard_iface , int max_if_num )
2010-12-13 11:19:28 +00:00
{
2011-02-18 12:33:20 +00:00
struct bat_priv * bat_priv = netdev_priv ( hard_iface - > soft_iface ) ;
2010-12-13 11:19:28 +00:00
struct hashtable_t * hash = bat_priv - > orig_hash ;
2011-02-18 12:28:09 +00:00
struct hlist_node * node ;
2010-12-13 11:19:28 +00:00
struct hlist_head * head ;
2011-02-18 12:33:20 +00:00
struct hard_iface * hard_iface_tmp ;
2010-12-13 11:19:28 +00:00
struct orig_node * orig_node ;
2011-10-05 17:05:25 +02:00
uint32_t i ;
int ret ;
2010-12-13 11:19:28 +00:00
/* resize all orig nodes because orig_node->bcast_own(_sum) depend on
2012-05-12 02:09:43 +02:00
* if_num
*/
2010-12-13 11:19:28 +00:00
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
2011-01-19 20:01:40 +00:00
rcu_read_lock ( ) ;
2011-02-18 12:28:09 +00:00
hlist_for_each_entry_rcu ( orig_node , node , head , hash_entry ) {
2011-01-19 20:01:42 +00:00
spin_lock_bh ( & orig_node - > ogm_cnt_lock ) ;
2012-05-12 18:34:00 +02:00
ret = batadv_orig_node_del_if ( orig_node , max_if_num ,
hard_iface - > if_num ) ;
2011-01-19 20:01:42 +00:00
spin_unlock_bh ( & orig_node - > ogm_cnt_lock ) ;
2010-12-13 11:19:28 +00:00
2012-05-05 13:27:28 +02:00
if ( ret = = - ENOMEM )
2010-12-13 11:19:28 +00:00
goto err ;
}
2011-01-19 20:01:40 +00:00
rcu_read_unlock ( ) ;
2010-12-13 11:19:28 +00:00
}
/* renumber remaining batman interfaces _inside_ of orig_hash_lock */
rcu_read_lock ( ) ;
2012-05-12 02:09:42 +02:00
list_for_each_entry_rcu ( hard_iface_tmp , & batadv_hardif_list , list ) {
2011-02-18 12:33:20 +00:00
if ( hard_iface_tmp - > if_status = = IF_NOT_IN_USE )
2010-12-13 11:19:28 +00:00
continue ;
2011-02-18 12:33:20 +00:00
if ( hard_iface = = hard_iface_tmp )
2010-12-13 11:19:28 +00:00
continue ;
2011-02-18 12:33:20 +00:00
if ( hard_iface - > soft_iface ! = hard_iface_tmp - > soft_iface )
2010-12-13 11:19:28 +00:00
continue ;
2011-02-18 12:33:20 +00:00
if ( hard_iface_tmp - > if_num > hard_iface - > if_num )
hard_iface_tmp - > if_num - - ;
2010-12-13 11:19:28 +00:00
}
rcu_read_unlock ( ) ;
2011-02-18 12:33:20 +00:00
hard_iface - > if_num = - 1 ;
2010-12-13 11:19:28 +00:00
return 0 ;
err :
2011-01-19 20:01:40 +00:00
rcu_read_unlock ( ) ;
2010-12-13 11:19:28 +00:00
return - ENOMEM ;
}