2017-11-19 17:05:11 +03:00
// SPDX-License-Identifier: GPL-2.0
2021-01-01 02:00:01 +03:00
/* Copyright (C) B.A.T.M.A.N. contributors:
2013-01-25 14:12:38 +04:00
*
* Martin Hundebøll , Jeppe Ledet - Pedersen
*/
2015-04-17 20:40:28 +03:00
# include "network-coding.h"
# include "main.h"
# include <linux/atomic.h>
2015-06-16 18:10:23 +03:00
# include <linux/bitops.h>
2015-04-17 20:40:28 +03:00
# include <linux/byteorder/generic.h>
# include <linux/compiler.h>
2022-01-21 19:14:44 +03:00
# include <linux/container_of.h>
2015-04-17 20:40:28 +03:00
# include <linux/errno.h>
# include <linux/etherdevice.h>
2017-11-19 19:12:02 +03:00
# include <linux/gfp.h>
2015-04-17 20:40:28 +03:00
# include <linux/if_ether.h>
# include <linux/if_packet.h>
# include <linux/init.h>
# include <linux/jhash.h>
# include <linux/jiffies.h>
2016-01-16 12:29:45 +03:00
# include <linux/kref.h>
2015-04-17 20:40:28 +03:00
# include <linux/list.h>
# include <linux/lockdep.h>
2017-11-19 19:12:04 +03:00
# include <linux/net.h>
2015-04-17 20:40:28 +03:00
# include <linux/netdevice.h>
# include <linux/printk.h>
2022-12-17 14:40:08 +03:00
# include <linux/random.h>
2015-04-17 20:40:28 +03:00
# include <linux/rculist.h>
# include <linux/rcupdate.h>
# include <linux/skbuff.h>
# include <linux/slab.h>
# include <linux/spinlock.h>
# include <linux/stddef.h>
# include <linux/string.h>
# include <linux/workqueue.h>
2017-12-21 12:17:41 +03:00
# include <uapi/linux/batadv_packet.h>
2013-01-25 14:12:39 +04:00
2013-01-25 14:12:40 +04:00
# include "hash.h"
2016-05-16 00:48:31 +03:00
# include "log.h"
2013-01-25 14:12:39 +04:00
# include "originator.h"
2013-01-25 14:12:43 +04:00
# include "routing.h"
2015-04-17 20:40:28 +03:00
# include "send.h"
2016-05-15 12:07:43 +03:00
# include "tvlv.h"
2013-01-25 14:12:38 +04:00
2013-01-25 14:12:40 +04:00
static struct lock_class_key batadv_nc_coding_hash_lock_class_key ;
2013-01-25 14:12:42 +04:00
static struct lock_class_key batadv_nc_decoding_hash_lock_class_key ;
2013-01-25 14:12:40 +04:00
2013-01-25 14:12:38 +04:00
static void batadv_nc_worker ( struct work_struct * work ) ;
2013-01-25 14:12:43 +04:00
static int batadv_nc_recv_coded_packet ( struct sk_buff * skb ,
struct batadv_hard_iface * recv_if ) ;
2013-01-25 14:12:38 +04:00
2013-09-27 20:03:39 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_init ( ) - one - time initialization for network coding
2015-10-31 14:29:30 +03:00
*
* Return : 0 on success or negative error number in case of failure
2013-09-27 20:03:39 +04:00
*/
int __init batadv_nc_init ( void )
{
/* Register our packet type */
2021-12-10 05:19:17 +03:00
return batadv_recv_handler_register ( BATADV_CODED ,
batadv_nc_recv_coded_packet ) ;
2013-09-27 20:03:39 +04:00
}
2013-01-25 14:12:38 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_start_timer ( ) - initialise the nc periodic worker
2013-01-25 14:12:38 +04:00
* @ bat_priv : the bat priv with all the soft interface information
*/
static void batadv_nc_start_timer ( struct batadv_priv * bat_priv )
{
queue_delayed_work ( batadv_event_workqueue , & bat_priv - > nc . work ,
msecs_to_jiffies ( 10 ) ) ;
}
2013-04-23 17:40:00 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_tvlv_container_update ( ) - update the network coding tvlv container
2013-04-23 17:40:00 +04:00
* after network coding setting change
* @ bat_priv : the bat priv with all the soft interface information
*/
static void batadv_nc_tvlv_container_update ( struct batadv_priv * bat_priv )
{
char nc_mode ;
nc_mode = atomic_read ( & bat_priv - > network_coding ) ;
switch ( nc_mode ) {
case 0 :
batadv_tvlv_container_unregister ( bat_priv , BATADV_TVLV_NC , 1 ) ;
break ;
case 1 :
batadv_tvlv_container_register ( bat_priv , BATADV_TVLV_NC , 1 ,
NULL , 0 ) ;
break ;
}
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_status_update ( ) - update the network coding tvlv container after
2013-04-23 17:40:00 +04:00
* network coding setting change
* @ net_dev : the soft interface net device
*/
void batadv_nc_status_update ( struct net_device * net_dev )
{
struct batadv_priv * bat_priv = netdev_priv ( net_dev ) ;
2014-05-10 20:56:37 +04:00
2013-04-23 17:40:00 +04:00
batadv_nc_tvlv_container_update ( bat_priv ) ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_tvlv_ogm_handler_v1 ( ) - process incoming nc tvlv container
2013-04-23 17:40:00 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ orig : the orig_node of the ogm
* @ flags : flags indicating the tvlv state ( see batadv_tvlv_handler_flags )
* @ tvlv_value : tvlv buffer containing the gateway data
* @ tvlv_value_len : tvlv buffer length
*/
static void batadv_nc_tvlv_ogm_handler_v1 ( struct batadv_priv * bat_priv ,
struct batadv_orig_node * orig ,
2015-05-26 19:34:26 +03:00
u8 flags ,
void * tvlv_value , u16 tvlv_value_len )
2013-04-23 17:40:00 +04:00
{
if ( flags & BATADV_TVLV_HANDLER_OGM_CIFNOTFND )
2015-06-16 18:10:23 +03:00
clear_bit ( BATADV_ORIG_CAPA_HAS_NC , & orig - > capabilities ) ;
2013-04-23 17:40:00 +04:00
else
2015-06-16 18:10:23 +03:00
set_bit ( BATADV_ORIG_CAPA_HAS_NC , & orig - > capabilities ) ;
2013-04-23 17:40:00 +04:00
}
2013-01-25 14:12:38 +04:00
/**
2020-06-01 21:13:21 +03:00
* batadv_nc_mesh_init ( ) - initialise coding hash table and start housekeeping
2013-01-25 14:12:38 +04:00
* @ bat_priv : the bat priv with all the soft interface information
2015-10-31 14:29:30 +03:00
*
* Return : 0 on success or negative error number in case of failure
2013-01-25 14:12:38 +04:00
*/
2013-09-27 20:03:39 +04:00
int batadv_nc_mesh_init ( struct batadv_priv * bat_priv )
2013-01-25 14:12:38 +04:00
{
2013-01-25 14:12:40 +04:00
bat_priv - > nc . timestamp_fwd_flush = jiffies ;
2013-01-25 14:12:42 +04:00
bat_priv - > nc . timestamp_sniffed_purge = jiffies ;
2013-01-25 14:12:40 +04:00
2013-01-25 14:12:42 +04:00
if ( bat_priv - > nc . coding_hash | | bat_priv - > nc . decoding_hash )
2013-01-25 14:12:40 +04:00
return 0 ;
bat_priv - > nc . coding_hash = batadv_hash_new ( 128 ) ;
if ( ! bat_priv - > nc . coding_hash )
goto err ;
batadv_hash_set_lock_class ( bat_priv - > nc . coding_hash ,
& batadv_nc_coding_hash_lock_class_key ) ;
2013-01-25 14:12:42 +04:00
bat_priv - > nc . decoding_hash = batadv_hash_new ( 128 ) ;
2021-10-24 16:13:56 +03:00
if ( ! bat_priv - > nc . decoding_hash ) {
batadv_hash_destroy ( bat_priv - > nc . coding_hash ) ;
2013-01-25 14:12:42 +04:00
goto err ;
2021-10-24 16:13:56 +03:00
}
2013-01-25 14:12:42 +04:00
2014-11-11 18:22:23 +03:00
batadv_hash_set_lock_class ( bat_priv - > nc . decoding_hash ,
2013-01-25 14:12:42 +04:00
& batadv_nc_decoding_hash_lock_class_key ) ;
2013-01-25 14:12:38 +04:00
INIT_DELAYED_WORK ( & bat_priv - > nc . work , batadv_nc_worker ) ;
batadv_nc_start_timer ( bat_priv ) ;
2013-04-23 17:40:00 +04:00
batadv_tvlv_handler_register ( bat_priv , batadv_nc_tvlv_ogm_handler_v1 ,
2022-12-27 22:34:06 +03:00
NULL , NULL , BATADV_TVLV_NC , 1 ,
2013-04-23 17:40:00 +04:00
BATADV_TVLV_HANDLER_OGM_CIFNOTFND ) ;
batadv_nc_tvlv_container_update ( bat_priv ) ;
2013-01-25 14:12:38 +04:00
return 0 ;
2013-01-25 14:12:40 +04:00
err :
return - ENOMEM ;
2013-01-25 14:12:38 +04:00
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_init_bat_priv ( ) - initialise the nc specific bat_priv variables
2013-01-25 14:12:38 +04:00
* @ bat_priv : the bat priv with all the soft interface information
*/
void batadv_nc_init_bat_priv ( struct batadv_priv * bat_priv )
{
2015-02-18 20:20:24 +03:00
atomic_set ( & bat_priv - > network_coding , 0 ) ;
2013-01-25 14:12:39 +04:00
bat_priv - > nc . min_tq = 200 ;
2013-01-25 14:12:40 +04:00
bat_priv - > nc . max_fwd_delay = 10 ;
2013-01-25 14:12:42 +04:00
bat_priv - > nc . max_buffer_time = 200 ;
2013-01-25 14:12:39 +04:00
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_init_orig ( ) - initialise the nc fields of an orig_node
2013-01-25 14:12:39 +04:00
* @ orig_node : the orig_node which is going to be initialised
*/
void batadv_nc_init_orig ( struct batadv_orig_node * orig_node )
{
INIT_LIST_HEAD ( & orig_node - > in_coding_list ) ;
INIT_LIST_HEAD ( & orig_node - > out_coding_list ) ;
spin_lock_init ( & orig_node - > in_coding_list_lock ) ;
spin_lock_init ( & orig_node - > out_coding_list_lock ) ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_node_release ( ) - release nc_node from lists and queue for free
* after rcu grace period
2016-01-16 12:29:45 +03:00
* @ ref : kref pointer of the nc_node
2013-01-25 14:12:39 +04:00
*/
2016-01-16 12:29:45 +03:00
static void batadv_nc_node_release ( struct kref * ref )
2013-01-25 14:12:39 +04:00
{
2016-01-16 12:29:45 +03:00
struct batadv_nc_node * nc_node ;
nc_node = container_of ( ref , struct batadv_nc_node , refcount ) ;
2016-01-17 13:01:09 +03:00
batadv_orig_node_put ( nc_node - > orig_node ) ;
2016-01-05 14:06:19 +03:00
kfree_rcu ( nc_node , rcu ) ;
2013-01-25 14:12:39 +04:00
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_node_put ( ) - decrement the nc_node refcounter and possibly
2016-01-05 14:06:19 +03:00
* release it
2016-01-16 12:29:45 +03:00
* @ nc_node : nc_node to be free ' d
2013-01-25 14:12:39 +04:00
*/
2016-01-17 13:01:22 +03:00
static void batadv_nc_node_put ( struct batadv_nc_node * nc_node )
2013-01-25 14:12:39 +04:00
{
2021-08-08 20:56:17 +03:00
if ( ! nc_node )
return ;
2016-01-16 12:29:45 +03:00
kref_put ( & nc_node - > refcount , batadv_nc_node_release ) ;
2013-01-25 14:12:39 +04:00
}
2013-01-25 14:12:40 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_path_release ( ) - release nc_path from lists and queue for free
* after rcu grace period
2016-01-16 12:29:46 +03:00
* @ ref : kref pointer of the nc_path
*/
static void batadv_nc_path_release ( struct kref * ref )
{
struct batadv_nc_path * nc_path ;
nc_path = container_of ( ref , struct batadv_nc_path , refcount ) ;
kfree_rcu ( nc_path , rcu ) ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_path_put ( ) - decrement the nc_path refcounter and possibly
2016-01-16 12:29:46 +03:00
* release it
* @ nc_path : nc_path to be free ' d
2013-01-25 14:12:40 +04:00
*/
2016-01-17 13:01:23 +03:00
static void batadv_nc_path_put ( struct batadv_nc_path * nc_path )
2013-01-25 14:12:40 +04:00
{
2021-08-08 20:56:17 +03:00
if ( ! nc_path )
return ;
2016-01-16 12:29:46 +03:00
kref_put ( & nc_path - > refcount , batadv_nc_path_release ) ;
2013-01-25 14:12:40 +04:00
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_packet_free ( ) - frees nc packet
2013-01-25 14:12:40 +04:00
* @ nc_packet : the nc packet to free
2020-07-31 21:33:00 +03:00
* @ dropped : whether the packet is freed because is dropped
2013-01-25 14:12:40 +04:00
*/
2016-07-17 22:04:00 +03:00
static void batadv_nc_packet_free ( struct batadv_nc_packet * nc_packet ,
bool dropped )
2013-01-25 14:12:40 +04:00
{
2016-07-17 22:04:00 +03:00
if ( dropped )
kfree_skb ( nc_packet - > skb ) ;
else
consume_skb ( nc_packet - > skb ) ;
2016-01-17 13:01:23 +03:00
batadv_nc_path_put ( nc_packet - > nc_path ) ;
2013-01-25 14:12:40 +04:00
kfree ( nc_packet ) ;
}
2013-01-25 14:12:39 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_to_purge_nc_node ( ) - checks whether an nc node has to be purged
2013-01-25 14:12:39 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ nc_node : the nc node to check
*
2015-09-15 20:00:48 +03:00
* Return : true if the entry has to be purged now , false otherwise
2013-01-25 14:12:39 +04:00
*/
static bool batadv_nc_to_purge_nc_node ( struct batadv_priv * bat_priv ,
struct batadv_nc_node * nc_node )
{
if ( atomic_read ( & bat_priv - > mesh_state ) ! = BATADV_MESH_ACTIVE )
return true ;
return batadv_has_timed_out ( nc_node - > last_seen , BATADV_NC_NODE_TIMEOUT ) ;
}
2013-01-25 14:12:40 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_to_purge_nc_path_coding ( ) - checks whether an nc path has timed out
2013-01-25 14:12:40 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ nc_path : the nc path to check
*
2015-09-15 20:00:48 +03:00
* Return : true if the entry has to be purged now , false otherwise
2013-01-25 14:12:40 +04:00
*/
static bool batadv_nc_to_purge_nc_path_coding ( struct batadv_priv * bat_priv ,
struct batadv_nc_path * nc_path )
{
if ( atomic_read ( & bat_priv - > mesh_state ) ! = BATADV_MESH_ACTIVE )
return true ;
/* purge the path when no packets has been added for 10 times the
* max_fwd_delay time
*/
return batadv_has_timed_out ( nc_path - > last_valid ,
bat_priv - > nc . max_fwd_delay * 10 ) ;
}
2013-01-25 14:12:42 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_to_purge_nc_path_decoding ( ) - checks whether an nc path has timed
* out
2013-01-25 14:12:42 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ nc_path : the nc path to check
*
2015-09-15 20:00:48 +03:00
* Return : true if the entry has to be purged now , false otherwise
2013-01-25 14:12:42 +04:00
*/
static bool batadv_nc_to_purge_nc_path_decoding ( struct batadv_priv * bat_priv ,
struct batadv_nc_path * nc_path )
{
if ( atomic_read ( & bat_priv - > mesh_state ) ! = BATADV_MESH_ACTIVE )
return true ;
/* purge the path when no packets has been added for 10 times the
* max_buffer time
*/
return batadv_has_timed_out ( nc_path - > last_valid ,
2015-02-18 17:19:20 +03:00
bat_priv - > nc . max_buffer_time * 10 ) ;
2013-01-25 14:12:42 +04:00
}
2013-01-25 14:12:39 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_purge_orig_nc_nodes ( ) - go through list of nc nodes and purge stale
2013-01-25 14:12:39 +04:00
* entries
* @ bat_priv : the bat priv with all the soft interface information
* @ list : list of nc nodes
* @ lock : nc node list lock
* @ to_purge : function in charge to decide whether an entry has to be purged or
* not . This function takes the nc node as argument and has to return
* a boolean value : true if the entry has to be deleted , false
* otherwise
*/
static void
batadv_nc_purge_orig_nc_nodes ( struct batadv_priv * bat_priv ,
struct list_head * list ,
spinlock_t * lock ,
bool ( * to_purge ) ( struct batadv_priv * ,
struct batadv_nc_node * ) )
{
struct batadv_nc_node * nc_node , * nc_node_tmp ;
/* For each nc_node in list */
spin_lock_bh ( lock ) ;
list_for_each_entry_safe ( nc_node , nc_node_tmp , list , list ) {
/* if an helper function has been passed as parameter,
* ask it if the entry has to be purged or not
*/
if ( to_purge & & ! to_purge ( bat_priv , nc_node ) )
continue ;
batadv_dbg ( BATADV_DBG_NC , bat_priv ,
" Removing nc_node %pM -> %pM \n " ,
nc_node - > addr , nc_node - > orig_node - > orig ) ;
list_del_rcu ( & nc_node - > list ) ;
2016-01-17 13:01:22 +03:00
batadv_nc_node_put ( nc_node ) ;
2013-01-25 14:12:39 +04:00
}
spin_unlock_bh ( lock ) ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_purge_orig ( ) - purges all nc node data attached of the given
2013-01-25 14:12:39 +04:00
* originator
* @ bat_priv : the bat priv with all the soft interface information
* @ orig_node : orig_node with the nc node entries to be purged
* @ to_purge : function in charge to decide whether an entry has to be purged or
* not . This function takes the nc node as argument and has to return
* a boolean value : true is the entry has to be deleted , false
* otherwise
*/
void batadv_nc_purge_orig ( struct batadv_priv * bat_priv ,
struct batadv_orig_node * orig_node ,
bool ( * to_purge ) ( struct batadv_priv * ,
struct batadv_nc_node * ) )
{
/* Check ingoing nc_node's of this orig_node */
batadv_nc_purge_orig_nc_nodes ( bat_priv , & orig_node - > in_coding_list ,
& orig_node - > in_coding_list_lock ,
to_purge ) ;
/* Check outgoing nc_node's of this orig_node */
batadv_nc_purge_orig_nc_nodes ( bat_priv , & orig_node - > out_coding_list ,
& orig_node - > out_coding_list_lock ,
to_purge ) ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_purge_orig_hash ( ) - traverse entire originator hash to check if
* they have timed out nc nodes
2013-01-25 14:12:39 +04:00
* @ bat_priv : the bat priv with all the soft interface information
*/
static void batadv_nc_purge_orig_hash ( struct batadv_priv * bat_priv )
{
struct batadv_hashtable * hash = bat_priv - > orig_hash ;
struct hlist_head * head ;
struct batadv_orig_node * orig_node ;
2015-05-26 19:34:26 +03:00
u32 i ;
2013-01-25 14:12:39 +04:00
if ( ! hash )
return ;
/* For each orig_node */
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( orig_node , head , hash_entry )
batadv_nc_purge_orig ( bat_priv , orig_node ,
batadv_nc_to_purge_nc_node ) ;
rcu_read_unlock ( ) ;
}
2013-01-25 14:12:38 +04:00
}
2013-01-25 14:12:40 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_purge_paths ( ) - traverse all nc paths part of the hash and remove
2013-01-25 14:12:40 +04:00
* unused ones
* @ bat_priv : the bat priv with all the soft interface information
* @ hash : hash table containing the nc paths to check
* @ to_purge : function in charge to decide whether an entry has to be purged or
* not . This function takes the nc node as argument and has to return
* a boolean value : true is the entry has to be deleted , false
* otherwise
*/
static void batadv_nc_purge_paths ( struct batadv_priv * bat_priv ,
struct batadv_hashtable * hash ,
bool ( * to_purge ) ( struct batadv_priv * ,
struct batadv_nc_path * ) )
{
struct hlist_head * head ;
struct hlist_node * node_tmp ;
struct batadv_nc_path * nc_path ;
spinlock_t * lock ; /* Protects lists in hash */
2015-05-26 19:34:26 +03:00
u32 i ;
2013-01-25 14:12:40 +04:00
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
lock = & hash - > list_locks [ i ] ;
/* For each nc_path in this bin */
spin_lock_bh ( lock ) ;
hlist_for_each_entry_safe ( nc_path , node_tmp , head , hash_entry ) {
/* if an helper function has been passed as parameter,
* ask it if the entry has to be purged or not
*/
if ( to_purge & & ! to_purge ( bat_priv , nc_path ) )
continue ;
/* purging an non-empty nc_path should never happen, but
* is observed under high CPU load . Delay the purging
* until next iteration to allow the packet_list to be
* emptied first .
*/
if ( ! unlikely ( list_empty ( & nc_path - > packet_list ) ) ) {
net_ratelimited_function ( printk ,
KERN_WARNING
" Skipping free of non-empty nc_path (%pM -> %pM)! \n " ,
nc_path - > prev_hop ,
nc_path - > next_hop ) ;
continue ;
}
/* nc_path is unused, so remove it */
batadv_dbg ( BATADV_DBG_NC , bat_priv ,
" Remove nc_path %pM -> %pM \n " ,
nc_path - > prev_hop , nc_path - > next_hop ) ;
hlist_del_rcu ( & nc_path - > hash_entry ) ;
2016-01-17 13:01:23 +03:00
batadv_nc_path_put ( nc_path ) ;
2013-01-25 14:12:40 +04:00
}
spin_unlock_bh ( lock ) ;
}
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_hash_key_gen ( ) - computes the nc_path hash key
2013-01-25 14:12:40 +04:00
* @ key : buffer to hold the final hash key
* @ src : source ethernet mac address going into the hash key
* @ dst : destination ethernet mac address going into the hash key
*/
static void batadv_nc_hash_key_gen ( struct batadv_nc_path * key , const char * src ,
const char * dst )
{
memcpy ( key - > prev_hop , src , sizeof ( key - > prev_hop ) ) ;
memcpy ( key - > next_hop , dst , sizeof ( key - > next_hop ) ) ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_hash_choose ( ) - compute the hash value for an nc path
2013-01-25 14:12:40 +04:00
* @ data : data to hash
* @ size : size of the hash table
*
2015-09-15 20:00:48 +03:00
* Return : the selected index in the hash table for the given data .
2013-01-25 14:12:40 +04:00
*/
2015-05-26 19:34:26 +03:00
static u32 batadv_nc_hash_choose ( const void * data , u32 size )
2013-01-25 14:12:40 +04:00
{
const struct batadv_nc_path * nc_path = data ;
2015-05-26 19:34:26 +03:00
u32 hash = 0 ;
2013-01-25 14:12:40 +04:00
2015-03-01 11:46:18 +03:00
hash = jhash ( & nc_path - > prev_hop , sizeof ( nc_path - > prev_hop ) , hash ) ;
hash = jhash ( & nc_path - > next_hop , sizeof ( nc_path - > next_hop ) , hash ) ;
2013-01-25 14:12:40 +04:00
return hash % size ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_hash_compare ( ) - comparing function used in the network coding hash
2013-01-25 14:12:40 +04:00
* tables
* @ node : node in the local table
* @ data2 : second object to compare the node to
*
2016-02-22 23:02:39 +03:00
* Return : true if the two entry are the same , false otherwise
2013-01-25 14:12:40 +04:00
*/
2016-02-22 23:02:39 +03:00
static bool batadv_nc_hash_compare ( const struct hlist_node * node ,
const void * data2 )
2013-01-25 14:12:40 +04:00
{
const struct batadv_nc_path * nc_path1 , * nc_path2 ;
nc_path1 = container_of ( node , struct batadv_nc_path , hash_entry ) ;
nc_path2 = data2 ;
/* Return 1 if the two keys are identical */
2016-03-11 16:01:10 +03:00
if ( ! batadv_compare_eth ( nc_path1 - > prev_hop , nc_path2 - > prev_hop ) )
2016-02-22 23:02:39 +03:00
return false ;
2013-01-25 14:12:40 +04:00
2016-03-11 16:01:10 +03:00
if ( ! batadv_compare_eth ( nc_path1 - > next_hop , nc_path2 - > next_hop ) )
2016-02-22 23:02:39 +03:00
return false ;
2013-01-25 14:12:40 +04:00
2016-02-22 23:02:39 +03:00
return true ;
2013-01-25 14:12:40 +04:00
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_hash_find ( ) - search for an existing nc path and return it
2013-01-25 14:12:40 +04:00
* @ hash : hash table containing the nc path
* @ data : search key
*
2015-09-15 20:00:48 +03:00
* Return : the nc_path if found , NULL otherwise .
2013-01-25 14:12:40 +04:00
*/
static struct batadv_nc_path *
batadv_nc_hash_find ( struct batadv_hashtable * hash ,
void * data )
{
struct hlist_head * head ;
struct batadv_nc_path * nc_path , * nc_path_tmp = NULL ;
int index ;
if ( ! hash )
return NULL ;
index = batadv_nc_hash_choose ( data , hash - > size ) ;
head = & hash - > table [ index ] ;
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( nc_path , head , hash_entry ) {
if ( ! batadv_nc_hash_compare ( & nc_path - > hash_entry , data ) )
continue ;
2016-01-16 12:29:46 +03:00
if ( ! kref_get_unless_zero ( & nc_path - > refcount ) )
2013-01-25 14:12:40 +04:00
continue ;
nc_path_tmp = nc_path ;
break ;
}
rcu_read_unlock ( ) ;
return nc_path_tmp ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_send_packet ( ) - send non - coded packet and free nc_packet struct
2013-01-25 14:12:40 +04:00
* @ nc_packet : the nc packet to send
*/
static void batadv_nc_send_packet ( struct batadv_nc_packet * nc_packet )
{
2016-01-16 11:40:15 +03:00
batadv_send_unicast_skb ( nc_packet - > skb , nc_packet - > neigh_node ) ;
2013-01-25 14:12:40 +04:00
nc_packet - > skb = NULL ;
2016-07-17 22:04:00 +03:00
batadv_nc_packet_free ( nc_packet , false ) ;
2013-01-25 14:12:40 +04:00
}
2013-01-25 14:12:42 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_sniffed_purge ( ) - Checks timestamp of given sniffed nc_packet .
2013-01-25 14:12:42 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ nc_path : the nc path the packet belongs to
* @ nc_packet : the nc packet to be checked
*
* Checks whether the given sniffed ( overheard ) nc_packet has hit its buffering
* timeout . If so , the packet is no longer kept and the entry deleted from the
* queue . Has to be called with the appropriate locks .
*
2015-09-15 20:00:48 +03:00
* Return : false as soon as the entry in the fifo queue has not been timed out
2013-01-25 14:12:42 +04:00
* yet and true otherwise .
*/
static bool batadv_nc_sniffed_purge ( struct batadv_priv * bat_priv ,
struct batadv_nc_path * nc_path ,
struct batadv_nc_packet * nc_packet )
{
unsigned long timeout = bat_priv - > nc . max_buffer_time ;
bool res = false ;
2015-06-21 15:45:14 +03:00
lockdep_assert_held ( & nc_path - > packet_list_lock ) ;
2013-01-25 14:12:42 +04:00
/* Packets are added to tail, so the remaining packets did not time
* out and we can stop processing the current queue
*/
if ( atomic_read ( & bat_priv - > mesh_state ) = = BATADV_MESH_ACTIVE & &
! batadv_has_timed_out ( nc_packet - > timestamp , timeout ) )
goto out ;
/* purge nc packet */
list_del ( & nc_packet - > list ) ;
2016-07-17 22:04:00 +03:00
batadv_nc_packet_free ( nc_packet , true ) ;
2013-01-25 14:12:42 +04:00
res = true ;
out :
return res ;
}
2013-01-25 14:12:40 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_fwd_flush ( ) - Checks the timestamp of the given nc packet .
2013-01-25 14:12:40 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ nc_path : the nc path the packet belongs to
* @ nc_packet : the nc packet to be checked
*
* Checks whether the given nc packet has hit its forward timeout . If so , the
* packet is no longer delayed , immediately sent and the entry deleted from the
* queue . Has to be called with the appropriate locks .
*
2015-09-15 20:00:48 +03:00
* Return : false as soon as the entry in the fifo queue has not been timed out
2013-01-25 14:12:40 +04:00
* yet and true otherwise .
*/
static bool batadv_nc_fwd_flush ( struct batadv_priv * bat_priv ,
struct batadv_nc_path * nc_path ,
struct batadv_nc_packet * nc_packet )
{
unsigned long timeout = bat_priv - > nc . max_fwd_delay ;
2015-06-21 15:45:14 +03:00
lockdep_assert_held ( & nc_path - > packet_list_lock ) ;
2013-01-25 14:12:40 +04:00
/* Packets are added to tail, so the remaining packets did not time
* out and we can stop processing the current queue
*/
if ( atomic_read ( & bat_priv - > mesh_state ) = = BATADV_MESH_ACTIVE & &
! batadv_has_timed_out ( nc_packet - > timestamp , timeout ) )
return false ;
/* Send packet */
batadv_inc_counter ( bat_priv , BATADV_CNT_FORWARD ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_FORWARD_BYTES ,
nc_packet - > skb - > len + ETH_HLEN ) ;
list_del ( & nc_packet - > list ) ;
batadv_nc_send_packet ( nc_packet ) ;
return true ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_process_nc_paths ( ) - traverse given nc packet pool and free timed
* out nc packets
2013-01-25 14:12:40 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ hash : to be processed hash table
* @ process_fn : Function called to process given nc packet . Should return true
* to encourage this function to proceed with the next packet .
* Otherwise the rest of the current queue is skipped .
*/
static void
batadv_nc_process_nc_paths ( struct batadv_priv * bat_priv ,
struct batadv_hashtable * hash ,
bool ( * process_fn ) ( struct batadv_priv * ,
struct batadv_nc_path * ,
struct batadv_nc_packet * ) )
{
struct hlist_head * head ;
struct batadv_nc_packet * nc_packet , * nc_packet_tmp ;
struct batadv_nc_path * nc_path ;
bool ret ;
int i ;
if ( ! hash )
return ;
/* Loop hash table bins */
for ( i = 0 ; i < hash - > size ; i + + ) {
head = & hash - > table [ i ] ;
/* Loop coding paths */
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( nc_path , head , hash_entry ) {
/* Loop packets */
spin_lock_bh ( & nc_path - > packet_list_lock ) ;
list_for_each_entry_safe ( nc_packet , nc_packet_tmp ,
& nc_path - > packet_list , list ) {
ret = process_fn ( bat_priv , nc_path , nc_packet ) ;
if ( ! ret )
break ;
}
spin_unlock_bh ( & nc_path - > packet_list_lock ) ;
}
rcu_read_unlock ( ) ;
}
}
2013-01-25 14:12:38 +04:00
/**
2020-06-01 21:13:21 +03:00
* batadv_nc_worker ( ) - periodic task for housekeeping related to network
2017-12-02 21:51:47 +03:00
* coding
2013-01-25 14:12:38 +04:00
* @ work : kernel work struct
*/
static void batadv_nc_worker ( struct work_struct * work )
{
struct delayed_work * delayed_work ;
struct batadv_priv_nc * priv_nc ;
struct batadv_priv * bat_priv ;
2013-01-25 14:12:40 +04:00
unsigned long timeout ;
2013-01-25 14:12:38 +04:00
2015-12-28 18:43:37 +03:00
delayed_work = to_delayed_work ( work ) ;
2013-01-25 14:12:38 +04:00
priv_nc = container_of ( delayed_work , struct batadv_priv_nc , work ) ;
bat_priv = container_of ( priv_nc , struct batadv_priv , nc ) ;
2013-01-25 14:12:39 +04:00
batadv_nc_purge_orig_hash ( bat_priv ) ;
2013-01-25 14:12:40 +04:00
batadv_nc_purge_paths ( bat_priv , bat_priv - > nc . coding_hash ,
batadv_nc_to_purge_nc_path_coding ) ;
2013-01-25 14:12:42 +04:00
batadv_nc_purge_paths ( bat_priv , bat_priv - > nc . decoding_hash ,
batadv_nc_to_purge_nc_path_decoding ) ;
2013-01-25 14:12:40 +04:00
timeout = bat_priv - > nc . max_fwd_delay ;
if ( batadv_has_timed_out ( bat_priv - > nc . timestamp_fwd_flush , timeout ) ) {
batadv_nc_process_nc_paths ( bat_priv , bat_priv - > nc . coding_hash ,
batadv_nc_fwd_flush ) ;
bat_priv - > nc . timestamp_fwd_flush = jiffies ;
}
2013-01-25 14:12:39 +04:00
2013-01-25 14:12:42 +04:00
if ( batadv_has_timed_out ( bat_priv - > nc . timestamp_sniffed_purge ,
bat_priv - > nc . max_buffer_time ) ) {
batadv_nc_process_nc_paths ( bat_priv , bat_priv - > nc . decoding_hash ,
batadv_nc_sniffed_purge ) ;
bat_priv - > nc . timestamp_sniffed_purge = jiffies ;
}
2013-01-25 14:12:38 +04:00
/* Schedule a new check */
batadv_nc_start_timer ( bat_priv ) ;
}
2013-01-25 14:12:39 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_can_nc_with_orig ( ) - checks whether the given orig node is suitable
* for coding or not
2013-01-25 14:12:39 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ orig_node : neighboring orig node which may be used as nc candidate
* @ ogm_packet : incoming ogm packet also used for the checks
*
2015-09-15 20:00:48 +03:00
* Return : true if :
2013-01-25 14:12:39 +04:00
* 1 ) The OGM must have the most recent sequence number .
* 2 ) The TTL must be decremented by one and only one .
* 3 ) The OGM must be received from the first hop from orig_node .
* 4 ) The TQ value of the OGM must be above bat_priv - > nc . min_tq .
*/
static bool batadv_can_nc_with_orig ( struct batadv_priv * bat_priv ,
struct batadv_orig_node * orig_node ,
struct batadv_ogm_packet * ogm_packet )
{
2013-11-13 22:14:47 +04:00
struct batadv_orig_ifinfo * orig_ifinfo ;
2015-05-26 19:34:26 +03:00
u32 last_real_seqno ;
u8 last_ttl ;
2013-11-13 22:14:47 +04:00
orig_ifinfo = batadv_orig_ifinfo_get ( orig_node , BATADV_IF_DEFAULT ) ;
if ( ! orig_ifinfo )
2013-01-25 14:12:39 +04:00
return false ;
2013-11-13 22:14:47 +04:00
last_ttl = orig_ifinfo - > last_ttl ;
last_real_seqno = orig_ifinfo - > last_real_seqno ;
2016-01-17 13:01:13 +03:00
batadv_orig_ifinfo_put ( orig_ifinfo ) ;
2013-11-13 22:14:47 +04:00
if ( last_real_seqno ! = ntohl ( ogm_packet - > seqno ) )
return false ;
if ( last_ttl ! = ogm_packet - > ttl + 1 )
2013-01-25 14:12:39 +04:00
return false ;
if ( ! batadv_compare_eth ( ogm_packet - > orig , ogm_packet - > prev_sender ) )
return false ;
if ( ogm_packet - > tq < bat_priv - > nc . min_tq )
return false ;
return true ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_find_nc_node ( ) - search for an existing nc node and return it
2013-01-25 14:12:39 +04:00
* @ orig_node : orig node originating the ogm packet
* @ orig_neigh_node : neighboring orig node from which we received the ogm packet
* ( can be equal to orig_node )
* @ in_coding : traverse incoming or outgoing network coding list
*
2015-09-15 20:00:48 +03:00
* Return : the nc_node if found , NULL otherwise .
2013-01-25 14:12:39 +04:00
*/
2016-03-05 17:56:01 +03:00
static struct batadv_nc_node *
batadv_nc_find_nc_node ( struct batadv_orig_node * orig_node ,
struct batadv_orig_node * orig_neigh_node ,
bool in_coding )
2013-01-25 14:12:39 +04:00
{
struct batadv_nc_node * nc_node , * nc_node_out = NULL ;
struct list_head * list ;
if ( in_coding )
list = & orig_neigh_node - > in_coding_list ;
else
list = & orig_neigh_node - > out_coding_list ;
/* Traverse list of nc_nodes to orig_node */
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( nc_node , list , list ) {
if ( ! batadv_compare_eth ( nc_node - > addr , orig_node - > orig ) )
continue ;
2016-01-16 12:29:45 +03:00
if ( ! kref_get_unless_zero ( & nc_node - > refcount ) )
2013-01-25 14:12:39 +04:00
continue ;
/* Found a match */
nc_node_out = nc_node ;
break ;
}
rcu_read_unlock ( ) ;
return nc_node_out ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_get_nc_node ( ) - retrieves an nc node or creates the entry if it was
2013-01-25 14:12:39 +04:00
* not found
* @ bat_priv : the bat priv with all the soft interface information
* @ orig_node : orig node originating the ogm packet
* @ orig_neigh_node : neighboring orig node from which we received the ogm packet
* ( can be equal to orig_node )
* @ in_coding : traverse incoming or outgoing network coding list
*
2015-09-15 20:00:48 +03:00
* Return : the nc_node if found or created , NULL in case of an error .
2013-01-25 14:12:39 +04:00
*/
2016-03-05 17:56:01 +03:00
static struct batadv_nc_node *
batadv_nc_get_nc_node ( struct batadv_priv * bat_priv ,
struct batadv_orig_node * orig_node ,
struct batadv_orig_node * orig_neigh_node ,
bool in_coding )
2013-01-25 14:12:39 +04:00
{
struct batadv_nc_node * nc_node ;
spinlock_t * lock ; /* Used to lock list selected by "int in_coding" */
struct list_head * list ;
2018-08-12 22:04:42 +03:00
/* Select ingoing or outgoing coding node */
if ( in_coding ) {
lock = & orig_neigh_node - > in_coding_list_lock ;
list = & orig_neigh_node - > in_coding_list ;
} else {
lock = & orig_neigh_node - > out_coding_list_lock ;
list = & orig_neigh_node - > out_coding_list ;
}
spin_lock_bh ( lock ) ;
2013-01-25 14:12:39 +04:00
/* Check if nc_node is already added */
nc_node = batadv_nc_find_nc_node ( orig_node , orig_neigh_node , in_coding ) ;
/* Node found */
if ( nc_node )
2018-08-12 22:04:42 +03:00
goto unlock ;
2013-01-25 14:12:39 +04:00
nc_node = kzalloc ( sizeof ( * nc_node ) , GFP_ATOMIC ) ;
if ( ! nc_node )
2018-08-12 22:04:42 +03:00
goto unlock ;
2013-01-25 14:12:39 +04:00
/* Initialize nc_node */
INIT_LIST_HEAD ( & nc_node - > list ) ;
2016-01-16 12:29:45 +03:00
kref_init ( & nc_node - > refcount ) ;
2016-07-15 18:39:21 +03:00
ether_addr_copy ( nc_node - > addr , orig_node - > orig ) ;
kref_get ( & orig_neigh_node - > refcount ) ;
nc_node - > orig_node = orig_neigh_node ;
2013-01-25 14:12:39 +04:00
batadv_dbg ( BATADV_DBG_NC , bat_priv , " Adding nc_node %pM -> %pM \n " ,
nc_node - > addr , nc_node - > orig_node - > orig ) ;
/* Add nc_node to orig_node */
2016-07-15 18:39:30 +03:00
kref_get ( & nc_node - > refcount ) ;
2013-01-25 14:12:39 +04:00
list_add_tail_rcu ( & nc_node - > list , list ) ;
2018-08-12 22:04:42 +03:00
unlock :
2013-01-25 14:12:39 +04:00
spin_unlock_bh ( lock ) ;
return nc_node ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_update_nc_node ( ) - updates stored incoming and outgoing nc node
2015-05-31 11:10:20 +03:00
* structs ( best called on incoming OGMs )
2013-01-25 14:12:39 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ orig_node : orig node originating the ogm packet
* @ orig_neigh_node : neighboring orig node from which we received the ogm packet
* ( can be equal to orig_node )
* @ ogm_packet : incoming ogm packet
* @ is_single_hop_neigh : orig_node is a single hop neighbor
*/
void batadv_nc_update_nc_node ( struct batadv_priv * bat_priv ,
struct batadv_orig_node * orig_node ,
struct batadv_orig_node * orig_neigh_node ,
struct batadv_ogm_packet * ogm_packet ,
int is_single_hop_neigh )
{
2015-06-09 21:50:49 +03:00
struct batadv_nc_node * in_nc_node = NULL ;
struct batadv_nc_node * out_nc_node = NULL ;
2013-01-25 14:12:39 +04:00
/* Check if network coding is enabled */
if ( ! atomic_read ( & bat_priv - > network_coding ) )
goto out ;
2013-04-23 17:40:00 +04:00
/* check if orig node is network coding enabled */
2015-06-16 18:10:23 +03:00
if ( ! test_bit ( BATADV_ORIG_CAPA_HAS_NC , & orig_node - > capabilities ) )
2013-04-23 17:40:00 +04:00
goto out ;
2013-01-25 14:12:39 +04:00
/* accept ogms from 'good' neighbors and single hop neighbors */
if ( ! batadv_can_nc_with_orig ( bat_priv , orig_node , ogm_packet ) & &
! is_single_hop_neigh )
goto out ;
/* Add orig_node as in_nc_node on hop */
in_nc_node = batadv_nc_get_nc_node ( bat_priv , orig_node ,
orig_neigh_node , true ) ;
if ( ! in_nc_node )
goto out ;
in_nc_node - > last_seen = jiffies ;
/* Add hop as out_nc_node on orig_node */
out_nc_node = batadv_nc_get_nc_node ( bat_priv , orig_neigh_node ,
orig_node , false ) ;
if ( ! out_nc_node )
goto out ;
out_nc_node - > last_seen = jiffies ;
out :
2021-08-08 20:11:08 +03:00
batadv_nc_node_put ( in_nc_node ) ;
batadv_nc_node_put ( out_nc_node ) ;
2013-01-25 14:12:39 +04:00
}
2013-01-25 14:12:40 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_get_path ( ) - get existing nc_path or allocate a new one
2013-01-25 14:12:40 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ hash : hash table containing the nc path
* @ src : ethernet source address - first half of the nc path search key
* @ dst : ethernet destination address - second half of the nc path search key
*
2015-09-15 20:00:48 +03:00
* Return : pointer to nc_path if the path was found or created , returns NULL
2013-01-25 14:12:40 +04:00
* on error .
*/
static struct batadv_nc_path * batadv_nc_get_path ( struct batadv_priv * bat_priv ,
struct batadv_hashtable * hash ,
2015-05-26 19:34:26 +03:00
u8 * src ,
u8 * dst )
2013-01-25 14:12:40 +04:00
{
int hash_added ;
struct batadv_nc_path * nc_path , nc_path_key ;
batadv_nc_hash_key_gen ( & nc_path_key , src , dst ) ;
/* Search for existing nc_path */
nc_path = batadv_nc_hash_find ( hash , ( void * ) & nc_path_key ) ;
if ( nc_path ) {
/* Set timestamp to delay removal of nc_path */
nc_path - > last_valid = jiffies ;
return nc_path ;
}
/* No existing nc_path was found; create a new */
nc_path = kzalloc ( sizeof ( * nc_path ) , GFP_ATOMIC ) ;
if ( ! nc_path )
return NULL ;
/* Initialize nc_path */
INIT_LIST_HEAD ( & nc_path - > packet_list ) ;
spin_lock_init ( & nc_path - > packet_list_lock ) ;
2016-01-16 12:29:46 +03:00
kref_init ( & nc_path - > refcount ) ;
2013-01-25 14:12:40 +04:00
nc_path - > last_valid = jiffies ;
2014-01-22 03:42:11 +04:00
ether_addr_copy ( nc_path - > next_hop , dst ) ;
ether_addr_copy ( nc_path - > prev_hop , src ) ;
2013-01-25 14:12:40 +04:00
batadv_dbg ( BATADV_DBG_NC , bat_priv , " Adding nc_path %pM -> %pM \n " ,
nc_path - > prev_hop ,
nc_path - > next_hop ) ;
/* Add nc_path to hash table */
2016-07-15 18:39:31 +03:00
kref_get ( & nc_path - > refcount ) ;
2013-01-25 14:12:40 +04:00
hash_added = batadv_hash_add ( hash , batadv_nc_hash_compare ,
batadv_nc_hash_choose , & nc_path_key ,
& nc_path - > hash_entry ) ;
if ( hash_added < 0 ) {
kfree ( nc_path ) ;
return NULL ;
}
return nc_path ;
}
2013-01-25 14:12:41 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_random_weight_tq ( ) - scale the receivers TQ - value to avoid unfair
2013-01-25 14:12:41 +04:00
* selection of a receiver with slightly lower TQ than the other
* @ tq : to be weighted tq value
2015-10-31 14:29:30 +03:00
*
* Return : scaled tq value
2013-01-25 14:12:41 +04:00
*/
2015-05-26 19:34:26 +03:00
static u8 batadv_nc_random_weight_tq ( u8 tq )
2013-01-25 14:12:41 +04:00
{
/* randomize the estimated packet loss (max TQ - estimated TQ) */
2022-10-10 05:44:02 +03:00
u8 rand_tq = get_random_u32_below ( BATADV_TQ_MAX_VALUE + 1 - tq ) ;
2013-01-25 14:12:41 +04:00
/* convert to (randomized) estimated tq again */
return BATADV_TQ_MAX_VALUE - rand_tq ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_memxor ( ) - XOR destination with source
2013-01-25 14:12:41 +04:00
* @ dst : byte array to XOR into
* @ src : byte array to XOR from
* @ len : length of destination array
*/
static void batadv_nc_memxor ( char * dst , const char * src , unsigned int len )
{
unsigned int i ;
for ( i = 0 ; i < len ; + + i )
dst [ i ] ^ = src [ i ] ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_code_packets ( ) - code a received unicast_packet with an nc packet
2013-01-25 14:12:41 +04:00
* into a coded_packet and send it
* @ bat_priv : the bat priv with all the soft interface information
* @ skb : data skb to forward
* @ ethhdr : pointer to the ethernet header inside the skb
* @ nc_packet : structure containing the packet to the skb can be coded with
* @ neigh_node : next hop to forward packet to
*
2015-09-15 20:00:48 +03:00
* Return : true if both packets are consumed , false otherwise .
2013-01-25 14:12:41 +04:00
*/
static bool batadv_nc_code_packets ( struct batadv_priv * bat_priv ,
struct sk_buff * skb ,
struct ethhdr * ethhdr ,
struct batadv_nc_packet * nc_packet ,
struct batadv_neigh_node * neigh_node )
{
2015-05-26 19:34:26 +03:00
u8 tq_weighted_neigh , tq_weighted_coding , tq_tmp ;
2013-01-25 14:12:41 +04:00
struct sk_buff * skb_dest , * skb_src ;
struct batadv_unicast_packet * packet1 ;
struct batadv_unicast_packet * packet2 ;
struct batadv_coded_packet * coded_packet ;
2016-01-16 11:40:15 +03:00
struct batadv_neigh_node * neigh_tmp , * router_neigh , * first_dest ;
struct batadv_neigh_node * router_coding = NULL , * second_dest ;
2013-11-13 22:14:46 +04:00
struct batadv_neigh_ifinfo * router_neigh_ifinfo = NULL ;
struct batadv_neigh_ifinfo * router_coding_ifinfo = NULL ;
2016-01-16 11:40:15 +03:00
u8 * first_source , * second_source ;
2013-01-25 14:12:41 +04:00
__be32 packet_id1 , packet_id2 ;
size_t count ;
bool res = false ;
int coding_len ;
int unicast_size = sizeof ( * packet1 ) ;
int coded_size = sizeof ( * coded_packet ) ;
int header_add = coded_size - unicast_size ;
2013-11-13 22:14:47 +04:00
/* TODO: do we need to consider the outgoing interface for
* coded packets ?
*/
router_neigh = batadv_orig_router_get ( neigh_node - > orig_node ,
BATADV_IF_DEFAULT ) ;
2013-01-25 14:12:41 +04:00
if ( ! router_neigh )
goto out ;
2013-11-13 22:14:46 +04:00
router_neigh_ifinfo = batadv_neigh_ifinfo_get ( router_neigh ,
BATADV_IF_DEFAULT ) ;
if ( ! router_neigh_ifinfo )
goto out ;
2013-01-25 14:12:41 +04:00
neigh_tmp = nc_packet - > neigh_node ;
2013-11-13 22:14:47 +04:00
router_coding = batadv_orig_router_get ( neigh_tmp - > orig_node ,
BATADV_IF_DEFAULT ) ;
2013-01-25 14:12:41 +04:00
if ( ! router_coding )
goto out ;
2013-11-13 22:14:46 +04:00
router_coding_ifinfo = batadv_neigh_ifinfo_get ( router_coding ,
BATADV_IF_DEFAULT ) ;
if ( ! router_coding_ifinfo )
goto out ;
tq_tmp = router_neigh_ifinfo - > bat_iv . tq_avg ;
tq_weighted_neigh = batadv_nc_random_weight_tq ( tq_tmp ) ;
tq_tmp = router_coding_ifinfo - > bat_iv . tq_avg ;
tq_weighted_coding = batadv_nc_random_weight_tq ( tq_tmp ) ;
2013-01-25 14:12:41 +04:00
/* Select one destination for the MAC-header dst-field based on
* weighted TQ - values .
*/
if ( tq_weighted_neigh > = tq_weighted_coding ) {
/* Destination from nc_packet is selected for MAC-header */
2016-01-16 11:40:15 +03:00
first_dest = nc_packet - > neigh_node ;
2013-01-25 14:12:41 +04:00
first_source = nc_packet - > nc_path - > prev_hop ;
2016-01-16 11:40:15 +03:00
second_dest = neigh_node ;
2013-01-25 14:12:41 +04:00
second_source = ethhdr - > h_source ;
packet1 = ( struct batadv_unicast_packet * ) nc_packet - > skb - > data ;
packet2 = ( struct batadv_unicast_packet * ) skb - > data ;
packet_id1 = nc_packet - > packet_id ;
packet_id2 = batadv_skb_crc32 ( skb ,
skb - > data + sizeof ( * packet2 ) ) ;
} else {
/* Destination for skb is selected for MAC-header */
2016-01-16 11:40:15 +03:00
first_dest = neigh_node ;
2013-01-25 14:12:41 +04:00
first_source = ethhdr - > h_source ;
2016-01-16 11:40:15 +03:00
second_dest = nc_packet - > neigh_node ;
2013-01-25 14:12:41 +04:00
second_source = nc_packet - > nc_path - > prev_hop ;
packet1 = ( struct batadv_unicast_packet * ) skb - > data ;
packet2 = ( struct batadv_unicast_packet * ) nc_packet - > skb - > data ;
packet_id1 = batadv_skb_crc32 ( skb ,
skb - > data + sizeof ( * packet1 ) ) ;
packet_id2 = nc_packet - > packet_id ;
}
/* Instead of zero padding the smallest data buffer, we
* code into the largest .
*/
if ( skb - > len < = nc_packet - > skb - > len ) {
skb_dest = nc_packet - > skb ;
skb_src = skb ;
} else {
skb_dest = skb ;
skb_src = nc_packet - > skb ;
}
/* coding_len is used when decoding the packet shorter packet */
coding_len = skb_src - > len - unicast_size ;
if ( skb_linearize ( skb_dest ) < 0 | | skb_linearize ( skb_src ) < 0 )
goto out ;
skb_push ( skb_dest , header_add ) ;
coded_packet = ( struct batadv_coded_packet * ) skb_dest - > data ;
skb_reset_mac_header ( skb_dest ) ;
2013-12-02 23:38:31 +04:00
coded_packet - > packet_type = BATADV_CODED ;
coded_packet - > version = BATADV_COMPAT_VERSION ;
coded_packet - > ttl = packet1 - > ttl ;
2013-01-25 14:12:41 +04:00
/* Info about first unicast packet */
2014-01-22 03:42:11 +04:00
ether_addr_copy ( coded_packet - > first_source , first_source ) ;
ether_addr_copy ( coded_packet - > first_orig_dest , packet1 - > dest ) ;
2013-01-25 14:12:41 +04:00
coded_packet - > first_crc = packet_id1 ;
coded_packet - > first_ttvn = packet1 - > ttvn ;
/* Info about second unicast packet */
2016-01-16 11:40:15 +03:00
ether_addr_copy ( coded_packet - > second_dest , second_dest - > addr ) ;
2014-01-22 03:42:11 +04:00
ether_addr_copy ( coded_packet - > second_source , second_source ) ;
ether_addr_copy ( coded_packet - > second_orig_dest , packet2 - > dest ) ;
2013-01-25 14:12:41 +04:00
coded_packet - > second_crc = packet_id2 ;
2013-12-02 23:38:31 +04:00
coded_packet - > second_ttl = packet2 - > ttl ;
2013-01-25 14:12:41 +04:00
coded_packet - > second_ttvn = packet2 - > ttvn ;
coded_packet - > coded_len = htons ( coding_len ) ;
/* This is where the magic happens: Code skb_src into skb_dest */
batadv_nc_memxor ( skb_dest - > data + coded_size ,
skb_src - > data + unicast_size , coding_len ) ;
/* Update counters accordingly */
if ( BATADV_SKB_CB ( skb_src ) - > decoded & &
BATADV_SKB_CB ( skb_dest ) - > decoded ) {
/* Both packets are recoded */
count = skb_src - > len + ETH_HLEN ;
count + = skb_dest - > len + ETH_HLEN ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_RECODE , 2 ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_RECODE_BYTES , count ) ;
} else if ( ! BATADV_SKB_CB ( skb_src ) - > decoded & &
! BATADV_SKB_CB ( skb_dest ) - > decoded ) {
/* Both packets are newly coded */
count = skb_src - > len + ETH_HLEN ;
count + = skb_dest - > len + ETH_HLEN ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_CODE , 2 ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_CODE_BYTES , count ) ;
} else if ( BATADV_SKB_CB ( skb_src ) - > decoded & &
! BATADV_SKB_CB ( skb_dest ) - > decoded ) {
/* skb_src recoded and skb_dest is newly coded */
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_RECODE ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_RECODE_BYTES ,
skb_src - > len + ETH_HLEN ) ;
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_CODE ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_CODE_BYTES ,
skb_dest - > len + ETH_HLEN ) ;
} else if ( ! BATADV_SKB_CB ( skb_src ) - > decoded & &
BATADV_SKB_CB ( skb_dest ) - > decoded ) {
/* skb_src is newly coded and skb_dest is recoded */
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_CODE ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_CODE_BYTES ,
skb_src - > len + ETH_HLEN ) ;
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_RECODE ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_RECODE_BYTES ,
skb_dest - > len + ETH_HLEN ) ;
}
/* skb_src is now coded into skb_dest, so free it */
2016-07-17 22:04:00 +03:00
consume_skb ( skb_src ) ;
2013-01-25 14:12:41 +04:00
/* avoid duplicate free of skb from nc_packet */
nc_packet - > skb = NULL ;
2016-07-17 22:04:00 +03:00
batadv_nc_packet_free ( nc_packet , false ) ;
2013-01-25 14:12:41 +04:00
/* Send the coded packet and return true */
2016-01-16 11:40:15 +03:00
batadv_send_unicast_skb ( skb_dest , first_dest ) ;
2013-01-25 14:12:41 +04:00
res = true ;
out :
2021-08-08 20:11:08 +03:00
batadv_neigh_node_put ( router_neigh ) ;
batadv_neigh_node_put ( router_coding ) ;
batadv_neigh_ifinfo_put ( router_neigh_ifinfo ) ;
batadv_neigh_ifinfo_put ( router_coding_ifinfo ) ;
2013-01-25 14:12:41 +04:00
return res ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_coding_possible ( ) - true if a decoded skb is available at dst .
2013-01-25 14:12:41 +04:00
* @ skb : data skb to forward
* @ dst : destination mac address of the other skb to code with
* @ src : source mac address of skb
*
* Whenever we network code a packet we have to check whether we received it in
* a network coded form . If so , we may not be able to use it for coding because
* some neighbors may also have received ( overheard ) the packet in the network
* coded form without being able to decode it . It is hard to know which of the
* neighboring nodes was able to decode the packet , therefore we can only
* re - code the packet if the source of the previous encoded packet is involved .
* Since the source encoded the packet we can be certain it has all necessary
* decode information .
*
2015-09-15 20:00:48 +03:00
* Return : true if coding of a decoded packet is allowed .
2013-01-25 14:12:41 +04:00
*/
2015-05-26 19:34:26 +03:00
static bool batadv_nc_skb_coding_possible ( struct sk_buff * skb , u8 * dst , u8 * src )
2013-01-25 14:12:41 +04:00
{
if ( BATADV_SKB_CB ( skb ) - > decoded & & ! batadv_compare_eth ( dst , src ) )
return false ;
2014-09-01 16:37:25 +04:00
return true ;
2013-01-25 14:12:41 +04:00
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_path_search ( ) - Find the coding path matching in_nc_node and
2013-01-25 14:12:41 +04:00
* out_nc_node to retrieve a buffered packet that can be used for coding .
* @ bat_priv : the bat priv with all the soft interface information
* @ in_nc_node : pointer to skb next hop ' s neighbor nc node
* @ out_nc_node : pointer to skb source ' s neighbor nc node
* @ skb : data skb to forward
* @ eth_dst : next hop mac address of skb
*
2015-09-15 20:00:48 +03:00
* Return : true if coding of a decoded skb is allowed .
2013-01-25 14:12:41 +04:00
*/
static struct batadv_nc_packet *
batadv_nc_path_search ( struct batadv_priv * bat_priv ,
struct batadv_nc_node * in_nc_node ,
struct batadv_nc_node * out_nc_node ,
struct sk_buff * skb ,
2015-05-26 19:34:26 +03:00
u8 * eth_dst )
2013-01-25 14:12:41 +04:00
{
struct batadv_nc_path * nc_path , nc_path_key ;
struct batadv_nc_packet * nc_packet_out = NULL ;
struct batadv_nc_packet * nc_packet , * nc_packet_tmp ;
struct batadv_hashtable * hash = bat_priv - > nc . coding_hash ;
int idx ;
if ( ! hash )
return NULL ;
/* Create almost path key */
batadv_nc_hash_key_gen ( & nc_path_key , in_nc_node - > addr ,
out_nc_node - > addr ) ;
idx = batadv_nc_hash_choose ( & nc_path_key , hash - > size ) ;
/* Check for coding opportunities in this nc_path */
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( nc_path , & hash - > table [ idx ] , hash_entry ) {
if ( ! batadv_compare_eth ( nc_path - > prev_hop , in_nc_node - > addr ) )
continue ;
if ( ! batadv_compare_eth ( nc_path - > next_hop , out_nc_node - > addr ) )
continue ;
spin_lock_bh ( & nc_path - > packet_list_lock ) ;
if ( list_empty ( & nc_path - > packet_list ) ) {
spin_unlock_bh ( & nc_path - > packet_list_lock ) ;
continue ;
}
list_for_each_entry_safe ( nc_packet , nc_packet_tmp ,
& nc_path - > packet_list , list ) {
if ( ! batadv_nc_skb_coding_possible ( nc_packet - > skb ,
eth_dst ,
in_nc_node - > addr ) )
continue ;
/* Coding opportunity is found! */
list_del ( & nc_packet - > list ) ;
nc_packet_out = nc_packet ;
break ;
}
spin_unlock_bh ( & nc_path - > packet_list_lock ) ;
break ;
}
rcu_read_unlock ( ) ;
return nc_packet_out ;
}
/**
2020-06-01 21:13:21 +03:00
* batadv_nc_skb_src_search ( ) - Loops through the list of neighboring nodes of
2017-12-02 21:51:47 +03:00
* the skb ' s sender ( may be equal to the originator ) .
2013-01-25 14:12:41 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ skb : data skb to forward
* @ eth_dst : next hop mac address of skb
* @ eth_src : source mac address of skb
* @ in_nc_node : pointer to skb next hop ' s neighbor nc node
*
2015-09-15 20:00:48 +03:00
* Return : an nc packet if a suitable coding packet was found , NULL otherwise .
2013-01-25 14:12:41 +04:00
*/
static struct batadv_nc_packet *
batadv_nc_skb_src_search ( struct batadv_priv * bat_priv ,
struct sk_buff * skb ,
2015-05-26 19:34:26 +03:00
u8 * eth_dst ,
u8 * eth_src ,
2013-01-25 14:12:41 +04:00
struct batadv_nc_node * in_nc_node )
{
struct batadv_orig_node * orig_node ;
struct batadv_nc_node * out_nc_node ;
struct batadv_nc_packet * nc_packet = NULL ;
orig_node = batadv_orig_hash_find ( bat_priv , eth_src ) ;
if ( ! orig_node )
return NULL ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( out_nc_node ,
& orig_node - > out_coding_list , list ) {
/* Check if the skb is decoded and if recoding is possible */
if ( ! batadv_nc_skb_coding_possible ( skb ,
out_nc_node - > addr , eth_src ) )
continue ;
/* Search for an opportunity in this nc_path */
nc_packet = batadv_nc_path_search ( bat_priv , in_nc_node ,
out_nc_node , skb , eth_dst ) ;
if ( nc_packet )
break ;
}
rcu_read_unlock ( ) ;
2016-01-17 13:01:09 +03:00
batadv_orig_node_put ( orig_node ) ;
2013-01-25 14:12:41 +04:00
return nc_packet ;
}
2013-01-25 14:12:42 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_store_before_coding ( ) - set the ethernet src and dst of the
2013-01-25 14:12:42 +04:00
* unicast skb before it is stored for use in later decoding
* @ bat_priv : the bat priv with all the soft interface information
* @ skb : data skb to store
* @ eth_dst_new : new destination mac address of skb
*/
static void batadv_nc_skb_store_before_coding ( struct batadv_priv * bat_priv ,
struct sk_buff * skb ,
2015-05-26 19:34:26 +03:00
u8 * eth_dst_new )
2013-01-25 14:12:42 +04:00
{
struct ethhdr * ethhdr ;
/* Copy skb header to change the mac header */
2014-06-12 02:36:26 +04:00
skb = pskb_copy_for_clone ( skb , GFP_ATOMIC ) ;
2013-01-25 14:12:42 +04:00
if ( ! skb )
return ;
/* Set the mac header as if we actually sent the packet uncoded */
2013-04-08 17:08:18 +04:00
ethhdr = eth_hdr ( skb ) ;
2014-01-22 03:42:11 +04:00
ether_addr_copy ( ethhdr - > h_source , ethhdr - > h_dest ) ;
ether_addr_copy ( ethhdr - > h_dest , eth_dst_new ) ;
2013-01-25 14:12:42 +04:00
/* Set data pointer to MAC header to mimic packets from our tx path */
skb_push ( skb , ETH_HLEN ) ;
/* Add the packet to the decoding packet pool */
batadv_nc_skb_store_for_decoding ( bat_priv , skb ) ;
/* batadv_nc_skb_store_for_decoding() clones the skb, so we must free
* our ref
*/
2016-07-17 22:04:00 +03:00
consume_skb ( skb ) ;
2013-01-25 14:12:42 +04:00
}
2013-01-25 14:12:41 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_dst_search ( ) - Loops through list of neighboring nodes to dst .
2013-01-25 14:12:41 +04:00
* @ skb : data skb to forward
* @ neigh_node : next hop to forward packet to
* @ ethhdr : pointer to the ethernet header inside the skb
*
2020-06-01 21:13:21 +03:00
* Loops through the list of neighboring nodes the next hop has a good
* connection to ( receives OGMs with a sufficient quality ) . We need to find a
* neighbor of our next hop that potentially sent a packet which our next hop
* also received ( overheard ) and has stored for later decoding .
2013-01-25 14:12:41 +04:00
*
2015-09-15 20:00:48 +03:00
* Return : true if the skb was consumed ( encoded packet sent ) or false otherwise
2013-01-25 14:12:41 +04:00
*/
static bool batadv_nc_skb_dst_search ( struct sk_buff * skb ,
struct batadv_neigh_node * neigh_node ,
struct ethhdr * ethhdr )
{
struct net_device * netdev = neigh_node - > if_incoming - > soft_iface ;
struct batadv_priv * bat_priv = netdev_priv ( netdev ) ;
struct batadv_orig_node * orig_node = neigh_node - > orig_node ;
struct batadv_nc_node * nc_node ;
struct batadv_nc_packet * nc_packet = NULL ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( nc_node , & orig_node - > in_coding_list , list ) {
/* Search for coding opportunity with this in_nc_node */
nc_packet = batadv_nc_skb_src_search ( bat_priv , skb ,
neigh_node - > addr ,
ethhdr - > h_source , nc_node ) ;
/* Opportunity was found, so stop searching */
if ( nc_packet )
break ;
}
rcu_read_unlock ( ) ;
if ( ! nc_packet )
return false ;
2013-01-25 14:12:42 +04:00
/* Save packets for later decoding */
batadv_nc_skb_store_before_coding ( bat_priv , skb ,
neigh_node - > addr ) ;
batadv_nc_skb_store_before_coding ( bat_priv , nc_packet - > skb ,
nc_packet - > neigh_node - > addr ) ;
2013-01-25 14:12:41 +04:00
/* Code and send packets */
if ( batadv_nc_code_packets ( bat_priv , skb , ethhdr , nc_packet ,
neigh_node ) )
return true ;
/* out of mem ? Coding failed - we have to free the buffered packet
* to avoid memleaks . The skb passed as argument will be dealt with
* by the calling function .
*/
batadv_nc_send_packet ( nc_packet ) ;
return false ;
}
2013-01-25 14:12:40 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_add_to_path ( ) - buffer skb for later encoding / decoding
2013-01-25 14:12:40 +04:00
* @ skb : skb to add to path
* @ nc_path : path to add skb to
* @ neigh_node : next hop to forward packet to
* @ packet_id : checksum to identify packet
*
2015-09-15 20:00:48 +03:00
* Return : true if the packet was buffered or false in case of an error .
2013-01-25 14:12:40 +04:00
*/
static bool batadv_nc_skb_add_to_path ( struct sk_buff * skb ,
struct batadv_nc_path * nc_path ,
struct batadv_neigh_node * neigh_node ,
__be32 packet_id )
{
struct batadv_nc_packet * nc_packet ;
nc_packet = kzalloc ( sizeof ( * nc_packet ) , GFP_ATOMIC ) ;
if ( ! nc_packet )
return false ;
/* Initialize nc_packet */
nc_packet - > timestamp = jiffies ;
nc_packet - > packet_id = packet_id ;
nc_packet - > skb = skb ;
nc_packet - > neigh_node = neigh_node ;
nc_packet - > nc_path = nc_path ;
/* Add coding packet to list */
spin_lock_bh ( & nc_path - > packet_list_lock ) ;
list_add_tail ( & nc_packet - > list , & nc_path - > packet_list ) ;
spin_unlock_bh ( & nc_path - > packet_list_lock ) ;
return true ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_forward ( ) - try to code a packet or add it to the coding packet
2013-01-25 14:12:40 +04:00
* buffer
* @ skb : data skb to forward
* @ neigh_node : next hop to forward packet to
*
2015-09-15 20:00:48 +03:00
* Return : true if the skb was consumed ( encoded packet sent ) or false otherwise
2013-01-25 14:12:40 +04:00
*/
bool batadv_nc_skb_forward ( struct sk_buff * skb ,
2013-04-20 15:54:39 +04:00
struct batadv_neigh_node * neigh_node )
2013-01-25 14:12:40 +04:00
{
const struct net_device * netdev = neigh_node - > if_incoming - > soft_iface ;
struct batadv_priv * bat_priv = netdev_priv ( netdev ) ;
struct batadv_unicast_packet * packet ;
struct batadv_nc_path * nc_path ;
2013-04-20 15:54:39 +04:00
struct ethhdr * ethhdr = eth_hdr ( skb ) ;
2013-01-25 14:12:40 +04:00
__be32 packet_id ;
u8 * payload ;
/* Check if network coding is enabled */
if ( ! atomic_read ( & bat_priv - > network_coding ) )
goto out ;
/* We only handle unicast packets */
payload = skb_network_header ( skb ) ;
packet = ( struct batadv_unicast_packet * ) payload ;
2013-12-02 23:38:31 +04:00
if ( packet - > packet_type ! = BATADV_UNICAST )
2013-01-25 14:12:40 +04:00
goto out ;
2013-01-25 14:12:41 +04:00
/* Try to find a coding opportunity and send the skb if one is found */
if ( batadv_nc_skb_dst_search ( skb , neigh_node , ethhdr ) )
return true ;
2013-01-25 14:12:40 +04:00
/* Find or create a nc_path for this src-dst pair */
nc_path = batadv_nc_get_path ( bat_priv ,
bat_priv - > nc . coding_hash ,
ethhdr - > h_source ,
neigh_node - > addr ) ;
if ( ! nc_path )
goto out ;
/* Add skb to nc_path */
packet_id = batadv_skb_crc32 ( skb , payload + sizeof ( * packet ) ) ;
if ( ! batadv_nc_skb_add_to_path ( skb , nc_path , neigh_node , packet_id ) )
goto free_nc_path ;
/* Packet is consumed */
return true ;
free_nc_path :
2016-01-17 13:01:23 +03:00
batadv_nc_path_put ( nc_path ) ;
2013-01-25 14:12:40 +04:00
out :
/* Packet is not consumed */
return false ;
}
2013-01-25 14:12:42 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_store_for_decoding ( ) - save a clone of the skb which can be
* used when decoding coded packets
2013-01-25 14:12:42 +04:00
* @ bat_priv : the bat priv with all the soft interface information
* @ skb : data skb to store
*/
void batadv_nc_skb_store_for_decoding ( struct batadv_priv * bat_priv ,
struct sk_buff * skb )
{
struct batadv_unicast_packet * packet ;
struct batadv_nc_path * nc_path ;
2013-04-08 17:08:18 +04:00
struct ethhdr * ethhdr = eth_hdr ( skb ) ;
2013-01-25 14:12:42 +04:00
__be32 packet_id ;
u8 * payload ;
/* Check if network coding is enabled */
if ( ! atomic_read ( & bat_priv - > network_coding ) )
goto out ;
/* Check for supported packet type */
payload = skb_network_header ( skb ) ;
packet = ( struct batadv_unicast_packet * ) payload ;
2013-12-02 23:38:31 +04:00
if ( packet - > packet_type ! = BATADV_UNICAST )
2013-01-25 14:12:42 +04:00
goto out ;
/* Find existing nc_path or create a new */
nc_path = batadv_nc_get_path ( bat_priv ,
bat_priv - > nc . decoding_hash ,
ethhdr - > h_source ,
ethhdr - > h_dest ) ;
if ( ! nc_path )
goto out ;
/* Clone skb and adjust skb->data to point at batman header */
skb = skb_clone ( skb , GFP_ATOMIC ) ;
if ( unlikely ( ! skb ) )
goto free_nc_path ;
if ( unlikely ( ! pskb_may_pull ( skb , ETH_HLEN ) ) )
goto free_skb ;
if ( unlikely ( ! skb_pull_rcsum ( skb , ETH_HLEN ) ) )
goto free_skb ;
/* Add skb to nc_path */
packet_id = batadv_skb_crc32 ( skb , payload + sizeof ( * packet ) ) ;
if ( ! batadv_nc_skb_add_to_path ( skb , nc_path , NULL , packet_id ) )
goto free_skb ;
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_BUFFER ) ;
return ;
free_skb :
kfree_skb ( skb ) ;
free_nc_path :
2016-01-17 13:01:23 +03:00
batadv_nc_path_put ( nc_path ) ;
2013-01-25 14:12:42 +04:00
out :
return ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_store_sniffed_unicast ( ) - check if a received unicast packet
2013-01-25 14:12:42 +04:00
* should be saved in the decoding buffer and , if so , store it there
* @ bat_priv : the bat priv with all the soft interface information
* @ skb : unicast skb to store
*/
void batadv_nc_skb_store_sniffed_unicast ( struct batadv_priv * bat_priv ,
struct sk_buff * skb )
{
2013-04-08 17:08:18 +04:00
struct ethhdr * ethhdr = eth_hdr ( skb ) ;
2013-01-25 14:12:42 +04:00
2013-04-23 04:32:51 +04:00
if ( batadv_is_my_mac ( bat_priv , ethhdr - > h_dest ) )
2013-01-25 14:12:42 +04:00
return ;
/* Set data pointer to MAC header to mimic packets from our tx path */
skb_push ( skb , ETH_HLEN ) ;
batadv_nc_skb_store_for_decoding ( bat_priv , skb ) ;
}
2013-01-25 14:12:43 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_skb_decode_packet ( ) - decode given skb using the decode data stored
2013-01-25 14:12:43 +04:00
* in nc_packet
2013-04-23 04:32:51 +04:00
* @ bat_priv : the bat priv with all the soft interface information
2013-01-25 14:12:43 +04:00
* @ skb : unicast skb to decode
* @ nc_packet : decode data needed to decode the skb
*
2015-09-15 20:00:48 +03:00
* Return : pointer to decoded unicast packet if the packet was decoded or NULL
2013-01-25 14:12:43 +04:00
* in case of an error .
*/
static struct batadv_unicast_packet *
2013-04-23 04:32:51 +04:00
batadv_nc_skb_decode_packet ( struct batadv_priv * bat_priv , struct sk_buff * skb ,
2013-01-25 14:12:43 +04:00
struct batadv_nc_packet * nc_packet )
{
const int h_size = sizeof ( struct batadv_unicast_packet ) ;
const int h_diff = sizeof ( struct batadv_coded_packet ) - h_size ;
struct batadv_unicast_packet * unicast_packet ;
struct batadv_coded_packet coded_packet_tmp ;
struct ethhdr * ethhdr , ethhdr_tmp ;
2015-05-26 19:34:26 +03:00
u8 * orig_dest , ttl , ttvn ;
2013-01-25 14:12:43 +04:00
unsigned int coding_len ;
2013-05-07 15:25:02 +04:00
int err ;
2013-01-25 14:12:43 +04:00
/* Save headers temporarily */
memcpy ( & coded_packet_tmp , skb - > data , sizeof ( coded_packet_tmp ) ) ;
memcpy ( & ethhdr_tmp , skb_mac_header ( skb ) , sizeof ( ethhdr_tmp ) ) ;
if ( skb_cow ( skb , 0 ) < 0 )
return NULL ;
if ( unlikely ( ! skb_pull_rcsum ( skb , h_diff ) ) )
return NULL ;
/* Data points to batman header, so set mac header 14 bytes before
* and network to data
*/
skb_set_mac_header ( skb , - ETH_HLEN ) ;
skb_reset_network_header ( skb ) ;
/* Reconstruct original mac header */
2013-04-08 17:08:18 +04:00
ethhdr = eth_hdr ( skb ) ;
2014-01-06 20:09:46 +04:00
* ethhdr = ethhdr_tmp ;
2013-01-25 14:12:43 +04:00
/* Select the correct unicast header information based on the location
* of our mac address in the coded_packet header
*/
2013-04-23 04:32:51 +04:00
if ( batadv_is_my_mac ( bat_priv , coded_packet_tmp . second_dest ) ) {
2013-01-25 14:12:43 +04:00
/* If we are the second destination the packet was overheard,
* so the Ethernet address must be copied to h_dest and
* pkt_type changed from PACKET_OTHERHOST to PACKET_HOST
*/
2014-01-22 03:42:11 +04:00
ether_addr_copy ( ethhdr - > h_dest , coded_packet_tmp . second_dest ) ;
2013-01-25 14:12:43 +04:00
skb - > pkt_type = PACKET_HOST ;
orig_dest = coded_packet_tmp . second_orig_dest ;
ttl = coded_packet_tmp . second_ttl ;
ttvn = coded_packet_tmp . second_ttvn ;
} else {
orig_dest = coded_packet_tmp . first_orig_dest ;
2013-12-02 23:38:31 +04:00
ttl = coded_packet_tmp . ttl ;
2013-01-25 14:12:43 +04:00
ttvn = coded_packet_tmp . first_ttvn ;
}
coding_len = ntohs ( coded_packet_tmp . coded_len ) ;
if ( coding_len > skb - > len )
return NULL ;
/* Here the magic is reversed:
* extract the missing packet from the received coded packet
*/
batadv_nc_memxor ( skb - > data + h_size ,
nc_packet - > skb - > data + h_size ,
coding_len ) ;
/* Resize decoded skb if decoded with larger packet */
2013-05-07 15:25:02 +04:00
if ( nc_packet - > skb - > len > coding_len + h_size ) {
err = pskb_trim_rcsum ( skb , coding_len + h_size ) ;
if ( err )
return NULL ;
}
2013-01-25 14:12:43 +04:00
/* Create decoded unicast packet */
unicast_packet = ( struct batadv_unicast_packet * ) skb - > data ;
2013-12-02 23:38:31 +04:00
unicast_packet - > packet_type = BATADV_UNICAST ;
unicast_packet - > version = BATADV_COMPAT_VERSION ;
unicast_packet - > ttl = ttl ;
2014-01-22 03:42:11 +04:00
ether_addr_copy ( unicast_packet - > dest , orig_dest ) ;
2013-01-25 14:12:43 +04:00
unicast_packet - > ttvn = ttvn ;
2016-07-17 22:04:00 +03:00
batadv_nc_packet_free ( nc_packet , false ) ;
2013-01-25 14:12:43 +04:00
return unicast_packet ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_find_decoding_packet ( ) - search through buffered decoding data to
2013-01-25 14:12:43 +04:00
* find the data needed to decode the coded packet
* @ bat_priv : the bat priv with all the soft interface information
* @ ethhdr : pointer to the ethernet header inside the coded packet
* @ coded : coded packet we try to find decode data for
*
2015-09-15 20:00:48 +03:00
* Return : pointer to nc packet if the needed data was found or NULL otherwise .
2013-01-25 14:12:43 +04:00
*/
static struct batadv_nc_packet *
batadv_nc_find_decoding_packet ( struct batadv_priv * bat_priv ,
struct ethhdr * ethhdr ,
struct batadv_coded_packet * coded )
{
struct batadv_hashtable * hash = bat_priv - > nc . decoding_hash ;
struct batadv_nc_packet * tmp_nc_packet , * nc_packet = NULL ;
struct batadv_nc_path * nc_path , nc_path_key ;
2015-05-26 19:34:26 +03:00
u8 * dest , * source ;
2013-01-25 14:12:43 +04:00
__be32 packet_id ;
int index ;
if ( ! hash )
return NULL ;
/* Select the correct packet id based on the location of our mac-addr */
dest = ethhdr - > h_source ;
2013-04-23 04:32:51 +04:00
if ( ! batadv_is_my_mac ( bat_priv , coded - > second_dest ) ) {
2013-01-25 14:12:43 +04:00
source = coded - > second_source ;
packet_id = coded - > second_crc ;
} else {
source = coded - > first_source ;
packet_id = coded - > first_crc ;
}
batadv_nc_hash_key_gen ( & nc_path_key , source , dest ) ;
index = batadv_nc_hash_choose ( & nc_path_key , hash - > size ) ;
/* Search for matching coding path */
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( nc_path , & hash - > table [ index ] , hash_entry ) {
/* Find matching nc_packet */
spin_lock_bh ( & nc_path - > packet_list_lock ) ;
list_for_each_entry ( tmp_nc_packet ,
& nc_path - > packet_list , list ) {
if ( packet_id = = tmp_nc_packet - > packet_id ) {
list_del ( & tmp_nc_packet - > list ) ;
nc_packet = tmp_nc_packet ;
break ;
}
}
spin_unlock_bh ( & nc_path - > packet_list_lock ) ;
if ( nc_packet )
break ;
}
rcu_read_unlock ( ) ;
if ( ! nc_packet )
batadv_dbg ( BATADV_DBG_NC , bat_priv ,
" No decoding packet found for %u \n " , packet_id ) ;
return nc_packet ;
}
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_recv_coded_packet ( ) - try to decode coded packet and enqueue the
2013-01-25 14:12:43 +04:00
* resulting unicast packet
* @ skb : incoming coded packet
* @ recv_if : pointer to interface this packet was received on
2015-10-31 14:29:30 +03:00
*
* Return : NET_RX_SUCCESS if the packet has been consumed or NET_RX_DROP
* otherwise .
2013-01-25 14:12:43 +04:00
*/
static int batadv_nc_recv_coded_packet ( struct sk_buff * skb ,
struct batadv_hard_iface * recv_if )
{
struct batadv_priv * bat_priv = netdev_priv ( recv_if - > soft_iface ) ;
struct batadv_unicast_packet * unicast_packet ;
struct batadv_coded_packet * coded_packet ;
struct batadv_nc_packet * nc_packet ;
struct ethhdr * ethhdr ;
int hdr_size = sizeof ( * coded_packet ) ;
/* Check if network coding is enabled */
if ( ! atomic_read ( & bat_priv - > network_coding ) )
2016-07-17 22:04:04 +03:00
goto free_skb ;
2013-01-25 14:12:43 +04:00
/* Make sure we can access (and remove) header */
if ( unlikely ( ! pskb_may_pull ( skb , hdr_size ) ) )
2016-07-17 22:04:04 +03:00
goto free_skb ;
2013-01-25 14:12:43 +04:00
coded_packet = ( struct batadv_coded_packet * ) skb - > data ;
2013-04-08 17:08:18 +04:00
ethhdr = eth_hdr ( skb ) ;
2013-01-25 14:12:43 +04:00
/* Verify frame is destined for us */
2013-04-23 04:32:51 +04:00
if ( ! batadv_is_my_mac ( bat_priv , ethhdr - > h_dest ) & &
! batadv_is_my_mac ( bat_priv , coded_packet - > second_dest ) )
2016-07-17 22:04:04 +03:00
goto free_skb ;
2013-01-25 14:12:43 +04:00
/* Update stat counter */
2013-04-23 04:32:51 +04:00
if ( batadv_is_my_mac ( bat_priv , coded_packet - > second_dest ) )
2013-01-25 14:12:43 +04:00
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_SNIFFED ) ;
nc_packet = batadv_nc_find_decoding_packet ( bat_priv , ethhdr ,
coded_packet ) ;
if ( ! nc_packet ) {
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_DECODE_FAILED ) ;
2016-07-17 22:04:04 +03:00
goto free_skb ;
2013-01-25 14:12:43 +04:00
}
/* Make skb's linear, because decoding accesses the entire buffer */
if ( skb_linearize ( skb ) < 0 )
goto free_nc_packet ;
if ( skb_linearize ( nc_packet - > skb ) < 0 )
goto free_nc_packet ;
/* Decode the packet */
2013-04-23 04:32:51 +04:00
unicast_packet = batadv_nc_skb_decode_packet ( bat_priv , skb , nc_packet ) ;
2013-01-25 14:12:43 +04:00
if ( ! unicast_packet ) {
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_DECODE_FAILED ) ;
goto free_nc_packet ;
}
/* Mark packet as decoded to do correct recoding when forwarding */
BATADV_SKB_CB ( skb ) - > decoded = true ;
batadv_inc_counter ( bat_priv , BATADV_CNT_NC_DECODE ) ;
batadv_add_counter ( bat_priv , BATADV_CNT_NC_DECODE_BYTES ,
skb - > len + ETH_HLEN ) ;
return batadv_recv_unicast_packet ( skb , recv_if ) ;
free_nc_packet :
2016-07-17 22:04:00 +03:00
batadv_nc_packet_free ( nc_packet , true ) ;
2016-07-17 22:04:04 +03:00
free_skb :
kfree_skb ( skb ) ;
2013-01-25 14:12:43 +04:00
return NET_RX_DROP ;
}
2013-01-25 14:12:38 +04:00
/**
2017-12-02 21:51:47 +03:00
* batadv_nc_mesh_free ( ) - clean up network coding memory
2013-01-25 14:12:38 +04:00
* @ bat_priv : the bat priv with all the soft interface information
*/
2013-09-27 20:03:39 +04:00
void batadv_nc_mesh_free ( struct batadv_priv * bat_priv )
2013-01-25 14:12:38 +04:00
{
2013-04-23 17:40:00 +04:00
batadv_tvlv_container_unregister ( bat_priv , BATADV_TVLV_NC , 1 ) ;
batadv_tvlv_handler_unregister ( bat_priv , BATADV_TVLV_NC , 1 ) ;
2013-01-25 14:12:38 +04:00
cancel_delayed_work_sync ( & bat_priv - > nc . work ) ;
2013-01-25 14:12:42 +04:00
2013-01-25 14:12:40 +04:00
batadv_nc_purge_paths ( bat_priv , bat_priv - > nc . coding_hash , NULL ) ;
batadv_hash_destroy ( bat_priv - > nc . coding_hash ) ;
2013-01-25 14:12:42 +04:00
batadv_nc_purge_paths ( bat_priv , bat_priv - > nc . decoding_hash , NULL ) ;
batadv_hash_destroy ( bat_priv - > nc . decoding_hash ) ;
2013-01-25 14:12:38 +04:00
}