2005-10-16 18:33:22 +04:00
/*
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
* Copyright ( C ) 2004 - 2005 Red Hat , Inc . All rights reserved .
*
* This file is part of the device - mapper userspace tools .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU General Public License v .2 .
*
* 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 . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "lib.h"
2005-10-17 02:57:20 +04:00
struct dm_hash_node {
struct dm_hash_node * next ;
2005-10-16 18:33:22 +04:00
void * data ;
2006-01-04 19:07:27 +03:00
unsigned keylen ;
2005-10-16 18:33:22 +04:00
char key [ 0 ] ;
} ;
2005-10-17 02:57:20 +04:00
struct dm_hash_table {
2006-01-04 19:07:27 +03:00
unsigned num_nodes ;
unsigned num_slots ;
2005-10-17 02:57:20 +04:00
struct dm_hash_node * * slots ;
2005-10-16 18:33:22 +04:00
} ;
/* Permutation of the Integers 0 through 255 */
static unsigned char _nums [ ] = {
1 , 14 , 110 , 25 , 97 , 174 , 132 , 119 , 138 , 170 , 125 , 118 , 27 , 233 , 140 , 51 ,
87 , 197 , 177 , 107 , 234 , 169 , 56 , 68 , 30 , 7 , 173 , 73 , 188 , 40 , 36 , 65 ,
49 , 213 , 104 , 190 , 57 , 211 , 148 , 223 , 48 , 115 , 15 , 2 , 67 , 186 , 210 , 28 ,
12 , 181 , 103 , 70 , 22 , 58 , 75 , 78 , 183 , 167 , 238 , 157 , 124 , 147 , 172 ,
144 ,
176 , 161 , 141 , 86 , 60 , 66 , 128 , 83 , 156 , 241 , 79 , 46 , 168 , 198 , 41 , 254 ,
178 , 85 , 253 , 237 , 250 , 154 , 133 , 88 , 35 , 206 , 95 , 116 , 252 , 192 , 54 ,
221 ,
102 , 218 , 255 , 240 , 82 , 106 , 158 , 201 , 61 , 3 , 89 , 9 , 42 , 155 , 159 , 93 ,
166 , 80 , 50 , 34 , 175 , 195 , 100 , 99 , 26 , 150 , 16 , 145 , 4 , 33 , 8 , 189 ,
121 , 64 , 77 , 72 , 208 , 245 , 130 , 122 , 143 , 55 , 105 , 134 , 29 , 164 , 185 ,
194 ,
193 , 239 , 101 , 242 , 5 , 171 , 126 , 11 , 74 , 59 , 137 , 228 , 108 , 191 , 232 ,
139 ,
6 , 24 , 81 , 20 , 127 , 17 , 91 , 92 , 251 , 151 , 225 , 207 , 21 , 98 , 113 , 112 ,
84 , 226 , 18 , 214 , 199 , 187 , 13 , 32 , 94 , 220 , 224 , 212 , 247 , 204 , 196 ,
43 ,
249 , 236 , 45 , 244 , 111 , 182 , 153 , 136 , 129 , 90 , 217 , 202 , 19 , 165 , 231 ,
71 ,
230 , 142 , 96 , 227 , 62 , 179 , 246 , 114 , 162 , 53 , 160 , 215 , 205 , 180 , 47 ,
109 ,
44 , 38 , 31 , 149 , 135 , 0 , 216 , 52 , 63 , 23 , 37 , 69 , 39 , 117 , 146 , 184 ,
163 , 200 , 222 , 235 , 248 , 243 , 219 , 10 , 152 , 131 , 123 , 229 , 203 , 76 , 120 ,
209
} ;
2006-01-04 19:07:27 +03:00
static struct dm_hash_node * _create_node ( const char * str , unsigned len )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * n = dm_malloc ( sizeof ( * n ) + len ) ;
2005-10-16 18:33:22 +04:00
if ( n ) {
memcpy ( n - > key , str , len ) ;
n - > keylen = len ;
}
return n ;
}
2007-04-27 18:52:41 +04:00
static unsigned long _hash ( const char * str , unsigned len )
2005-10-16 18:33:22 +04:00
{
2006-01-04 19:07:27 +03:00
unsigned long h = 0 , g ;
unsigned i ;
2005-10-16 18:33:22 +04:00
for ( i = 0 ; i < len ; i + + ) {
h < < = 4 ;
2007-04-27 18:52:41 +04:00
h + = _nums [ ( unsigned char ) * str + + ] ;
2005-10-16 18:33:22 +04:00
g = h & ( ( unsigned long ) 0xf < < 16u ) ;
if ( g ) {
h ^ = g > > 16u ;
h ^ = g > > 5u ;
}
}
return h ;
}
2005-10-17 02:57:20 +04:00
struct dm_hash_table * dm_hash_create ( unsigned size_hint )
2005-10-16 18:33:22 +04:00
{
size_t len ;
unsigned new_size = 16u ;
2005-10-17 02:57:20 +04:00
struct dm_hash_table * hc = dm_malloc ( sizeof ( * hc ) ) ;
2005-10-16 18:33:22 +04:00
if ( ! hc ) {
stack ;
return 0 ;
}
memset ( hc , 0 , sizeof ( * hc ) ) ;
/* round size hint up to a power of two */
while ( new_size < size_hint )
new_size = new_size < < 1 ;
hc - > num_slots = new_size ;
len = sizeof ( * ( hc - > slots ) ) * new_size ;
2005-10-17 02:57:20 +04:00
if ( ! ( hc - > slots = dm_malloc ( len ) ) ) {
2005-10-16 18:33:22 +04:00
stack ;
goto bad ;
}
memset ( hc - > slots , 0 , len ) ;
return hc ;
bad :
2005-10-17 02:57:20 +04:00
dm_free ( hc - > slots ) ;
dm_free ( hc ) ;
2005-10-16 18:33:22 +04:00
return 0 ;
}
2005-10-17 02:57:20 +04:00
static void _free_nodes ( struct dm_hash_table * t )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * c , * n ;
2006-01-04 19:07:27 +03:00
unsigned i ;
2005-10-16 18:33:22 +04:00
for ( i = 0 ; i < t - > num_slots ; i + + )
for ( c = t - > slots [ i ] ; c ; c = n ) {
n = c - > next ;
2005-10-17 02:57:20 +04:00
dm_free ( c ) ;
2005-10-16 18:33:22 +04:00
}
}
2005-10-17 02:57:20 +04:00
void dm_hash_destroy ( struct dm_hash_table * t )
2005-10-16 18:33:22 +04:00
{
_free_nodes ( t ) ;
2005-10-17 02:57:20 +04:00
dm_free ( t - > slots ) ;
dm_free ( t ) ;
2005-10-16 18:33:22 +04:00
}
2006-01-04 19:07:27 +03:00
static struct dm_hash_node * * _find ( struct dm_hash_table * t , const char * key ,
uint32_t len )
2005-10-16 18:33:22 +04:00
{
unsigned h = _hash ( key , len ) & ( t - > num_slots - 1 ) ;
2005-10-17 02:57:20 +04:00
struct dm_hash_node * * c ;
2005-10-16 18:33:22 +04:00
for ( c = & t - > slots [ h ] ; * c ; c = & ( ( * c ) - > next ) )
if ( ! memcmp ( key , ( * c ) - > key , len ) )
break ;
return c ;
}
2005-10-17 02:57:20 +04:00
void * dm_hash_lookup_binary ( struct dm_hash_table * t , const char * key ,
2005-10-16 18:33:22 +04:00
uint32_t len )
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * * c = _find ( t , key , len ) ;
2006-01-04 19:07:27 +03:00
2005-10-16 18:33:22 +04:00
return * c ? ( * c ) - > data : 0 ;
}
2005-10-17 02:57:20 +04:00
int dm_hash_insert_binary ( struct dm_hash_table * t , const char * key ,
2006-01-04 19:07:27 +03:00
uint32_t len , void * data )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * * c = _find ( t , key , len ) ;
2005-10-16 18:33:22 +04:00
if ( * c )
( * c ) - > data = data ;
else {
2005-10-17 02:57:20 +04:00
struct dm_hash_node * n = _create_node ( key , len ) ;
2005-10-16 18:33:22 +04:00
if ( ! n )
return 0 ;
n - > data = data ;
n - > next = 0 ;
* c = n ;
t - > num_nodes + + ;
}
return 1 ;
}
2005-10-17 02:57:20 +04:00
void dm_hash_remove_binary ( struct dm_hash_table * t , const char * key ,
2005-10-16 18:33:22 +04:00
uint32_t len )
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * * c = _find ( t , key , len ) ;
2005-10-16 18:33:22 +04:00
if ( * c ) {
2005-10-17 02:57:20 +04:00
struct dm_hash_node * old = * c ;
2005-10-16 18:33:22 +04:00
* c = ( * c ) - > next ;
2005-10-17 02:57:20 +04:00
dm_free ( old ) ;
2005-10-16 18:33:22 +04:00
t - > num_nodes - - ;
}
}
2005-10-17 02:57:20 +04:00
void * dm_hash_lookup ( struct dm_hash_table * t , const char * key )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
return dm_hash_lookup_binary ( t , key , strlen ( key ) + 1 ) ;
2005-10-16 18:33:22 +04:00
}
2005-10-17 02:57:20 +04:00
int dm_hash_insert ( struct dm_hash_table * t , const char * key , void * data )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
return dm_hash_insert_binary ( t , key , strlen ( key ) + 1 , data ) ;
2005-10-16 18:33:22 +04:00
}
2005-10-17 02:57:20 +04:00
void dm_hash_remove ( struct dm_hash_table * t , const char * key )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
dm_hash_remove_binary ( t , key , strlen ( key ) + 1 ) ;
2005-10-16 18:33:22 +04:00
}
2005-10-17 02:57:20 +04:00
unsigned dm_hash_get_num_entries ( struct dm_hash_table * t )
2005-10-16 18:33:22 +04:00
{
return t - > num_nodes ;
}
2005-10-17 02:57:20 +04:00
void dm_hash_iter ( struct dm_hash_table * t , dm_hash_iterate_fn f )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * c ;
2006-01-04 19:07:27 +03:00
unsigned i ;
2005-10-16 18:33:22 +04:00
for ( i = 0 ; i < t - > num_slots ; i + + )
for ( c = t - > slots [ i ] ; c ; c = c - > next )
f ( c - > data ) ;
}
2005-10-17 02:57:20 +04:00
void dm_hash_wipe ( struct dm_hash_table * t )
2005-10-16 18:33:22 +04:00
{
_free_nodes ( t ) ;
2005-10-17 02:57:20 +04:00
memset ( t - > slots , 0 , sizeof ( struct dm_hash_node * ) * t - > num_slots ) ;
2006-01-04 19:07:27 +03:00
t - > num_nodes = 0u ;
2005-10-16 18:33:22 +04:00
}
2007-01-22 18:03:57 +03:00
char * dm_hash_get_key ( struct dm_hash_table * t __attribute ( ( unused ) ) ,
struct dm_hash_node * n )
2005-10-16 18:33:22 +04:00
{
return n - > key ;
}
2007-01-22 18:03:57 +03:00
void * dm_hash_get_data ( struct dm_hash_table * t __attribute ( ( unused ) ) ,
struct dm_hash_node * n )
2005-10-16 18:33:22 +04:00
{
return n - > data ;
}
2005-10-17 02:57:20 +04:00
static struct dm_hash_node * _next_slot ( struct dm_hash_table * t , unsigned s )
2005-10-16 18:33:22 +04:00
{
2005-10-17 02:57:20 +04:00
struct dm_hash_node * c = NULL ;
2006-01-04 19:07:27 +03:00
unsigned i ;
2005-10-16 18:33:22 +04:00
for ( i = s ; i < t - > num_slots & & ! c ; i + + )
c = t - > slots [ i ] ;
return c ;
}
2005-10-17 02:57:20 +04:00
struct dm_hash_node * dm_hash_get_first ( struct dm_hash_table * t )
2005-10-16 18:33:22 +04:00
{
return _next_slot ( t , 0 ) ;
}
2005-10-17 02:57:20 +04:00
struct dm_hash_node * dm_hash_get_next ( struct dm_hash_table * t , struct dm_hash_node * n )
2005-10-16 18:33:22 +04:00
{
unsigned h = _hash ( n - > key , n - > keylen ) & ( t - > num_slots - 1 ) ;
2006-01-04 19:07:27 +03:00
2005-10-16 18:33:22 +04:00
return n - > next ? n - > next : _next_slot ( t , h + 1 ) ;
}