55
54
#include <drizzled/item/null.h>
56
55
#include <drizzled/temporal.h>
58
#include <drizzled/refresh_version.h>
60
#include <drizzled/table/singular.h>
62
#include <drizzled/table_proto.h>
63
#include <drizzled/typelib.h>
57
#include "drizzled/table_proto.h"
65
59
using namespace std;
64
extern pid_t current_pid;
70
65
extern plugin::StorageEngine *heap_engine;
71
66
extern plugin::StorageEngine *myisam_engine;
73
68
/* Functions defined in this cursor */
70
void open_table_error(TableShare *share, int error, int db_errno,
71
myf errortype, int errarg);
75
73
/*************************************************************************/
77
// @note this should all be the destructor
78
int Table::delete_table(bool free_share)
75
/* Get column name from column hash */
77
static unsigned char *get_field_name(Field **buff, size_t *length, bool)
79
*length= (uint32_t) strlen((*buff)->field_name);
80
return (unsigned char*) (*buff)->field_name;
83
static TABLE_CATEGORY get_table_category(const LEX_STRING *db)
87
return TABLE_CATEGORY_USER;
92
Allocate a setup TableShare structure
96
TableList Take database and table name from there
97
key Table cache key (db \0 table_name \0...)
98
key_length Length of key
101
0 Error (out of memory)
105
TableShare *alloc_table_share(TableList *table_list, char *key,
108
memory::Root mem_root;
110
char *key_buff, *path_buff;
111
char path[FN_REFLEN];
112
uint32_t path_length;
114
path_length= build_table_filename(path, sizeof(path) - 1,
116
table_list->table_name, false);
117
memory::init_sql_alloc(&mem_root, TABLE_ALLOC_BLOCK_SIZE, 0);
118
if (multi_alloc_root(&mem_root,
119
&share, sizeof(*share),
120
&key_buff, key_length,
121
&path_buff, path_length + 1,
124
memset(share, 0, sizeof(*share));
126
share->set_table_cache_key(key_buff, key, key_length);
128
share->path.str= path_buff;
129
share->path.length= path_length;
130
strcpy(share->path.str, path);
131
share->normalized_path.str= share->path.str;
132
share->normalized_path.length= path_length;
134
share->version= refresh_version;
136
memcpy(&share->mem_root, &mem_root, sizeof(mem_root));
137
pthread_mutex_init(&share->mutex, MY_MUTEX_INIT_FAST);
138
pthread_cond_init(&share->cond, NULL);
144
static enum_field_types proto_field_type_to_drizzle_type(uint32_t proto_field_type)
146
enum_field_types field_type;
148
switch(proto_field_type)
150
case message::Table::Field::INTEGER:
151
field_type= DRIZZLE_TYPE_LONG;
153
case message::Table::Field::DOUBLE:
154
field_type= DRIZZLE_TYPE_DOUBLE;
156
case message::Table::Field::TIMESTAMP:
157
field_type= DRIZZLE_TYPE_TIMESTAMP;
159
case message::Table::Field::BIGINT:
160
field_type= DRIZZLE_TYPE_LONGLONG;
162
case message::Table::Field::DATETIME:
163
field_type= DRIZZLE_TYPE_DATETIME;
165
case message::Table::Field::DATE:
166
field_type= DRIZZLE_TYPE_DATE;
168
case message::Table::Field::VARCHAR:
169
field_type= DRIZZLE_TYPE_VARCHAR;
171
case message::Table::Field::DECIMAL:
172
field_type= DRIZZLE_TYPE_DECIMAL;
174
case message::Table::Field::ENUM:
175
field_type= DRIZZLE_TYPE_ENUM;
177
case message::Table::Field::BLOB:
178
field_type= DRIZZLE_TYPE_BLOB;
181
field_type= DRIZZLE_TYPE_LONG; /* Set value to kill GCC warning */
188
static Item *default_value_item(enum_field_types field_type,
189
const CHARSET_INFO *charset,
190
bool default_null, const string *default_value,
191
const string *default_bin_value)
193
Item *default_item= NULL;
198
return new Item_null();
203
case DRIZZLE_TYPE_LONG:
204
case DRIZZLE_TYPE_LONGLONG:
205
default_item= new Item_int(default_value->c_str(),
206
(int64_t) internal::my_strtoll10(default_value->c_str(),
209
default_value->length());
211
case DRIZZLE_TYPE_DOUBLE:
212
default_item= new Item_float(default_value->c_str(),
213
default_value->length());
215
case DRIZZLE_TYPE_NULL:
217
case DRIZZLE_TYPE_TIMESTAMP:
218
case DRIZZLE_TYPE_DATETIME:
219
case DRIZZLE_TYPE_DATE:
220
if (default_value->compare("NOW()") == 0)
222
case DRIZZLE_TYPE_ENUM:
223
default_item= new Item_string(default_value->c_str(),
224
default_value->length(),
225
system_charset_info);
227
case DRIZZLE_TYPE_VARCHAR:
228
case DRIZZLE_TYPE_BLOB: /* Blob is here due to TINYTEXT. Feel the hate. */
229
if (charset==&my_charset_bin)
231
default_item= new Item_string(default_bin_value->c_str(),
232
default_bin_value->length(),
237
default_item= new Item_string(default_value->c_str(),
238
default_value->length(),
239
system_charset_info);
242
case DRIZZLE_TYPE_DECIMAL:
243
default_item= new Item_decimal(default_value->c_str(),
244
default_value->length(),
245
system_charset_info);
252
int parse_table_proto(Session& session,
253
message::Table &table,
258
if (! table.IsInitialized())
260
my_error(ER_CORRUPT_TABLE_DEFINITION, MYF(0), table.InitializationErrorString().c_str());
261
return ER_CORRUPT_TABLE_DEFINITION;
264
share->setTableProto(new(nothrow) message::Table(table));
266
share->storage_engine= plugin::StorageEngine::findByName(session, table.engine().name());
267
assert(share->storage_engine); // We use an assert() here because we should never get this far and still have no suitable engine.
269
message::Table::TableOptions table_options;
271
if (table.has_options())
272
table_options= table.options();
274
uint32_t db_create_options= 0;
276
if (table_options.has_pack_keys())
278
if (table_options.pack_keys())
279
db_create_options|= HA_OPTION_PACK_KEYS;
281
db_create_options|= HA_OPTION_NO_PACK_KEYS;
284
if (table_options.pack_record())
285
db_create_options|= HA_OPTION_PACK_RECORD;
287
/* db_create_options was stored as 2 bytes in FRM
288
Any HA_OPTION_ that doesn't fit into 2 bytes was silently truncated away.
290
share->db_create_options= (db_create_options & 0x0000FFFF);
291
share->db_options_in_use= share->db_create_options;
293
share->row_type= table_options.has_row_type() ?
294
(enum row_type) table_options.row_type() : ROW_TYPE_DEFAULT;
296
share->block_size= table_options.has_block_size() ?
297
table_options.block_size() : 0;
299
share->table_charset= get_charset(table_options.has_collation_id()?
300
table_options.collation_id() : 0);
302
if (!share->table_charset)
304
/* unknown charset in head[38] or pre-3.23 frm */
305
if (use_mb(default_charset_info))
307
/* Warn that we may be changing the size of character columns */
308
errmsg_printf(ERRMSG_LVL_WARN,
309
_("'%s' had no or invalid character set, "
310
"and default character set is multi-byte, "
311
"so character column sizes may have changed"),
314
share->table_charset= default_charset_info;
317
share->db_record_offset= 1;
319
share->blob_ptr_size= portable_sizeof_char_ptr; // more bonghits.
321
share->keys= table.indexes_size();
324
for (int indx= 0; indx < table.indexes_size(); indx++)
325
share->key_parts+= table.indexes(indx).index_part_size();
327
share->key_info= (KEY*) alloc_root(&share->mem_root,
328
table.indexes_size() * sizeof(KEY)
329
+share->key_parts*sizeof(KEY_PART_INFO));
331
KEY_PART_INFO *key_part;
333
key_part= reinterpret_cast<KEY_PART_INFO*>
334
(share->key_info+table.indexes_size());
337
ulong *rec_per_key= (ulong*) alloc_root(&share->mem_root,
338
sizeof(ulong*)*share->key_parts);
340
share->keynames.count= table.indexes_size();
341
share->keynames.name= NULL;
342
share->keynames.type_names= (const char**)
343
alloc_root(&share->mem_root, sizeof(char*) * (table.indexes_size()+1));
345
share->keynames.type_lengths= (unsigned int*)
346
alloc_root(&share->mem_root,
347
sizeof(unsigned int) * (table.indexes_size()+1));
349
share->keynames.type_names[share->keynames.count]= NULL;
350
share->keynames.type_lengths[share->keynames.count]= 0;
352
KEY* keyinfo= share->key_info;
353
for (int keynr= 0; keynr < table.indexes_size(); keynr++, keyinfo++)
355
message::Table::Index indx= table.indexes(keynr);
360
if (indx.is_unique())
361
keyinfo->flags|= HA_NOSAME;
363
if (indx.has_options())
365
message::Table::Index::IndexOptions indx_options= indx.options();
366
if (indx_options.pack_key())
367
keyinfo->flags|= HA_PACK_KEY;
369
if (indx_options.var_length_key())
370
keyinfo->flags|= HA_VAR_LENGTH_PART;
372
if (indx_options.null_part_key())
373
keyinfo->flags|= HA_NULL_PART_KEY;
375
if (indx_options.binary_pack_key())
376
keyinfo->flags|= HA_BINARY_PACK_KEY;
378
if (indx_options.has_partial_segments())
379
keyinfo->flags|= HA_KEY_HAS_PART_KEY_SEG;
381
if (indx_options.auto_generated_key())
382
keyinfo->flags|= HA_GENERATED_KEY;
384
if (indx_options.has_key_block_size())
386
keyinfo->flags|= HA_USES_BLOCK_SIZE;
387
keyinfo->block_size= indx_options.key_block_size();
391
keyinfo->block_size= 0;
397
case message::Table::Index::UNKNOWN_INDEX:
398
keyinfo->algorithm= HA_KEY_ALG_UNDEF;
400
case message::Table::Index::BTREE:
401
keyinfo->algorithm= HA_KEY_ALG_BTREE;
403
case message::Table::Index::HASH:
404
keyinfo->algorithm= HA_KEY_ALG_HASH;
408
/* TODO: suitable warning ? */
409
keyinfo->algorithm= HA_KEY_ALG_UNDEF;
413
keyinfo->key_length= indx.key_length();
415
keyinfo->key_parts= indx.index_part_size();
417
keyinfo->key_part= key_part;
418
keyinfo->rec_per_key= rec_per_key;
420
for (unsigned int partnr= 0;
421
partnr < keyinfo->key_parts;
422
partnr++, key_part++)
424
message::Table::Index::IndexPart part;
425
part= indx.index_part(partnr);
429
key_part->field= NULL;
430
key_part->fieldnr= part.fieldnr() + 1; // start from 1.
431
key_part->null_bit= 0;
432
/* key_part->null_offset is only set if null_bit (see later) */
433
/* key_part->key_type= */ /* I *THINK* this may be okay.... */
434
/* key_part->type ???? */
435
key_part->key_part_flag= 0;
436
if (part.has_in_reverse_order())
437
key_part->key_part_flag= part.in_reverse_order()? HA_REVERSE_SORT : 0;
439
key_part->length= part.compare_length();
441
key_part->store_length= key_part->length;
443
/* key_part->offset is set later */
444
key_part->key_type= part.key_type();
447
if (! indx.has_comment())
449
keyinfo->comment.length= 0;
450
keyinfo->comment.str= NULL;
454
keyinfo->flags|= HA_USES_COMMENT;
455
keyinfo->comment.length= indx.comment().length();
456
keyinfo->comment.str= strmake_root(&share->mem_root,
457
indx.comment().c_str(),
458
keyinfo->comment.length);
461
keyinfo->name= strmake_root(&share->mem_root,
463
indx.name().length());
465
share->keynames.type_names[keynr]= keyinfo->name;
466
share->keynames.type_lengths[keynr]= indx.name().length();
469
share->keys_for_keyread.reset();
470
set_prefix(share->keys_in_use, share->keys);
472
share->fields= table.field_size();
474
share->field= (Field**) alloc_root(&share->mem_root,
475
((share->fields+1) * sizeof(Field*)));
476
share->field[share->fields]= NULL;
478
uint32_t null_fields= 0;
481
uint32_t *field_offsets= (uint32_t*)malloc(share->fields * sizeof(uint32_t));
482
uint32_t *field_pack_length=(uint32_t*)malloc(share->fields*sizeof(uint32_t));
484
assert(field_offsets && field_pack_length); // TODO: fixme
486
uint32_t interval_count= 0;
487
uint32_t interval_parts= 0;
489
uint32_t stored_columns_reclength= 0;
491
for (unsigned int fieldnr= 0; fieldnr < share->fields; fieldnr++)
493
message::Table::Field pfield= table.field(fieldnr);
494
if (pfield.constraints().is_nullable())
497
enum_field_types drizzle_field_type=
498
proto_field_type_to_drizzle_type(pfield.type());
500
field_offsets[fieldnr]= stored_columns_reclength;
502
/* the below switch is very similar to
503
CreateField::create_length_to_internal_length in field.cc
504
(which should one day be replace by just this code)
506
switch(drizzle_field_type)
508
case DRIZZLE_TYPE_BLOB:
509
case DRIZZLE_TYPE_VARCHAR:
511
message::Table::Field::StringFieldOptions field_options= pfield.string_options();
513
const CHARSET_INFO *cs= get_charset(field_options.has_collation_id() ?
514
field_options.collation_id() : 0);
517
cs= default_charset_info;
519
field_pack_length[fieldnr]= calc_pack_length(drizzle_field_type,
520
field_options.length() * cs->mbmaxlen);
523
case DRIZZLE_TYPE_ENUM:
525
message::Table::Field::SetFieldOptions field_options= pfield.set_options();
527
field_pack_length[fieldnr]=
528
get_enum_pack_length(field_options.field_value_size());
531
interval_parts+= field_options.field_value_size();
534
case DRIZZLE_TYPE_DECIMAL:
536
message::Table::Field::NumericFieldOptions fo= pfield.numeric_options();
538
field_pack_length[fieldnr]= my_decimal_get_binary_size(fo.precision(), fo.scale());
542
/* Zero is okay here as length is fixed for other types. */
543
field_pack_length[fieldnr]= calc_pack_length(drizzle_field_type, 0);
546
share->reclength+= field_pack_length[fieldnr];
547
stored_columns_reclength+= field_pack_length[fieldnr];
550
/* data_offset added to stored_rec_length later */
551
share->stored_rec_length= stored_columns_reclength;
553
share->null_fields= null_fields;
555
ulong null_bits= null_fields;
556
if (! table_options.pack_record())
558
ulong data_offset= (null_bits + 7)/8;
561
share->reclength+= data_offset;
562
share->stored_rec_length+= data_offset;
564
ulong rec_buff_length;
566
rec_buff_length= ALIGN_SIZE(share->reclength + 1);
567
share->rec_buff_length= rec_buff_length;
569
unsigned char* record= NULL;
571
if (! (record= (unsigned char *) alloc_root(&share->mem_root,
575
memset(record, 0, rec_buff_length);
579
if (! table_options.pack_record())
581
null_count++; // one bit for delete mark.
585
share->default_values= record;
589
share->intervals= (TYPELIB *) alloc_root(&share->mem_root,
590
interval_count*sizeof(TYPELIB));
593
share->intervals= NULL;
595
share->fieldnames.type_names= (const char **) alloc_root(&share->mem_root,
596
(share->fields + 1) * sizeof(char*));
598
share->fieldnames.type_lengths= (unsigned int *) alloc_root(&share->mem_root,
599
(share->fields + 1) * sizeof(unsigned int));
601
share->fieldnames.type_names[share->fields]= NULL;
602
share->fieldnames.type_lengths[share->fields]= 0;
603
share->fieldnames.count= share->fields;
606
/* Now fix the TYPELIBs for the intervals (enum values)
610
uint32_t interval_nr= 0;
612
for (unsigned int fieldnr= 0; fieldnr < share->fields; fieldnr++)
614
message::Table::Field pfield= table.field(fieldnr);
617
share->fieldnames.type_names[fieldnr]= strmake_root(&share->mem_root,
618
pfield.name().c_str(),
619
pfield.name().length());
621
share->fieldnames.type_lengths[fieldnr]= pfield.name().length();
624
if (pfield.type() != message::Table::Field::ENUM)
627
message::Table::Field::SetFieldOptions field_options= pfield.set_options();
629
const CHARSET_INFO *charset= get_charset(field_options.has_collation_id() ?
630
field_options.collation_id() : 0);
633
charset= default_charset_info;
635
TYPELIB *t= &(share->intervals[interval_nr]);
637
t->type_names= (const char**)alloc_root(&share->mem_root,
638
(field_options.field_value_size() + 1) * sizeof(char*));
640
t->type_lengths= (unsigned int*) alloc_root(&share->mem_root,
641
(field_options.field_value_size() + 1) * sizeof(unsigned int));
643
t->type_names[field_options.field_value_size()]= NULL;
644
t->type_lengths[field_options.field_value_size()]= 0;
646
t->count= field_options.field_value_size();
649
for (int n= 0; n < field_options.field_value_size(); n++)
651
t->type_names[n]= strmake_root(&share->mem_root,
652
field_options.field_value(n).c_str(),
653
field_options.field_value(n).length());
656
* Go ask the charset what the length is as for "" length=1
657
* and there's stripping spaces or some other crack going on.
660
lengthsp= charset->cset->lengthsp(charset,
662
field_options.field_value(n).length());
663
t->type_lengths[n]= lengthsp;
669
/* and read the fields */
672
bool use_hash= share->fields >= MAX_FIELDS_BEFORE_HASH;
675
use_hash= ! hash_init(&share->name_hash,
680
(hash_get_key) get_field_name,
684
unsigned char* null_pos= record;;
685
int null_bit_pos= (table_options.pack_record()) ? 0 : 1;
687
for (unsigned int fieldnr= 0; fieldnr < share->fields; fieldnr++)
689
message::Table::Field pfield= table.field(fieldnr);
691
enum column_format_type column_format= COLUMN_FORMAT_TYPE_DEFAULT;
693
switch (pfield.format())
695
case message::Table::Field::DefaultFormat:
696
column_format= COLUMN_FORMAT_TYPE_DEFAULT;
698
case message::Table::Field::FixedFormat:
699
column_format= COLUMN_FORMAT_TYPE_FIXED;
701
case message::Table::Field::DynamicFormat:
702
column_format= COLUMN_FORMAT_TYPE_DYNAMIC;
708
Field::utype unireg_type= Field::NONE;
710
if (pfield.has_numeric_options() &&
711
pfield.numeric_options().is_autoincrement())
713
unireg_type= Field::NEXT_NUMBER;
716
if (pfield.has_options() &&
717
pfield.options().has_default_value() &&
718
pfield.options().default_value().compare("NOW()") == 0)
720
if (pfield.options().has_update_value() &&
721
pfield.options().update_value().compare("NOW()") == 0)
723
unireg_type= Field::TIMESTAMP_DNUN_FIELD;
725
else if (! pfield.options().has_update_value())
727
unireg_type= Field::TIMESTAMP_DN_FIELD;
730
assert(1); // Invalid update value.
732
else if (pfield.has_options() &&
733
pfield.options().has_update_value() &&
734
pfield.options().update_value().compare("NOW()") == 0)
736
unireg_type= Field::TIMESTAMP_UN_FIELD;
740
if (!pfield.has_comment())
742
comment.str= (char*)"";
747
size_t len= pfield.comment().length();
748
const char* str= pfield.comment().c_str();
750
comment.str= strmake_root(&share->mem_root, str, len);
754
enum_field_types field_type;
756
field_type= proto_field_type_to_drizzle_type(pfield.type());
758
const CHARSET_INFO *charset= &my_charset_bin;
760
if (field_type == DRIZZLE_TYPE_BLOB ||
761
field_type == DRIZZLE_TYPE_VARCHAR)
763
message::Table::Field::StringFieldOptions field_options= pfield.string_options();
765
charset= get_charset(field_options.has_collation_id() ?
766
field_options.collation_id() : 0);
769
charset= default_charset_info;
772
if (field_type == DRIZZLE_TYPE_ENUM)
774
message::Table::Field::SetFieldOptions field_options= pfield.set_options();
776
charset= get_charset(field_options.has_collation_id()?
777
field_options.collation_id() : 0);
780
charset= default_charset_info;
784
if (field_type == DRIZZLE_TYPE_DECIMAL
785
|| field_type == DRIZZLE_TYPE_DOUBLE)
787
message::Table::Field::NumericFieldOptions fo= pfield.numeric_options();
789
if (! pfield.has_numeric_options() || ! fo.has_scale())
792
We don't write the default to table proto so
793
if no decimals specified for DOUBLE, we use the default.
795
decimals= NOT_FIXED_DEC;
799
if (fo.scale() > DECIMAL_MAX_SCALE)
804
decimals= static_cast<uint8_t>(fo.scale());
808
Item *default_value= NULL;
810
if (pfield.options().has_default_value() ||
811
pfield.options().has_default_null() ||
812
pfield.options().has_default_bin_value())
814
default_value= default_value_item(field_type,
816
pfield.options().default_null(),
817
&pfield.options().default_value(),
818
&pfield.options().default_bin_value());
822
Table temp_table; /* Use this so that BLOB DEFAULT '' works */
823
memset(&temp_table, 0, sizeof(temp_table));
825
temp_table.in_use= &session;
826
temp_table.s->db_low_byte_first= true; //Cursor->low_byte_first();
827
temp_table.s->blob_ptr_size= portable_sizeof_char_ptr;
829
uint32_t field_length= 0; //Assignment is for compiler complaint.
833
case DRIZZLE_TYPE_BLOB:
834
case DRIZZLE_TYPE_VARCHAR:
836
message::Table::Field::StringFieldOptions field_options= pfield.string_options();
838
charset= get_charset(field_options.has_collation_id() ?
839
field_options.collation_id() : 0);
842
charset= default_charset_info;
844
field_length= field_options.length() * charset->mbmaxlen;
847
case DRIZZLE_TYPE_DOUBLE:
849
message::Table::Field::NumericFieldOptions fo= pfield.numeric_options();
850
if (!fo.has_precision() && !fo.has_scale())
852
field_length= DBL_DIG+7;
856
field_length= fo.precision();
858
if (field_length < decimals &&
859
decimals != NOT_FIXED_DEC)
861
my_error(ER_M_BIGGER_THAN_D, MYF(0), pfield.name().c_str());
867
case DRIZZLE_TYPE_DECIMAL:
869
message::Table::Field::NumericFieldOptions fo= pfield.numeric_options();
871
field_length= my_decimal_precision_to_length(fo.precision(), fo.scale(),
875
case DRIZZLE_TYPE_TIMESTAMP:
876
case DRIZZLE_TYPE_DATETIME:
877
field_length= DateTime::MAX_STRING_LENGTH;
879
case DRIZZLE_TYPE_DATE:
880
field_length= Date::MAX_STRING_LENGTH;
882
case DRIZZLE_TYPE_ENUM:
886
message::Table::Field::SetFieldOptions fo= pfield.set_options();
888
for(int valnr= 0; valnr < fo.field_value_size(); valnr++)
890
if (fo.field_value(valnr).length() > field_length)
891
field_length= charset->cset->numchars(charset,
892
fo.field_value(valnr).c_str(),
893
fo.field_value(valnr).c_str()
894
+ fo.field_value(valnr).length())
899
case DRIZZLE_TYPE_LONG:
901
uint32_t sign_len= pfield.constraints().is_unsigned() ? 0 : 1;
902
field_length= MAX_INT_WIDTH+sign_len;
905
case DRIZZLE_TYPE_LONGLONG:
906
field_length= MAX_BIGINT_WIDTH;
908
case DRIZZLE_TYPE_NULL:
909
abort(); // Programming error
912
Field* f= make_field(share,
914
record + field_offsets[fieldnr] + data_offset,
916
pfield.constraints().is_nullable(),
922
(Field::utype) MTYP_TYPENR(unireg_type),
923
((field_type == DRIZZLE_TYPE_ENUM) ?
924
share->intervals + (interval_nr++)
926
share->fieldnames.type_names[fieldnr]);
928
share->field[fieldnr]= f;
930
f->init(&temp_table); /* blob default values need table obj */
932
if (! (f->flags & NOT_NULL_FLAG))
934
*f->null_ptr|= f->null_bit;
935
if (! (null_bit_pos= (null_bit_pos + 1) & 7)) /* @TODO Ugh. */
942
enum_check_fields old_count_cuted_fields= session.count_cuted_fields;
943
session.count_cuted_fields= CHECK_FIELD_WARN;
944
int res= default_value->save_in_field(f, 1);
945
session.count_cuted_fields= old_count_cuted_fields;
946
if (res != 0 && res != 3) /* @TODO Huh? */
948
my_error(ER_INVALID_DEFAULT, MYF(0), f->field_name);
953
else if (f->real_type() == DRIZZLE_TYPE_ENUM &&
954
(f->flags & NOT_NULL_FLAG))
957
f->store((int64_t) 1, true);
962
/* hack to undo f->init() */
966
f->field_index= fieldnr;
968
if (! default_value &&
969
! (f->unireg_check==Field::NEXT_NUMBER) &&
970
(f->flags & NOT_NULL_FLAG) &&
971
(f->real_type() != DRIZZLE_TYPE_TIMESTAMP))
973
f->flags|= NO_DEFAULT_VALUE_FLAG;
976
if (f->unireg_check == Field::NEXT_NUMBER)
977
share->found_next_number_field= &(share->field[fieldnr]);
979
if (share->timestamp_field == f)
980
share->timestamp_field_offset= fieldnr;
982
if (use_hash) /* supposedly this never fails... but comments lie */
983
(void) my_hash_insert(&share->name_hash,
984
(unsigned char*)&(share->field[fieldnr]));
988
keyinfo= share->key_info;
989
for (unsigned int keynr= 0; keynr < share->keys; keynr++, keyinfo++)
991
key_part= keyinfo->key_part;
993
for (unsigned int partnr= 0;
994
partnr < keyinfo->key_parts;
995
partnr++, key_part++)
998
* Fix up key_part->offset by adding data_offset.
999
* We really should compute offset as well.
1000
* But at least this way we are a little better.
1002
key_part->offset= field_offsets[key_part->fieldnr-1] + data_offset;
1007
We need to set the unused bits to 1. If the number of bits is a multiple
1008
of 8 there are no unused bits.
1012
*(record + null_count / 8)|= ~(((unsigned char) 1 << (null_count & 7)) - 1);
1014
share->null_bytes= (null_pos - (unsigned char*) record + (null_bit_pos + 7) / 8);
1016
share->last_null_bit_pos= null_bit_pos;
1018
free(field_offsets);
1019
field_offsets= NULL;
1020
free(field_pack_length);
1021
field_pack_length= NULL;
1024
if (share->key_parts)
1026
uint32_t primary_key= (uint32_t) (find_type((char*) "PRIMARY",
1027
&share->keynames, 3) - 1); /* @TODO Huh? */
1029
keyinfo= share->key_info;
1030
key_part= keyinfo->key_part;
1032
for (uint32_t key= 0; key < share->keys; key++,keyinfo++)
1034
uint32_t usable_parts= 0;
1036
if (primary_key >= MAX_KEY && (keyinfo->flags & HA_NOSAME))
1039
If the UNIQUE key doesn't have NULL columns and is not a part key
1040
declare this as a primary key.
1043
for (uint32_t i= 0; i < keyinfo->key_parts; i++)
1045
uint32_t fieldnr= key_part[i].fieldnr;
1047
share->field[fieldnr-1]->null_ptr ||
1048
share->field[fieldnr-1]->key_length() != key_part[i].length)
1050
primary_key= MAX_KEY; // Can't be used
1056
for (uint32_t i= 0 ; i < keyinfo->key_parts ; key_part++,i++)
1059
if (! key_part->fieldnr)
1061
abort(); // goto err;
1063
field= key_part->field= share->field[key_part->fieldnr-1];
1064
key_part->type= field->key_type();
1065
if (field->null_ptr)
1067
key_part->null_offset=(uint32_t) ((unsigned char*) field->null_ptr -
1068
share->default_values);
1069
key_part->null_bit= field->null_bit;
1070
key_part->store_length+=HA_KEY_NULL_LENGTH;
1071
keyinfo->flags|=HA_NULL_PART_KEY;
1072
keyinfo->extra_length+= HA_KEY_NULL_LENGTH;
1073
keyinfo->key_length+= HA_KEY_NULL_LENGTH;
1075
if (field->type() == DRIZZLE_TYPE_BLOB ||
1076
field->real_type() == DRIZZLE_TYPE_VARCHAR)
1078
if (field->type() == DRIZZLE_TYPE_BLOB)
1079
key_part->key_part_flag|= HA_BLOB_PART;
1081
key_part->key_part_flag|= HA_VAR_LENGTH_PART;
1082
keyinfo->extra_length+=HA_KEY_BLOB_LENGTH;
1083
key_part->store_length+=HA_KEY_BLOB_LENGTH;
1084
keyinfo->key_length+= HA_KEY_BLOB_LENGTH;
1086
if (i == 0 && key != primary_key)
1087
field->flags |= (((keyinfo->flags & HA_NOSAME) &&
1088
(keyinfo->key_parts == 1)) ?
1089
UNIQUE_KEY_FLAG : MULTIPLE_KEY_FLAG);
1091
field->key_start.set(key);
1092
if (field->key_length() == key_part->length &&
1093
!(field->flags & BLOB_FLAG))
1095
enum ha_key_alg algo= share->key_info[key].algorithm;
1096
if (share->db_type()->index_flags(algo) & HA_KEYREAD_ONLY)
1098
share->keys_for_keyread.set(key);
1099
field->part_of_key.set(key);
1100
field->part_of_key_not_clustered.set(key);
1102
if (share->db_type()->index_flags(algo) & HA_READ_ORDER)
1103
field->part_of_sortkey.set(key);
1105
if (!(key_part->key_part_flag & HA_REVERSE_SORT) &&
1107
usable_parts++; // For FILESORT
1108
field->flags|= PART_KEY_FLAG;
1109
if (key == primary_key)
1111
field->flags|= PRI_KEY_FLAG;
1113
If this field is part of the primary key and all keys contains
1114
the primary key, then we can use any key to find this column
1116
if (share->storage_engine->check_flag(HTON_BIT_PRIMARY_KEY_IN_READ_INDEX))
1118
field->part_of_key= share->keys_in_use;
1119
if (field->part_of_sortkey.test(key))
1120
field->part_of_sortkey= share->keys_in_use;
1123
if (field->key_length() != key_part->length)
1125
key_part->key_part_flag|= HA_PART_KEY_SEG;
1128
keyinfo->usable_key_parts= usable_parts; // Filesort
1130
set_if_bigger(share->max_key_length,keyinfo->key_length+
1131
keyinfo->key_parts);
1132
share->total_key_length+= keyinfo->key_length;
1134
if (keyinfo->flags & HA_NOSAME)
1136
set_if_bigger(share->max_unique_length,keyinfo->key_length);
1139
if (primary_key < MAX_KEY &&
1140
(share->keys_in_use.test(primary_key)))
1142
share->primary_key= primary_key;
1144
If we are using an integer as the primary key then allow the user to
1145
refer to it as '_rowid'
1147
if (share->key_info[primary_key].key_parts == 1)
1149
Field *field= share->key_info[primary_key].key_part[0].field;
1150
if (field && field->result_type() == INT_RESULT)
1152
/* note that fieldnr here (and rowid_field_offset) starts from 1 */
1153
share->rowid_field_offset= (share->key_info[primary_key].key_part[0].
1159
share->primary_key = MAX_KEY; // we do not have a primary key
1162
share->primary_key= MAX_KEY;
1164
if (share->found_next_number_field)
1166
Field *reg_field= *share->found_next_number_field;
1167
if ((int) (share->next_number_index= (uint32_t)
1168
find_ref_key(share->key_info, share->keys,
1169
share->default_values, reg_field,
1170
&share->next_number_key_offset,
1171
&share->next_number_keypart)) < 0)
1173
/* Wrong field definition */
1178
reg_field->flags |= AUTO_INCREMENT_FLAG;
1181
if (share->blob_fields)
1186
/* Store offsets to blob fields to find them fast */
1187
if (!(share->blob_field= save=
1188
(uint*) alloc_root(&share->mem_root,
1189
(uint32_t) (share->blob_fields* sizeof(uint32_t)))))
1191
for (k= 0, ptr= share->field ; *ptr ; ptr++, k++)
1193
if ((*ptr)->flags & BLOB_FLAG)
1198
share->db_low_byte_first= true; // @todo Question this.
1199
share->column_bitmap_size= bitmap_buffer_size(share->fields);
1201
my_bitmap_map *bitmaps;
1203
if (!(bitmaps= (my_bitmap_map*) alloc_root(&share->mem_root,
1204
share->column_bitmap_size)))
1206
share->all_set.init(bitmaps, share->fields);
1207
share->all_set.setAll();
1213
free(field_offsets);
1214
if (field_pack_length)
1215
free(field_pack_length);
1217
share->error= error;
1218
share->open_errno= errno;
1220
hash_free(&share->name_hash);
1221
share->open_table_error(error, share->open_errno, 0);
1227
Read table definition from a binary / text based .frm cursor
1231
session Thread Cursor
1232
share Fill this with table definition
1235
This function is called when the table definition is not cached in
1237
The data is returned in 'share', which is alloced by
1238
alloc_table_share().. The code assumes that share is initialized.
1242
1 Error (see open_table_error)
1243
2 Error (see open_table_error)
1244
3 Wrong data in .frm cursor
1245
4 Error (see open_table_error)
1246
5 Error (see open_table_error: charset unavailable)
1247
6 Unknown .frm version
1250
int open_table_def(Session& session, TableShare *share)
1258
message::Table table;
1260
error= plugin::StorageEngine::getTableDefinition(session, share->normalized_path.str,
1262
share->table_name.str,
1266
if (error != EEXIST)
1275
if (not table.IsInitialized())
1283
error= parse_table_proto(session, table, share);
1285
share->table_category= get_table_category(& share->db);
1288
session.status_var.opened_shares++;
1291
if (error && !error_given)
1293
share->error= error;
1294
share->open_table_error(error, (share->open_errno= errno), 0);
1302
Open a table based on a TableShare
1305
open_table_from_share()
1306
session Thread Cursor
1307
share Table definition
1308
alias Alias for table
1309
db_stat open flags (for example HA_OPEN_KEYFILE|
1310
HA_OPEN_RNDFILE..) can be 0 (example in
1312
ha_open_flags HA_OPEN_ABORT_IF_LOCKED etc..
1313
outparam result table
1317
1 Error (see open_table_error)
1318
2 Error (see open_table_error)
1319
3 Wrong data in .frm cursor
1320
4 Error (see open_table_error)
1321
5 Error (see open_table_error: charset unavailable)
1322
7 Table definition has changed in engine
1325
int open_table_from_share(Session *session, TableShare *share, const char *alias,
1326
uint32_t db_stat, uint32_t ha_open_flags,
1330
uint32_t records, i, bitmap_size;
1331
bool error_reported= false;
1332
unsigned char *record, *bitmaps;
1335
/* Parsing of partitioning information from .frm needs session->lex set up. */
1336
assert(session->lex->is_lex_started);
1339
outparam->resetTable(session, share, db_stat);
1342
if (not (outparam->alias= strdup(alias)))
1345
/* Allocate Cursor */
1346
if (not (outparam->cursor= share->db_type()->getCursor(*share, &outparam->mem_root)))
1351
if ((db_stat & HA_OPEN_KEYFILE))
1356
if (!(record= (unsigned char*) alloc_root(&outparam->mem_root,
1357
share->rec_buff_length * records)))
1362
/* We are probably in hard repair, and the buffers should not be used */
1363
outparam->record[0]= outparam->record[1]= share->default_values;
1367
outparam->record[0]= record;
1369
outparam->record[1]= record+ share->rec_buff_length;
1371
outparam->record[1]= outparam->record[0]; // Safety
1376
We need this because when we read var-length rows, we are not updating
1377
bytes after end of varchar
1381
memcpy(outparam->record[0], share->default_values, share->rec_buff_length);
1382
memcpy(outparam->record[1], share->default_values, share->null_bytes);
1384
memcpy(outparam->record[1], share->default_values,
1385
share->rec_buff_length);
1389
if (!(field_ptr = (Field **) alloc_root(&outparam->mem_root,
1390
(uint32_t) ((share->fields+1)*
1394
outparam->field= field_ptr;
1396
record= (unsigned char*) outparam->record[0]-1; /* Fieldstart = 1 */
1398
outparam->null_flags= (unsigned char*) record+1;
1400
/* Setup copy of fields from share, but use the right alias and record */
1401
for (i= 0 ; i < share->fields; i++, field_ptr++)
1403
if (!((*field_ptr)= share->field[i]->clone(&outparam->mem_root, outparam)))
1406
(*field_ptr)= 0; // End marker
1408
if (share->found_next_number_field)
1409
outparam->found_next_number_field=
1410
outparam->field[(uint32_t) (share->found_next_number_field - share->field)];
1411
if (share->timestamp_field)
1412
outparam->timestamp_field= (Field_timestamp*) outparam->field[share->timestamp_field_offset];
1415
/* Fix key->name and key_part->field */
1416
if (share->key_parts)
1418
KEY *key_info, *key_info_end;
1419
KEY_PART_INFO *key_part;
1421
n_length= share->keys*sizeof(KEY) + share->key_parts*sizeof(KEY_PART_INFO);
1422
if (!(key_info= (KEY*) alloc_root(&outparam->mem_root, n_length)))
1424
outparam->key_info= key_info;
1425
key_part= (reinterpret_cast<KEY_PART_INFO*> (key_info+share->keys));
1427
memcpy(key_info, share->key_info, sizeof(*key_info)*share->keys);
1428
memcpy(key_part, share->key_info[0].key_part, (sizeof(*key_part) *
1431
for (key_info_end= key_info + share->keys ;
1432
key_info < key_info_end ;
1435
KEY_PART_INFO *key_part_end;
1437
key_info->table= outparam;
1438
key_info->key_part= key_part;
1440
for (key_part_end= key_part+ key_info->key_parts ;
1441
key_part < key_part_end ;
1444
Field *field= key_part->field= outparam->field[key_part->fieldnr-1];
1446
if (field->key_length() != key_part->length &&
1447
!(field->flags & BLOB_FLAG))
1450
We are using only a prefix of the column as a key:
1451
Create a new field for the key part that matches the index
1453
field= key_part->field=field->new_field(&outparam->mem_root,
1455
field->field_length= key_part->length;
1461
/* Allocate bitmaps */
1463
bitmap_size= share->column_bitmap_size;
1464
if (!(bitmaps= (unsigned char*) alloc_root(&outparam->mem_root, bitmap_size*3)))
1466
outparam->def_read_set.init((my_bitmap_map*) bitmaps, share->fields);
1467
outparam->def_write_set.init((my_bitmap_map*) (bitmaps+bitmap_size), share->fields);
1468
outparam->tmp_set.init((my_bitmap_map*) (bitmaps+bitmap_size*2), share->fields);
1469
outparam->default_column_bitmaps();
1471
/* The table struct is now initialized; Open the table */
1476
if ((ha_err= (outparam->cursor->
1477
ha_open(outparam, share->normalized_path.str,
1478
(db_stat & HA_READ_ONLY ? O_RDONLY : O_RDWR),
1479
(db_stat & HA_OPEN_TEMPORARY ? HA_OPEN_TMP_TABLE :
1480
(db_stat & HA_WAIT_IF_LOCKED) ? HA_OPEN_WAIT_IF_LOCKED :
1481
(db_stat & (HA_ABORT_IF_LOCKED | HA_GET_INFO)) ?
1482
HA_OPEN_ABORT_IF_LOCKED :
1483
HA_OPEN_IGNORE_IF_LOCKED) | ha_open_flags))))
1487
case HA_ERR_NO_SUCH_TABLE:
1489
The table did not exists in storage engine, use same error message
1490
as if the .frm cursor didn't exist
1497
Too many files opened, use same error message as if the .frm
1504
outparam->print_error(ha_err, MYF(0));
1505
error_reported= true;
1506
if (ha_err == HA_ERR_TABLE_DEF_CHANGED)
1514
#if defined(HAVE_purify)
1515
memset(bitmaps, 0, bitmap_size*3);
1518
session->status_var.opened_tables++;
1523
if (!error_reported)
1524
share->open_table_error(error, errno, 0);
1525
delete outparam->cursor;
1526
outparam->cursor= 0; // For easier error checking
1527
outparam->db_stat= 0;
1528
free_root(&outparam->mem_root, MYF(0)); // Safe to call on zeroed root
1529
free((char*) outparam->alias);
1533
bool Table::fill_item_list(List<Item> *item_list) const
1536
All Item_field's created using a direct pointer to a field
1537
are fixed in Item_field constructor.
1539
for (Field **ptr= field; *ptr; ptr++)
1541
Item_field *item= new Item_field(*ptr);
1542
if (!item || item_list->push_back(item))
1548
int Table::closefrm(bool free_share)
83
1553
error= cursor->close();
1554
free((char*) alias);
88
1558
for (Field **ptr=field ; *ptr ; ptr++)
1563
cursor= 0; /* For easier errorchecking */
1566
if (s->tmp_table == STANDARD_TABLE)
1567
TableShare::release(s);
1569
s->free_table_share();
1571
free_root(&mem_root, MYF(0));
106
mem_root.free_root(MYF(0));
110
1577
void Table::resetTable(Session *session,
111
1578
TableShare *share,
112
1579
uint32_t db_stat_arg)
1376
3016
session->mem_root= mem_root_save;
3017
table->free_tmp_table(session);
1382
3021
/****************************************************************************/
1384
void Table::column_bitmaps_set(boost::dynamic_bitset<>& read_set_arg,
1385
boost::dynamic_bitset<>& write_set_arg)
1387
read_set= &read_set_arg;
1388
write_set= &write_set_arg;
1392
const boost::dynamic_bitset<> Table::use_all_columns(boost::dynamic_bitset<>& in_map)
1394
const boost::dynamic_bitset<> old= in_map;
1395
in_map= getShare()->all_set;
3024
Create a reduced Table object with properly set up Field list from a
3025
list of field definitions.
3027
The created table doesn't have a table Cursor associated with
3028
it, has no keys, no group/distinct, no copy_funcs array.
3029
The sole purpose of this Table object is to use the power of Field
3030
class to read/write data to/from table->record[0]. Then one can store
3031
the record in any container (RB tree, hash, etc).
3032
The table is created in Session mem_root, so are the table's fields.
3033
Consequently, if you don't BLOB fields, you don't need to free it.
3035
@param session connection handle
3036
@param field_list list of column definitions
3039
0 if out of memory, Table object in case of success
3042
Table *create_virtual_tmp_table(Session *session, List<CreateField> &field_list)
3044
uint32_t field_count= field_list.elements;
3045
uint32_t blob_count= 0;
3047
CreateField *cdef; /* column definition */
3048
uint32_t record_length= 0;
3049
uint32_t null_count= 0; /* number of columns which may be null */
3050
uint32_t null_pack_length; /* NULL representation array length */
3051
uint32_t *blob_field;
3052
unsigned char *bitmaps;
3056
if (!multi_alloc_root(session->mem_root,
3057
&table, sizeof(*table),
3058
&share, sizeof(*share),
3059
&field, (field_count + 1) * sizeof(Field*),
3060
&blob_field, (field_count+1) *sizeof(uint32_t),
3061
&bitmaps, bitmap_buffer_size(field_count)*2,
3065
memset(table, 0, sizeof(*table));
3066
memset(share, 0, sizeof(*share));
3067
table->field= field;
3069
share->blob_field= blob_field;
3070
share->fields= field_count;
3071
share->blob_ptr_size= portable_sizeof_char_ptr;
3072
table->setup_tmp_table_column_bitmaps(bitmaps);
3074
/* Create all fields and calculate the total length of record */
3075
List_iterator_fast<CreateField> it(field_list);
3076
while ((cdef= it++))
3078
*field= make_field(share,
3082
(cdef->flags & NOT_NULL_FLAG) ? false : true,
3083
(unsigned char *) ((cdef->flags & NOT_NULL_FLAG) ? 0 : ""),
3084
(cdef->flags & NOT_NULL_FLAG) ? 0 : 1,
3093
(*field)->init(table);
3094
record_length+= (*field)->pack_length();
3095
if (! ((*field)->flags & NOT_NULL_FLAG))
3098
if ((*field)->flags & BLOB_FLAG)
3099
share->blob_field[blob_count++]= (uint32_t) (field - table->field);
3103
*field= NULL; /* mark the end of the list */
3104
share->blob_field[blob_count]= 0; /* mark the end of the list */
3105
share->blob_fields= blob_count;
3107
null_pack_length= (null_count + 7)/8;
3108
share->reclength= record_length + null_pack_length;
3109
share->rec_buff_length= ALIGN_SIZE(share->reclength + 1);
3110
table->record[0]= (unsigned char*) session->alloc(share->rec_buff_length);
3111
if (!table->record[0])
3114
if (null_pack_length)
3116
table->null_flags= (unsigned char*) table->record[0];
3117
share->null_fields= null_count;
3118
share->null_bytes= null_pack_length;
3121
table->in_use= session; /* field->reset() may access table->in_use */
3123
/* Set up field pointers */
3124
unsigned char *null_pos= table->record[0];
3125
unsigned char *field_pos= null_pos + share->null_bytes;
3126
uint32_t null_bit= 1;
3128
for (field= table->field; *field; ++field)
3130
Field *cur_field= *field;
3131
if ((cur_field->flags & NOT_NULL_FLAG))
3132
cur_field->move_field(field_pos);
3135
cur_field->move_field(field_pos, (unsigned char*) null_pos, null_bit);
3137
if (null_bit == (1 << 8))
3145
field_pos+= cur_field->pack_length();
3150
for (field= table->field; *field; ++field)
3151
delete *field; /* just invokes field destructor */
3155
bool Table::open_tmp_table()
3158
if ((error=cursor->ha_open(this, s->table_name.str,O_RDWR,
3159
HA_OPEN_TMP_TABLE | HA_OPEN_INTERNAL_TABLE)))
3161
print_error(error, MYF(0));
3165
(void) cursor->extra(HA_EXTRA_QUICK); /* Faster */
3171
Create MyISAM temporary table
3174
create_myisam_tmp_table()
3175
keyinfo Description of the index (there is always one index)
3176
start_recinfo MyISAM's column descriptions
3177
recinfo INOUT End of MyISAM's column descriptions
3181
Create a MyISAM temporary table according to passed description. The is
3182
assumed to have one unique index or constraint.
3184
The passed array or MI_COLUMNDEF structures must have this form:
3186
1. 1-byte column (afaiu for 'deleted' flag) (note maybe not 1-byte
3187
when there are many nullable columns)
3189
3. One free MI_COLUMNDEF element (*recinfo points here)
3191
This function may use the free element to create hash column for unique
3199
bool Table::create_myisam_tmp_table(KEY *keyinfo,
3200
MI_COLUMNDEF *start_recinfo,
3201
MI_COLUMNDEF **recinfo,
3206
MI_UNIQUEDEF uniquedef;
3207
TableShare *share= s;
3210
{ // Get keys for ni_create
3211
bool using_unique_constraint= false;
3212
HA_KEYSEG *seg= (HA_KEYSEG*) alloc_root(&this->mem_root,
3213
sizeof(*seg) * keyinfo->key_parts);
3217
memset(seg, 0, sizeof(*seg) * keyinfo->key_parts);
3218
if (keyinfo->key_length >= cursor->getEngine()->max_key_length() ||
3219
keyinfo->key_parts > cursor->getEngine()->max_key_parts() ||
3222
/* Can't create a key; Make a unique constraint instead of a key */
3225
using_unique_constraint= true;
3226
memset(&uniquedef, 0, sizeof(uniquedef));
3227
uniquedef.keysegs=keyinfo->key_parts;
3229
uniquedef.null_are_equal=1;
3231
/* Create extra column for hash value */
3232
memset(*recinfo, 0, sizeof(**recinfo));
3233
(*recinfo)->type= FIELD_CHECK;
3234
(*recinfo)->length=MI_UNIQUE_HASH_LENGTH;
3236
share->reclength+=MI_UNIQUE_HASH_LENGTH;
3240
/* Create an unique key */
3241
memset(&keydef, 0, sizeof(keydef));
3242
keydef.flag=HA_NOSAME | HA_BINARY_PACK_KEY | HA_PACK_KEY;
3243
keydef.keysegs= keyinfo->key_parts;
3246
for (uint32_t i= 0; i < keyinfo->key_parts ; i++,seg++)
3248
Field *key_field=keyinfo->key_part[i].field;
3250
seg->language= key_field->charset()->number;
3251
seg->length= keyinfo->key_part[i].length;
3252
seg->start= keyinfo->key_part[i].offset;
3253
if (key_field->flags & BLOB_FLAG)
3255
seg->type= ((keyinfo->key_part[i].key_type & 1 /* binary */) ?
3256
HA_KEYTYPE_VARBINARY2 : HA_KEYTYPE_VARTEXT2);
3257
seg->bit_start= (uint8_t)(key_field->pack_length()
3258
- share->blob_ptr_size);
3259
seg->flag= HA_BLOB_PART;
3260
seg->length= 0; // Whole blob in unique constraint
3264
seg->type= keyinfo->key_part[i].type;
3266
if (!(key_field->flags & NOT_NULL_FLAG))
3268
seg->null_bit= key_field->null_bit;
3269
seg->null_pos= (uint32_t) (key_field->null_ptr - (unsigned char*) record[0]);
3271
We are using a GROUP BY on something that contains NULL
3272
In this case we have to tell MyISAM that two NULL should
3273
on INSERT be regarded at the same value
3275
if (! using_unique_constraint)
3276
keydef.flag|= HA_NULL_ARE_EQUAL;
3280
MI_CREATE_INFO create_info;
3281
memset(&create_info, 0, sizeof(create_info));
3283
if ((options & (OPTION_BIG_TABLES | SELECT_SMALL_RESULT)) ==
3285
create_info.data_file_length= ~(uint64_t) 0;
3287
if ((error=mi_create(share->table_name.str, share->keys, &keydef,
3288
(uint32_t) (*recinfo-start_recinfo),
3290
share->uniques, &uniquedef,
3292
HA_CREATE_TMP_TABLE)))
3294
print_error(error, MYF(0));
3298
status_var_increment(in_use->status_var.created_tmp_disk_tables);
3299
share->db_record_offset= 1;
3306
void Table::free_tmp_table(Session *session)
3308
memory::Root own_root= mem_root;
3309
const char *save_proc_info;
3311
save_proc_info=session->get_proc_info();
3312
session->set_proc_info("removing tmp table");
3314
// Release latches since this can take a long time
3315
plugin::TransactionalStorageEngine::releaseTemporaryLatches(session);
3320
cursor->closeMarkForDelete(s->table_name.str);
3322
s->db_type()->doDropTable(*session, s->table_name.str);
3328
for (Field **ptr= field ; *ptr ; ptr++)
3332
free_root(&own_root, MYF(0)); /* the table is allocated in its own root */
3333
session->set_proc_info(save_proc_info);
3337
If a HEAP table gets full, create a MyISAM table and copy all rows
3341
bool create_myisam_from_heap(Session *session, Table *table,
3342
MI_COLUMNDEF *start_recinfo,
3343
MI_COLUMNDEF **recinfo,
3344
int error, bool ignore_last_dupp_key_error)
3348
const char *save_proc_info;
3351
if (table->s->db_type() != heap_engine ||
3352
error != HA_ERR_RECORD_FILE_FULL)
3354
table->print_error(error, MYF(0));
3358
// Release latches since this can take a long time
3359
plugin::TransactionalStorageEngine::releaseTemporaryLatches(session);
3363
new_table.s= &share;
3364
new_table.s->storage_engine= myisam_engine;
3365
if (not (new_table.cursor= new_table.s->db_type()->getCursor(share, &new_table.mem_root)))
3366
return true; // End of memory
3368
save_proc_info=session->get_proc_info();
3369
session->set_proc_info("converting HEAP to MyISAM");
3371
if (new_table.create_myisam_tmp_table(table->key_info, start_recinfo,
3372
recinfo, session->lex->select_lex.options |
3375
if (new_table.open_tmp_table())
3377
if (table->cursor->indexes_are_disabled())
3378
new_table.cursor->ha_disable_indexes(HA_KEY_SWITCH_ALL);
3379
table->cursor->ha_index_or_rnd_end();
3380
table->cursor->ha_rnd_init(1);
3383
new_table.cursor->extra(HA_EXTRA_NO_ROWS);
3384
new_table.no_rows=1;
3387
/* HA_EXTRA_WRITE_CACHE can stay until close, no need to disable it */
3388
new_table.cursor->extra(HA_EXTRA_WRITE_CACHE);
3391
copy all old rows from heap table to MyISAM table
3392
This is the only code that uses record[1] to read/write but this
3393
is safe as this is a temporary MyISAM table without timestamp/autoincrement.
3395
while (!table->cursor->rnd_next(new_table.record[1]))
3397
write_err= new_table.cursor->ha_write_row(new_table.record[1]);
3401
/* copy row that filled HEAP table */
3402
if ((write_err=new_table.cursor->ha_write_row(table->record[0])))
3404
if (new_table.cursor->is_fatal_error(write_err, HA_CHECK_DUP) ||
3405
!ignore_last_dupp_key_error)
3409
/* remove heap table and change to use myisam table */
3410
(void) table->cursor->ha_rnd_end();
3411
(void) table->cursor->close(); // This deletes the table !
3412
delete table->cursor;
3413
table->cursor= NULL;
3414
new_table.s= table->s; // Keep old share
3418
table->cursor->change_table_ptr(table, table->s);
3419
table->use_all_columns();
3422
const char *new_proc_info=
3423
(!strcmp(save_proc_info,"Copying to tmp table") ?
3424
"Copying to tmp table on disk" : save_proc_info);
3425
session->set_proc_info(new_proc_info);
3430
table->print_error(write_err, MYF(0));
3431
(void) table->cursor->ha_rnd_end();
3432
(void) new_table.cursor->close();
3434
new_table.s->db_type()->doDropTable(*session, new_table.s->table_name.str);
3436
delete new_table.cursor;
3437
session->set_proc_info(save_proc_info);
3438
table->mem_root= new_table.mem_root;
3442
my_bitmap_map *Table::use_all_columns(MyBitmap *bitmap)
3444
my_bitmap_map *old= bitmap->getBitmap();
3445
bitmap->setBitmap(s->all_set.getBitmap());
1399
void Table::restore_column_map(const boost::dynamic_bitset<>& old)
3449
void Table::restore_column_map(my_bitmap_map *old)
1401
for (boost::dynamic_bitset<>::size_type i= 0; i < old.size(); i++)
3451
read_set->setBitmap(old);
1414
3454
uint32_t Table::find_shortest_key(const key_map *usable_keys)