Commit d1308443 authored by Sergey Glukhov's avatar Sergey Glukhov

automerge

parents 8ac55785 61753a07
...@@ -21,6 +21,40 @@ ...@@ -21,6 +21,40 @@
extern "C" { extern "C" {
#endif #endif
/*
There was a problem on MacOSX with a shared object ha_example.so.
It used hash_search(). During build of ha_example.so no libmysys
was specified. Since MacOSX had a hash_search() in the system
library, it built the shared object so that the dynamic linker
linked hash_search() to the system library, which caused a crash
when called. To come around this, we renamed hash_search() to
my_hash_search(), as we did long ago with hash_insert() and
hash_reset(). However, this time we made the move complete with
all names. To keep compatibility, we redefine the old names.
Since every C and C++ file, that uses HASH, needs to include
this file, the change is complete. Both names could be used
in the code, but the my_* versions are recommended now.
*/
#define hash_get_key my_hash_get_key
#define hash_free_key my_hash_free_key
#define hash_init my_hash_init
#define hash_init2 my_hash_init2
#define _hash_init _my_hash_init
#define hash_free my_hash_free
#define hash_reset my_hash_reset
#define hash_element my_hash_element
#define hash_search my_hash_search
#define hash_first my_hash_first
#define hash_next my_hash_next
#define hash_insert my_hash_insert
#define hash_delete my_hash_delete
#define hash_update my_hash_update
#define hash_replace my_hash_replace
#define hash_check my_hash_check
#define hash_clear my_hash_clear
#define hash_inited my_hash_inited
#define hash_init_opt my_hash_init_opt
/* /*
Overhead to store an element in hash Overhead to store an element in hash
Can be used to approximate memory consumption for a hash Can be used to approximate memory consumption for a hash
...@@ -30,8 +64,8 @@ extern "C" { ...@@ -30,8 +64,8 @@ extern "C" {
/* flags for hash_init */ /* flags for hash_init */
#define HASH_UNIQUE 1 /* hash_insert fails on duplicate key */ #define HASH_UNIQUE 1 /* hash_insert fails on duplicate key */
typedef uchar *(*hash_get_key)(const uchar *,size_t*,my_bool); typedef uchar *(*my_hash_get_key)(const uchar *,size_t*,my_bool);
typedef void (*hash_free_key)(void *); typedef void (*my_hash_free_key)(void *);
typedef struct st_hash { typedef struct st_hash {
size_t key_offset,key_length; /* Length of key if const length */ size_t key_offset,key_length; /* Length of key if const length */
...@@ -39,7 +73,7 @@ typedef struct st_hash { ...@@ -39,7 +73,7 @@ typedef struct st_hash {
ulong records; ulong records;
uint flags; uint flags;
DYNAMIC_ARRAY array; /* Place for hash_keys */ DYNAMIC_ARRAY array; /* Place for hash_keys */
hash_get_key get_key; my_hash_get_key get_key;
void (*free)(void *); void (*free)(void *);
CHARSET_INFO *charset; CHARSET_INFO *charset;
} HASH; } HASH;
...@@ -47,30 +81,34 @@ typedef struct st_hash { ...@@ -47,30 +81,34 @@ typedef struct st_hash {
/* A search iterator state */ /* A search iterator state */
typedef uint HASH_SEARCH_STATE; typedef uint HASH_SEARCH_STATE;
#define hash_init(A,B,C,D,E,F,G,H) _hash_init(A,0,B,C,D,E,F,G,H CALLER_INFO) #define my_hash_init(A,B,C,D,E,F,G,H) \
#define hash_init2(A,B,C,D,E,F,G,H,I) _hash_init(A,B,C,D,E,F,G,H,I CALLER_INFO) _my_hash_init(A,0,B,C,D,E,F,G,H CALLER_INFO)
my_bool _hash_init(HASH *hash, uint growth_size,CHARSET_INFO *charset, #define my_hash_init2(A,B,C,D,E,F,G,H,I) \
_my_hash_init(A,B,C,D,E,F,G,H,I CALLER_INFO)
my_bool _my_hash_init(HASH *hash, uint growth_size, CHARSET_INFO *charset,
ulong default_array_elements, size_t key_offset, ulong default_array_elements, size_t key_offset,
size_t key_length, hash_get_key get_key, size_t key_length, my_hash_get_key get_key,
void (*free_element)(void*), uint flags CALLER_INFO_PROTO); void (*free_element)(void*),
void hash_free(HASH *tree); uint flags CALLER_INFO_PROTO);
void my_hash_free(HASH *tree);
void my_hash_reset(HASH *hash); void my_hash_reset(HASH *hash);
uchar *hash_element(HASH *hash,ulong idx); uchar *my_hash_element(HASH *hash, ulong idx);
uchar *hash_search(const HASH *info, const uchar *key, size_t length); uchar *my_hash_search(const HASH *info, const uchar *key, size_t length);
uchar *hash_first(const HASH *info, const uchar *key, size_t length, uchar *my_hash_first(const HASH *info, const uchar *key, size_t length,
HASH_SEARCH_STATE *state); HASH_SEARCH_STATE *state);
uchar *hash_next(const HASH *info, const uchar *key, size_t length, uchar *my_hash_next(const HASH *info, const uchar *key, size_t length,
HASH_SEARCH_STATE *state); HASH_SEARCH_STATE *state);
my_bool my_hash_insert(HASH *info,const uchar *data); my_bool my_hash_insert(HASH *info, const uchar *data);
my_bool hash_delete(HASH *hash,uchar *record); my_bool my_hash_delete(HASH *hash, uchar *record);
my_bool hash_update(HASH *hash,uchar *record,uchar *old_key,size_t old_key_length); my_bool my_hash_update(HASH *hash, uchar *record, uchar *old_key,
void hash_replace(HASH *hash, HASH_SEARCH_STATE *state, uchar *new_row); size_t old_key_length);
my_bool hash_check(HASH *hash); /* Only in debug library */ void my_hash_replace(HASH *hash, HASH_SEARCH_STATE *state, uchar *new_row);
my_bool my_hash_check(HASH *hash); /* Only in debug library */
#define hash_clear(H) bzero((char*) (H),sizeof(*(H))) #define my_hash_clear(H) bzero((char*) (H), sizeof(*(H)))
#define hash_inited(H) ((H)->array.buffer != 0) #define my_hash_inited(H) ((H)->array.buffer != 0)
#define hash_init_opt(A,B,C,D,E,F,G,H) \ #define my_hash_init_opt(A,B,C,D,E,F,G,H) \
(!hash_inited(A) && _hash_init(A,0,B,C,D,E,F,G, H CALLER_INFO)) (!my_hash_inited(A) && _my_hash_init(A,0,B,C,D,E,F,G, H CALLER_INFO))
#ifdef __cplusplus #ifdef __cplusplus
} }
......
...@@ -33,7 +33,7 @@ typedef struct st_hash_info { ...@@ -33,7 +33,7 @@ typedef struct st_hash_info {
uchar *data; /* data for current entry */ uchar *data; /* data for current entry */
} HASH_LINK; } HASH_LINK;
static uint hash_mask(uint hashnr,uint buffmax,uint maxlength); static uint my_hash_mask(uint hashnr, uint buffmax, uint maxlength);
static void movelink(HASH_LINK *array,uint pos,uint next_link,uint newlink); static void movelink(HASH_LINK *array,uint pos,uint next_link,uint newlink);
static int hashcmp(const HASH *hash, HASH_LINK *pos, const uchar *key, static int hashcmp(const HASH *hash, HASH_LINK *pos, const uchar *key,
size_t length); size_t length);
...@@ -46,19 +46,19 @@ static uint calc_hash(const HASH *hash, const uchar *key, size_t length) ...@@ -46,19 +46,19 @@ static uint calc_hash(const HASH *hash, const uchar *key, size_t length)
} }
my_bool my_bool
_hash_init(HASH *hash,uint growth_size, CHARSET_INFO *charset, _my_hash_init(HASH *hash, uint growth_size, CHARSET_INFO *charset,
ulong size, size_t key_offset, size_t key_length, ulong size, size_t key_offset, size_t key_length,
hash_get_key get_key, my_hash_get_key get_key,
void (*free_element)(void*),uint flags CALLER_INFO_PROTO) void (*free_element)(void*), uint flags CALLER_INFO_PROTO)
{ {
DBUG_ENTER("hash_init"); DBUG_ENTER("my_hash_init");
DBUG_PRINT("enter",("hash: 0x%lx size: %u", (long) hash, (uint) size)); DBUG_PRINT("enter",("hash: 0x%lx size: %u", (long) hash, (uint) size));
hash->records=0; hash->records=0;
if (my_init_dynamic_array_ci(&hash->array, sizeof(HASH_LINK), size, if (my_init_dynamic_array_ci(&hash->array, sizeof(HASH_LINK), size,
growth_size)) growth_size))
{ {
hash->free=0; /* Allow call to hash_free */ hash->free=0; /* Allow call to my_hash_free */
DBUG_RETURN(1); DBUG_RETURN(1);
} }
hash->key_offset=key_offset; hash->key_offset=key_offset;
...@@ -76,14 +76,14 @@ _hash_init(HASH *hash,uint growth_size, CHARSET_INFO *charset, ...@@ -76,14 +76,14 @@ _hash_init(HASH *hash,uint growth_size, CHARSET_INFO *charset,
Call hash->free on all elements in hash. Call hash->free on all elements in hash.
SYNOPSIS SYNOPSIS
hash_free_elements() my_hash_free_elements()
hash hash table hash hash table
NOTES: NOTES:
Sets records to 0 Sets records to 0
*/ */
static inline void hash_free_elements(HASH *hash) static inline void my_hash_free_elements(HASH *hash)
{ {
if (hash->free) if (hash->free)
{ {
...@@ -100,18 +100,18 @@ static inline void hash_free_elements(HASH *hash) ...@@ -100,18 +100,18 @@ static inline void hash_free_elements(HASH *hash)
Free memory used by hash. Free memory used by hash.
SYNOPSIS SYNOPSIS
hash_free() my_hash_free()
hash the hash to delete elements of hash the hash to delete elements of
NOTES: Hash can't be reused without calling hash_init again. NOTES: Hash can't be reused without calling my_hash_init again.
*/ */
void hash_free(HASH *hash) void my_hash_free(HASH *hash)
{ {
DBUG_ENTER("hash_free"); DBUG_ENTER("my_hash_free");
DBUG_PRINT("enter",("hash: 0x%lx", (long) hash)); DBUG_PRINT("enter",("hash: 0x%lx", (long) hash));
hash_free_elements(hash); my_hash_free_elements(hash);
hash->free= 0; hash->free= 0;
delete_dynamic(&hash->array); delete_dynamic(&hash->array);
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
...@@ -131,7 +131,7 @@ void my_hash_reset(HASH *hash) ...@@ -131,7 +131,7 @@ void my_hash_reset(HASH *hash)
DBUG_ENTER("my_hash_reset"); DBUG_ENTER("my_hash_reset");
DBUG_PRINT("enter",("hash: 0x%lxd", (long) hash)); DBUG_PRINT("enter",("hash: 0x%lxd", (long) hash));
hash_free_elements(hash); my_hash_free_elements(hash);
reset_dynamic(&hash->array); reset_dynamic(&hash->array);
/* Set row pointers so that the hash can be reused at once */ /* Set row pointers so that the hash can be reused at once */
hash->blength= 1; hash->blength= 1;
...@@ -146,7 +146,7 @@ void my_hash_reset(HASH *hash) ...@@ -146,7 +146,7 @@ void my_hash_reset(HASH *hash)
*/ */
static inline char* static inline char*
hash_key(const HASH *hash, const uchar *record, size_t *length, my_hash_key(const HASH *hash, const uchar *record, size_t *length,
my_bool first) my_bool first)
{ {
if (hash->get_key) if (hash->get_key)
...@@ -157,18 +157,18 @@ hash_key(const HASH *hash, const uchar *record, size_t *length, ...@@ -157,18 +157,18 @@ hash_key(const HASH *hash, const uchar *record, size_t *length,
/* Calculate pos according to keys */ /* Calculate pos according to keys */
static uint hash_mask(uint hashnr,uint buffmax,uint maxlength) static uint my_hash_mask(uint hashnr, uint buffmax, uint maxlength)
{ {
if ((hashnr & (buffmax-1)) < maxlength) return (hashnr & (buffmax-1)); if ((hashnr & (buffmax-1)) < maxlength) return (hashnr & (buffmax-1));
return (hashnr & ((buffmax >> 1) -1)); return (hashnr & ((buffmax >> 1) -1));
} }
static uint hash_rec_mask(const HASH *hash, HASH_LINK *pos, static uint my_hash_rec_mask(const HASH *hash, HASH_LINK *pos,
uint buffmax, uint maxlength) uint buffmax, uint maxlength)
{ {
size_t length; size_t length;
uchar *key= (uchar*) hash_key(hash,pos->data,&length,0); uchar *key= (uchar*) my_hash_key(hash, pos->data, &length, 0);
return hash_mask(calc_hash(hash,key,length),buffmax,maxlength); return my_hash_mask(calc_hash(hash, key, length), buffmax, maxlength);
} }
...@@ -181,15 +181,15 @@ inline ...@@ -181,15 +181,15 @@ inline
unsigned int rec_hashnr(HASH *hash,const uchar *record) unsigned int rec_hashnr(HASH *hash,const uchar *record)
{ {
size_t length; size_t length;
uchar *key= (uchar*) hash_key(hash,record,&length,0); uchar *key= (uchar*) my_hash_key(hash, record, &length, 0);
return calc_hash(hash,key,length); return calc_hash(hash,key,length);
} }
uchar* hash_search(const HASH *hash, const uchar *key, size_t length) uchar* my_hash_search(const HASH *hash, const uchar *key, size_t length)
{ {
HASH_SEARCH_STATE state; HASH_SEARCH_STATE state;
return hash_first(hash, key, length, &state); return my_hash_first(hash, key, length, &state);
} }
/* /*
...@@ -199,18 +199,18 @@ uchar* hash_search(const HASH *hash, const uchar *key, size_t length) ...@@ -199,18 +199,18 @@ uchar* hash_search(const HASH *hash, const uchar *key, size_t length)
Assigns the number of the found record to HASH_SEARCH_STATE state Assigns the number of the found record to HASH_SEARCH_STATE state
*/ */
uchar* hash_first(const HASH *hash, const uchar *key, size_t length, uchar* my_hash_first(const HASH *hash, const uchar *key, size_t length,
HASH_SEARCH_STATE *current_record) HASH_SEARCH_STATE *current_record)
{ {
HASH_LINK *pos; HASH_LINK *pos;
uint flag,idx; uint flag,idx;
DBUG_ENTER("hash_first"); DBUG_ENTER("my_hash_first");
flag=1; flag=1;
if (hash->records) if (hash->records)
{ {
idx=hash_mask(calc_hash(hash,key,length ? length : hash->key_length), idx= my_hash_mask(calc_hash(hash, key, length ? length : hash->key_length),
hash->blength,hash->records); hash->blength, hash->records);
do do
{ {
pos= dynamic_element(&hash->array,idx,HASH_LINK*); pos= dynamic_element(&hash->array,idx,HASH_LINK*);
...@@ -223,7 +223,7 @@ uchar* hash_first(const HASH *hash, const uchar *key, size_t length, ...@@ -223,7 +223,7 @@ uchar* hash_first(const HASH *hash, const uchar *key, size_t length,
if (flag) if (flag)
{ {
flag=0; /* Reset flag */ flag=0; /* Reset flag */
if (hash_rec_mask(hash,pos,hash->blength,hash->records) != idx) if (my_hash_rec_mask(hash, pos, hash->blength, hash->records) != idx)
break; /* Wrong link */ break; /* Wrong link */
} }
} }
...@@ -234,9 +234,9 @@ uchar* hash_first(const HASH *hash, const uchar *key, size_t length, ...@@ -234,9 +234,9 @@ uchar* hash_first(const HASH *hash, const uchar *key, size_t length,
} }
/* Get next record with identical key */ /* Get next record with identical key */
/* Can only be called if previous calls was hash_search */ /* Can only be called if previous calls was my_hash_search */
uchar* hash_next(const HASH *hash, const uchar *key, size_t length, uchar* my_hash_next(const HASH *hash, const uchar *key, size_t length,
HASH_SEARCH_STATE *current_record) HASH_SEARCH_STATE *current_record)
{ {
HASH_LINK *pos; HASH_LINK *pos;
...@@ -297,7 +297,7 @@ static int hashcmp(const HASH *hash, HASH_LINK *pos, const uchar *key, ...@@ -297,7 +297,7 @@ static int hashcmp(const HASH *hash, HASH_LINK *pos, const uchar *key,
size_t length) size_t length)
{ {
size_t rec_keylength; size_t rec_keylength;
uchar *rec_key= (uchar*) hash_key(hash,pos->data,&rec_keylength,1); uchar *rec_key= (uchar*) my_hash_key(hash, pos->data, &rec_keylength, 1);
return ((length && length != rec_keylength) || return ((length && length != rec_keylength) ||
my_strnncoll(hash->charset, (uchar*) rec_key, rec_keylength, my_strnncoll(hash->charset, (uchar*) rec_key, rec_keylength,
(uchar*) key, rec_keylength)); (uchar*) key, rec_keylength));
...@@ -306,7 +306,7 @@ static int hashcmp(const HASH *hash, HASH_LINK *pos, const uchar *key, ...@@ -306,7 +306,7 @@ static int hashcmp(const HASH *hash, HASH_LINK *pos, const uchar *key,
/* Write a hash-key to the hash-index */ /* Write a hash-key to the hash-index */
my_bool my_hash_insert(HASH *info,const uchar *record) my_bool my_hash_insert(HASH *info, const uchar *record)
{ {
int flag; int flag;
size_t idx; size_t idx;
...@@ -321,8 +321,8 @@ my_bool my_hash_insert(HASH *info,const uchar *record) ...@@ -321,8 +321,8 @@ my_bool my_hash_insert(HASH *info,const uchar *record)
if (HASH_UNIQUE & info->flags) if (HASH_UNIQUE & info->flags)
{ {
uchar *key= (uchar*) hash_key(info, record, &idx, 1); uchar *key= (uchar*) my_hash_key(info, record, &idx, 1);
if (hash_search(info, key, idx)) if (my_hash_search(info, key, idx))
return(TRUE); /* Duplicate entry */ return(TRUE); /* Duplicate entry */
} }
...@@ -341,7 +341,7 @@ my_bool my_hash_insert(HASH *info,const uchar *record) ...@@ -341,7 +341,7 @@ my_bool my_hash_insert(HASH *info,const uchar *record)
pos=data+idx; pos=data+idx;
hash_nr=rec_hashnr(info,pos->data); hash_nr=rec_hashnr(info,pos->data);
if (flag == 0) /* First loop; Check if ok */ if (flag == 0) /* First loop; Check if ok */
if (hash_mask(hash_nr,info->blength,info->records) != first_index) if (my_hash_mask(hash_nr, info->blength, info->records) != first_index)
break; break;
if (!(hash_nr & halfbuff)) if (!(hash_nr & halfbuff))
{ /* Key will not move */ { /* Key will not move */
...@@ -413,7 +413,7 @@ my_bool my_hash_insert(HASH *info,const uchar *record) ...@@ -413,7 +413,7 @@ my_bool my_hash_insert(HASH *info,const uchar *record)
} }
/* Check if we are at the empty position */ /* Check if we are at the empty position */
idx=hash_mask(rec_hashnr(info,record),info->blength,info->records+1); idx= my_hash_mask(rec_hashnr(info, record), info->blength, info->records + 1);
pos=data+idx; pos=data+idx;
if (pos == empty) if (pos == empty)
{ {
...@@ -424,7 +424,7 @@ my_bool my_hash_insert(HASH *info,const uchar *record) ...@@ -424,7 +424,7 @@ my_bool my_hash_insert(HASH *info,const uchar *record)
{ {
/* Check if more records in same hash-nr family */ /* Check if more records in same hash-nr family */
empty[0]=pos[0]; empty[0]=pos[0];
gpos=data+hash_rec_mask(info,pos,info->blength,info->records+1); gpos= data + my_hash_rec_mask(info, pos, info->blength, info->records + 1);
if (pos == gpos) if (pos == gpos)
{ {
pos->data=(uchar*) record; pos->data=(uchar*) record;
...@@ -449,18 +449,18 @@ my_bool my_hash_insert(HASH *info,const uchar *record) ...@@ -449,18 +449,18 @@ my_bool my_hash_insert(HASH *info,const uchar *record)
** if there is a free-function it's called for record if found ** if there is a free-function it's called for record if found
******************************************************************************/ ******************************************************************************/
my_bool hash_delete(HASH *hash,uchar *record) my_bool my_hash_delete(HASH *hash, uchar *record)
{ {
uint blength,pos2,pos_hashnr,lastpos_hashnr,idx,empty_index; uint blength,pos2,pos_hashnr,lastpos_hashnr,idx,empty_index;
HASH_LINK *data,*lastpos,*gpos,*pos,*pos3,*empty; HASH_LINK *data,*lastpos,*gpos,*pos,*pos3,*empty;
DBUG_ENTER("hash_delete"); DBUG_ENTER("my_hash_delete");
if (!hash->records) if (!hash->records)
DBUG_RETURN(1); DBUG_RETURN(1);
blength=hash->blength; blength=hash->blength;
data=dynamic_element(&hash->array,0,HASH_LINK*); data=dynamic_element(&hash->array,0,HASH_LINK*);
/* Search after record with key */ /* Search after record with key */
pos=data+ hash_mask(rec_hashnr(hash,record),blength,hash->records); pos= data + my_hash_mask(rec_hashnr(hash, record), blength, hash->records);
gpos = 0; gpos = 0;
while (pos->data != record) while (pos->data != record)
...@@ -491,7 +491,7 @@ my_bool hash_delete(HASH *hash,uchar *record) ...@@ -491,7 +491,7 @@ my_bool hash_delete(HASH *hash,uchar *record)
/* Move the last key (lastpos) */ /* Move the last key (lastpos) */
lastpos_hashnr=rec_hashnr(hash,lastpos->data); lastpos_hashnr=rec_hashnr(hash,lastpos->data);
/* pos is where lastpos should be */ /* pos is where lastpos should be */
pos=data+hash_mask(lastpos_hashnr,hash->blength,hash->records); pos= data + my_hash_mask(lastpos_hashnr, hash->blength, hash->records);
if (pos == empty) /* Move to empty position. */ if (pos == empty) /* Move to empty position. */
{ {
empty[0]=lastpos[0]; empty[0]=lastpos[0];
...@@ -499,7 +499,7 @@ my_bool hash_delete(HASH *hash,uchar *record) ...@@ -499,7 +499,7 @@ my_bool hash_delete(HASH *hash,uchar *record)
} }
pos_hashnr=rec_hashnr(hash,pos->data); pos_hashnr=rec_hashnr(hash,pos->data);
/* pos3 is where the pos should be */ /* pos3 is where the pos should be */
pos3= data+hash_mask(pos_hashnr,hash->blength,hash->records); pos3= data + my_hash_mask(pos_hashnr, hash->blength, hash->records);
if (pos != pos3) if (pos != pos3)
{ /* pos is on wrong posit */ { /* pos is on wrong posit */
empty[0]=pos[0]; /* Save it here */ empty[0]=pos[0]; /* Save it here */
...@@ -507,8 +507,8 @@ my_bool hash_delete(HASH *hash,uchar *record) ...@@ -507,8 +507,8 @@ my_bool hash_delete(HASH *hash,uchar *record)
movelink(data,(uint) (pos-data),(uint) (pos3-data),empty_index); movelink(data,(uint) (pos-data),(uint) (pos3-data),empty_index);
goto exit; goto exit;
} }
pos2= hash_mask(lastpos_hashnr,blength,hash->records+1); pos2= my_hash_mask(lastpos_hashnr, blength, hash->records + 1);
if (pos2 == hash_mask(pos_hashnr,blength,hash->records+1)) if (pos2 == my_hash_mask(pos_hashnr, blength, hash->records + 1))
{ /* Identical key-positions */ { /* Identical key-positions */
if (pos2 != hash->records) if (pos2 != hash->records)
{ {
...@@ -536,26 +536,26 @@ exit: ...@@ -536,26 +536,26 @@ exit:
This is much more efficent than using a delete & insert. This is much more efficent than using a delete & insert.
*/ */
my_bool hash_update(HASH *hash, uchar *record, uchar *old_key, my_bool my_hash_update(HASH *hash, uchar *record, uchar *old_key,
size_t old_key_length) size_t old_key_length)
{ {
uint new_index,new_pos_index,blength,records,empty; uint new_index,new_pos_index,blength,records,empty;
size_t idx; size_t idx;
HASH_LINK org_link,*data,*previous,*pos; HASH_LINK org_link,*data,*previous,*pos;
DBUG_ENTER("hash_update"); DBUG_ENTER("my_hash_update");
if (HASH_UNIQUE & hash->flags) if (HASH_UNIQUE & hash->flags)
{ {
HASH_SEARCH_STATE state; HASH_SEARCH_STATE state;
uchar *found, *new_key= (uchar*) hash_key(hash, record, &idx, 1); uchar *found, *new_key= (uchar*) my_hash_key(hash, record, &idx, 1);
if ((found= hash_first(hash, new_key, idx, &state))) if ((found= my_hash_first(hash, new_key, idx, &state)))
{ {
do do
{ {
if (found != record) if (found != record)
DBUG_RETURN(1); /* Duplicate entry */ DBUG_RETURN(1); /* Duplicate entry */
} }
while ((found= hash_next(hash, new_key, idx, &state))); while ((found= my_hash_next(hash, new_key, idx, &state)));
} }
} }
...@@ -564,11 +564,11 @@ my_bool hash_update(HASH *hash, uchar *record, uchar *old_key, ...@@ -564,11 +564,11 @@ my_bool hash_update(HASH *hash, uchar *record, uchar *old_key,
/* Search after record with key */ /* Search after record with key */
idx=hash_mask(calc_hash(hash, old_key,(old_key_length ? idx= my_hash_mask(calc_hash(hash, old_key, (old_key_length ?
old_key_length : old_key_length :
hash->key_length)), hash->key_length)),
blength,records); blength, records);
new_index=hash_mask(rec_hashnr(hash,record),blength,records); new_index= my_hash_mask(rec_hashnr(hash, record), blength, records);
if (idx == new_index) if (idx == new_index)
DBUG_RETURN(0); /* Nothing to do (No record check) */ DBUG_RETURN(0); /* Nothing to do (No record check) */
previous=0; previous=0;
...@@ -618,7 +618,7 @@ my_bool hash_update(HASH *hash, uchar *record, uchar *old_key, ...@@ -618,7 +618,7 @@ my_bool hash_update(HASH *hash, uchar *record, uchar *old_key,
DBUG_RETURN(0); DBUG_RETURN(0);
} }
pos=data+new_index; pos=data+new_index;
new_pos_index=hash_rec_mask(hash,pos,blength,records); new_pos_index= my_hash_rec_mask(hash, pos, blength, records);
if (new_index != new_pos_index) if (new_index != new_pos_index)
{ /* Other record in wrong position */ { /* Other record in wrong position */
data[empty] = *pos; data[empty] = *pos;
...@@ -636,7 +636,7 @@ my_bool hash_update(HASH *hash, uchar *record, uchar *old_key, ...@@ -636,7 +636,7 @@ my_bool hash_update(HASH *hash, uchar *record, uchar *old_key,
} }
uchar *hash_element(HASH *hash,ulong idx) uchar *my_hash_element(HASH *hash, ulong idx)
{ {
if (idx < hash->records) if (idx < hash->records)
return dynamic_element(&hash->array,idx,HASH_LINK*)->data; return dynamic_element(&hash->array,idx,HASH_LINK*)->data;
...@@ -649,7 +649,8 @@ uchar *hash_element(HASH *hash,ulong idx) ...@@ -649,7 +649,8 @@ uchar *hash_element(HASH *hash,ulong idx)
isn't changed isn't changed
*/ */
void hash_replace(HASH *hash, HASH_SEARCH_STATE *current_record, uchar *new_row) void my_hash_replace(HASH *hash, HASH_SEARCH_STATE *current_record,
uchar *new_row)
{ {
if (*current_record != NO_RECORD) /* Safety */ if (*current_record != NO_RECORD) /* Safety */
dynamic_element(&hash->array, *current_record, HASH_LINK*)->data= new_row; dynamic_element(&hash->array, *current_record, HASH_LINK*)->data= new_row;
...@@ -658,7 +659,7 @@ void hash_replace(HASH *hash, HASH_SEARCH_STATE *current_record, uchar *new_row) ...@@ -658,7 +659,7 @@ void hash_replace(HASH *hash, HASH_SEARCH_STATE *current_record, uchar *new_row)
#ifndef DBUG_OFF #ifndef DBUG_OFF
my_bool hash_check(HASH *hash) my_bool my_hash_check(HASH *hash)
{ {
int error; int error;
uint i,rec_link,found,max_links,seek,links,idx; uint i,rec_link,found,max_links,seek,links,idx;
...@@ -671,7 +672,7 @@ my_bool hash_check(HASH *hash) ...@@ -671,7 +672,7 @@ my_bool hash_check(HASH *hash)
for (i=found=max_links=seek=0 ; i < records ; i++) for (i=found=max_links=seek=0 ; i < records ; i++)
{ {
if (hash_rec_mask(hash,data+i,blength,records) == i) if (my_hash_rec_mask(hash, data + i, blength, records) == i)
{ {
found++; seek++; links=1; found++; seek++; links=1;
for (idx=data[i].next ; for (idx=data[i].next ;
...@@ -687,10 +688,11 @@ my_bool hash_check(HASH *hash) ...@@ -687,10 +688,11 @@ my_bool hash_check(HASH *hash)
} }
hash_info=data+idx; hash_info=data+idx;
seek+= ++links; seek+= ++links;
if ((rec_link=hash_rec_mask(hash,hash_info,blength,records)) != i) if ((rec_link= my_hash_rec_mask(hash, hash_info,
blength, records)) != i)
{ {
DBUG_PRINT("error", DBUG_PRINT("error", ("Record in wrong link at %d: Start %d "
("Record in wrong link at %d: Start %d Record: 0x%lx Record-link %d", "Record: 0x%lx Record-link %d",
idx, i, (long) hash_info->data, rec_link)); idx, i, (long) hash_info->data, rec_link));
error=1; error=1;
} }
......
...@@ -373,6 +373,10 @@ int ha_finalize_handlerton(st_plugin_int *plugin) ...@@ -373,6 +373,10 @@ int ha_finalize_handlerton(st_plugin_int *plugin)
handlerton *hton= (handlerton *)plugin->data; handlerton *hton= (handlerton *)plugin->data;
DBUG_ENTER("ha_finalize_handlerton"); DBUG_ENTER("ha_finalize_handlerton");
/* hton can be NULL here, if ha_initialize_handlerton() failed. */
if (!hton)
goto end;
switch (hton->state) switch (hton->state)
{ {
case SHOW_OPTION_NO: case SHOW_OPTION_NO:
...@@ -401,8 +405,16 @@ int ha_finalize_handlerton(st_plugin_int *plugin) ...@@ -401,8 +405,16 @@ int ha_finalize_handlerton(st_plugin_int *plugin)
} }
} }
/*
In case a plugin is uninstalled and re-installed later, it should
reuse an array slot. Otherwise the number of uninstall/install
cycles would be limited.
*/
hton2plugin[hton->slot]= NULL;
my_free((uchar*)hton, MYF(0)); my_free((uchar*)hton, MYF(0));
end:
DBUG_RETURN(0); DBUG_RETURN(0);
} }
...@@ -437,6 +449,7 @@ int ha_initialize_handlerton(st_plugin_int *plugin) ...@@ -437,6 +449,7 @@ int ha_initialize_handlerton(st_plugin_int *plugin)
case SHOW_OPTION_YES: case SHOW_OPTION_YES:
{ {
uint tmp; uint tmp;
ulong fslot;
/* now check the db_type for conflict */ /* now check the db_type for conflict */
if (hton->db_type <= DB_TYPE_UNKNOWN || if (hton->db_type <= DB_TYPE_UNKNOWN ||
hton->db_type >= DB_TYPE_DEFAULT || hton->db_type >= DB_TYPE_DEFAULT ||
...@@ -461,7 +474,31 @@ int ha_initialize_handlerton(st_plugin_int *plugin) ...@@ -461,7 +474,31 @@ int ha_initialize_handlerton(st_plugin_int *plugin)
tmp= hton->savepoint_offset; tmp= hton->savepoint_offset;
hton->savepoint_offset= savepoint_alloc_size; hton->savepoint_offset= savepoint_alloc_size;
savepoint_alloc_size+= tmp; savepoint_alloc_size+= tmp;
/*
In case a plugin is uninstalled and re-installed later, it should
reuse an array slot. Otherwise the number of uninstall/install
cycles would be limited. So look for a free slot.
*/
DBUG_PRINT("plugin", ("total_ha: %lu", total_ha));
for (fslot= 0; fslot < total_ha; fslot++)
{
if (!hton2plugin[fslot])
break;
}
if (fslot < total_ha)
hton->slot= fslot;
else
{
if (total_ha >= MAX_HA)
{
sql_print_error("Too many plugins loaded. Limit is %lu. "
"Failed on '%s'", (ulong) MAX_HA, plugin->name.str);
goto err;
}
hton->slot= total_ha++; hton->slot= total_ha++;
}
hton2plugin[hton->slot]=plugin; hton2plugin[hton->slot]=plugin;
if (hton->prepare) if (hton->prepare)
total_ha_2pc++; total_ha_2pc++;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment