2017-11-19 15:05:11 +01:00
// SPDX-License-Identifier: GPL-2.0
2021-01-01 00:00:01 +01:00
/* Copyright (C) B.A.T.M.A.N. contributors:
2010-12-13 11:19:28 +00:00
*
* Simon Wunderlich , Marek Lindner
*/
# include "hash.h"
2015-04-17 19:40:28 +02:00
# include "main.h"
2017-11-19 17:12:02 +01:00
# include <linux/gfp.h>
2015-04-17 19:40:28 +02:00
# include <linux/lockdep.h>
# include <linux/slab.h>
2010-12-13 11:19:28 +00:00
/* clears the hash */
2012-06-05 22:31:28 +02:00
static void batadv_hash_init ( struct batadv_hashtable * hash )
2010-12-13 11:19:28 +00:00
{
2015-05-26 18:34:26 +02:00
u32 i ;
2010-12-13 11:19:28 +00:00
2012-06-17 16:27:22 +02:00
for ( i = 0 ; i < hash - > size ; i + + ) {
2010-12-13 11:19:28 +00:00
INIT_HLIST_HEAD ( & hash - > table [ i ] ) ;
2011-01-19 20:01:40 +00:00
spin_lock_init ( & hash - > list_locks [ i ] ) ;
}
2018-10-30 22:01:25 +01:00
atomic_set ( & hash - > generation , 0 ) ;
2010-12-13 11:19:28 +00:00
}
2017-12-02 19:51:53 +01:00
/**
* batadv_hash_destroy ( ) - Free only the hashtable and the hash itself
* @ hash : hash object to destroy
*/
2012-06-05 22:31:28 +02:00
void batadv_hash_destroy ( struct batadv_hashtable * hash )
2010-12-13 11:19:28 +00:00
{
2011-01-19 20:01:40 +00:00
kfree ( hash - > list_locks ) ;
2010-12-13 11:19:28 +00:00
kfree ( hash - > table ) ;
kfree ( hash ) ;
}
2017-12-02 19:51:53 +01:00
/**
* batadv_hash_new ( ) - Allocates and clears the hashtable
* @ size : number of hash buckets to allocate
*
* Return : newly allocated hashtable , NULL on errors
*/
2015-05-26 18:34:26 +02:00
struct batadv_hashtable * batadv_hash_new ( u32 size )
2010-12-13 11:19:28 +00:00
{
2012-06-05 22:31:28 +02:00
struct batadv_hashtable * hash ;
2010-12-13 11:19:28 +00:00
2011-05-14 23:14:54 +02:00
hash = kmalloc ( sizeof ( * hash ) , GFP_ATOMIC ) ;
2010-12-13 11:19:28 +00:00
if ( ! hash )
return NULL ;
2014-05-24 06:43:47 +02:00
hash - > table = kmalloc_array ( size , sizeof ( * hash - > table ) , GFP_ATOMIC ) ;
2011-01-19 20:01:40 +00:00
if ( ! hash - > table )
goto free_hash ;
2010-12-13 11:19:28 +00:00
2014-05-24 06:43:47 +02:00
hash - > list_locks = kmalloc_array ( size , sizeof ( * hash - > list_locks ) ,
GFP_ATOMIC ) ;
2011-01-19 20:01:40 +00:00
if ( ! hash - > list_locks )
goto free_table ;
2010-12-13 11:19:28 +00:00
2011-01-19 20:01:40 +00:00
hash - > size = size ;
2012-05-12 18:33:58 +02:00
batadv_hash_init ( hash ) ;
2010-12-13 11:19:28 +00:00
return hash ;
2011-01-19 20:01:40 +00:00
free_table :
kfree ( hash - > table ) ;
free_hash :
kfree ( hash ) ;
return NULL ;
}
2012-03-29 12:38:20 +02:00
2017-12-02 19:51:53 +01:00
/**
* batadv_hash_set_lock_class ( ) - Set specific lockdep class for hash spinlocks
* @ hash : hash object to modify
* @ key : lockdep class key address
*/
2012-06-05 22:31:28 +02:00
void batadv_hash_set_lock_class ( struct batadv_hashtable * hash ,
2012-03-29 12:38:20 +02:00
struct lock_class_key * key )
{
2015-05-26 18:34:26 +02:00
u32 i ;
2012-03-29 12:38:20 +02:00
for ( i = 0 ; i < hash - > size ; i + + )
lockdep_set_class ( & hash - > list_locks [ i ] , key ) ;
}