2010-12-13 14:19:28 +03:00
/*
2012-01-01 03:41:38 +04:00
* Copyright ( C ) 2007 - 2012 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
*
*/
# include "main.h"
# include "bat_sysfs.h"
# include "bat_debugfs.h"
# include "routing.h"
# include "send.h"
# include "originator.h"
# include "soft-interface.h"
# include "icmp_socket.h"
# include "translation-table.h"
# include "hard-interface.h"
# include "gateway_client.h"
2012-01-22 23:00:19 +04:00
# include "bridge_loop_avoidance.h"
2010-12-13 14:19:28 +03:00
# include "vis.h"
# include "hash.h"
2011-11-28 13:40:17 +04:00
# include "bat_algo.h"
2010-12-13 14:19:28 +03:00
2011-05-03 13:51:38 +04:00
/* List manipulations on hardif_list have to be rtnl_lock()'ed,
* list traversals just rcu - locked */
2011-02-18 15:33:19 +03:00
struct list_head hardif_list ;
2011-11-28 13:40:17 +04:00
char bat_routing_algo [ 20 ] = " BATMAN IV " ;
static struct hlist_head bat_algo_list ;
2010-12-13 14:19:28 +03:00
unsigned char broadcast_addr [ ] = { 0xff , 0xff , 0xff , 0xff , 0xff , 0xff } ;
struct workqueue_struct * bat_event_workqueue ;
static int __init batman_init ( void )
{
2011-02-18 15:33:19 +03:00
INIT_LIST_HEAD ( & hardif_list ) ;
2011-11-28 13:40:17 +04:00
INIT_HLIST_HEAD ( & bat_algo_list ) ;
bat_iv_init ( ) ;
2010-12-13 14:19:28 +03:00
/* the name should not be longer than 10 chars - see
* http : //lwn.net/Articles/23634/ */
bat_event_workqueue = create_singlethread_workqueue ( " bat_events " ) ;
if ( ! bat_event_workqueue )
return - ENOMEM ;
bat_socket_init ( ) ;
debugfs_init ( ) ;
register_netdevice_notifier ( & hard_if_notifier ) ;
2012-03-07 12:07:45 +04:00
pr_info ( " B.A.T.M.A.N. advanced %s (compatibility version %i) loaded \n " ,
SOURCE_VERSION , COMPAT_VERSION ) ;
2010-12-13 14:19:28 +03:00
return 0 ;
}
static void __exit batman_exit ( void )
{
debugfs_destroy ( ) ;
unregister_netdevice_notifier ( & hard_if_notifier ) ;
hardif_remove_interfaces ( ) ;
flush_workqueue ( bat_event_workqueue ) ;
destroy_workqueue ( bat_event_workqueue ) ;
bat_event_workqueue = NULL ;
rcu_barrier ( ) ;
}
int mesh_init ( struct net_device * soft_iface )
{
struct bat_priv * bat_priv = netdev_priv ( soft_iface ) ;
spin_lock_init ( & bat_priv - > forw_bat_list_lock ) ;
spin_lock_init ( & bat_priv - > forw_bcast_list_lock ) ;
2011-04-27 16:27:44 +04:00
spin_lock_init ( & bat_priv - > tt_changes_list_lock ) ;
spin_lock_init ( & bat_priv - > tt_req_list_lock ) ;
2011-04-27 16:27:57 +04:00
spin_lock_init ( & bat_priv - > tt_roam_list_lock ) ;
2011-04-27 16:27:44 +04:00
spin_lock_init ( & bat_priv - > tt_buff_lock ) ;
2010-12-13 14:19:28 +03:00
spin_lock_init ( & bat_priv - > gw_list_lock ) ;
spin_lock_init ( & bat_priv - > vis_hash_lock ) ;
spin_lock_init ( & bat_priv - > vis_list_lock ) ;
INIT_HLIST_HEAD ( & bat_priv - > forw_bat_list ) ;
INIT_HLIST_HEAD ( & bat_priv - > forw_bcast_list ) ;
INIT_HLIST_HEAD ( & bat_priv - > gw_list ) ;
2011-04-27 16:27:44 +04:00
INIT_LIST_HEAD ( & bat_priv - > tt_changes_list ) ;
INIT_LIST_HEAD ( & bat_priv - > tt_req_list ) ;
2011-04-27 16:27:57 +04:00
INIT_LIST_HEAD ( & bat_priv - > tt_roam_list ) ;
2010-12-13 14:19:28 +03:00
if ( originator_init ( bat_priv ) < 1 )
goto err ;
2011-04-27 16:27:44 +04:00
if ( tt_init ( bat_priv ) < 1 )
2010-12-13 14:19:28 +03:00
goto err ;
2011-07-07 17:35:35 +04:00
tt_local_add ( soft_iface , soft_iface - > dev_addr , NULL_IFINDEX ) ;
2010-12-13 14:19:28 +03:00
if ( vis_init ( bat_priv ) < 1 )
goto err ;
2012-01-22 23:00:19 +04:00
if ( bla_init ( bat_priv ) < 1 )
goto err ;
2011-04-27 02:22:00 +04:00
atomic_set ( & bat_priv - > gw_reselect , 0 ) ;
2010-12-13 14:19:28 +03:00
atomic_set ( & bat_priv - > mesh_state , MESH_ACTIVE ) ;
goto end ;
err :
mesh_free ( soft_iface ) ;
return - 1 ;
end :
return 0 ;
}
void mesh_free ( struct net_device * soft_iface )
{
struct bat_priv * bat_priv = netdev_priv ( soft_iface ) ;
atomic_set ( & bat_priv - > mesh_state , MESH_DEACTIVATING ) ;
purge_outstanding_packets ( bat_priv , NULL ) ;
vis_quit ( bat_priv ) ;
gw_node_purge ( bat_priv ) ;
originator_free ( bat_priv ) ;
2011-04-27 16:27:44 +04:00
tt_free ( bat_priv ) ;
2010-12-13 14:19:28 +03:00
2012-01-22 23:00:19 +04:00
bla_free ( bat_priv ) ;
2010-12-13 14:19:28 +03:00
atomic_set ( & bat_priv - > mesh_state , MESH_INACTIVE ) ;
}
void inc_module_count ( void )
{
try_module_get ( THIS_MODULE ) ;
}
void dec_module_count ( void )
{
module_put ( THIS_MODULE ) ;
}
2011-05-15 01:14:50 +04:00
int is_my_mac ( const uint8_t * addr )
2010-12-13 14:19:28 +03:00
{
2011-05-15 01:14:50 +04:00
const struct hard_iface * hard_iface ;
2010-12-13 14:19:28 +03:00
rcu_read_lock ( ) ;
2011-02-18 15:33:20 +03:00
list_for_each_entry_rcu ( hard_iface , & hardif_list , list ) {
if ( hard_iface - > if_status ! = IF_ACTIVE )
2010-12-13 14:19:28 +03:00
continue ;
2011-02-18 15:33:20 +03:00
if ( compare_eth ( hard_iface - > net_dev - > dev_addr , addr ) ) {
2010-12-13 14:19:28 +03:00
rcu_read_unlock ( ) ;
return 1 ;
}
}
rcu_read_unlock ( ) ;
return 0 ;
}
2011-11-28 13:40:17 +04:00
static struct bat_algo_ops * bat_algo_get ( char * name )
{
struct bat_algo_ops * bat_algo_ops = NULL , * bat_algo_ops_tmp ;
struct hlist_node * node ;
hlist_for_each_entry ( bat_algo_ops_tmp , node , & bat_algo_list , list ) {
if ( strcmp ( bat_algo_ops_tmp - > name , name ) ! = 0 )
continue ;
bat_algo_ops = bat_algo_ops_tmp ;
break ;
}
return bat_algo_ops ;
}
int bat_algo_register ( struct bat_algo_ops * bat_algo_ops )
{
struct bat_algo_ops * bat_algo_ops_tmp ;
int ret = - 1 ;
bat_algo_ops_tmp = bat_algo_get ( bat_algo_ops - > name ) ;
if ( bat_algo_ops_tmp ) {
2012-03-07 12:07:45 +04:00
pr_info ( " Trying to register already registered routing algorithm: %s \n " ,
bat_algo_ops - > name ) ;
2011-11-28 13:40:17 +04:00
goto out ;
}
2011-11-28 17:31:55 +04:00
/* all algorithms must implement all ops (for now) */
2012-02-07 13:20:45 +04:00
if ( ! bat_algo_ops - > bat_iface_enable | |
2011-11-28 17:31:55 +04:00
! bat_algo_ops - > bat_ogm_init_primary | |
! bat_algo_ops - > bat_ogm_update_mac | |
! bat_algo_ops - > bat_ogm_schedule | |
! bat_algo_ops - > bat_ogm_emit | |
! bat_algo_ops - > bat_ogm_receive ) {
pr_info ( " Routing algo '%s' does not implement required ops \n " ,
bat_algo_ops - > name ) ;
goto out ;
}
2011-11-28 13:40:17 +04:00
INIT_HLIST_NODE ( & bat_algo_ops - > list ) ;
hlist_add_head ( & bat_algo_ops - > list , & bat_algo_list ) ;
ret = 0 ;
out :
return ret ;
}
int bat_algo_select ( struct bat_priv * bat_priv , char * name )
{
struct bat_algo_ops * bat_algo_ops ;
int ret = - 1 ;
bat_algo_ops = bat_algo_get ( name ) ;
if ( ! bat_algo_ops )
goto out ;
bat_priv - > bat_algo_ops = bat_algo_ops ;
ret = 0 ;
out :
return ret ;
}
int bat_algo_seq_print_text ( struct seq_file * seq , void * offset )
{
struct bat_algo_ops * bat_algo_ops ;
struct hlist_node * node ;
seq_printf ( seq , " Available routing algorithms: \n " ) ;
hlist_for_each_entry ( bat_algo_ops , node , & bat_algo_list , list ) {
seq_printf ( seq , " %s \n " , bat_algo_ops - > name ) ;
}
return 0 ;
}
2011-12-10 15:45:53 +04:00
static int param_set_ra ( const char * val , const struct kernel_param * kp )
{
struct bat_algo_ops * bat_algo_ops ;
bat_algo_ops = bat_algo_get ( ( char * ) val ) ;
if ( ! bat_algo_ops ) {
pr_err ( " Routing algorithm '%s' is not supported \n " , val ) ;
return - EINVAL ;
}
return param_set_copystring ( val , kp ) ;
}
static const struct kernel_param_ops param_ops_ra = {
. set = param_set_ra ,
. get = param_get_string ,
} ;
static struct kparam_string __param_string_ra = {
. maxlen = sizeof ( bat_routing_algo ) ,
. string = bat_routing_algo ,
} ;
module_param_cb ( routing_algo , & param_ops_ra , & __param_string_ra , 0644 ) ;
2010-12-13 14:19:28 +03:00
module_init ( batman_init ) ;
module_exit ( batman_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( DRIVER_AUTHOR ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_SUPPORTED_DEVICE ( DRIVER_DEVICE ) ;
MODULE_VERSION ( SOURCE_VERSION ) ;