2010-12-13 11:19:28 +00:00
/*
2011-01-27 10:38:15 +01:00
* Copyright ( C ) 2007 - 2011 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
*
*/
# ifndef _NET_BATMAN_ADV_ORIGINATOR_H_
# define _NET_BATMAN_ADV_ORIGINATOR_H_
2011-02-18 12:28:09 +00:00
# include "hash.h"
2010-12-13 11:19:28 +00:00
int originator_init ( struct bat_priv * bat_priv ) ;
void originator_free ( struct bat_priv * bat_priv ) ;
void purge_orig_ref ( struct bat_priv * bat_priv ) ;
2011-02-18 12:28:10 +00:00
void orig_node_free_ref ( struct orig_node * orig_node ) ;
2010-12-13 11:19:28 +00:00
struct orig_node * get_orig_node ( struct bat_priv * bat_priv , uint8_t * addr ) ;
2010-12-12 21:57:10 +00:00
struct neigh_node * create_neighbor ( struct orig_node * orig_node ,
struct orig_node * orig_neigh_node ,
uint8_t * neigh ,
2011-02-18 12:33:20 +00:00
struct hard_iface * if_incoming ) ;
2011-02-10 14:33:53 +00:00
void neigh_node_free_ref ( struct neigh_node * neigh_node ) ;
2011-03-14 22:43:37 +00:00
struct neigh_node * orig_node_get_router ( struct orig_node * orig_node ) ;
2010-12-13 11:19:28 +00:00
int orig_seq_print_text ( struct seq_file * seq , void * offset ) ;
2011-02-18 12:33:20 +00:00
int orig_hash_add_if ( struct hard_iface * hard_iface , int max_if_num ) ;
int orig_hash_del_if ( struct hard_iface * hard_iface , int max_if_num ) ;
2010-12-13 11:19:28 +00:00
/* returns 1 if they are the same originator */
2011-02-18 12:28:09 +00:00
static inline int compare_orig ( struct hlist_node * node , void * data2 )
2010-12-13 11:19:28 +00:00
{
2011-02-18 12:28:09 +00:00
void * data1 = container_of ( node , struct orig_node , hash_entry ) ;
2010-12-13 11:19:28 +00:00
return ( memcmp ( data1 , data2 , ETH_ALEN ) = = 0 ? 1 : 0 ) ;
}
/* hashfunction to choose an entry in a hash table of given size */
/* hash algorithm from http://en.wikipedia.org/wiki/Hash_table */
static inline int choose_orig ( void * data , int32_t size )
{
unsigned char * key = data ;
uint32_t hash = 0 ;
size_t i ;
for ( i = 0 ; i < 6 ; i + + ) {
hash + = key [ i ] ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
}
hash + = ( hash < < 3 ) ;
hash ^ = ( hash > > 11 ) ;
hash + = ( hash < < 15 ) ;
return hash % size ;
}
2011-02-18 12:28:09 +00:00
static inline struct orig_node * orig_hash_find ( struct bat_priv * bat_priv ,
void * data )
{
struct hashtable_t * hash = bat_priv - > orig_hash ;
struct hlist_head * head ;
struct hlist_node * node ;
struct orig_node * orig_node , * orig_node_tmp = NULL ;
int index ;
if ( ! hash )
return NULL ;
index = choose_orig ( data , hash - > size ) ;
head = & hash - > table [ index ] ;
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( orig_node , node , head , hash_entry ) {
if ( ! compare_eth ( orig_node , data ) )
continue ;
2011-02-18 12:28:10 +00:00
if ( ! atomic_inc_not_zero ( & orig_node - > refcount ) )
continue ;
2011-02-18 12:28:09 +00:00
orig_node_tmp = orig_node ;
break ;
}
rcu_read_unlock ( ) ;
return orig_node_tmp ;
}
2010-12-13 11:19:28 +00:00
# endif /* _NET_BATMAN_ADV_ORIGINATOR_H_ */