28
using namespace drizzled;
28
static uchar *next_free_record_pos(HP_SHARE *info);
30
29
static HASH_INFO *hp_find_free_hash(HP_SHARE *info, HP_BLOCK *block,
33
int heap_write(HP_INFO *info, const unsigned char *record)
32
int heap_write(HP_INFO *info, const uchar *record)
35
34
HP_KEYDEF *keydef, *end;
37
36
HP_SHARE *share=info->s;
38
uint32_t rec_length, chunk_count;
40
if ((share->records >= share->max_records && share->max_records) ||
41
(share->recordspace.total_data_length + share->index_length >= share->max_table_size))
37
DBUG_ENTER("heap_write");
39
if (info->mode & O_RDONLY)
43
return(errno=HA_ERR_RECORD_FILE_FULL);
41
DBUG_RETURN(my_errno=EACCES);
46
rec_length = hp_get_encoded_data_length(share, record, &chunk_count);
48
if (!(pos=hp_allocate_chunkset(&share->recordspace, chunk_count)))
44
if (!(pos=next_free_record_pos(share)))
45
DBUG_RETURN(my_errno);
52
48
for (keydef = share->keydef, end = keydef + share->keys; keydef < end;
59
hp_copy_record_data_to_chunkset(share, record, pos);
55
memcpy(pos,record,(size_t) share->reclength);
56
pos[share->reclength]=1; /* Mark record as not deleted */
61
57
if (++share->records == share->blength)
62
58
share->blength+= share->blength;
64
59
info->current_ptr=pos;
65
60
info->current_hash_ptr=0;
66
61
info->update|=HA_STATE_AKTIV;
62
#if !defined(DBUG_OFF) && defined(EXTRA_HEAP_DEBUG)
63
DBUG_EXECUTE("check_heap",heap_check_heap(info, 0););
67
65
if (share->auto_key)
68
66
heap_update_auto_increment(info, record);
70
if (my_errno == HA_ERR_FOUND_DUPP_KEY)
71
DBUG_PRINT("info",("Duplicate key: %d", (int) (keydef - share->keydef)));
72
72
info->errkey= keydef - share->keydef;
74
74
We don't need to delete non-inserted key from rb-tree. Also, if
85
85
if ((*keydef->delete_key)(info, keydef, record, pos, 0))
90
hp_free_chunks(&share->recordspace, pos);
91
*((uchar**) pos)=share->del_link;
93
pos[share->reclength]=0; /* Record deleted */
95
DBUG_RETURN(my_errno);
96
Write a key to rb_tree-index
99
Write a key to rb_tree-index
99
int hp_rb_write_key(HP_INFO *info, HP_KEYDEF *keyinfo, const unsigned char *record,
100
unsigned char *recpos)
102
int hp_rb_write_key(HP_INFO *info, HP_KEYDEF *keyinfo, const uchar *record,
102
105
heap_rb_param custom_arg;
103
uint32_t old_allocated;
105
108
custom_arg.keyseg= keyinfo->seg;
106
109
custom_arg.key_length= hp_rb_make_key(keyinfo, info->recbuf, record, recpos);
118
121
if (!tree_insert(&keyinfo->rb_tree, (void*)info->recbuf,
119
122
custom_arg.key_length, &custom_arg))
121
errno= HA_ERR_FOUND_DUPP_KEY;
124
my_errno= HA_ERR_FOUND_DUPP_KEY;
124
127
info->s->index_length+= (keyinfo->rb_tree.allocated-old_allocated);
131
/* Find where to place new record */
133
static uchar *next_free_record_pos(HP_SHARE *info)
138
DBUG_ENTER("next_free_record_pos");
143
info->del_link= *((uchar**) pos);
145
DBUG_PRINT("exit",("Used old position: 0x%lx",(long) pos));
148
if (!(block_pos=(info->records % info->block.records_in_block)))
150
if ((info->records > info->max_records && info->max_records) ||
151
(info->data_length + info->index_length >= info->max_table_size))
153
my_errno=HA_ERR_RECORD_FILE_FULL;
156
if (hp_get_new_block(&info->block,&length))
158
info->data_length+=length;
160
DBUG_PRINT("exit",("Used new position: 0x%lx",
161
(long) ((uchar*) info->block.level_info[0].last_blocks+
162
block_pos * info->block.recbuffer)));
163
DBUG_RETURN((uchar*) info->block.level_info[0].last_blocks+
164
block_pos*info->block.recbuffer);
129
169
Write a hash-key to the hash-index
131
171
info Heap table info
133
173
record Table record to added
134
recpos Memory buffer where the table record will be stored if added
174
recpos Memory buffer where the table record will be stored if added
137
Hash index uses HP_BLOCK structure as a 'growable array' of HASH_INFO
177
Hash index uses HP_BLOCK structure as a 'growable array' of HASH_INFO
138
178
structs. Array size == number of entries in hash index.
139
179
hp_mask(hp_rec_hashnr()) maps hash entries values to hash array positions.
140
180
If there are several hash entries with the same hash array position P,
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
181
they are connected in a linked list via HASH_INFO::next_key. The first
182
list element is located at position P, next elements are located at
143
183
positions for which there is no record that should be located at that
144
184
position. The order of elements in the list is arbitrary.
148
188
-1 - Out of memory
149
HA_ERR_FOUND_DUPP_KEY - Duplicate record on unique key. The record was
189
HA_ERR_FOUND_DUPP_KEY - Duplicate record on unique key. The record was
150
190
still added and the caller must call hp_delete_key for it.
153
193
int hp_write_key(HP_INFO *info, HP_KEYDEF *keyinfo,
154
const unsigned char *record, unsigned char *recpos)
194
const uchar *record, uchar *recpos)
156
196
HP_SHARE *share = info->s;
158
uint32_t halfbuff,hashnr,first_index;
159
unsigned char *ptr_to_rec= NULL,*ptr_to_rec2= NULL;
160
HASH_INFO *empty, *gpos= NULL, *gpos2= NULL, *pos;
198
ulong halfbuff,hashnr,first_index;
199
uchar *ptr_to_rec,*ptr_to_rec2;
200
HASH_INFO *empty,*gpos,*gpos2,*pos;
201
DBUG_ENTER("hp_write_key");
163
204
if (!(empty= hp_find_free_hash(share,&keyinfo->block,share->records)))
164
return(-1); /* No more memory */
205
DBUG_RETURN(-1); /* No more memory */
165
206
halfbuff= (long) share->blength >> 1;
166
207
pos= hp_find_hash(&keyinfo->block,(first_index=share->records-halfbuff));
169
210
We're about to add one more hash array position, with hash_mask=#records.
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
211
The number of hash positions will change and some entries might need to
212
be relocated to the newly added position. Those entries are currently
213
members of the list that starts at #first_index position (this is
173
214
guaranteed by properties of hp_mask(hp_rec_hashnr(X)) mapping function)
174
215
At #first_index position currently there may be either:
175
216
a) An entry with hashnr != first_index. We don't need to move it.
177
218
b) A list of items with hash_mask=first_index. The list contains entries
179
1) entries that should be relocated to the list that starts at new
220
1) entries that should be relocated to the list that starts at new
180
221
position we're adding ('uppper' list)
181
2) entries that should be left in the list starting at #first_index
222
2) entries that should be left in the list starting at #first_index
182
223
position ('lower' list)
184
225
if (pos != empty) /* If some records */