28
using namespace drizzled;
28
30
static HASH_INFO *hp_find_free_hash(HP_SHARE *info, HP_BLOCK *block,
31
int heap_write(HP_INFO *info, const uchar *record)
33
int heap_write(HP_INFO *info, const unsigned char *record)
33
35
HP_KEYDEF *keydef, *end;
35
37
HP_SHARE *share=info->s;
36
uint rec_length, chunk_count;
38
uint32_t rec_length, chunk_count;
38
40
if ((share->records >= share->max_records && share->max_records) ||
39
41
(share->recordspace.total_data_length + share->index_length >= share->max_table_size))
41
return(my_errno=HA_ERR_RECORD_FILE_FULL);
43
return(errno=HA_ERR_RECORD_FILE_FULL);
44
46
rec_length = hp_get_encoded_data_length(share, record, &chunk_count);
46
48
if (!(pos=hp_allocate_chunkset(&share->recordspace, chunk_count)))
50
52
for (keydef = share->keydef, end = keydef + share->keys; keydef < end;
74
76
either. Otherwise for HASH index on HA_ERR_FOUND_DUPP_KEY the key
75
77
was inserted and we have to delete it.
77
if (keydef->algorithm == HA_KEY_ALG_BTREE || my_errno == ENOMEM)
79
if (keydef->algorithm == HA_KEY_ALG_BTREE || errno == ENOMEM)
83
85
if ((*keydef->delete_key)(info, keydef, record, pos, 0))
88
90
hp_free_chunks(&share->recordspace, pos);
94
Write a key to rb_tree-index
96
Write a key to rb_tree-index
97
int hp_rb_write_key(HP_INFO *info, HP_KEYDEF *keyinfo, const uchar *record,
99
int hp_rb_write_key(HP_INFO *info, HP_KEYDEF *keyinfo, const unsigned char *record,
100
unsigned char *recpos)
100
102
heap_rb_param custom_arg;
103
uint32_t old_allocated;
103
105
custom_arg.keyseg= keyinfo->seg;
104
106
custom_arg.key_length= hp_rb_make_key(keyinfo, info->recbuf, record, recpos);
116
118
if (!tree_insert(&keyinfo->rb_tree, (void*)info->recbuf,
117
119
custom_arg.key_length, &custom_arg))
119
my_errno= HA_ERR_FOUND_DUPP_KEY;
121
errno= HA_ERR_FOUND_DUPP_KEY;
122
124
info->s->index_length+= (keyinfo->rb_tree.allocated-old_allocated);
129
131
info Heap table info
131
133
record Table record to added
132
recpos Memory buffer where the table record will be stored if added
134
recpos Memory buffer where the table record will be stored if added
135
Hash index uses HP_BLOCK structure as a 'growable array' of HASH_INFO
137
Hash index uses HP_BLOCK structure as a 'growable array' of HASH_INFO
136
138
structs. Array size == number of entries in hash index.
137
139
hp_mask(hp_rec_hashnr()) maps hash entries values to hash array positions.
138
140
If there are several hash entries with the same hash array position P,
139
they are connected in a linked list via HASH_INFO::next_key. The first
140
list element is located at position P, next elements are located at
141
they are connected in a linked list via HASH_INFO::next_key. The first
142
list element is located at position P, next elements are located at
141
143
positions for which there is no record that should be located at that
142
144
position. The order of elements in the list is arbitrary.
146
148
-1 - Out of memory
147
HA_ERR_FOUND_DUPP_KEY - Duplicate record on unique key. The record was
149
HA_ERR_FOUND_DUPP_KEY - Duplicate record on unique key. The record was
148
150
still added and the caller must call hp_delete_key for it.
151
153
int hp_write_key(HP_INFO *info, HP_KEYDEF *keyinfo,
152
const uchar *record, uchar *recpos)
154
const unsigned char *record, unsigned char *recpos)
154
156
HP_SHARE *share = info->s;
156
158
uint32_t halfbuff,hashnr,first_index;
157
uchar *ptr_to_rec= NULL,*ptr_to_rec2= NULL;
159
unsigned char *ptr_to_rec= NULL,*ptr_to_rec2= NULL;
158
160
HASH_INFO *empty, *gpos= NULL, *gpos2= NULL, *pos;
162
164
return(-1); /* No more memory */
163
165
halfbuff= (long) share->blength >> 1;
164
166
pos= hp_find_hash(&keyinfo->block,(first_index=share->records-halfbuff));
167
169
We're about to add one more hash array position, with hash_mask=#records.
168
The number of hash positions will change and some entries might need to
169
be relocated to the newly added position. Those entries are currently
170
members of the list that starts at #first_index position (this is
170
The number of hash positions will change and some entries might need to
171
be relocated to the newly added position. Those entries are currently
172
members of the list that starts at #first_index position (this is
171
173
guaranteed by properties of hp_mask(hp_rec_hashnr(X)) mapping function)
172
174
At #first_index position currently there may be either:
173
175
a) An entry with hashnr != first_index. We don't need to move it.
175
177
b) A list of items with hash_mask=first_index. The list contains entries
177
1) entries that should be relocated to the list that starts at new
179
1) entries that should be relocated to the list that starts at new
178
180
position we're adding ('uppper' list)
179
2) entries that should be left in the list starting at #first_index
181
2) entries that should be left in the list starting at #first_index
180
182
position ('lower' list)
182
184
if (pos != empty) /* If some records */
186
188
hashnr = hp_rec_hashnr(keyinfo, pos->ptr_to_rec);
190
First loop, bail out if we're dealing with case a) from above
192
First loop, bail out if we're dealing with case a) from above
193
195
if (hp_mask(hashnr, share->blength, share->records) != first_index)
331
333
if (! hp_rec_key_cmp(keyinfo, record, pos->ptr_to_rec, 1))
333
return(my_errno=HA_ERR_FOUND_DUPP_KEY);
335
return(errno=HA_ERR_FOUND_DUPP_KEY);
335
337
} while ((pos=pos->next_key));
355
357
info->index_length+=length;
357
359
block->last_allocated=records+1;
358
return((HASH_INFO*) ((uchar*) block->level_info[0].last_blocks+
360
return((HASH_INFO*) ((unsigned char*) block->level_info[0].last_blocks+
359
361
block_pos*block->recbuffer));