1
/*****************************************************************************
3
Copyright (c) 2005, 2009, Innobase Oy. All Rights Reserved.
5
This program is free software; you can redistribute it and/or modify it under
6
the terms of the GNU General Public License as published by the Free Software
7
Foundation; version 2 of the License.
9
This program is distributed in the hope that it will be useful, but WITHOUT
10
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
13
You should have received a copy of the GNU General Public License along with
14
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
15
Place, Suite 330, Boston, MA 02111-1307 USA
17
*****************************************************************************/
19
/**************************************************//**
20
@file handler/handler0alter.cc
22
*******************************************************/
25
#include <drizzled/error.h>
26
#include "drizzled/charset_info.h"
27
#include <drizzled/field.h>
28
#include <drizzled/table.h>
29
#include <drizzled/field/varstring.h>
30
#include "drizzled/internal/my_sys.h"
34
#include "row0merge.h"
38
#include "ha_prototypes.h"
39
#include "handler0alter.h"
42
#include "ha_innodb.h"
43
#include "handler0vars.h"
45
/*************************************************************//**
46
Copies an InnoDB column to a MySQL field. This function is
47
adapted from row_sel_field_store_in_mysql_format(). */
50
innobase_col_to_mysql(
51
/*==================*/
52
const dict_col_t* col, /*!< in: InnoDB column */
53
const unsigned char* data, /*!< in: InnoDB column data */
54
ulint len, /*!< in: length of data, in bytes */
55
Field* field) /*!< in/out: MySQL field */
58
unsigned char* dest = field->ptr;
59
ulint flen = field->pack_length();
65
/* Convert integer data from Innobase to little-endian
66
format, sign bit restored to normal */
68
for (ptr = dest + len; ptr != dest; ) {
72
if (!(field->flags & UNSIGNED_FLAG)) {
73
((byte*) dest)[len - 1] ^= 0x80;
83
if (field->type() == DRIZZLE_TYPE_VARCHAR) {
84
/* This is a >= 5.0.3 type true VARCHAR. Store the
85
length of the data to the first byte or the first
88
dest = row_mysql_store_true_var_len(
89
dest, len, flen - field->key_length());
92
/* Copy the actual data */
93
memcpy(dest, data, len);
97
/* Store a pointer to the BLOB buffer to dest: the BLOB was
98
already copied to the buffer in row_sel_store_mysql_rec */
100
row_mysql_store_blob_ref(dest, flen, data, len);
106
ut_ad(col->mbmaxlen >= col->mbminlen);
107
ut_ad(col->mbmaxlen > col->mbminlen || flen == len);
108
memcpy(dest, data, len);
114
/* These column types should never be shipped to MySQL. */
122
/* Above are the valid column types for MySQL data. */
124
#else /* UNIV_DEBUG */
126
#endif /* UNIV_DEBUG */
127
memcpy(dest, data, len);
131
/*************************************************************//**
132
Copies an InnoDB record to table->getInsertRecord(). */
133
extern "C" UNIV_INTERN
135
innobase_rec_to_mysql(
136
/*==================*/
137
Table* table, /*!< in/out: MySQL table */
138
const rec_t* rec, /*!< in: record */
139
const dict_index_t* index, /*!< in: index */
140
const ulint* offsets) /*!< in: rec_get_offsets(
143
uint n_fields = table->getShare()->sizeFields();
146
ut_ad(n_fields == dict_table_get_n_user_cols(index->table));
148
for (i = 0; i < n_fields; i++) {
149
Field* field = table->getField(i);
152
const unsigned char* ifield;
156
ipos = dict_index_get_nth_col_pos(index, i);
158
if (UNIV_UNLIKELY(ipos == ULINT_UNDEFINED)) {
164
ifield = rec_get_nth_field(rec, offsets, ipos, &ilen);
166
/* Assign the NULL flag */
167
if (ilen == UNIV_SQL_NULL) {
168
ut_ad(field->real_maybe_null());
172
field->set_notnull();
174
innobase_col_to_mysql(
176
dict_index_get_nth_field(index, ipos)),
177
ifield, ilen, field);
181
/*************************************************************//**
182
Resets table->getInsertRecord(). */
183
extern "C" UNIV_INTERN
187
Table* table) /*!< in/out: MySQL table */
189
uint n_fields = table->getShare()->sizeFields();
192
for (i = 0; i < n_fields; i++) {
193
table->getField(i)->set_default();
197
#if 0 // This is a part of the fast index code.
198
/******************************************************************//**
199
Removes the filename encoding of a database and table name. */
202
innobase_convert_tablename(
203
/*=======================*/
204
char* s) /*!< in: identifier; out: decoded identifier */
207
char* slash = strchr(s, '/');
211
/* Temporarily replace the '/' with NUL. */
213
strncpy(s, s, slash - s + 1);
217
/* Append a '.' after the database name. */
220
/* Convert the table name. */
221
strncpy(t, slash, slash - t + strlen(slash));
226
/*******************************************************************//**
227
This function checks that index keys are sensible.
228
@return 0 or error number */
231
innobase_check_index_keys(
232
/*======================*/
233
const KeyInfo* key_info, /*!< in: Indexes to be created */
234
ulint num_of_keys) /*!< in: Number of indexes to
242
for (key_num = 0; key_num < num_of_keys; key_num++) {
243
const KeyInfo& key = key_info[key_num];
245
/* Check that the same index name does not appear
246
twice in indexes to be created. */
248
for (ulint i = 0; i < key_num; i++) {
249
const KeyInfo& key2 = key_info[i];
251
if (0 == strcmp(key.name, key2.name)) {
252
errmsg_printf(ERRMSG_LVL_ERROR, "InnoDB: key name `%s` appears"
253
" twice in CREATE INDEX\n",
256
return(ER_WRONG_NAME_FOR_INDEX);
260
/* Check that MySQL does not try to create a column
261
prefix index field on an inappropriate data type and
262
that the same colum does not appear twice in the index. */
264
for (ulint i = 0; i < key.key_parts; i++) {
265
const KeyPartInfo& key_part1
271
switch (get_innobase_type_from_mysql_type(
272
&is_unsigned, field)) {
279
if (field->type() == DRIZZLE_TYPE_VARCHAR) {
281
>= field->pack_length()
282
- ((Field_varstring*) field)
288
>= field->pack_length()) {
293
errmsg_printf(ERRMSG_LVL_ERROR, "InnoDB: MySQL is trying to"
294
" create a column prefix"
296
" inappropriate data type."
301
return(ER_WRONG_KEY_COLUMN);
304
for (ulint j = 0; j < i; j++) {
305
const KeyPartInfo& key_part2
308
if (strcmp(key_part1.field->field_name,
309
key_part2.field->field_name)) {
313
errmsg_printf(ERRMSG_LVL_ERROR, "InnoDB: column `%s`"
314
" is not allowed to occur"
315
" twice in index `%s`.\n",
316
key_part1.field->field_name,
318
return(ER_WRONG_KEY_COLUMN);
326
/*******************************************************************//**
327
Create index field definition for key part */
330
innobase_create_index_field_def(
331
/*============================*/
332
KeyPartInfo* key_part, /*!< in: MySQL key definition */
333
mem_heap_t* heap, /*!< in: memory heap */
334
merge_index_field_t* index_field) /*!< out: index field
335
definition for key_part */
344
field = key_part->field;
347
col_type = get_innobase_type_from_mysql_type(&is_unsigned, field);
349
if (DATA_BLOB == col_type
350
|| (key_part->length < field->pack_length()
351
&& field->type() != DRIZZLE_TYPE_VARCHAR)
352
|| (field->type() == DRIZZLE_TYPE_VARCHAR
353
&& key_part->length < field->pack_length()
354
- ((Field_varstring*)field)->length_bytes)) {
356
index_field->prefix_len = key_part->length;
358
index_field->prefix_len = 0;
361
index_field->field_name = mem_heap_strdup(heap, field->field_name);
366
/*******************************************************************//**
367
Create index definition for key */
370
innobase_create_index_def(
371
/*======================*/
372
KeyInfo* key, /*!< in: key definition */
373
bool new_primary, /*!< in: TRUE=generating
376
bool key_primary, /*!< in: TRUE if this key
378
merge_index_def_t* index, /*!< out: index definition */
379
mem_heap_t* heap) /*!< in: heap where memory
384
ulint n_fields = key->key_parts;
387
index->fields = (merge_index_field_t*) mem_heap_alloc(
388
heap, n_fields * sizeof *index->fields);
391
index->n_fields = n_fields;
392
len = strlen(key->name) + 1;
393
index->name = index_name = (char*) mem_heap_alloc(heap,
396
if (UNIV_LIKELY(!new_primary)) {
397
*index_name++ = TEMP_INDEX_PREFIX;
400
memcpy(index_name, key->name, len);
402
if (key->flags & HA_NOSAME) {
403
index->ind_type |= DICT_UNIQUE;
407
index->ind_type |= DICT_CLUSTERED;
410
for (i = 0; i < n_fields; i++) {
411
innobase_create_index_field_def(&key->key_part[i], heap,
418
/*******************************************************************//**
419
Copy index field definition */
422
innobase_copy_index_field_def(
423
/*==========================*/
424
const dict_field_t* field, /*!< in: definition to copy */
425
merge_index_field_t* index_field) /*!< out: copied definition */
427
assert(field != NULL);
428
assert(index_field != NULL);
430
index_field->field_name = field->name;
431
index_field->prefix_len = field->prefix_len;
436
/*******************************************************************//**
437
Copy index definition for the index */
440
innobase_copy_index_def(
441
/*====================*/
442
const dict_index_t* index, /*!< in: index definition to copy */
443
merge_index_def_t* new_index,/*!< out: Index definition */
444
mem_heap_t* heap) /*!< in: heap where allocated */
449
/* Note that we take only those fields that user defined to be
450
in the index. In the internal representation more colums were
451
added and those colums are not copied .*/
453
n_fields = index->n_user_defined_cols;
455
new_index->fields = (merge_index_field_t*) mem_heap_alloc(
456
heap, n_fields * sizeof *new_index->fields);
458
/* When adding a PRIMARY KEY, we may convert a previous
459
clustered index to a secondary index (UNIQUE NOT NULL). */
460
new_index->ind_type = index->type & ~DICT_CLUSTERED;
461
new_index->n_fields = n_fields;
462
new_index->name = index->name;
464
for (i = 0; i < n_fields; i++) {
465
innobase_copy_index_field_def(&index->fields[i],
466
&new_index->fields[i]);
472
/*******************************************************************//**
473
Create an index table where indexes are ordered as follows:
475
IF a new primary key is defined for the table THEN
478
2) Original secondary indexes
479
3) New secondary indexes
483
1) All new indexes in the order they arrive from MySQL
488
@return key definitions or NULL */
492
innobase_create_key_def(
493
/*====================*/
494
trx_t* trx, /*!< in: trx */
495
const dict_table_t*table, /*!< in: table definition */
496
mem_heap_t* heap, /*!< in: heap where space for key
497
definitions are allocated */
498
KeyInfo* key_info, /*!< in: Indexes to be created */
499
ulint& n_keys) /*!< in/out: Number of indexes to
503
merge_index_def_t* indexdef;
504
merge_index_def_t* indexdefs;
507
indexdef = indexdefs = (merge_index_def_t*)
508
mem_heap_alloc(heap, sizeof *indexdef
509
* (n_keys + UT_LIST_GET_LEN(table->indexes)));
511
/* If there is a primary key, it is always the first index
512
defined for the table. */
514
new_primary = !my_strcasecmp(system_charset_info,
515
key_info->name, "PRIMARY");
517
/* If there is a UNIQUE INDEX consisting entirely of NOT NULL
518
columns, MySQL will treat it as a PRIMARY KEY unless the
519
table already has one. */
521
if (!new_primary && (key_info->flags & HA_NOSAME)
522
&& row_table_got_default_clust_index(table)) {
523
uint key_part = key_info->key_parts;
528
if (key_info->key_part[key_part].null_bit == 0) {
536
const dict_index_t* index;
538
/* Create the PRIMARY key index definition */
539
innobase_create_index_def(&key_info[i++], TRUE, TRUE,
542
row_mysql_lock_data_dictionary(trx);
544
index = dict_table_get_first_index(table);
546
/* Copy the index definitions of the old table. Skip
547
the old clustered index if it is a generated clustered
548
index or a PRIMARY KEY. If the clustered index is a
549
UNIQUE INDEX, it must be converted to a secondary index. */
551
if (dict_index_get_nth_col(index, 0)->mtype == DATA_SYS
552
|| !my_strcasecmp(system_charset_info,
553
index->name, "PRIMARY")) {
554
index = dict_table_get_next_index(index);
558
innobase_copy_index_def(index, indexdef++, heap);
559
index = dict_table_get_next_index(index);
562
row_mysql_unlock_data_dictionary(trx);
565
/* Create definitions for added secondary indexes. */
568
innobase_create_index_def(&key_info[i++], new_primary, FALSE,
572
n_keys = indexdef - indexdefs;
577
/*******************************************************************//**
578
Create a temporary tablename using query id, thread id, and id
579
@return temporary tablename */
582
innobase_create_temporary_tablename(
583
/*================================*/
584
mem_heap_t* heap, /*!< in: memory heap */
585
char id, /*!< in: identifier [0-9a-zA-Z] */
586
const char* table_name) /*!< in: table name */
590
static const char suffix[] = "@0023 "; /* "# " */
592
len = strlen(table_name);
594
name = (char*) mem_heap_alloc(heap, len + sizeof suffix);
595
memcpy(name, table_name, len);
596
memcpy(name + len, suffix, sizeof suffix);
597
name[len + (sizeof suffix - 2)] = id;
603
/*******************************************************************//**
605
@return 0 or error number */
608
ha_innobase::add_index(
609
/*===================*/
611
Table* i_table, /*!< in: Table where indexes are created */
612
KeyInfo* key_info, /*!< in: Indexes to be created */
613
uint num_of_keys) /*!< in: Number of indexes to be created */
615
dict_index_t** index; /*!< Index to be created */
616
dict_table_t* innodb_table; /*!< InnoDB table in dictionary */
617
dict_table_t* indexed_table; /*!< Table where indexes are created */
618
merge_index_def_t* index_defs; /*!< Index definitions */
619
mem_heap_t* heap; /*!< Heap for index definitions */
620
trx_t* trx; /*!< Transaction */
622
ulint num_created = 0;
623
ibool dict_locked = FALSE;
631
if (srv_created_new_raw || srv_force_recovery) {
632
return(HA_ERR_WRONG_COMMAND);
635
update_session(session);
637
heap = mem_heap_create(1024);
639
/* In case MySQL calls this in the middle of a SELECT query, release
640
possible adaptive hash latch to avoid deadlocks of threads. */
641
trx_search_latch_release_if_reserved(prebuilt->trx);
642
trx_start_if_not_started(prebuilt->trx);
644
/* Create a background transaction for the operations on
645
the data dictionary tables. */
646
trx = innobase_trx_allocate(user_session);
647
trx_start_if_not_started(trx);
649
innodb_table = indexed_table
650
= dict_table_get(prebuilt->table->name, FALSE);
652
/* Check that index keys are sensible */
654
error = innobase_check_index_keys(key_info, num_of_keys);
656
if (UNIV_UNLIKELY(error)) {
659
trx_general_rollback_for_mysql(trx, FALSE, NULL);
660
trx_free_for_mysql(trx);
661
trx_commit_for_mysql(prebuilt->trx);
665
/* Create table containing all indexes to be built in this
666
alter table add index so that they are in the correct order
669
num_of_idx = num_of_keys;
671
index_defs = innobase_create_key_def(
672
trx, innodb_table, heap, key_info, num_of_idx);
674
new_primary = DICT_CLUSTERED & index_defs[0].ind_type;
676
/* Allocate memory for dictionary index definitions */
678
index = (dict_index_t**) mem_heap_alloc(
679
heap, num_of_idx * sizeof *index);
681
/* Flag this transaction as a dictionary operation, so that
682
the data dictionary will be locked in crash recovery. */
683
trx_set_dict_operation(trx, TRX_DICT_OP_INDEX);
685
/* Acquire a lock on the table before creating any indexes. */
686
error = row_merge_lock_table(prebuilt->trx, innodb_table,
687
new_primary ? LOCK_X : LOCK_S);
689
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
694
/* Latch the InnoDB data dictionary exclusively so that no deadlocks
695
or lock waits can happen in it during an index create operation. */
697
row_mysql_lock_data_dictionary(trx);
700
/* If a new primary key is defined for the table we need
701
to drop the original table and rebuild all indexes. */
703
if (UNIV_UNLIKELY(new_primary)) {
704
/* This transaction should be the only one
705
operating on the table. */
706
ut_a(innodb_table->n_mysql_handles_opened == 1);
708
char* new_table_name = innobase_create_temporary_tablename(
709
heap, '1', innodb_table->name);
711
/* Clone the table. */
712
trx_set_dict_operation(trx, TRX_DICT_OP_TABLE);
713
indexed_table = row_merge_create_temporary_table(
714
new_table_name, index_defs, innodb_table, trx);
716
if (!indexed_table) {
718
switch (trx->error_state) {
719
case DB_TABLESPACE_ALREADY_EXISTS:
720
case DB_DUPLICATE_KEY:
721
innobase_convert_tablename(new_table_name);
722
my_error(HA_ERR_TABLE_EXIST, MYF(0),
724
error = HA_ERR_TABLE_EXIST;
727
error = convert_error_code_to_mysql(
728
trx->error_state, innodb_table->flags,
732
row_mysql_unlock_data_dictionary(trx);
736
trx->table_id = indexed_table->id;
739
/* Create the indexes in SYS_INDEXES and load into dictionary. */
741
for (ulint i = 0; i < num_of_idx; i++) {
743
index[i] = row_merge_create_index(trx, indexed_table,
747
error = trx->error_state;
754
ut_ad(error == DB_SUCCESS);
756
/* Commit the data dictionary transaction in order to release
757
the table locks on the system tables. Unfortunately, this
758
means that if MySQL crashes while creating a new primary key
759
inside row_merge_build_indexes(), indexed_table will not be
760
dropped on crash recovery. Thus, it will become orphaned. */
761
trx_commit_for_mysql(trx);
763
row_mysql_unlock_data_dictionary(trx);
766
ut_a(trx->n_active_thrs == 0);
767
ut_a(UT_LIST_GET_LEN(trx->signals) == 0);
769
if (UNIV_UNLIKELY(new_primary)) {
770
/* A primary key is to be built. Acquire an exclusive
771
table lock also on the table that is being created. */
772
ut_ad(indexed_table != innodb_table);
774
error = row_merge_lock_table(prebuilt->trx, indexed_table,
777
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
783
/* Read the clustered index of the table and build indexes
784
based on this information using temporary files and merge sort. */
785
error = row_merge_build_indexes(prebuilt->trx,
786
innodb_table, indexed_table,
787
index, num_of_idx, i_table);
791
/* TODO: At the moment we can't handle the following statement
792
in our debugging code below:
794
alter table t drop index b, add index (b);
796
The fix will have to parse the SQL and note that the index
797
being added has the same name as the the one being dropped and
798
ignore that in the dup index check.*/
799
//dict_table_check_for_dup_indexes(prebuilt->table);
802
/* After an error, remove all those index definitions from the
803
dictionary which were defined. */
806
const char* old_name;
810
row_mysql_lock_data_dictionary(trx);
814
error = row_merge_rename_indexes(trx, indexed_table);
816
if (error != DB_SUCCESS) {
817
row_merge_drop_indexes(trx, indexed_table,
824
/* If a new primary key was defined for the table and
825
there was no error at this point, we can now rename
826
the old table as a temporary table, rename the new
827
temporary table as the old table and drop the old table. */
828
old_name = innodb_table->name;
829
tmp_name = innobase_create_temporary_tablename(heap, '2',
832
error = row_merge_rename_tables(innodb_table, indexed_table,
835
if (error != DB_SUCCESS) {
837
row_merge_drop_table(trx, indexed_table);
840
case DB_TABLESPACE_ALREADY_EXISTS:
841
case DB_DUPLICATE_KEY:
842
innobase_convert_tablename(tmp_name);
843
my_error(HA_ERR_TABLE_EXIST, MYF(0), tmp_name);
844
error = HA_ERR_TABLE_EXIST;
852
trx_commit_for_mysql(prebuilt->trx);
853
row_prebuilt_free(prebuilt, TRUE);
854
prebuilt = row_create_prebuilt(indexed_table);
856
indexed_table->n_mysql_handles_opened++;
858
error = row_merge_drop_table(trx, innodb_table);
861
case DB_TOO_BIG_RECORD:
862
my_error(HA_ERR_TO_BIG_ROW, MYF(0));
864
case DB_PRIMARY_KEY_IS_NULL:
865
my_error(ER_PRIMARY_CANT_HAVE_NULL, MYF(0));
867
case DB_DUPLICATE_KEY:
869
prebuilt->trx->error_info = NULL;
873
row_merge_drop_table(trx, indexed_table);
876
row_mysql_lock_data_dictionary(trx);
880
row_merge_drop_indexes(trx, indexed_table,
885
error = convert_error_code_to_mysql(error,
891
trx_commit_for_mysql(trx);
893
trx_commit_for_mysql(prebuilt->trx);
897
row_mysql_unlock_data_dictionary(trx);
900
trx_free_for_mysql(trx);
902
/* There might be work for utility threads.*/
903
srv_active_wake_master_thread();
908
/*******************************************************************//**
909
Prepare to drop some indexes of a table.
910
@return 0 or error number */
913
ha_innobase::prepare_drop_index(
914
/*============================*/
916
Table* i_table, /*!< in: Table where indexes are dropped */
917
uint* key_num, /*!< in: Key nums to be dropped */
918
uint num_of_keys) /*!< in: Number of keys to be dropped */
927
if (srv_created_new_raw || srv_force_recovery) {
928
return(HA_ERR_WRONG_COMMAND);
931
update_session(session);
933
trx_search_latch_release_if_reserved(prebuilt->trx);
936
/* Test and mark all the indexes to be dropped */
938
row_mysql_lock_data_dictionary(trx);
940
/* Check that none of the indexes have previously been flagged
943
const dict_index_t* index
944
= dict_table_get_first_index(prebuilt->table);
946
ut_a(!index->to_be_dropped);
947
index = dict_table_get_next_index(index);
951
for (n_key = 0; n_key < num_of_keys; n_key++) {
955
key = i_table->key_info + key_num[n_key];
956
index = dict_table_get_index_on_name_and_min_id(
957
prebuilt->table, key->name);
960
errmsg_printf(ERRMSG_LVL_ERROR, "InnoDB could not find key n:o %u "
961
"with name %s for table %s",
963
key ? key->name : "NULL",
964
prebuilt->table->name);
966
err = HA_ERR_KEY_NOT_FOUND;
970
/* Refuse to drop the clustered index. It would be
971
better to automatically generate a clustered index,
972
but drizzled::alter_table() will call this method only
973
after ha_innobase::add_index(). */
975
if (dict_index_is_clust(index)) {
976
my_error(ER_REQUIRES_PRIMARY_KEY, MYF(0));
981
index->to_be_dropped = TRUE;
984
/* If FOREIGN_KEY_CHECK = 1 you may not drop an index defined
985
for a foreign key constraint because InnoDB requires that both
986
tables contain indexes for the constraint. Note that CREATE
987
INDEX id ON table does a CREATE INDEX and DROP INDEX, and we
988
can ignore here foreign keys because a new index for the
989
foreign key has already been created.
991
We check for the foreign key constraints after marking the
992
candidate indexes for deletion, because when we check for an
993
equivalent foreign index we don't want to select an index that
996
if (trx->check_foreigns
997
&& session_sql_command(user_session) != SQLCOM_CREATE_INDEX) {
1000
for (index = dict_table_get_first_index(prebuilt->table);
1002
index = dict_table_get_next_index(index)) {
1003
dict_foreign_t* foreign;
1005
if (!index->to_be_dropped) {
1010
/* Check if the index is referenced. */
1011
foreign = dict_table_get_referenced_constraint(
1012
prebuilt->table, index);
1016
trx_set_detailed_error(
1018
"Index needed in foreign key "
1021
trx->error_info = index;
1023
err = HA_ERR_DROP_INDEX_FK;
1026
/* Check if this index references some
1028
foreign = dict_table_get_foreign_constraint(
1029
prebuilt->table, index);
1032
ut_a(foreign->foreign_index == index);
1034
/* Search for an equivalent index that
1035
the foreign key constraint could use
1036
if this index were to be deleted. */
1037
if (!dict_foreign_find_equiv_index(
1045
} else if (session_sql_command(user_session) == SQLCOM_CREATE_INDEX) {
1046
/* This is a drop of a foreign key constraint index that
1047
was created by MySQL when the constraint was added. MySQL
1048
does this when the user creates an index explicitly which
1049
can be used in place of the automatically generated index. */
1051
dict_index_t* index;
1053
for (index = dict_table_get_first_index(prebuilt->table);
1055
index = dict_table_get_next_index(index)) {
1056
dict_foreign_t* foreign;
1058
if (!index->to_be_dropped) {
1063
/* Check if this index references some other table */
1064
foreign = dict_table_get_foreign_constraint(
1065
prebuilt->table, index);
1067
if (foreign == NULL) {
1072
ut_a(foreign->foreign_index == index);
1074
/* Search for an equivalent index that the
1075
foreign key constraint could use if this index
1076
were to be deleted. */
1078
if (!dict_foreign_find_equiv_index(foreign)) {
1079
trx_set_detailed_error(
1081
"Index needed in foreign key "
1084
trx->error_info = foreign->foreign_index;
1086
err = HA_ERR_DROP_INDEX_FK;
1094
/* Undo our changes since there was some sort of error. */
1096
= dict_table_get_first_index(prebuilt->table);
1099
index->to_be_dropped = FALSE;
1100
index = dict_table_get_next_index(index);
1104
row_mysql_unlock_data_dictionary(trx);
1109
/*******************************************************************//**
1110
Drop the indexes that were passed to a successful prepare_drop_index().
1111
@return 0 or error number */
1114
ha_innobase::final_drop_index(
1115
/*==========================*/
1117
Table* ) /*!< in: Table where indexes are dropped */
1119
dict_index_t* index; /*!< Index to be dropped */
1120
trx_t* trx; /*!< Transaction */
1123
if (srv_created_new_raw || srv_force_recovery) {
1124
return(HA_ERR_WRONG_COMMAND);
1127
update_session(session);
1129
trx_search_latch_release_if_reserved(prebuilt->trx);
1130
trx_start_if_not_started(prebuilt->trx);
1132
/* Create a background transaction for the operations on
1133
the data dictionary tables. */
1134
trx = innobase_trx_allocate(user_session);
1135
trx_start_if_not_started(trx);
1137
/* Flag this transaction as a dictionary operation, so that
1138
the data dictionary will be locked in crash recovery. */
1139
trx_set_dict_operation(trx, TRX_DICT_OP_INDEX);
1141
/* Lock the table exclusively, to ensure that no active
1142
transaction depends on an index that is being dropped. */
1143
err = convert_error_code_to_mysql(
1144
row_merge_lock_table(prebuilt->trx, prebuilt->table, LOCK_X),
1145
prebuilt->table->flags, user_session);
1147
row_mysql_lock_data_dictionary(trx);
1149
if (UNIV_UNLIKELY(err)) {
1151
/* Unmark the indexes to be dropped. */
1152
for (index = dict_table_get_first_index(prebuilt->table);
1153
index; index = dict_table_get_next_index(index)) {
1155
index->to_be_dropped = FALSE;
1161
/* Drop indexes marked to be dropped */
1163
index = dict_table_get_first_index(prebuilt->table);
1166
dict_index_t* next_index;
1168
next_index = dict_table_get_next_index(index);
1170
if (index->to_be_dropped) {
1172
row_merge_drop_index(index, prebuilt->table, trx);
1178
/* Check that all flagged indexes were dropped. */
1179
for (index = dict_table_get_first_index(prebuilt->table);
1180
index; index = dict_table_get_next_index(index)) {
1181
ut_a(!index->to_be_dropped);
1185
dict_table_check_for_dup_indexes(prebuilt->table);
1189
trx_commit_for_mysql(trx);
1190
trx_commit_for_mysql(prebuilt->trx);
1191
row_mysql_unlock_data_dictionary(trx);
1193
/* Flush the log to reduce probability that the .frm files and
1194
the InnoDB data dictionary get out-of-sync if the user runs
1195
with innodb_flush_log_at_trx_commit = 0 */
1197
log_buffer_flush_to_disk();
1199
trx_free_for_mysql(trx);
1201
/* Tell the InnoDB server that there might be work for
1204
srv_active_wake_master_thread();