28
static uchar *next_free_record_pos(HP_SHARE *info);
28
29
static HASH_INFO *hp_find_free_hash(HP_SHARE *info, HP_BLOCK *block,
31
int heap_write(HP_INFO *info, const unsigned char *record)
32
int heap_write(HP_INFO *info, const uchar *record)
33
34
HP_KEYDEF *keydef, *end;
35
36
HP_SHARE *share=info->s;
36
uint32_t rec_length, chunk_count;
38
if ((share->records >= share->max_records && share->max_records) ||
39
(share->recordspace.total_data_length + share->index_length >= share->max_table_size))
41
return(errno=HA_ERR_RECORD_FILE_FULL);
44
rec_length = hp_get_encoded_data_length(share, record, &chunk_count);
46
if (!(pos=hp_allocate_chunkset(&share->recordspace, chunk_count)))
37
if (!(pos=next_free_record_pos(share)))
50
41
for (keydef = share->keydef, end = keydef + share->keys; keydef < end;
57
hp_copy_record_data_to_chunkset(share, record, pos);
48
memcpy(pos,record,(size_t) share->reclength);
49
pos[share->reclength]=1; /* Mark record as not deleted */
59
50
if (++share->records == share->blength)
60
51
share->blength+= share->blength;
62
52
info->current_ptr=pos;
63
53
info->current_hash_ptr=0;
64
54
info->update|=HA_STATE_AKTIV;
83
73
if ((*keydef->delete_key)(info, keydef, record, pos, 0))
88
hp_free_chunks(&share->recordspace, pos);
79
*((uchar**) pos)=share->del_link;
81
pos[share->reclength]=0; /* Record deleted */
94
Write a key to rb_tree-index
87
Write a key to rb_tree-index
97
int hp_rb_write_key(HP_INFO *info, HP_KEYDEF *keyinfo, const unsigned char *record,
98
unsigned char *recpos)
90
int hp_rb_write_key(HP_INFO *info, HP_KEYDEF *keyinfo, const uchar *record,
100
93
heap_rb_param custom_arg;
101
uint32_t old_allocated;
103
96
custom_arg.keyseg= keyinfo->seg;
104
97
custom_arg.key_length= hp_rb_make_key(keyinfo, info->recbuf, record, recpos);
116
109
if (!tree_insert(&keyinfo->rb_tree, (void*)info->recbuf,
117
110
custom_arg.key_length, &custom_arg))
119
errno= HA_ERR_FOUND_DUPP_KEY;
112
my_errno= HA_ERR_FOUND_DUPP_KEY;
122
115
info->s->index_length+= (keyinfo->rb_tree.allocated-old_allocated);
119
/* Find where to place new record */
121
static uchar *next_free_record_pos(HP_SHARE *info)
130
info->del_link= *((uchar**) pos);
134
if (!(block_pos=(info->records % info->block.records_in_block)))
136
if ((info->records > info->max_records && info->max_records) ||
137
(info->data_length + info->index_length >= info->max_table_size))
139
my_errno=HA_ERR_RECORD_FILE_FULL;
142
if (hp_get_new_block(&info->block,&length))
144
info->data_length+=length;
146
return((uchar*) info->block.level_info[0].last_blocks+
147
block_pos*info->block.recbuffer);
127
152
Write a hash-key to the hash-index
129
154
info Heap table info
131
156
record Table record to added
132
recpos Memory buffer where the table record will be stored if added
157
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
160
Hash index uses HP_BLOCK structure as a 'growable array' of HASH_INFO
136
161
structs. Array size == number of entries in hash index.
137
162
hp_mask(hp_rec_hashnr()) maps hash entries values to hash array positions.
138
163
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
164
they are connected in a linked list via HASH_INFO::next_key. The first
165
list element is located at position P, next elements are located at
141
166
positions for which there is no record that should be located at that
142
167
position. The order of elements in the list is arbitrary.
146
171
-1 - Out of memory
147
HA_ERR_FOUND_DUPP_KEY - Duplicate record on unique key. The record was
172
HA_ERR_FOUND_DUPP_KEY - Duplicate record on unique key. The record was
148
173
still added and the caller must call hp_delete_key for it.
151
176
int hp_write_key(HP_INFO *info, HP_KEYDEF *keyinfo,
152
const unsigned char *record, unsigned char *recpos)
177
const uchar *record, uchar *recpos)
154
179
HP_SHARE *share = info->s;
156
uint32_t halfbuff,hashnr,first_index;
157
unsigned char *ptr_to_rec= NULL,*ptr_to_rec2= NULL;
181
ulong halfbuff,hashnr,first_index;
182
uchar *ptr_to_rec= NULL,*ptr_to_rec2= NULL;
158
183
HASH_INFO *empty, *gpos= NULL, *gpos2= NULL, *pos;
162
187
return(-1); /* No more memory */
163
188
halfbuff= (long) share->blength >> 1;
164
189
pos= hp_find_hash(&keyinfo->block,(first_index=share->records-halfbuff));
167
192
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
193
The number of hash positions will change and some entries might need to
194
be relocated to the newly added position. Those entries are currently
195
members of the list that starts at #first_index position (this is
171
196
guaranteed by properties of hp_mask(hp_rec_hashnr(X)) mapping function)
172
197
At #first_index position currently there may be either:
173
198
a) An entry with hashnr != first_index. We don't need to move it.
175
200
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
202
1) entries that should be relocated to the list that starts at new
178
203
position we're adding ('uppper' list)
179
2) entries that should be left in the list starting at #first_index
204
2) entries that should be left in the list starting at #first_index
180
205
position ('lower' list)
182
207
if (pos != empty) /* If some records */