1
/******************************************************
4
(c) 2005-2008 Innobase Oy
5
*******************************************************/
7
#include <mysql_priv.h>
8
#include <mysqld_error.h>
12
#include "row0merge.h"
16
#include "ha_prototypes.h"
17
#include "handler0alter.h"
20
#include "ha_innodb.h"
22
/*****************************************************************
23
Copies an InnoDB column to a MySQL field. This function is
24
adapted from row_sel_field_store_in_mysql_format(). */
27
innobase_col_to_mysql(
28
/*==================*/
29
const dict_col_t* col, /* in: InnoDB column */
30
const uchar* data, /* in: InnoDB column data */
31
ulint len, /* in: length of data, in bytes */
32
Field* field) /* in/out: MySQL field */
35
uchar* dest = field->ptr;
36
ulint flen = field->pack_length();
42
/* Convert integer data from Innobase to little-endian
43
format, sign bit restored to normal */
45
for (ptr = dest + len; ptr != dest; ) {
49
if (!(field->flags & UNSIGNED_FLAG)) {
50
((byte*) dest)[len - 1] ^= 0x80;
60
if (field->type() == MYSQL_TYPE_VARCHAR) {
61
/* This is a >= 5.0.3 type true VARCHAR. Store the
62
length of the data to the first byte or the first
65
dest = row_mysql_store_true_var_len(
66
dest, len, flen - field->key_length());
69
/* Copy the actual data */
70
memcpy(dest, data, len);
74
/* Store a pointer to the BLOB buffer to dest: the BLOB was
75
already copied to the buffer in row_sel_store_mysql_rec */
77
row_mysql_store_blob_ref(dest, flen, data, len);
83
ut_ad(col->mbmaxlen >= col->mbminlen);
84
ut_ad(col->mbmaxlen > col->mbminlen || flen == len);
85
memcpy(dest, data, len);
91
/* These column types should never be shipped to MySQL. */
99
/* Above are the valid column types for MySQL data. */
101
#else /* UNIV_DEBUG */
103
#endif /* UNIV_DEBUG */
104
memcpy(dest, data, len);
108
/*****************************************************************
109
Copies an InnoDB record to table->record[0]. */
110
extern "C" UNIV_INTERN
112
innobase_rec_to_mysql(
113
/*==================*/
114
TABLE* table, /* in/out: MySQL table */
115
const rec_t* rec, /* in: record */
116
const dict_index_t* index, /* in: index */
117
const ulint* offsets) /* in: rec_get_offsets(
120
uint n_fields = table->s->fields;
123
ut_ad(n_fields == dict_table_get_n_user_cols(index->table));
125
for (i = 0; i < n_fields; i++) {
126
Field* field = table->field[i];
133
ipos = dict_index_get_nth_col_pos(index, i);
135
if (UNIV_UNLIKELY(ipos == ULINT_UNDEFINED)) {
141
ifield = rec_get_nth_field(rec, offsets, ipos, &ilen);
143
/* Assign the NULL flag */
144
if (ilen == UNIV_SQL_NULL) {
145
ut_ad(field->real_maybe_null());
149
field->set_notnull();
151
innobase_col_to_mysql(
153
dict_index_get_nth_field(index, ipos)),
154
ifield, ilen, field);
158
/*****************************************************************
159
Resets table->record[0]. */
160
extern "C" UNIV_INTERN
164
TABLE* table) /* in/out: MySQL table */
166
uint n_fields = table->s->fields;
169
for (i = 0; i < n_fields; i++) {
170
table->field[i]->set_default();
174
/**********************************************************************
175
Removes the filename encoding of a database and table name. */
178
innobase_convert_tablename(
179
/*=======================*/
180
char* s) /* in: identifier; out: decoded identifier */
184
char* slash = strchr(s, '/');
188
/* Temporarily replace the '/' with NUL. */
190
/* Convert the database name. */
191
strconvert(&my_charset_filename, s, system_charset_info,
192
s, slash - s + 1, &errors);
196
/* Append a '.' after the database name. */
199
/* Convert the table name. */
200
strconvert(&my_charset_filename, slash, system_charset_info,
201
t, slash - t + strlen(slash), &errors);
203
strconvert(&my_charset_filename, s,
204
system_charset_info, s, strlen(s), &errors);
208
/***********************************************************************
209
This function checks that index keys are sensible. */
212
innobase_check_index_keys(
213
/*======================*/
214
/* out: 0 or error number */
215
const KEY* key_info, /* in: Indexes to be created */
216
ulint num_of_keys) /* in: Number of indexes to
224
for (key_num = 0; key_num < num_of_keys; key_num++) {
225
const KEY& key = key_info[key_num];
227
/* Check that the same index name does not appear
228
twice in indexes to be created. */
230
for (ulint i = 0; i < key_num; i++) {
231
const KEY& key2 = key_info[i];
233
if (0 == strcmp(key.name, key2.name)) {
234
sql_print_error("InnoDB: key name `%s` appears"
235
" twice in CREATE INDEX\n",
238
return(ER_WRONG_NAME_FOR_INDEX);
242
/* Check that MySQL does not try to create a column
243
prefix index field on an inappropriate data type and
244
that the same colum does not appear twice in the index. */
246
for (ulint i = 0; i < key.key_parts; i++) {
247
const KEY_PART_INFO& key_part1
253
switch (get_innobase_type_from_mysql_type(
254
&is_unsigned, field)) {
261
if (field->type() == MYSQL_TYPE_VARCHAR) {
263
>= field->pack_length()
264
- ((Field_varstring*) field)
270
>= field->pack_length()) {
275
sql_print_error("InnoDB: MySQL is trying to"
276
" create a column prefix"
278
" inappropriate data type."
283
return(ER_WRONG_KEY_COLUMN);
286
for (ulint j = 0; j < i; j++) {
287
const KEY_PART_INFO& key_part2
290
if (strcmp(key_part1.field->field_name,
291
key_part2.field->field_name)) {
295
sql_print_error("InnoDB: column `%s`"
296
" is not allowed to occur"
297
" twice in index `%s`.\n",
298
key_part1.field->field_name,
300
return(ER_WRONG_KEY_COLUMN);
308
/***********************************************************************
309
Create index field definition for key part */
312
innobase_create_index_field_def(
313
/*============================*/
314
KEY_PART_INFO* key_part, /* in: MySQL key definition */
315
mem_heap_t* heap, /* in: memory heap */
316
merge_index_field_t* index_field) /* out: index field
317
definition for key_part */
323
DBUG_ENTER("innobase_create_index_field_def");
328
field = key_part->field;
331
col_type = get_innobase_type_from_mysql_type(&is_unsigned, field);
333
if (DATA_BLOB == col_type
334
|| (key_part->length < field->pack_length()
335
&& field->type() != MYSQL_TYPE_VARCHAR)
336
|| (field->type() == MYSQL_TYPE_VARCHAR
337
&& key_part->length < field->pack_length()
338
- ((Field_varstring*)field)->length_bytes)) {
340
index_field->prefix_len = key_part->length;
342
index_field->prefix_len = 0;
345
index_field->field_name = mem_heap_strdup(heap, field->field_name);
350
/***********************************************************************
351
Create index definition for key */
354
innobase_create_index_def(
355
/*======================*/
356
KEY* key, /* in: key definition */
357
bool new_primary, /* in: TRUE=generating
360
bool key_primary, /* in: TRUE if this key
362
merge_index_def_t* index, /* out: index definition */
363
mem_heap_t* heap) /* in: heap where memory
368
ulint n_fields = key->key_parts;
371
DBUG_ENTER("innobase_create_index_def");
373
index->fields = (merge_index_field_t*) mem_heap_alloc(
374
heap, n_fields * sizeof *index->fields);
377
index->n_fields = n_fields;
378
len = strlen(key->name) + 1;
379
index->name = index_name = (char*) mem_heap_alloc(heap,
382
if (UNIV_LIKELY(!new_primary)) {
383
*index_name++ = TEMP_INDEX_PREFIX;
386
memcpy(index_name, key->name, len);
388
if (key->flags & HA_NOSAME) {
389
index->ind_type |= DICT_UNIQUE;
393
index->ind_type |= DICT_CLUSTERED;
396
for (i = 0; i < n_fields; i++) {
397
innobase_create_index_field_def(&key->key_part[i], heap,
404
/***********************************************************************
405
Copy index field definition */
408
innobase_copy_index_field_def(
409
/*==========================*/
410
const dict_field_t* field, /* in: definition to copy */
411
merge_index_field_t* index_field) /* out: copied definition */
413
DBUG_ENTER("innobase_copy_index_field_def");
414
DBUG_ASSERT(field != NULL);
415
DBUG_ASSERT(index_field != NULL);
417
index_field->field_name = field->name;
418
index_field->prefix_len = field->prefix_len;
423
/***********************************************************************
424
Copy index definition for the index */
427
innobase_copy_index_def(
428
/*====================*/
429
const dict_index_t* index, /* in: index definition to copy */
430
merge_index_def_t* new_index,/* out: Index definition */
431
mem_heap_t* heap) /* in: heap where allocated */
436
DBUG_ENTER("innobase_copy_index_def");
438
/* Note that we take only those fields that user defined to be
439
in the index. In the internal representation more colums were
440
added and those colums are not copied .*/
442
n_fields = index->n_user_defined_cols;
444
new_index->fields = (merge_index_field_t*) mem_heap_alloc(
445
heap, n_fields * sizeof *new_index->fields);
447
/* When adding a PRIMARY KEY, we may convert a previous
448
clustered index to a secondary index (UNIQUE NOT NULL). */
449
new_index->ind_type = index->type & ~DICT_CLUSTERED;
450
new_index->n_fields = n_fields;
451
new_index->name = index->name;
453
for (i = 0; i < n_fields; i++) {
454
innobase_copy_index_field_def(&index->fields[i],
455
&new_index->fields[i]);
461
/***********************************************************************
462
Create an index table where indexes are ordered as follows:
464
IF a new primary key is defined for the table THEN
467
2) Original secondary indexes
468
3) New secondary indexes
472
1) All new indexes in the order they arrive from MySQL
479
innobase_create_key_def(
480
/*====================*/
481
/* out: key definitions or NULL */
482
trx_t* trx, /* in: trx */
483
const dict_table_t*table, /* in: table definition */
484
mem_heap_t* heap, /* in: heap where space for key
485
definitions are allocated */
486
KEY* key_info, /* in: Indexes to be created */
487
ulint& n_keys) /* in/out: Number of indexes to
491
merge_index_def_t* indexdef;
492
merge_index_def_t* indexdefs;
495
DBUG_ENTER("innobase_create_key_def");
497
indexdef = indexdefs = (merge_index_def_t*)
498
mem_heap_alloc(heap, sizeof *indexdef
499
* (n_keys + UT_LIST_GET_LEN(table->indexes)));
501
/* If there is a primary key, it is always the first index
502
defined for the table. */
504
new_primary = !my_strcasecmp(system_charset_info,
505
key_info->name, "PRIMARY");
507
/* If there is a UNIQUE INDEX consisting entirely of NOT NULL
508
columns, MySQL will treat it as a PRIMARY KEY unless the
509
table already has one. */
511
if (!new_primary && (key_info->flags & HA_NOSAME)
512
&& row_table_got_default_clust_index(table)) {
513
uint key_part = key_info->key_parts;
518
if (key_info->key_part[key_part].key_type
519
& FIELDFLAG_MAYBE_NULL) {
527
const dict_index_t* index;
529
/* Create the PRIMARY key index definition */
530
innobase_create_index_def(&key_info[i++], TRUE, TRUE,
533
row_mysql_lock_data_dictionary(trx);
535
index = dict_table_get_first_index(table);
537
/* Copy the index definitions of the old table. Skip
538
the old clustered index if it is a generated clustered
539
index or a PRIMARY KEY. If the clustered index is a
540
UNIQUE INDEX, it must be converted to a secondary index. */
542
if (dict_index_get_nth_col(index, 0)->mtype == DATA_SYS
543
|| !my_strcasecmp(system_charset_info,
544
index->name, "PRIMARY")) {
545
index = dict_table_get_next_index(index);
549
innobase_copy_index_def(index, indexdef++, heap);
550
index = dict_table_get_next_index(index);
553
row_mysql_unlock_data_dictionary(trx);
556
/* Create definitions for added secondary indexes. */
559
innobase_create_index_def(&key_info[i++], new_primary, FALSE,
563
n_keys = indexdef - indexdefs;
565
DBUG_RETURN(indexdefs);
568
/***********************************************************************
569
Create a temporary tablename using query id, thread id, and id */
572
innobase_create_temporary_tablename(
573
/*================================*/
574
/* out: temporary tablename */
575
mem_heap_t* heap, /* in: memory heap */
576
char id, /* in: identifier [0-9a-zA-Z] */
577
const char* table_name) /* in: table name */
581
static const char suffix[] = "@0023 "; /* "# " */
583
len = strlen(table_name);
585
name = (char*) mem_heap_alloc(heap, len + sizeof suffix);
586
memcpy(name, table_name, len);
587
memcpy(name + len, suffix, sizeof suffix);
588
name[len + (sizeof suffix - 2)] = id;
593
/***********************************************************************
597
ha_innobase::add_index(
598
/*===================*/
599
/* out: 0 or error number */
600
TABLE* table, /* in: Table where indexes are created */
601
KEY* key_info, /* in: Indexes to be created */
602
uint num_of_keys) /* in: Number of indexes to be created */
604
dict_index_t** index; /* Index to be created */
605
dict_table_t* innodb_table; /* InnoDB table in dictionary */
606
dict_table_t* indexed_table; /* Table where indexes are created */
607
merge_index_def_t* index_defs; /* Index definitions */
608
mem_heap_t* heap; /* Heap for index definitions */
609
trx_t* trx; /* Transaction */
611
ulint num_created = 0;
612
ibool dict_locked = FALSE;
616
DBUG_ENTER("ha_innobase::add_index");
621
if (srv_created_new_raw || srv_force_recovery) {
622
DBUG_RETURN(HA_ERR_WRONG_COMMAND);
627
heap = mem_heap_create(1024);
629
/* In case MySQL calls this in the middle of a SELECT query, release
630
possible adaptive hash latch to avoid deadlocks of threads. */
631
trx_search_latch_release_if_reserved(prebuilt->trx);
633
/* Create a background transaction for the operations on
634
the data dictionary tables. */
635
trx = trx_allocate_for_mysql();
636
trx_start_if_not_started(trx);
638
trans_register_ha(user_thd, FALSE, ht);
639
prebuilt->trx->active_trans = 1;
641
trx->mysql_thd = user_thd;
642
trx->mysql_query_str = thd_query(user_thd);
644
innodb_table = indexed_table
645
= dict_table_get(prebuilt->table->name, FALSE);
647
/* Check that index keys are sensible */
649
error = innobase_check_index_keys(key_info, num_of_keys);
651
if (UNIV_UNLIKELY(error)) {
654
trx_general_rollback_for_mysql(trx, FALSE, NULL);
655
trx_free_for_mysql(trx);
656
trx_commit_for_mysql(prebuilt->trx);
660
/* Create table containing all indexes to be built in this
661
alter table add index so that they are in the correct order
664
num_of_idx = num_of_keys;
666
index_defs = innobase_create_key_def(
667
trx, innodb_table, heap, key_info, num_of_idx);
669
new_primary = DICT_CLUSTERED & index_defs[0].ind_type;
671
/* Allocate memory for dictionary index definitions */
673
index = (dict_index_t**) mem_heap_alloc(
674
heap, num_of_idx * sizeof *index);
676
/* Flag this transaction as a dictionary operation, so that
677
the data dictionary will be locked in crash recovery. */
678
trx_set_dict_operation(trx, TRX_DICT_OP_INDEX);
680
/* Acquire a lock on the table before creating any indexes. */
681
error = row_merge_lock_table(prebuilt->trx, innodb_table,
682
new_primary ? LOCK_X : LOCK_S);
684
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
689
/* Latch the InnoDB data dictionary exclusively so that no deadlocks
690
or lock waits can happen in it during an index create operation. */
692
row_mysql_lock_data_dictionary(trx);
695
/* If a new primary key is defined for the table we need
696
to drop the original table and rebuild all indexes. */
698
if (UNIV_UNLIKELY(new_primary)) {
699
/* This transaction should be the only one
700
operating on the table. */
701
ut_a(innodb_table->n_mysql_handles_opened == 1);
703
char* new_table_name = innobase_create_temporary_tablename(
704
heap, '1', innodb_table->name);
706
/* Clone the table. */
707
trx_set_dict_operation(trx, TRX_DICT_OP_TABLE);
708
indexed_table = row_merge_create_temporary_table(
709
new_table_name, index_defs, innodb_table, trx);
711
if (!indexed_table) {
713
switch (trx->error_state) {
714
case DB_TABLESPACE_ALREADY_EXISTS:
715
case DB_DUPLICATE_KEY:
716
innobase_convert_tablename(new_table_name);
717
my_error(HA_ERR_TABLE_EXIST, MYF(0),
719
error = HA_ERR_TABLE_EXIST;
722
error = convert_error_code_to_mysql(
723
trx->error_state, innodb_table->flags,
727
row_mysql_unlock_data_dictionary(trx);
731
trx->table_id = indexed_table->id;
734
/* Create the indexes in SYS_INDEXES and load into dictionary. */
736
for (ulint i = 0; i < num_of_idx; i++) {
738
index[i] = row_merge_create_index(trx, indexed_table,
742
error = trx->error_state;
749
ut_ad(error == DB_SUCCESS);
751
/* Commit the data dictionary transaction in order to release
752
the table locks on the system tables. Unfortunately, this
753
means that if MySQL crashes while creating a new primary key
754
inside row_merge_build_indexes(), indexed_table will not be
755
dropped on crash recovery. Thus, it will become orphaned. */
756
trx_commit_for_mysql(trx);
758
row_mysql_unlock_data_dictionary(trx);
761
ut_a(trx->n_active_thrs == 0);
762
ut_a(UT_LIST_GET_LEN(trx->signals) == 0);
764
if (UNIV_UNLIKELY(new_primary)) {
765
/* A primary key is to be built. Acquire an exclusive
766
table lock also on the table that is being created. */
767
ut_ad(indexed_table != innodb_table);
769
error = row_merge_lock_table(prebuilt->trx, indexed_table,
772
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
778
/* Read the clustered index of the table and build indexes
779
based on this information using temporary files and merge sort. */
780
error = row_merge_build_indexes(prebuilt->trx,
781
innodb_table, indexed_table,
782
index, num_of_idx, table);
786
/* TODO: At the moment we can't handle the following statement
787
in our debugging code below:
789
alter table t drop index b, add index (b);
791
The fix will have to parse the SQL and note that the index
792
being added has the same name as the the one being dropped and
793
ignore that in the dup index check.*/
794
//dict_table_check_for_dup_indexes(prebuilt->table);
797
/* After an error, remove all those index definitions from the
798
dictionary which were defined. */
801
const char* old_name;
807
error = row_merge_rename_indexes(trx, indexed_table);
809
if (error != DB_SUCCESS) {
810
row_merge_drop_indexes(trx, indexed_table,
817
/* If a new primary key was defined for the table and
818
there was no error at this point, we can now rename
819
the old table as a temporary table, rename the new
820
temporary table as the old table and drop the old table. */
821
old_name = innodb_table->name;
822
tmp_name = innobase_create_temporary_tablename(heap, '2',
825
row_mysql_lock_data_dictionary(trx);
828
error = row_merge_rename_tables(innodb_table, indexed_table,
831
if (error != DB_SUCCESS) {
833
row_merge_drop_table(trx, indexed_table);
836
case DB_TABLESPACE_ALREADY_EXISTS:
837
case DB_DUPLICATE_KEY:
838
innobase_convert_tablename(tmp_name);
839
my_error(HA_ERR_TABLE_EXIST, MYF(0), tmp_name);
840
error = HA_ERR_TABLE_EXIST;
848
trx_commit_for_mysql(prebuilt->trx);
849
row_prebuilt_free(prebuilt, TRUE);
850
prebuilt = row_create_prebuilt(indexed_table);
852
indexed_table->n_mysql_handles_opened++;
854
error = row_merge_drop_table(trx, innodb_table);
857
case DB_TOO_BIG_RECORD:
858
my_error(HA_ERR_TO_BIG_ROW, MYF(0));
860
case DB_PRIMARY_KEY_IS_NULL:
861
my_error(ER_PRIMARY_CANT_HAVE_NULL, MYF(0));
863
case DB_DUPLICATE_KEY:
865
prebuilt->trx->error_info = NULL;
869
row_merge_drop_table(trx, indexed_table);
871
row_merge_drop_indexes(trx, indexed_table,
876
error = convert_error_code_to_mysql(error,
882
trx_commit_for_mysql(trx);
884
trx_commit_for_mysql(prebuilt->trx);
888
row_mysql_unlock_data_dictionary(trx);
891
trx_free_for_mysql(trx);
893
/* There might be work for utility threads.*/
894
srv_active_wake_master_thread();
899
/***********************************************************************
900
Prepare to drop some indexes of a table. */
903
ha_innobase::prepare_drop_index(
904
/*============================*/
905
/* out: 0 or error number */
906
TABLE* table, /* in: Table where indexes are dropped */
907
uint* key_num, /* in: Key nums to be dropped */
908
uint num_of_keys) /* in: Number of keys to be dropped */
914
DBUG_ENTER("ha_innobase::prepare_drop_index");
918
if (srv_created_new_raw || srv_force_recovery) {
919
DBUG_RETURN(HA_ERR_WRONG_COMMAND);
924
trx_search_latch_release_if_reserved(prebuilt->trx);
927
/* Test and mark all the indexes to be dropped */
929
row_mysql_lock_data_dictionary(trx);
931
/* Check that none of the indexes have previously been flagged
934
const dict_index_t* index
935
= dict_table_get_first_index(prebuilt->table);
937
ut_a(!index->to_be_dropped);
938
index = dict_table_get_next_index(index);
942
for (n_key = 0; n_key < num_of_keys; n_key++) {
946
key = table->key_info + key_num[n_key];
947
index = dict_table_get_index_on_name_and_min_id(
948
prebuilt->table, key->name);
951
sql_print_error("InnoDB could not find key n:o %u "
952
"with name %s for table %s",
954
key ? key->name : "NULL",
955
prebuilt->table->name);
957
err = HA_ERR_KEY_NOT_FOUND;
961
/* Refuse to drop the clustered index. It would be
962
better to automatically generate a clustered index,
963
but mysql_alter_table() will call this method only
964
after ha_innobase::add_index(). */
966
if (dict_index_is_clust(index)) {
967
my_error(ER_REQUIRES_PRIMARY_KEY, MYF(0));
972
index->to_be_dropped = TRUE;
975
/* If FOREIGN_KEY_CHECK = 1 you may not drop an index defined
976
for a foreign key constraint because InnoDB requires that both
977
tables contain indexes for the constraint. Note that CREATE
978
INDEX id ON table does a CREATE INDEX and DROP INDEX, and we
979
can ignore here foreign keys because a new index for the
980
foreign key has already been created.
982
We check for the foreign key constraints after marking the
983
candidate indexes for deletion, because when we check for an
984
equivalent foreign index we don't want to select an index that
987
if (trx->check_foreigns
988
&& thd_sql_command(user_thd) != SQLCOM_CREATE_INDEX) {
990
= dict_table_get_first_index(prebuilt->table);
993
dict_foreign_t* foreign;
995
if (!index->to_be_dropped) {
1000
/* Check if the index is referenced. */
1001
foreign = dict_table_get_referenced_constraint(
1002
prebuilt->table, index);
1006
trx_set_detailed_error(
1008
"Index needed in foreign key "
1011
trx->error_info = index;
1013
err = HA_ERR_DROP_INDEX_FK;
1016
/* Check if this index references some
1018
foreign = dict_table_get_foreign_constraint(
1019
prebuilt->table, index);
1022
ut_a(foreign->foreign_index == index);
1024
/* Search for an equivalent index that
1025
the foreign key contraint could use
1026
if this index were to be deleted. */
1027
if (!dict_table_find_equivalent_index(
1029
foreign->foreign_index)) {
1037
index = dict_table_get_next_index(index);
1043
/* Undo our changes since there was some sort of error. */
1045
= dict_table_get_first_index(prebuilt->table);
1048
index->to_be_dropped = FALSE;
1049
index = dict_table_get_next_index(index);
1053
row_mysql_unlock_data_dictionary(trx);
1058
/***********************************************************************
1059
Drop the indexes that were passed to a successful prepare_drop_index(). */
1062
ha_innobase::final_drop_index(
1063
/*==========================*/
1064
/* out: 0 or error number */
1065
TABLE* table) /* in: Table where indexes are dropped */
1067
dict_index_t* index; /* Index to be dropped */
1068
trx_t* trx; /* Transaction */
1071
DBUG_ENTER("ha_innobase::final_drop_index");
1074
if (srv_created_new_raw || srv_force_recovery) {
1075
DBUG_RETURN(HA_ERR_WRONG_COMMAND);
1080
trx_search_latch_release_if_reserved(prebuilt->trx);
1082
/* Create a background transaction for the operations on
1083
the data dictionary tables. */
1084
trx = trx_allocate_for_mysql();
1085
trx_start_if_not_started(trx);
1087
trans_register_ha(user_thd, FALSE, ht);
1088
prebuilt->trx->active_trans = 1;
1090
trx->mysql_thd = user_thd;
1091
trx->mysql_query_str = thd_query(user_thd);
1093
/* Flag this transaction as a dictionary operation, so that
1094
the data dictionary will be locked in crash recovery. */
1095
trx_set_dict_operation(trx, TRX_DICT_OP_INDEX);
1097
/* Lock the table exclusively, to ensure that no active
1098
transaction depends on an index that is being dropped. */
1099
err = convert_error_code_to_mysql(
1100
row_merge_lock_table(prebuilt->trx, prebuilt->table, LOCK_X),
1101
prebuilt->table->flags, user_thd);
1103
if (UNIV_UNLIKELY(err)) {
1105
/* Unmark the indexes to be dropped. */
1106
row_mysql_lock_data_dictionary(trx);
1108
for (index = dict_table_get_first_index(prebuilt->table);
1109
index; index = dict_table_get_next_index(index)) {
1111
index->to_be_dropped = FALSE;
1114
row_mysql_unlock_data_dictionary(trx);
1118
/* Drop indexes marked to be dropped */
1120
row_mysql_lock_data_dictionary(trx);
1122
index = dict_table_get_first_index(prebuilt->table);
1125
dict_index_t* next_index;
1127
next_index = dict_table_get_next_index(index);
1129
if (index->to_be_dropped) {
1131
row_merge_drop_index(index, prebuilt->table, trx);
1137
/* Check that all flagged indexes were dropped. */
1138
for (index = dict_table_get_first_index(prebuilt->table);
1139
index; index = dict_table_get_next_index(index)) {
1140
ut_a(!index->to_be_dropped);
1144
dict_table_check_for_dup_indexes(prebuilt->table);
1146
row_mysql_unlock_data_dictionary(trx);
1149
trx_commit_for_mysql(trx);
1150
trx_commit_for_mysql(prebuilt->trx);
1152
/* Flush the log to reduce probability that the .frm files and
1153
the InnoDB data dictionary get out-of-sync if the user runs
1154
with innodb_flush_log_at_trx_commit = 0 */
1156
log_buffer_flush_to_disk();
1158
trx_free_for_mysql(trx);
1160
/* Tell the InnoDB server that there might be work for
1163
srv_active_wake_master_thread();