2010-12-13 14:19:28 +03:00
/*
2011-01-27 12:38:15 +03:00
* Copyright ( C ) 2007 - 2011 B . A . T . M . A . N . contributors :
2010-12-13 14:19:28 +03: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 15:28:09 +03:00
# include "hash.h"
2010-12-13 14:19:28 +03: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 15:28:10 +03:00
void orig_node_free_ref ( struct orig_node * orig_node ) ;
2011-05-15 01:14:50 +04:00
struct orig_node * get_orig_node ( struct bat_priv * bat_priv , const uint8_t * addr ) ;
2010-12-13 00:57:10 +03:00
struct neigh_node * create_neighbor ( struct orig_node * orig_node ,
struct orig_node * orig_neigh_node ,
2011-05-15 01:14:50 +04:00
const uint8_t * neigh ,
2011-02-18 15:33:20 +03:00
struct hard_iface * if_incoming ) ;
2011-02-10 17:33:53 +03:00
void neigh_node_free_ref ( struct neigh_node * neigh_node ) ;
2011-03-15 01:43:37 +03:00
struct neigh_node * orig_node_get_router ( struct orig_node * orig_node ) ;
2010-12-13 14:19:28 +03:00
int orig_seq_print_text ( struct seq_file * seq , void * offset ) ;
2011-02-18 15:33:20 +03: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 14:19:28 +03:00
/* hashfunction to choose an entry in a hash table of given size */
/* hash algorithm from http://en.wikipedia.org/wiki/Hash_table */
2011-05-15 01:14:50 +04:00
static inline int choose_orig ( const void * data , int32_t size )
2010-12-13 14:19:28 +03:00
{
2011-05-15 01:14:50 +04:00
const unsigned char * key = data ;
2010-12-13 14:19:28 +03:00
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 15:28:09 +03:00
static inline struct orig_node * orig_hash_find ( struct bat_priv * bat_priv ,
2011-05-15 01:14:50 +04:00
const void * data )
2011-02-18 15:28:09 +03:00
{
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 15:28:10 +03:00
if ( ! atomic_inc_not_zero ( & orig_node - > refcount ) )
continue ;
2011-02-18 15:28:09 +03:00
orig_node_tmp = orig_node ;
break ;
}
rcu_read_unlock ( ) ;
return orig_node_tmp ;
}
2010-12-13 14:19:28 +03:00
# endif /* _NET_BATMAN_ADV_ORIGINATOR_H_ */