1
/******************************************************
2
The simple hash table utility
6
Created 5/20/1997 Heikki Tuuri
7
*******************************************************/
14
#include "sync0sync.h"
16
typedef struct hash_table_struct hash_table_t;
17
typedef struct hash_cell_struct hash_cell_t;
19
typedef void* hash_node_t;
21
/* Fix Bug #13859: symbol collision between imap/mysql */
22
#define hash_create hash0_create
24
/*****************************************************************
25
Creates a hash table with >= n array cells. The actual number
26
of cells is chosen to be a prime number slightly bigger than n. */
31
/* out, own: created table */
32
ulint n); /* in: number of array cells */
33
/*****************************************************************
34
Creates a mutex array to protect a hash table. */
37
hash_create_mutexes_func(
38
/*=====================*/
39
hash_table_t* table, /* in: hash table */
40
#ifdef UNIV_SYNC_DEBUG
41
ulint sync_level, /* in: latching order level of the
42
mutexes: used in the debug version */
43
#endif /* UNIV_SYNC_DEBUG */
44
ulint n_mutexes); /* in: number of mutexes */
45
#ifdef UNIV_SYNC_DEBUG
46
# define hash_create_mutexes(t,n,level) hash_create_mutexes_func(t,level,n)
47
#else /* UNIV_SYNC_DEBUG */
48
# define hash_create_mutexes(t,n,level) hash_create_mutexes_func(t,n)
49
#endif /* UNIV_SYNC_DEBUG */
51
/*****************************************************************
52
Frees a hash table. */
57
hash_table_t* table); /* in, own: hash table */
58
/******************************************************************
59
Calculates the hash value from a folded value. */
64
/* out: hashed value */
65
ulint fold, /* in: folded value */
66
hash_table_t* table); /* in: hash table */
67
/************************************************************************
68
Assert that the mutex for the table in a hash operation is owned. */
69
#ifdef UNIV_SYNC_DEBUG
70
# define HASH_ASSERT_OWNED(TABLE, FOLD) \
71
ut_ad(!(TABLE)->mutexes || mutex_own(hash_get_mutex(TABLE, FOLD)));
73
# define HASH_ASSERT_OWNED(TABLE, FOLD)
76
/***********************************************************************
77
Inserts a struct to a hash table. */
79
#define HASH_INSERT(TYPE, NAME, TABLE, FOLD, DATA)\
81
hash_cell_t* cell3333;\
84
HASH_ASSERT_OWNED(TABLE, FOLD)\
88
cell3333 = hash_get_nth_cell(TABLE, hash_calc_hash(FOLD, TABLE));\
90
if (cell3333->node == NULL) {\
91
cell3333->node = DATA;\
93
struct3333 = cell3333->node;\
95
while (struct3333->NAME != NULL) {\
97
struct3333 = struct3333->NAME;\
100
struct3333->NAME = DATA;\
104
#ifdef UNIV_HASH_DEBUG
105
# define HASH_ASSERT_VALID(DATA) ut_a((void*) (DATA) != (void*) -1)
106
# define HASH_INVALIDATE(DATA, NAME) DATA->NAME = (void*) -1
108
# define HASH_ASSERT_VALID(DATA) do {} while (0)
109
# define HASH_INVALIDATE(DATA, NAME) do {} while (0)
112
/***********************************************************************
113
Deletes a struct from a hash table. */
115
#define HASH_DELETE(TYPE, NAME, TABLE, FOLD, DATA)\
117
hash_cell_t* cell3333;\
120
HASH_ASSERT_OWNED(TABLE, FOLD)\
122
cell3333 = hash_get_nth_cell(TABLE, hash_calc_hash(FOLD, TABLE));\
124
if (cell3333->node == DATA) {\
125
HASH_ASSERT_VALID(DATA->NAME);\
126
cell3333->node = DATA->NAME;\
128
struct3333 = cell3333->node;\
130
while (struct3333->NAME != DATA) {\
132
struct3333 = struct3333->NAME;\
136
struct3333->NAME = DATA->NAME;\
138
HASH_INVALIDATE(DATA, NAME);\
141
/***********************************************************************
142
Gets the first struct in a hash chain, NULL if none. */
144
#define HASH_GET_FIRST(TABLE, HASH_VAL)\
145
(hash_get_nth_cell(TABLE, HASH_VAL)->node)
147
/***********************************************************************
148
Gets the next struct in a hash chain, NULL if none. */
150
#define HASH_GET_NEXT(NAME, DATA) ((DATA)->NAME)
152
/************************************************************************
153
Looks for a struct in a hash table. */
154
#define HASH_SEARCH(NAME, TABLE, FOLD, TYPE, DATA, TEST)\
157
HASH_ASSERT_OWNED(TABLE, FOLD)\
159
(DATA) = (TYPE) HASH_GET_FIRST(TABLE, hash_calc_hash(FOLD, TABLE));\
160
HASH_ASSERT_VALID(DATA);\
162
while ((DATA) != NULL) {\
166
HASH_ASSERT_VALID(HASH_GET_NEXT(NAME, DATA));\
167
(DATA) = (TYPE) HASH_GET_NEXT(NAME, DATA);\
172
/****************************************************************
173
Gets the nth cell in a hash table. */
178
/* out: pointer to cell */
179
hash_table_t* table, /* in: hash table */
180
ulint n); /* in: cell index */
182
/*****************************************************************
183
Clears a hash table so that all the cells become empty. */
188
hash_table_t* table); /* in/out: hash table */
190
/*****************************************************************
191
Returns the number of cells in a hash table. */
196
/* out: number of cells */
197
hash_table_t* table); /* in: table */
198
/***********************************************************************
199
Deletes a struct which is stored in the heap of the hash table, and compacts
200
the heap. The fold value must be stored in the struct NODE in a field named
203
#define HASH_DELETE_AND_COMPACT(TYPE, NAME, TABLE, NODE)\
207
hash_cell_t* cell111;\
210
fold111 = (NODE)->fold;\
212
HASH_DELETE(TYPE, NAME, TABLE, fold111, NODE);\
214
top_node111 = (TYPE*)mem_heap_get_top(\
215
hash_get_heap(TABLE, fold111),\
218
/* If the node to remove is not the top node in the heap, compact the\
219
heap of nodes by moving the top node in the place of NODE. */\
221
if (NODE != top_node111) {\
223
/* Copy the top node in place of NODE */\
225
*(NODE) = *top_node111;\
227
cell111 = hash_get_nth_cell(TABLE,\
228
hash_calc_hash(top_node111->fold, TABLE));\
230
/* Look for the pointer to the top node, to update it */\
232
if (cell111->node == top_node111) {\
233
/* The top node is the first in the chain */\
235
cell111->node = NODE;\
237
/* We have to look for the predecessor of the top\
239
node111 = cell111->node;\
241
while (top_node111 != HASH_GET_NEXT(NAME, node111)) {\
243
node111 = HASH_GET_NEXT(NAME, node111);\
246
/* Now we have the predecessor node */\
248
node111->NAME = NODE;\
252
/* Free the space occupied by the top node */\
254
mem_heap_free_top(hash_get_heap(TABLE, fold111), sizeof(TYPE));\
257
/********************************************************************
258
Move all hash table entries from OLD_TABLE to NEW_TABLE.*/
260
#define HASH_MIGRATE(OLD_TABLE, NEW_TABLE, NODE_TYPE, PTR_NAME, FOLD_FUNC) \
263
ulint cell_count2222;\
265
cell_count2222 = hash_get_n_cells(OLD_TABLE);\
267
for (i2222 = 0; i2222 < cell_count2222; i2222++) {\
268
NODE_TYPE* node2222 = HASH_GET_FIRST((OLD_TABLE), i2222);\
271
NODE_TYPE* next2222 = node2222->PTR_NAME;\
272
ulint fold2222 = FOLD_FUNC(node2222);\
274
HASH_INSERT(NODE_TYPE, PTR_NAME, (NEW_TABLE),\
275
fold2222, node2222);\
277
node2222 = next2222;\
283
/****************************************************************
284
Gets the mutex index for a fold value in a hash table. */
289
/* out: mutex number */
290
hash_table_t* table, /* in: hash table */
291
ulint fold); /* in: fold */
292
/****************************************************************
293
Gets the nth heap in a hash table. */
299
hash_table_t* table, /* in: hash table */
300
ulint i); /* in: index of the heap */
301
/****************************************************************
302
Gets the heap for a fold value in a hash table. */
308
hash_table_t* table, /* in: hash table */
309
ulint fold); /* in: fold */
310
/****************************************************************
311
Gets the nth mutex in a hash table. */
317
hash_table_t* table, /* in: hash table */
318
ulint i); /* in: index of the mutex */
319
/****************************************************************
320
Gets the mutex for a fold value in a hash table. */
326
hash_table_t* table, /* in: hash table */
327
ulint fold); /* in: fold */
328
/****************************************************************
329
Reserves the mutex for a fold value in a hash table. */
334
hash_table_t* table, /* in: hash table */
335
ulint fold); /* in: fold */
336
/****************************************************************
337
Releases the mutex for a fold value in a hash table. */
342
hash_table_t* table, /* in: hash table */
343
ulint fold); /* in: fold */
344
/****************************************************************
345
Reserves all the mutexes of a hash table, in an ascending order. */
348
hash_mutex_enter_all(
349
/*=================*/
350
hash_table_t* table); /* in: hash table */
351
/****************************************************************
352
Releases all the mutexes of a hash table. */
357
hash_table_t* table); /* in: hash table */
360
struct hash_cell_struct{
361
void* node; /* hash chain node, NULL if none */
364
/* The hash table structure */
365
struct hash_table_struct {
367
ibool adaptive;/* TRUE if this is the hash table of the
368
adaptive hash index */
369
#endif /* UNIV_DEBUG */
370
ulint n_cells;/* number of cells in the hash table */
371
hash_cell_t* array; /* pointer to cell array */
372
ulint n_mutexes;/* if mutexes != NULL, then the number of
373
mutexes, must be a power of 2 */
374
mutex_t* mutexes;/* NULL, or an array of mutexes used to
375
protect segments of the hash table */
376
mem_heap_t** heaps; /* if this is non-NULL, hash chain nodes for
377
external chaining can be allocated from these
378
memory heaps; there are then n_mutexes many of
384
#define HASH_TABLE_MAGIC_N 76561114
387
#include "hash0hash.ic"