mirror of
https://github.com/MariaDB/server.git
synced 2025-01-19 21:42:35 +01:00
6381 lines
178 KiB
C++
6381 lines
178 KiB
C++
/*****************************************************************************
|
|
|
|
Copyright (c) 2005, 2016, Oracle and/or its affiliates. All Rights Reserved.
|
|
Copyright (c) 2013, 2017, MariaDB Corporation. All Rights Reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify it under
|
|
the terms of the GNU General Public License as published by the Free Software
|
|
Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
this program; if not, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Suite 500, Boston, MA 02110-1335 USA
|
|
|
|
*****************************************************************************/
|
|
|
|
/**************************************************//**
|
|
@file handler/handler0alter.cc
|
|
Smart ALTER TABLE
|
|
*******************************************************/
|
|
|
|
#include <my_global.h>
|
|
#include <unireg.h>
|
|
#include <mysqld_error.h>
|
|
#include <log.h>
|
|
#include <debug_sync.h>
|
|
#include <innodb_priv.h>
|
|
#include <sql_alter.h>
|
|
#include <sql_class.h>
|
|
#include <sql_table.h>
|
|
|
|
#include "dict0crea.h"
|
|
#include "dict0dict.h"
|
|
#include "dict0priv.h"
|
|
#include "dict0stats.h"
|
|
#include "dict0stats_bg.h"
|
|
#include "log0log.h"
|
|
#include "rem0types.h"
|
|
#include "row0log.h"
|
|
#include "row0merge.h"
|
|
#include "srv0srv.h"
|
|
#include "trx0trx.h"
|
|
#include "trx0roll.h"
|
|
#include "ha_prototypes.h"
|
|
#include "handler0alter.h"
|
|
#include "srv0mon.h"
|
|
#include "fts0priv.h"
|
|
#include "pars0pars.h"
|
|
#include "row0sel.h"
|
|
#include "ha_innodb.h"
|
|
#ifdef WITH_WSREP
|
|
//#include "wsrep_api.h"
|
|
#include <sql_acl.h> // PROCESS_ACL
|
|
#endif
|
|
|
|
/** Operations for creating secondary indexes (no rebuild needed) */
|
|
static const Alter_inplace_info::HA_ALTER_FLAGS INNOBASE_ONLINE_CREATE
|
|
= Alter_inplace_info::ADD_INDEX
|
|
| Alter_inplace_info::ADD_UNIQUE_INDEX;
|
|
|
|
/** Operations for rebuilding a table in place */
|
|
static const Alter_inplace_info::HA_ALTER_FLAGS INNOBASE_ALTER_REBUILD
|
|
= Alter_inplace_info::ADD_PK_INDEX
|
|
| Alter_inplace_info::DROP_PK_INDEX
|
|
| Alter_inplace_info::CHANGE_CREATE_OPTION
|
|
/* CHANGE_CREATE_OPTION needs to check innobase_need_rebuild() */
|
|
| Alter_inplace_info::ALTER_COLUMN_NULLABLE
|
|
| Alter_inplace_info::ALTER_COLUMN_NOT_NULLABLE
|
|
| Alter_inplace_info::ALTER_COLUMN_ORDER
|
|
| Alter_inplace_info::DROP_COLUMN
|
|
| Alter_inplace_info::ADD_COLUMN
|
|
| Alter_inplace_info::RECREATE_TABLE
|
|
/*
|
|
| Alter_inplace_info::ALTER_COLUMN_TYPE
|
|
| Alter_inplace_info::ALTER_COLUMN_EQUAL_PACK_LENGTH
|
|
*/
|
|
;
|
|
|
|
/** Operations that require changes to data */
|
|
static const Alter_inplace_info::HA_ALTER_FLAGS INNOBASE_ALTER_DATA
|
|
= INNOBASE_ONLINE_CREATE | INNOBASE_ALTER_REBUILD;
|
|
|
|
/** Operations for altering a table that InnoDB does not care about */
|
|
static const Alter_inplace_info::HA_ALTER_FLAGS INNOBASE_INPLACE_IGNORE
|
|
= Alter_inplace_info::ALTER_COLUMN_DEFAULT
|
|
| Alter_inplace_info::ALTER_PARTITIONED
|
|
| Alter_inplace_info::ALTER_COLUMN_COLUMN_FORMAT
|
|
| Alter_inplace_info::ALTER_COLUMN_STORAGE_TYPE
|
|
| Alter_inplace_info::ALTER_RENAME;
|
|
|
|
/** Operations on foreign key definitions (changing the schema only) */
|
|
static const Alter_inplace_info::HA_ALTER_FLAGS INNOBASE_FOREIGN_OPERATIONS
|
|
= Alter_inplace_info::DROP_FOREIGN_KEY
|
|
| Alter_inplace_info::ADD_FOREIGN_KEY;
|
|
|
|
/** Operations that InnoDB cares about and can perform without rebuild */
|
|
static const Alter_inplace_info::HA_ALTER_FLAGS INNOBASE_ALTER_NOREBUILD
|
|
= INNOBASE_ONLINE_CREATE
|
|
| INNOBASE_FOREIGN_OPERATIONS
|
|
| Alter_inplace_info::DROP_INDEX
|
|
| Alter_inplace_info::DROP_UNIQUE_INDEX
|
|
| Alter_inplace_info::ALTER_COLUMN_NAME;
|
|
|
|
/* Report an InnoDB error to the client by invoking my_error(). */
|
|
static UNIV_COLD MY_ATTRIBUTE((nonnull))
|
|
void
|
|
my_error_innodb(
|
|
/*============*/
|
|
dberr_t error, /*!< in: InnoDB error code */
|
|
const char* table, /*!< in: table name */
|
|
ulint flags) /*!< in: table flags */
|
|
{
|
|
switch (error) {
|
|
case DB_MISSING_HISTORY:
|
|
my_error(ER_TABLE_DEF_CHANGED, MYF(0));
|
|
break;
|
|
case DB_RECORD_NOT_FOUND:
|
|
my_error(ER_KEY_NOT_FOUND, MYF(0), table);
|
|
break;
|
|
case DB_DEADLOCK:
|
|
my_error(ER_LOCK_DEADLOCK, MYF(0));
|
|
break;
|
|
case DB_LOCK_WAIT_TIMEOUT:
|
|
my_error(ER_LOCK_WAIT_TIMEOUT, MYF(0));
|
|
break;
|
|
case DB_INTERRUPTED:
|
|
my_error(ER_QUERY_INTERRUPTED, MYF(0));
|
|
break;
|
|
case DB_OUT_OF_MEMORY:
|
|
my_error(ER_OUT_OF_RESOURCES, MYF(0));
|
|
break;
|
|
case DB_OUT_OF_FILE_SPACE:
|
|
my_error(ER_RECORD_FILE_FULL, MYF(0), table);
|
|
break;
|
|
case DB_TEMP_FILE_WRITE_FAILURE:
|
|
my_error(ER_GET_ERRMSG, MYF(0),
|
|
DB_TEMP_FILE_WRITE_FAILURE,
|
|
ut_strerr(DB_TEMP_FILE_WRITE_FAILURE),
|
|
"InnoDB");
|
|
break;
|
|
case DB_TOO_BIG_INDEX_COL:
|
|
my_error(ER_INDEX_COLUMN_TOO_LONG, MYF(0),
|
|
DICT_MAX_FIELD_LEN_BY_FORMAT_FLAG(flags));
|
|
break;
|
|
case DB_TOO_MANY_CONCURRENT_TRXS:
|
|
my_error(ER_TOO_MANY_CONCURRENT_TRXS, MYF(0));
|
|
break;
|
|
case DB_LOCK_TABLE_FULL:
|
|
my_error(ER_LOCK_TABLE_FULL, MYF(0));
|
|
break;
|
|
case DB_UNDO_RECORD_TOO_BIG:
|
|
my_error(ER_UNDO_RECORD_TOO_BIG, MYF(0));
|
|
break;
|
|
case DB_CORRUPTION:
|
|
my_error(ER_NOT_KEYFILE, MYF(0), table);
|
|
break;
|
|
case DB_TOO_BIG_RECORD:
|
|
my_error(ER_TOO_BIG_ROWSIZE, MYF(0),
|
|
page_get_free_space_of_empty(
|
|
flags & DICT_TF_COMPACT) / 2);
|
|
break;
|
|
case DB_INVALID_NULL:
|
|
/* TODO: report the row, as we do for DB_DUPLICATE_KEY */
|
|
my_error(ER_INVALID_USE_OF_NULL, MYF(0));
|
|
break;
|
|
case DB_TABLESPACE_EXISTS:
|
|
my_error(ER_TABLESPACE_EXISTS, MYF(0), table);
|
|
break;
|
|
|
|
#ifdef UNIV_DEBUG
|
|
case DB_SUCCESS:
|
|
case DB_DUPLICATE_KEY:
|
|
case DB_ONLINE_LOG_TOO_BIG:
|
|
/* These codes should not be passed here. */
|
|
ut_error;
|
|
#endif /* UNIV_DEBUG */
|
|
default:
|
|
my_error(ER_GET_ERRNO, MYF(0), error, "InnoDB");
|
|
break;
|
|
}
|
|
}
|
|
|
|
/** Determine if fulltext indexes exist in a given table.
|
|
@param table MySQL table
|
|
@return whether fulltext indexes exist on the table */
|
|
static
|
|
bool
|
|
innobase_fulltext_exist(
|
|
/*====================*/
|
|
const TABLE* table)
|
|
{
|
|
for (uint i = 0; i < table->s->keys; i++) {
|
|
if (table->key_info[i].flags & HA_FULLTEXT) {
|
|
return(true);
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Determine if ALTER TABLE needs to rebuild the table.
|
|
@param ha_alter_info the DDL operation
|
|
@param altered_table MySQL original table
|
|
@return whether it is necessary to rebuild the table */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_need_rebuild(
|
|
/*==================*/
|
|
const Alter_inplace_info* ha_alter_info,
|
|
const TABLE* altered_table)
|
|
{
|
|
Alter_inplace_info::HA_ALTER_FLAGS alter_inplace_flags =
|
|
ha_alter_info->handler_flags & ~(INNOBASE_INPLACE_IGNORE);
|
|
|
|
if (alter_inplace_flags
|
|
== Alter_inplace_info::CHANGE_CREATE_OPTION
|
|
&& !(ha_alter_info->create_info->used_fields
|
|
& (HA_CREATE_USED_ROW_FORMAT
|
|
| HA_CREATE_USED_KEY_BLOCK_SIZE))) {
|
|
/* Any other CHANGE_CREATE_OPTION than changing
|
|
ROW_FORMAT or KEY_BLOCK_SIZE is ignored. */
|
|
return(false);
|
|
}
|
|
|
|
/* If alter table changes column name and adds a new
|
|
index, we need to check is this new index created
|
|
to new column name. This is because column name
|
|
changes are done normally after creating indexes. */
|
|
if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME) &&
|
|
((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_INDEX) ||
|
|
(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_FOREIGN_KEY))) {
|
|
for (ulint i = 0; i < ha_alter_info->index_add_count; i++) {
|
|
const KEY* key = &ha_alter_info->key_info_buffer[
|
|
ha_alter_info->index_add_buffer[i]];
|
|
|
|
for (ulint j = 0; j < key->user_defined_key_parts; j++) {
|
|
const KEY_PART_INFO* key_part = &(key->key_part[j]);
|
|
const Field* field = altered_table->field[key_part->fieldnr];
|
|
|
|
/* Field used on added index is renamed on
|
|
this same alter table. We need table
|
|
rebuild. */
|
|
if (field && field->flags & FIELD_IS_RENAMED) {
|
|
return (true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return(!!(ha_alter_info->handler_flags & INNOBASE_ALTER_REBUILD));
|
|
}
|
|
|
|
/** Check if InnoDB supports a particular alter table in-place
|
|
@param altered_table TABLE object for new version of table.
|
|
@param ha_alter_info Structure describing changes to be done
|
|
by ALTER TABLE and holding data used during in-place alter.
|
|
|
|
@retval HA_ALTER_INPLACE_NOT_SUPPORTED Not supported
|
|
@retval HA_ALTER_INPLACE_NO_LOCK Supported
|
|
@retval HA_ALTER_INPLACE_SHARED_LOCK_AFTER_PREPARE Supported, but requires
|
|
lock during main phase and exclusive lock during prepare phase.
|
|
@retval HA_ALTER_INPLACE_NO_LOCK_AFTER_PREPARE Supported, prepare phase
|
|
requires exclusive lock (any transactions that have accessed the table
|
|
must commit or roll back first, and no transactions can access the table
|
|
while prepare_inplace_alter_table() is executing)
|
|
*/
|
|
UNIV_INTERN
|
|
enum_alter_inplace_result
|
|
ha_innobase::check_if_supported_inplace_alter(
|
|
/*==========================================*/
|
|
TABLE* altered_table,
|
|
Alter_inplace_info* ha_alter_info)
|
|
{
|
|
DBUG_ENTER("check_if_supported_inplace_alter");
|
|
|
|
if (high_level_read_only) {
|
|
ha_alter_info->unsupported_reason =
|
|
innobase_get_err_msg(ER_READ_ONLY_MODE);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
} else if (srv_created_new_raw || srv_force_recovery) {
|
|
|
|
ha_alter_info->unsupported_reason =
|
|
innobase_get_err_msg(ER_READ_ONLY_MODE);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
if (altered_table->s->stored_fields > REC_MAX_N_USER_FIELDS) {
|
|
/* Deny the inplace ALTER TABLE. MySQL will try to
|
|
re-create the table and ha_innobase::create() will
|
|
return an error too. This is how we effectively
|
|
deny adding too many columns to a table. */
|
|
ha_alter_info->unsupported_reason =
|
|
innobase_get_err_msg(ER_TOO_MANY_FIELDS);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
update_thd();
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
|
|
/* Change on engine specific table options require rebuild of the
|
|
table */
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::CHANGE_CREATE_OPTION) {
|
|
ha_table_option_struct *new_options= ha_alter_info->create_info->option_struct;
|
|
ha_table_option_struct *old_options= table->s->option_struct;
|
|
|
|
if (new_options->page_compressed != old_options->page_compressed ||
|
|
new_options->page_compression_level != old_options->page_compression_level ||
|
|
new_options->atomic_writes != old_options->atomic_writes) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
if (new_options->encryption != old_options->encryption ||
|
|
new_options->encryption_key_id != old_options->encryption_key_id) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
}
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& ~(INNOBASE_INPLACE_IGNORE
|
|
| INNOBASE_ALTER_NOREBUILD
|
|
| INNOBASE_ALTER_REBUILD)) {
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& (Alter_inplace_info::ALTER_COLUMN_EQUAL_PACK_LENGTH
|
|
| Alter_inplace_info::ALTER_COLUMN_TYPE))
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_COLUMN_TYPE);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* Only support online add foreign key constraint when
|
|
check_foreigns is turned off */
|
|
if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_FOREIGN_KEY)
|
|
&& prebuilt->trx->check_foreigns) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_FK_CHECK);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
if (!(ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE)) {
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NO_LOCK);
|
|
}
|
|
|
|
/* Only support NULL -> NOT NULL change if strict table sql_mode
|
|
is set. Fall back to COPY for conversion if not strict tables.
|
|
In-Place will fail with an error when trying to convert
|
|
NULL to a NOT NULL value. */
|
|
if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NOT_NULLABLE)
|
|
&& !thd_is_strict_mode(user_thd)) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_NOT_NULL);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* InnoDB cannot IGNORE when creating unique indexes. IGNORE
|
|
should silently delete some duplicate rows. Our inplace_alter
|
|
code will not delete anything from existing indexes. */
|
|
if (ha_alter_info->ignore
|
|
&& (ha_alter_info->handler_flags
|
|
& (Alter_inplace_info::ADD_PK_INDEX
|
|
| Alter_inplace_info::ADD_UNIQUE_INDEX))) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_IGNORE);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* DROP PRIMARY KEY is only allowed in combination with ADD
|
|
PRIMARY KEY. */
|
|
if ((ha_alter_info->handler_flags
|
|
& (Alter_inplace_info::ADD_PK_INDEX
|
|
| Alter_inplace_info::DROP_PK_INDEX))
|
|
== Alter_inplace_info::DROP_PK_INDEX) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_NOPK);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* If a column change from NOT NULL to NULL,
|
|
and there's a implict pk on this column. the
|
|
table should be rebuild. The change should
|
|
only go through the "Copy" method.*/
|
|
if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NULLABLE)) {
|
|
uint primary_key = altered_table->s->primary_key;
|
|
|
|
/* See if MYSQL table has no pk but we do.*/
|
|
if (UNIV_UNLIKELY(primary_key >= MAX_KEY)
|
|
&& !row_table_got_default_clust_index(prebuilt->table)) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_PRIMARY_CANT_HAVE_NULL);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
}
|
|
|
|
/*
|
|
InnoDB in different MariaDB versions was generating different mtype
|
|
codes for certain types. In some cases the signed/unsigned bit was
|
|
generated differently too.
|
|
|
|
Online ALTER would change the mtype/unsigned_flag (to what the
|
|
current code generates) without changing the underlying data
|
|
represenation, and it might result in data corruption.
|
|
|
|
Don't do online ALTER if mtype/unsigned_flag are wrong.
|
|
*/
|
|
for (ulint i = 0, icol= 0; i < table->s->fields; i++) {
|
|
const Field* field = table->field[i];
|
|
const dict_col_t* col = dict_table_get_nth_col(prebuilt->table, icol);
|
|
ulint unsigned_flag;
|
|
if (!field->stored_in_db())
|
|
continue;
|
|
icol++;
|
|
|
|
if (col->mtype != get_innobase_type_from_mysql_type(&unsigned_flag, field)) {
|
|
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
if ((col->prtype & DATA_UNSIGNED) != unsigned_flag) {
|
|
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
}
|
|
|
|
/* If we have column that has changed from NULL -> NOT NULL
|
|
and column default has changed we need to do additional
|
|
check. */
|
|
if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NOT_NULLABLE) &&
|
|
(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_DEFAULT)) {
|
|
Alter_info *alter_info = ha_alter_info->alter_info;
|
|
List_iterator<Create_field> def_it(alter_info->create_list);
|
|
Create_field *def;
|
|
while ((def=def_it++)) {
|
|
|
|
/* If this is first column definition whose SQL type
|
|
is TIMESTAMP and it is defined as NOT NULL and
|
|
it has either constant default or function default
|
|
we must use "Copy" method. */
|
|
if (is_timestamp_type(def->sql_type)) {
|
|
if ((def->flags & NOT_NULL_FLAG) != 0 && // NOT NULL
|
|
(def->default_value != NULL || // constant default ?
|
|
def->unireg_check != Field::NONE)) { // function default
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_NOT_NULL);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
ulint n_indexes = UT_LIST_GET_LEN((prebuilt->table)->indexes);
|
|
|
|
/* If InnoDB dictionary and MySQL frm file are not consistent
|
|
use "Copy" method. */
|
|
if (prebuilt->table->dict_frm_mismatch) {
|
|
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_NO_SUCH_INDEX);
|
|
ib_push_frm_error(user_thd, prebuilt->table, altered_table,
|
|
n_indexes, true);
|
|
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* We should be able to do the operation in-place.
|
|
See if we can do it online (LOCK=NONE). */
|
|
bool online = true;
|
|
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
|
|
/* Fix the key parts. */
|
|
for (KEY* new_key = ha_alter_info->key_info_buffer;
|
|
new_key < ha_alter_info->key_info_buffer
|
|
+ ha_alter_info->key_count;
|
|
new_key++) {
|
|
for (KEY_PART_INFO* key_part = new_key->key_part;
|
|
key_part < new_key->key_part + new_key->user_defined_key_parts;
|
|
key_part++) {
|
|
const Create_field* new_field;
|
|
|
|
DBUG_ASSERT(key_part->fieldnr
|
|
< altered_table->s->fields);
|
|
|
|
cf_it.rewind();
|
|
for (uint fieldnr = 0; (new_field = cf_it++);
|
|
fieldnr++) {
|
|
if (fieldnr == key_part->fieldnr) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
DBUG_ASSERT(new_field);
|
|
|
|
key_part->field = altered_table->field[
|
|
key_part->fieldnr];
|
|
/* In some special cases InnoDB emits "false"
|
|
duplicate key errors with NULL key values. Let
|
|
us play safe and ensure that we can correctly
|
|
print key values even in such cases .*/
|
|
key_part->null_offset = key_part->field->null_offset();
|
|
key_part->null_bit = key_part->field->null_bit;
|
|
|
|
if (new_field->field) {
|
|
/* This is an existing column. */
|
|
continue;
|
|
}
|
|
|
|
/* This is an added column. */
|
|
DBUG_ASSERT(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_COLUMN);
|
|
|
|
/* We cannot replace a hidden FTS_DOC_ID
|
|
with a user-visible FTS_DOC_ID. */
|
|
if (prebuilt->table->fts
|
|
&& innobase_fulltext_exist(altered_table)
|
|
&& !my_strcasecmp(
|
|
system_charset_info,
|
|
key_part->field->field_name,
|
|
FTS_DOC_ID_COL_NAME)) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_HIDDEN_FTS);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
DBUG_ASSERT((MTYP_TYPENR(key_part->field->unireg_check)
|
|
== Field::NEXT_NUMBER)
|
|
== !!(key_part->field->flags
|
|
& AUTO_INCREMENT_FLAG));
|
|
|
|
if (key_part->field->flags & AUTO_INCREMENT_FLAG) {
|
|
/* We cannot assign an AUTO_INCREMENT
|
|
column values during online ALTER. */
|
|
DBUG_ASSERT(key_part->field == altered_table
|
|
-> found_next_number_field);
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_AUTOINC);
|
|
online = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
DBUG_ASSERT(!prebuilt->table->fts || prebuilt->table->fts->doc_col
|
|
<= table->s->stored_fields);
|
|
DBUG_ASSERT(!prebuilt->table->fts || prebuilt->table->fts->doc_col
|
|
< dict_table_get_n_user_cols(prebuilt->table));
|
|
|
|
if (prebuilt->table->fts
|
|
&& innobase_fulltext_exist(altered_table)) {
|
|
/* FULLTEXT indexes are supposed to remain. */
|
|
/* Disallow DROP INDEX FTS_DOC_ID_INDEX */
|
|
|
|
for (uint i = 0; i < ha_alter_info->index_drop_count; i++) {
|
|
if (!my_strcasecmp(
|
|
system_charset_info,
|
|
ha_alter_info->index_drop_buffer[i]->name,
|
|
FTS_DOC_ID_INDEX_NAME)) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_CHANGE_FTS);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
}
|
|
|
|
/* InnoDB can have a hidden FTS_DOC_ID_INDEX on a
|
|
visible FTS_DOC_ID column as well. Prevent dropping or
|
|
renaming the FTS_DOC_ID. */
|
|
|
|
for (Field** fp = table->field; *fp; fp++) {
|
|
if (!((*fp)->flags
|
|
& (FIELD_IS_RENAMED | FIELD_IS_DROPPED))) {
|
|
continue;
|
|
}
|
|
|
|
if (!my_strcasecmp(
|
|
system_charset_info,
|
|
(*fp)->field_name,
|
|
FTS_DOC_ID_COL_NAME)) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_CHANGE_FTS);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
}
|
|
}
|
|
|
|
prebuilt->trx->will_lock++;
|
|
|
|
if (!online) {
|
|
/* We already determined that only a non-locking
|
|
operation is possible. */
|
|
} else if (((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_PK_INDEX)
|
|
|| innobase_need_rebuild(ha_alter_info, table))
|
|
&& (innobase_fulltext_exist(altered_table))) {
|
|
/* Refuse to rebuild the table online, if
|
|
fulltext indexes are to survive the rebuild. */
|
|
online = false;
|
|
/* If the table already contains fulltext indexes,
|
|
refuse to rebuild the table natively altogether. */
|
|
if (prebuilt->table->fts) {
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_INNODB_FT_LIMIT);
|
|
DBUG_RETURN(HA_ALTER_INPLACE_NOT_SUPPORTED);
|
|
}
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_FTS);
|
|
} else if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_INDEX)) {
|
|
/* Building a full-text index requires a lock.
|
|
We could do without a lock if the table already contains
|
|
an FTS_DOC_ID column, but in that case we would have
|
|
to apply the modification log to the full-text indexes. */
|
|
|
|
for (uint i = 0; i < ha_alter_info->index_add_count; i++) {
|
|
const KEY* key =
|
|
&ha_alter_info->key_info_buffer[
|
|
ha_alter_info->index_add_buffer[i]];
|
|
if (key->flags & HA_FULLTEXT) {
|
|
DBUG_ASSERT(!(key->flags & HA_KEYFLAG_MASK
|
|
& ~(HA_FULLTEXT
|
|
| HA_PACK_KEY
|
|
| HA_GENERATED_KEY
|
|
| HA_BINARY_PACK_KEY)));
|
|
ha_alter_info->unsupported_reason = innobase_get_err_msg(
|
|
ER_ALTER_OPERATION_NOT_SUPPORTED_REASON_FTS);
|
|
online = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(online
|
|
? HA_ALTER_INPLACE_NO_LOCK_AFTER_PREPARE
|
|
: HA_ALTER_INPLACE_SHARED_LOCK_AFTER_PREPARE);
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Initialize the dict_foreign_t structure with supplied info
|
|
@return true if added, false if duplicate foreign->id */
|
|
static MY_ATTRIBUTE((nonnull(1,3,5,7)))
|
|
bool
|
|
innobase_init_foreign(
|
|
/*==================*/
|
|
dict_foreign_t* foreign, /*!< in/out: structure to
|
|
initialize */
|
|
char* constraint_name, /*!< in/out: constraint name if
|
|
exists */
|
|
dict_table_t* table, /*!< in: foreign table */
|
|
dict_index_t* index, /*!< in: foreign key index */
|
|
const char** column_names, /*!< in: foreign key column
|
|
names */
|
|
ulint num_field, /*!< in: number of columns */
|
|
const char* referenced_table_name, /*!< in: referenced table
|
|
name */
|
|
dict_table_t* referenced_table, /*!< in: referenced table */
|
|
dict_index_t* referenced_index, /*!< in: referenced index */
|
|
const char** referenced_column_names,/*!< in: referenced column
|
|
names */
|
|
ulint referenced_num_field) /*!< in: number of referenced
|
|
columns */
|
|
{
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
|
|
if (constraint_name) {
|
|
ulint db_len;
|
|
|
|
/* Catenate 'databasename/' to the constraint name specified
|
|
by the user: we conceive the constraint as belonging to the
|
|
same MySQL 'database' as the table itself. We store the name
|
|
to foreign->id. */
|
|
|
|
db_len = dict_get_db_name_len(table->name);
|
|
|
|
foreign->id = static_cast<char*>(mem_heap_alloc(
|
|
foreign->heap, db_len + strlen(constraint_name) + 2));
|
|
|
|
ut_memcpy(foreign->id, table->name, db_len);
|
|
foreign->id[db_len] = '/';
|
|
strcpy(foreign->id + db_len + 1, constraint_name);
|
|
|
|
/* Check if any existing foreign key has the same id,
|
|
this is needed only if user supplies the constraint name */
|
|
|
|
if (table->foreign_set.find(foreign)
|
|
!= table->foreign_set.end()) {
|
|
return(false);
|
|
}
|
|
}
|
|
|
|
foreign->foreign_table = table;
|
|
foreign->foreign_table_name = mem_heap_strdup(
|
|
foreign->heap, table->name);
|
|
dict_mem_foreign_table_name_lookup_set(foreign, TRUE);
|
|
|
|
foreign->foreign_index = index;
|
|
foreign->n_fields = (unsigned int) num_field;
|
|
|
|
foreign->foreign_col_names = static_cast<const char**>(
|
|
mem_heap_alloc(foreign->heap, num_field * sizeof(void*)));
|
|
|
|
for (ulint i = 0; i < foreign->n_fields; i++) {
|
|
foreign->foreign_col_names[i] = mem_heap_strdup(
|
|
foreign->heap, column_names[i]);
|
|
}
|
|
|
|
foreign->referenced_index = referenced_index;
|
|
foreign->referenced_table = referenced_table;
|
|
|
|
foreign->referenced_table_name = mem_heap_strdup(
|
|
foreign->heap, referenced_table_name);
|
|
dict_mem_referenced_table_name_lookup_set(foreign, TRUE);
|
|
|
|
foreign->referenced_col_names = static_cast<const char**>(
|
|
mem_heap_alloc(foreign->heap,
|
|
referenced_num_field * sizeof(void*)));
|
|
|
|
for (ulint i = 0; i < foreign->n_fields; i++) {
|
|
foreign->referenced_col_names[i]
|
|
= mem_heap_strdup(foreign->heap,
|
|
referenced_column_names[i]);
|
|
}
|
|
|
|
return(true);
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Check whether the foreign key options is legit
|
|
@return true if it is */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_check_fk_option(
|
|
/*=====================*/
|
|
const dict_foreign_t* foreign) /*!< in: foreign key */
|
|
{
|
|
if (!foreign->foreign_index) {
|
|
return(true);
|
|
}
|
|
|
|
if (foreign->type & (DICT_FOREIGN_ON_UPDATE_SET_NULL
|
|
| DICT_FOREIGN_ON_DELETE_SET_NULL)) {
|
|
|
|
for (ulint j = 0; j < foreign->n_fields; j++) {
|
|
if ((dict_index_get_nth_col(
|
|
foreign->foreign_index, j)->prtype)
|
|
& DATA_NOT_NULL) {
|
|
|
|
/* It is not sensible to define
|
|
SET NULL if the column is not
|
|
allowed to be NULL! */
|
|
return(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
return(true);
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Set foreign key options
|
|
@return true if successfully set */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_set_foreign_key_option(
|
|
/*============================*/
|
|
dict_foreign_t* foreign, /*!< in:InnoDB Foreign key */
|
|
Foreign_key* fk_key) /*!< in: Foreign key info from
|
|
MySQL */
|
|
{
|
|
ut_ad(!foreign->type);
|
|
|
|
switch (fk_key->delete_opt) {
|
|
case Foreign_key::FK_OPTION_NO_ACTION:
|
|
case Foreign_key::FK_OPTION_RESTRICT:
|
|
case Foreign_key::FK_OPTION_DEFAULT:
|
|
foreign->type = DICT_FOREIGN_ON_DELETE_NO_ACTION;
|
|
break;
|
|
case Foreign_key::FK_OPTION_CASCADE:
|
|
foreign->type = DICT_FOREIGN_ON_DELETE_CASCADE;
|
|
break;
|
|
case Foreign_key::FK_OPTION_SET_NULL:
|
|
foreign->type = DICT_FOREIGN_ON_DELETE_SET_NULL;
|
|
break;
|
|
}
|
|
|
|
switch (fk_key->update_opt) {
|
|
case Foreign_key::FK_OPTION_NO_ACTION:
|
|
case Foreign_key::FK_OPTION_RESTRICT:
|
|
case Foreign_key::FK_OPTION_DEFAULT:
|
|
foreign->type |= DICT_FOREIGN_ON_UPDATE_NO_ACTION;
|
|
break;
|
|
case Foreign_key::FK_OPTION_CASCADE:
|
|
foreign->type |= DICT_FOREIGN_ON_UPDATE_CASCADE;
|
|
break;
|
|
case Foreign_key::FK_OPTION_SET_NULL:
|
|
foreign->type |= DICT_FOREIGN_ON_UPDATE_SET_NULL;
|
|
break;
|
|
}
|
|
|
|
return(innobase_check_fk_option(foreign));
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Check if a foreign key constraint can make use of an index
|
|
that is being created.
|
|
@return useable index, or NULL if none found */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
const KEY*
|
|
innobase_find_equiv_index(
|
|
/*======================*/
|
|
const char*const* col_names,
|
|
/*!< in: column names */
|
|
uint n_cols, /*!< in: number of columns */
|
|
const KEY* keys, /*!< in: index information */
|
|
const uint* add, /*!< in: indexes being created */
|
|
uint n_add) /*!< in: number of indexes to create */
|
|
{
|
|
for (uint i = 0; i < n_add; i++) {
|
|
const KEY* key = &keys[add[i]];
|
|
|
|
if (key->user_defined_key_parts < n_cols) {
|
|
no_match:
|
|
continue;
|
|
}
|
|
|
|
for (uint j = 0; j < n_cols; j++) {
|
|
const KEY_PART_INFO& key_part = key->key_part[j];
|
|
uint32 col_len
|
|
= key_part.field->pack_length();
|
|
|
|
/* The MySQL pack length contains 1 or 2 bytes
|
|
length field for a true VARCHAR. */
|
|
|
|
if (key_part.field->type() == MYSQL_TYPE_VARCHAR) {
|
|
col_len -= static_cast<const Field_varstring*>(
|
|
key_part.field)->length_bytes;
|
|
}
|
|
|
|
if (key_part.length < col_len) {
|
|
|
|
/* Column prefix indexes cannot be
|
|
used for FOREIGN KEY constraints. */
|
|
goto no_match;
|
|
}
|
|
|
|
if (innobase_strcasecmp(col_names[j],
|
|
key_part.field->field_name)) {
|
|
/* Name mismatch */
|
|
goto no_match;
|
|
}
|
|
}
|
|
|
|
return(key);
|
|
}
|
|
|
|
return(NULL);
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Find an index whose first fields are the columns in the array
|
|
in the same order and is not marked for deletion
|
|
@return matching index, NULL if not found */
|
|
static MY_ATTRIBUTE((nonnull(1,2,6), warn_unused_result))
|
|
dict_index_t*
|
|
innobase_find_fk_index(
|
|
/*===================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
/*!< in: alter table info */
|
|
dict_table_t* table, /*!< in: table */
|
|
const char** col_names,
|
|
/*!< in: column names, or NULL
|
|
to use table->col_names */
|
|
dict_index_t** drop_index,
|
|
/*!< in: indexes to be dropped */
|
|
ulint n_drop_index,
|
|
/*!< in: size of drop_index[] */
|
|
const char** columns,/*!< in: array of column names */
|
|
ulint n_cols) /*!< in: number of columns */
|
|
{
|
|
dict_index_t* index;
|
|
|
|
index = dict_table_get_first_index(table);
|
|
|
|
while (index != NULL) {
|
|
if (!(index->type & DICT_FTS)
|
|
&& dict_foreign_qualify_index(
|
|
table, col_names, columns, n_cols,
|
|
index, NULL, true, 0, NULL, NULL, NULL)) {
|
|
for (ulint i = 0; i < n_drop_index; i++) {
|
|
if (index == drop_index[i]) {
|
|
/* Skip to-be-dropped indexes. */
|
|
goto next_rec;
|
|
}
|
|
}
|
|
|
|
return(index);
|
|
}
|
|
|
|
next_rec:
|
|
index = dict_table_get_next_index(index);
|
|
}
|
|
|
|
return(NULL);
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Create InnoDB foreign key structure from MySQL alter_info
|
|
@retval true if successful
|
|
@retval false on error (will call my_error()) */
|
|
static MY_ATTRIBUTE((nonnull(1,2,3,7,8), warn_unused_result))
|
|
bool
|
|
innobase_get_foreign_key_info(
|
|
/*==========================*/
|
|
Alter_inplace_info*
|
|
ha_alter_info, /*!< in: alter table info */
|
|
const TABLE_SHARE*
|
|
table_share, /*!< in: the TABLE_SHARE */
|
|
dict_table_t* table, /*!< in: table */
|
|
const char** col_names, /*!< in: column names, or NULL
|
|
to use table->col_names */
|
|
dict_index_t** drop_index, /*!< in: indexes to be dropped */
|
|
ulint n_drop_index, /*!< in: size of drop_index[] */
|
|
dict_foreign_t**add_fk, /*!< out: foreign constraint added */
|
|
ulint* n_add_fk, /*!< out: number of foreign
|
|
constraints added */
|
|
const trx_t* trx) /*!< in: user transaction */
|
|
{
|
|
Key* key;
|
|
Foreign_key* fk_key;
|
|
dict_table_t* referenced_table = NULL;
|
|
char* referenced_table_name = NULL;
|
|
ulint num_fk = 0;
|
|
Alter_info* alter_info = ha_alter_info->alter_info;
|
|
|
|
*n_add_fk = 0;
|
|
|
|
List_iterator<Key> key_iterator(alter_info->key_list);
|
|
|
|
while ((key=key_iterator++)) {
|
|
if (key->type != Key::FOREIGN_KEY) {
|
|
continue;
|
|
}
|
|
|
|
const char* column_names[MAX_NUM_FK_COLUMNS];
|
|
dict_index_t* index = NULL;
|
|
const char* referenced_column_names[MAX_NUM_FK_COLUMNS];
|
|
dict_index_t* referenced_index = NULL;
|
|
ulint num_col = 0;
|
|
ulint referenced_num_col = 0;
|
|
bool correct_option;
|
|
char* db_namep = NULL;
|
|
char* tbl_namep = NULL;
|
|
ulint db_name_len = 0;
|
|
ulint tbl_name_len = 0;
|
|
char db_name[MAX_DATABASE_NAME_LEN];
|
|
char tbl_name[MAX_TABLE_NAME_LEN];
|
|
|
|
fk_key = static_cast<Foreign_key*>(key);
|
|
|
|
if (fk_key->columns.elements > 0) {
|
|
ulint i = 0;
|
|
Key_part_spec* column;
|
|
List_iterator<Key_part_spec> key_part_iterator(
|
|
fk_key->columns);
|
|
|
|
/* Get all the foreign key column info for the
|
|
current table */
|
|
while ((column = key_part_iterator++)) {
|
|
column_names[i] = column->field_name.str;
|
|
ut_ad(i < MAX_NUM_FK_COLUMNS);
|
|
i++;
|
|
}
|
|
|
|
index = innobase_find_fk_index(
|
|
ha_alter_info,
|
|
table, col_names,
|
|
drop_index, n_drop_index,
|
|
column_names, i);
|
|
|
|
/* MySQL would add a index in the creation
|
|
list if no such index for foreign table,
|
|
so we have to use DBUG_EXECUTE_IF to simulate
|
|
the scenario */
|
|
DBUG_EXECUTE_IF("innodb_test_no_foreign_idx",
|
|
index = NULL;);
|
|
|
|
/* Check whether there exist such
|
|
index in the the index create clause */
|
|
if (!index && !innobase_find_equiv_index(
|
|
column_names, static_cast<uint>(i),
|
|
ha_alter_info->key_info_buffer,
|
|
ha_alter_info->index_add_buffer,
|
|
ha_alter_info->index_add_count)) {
|
|
my_error(
|
|
ER_FK_NO_INDEX_CHILD,
|
|
MYF(0),
|
|
fk_key->name.str
|
|
? fk_key->name.str : "",
|
|
table_share->table_name.str);
|
|
goto err_exit;
|
|
}
|
|
|
|
num_col = i;
|
|
}
|
|
|
|
add_fk[num_fk] = dict_mem_foreign_create();
|
|
|
|
#ifndef __WIN__
|
|
if(fk_key->ref_db.str) {
|
|
tablename_to_filename(fk_key->ref_db.str, db_name,
|
|
MAX_DATABASE_NAME_LEN);
|
|
db_namep = db_name;
|
|
db_name_len = strlen(db_name);
|
|
}
|
|
if (fk_key->ref_table.str) {
|
|
tablename_to_filename(fk_key->ref_table.str, tbl_name,
|
|
MAX_TABLE_NAME_LEN);
|
|
tbl_namep = tbl_name;
|
|
tbl_name_len = strlen(tbl_name);
|
|
}
|
|
#else
|
|
ut_ad(fk_key->ref_table.str);
|
|
tablename_to_filename(fk_key->ref_table.str, tbl_name,
|
|
MAX_TABLE_NAME_LEN);
|
|
innobase_casedn_str(tbl_name);
|
|
tbl_name_len = strlen(tbl_name);
|
|
tbl_namep = &tbl_name[0];
|
|
|
|
if (fk_key->ref_db.str != NULL) {
|
|
tablename_to_filename(fk_key->ref_db.str, db_name,
|
|
MAX_DATABASE_NAME_LEN);
|
|
innobase_casedn_str(db_name);
|
|
db_name_len = strlen(db_name);
|
|
db_namep = &db_name[0];
|
|
}
|
|
#endif
|
|
mutex_enter(&dict_sys->mutex);
|
|
|
|
referenced_table_name = dict_get_referenced_table(
|
|
table->name,
|
|
db_namep,
|
|
db_name_len,
|
|
tbl_namep,
|
|
tbl_name_len,
|
|
&referenced_table,
|
|
add_fk[num_fk]->heap);
|
|
|
|
/* Test the case when referenced_table failed to
|
|
open, if trx->check_foreigns is not set, we should
|
|
still be able to add the foreign key */
|
|
DBUG_EXECUTE_IF("innodb_test_open_ref_fail",
|
|
referenced_table = NULL;);
|
|
|
|
if (!referenced_table && trx->check_foreigns) {
|
|
mutex_exit(&dict_sys->mutex);
|
|
my_error(ER_FK_CANNOT_OPEN_PARENT,
|
|
MYF(0), tbl_namep);
|
|
|
|
goto err_exit;
|
|
}
|
|
|
|
if (fk_key->ref_columns.elements > 0) {
|
|
ulint i = 0;
|
|
Key_part_spec* column;
|
|
List_iterator<Key_part_spec> key_part_iterator(
|
|
fk_key->ref_columns);
|
|
|
|
while ((column = key_part_iterator++)) {
|
|
referenced_column_names[i] =
|
|
column->field_name.str;
|
|
ut_ad(i < MAX_NUM_FK_COLUMNS);
|
|
i++;
|
|
}
|
|
|
|
if (referenced_table) {
|
|
referenced_index =
|
|
dict_foreign_find_index(
|
|
referenced_table, 0,
|
|
referenced_column_names,
|
|
i, index,
|
|
TRUE, FALSE, NULL, NULL, NULL);
|
|
|
|
DBUG_EXECUTE_IF(
|
|
"innodb_test_no_reference_idx",
|
|
referenced_index = NULL;);
|
|
|
|
/* Check whether there exist such
|
|
index in the the index create clause */
|
|
if (!referenced_index) {
|
|
mutex_exit(&dict_sys->mutex);
|
|
my_error(ER_FK_NO_INDEX_PARENT, MYF(0),
|
|
fk_key->name.str
|
|
? fk_key->name.str : "",
|
|
tbl_namep);
|
|
goto err_exit;
|
|
}
|
|
} else {
|
|
ut_a(!trx->check_foreigns);
|
|
}
|
|
|
|
referenced_num_col = i;
|
|
} else {
|
|
/* Not possible to add a foreign key without a
|
|
referenced column */
|
|
mutex_exit(&dict_sys->mutex);
|
|
my_error(ER_CANNOT_ADD_FOREIGN, MYF(0), tbl_namep);
|
|
goto err_exit;
|
|
}
|
|
|
|
if (!innobase_init_foreign(
|
|
add_fk[num_fk], fk_key->name.str,
|
|
table, index, column_names,
|
|
num_col, referenced_table_name,
|
|
referenced_table, referenced_index,
|
|
referenced_column_names, referenced_num_col)) {
|
|
mutex_exit(&dict_sys->mutex);
|
|
my_error(
|
|
ER_DUP_CONSTRAINT_NAME,
|
|
MYF(0),
|
|
"FOREIGN KEY", add_fk[num_fk]->id);
|
|
goto err_exit;
|
|
}
|
|
|
|
mutex_exit(&dict_sys->mutex);
|
|
|
|
correct_option = innobase_set_foreign_key_option(
|
|
add_fk[num_fk], fk_key);
|
|
|
|
DBUG_EXECUTE_IF("innodb_test_wrong_fk_option",
|
|
correct_option = false;);
|
|
|
|
if (!correct_option) {
|
|
my_error(ER_FK_INCORRECT_OPTION,
|
|
MYF(0),
|
|
table_share->table_name.str,
|
|
add_fk[num_fk]->id);
|
|
goto err_exit;
|
|
}
|
|
|
|
num_fk++;
|
|
}
|
|
|
|
*n_add_fk = num_fk;
|
|
|
|
return(true);
|
|
err_exit:
|
|
for (ulint i = 0; i <= num_fk; i++) {
|
|
if (add_fk[i]) {
|
|
dict_foreign_free(add_fk[i]);
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Copies an InnoDB column to a MySQL field. This function is
|
|
adapted from row_sel_field_store_in_mysql_format(). */
|
|
static
|
|
void
|
|
innobase_col_to_mysql(
|
|
/*==================*/
|
|
const dict_col_t* col, /*!< in: InnoDB column */
|
|
const uchar* data, /*!< in: InnoDB column data */
|
|
ulint len, /*!< in: length of data, in bytes */
|
|
Field* field) /*!< in/out: MySQL field */
|
|
{
|
|
uchar* ptr;
|
|
uchar* dest = field->ptr;
|
|
ulint flen = field->pack_length();
|
|
|
|
switch (col->mtype) {
|
|
case DATA_INT:
|
|
ut_ad(len == flen);
|
|
|
|
/* Convert integer data from Innobase to little-endian
|
|
format, sign bit restored to normal */
|
|
|
|
for (ptr = dest + len; ptr != dest; ) {
|
|
*--ptr = *data++;
|
|
}
|
|
|
|
if (!(field->flags & UNSIGNED_FLAG)) {
|
|
((byte*) dest)[len - 1] ^= 0x80;
|
|
}
|
|
|
|
break;
|
|
|
|
case DATA_VARCHAR:
|
|
case DATA_VARMYSQL:
|
|
case DATA_BINARY:
|
|
field->reset();
|
|
|
|
if (field->type() == MYSQL_TYPE_VARCHAR) {
|
|
/* This is a >= 5.0.3 type true VARCHAR. Store the
|
|
length of the data to the first byte or the first
|
|
two bytes of dest. */
|
|
|
|
dest = row_mysql_store_true_var_len(
|
|
dest, len, flen - field->key_length());
|
|
}
|
|
|
|
/* Copy the actual data */
|
|
memcpy(dest, data, len);
|
|
break;
|
|
|
|
case DATA_BLOB:
|
|
/* Skip MySQL BLOBs when reporting an erroneous row
|
|
during index creation or table rebuild. */
|
|
field->set_null();
|
|
break;
|
|
|
|
#ifdef UNIV_DEBUG
|
|
case DATA_MYSQL:
|
|
ut_ad(flen >= len);
|
|
ut_ad(DATA_MBMAXLEN(col->mbminmaxlen)
|
|
>= DATA_MBMINLEN(col->mbminmaxlen));
|
|
memcpy(dest, data, len);
|
|
break;
|
|
|
|
default:
|
|
case DATA_SYS_CHILD:
|
|
case DATA_SYS:
|
|
/* These column types should never be shipped to MySQL. */
|
|
ut_ad(0);
|
|
|
|
case DATA_FLOAT:
|
|
case DATA_DOUBLE:
|
|
case DATA_DECIMAL:
|
|
/* Above are the valid column types for MySQL data. */
|
|
ut_ad(flen == len);
|
|
/* fall through */
|
|
case DATA_FIXBINARY:
|
|
case DATA_CHAR:
|
|
/* We may have flen > len when there is a shorter
|
|
prefix on the CHAR and BINARY column. */
|
|
ut_ad(flen >= len);
|
|
#else /* UNIV_DEBUG */
|
|
default:
|
|
#endif /* UNIV_DEBUG */
|
|
memcpy(dest, data, len);
|
|
}
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Copies an InnoDB record to table->record[0]. */
|
|
UNIV_INTERN
|
|
void
|
|
innobase_rec_to_mysql(
|
|
/*==================*/
|
|
struct TABLE* table, /*!< in/out: MySQL table */
|
|
const rec_t* rec, /*!< in: record */
|
|
const dict_index_t* index, /*!< in: index */
|
|
const ulint* offsets)/*!< in: rec_get_offsets(
|
|
rec, index, ...) */
|
|
{
|
|
uint n_fields = table->s->stored_fields;
|
|
uint sql_idx = 0;
|
|
|
|
ut_ad(n_fields == dict_table_get_n_user_cols(index->table)
|
|
- !!(DICT_TF2_FLAG_IS_SET(index->table,
|
|
DICT_TF2_FTS_HAS_DOC_ID)));
|
|
|
|
for (uint i = 0; i < n_fields; i++, sql_idx++) {
|
|
Field* field;
|
|
ulint ipos;
|
|
ulint ilen;
|
|
const uchar* ifield;
|
|
|
|
while (!((field= table->field[sql_idx])->stored_in_db()))
|
|
sql_idx++;
|
|
|
|
field->reset();
|
|
|
|
ipos = dict_index_get_nth_col_or_prefix_pos(index, i, TRUE,
|
|
NULL);
|
|
|
|
if (ipos == ULINT_UNDEFINED
|
|
|| rec_offs_nth_extern(offsets, ipos)) {
|
|
null_field:
|
|
field->set_null();
|
|
continue;
|
|
}
|
|
|
|
ifield = rec_get_nth_field(rec, offsets, ipos, &ilen);
|
|
|
|
/* Assign the NULL flag */
|
|
if (ilen == UNIV_SQL_NULL) {
|
|
ut_ad(field->real_maybe_null());
|
|
goto null_field;
|
|
}
|
|
|
|
field->set_notnull();
|
|
|
|
innobase_col_to_mysql(
|
|
dict_field_get_col(
|
|
dict_index_get_nth_field(index, ipos)),
|
|
ifield, ilen, field);
|
|
}
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Copies an InnoDB index entry to table->record[0]. */
|
|
UNIV_INTERN
|
|
void
|
|
innobase_fields_to_mysql(
|
|
/*=====================*/
|
|
struct TABLE* table, /*!< in/out: MySQL table */
|
|
const dict_index_t* index, /*!< in: InnoDB index */
|
|
const dfield_t* fields) /*!< in: InnoDB index fields */
|
|
{
|
|
uint n_fields = table->s->stored_fields;
|
|
uint sql_idx = 0;
|
|
|
|
ut_ad(n_fields == dict_table_get_n_user_cols(index->table)
|
|
- !!(DICT_TF2_FLAG_IS_SET(index->table,
|
|
DICT_TF2_FTS_HAS_DOC_ID)));
|
|
|
|
for (uint i = 0; i < n_fields; i++, sql_idx++) {
|
|
Field* field;
|
|
ulint ipos;
|
|
|
|
while (!((field= table->field[sql_idx])->stored_in_db()))
|
|
sql_idx++;
|
|
|
|
field->reset();
|
|
|
|
ipos = dict_index_get_nth_col_or_prefix_pos(index, i, TRUE,
|
|
NULL);
|
|
|
|
if (ipos == ULINT_UNDEFINED
|
|
|| dfield_is_ext(&fields[ipos])
|
|
|| dfield_is_null(&fields[ipos])) {
|
|
|
|
field->set_null();
|
|
} else {
|
|
field->set_notnull();
|
|
|
|
const dfield_t* df = &fields[ipos];
|
|
|
|
innobase_col_to_mysql(
|
|
dict_field_get_col(
|
|
dict_index_get_nth_field(index, ipos)),
|
|
static_cast<const uchar*>(dfield_get_data(df)),
|
|
dfield_get_len(df), field);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Copies an InnoDB row to table->record[0]. */
|
|
UNIV_INTERN
|
|
void
|
|
innobase_row_to_mysql(
|
|
/*==================*/
|
|
struct TABLE* table, /*!< in/out: MySQL table */
|
|
const dict_table_t* itab, /*!< in: InnoDB table */
|
|
const dtuple_t* row) /*!< in: InnoDB row */
|
|
{
|
|
uint n_fields = table->s->stored_fields;
|
|
uint sql_idx = 0;
|
|
|
|
/* The InnoDB row may contain an extra FTS_DOC_ID column at the end. */
|
|
ut_ad(row->n_fields == dict_table_get_n_cols(itab));
|
|
ut_ad(n_fields == row->n_fields - DATA_N_SYS_COLS
|
|
- !!(DICT_TF2_FLAG_IS_SET(itab, DICT_TF2_FTS_HAS_DOC_ID)));
|
|
|
|
for (uint i = 0; i < n_fields; i++, sql_idx++) {
|
|
Field* field;
|
|
const dfield_t* df = dtuple_get_nth_field(row, i);
|
|
|
|
while (!((field= table->field[sql_idx])->stored_in_db()))
|
|
sql_idx++;
|
|
|
|
field->reset();
|
|
|
|
if (dfield_is_ext(df) || dfield_is_null(df)) {
|
|
field->set_null();
|
|
} else {
|
|
field->set_notnull();
|
|
|
|
innobase_col_to_mysql(
|
|
dict_table_get_nth_col(itab, i),
|
|
static_cast<const uchar*>(dfield_get_data(df)),
|
|
dfield_get_len(df), field);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************//**
|
|
Resets table->record[0]. */
|
|
UNIV_INTERN
|
|
void
|
|
innobase_rec_reset(
|
|
/*===============*/
|
|
TABLE* table) /*!< in/out: MySQL table */
|
|
{
|
|
uint n_fields = table->s->fields;
|
|
uint i;
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
table->field[i]->set_default();
|
|
}
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
This function checks that index keys are sensible.
|
|
@return 0 or error number */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
int
|
|
innobase_check_index_keys(
|
|
/*======================*/
|
|
const Alter_inplace_info* info,
|
|
/*!< in: indexes to be created or dropped */
|
|
const dict_table_t* innodb_table)
|
|
/*!< in: Existing indexes */
|
|
{
|
|
for (uint key_num = 0; key_num < info->index_add_count;
|
|
key_num++) {
|
|
const KEY& key = info->key_info_buffer[
|
|
info->index_add_buffer[key_num]];
|
|
|
|
/* Check that the same index name does not appear
|
|
twice in indexes to be created. */
|
|
|
|
for (ulint i = 0; i < key_num; i++) {
|
|
const KEY& key2 = info->key_info_buffer[
|
|
info->index_add_buffer[i]];
|
|
|
|
if (0 == strcmp(key.name, key2.name)) {
|
|
my_error(ER_WRONG_NAME_FOR_INDEX, MYF(0),
|
|
key.name);
|
|
|
|
return(ER_WRONG_NAME_FOR_INDEX);
|
|
}
|
|
}
|
|
|
|
/* Check that the same index name does not already exist. */
|
|
|
|
const dict_index_t* index;
|
|
|
|
for (index = dict_table_get_first_index(innodb_table);
|
|
index; index = dict_table_get_next_index(index)) {
|
|
|
|
if (!strcmp(key.name, index->name)) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (index) {
|
|
/* If a key by the same name is being created and
|
|
dropped, the name clash is OK. */
|
|
for (uint i = 0; i < info->index_drop_count;
|
|
i++) {
|
|
const KEY* drop_key
|
|
= info->index_drop_buffer[i];
|
|
|
|
if (0 == strcmp(key.name, drop_key->name)) {
|
|
goto name_ok;
|
|
}
|
|
}
|
|
|
|
my_error(ER_WRONG_NAME_FOR_INDEX, MYF(0), key.name);
|
|
|
|
return(ER_WRONG_NAME_FOR_INDEX);
|
|
}
|
|
|
|
name_ok:
|
|
for (ulint i = 0; i < key.user_defined_key_parts; i++) {
|
|
const KEY_PART_INFO& key_part1
|
|
= key.key_part[i];
|
|
const Field* field
|
|
= key_part1.field;
|
|
ibool is_unsigned;
|
|
|
|
switch (get_innobase_type_from_mysql_type(
|
|
&is_unsigned, field)) {
|
|
default:
|
|
break;
|
|
case DATA_INT:
|
|
case DATA_FLOAT:
|
|
case DATA_DOUBLE:
|
|
case DATA_DECIMAL:
|
|
/* Check that MySQL does not try to
|
|
create a column prefix index field on
|
|
an inappropriate data type. */
|
|
|
|
if (field->type() == MYSQL_TYPE_VARCHAR) {
|
|
if (key_part1.length
|
|
>= field->pack_length()
|
|
- ((Field_varstring*) field)
|
|
->length_bytes) {
|
|
break;
|
|
}
|
|
} else {
|
|
if (key_part1.length
|
|
>= field->pack_length()) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
my_error(ER_WRONG_KEY_COLUMN, MYF(0), "InnoDB",
|
|
field->field_name);
|
|
return(ER_WRONG_KEY_COLUMN);
|
|
}
|
|
|
|
/* Check that the same column does not appear
|
|
twice in the index. */
|
|
|
|
for (ulint j = 0; j < i; j++) {
|
|
const KEY_PART_INFO& key_part2
|
|
= key.key_part[j];
|
|
|
|
if (key_part1.fieldnr != key_part2.fieldnr) {
|
|
continue;
|
|
}
|
|
|
|
my_error(ER_WRONG_KEY_COLUMN, MYF(0), "InnoDB",
|
|
field->field_name);
|
|
return(ER_WRONG_KEY_COLUMN);
|
|
}
|
|
}
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Create index field definition for key part */
|
|
static MY_ATTRIBUTE((nonnull(2,3)))
|
|
void
|
|
innobase_create_index_field_def(
|
|
/*============================*/
|
|
const TABLE* altered_table, /*!< in: MySQL table that is
|
|
being altered, or NULL
|
|
if a new clustered index is
|
|
not being created */
|
|
const KEY_PART_INFO* key_part, /*!< in: MySQL key definition */
|
|
index_field_t* index_field, /*!< out: index field
|
|
definition for key_part */
|
|
const Field** fields) /*!< in: MySQL table fields */
|
|
{
|
|
const Field* field;
|
|
ibool is_unsigned;
|
|
ulint col_type;
|
|
|
|
DBUG_ENTER("innobase_create_index_field_def");
|
|
|
|
ut_ad(key_part);
|
|
ut_ad(index_field);
|
|
|
|
field = altered_table
|
|
? altered_table->field[key_part->fieldnr]
|
|
: key_part->field;
|
|
ut_a(field);
|
|
|
|
index_field->col_no = key_part->fieldnr;
|
|
index_field->col_name = altered_table ? field->field_name : fields[key_part->fieldnr]->field_name;
|
|
|
|
col_type = get_innobase_type_from_mysql_type(&is_unsigned, field);
|
|
|
|
if (DATA_BLOB == col_type
|
|
|| (key_part->length < field->pack_length()
|
|
&& field->type() != MYSQL_TYPE_VARCHAR)
|
|
|| (field->type() == MYSQL_TYPE_VARCHAR
|
|
&& key_part->length < field->pack_length()
|
|
- ((Field_varstring*) field)->length_bytes)) {
|
|
|
|
index_field->prefix_len = key_part->length;
|
|
} else {
|
|
index_field->prefix_len = 0;
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Create index definition for key */
|
|
static MY_ATTRIBUTE((nonnull))
|
|
void
|
|
innobase_create_index_def(
|
|
/*======================*/
|
|
const TABLE* altered_table, /*!< in: MySQL table that is
|
|
being altered */
|
|
const KEY* keys, /*!< in: key definitions */
|
|
ulint key_number, /*!< in: MySQL key number */
|
|
bool new_clustered, /*!< in: true if generating
|
|
a new clustered index
|
|
on the table */
|
|
bool key_clustered, /*!< in: true if this is
|
|
the new clustered index */
|
|
index_def_t* index, /*!< out: index definition */
|
|
mem_heap_t* heap, /*!< in: heap where memory
|
|
is allocated */
|
|
const Field** fields) /*!< in: MySQL table fields
|
|
*/
|
|
{
|
|
const KEY* key = &keys[key_number];
|
|
ulint i;
|
|
ulint len;
|
|
ulint n_fields = key->user_defined_key_parts;
|
|
char* index_name;
|
|
|
|
DBUG_ENTER("innobase_create_index_def");
|
|
DBUG_ASSERT(!key_clustered || new_clustered);
|
|
|
|
index->fields = static_cast<index_field_t*>(
|
|
mem_heap_alloc(heap, n_fields * sizeof *index->fields));
|
|
|
|
memset(index->fields, 0, n_fields * sizeof *index->fields);
|
|
|
|
index->ind_type = 0;
|
|
index->key_number = key_number;
|
|
index->n_fields = n_fields;
|
|
len = strlen(key->name) + 1;
|
|
index->name = index_name = static_cast<char*>(
|
|
mem_heap_alloc(heap, len + !new_clustered));
|
|
|
|
if (!new_clustered) {
|
|
*index_name++ = TEMP_INDEX_PREFIX;
|
|
}
|
|
|
|
memcpy(index_name, key->name, len);
|
|
|
|
if (key->flags & HA_NOSAME) {
|
|
index->ind_type |= DICT_UNIQUE;
|
|
}
|
|
|
|
if (key_clustered) {
|
|
DBUG_ASSERT(!(key->flags & HA_FULLTEXT));
|
|
index->ind_type |= DICT_CLUSTERED;
|
|
} else if (key->flags & HA_FULLTEXT) {
|
|
DBUG_ASSERT(!(key->flags & HA_KEYFLAG_MASK
|
|
& ~(HA_FULLTEXT
|
|
| HA_PACK_KEY
|
|
| HA_BINARY_PACK_KEY)));
|
|
DBUG_ASSERT(!(key->flags & HA_NOSAME));
|
|
DBUG_ASSERT(!index->ind_type);
|
|
index->ind_type |= DICT_FTS;
|
|
}
|
|
|
|
if (!new_clustered) {
|
|
altered_table = NULL;
|
|
}
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
innobase_create_index_field_def(
|
|
altered_table, &key->key_part[i], &index->fields[i], fields);
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Check whether the table has the FTS_DOC_ID column
|
|
@return whether there exists an FTS_DOC_ID column */
|
|
static
|
|
bool
|
|
innobase_fts_check_doc_id_col(
|
|
/*==========================*/
|
|
const dict_table_t* table, /*!< in: InnoDB table with
|
|
fulltext index */
|
|
const TABLE* altered_table,
|
|
/*!< in: MySQL table with
|
|
fulltext index */
|
|
ulint* fts_doc_col_no)
|
|
/*!< out: The column number for
|
|
Doc ID, or ULINT_UNDEFINED
|
|
if it is of wrong type */
|
|
{
|
|
*fts_doc_col_no = ULINT_UNDEFINED;
|
|
|
|
const uint n_cols = altered_table->s->stored_fields;
|
|
uint sql_idx = 0;
|
|
uint i;
|
|
|
|
for (i = 0; i < n_cols; i++, sql_idx++) {
|
|
const Field* field;
|
|
while (!((field= altered_table->field[sql_idx])->
|
|
stored_in_db()))
|
|
sql_idx++;
|
|
if (my_strcasecmp(system_charset_info,
|
|
field->field_name, FTS_DOC_ID_COL_NAME)) {
|
|
continue;
|
|
}
|
|
|
|
if (strcmp(field->field_name, FTS_DOC_ID_COL_NAME)) {
|
|
my_error(ER_WRONG_COLUMN_NAME, MYF(0),
|
|
field->field_name);
|
|
} else if (field->type() != MYSQL_TYPE_LONGLONG
|
|
|| field->pack_length() != 8
|
|
|| field->real_maybe_null()
|
|
|| !(field->flags & UNSIGNED_FLAG)) {
|
|
my_error(ER_INNODB_FT_WRONG_DOCID_COLUMN, MYF(0),
|
|
field->field_name);
|
|
} else {
|
|
*fts_doc_col_no = i;
|
|
}
|
|
|
|
return(true);
|
|
}
|
|
|
|
if (!table) {
|
|
return(false);
|
|
}
|
|
|
|
for (; i + DATA_N_SYS_COLS < (uint) table->n_cols; i++) {
|
|
const char* name = dict_table_get_col_name(table, i);
|
|
|
|
if (strcmp(name, FTS_DOC_ID_COL_NAME) == 0) {
|
|
#ifdef UNIV_DEBUG
|
|
const dict_col_t* col;
|
|
|
|
col = dict_table_get_nth_col(table, i);
|
|
|
|
/* Because the FTS_DOC_ID does not exist in
|
|
the MySQL data dictionary, this must be the
|
|
internally created FTS_DOC_ID column. */
|
|
ut_ad(col->mtype == DATA_INT);
|
|
ut_ad(col->len == 8);
|
|
ut_ad(col->prtype & DATA_NOT_NULL);
|
|
ut_ad(col->prtype & DATA_UNSIGNED);
|
|
#endif /* UNIV_DEBUG */
|
|
*fts_doc_col_no = i;
|
|
return(true);
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Check whether the table has a unique index with FTS_DOC_ID_INDEX_NAME
|
|
on the Doc ID column.
|
|
@return the status of the FTS_DOC_ID index */
|
|
UNIV_INTERN
|
|
enum fts_doc_id_index_enum
|
|
innobase_fts_check_doc_id_index(
|
|
/*============================*/
|
|
const dict_table_t* table, /*!< in: table definition */
|
|
const TABLE* altered_table, /*!< in: MySQL table
|
|
that is being altered */
|
|
ulint* fts_doc_col_no) /*!< out: The column number for
|
|
Doc ID, or ULINT_UNDEFINED
|
|
if it is being created in
|
|
ha_alter_info */
|
|
{
|
|
const dict_index_t* index;
|
|
const dict_field_t* field;
|
|
|
|
if (altered_table) {
|
|
/* Check if a unique index with the name of
|
|
FTS_DOC_ID_INDEX_NAME is being created. */
|
|
|
|
for (uint i = 0; i < altered_table->s->keys; i++) {
|
|
const KEY& key = altered_table->key_info[i];
|
|
|
|
if (innobase_strcasecmp(
|
|
key.name, FTS_DOC_ID_INDEX_NAME)) {
|
|
continue;
|
|
}
|
|
|
|
if ((key.flags & HA_NOSAME)
|
|
&& key.user_defined_key_parts == 1
|
|
&& !strcmp(key.name, FTS_DOC_ID_INDEX_NAME)
|
|
&& !strcmp(key.key_part[0].field->field_name,
|
|
FTS_DOC_ID_COL_NAME)) {
|
|
if (fts_doc_col_no) {
|
|
*fts_doc_col_no = ULINT_UNDEFINED;
|
|
}
|
|
return(FTS_EXIST_DOC_ID_INDEX);
|
|
} else {
|
|
return(FTS_INCORRECT_DOC_ID_INDEX);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!table) {
|
|
return(FTS_NOT_EXIST_DOC_ID_INDEX);
|
|
}
|
|
|
|
for (index = dict_table_get_first_index(table);
|
|
index; index = dict_table_get_next_index(index)) {
|
|
|
|
/* Check if there exists a unique index with the name of
|
|
FTS_DOC_ID_INDEX_NAME */
|
|
if (innobase_strcasecmp(index->name, FTS_DOC_ID_INDEX_NAME)) {
|
|
continue;
|
|
}
|
|
|
|
if (!dict_index_is_unique(index)
|
|
|| dict_index_get_n_unique(index) > 1
|
|
|| strcmp(index->name, FTS_DOC_ID_INDEX_NAME)) {
|
|
return(FTS_INCORRECT_DOC_ID_INDEX);
|
|
}
|
|
|
|
/* Check whether the index has FTS_DOC_ID as its
|
|
first column */
|
|
field = dict_index_get_nth_field(index, 0);
|
|
|
|
/* The column would be of a BIGINT data type */
|
|
if (strcmp(field->name, FTS_DOC_ID_COL_NAME) == 0
|
|
&& field->col->mtype == DATA_INT
|
|
&& field->col->len == 8
|
|
&& field->col->prtype & DATA_NOT_NULL) {
|
|
if (fts_doc_col_no) {
|
|
*fts_doc_col_no = dict_col_get_no(field->col);
|
|
}
|
|
return(FTS_EXIST_DOC_ID_INDEX);
|
|
} else {
|
|
return(FTS_INCORRECT_DOC_ID_INDEX);
|
|
}
|
|
}
|
|
|
|
|
|
/* Not found */
|
|
return(FTS_NOT_EXIST_DOC_ID_INDEX);
|
|
}
|
|
/*******************************************************************//**
|
|
Check whether the table has a unique index with FTS_DOC_ID_INDEX_NAME
|
|
on the Doc ID column in MySQL create index definition.
|
|
@return FTS_EXIST_DOC_ID_INDEX if there exists the FTS_DOC_ID index,
|
|
FTS_INCORRECT_DOC_ID_INDEX if the FTS_DOC_ID index is of wrong format */
|
|
UNIV_INTERN
|
|
enum fts_doc_id_index_enum
|
|
innobase_fts_check_doc_id_index_in_def(
|
|
/*===================================*/
|
|
ulint n_key, /*!< in: Number of keys */
|
|
const KEY* key_info) /*!< in: Key definition */
|
|
{
|
|
/* Check whether there is a "FTS_DOC_ID_INDEX" in the to be built index
|
|
list */
|
|
for (ulint j = 0; j < n_key; j++) {
|
|
const KEY* key = &key_info[j];
|
|
|
|
if (innobase_strcasecmp(key->name, FTS_DOC_ID_INDEX_NAME)) {
|
|
continue;
|
|
}
|
|
|
|
/* Do a check on FTS DOC ID_INDEX, it must be unique,
|
|
named as "FTS_DOC_ID_INDEX" and on column "FTS_DOC_ID" */
|
|
if (!(key->flags & HA_NOSAME)
|
|
|| key->user_defined_key_parts != 1
|
|
|| strcmp(key->name, FTS_DOC_ID_INDEX_NAME)
|
|
|| strcmp(key->key_part[0].field->field_name,
|
|
FTS_DOC_ID_COL_NAME)) {
|
|
return(FTS_INCORRECT_DOC_ID_INDEX);
|
|
}
|
|
|
|
return(FTS_EXIST_DOC_ID_INDEX);
|
|
}
|
|
|
|
return(FTS_NOT_EXIST_DOC_ID_INDEX);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Create an index table where indexes are ordered as follows:
|
|
|
|
IF a new primary key is defined for the table THEN
|
|
|
|
1) New primary key
|
|
2) The remaining keys in key_info
|
|
|
|
ELSE
|
|
|
|
1) All new indexes in the order they arrive from MySQL
|
|
|
|
ENDIF
|
|
|
|
@return key definitions */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result, malloc))
|
|
index_def_t*
|
|
innobase_create_key_defs(
|
|
/*=====================*/
|
|
mem_heap_t* heap,
|
|
/*!< in/out: memory heap where space for key
|
|
definitions are allocated */
|
|
const Alter_inplace_info* ha_alter_info,
|
|
/*!< in: alter operation */
|
|
const TABLE* altered_table,
|
|
/*!< in: MySQL table that is being altered */
|
|
ulint& n_add,
|
|
/*!< in/out: number of indexes to be created */
|
|
ulint& n_fts_add,
|
|
/*!< out: number of FTS indexes to be created */
|
|
bool got_default_clust,
|
|
/*!< in: whether the table lacks a primary key */
|
|
ulint& fts_doc_id_col,
|
|
/*!< in: The column number for Doc ID */
|
|
bool& add_fts_doc_id,
|
|
/*!< in: whether we need to add new DOC ID
|
|
column for FTS index */
|
|
bool& add_fts_doc_idx,
|
|
/*!< in: whether we need to add new DOC ID
|
|
index for FTS index */
|
|
const TABLE* table)
|
|
/*!< in: MySQL table that is being altered */
|
|
{
|
|
index_def_t* indexdef;
|
|
index_def_t* indexdefs;
|
|
bool new_primary;
|
|
const uint*const add
|
|
= ha_alter_info->index_add_buffer;
|
|
const KEY*const key_info
|
|
= ha_alter_info->key_info_buffer;
|
|
|
|
DBUG_ENTER("innobase_create_key_defs");
|
|
DBUG_ASSERT(!add_fts_doc_id || add_fts_doc_idx);
|
|
DBUG_ASSERT(ha_alter_info->index_add_count == n_add);
|
|
|
|
/* If there is a primary key, it is always the first index
|
|
defined for the innodb_table. */
|
|
|
|
new_primary = n_add > 0
|
|
&& !my_strcasecmp(system_charset_info,
|
|
key_info[*add].name, "PRIMARY");
|
|
n_fts_add = 0;
|
|
|
|
/* If there is a UNIQUE INDEX consisting entirely of NOT NULL
|
|
columns and if the index does not contain column prefix(es)
|
|
(only prefix/part of the column is indexed), MySQL will treat the
|
|
index as a PRIMARY KEY unless the table already has one. */
|
|
|
|
ut_ad(altered_table->s->primary_key == 0
|
|
|| altered_table->s->primary_key == MAX_KEY);
|
|
|
|
if (got_default_clust && !new_primary) {
|
|
new_primary = (altered_table->s->primary_key != MAX_KEY);
|
|
}
|
|
|
|
const bool rebuild = new_primary || add_fts_doc_id
|
|
|| innobase_need_rebuild(ha_alter_info, table);
|
|
|
|
/* Reserve one more space if new_primary is true, and we might
|
|
need to add the FTS_DOC_ID_INDEX */
|
|
indexdef = indexdefs = static_cast<index_def_t*>(
|
|
mem_heap_alloc(
|
|
heap, sizeof *indexdef
|
|
* (ha_alter_info->key_count
|
|
+ rebuild
|
|
+ got_default_clust)));
|
|
|
|
if (rebuild) {
|
|
ulint primary_key_number;
|
|
|
|
if (new_primary) {
|
|
if (n_add == 0) {
|
|
DBUG_ASSERT(got_default_clust);
|
|
DBUG_ASSERT(altered_table->s->primary_key
|
|
== 0);
|
|
primary_key_number = 0;
|
|
} else {
|
|
primary_key_number = *add;
|
|
}
|
|
} else if (got_default_clust) {
|
|
/* Create the GEN_CLUST_INDEX */
|
|
index_def_t* index = indexdef++;
|
|
|
|
index->fields = NULL;
|
|
index->n_fields = 0;
|
|
index->ind_type = DICT_CLUSTERED;
|
|
index->name = mem_heap_strdup(
|
|
heap, innobase_index_reserve_name);
|
|
index->key_number = ~0;
|
|
primary_key_number = ULINT_UNDEFINED;
|
|
goto created_clustered;
|
|
} else {
|
|
primary_key_number = 0;
|
|
}
|
|
|
|
/* Create the PRIMARY key index definition */
|
|
innobase_create_index_def(
|
|
altered_table, key_info, primary_key_number,
|
|
TRUE, TRUE, indexdef++, heap, (const Field **)altered_table->field);
|
|
|
|
created_clustered:
|
|
n_add = 1;
|
|
|
|
for (ulint i = 0; i < ha_alter_info->key_count; i++) {
|
|
if (i == primary_key_number) {
|
|
continue;
|
|
}
|
|
/* Copy the index definitions. */
|
|
innobase_create_index_def(
|
|
altered_table, key_info, i, TRUE, FALSE,
|
|
indexdef, heap, (const Field **)altered_table->field);
|
|
|
|
if (indexdef->ind_type & DICT_FTS) {
|
|
n_fts_add++;
|
|
}
|
|
|
|
indexdef++;
|
|
n_add++;
|
|
}
|
|
|
|
if (n_fts_add > 0) {
|
|
if (!add_fts_doc_id
|
|
&& !innobase_fts_check_doc_id_col(
|
|
NULL, altered_table,
|
|
&fts_doc_id_col)) {
|
|
fts_doc_id_col =
|
|
altered_table->s->stored_fields;
|
|
add_fts_doc_id = true;
|
|
}
|
|
|
|
if (!add_fts_doc_idx) {
|
|
fts_doc_id_index_enum ret;
|
|
ulint doc_col_no;
|
|
|
|
ret = innobase_fts_check_doc_id_index(
|
|
NULL, altered_table, &doc_col_no);
|
|
|
|
/* This should have been checked before */
|
|
ut_ad(ret != FTS_INCORRECT_DOC_ID_INDEX);
|
|
|
|
if (ret == FTS_NOT_EXIST_DOC_ID_INDEX) {
|
|
add_fts_doc_idx = true;
|
|
} else {
|
|
ut_ad(ret == FTS_EXIST_DOC_ID_INDEX);
|
|
ut_ad(doc_col_no == ULINT_UNDEFINED
|
|
|| doc_col_no == fts_doc_id_col);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/* Create definitions for added secondary indexes. */
|
|
|
|
for (ulint i = 0; i < n_add; i++) {
|
|
innobase_create_index_def(
|
|
altered_table, key_info, add[i], FALSE, FALSE,
|
|
indexdef, heap, (const Field **)altered_table->field);
|
|
|
|
if (indexdef->ind_type & DICT_FTS) {
|
|
n_fts_add++;
|
|
}
|
|
|
|
indexdef++;
|
|
}
|
|
}
|
|
|
|
DBUG_ASSERT(indexdefs + n_add == indexdef);
|
|
|
|
if (add_fts_doc_idx) {
|
|
index_def_t* index = indexdef++;
|
|
|
|
index->fields = static_cast<index_field_t*>(
|
|
mem_heap_alloc(heap, sizeof *index->fields));
|
|
memset(index->fields, 0, sizeof *index->fields);
|
|
index->n_fields = 1;
|
|
index->fields->col_no = fts_doc_id_col;
|
|
index->fields->prefix_len = 0;
|
|
index->ind_type = DICT_UNIQUE;
|
|
|
|
if (rebuild) {
|
|
index->name = mem_heap_strdup(
|
|
heap, FTS_DOC_ID_INDEX_NAME);
|
|
ut_ad(!add_fts_doc_id
|
|
|| fts_doc_id_col == altered_table->s->stored_fields);
|
|
} else {
|
|
char* index_name;
|
|
index->name = index_name = static_cast<char*>(
|
|
mem_heap_alloc(
|
|
heap,
|
|
1 + sizeof FTS_DOC_ID_INDEX_NAME));
|
|
*index_name++ = TEMP_INDEX_PREFIX;
|
|
memcpy(index_name, FTS_DOC_ID_INDEX_NAME,
|
|
sizeof FTS_DOC_ID_INDEX_NAME);
|
|
}
|
|
|
|
/* TODO: assign a real MySQL key number for this */
|
|
index->key_number = ULINT_UNDEFINED;
|
|
n_add++;
|
|
}
|
|
|
|
DBUG_ASSERT(indexdef > indexdefs);
|
|
DBUG_ASSERT((ulint) (indexdef - indexdefs)
|
|
<= ha_alter_info->key_count
|
|
+ add_fts_doc_idx + got_default_clust);
|
|
DBUG_ASSERT(ha_alter_info->index_add_count <= n_add);
|
|
DBUG_RETURN(indexdefs);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Check each index column size, make sure they do not exceed the max limit
|
|
@return true if index column size exceeds limit */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_check_column_length(
|
|
/*=========================*/
|
|
ulint max_col_len, /*!< in: maximum column length */
|
|
const KEY* key_info) /*!< in: Indexes to be created */
|
|
{
|
|
for (ulint key_part = 0; key_part < key_info->user_defined_key_parts; key_part++) {
|
|
if (key_info->key_part[key_part].length > max_col_len) {
|
|
return(true);
|
|
}
|
|
}
|
|
return(false);
|
|
}
|
|
|
|
struct ha_innobase_inplace_ctx : public inplace_alter_handler_ctx
|
|
{
|
|
/** Dummy query graph */
|
|
que_thr_t* thr;
|
|
/** reference to the prebuilt struct of the creating instance */
|
|
row_prebuilt_t*&prebuilt;
|
|
/** InnoDB indexes being created */
|
|
dict_index_t** add_index;
|
|
/** MySQL key numbers for the InnoDB indexes that are being created */
|
|
const ulint* add_key_numbers;
|
|
/** number of InnoDB indexes being created */
|
|
ulint num_to_add_index;
|
|
/** InnoDB indexes being dropped */
|
|
dict_index_t** drop_index;
|
|
/** number of InnoDB indexes being dropped */
|
|
const ulint num_to_drop_index;
|
|
/** InnoDB foreign key constraints being dropped */
|
|
dict_foreign_t** drop_fk;
|
|
/** number of InnoDB foreign key constraints being dropped */
|
|
const ulint num_to_drop_fk;
|
|
/** InnoDB foreign key constraints being added */
|
|
dict_foreign_t** add_fk;
|
|
/** number of InnoDB foreign key constraints being dropped */
|
|
const ulint num_to_add_fk;
|
|
/** whether to create the indexes online */
|
|
bool online;
|
|
/** memory heap */
|
|
mem_heap_t* heap;
|
|
/** dictionary transaction */
|
|
trx_t* trx;
|
|
/** original table (if rebuilt, differs from indexed_table) */
|
|
dict_table_t* old_table;
|
|
/** table where the indexes are being created or dropped */
|
|
dict_table_t* new_table;
|
|
/** mapping of old column numbers to new ones, or NULL */
|
|
const ulint* col_map;
|
|
/** new column names, or NULL if nothing was renamed */
|
|
const char** col_names;
|
|
/** added AUTO_INCREMENT column position, or ULINT_UNDEFINED */
|
|
const ulint add_autoinc;
|
|
/** default values of ADD COLUMN, or NULL */
|
|
const dtuple_t* add_cols;
|
|
/** autoinc sequence to use */
|
|
ib_sequence_t sequence;
|
|
/** maximum auto-increment value */
|
|
ulonglong max_autoinc;
|
|
/** temporary table name to use for old table when renaming tables */
|
|
const char* tmp_name;
|
|
|
|
ha_innobase_inplace_ctx(row_prebuilt_t*& prebuilt_arg,
|
|
dict_index_t** drop_arg,
|
|
ulint num_to_drop_arg,
|
|
dict_foreign_t** drop_fk_arg,
|
|
ulint num_to_drop_fk_arg,
|
|
dict_foreign_t** add_fk_arg,
|
|
ulint num_to_add_fk_arg,
|
|
bool online_arg,
|
|
mem_heap_t* heap_arg,
|
|
dict_table_t* new_table_arg,
|
|
const char** col_names_arg,
|
|
ulint add_autoinc_arg,
|
|
ulonglong autoinc_col_min_value_arg,
|
|
ulonglong autoinc_col_max_value_arg) :
|
|
inplace_alter_handler_ctx(),
|
|
prebuilt (prebuilt_arg),
|
|
add_index (0), add_key_numbers (0), num_to_add_index (0),
|
|
drop_index (drop_arg), num_to_drop_index (num_to_drop_arg),
|
|
drop_fk (drop_fk_arg), num_to_drop_fk (num_to_drop_fk_arg),
|
|
add_fk (add_fk_arg), num_to_add_fk (num_to_add_fk_arg),
|
|
online (online_arg), heap (heap_arg), trx (0),
|
|
old_table (prebuilt_arg->table),
|
|
new_table (new_table_arg),
|
|
col_map (0), col_names (col_names_arg),
|
|
add_autoinc (add_autoinc_arg),
|
|
add_cols (0),
|
|
sequence(prebuilt->trx->mysql_thd,
|
|
autoinc_col_min_value_arg, autoinc_col_max_value_arg),
|
|
max_autoinc (0),
|
|
tmp_name (0)
|
|
{
|
|
#ifdef UNIV_DEBUG
|
|
for (ulint i = 0; i < num_to_add_index; i++) {
|
|
ut_ad(!add_index[i]->to_be_dropped);
|
|
}
|
|
for (ulint i = 0; i < num_to_drop_index; i++) {
|
|
ut_ad(drop_index[i]->to_be_dropped);
|
|
}
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
thr = pars_complete_graph_for_exec(NULL, prebuilt->trx, heap);
|
|
}
|
|
|
|
~ha_innobase_inplace_ctx()
|
|
{
|
|
mem_heap_free(heap);
|
|
}
|
|
|
|
/** Determine if the table will be rebuilt.
|
|
@return whether the table will be rebuilt */
|
|
bool need_rebuild () const { return(old_table != new_table); }
|
|
|
|
private:
|
|
// Disable copying
|
|
ha_innobase_inplace_ctx(const ha_innobase_inplace_ctx&);
|
|
ha_innobase_inplace_ctx& operator=(const ha_innobase_inplace_ctx&);
|
|
};
|
|
|
|
/********************************************************************//**
|
|
Drop any indexes that we were not able to free previously due to
|
|
open table handles. */
|
|
static
|
|
void
|
|
online_retry_drop_indexes_low(
|
|
/*==========================*/
|
|
dict_table_t* table, /*!< in/out: table */
|
|
trx_t* trx) /*!< in/out: transaction */
|
|
{
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
ut_ad(trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
ut_ad(trx_get_dict_operation(trx) == TRX_DICT_OP_INDEX);
|
|
|
|
/* We can have table->n_ref_count > 1, because other threads
|
|
may have prebuilt->table pointing to the table. However, these
|
|
other threads should be between statements, waiting for the
|
|
next statement to execute, or for a meta-data lock. */
|
|
ut_ad(table->n_ref_count >= 1);
|
|
|
|
if (table->drop_aborted) {
|
|
row_merge_drop_indexes(trx, table, TRUE);
|
|
}
|
|
}
|
|
|
|
/********************************************************************//**
|
|
Drop any indexes that we were not able to free previously due to
|
|
open table handles. */
|
|
static MY_ATTRIBUTE((nonnull))
|
|
void
|
|
online_retry_drop_indexes(
|
|
/*======================*/
|
|
dict_table_t* table, /*!< in/out: table */
|
|
THD* user_thd) /*!< in/out: MySQL connection */
|
|
{
|
|
if (table->drop_aborted) {
|
|
trx_t* trx = innobase_trx_allocate(user_thd);
|
|
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
online_retry_drop_indexes_low(table, trx);
|
|
trx_commit_for_mysql(trx);
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
trx_free_for_mysql(trx);
|
|
}
|
|
|
|
#ifdef UNIV_DEBUG
|
|
mutex_enter(&dict_sys->mutex);
|
|
dict_table_check_for_dup_indexes(table, CHECK_ALL_COMPLETE);
|
|
mutex_exit(&dict_sys->mutex);
|
|
ut_a(!table->drop_aborted);
|
|
#endif /* UNIV_DEBUG */
|
|
}
|
|
|
|
/********************************************************************//**
|
|
Commit a dictionary transaction and drop any indexes that we were not
|
|
able to free previously due to open table handles. */
|
|
static MY_ATTRIBUTE((nonnull))
|
|
void
|
|
online_retry_drop_indexes_with_trx(
|
|
/*===============================*/
|
|
dict_table_t* table, /*!< in/out: table */
|
|
trx_t* trx) /*!< in/out: transaction */
|
|
{
|
|
ut_ad(trx_state_eq(trx, TRX_STATE_NOT_STARTED));
|
|
ut_ad(trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
|
|
/* Now that the dictionary is being locked, check if we can
|
|
drop any incompletely created indexes that may have been left
|
|
behind in rollback_inplace_alter_table() earlier. */
|
|
if (table->drop_aborted) {
|
|
|
|
trx->table_id = 0;
|
|
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
|
|
|
|
online_retry_drop_indexes_low(table, trx);
|
|
trx_commit_for_mysql(trx);
|
|
}
|
|
}
|
|
|
|
/** Determines if InnoDB is dropping a foreign key constraint.
|
|
@param foreign the constraint
|
|
@param drop_fk constraints being dropped
|
|
@param n_drop_fk number of constraints that are being dropped
|
|
@return whether the constraint is being dropped */
|
|
inline MY_ATTRIBUTE((pure, nonnull, warn_unused_result))
|
|
bool
|
|
innobase_dropping_foreign(
|
|
/*======================*/
|
|
const dict_foreign_t* foreign,
|
|
dict_foreign_t** drop_fk,
|
|
ulint n_drop_fk)
|
|
{
|
|
while (n_drop_fk--) {
|
|
if (*drop_fk++ == foreign) {
|
|
return(true);
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/** Determines if an InnoDB FOREIGN KEY constraint depends on a
|
|
column that is being dropped or modified to NOT NULL.
|
|
@param user_table InnoDB table as it is before the ALTER operation
|
|
@param col_name Name of the column being altered
|
|
@param drop_fk constraints being dropped
|
|
@param n_drop_fk number of constraints that are being dropped
|
|
@param drop true=drop column, false=set NOT NULL
|
|
@retval true Not allowed (will call my_error())
|
|
@retval false Allowed
|
|
*/
|
|
static MY_ATTRIBUTE((pure, nonnull, warn_unused_result))
|
|
bool
|
|
innobase_check_foreigns_low(
|
|
/*========================*/
|
|
const dict_table_t* user_table,
|
|
dict_foreign_t** drop_fk,
|
|
ulint n_drop_fk,
|
|
const char* col_name,
|
|
bool drop)
|
|
{
|
|
dict_foreign_t* foreign;
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
|
|
/* Check if any FOREIGN KEY constraints are defined on this
|
|
column. */
|
|
|
|
for (dict_foreign_set::const_iterator it = user_table->foreign_set.begin();
|
|
it != user_table->foreign_set.end();
|
|
++it) {
|
|
|
|
foreign = *it;
|
|
|
|
if (!drop && !(foreign->type
|
|
& (DICT_FOREIGN_ON_DELETE_SET_NULL
|
|
| DICT_FOREIGN_ON_UPDATE_SET_NULL))) {
|
|
continue;
|
|
}
|
|
|
|
if (innobase_dropping_foreign(foreign, drop_fk, n_drop_fk)) {
|
|
continue;
|
|
}
|
|
|
|
for (unsigned f = 0; f < foreign->n_fields; f++) {
|
|
if (!strcmp(foreign->foreign_col_names[f],
|
|
col_name)) {
|
|
my_error(drop
|
|
? ER_FK_COLUMN_CANNOT_DROP
|
|
: ER_FK_COLUMN_NOT_NULL, MYF(0),
|
|
col_name, foreign->id);
|
|
return(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!drop) {
|
|
/* SET NULL clauses on foreign key constraints of
|
|
child tables affect the child tables, not the parent table.
|
|
The column can be NOT NULL in the parent table. */
|
|
return(false);
|
|
}
|
|
|
|
/* Check if any FOREIGN KEY constraints in other tables are
|
|
referring to the column that is being dropped. */
|
|
for (dict_foreign_set::const_iterator it
|
|
= user_table->referenced_set.begin();
|
|
it != user_table->referenced_set.end();
|
|
++it) {
|
|
|
|
foreign = *it;
|
|
|
|
if (innobase_dropping_foreign(foreign, drop_fk, n_drop_fk)) {
|
|
continue;
|
|
}
|
|
|
|
for (unsigned f = 0; f < foreign->n_fields; f++) {
|
|
char display_name[FN_REFLEN];
|
|
|
|
if (strcmp(foreign->referenced_col_names[f],
|
|
col_name)) {
|
|
continue;
|
|
}
|
|
|
|
char* buf_end = innobase_convert_name(
|
|
display_name, (sizeof display_name) - 1,
|
|
foreign->foreign_table_name,
|
|
strlen(foreign->foreign_table_name),
|
|
NULL, TRUE);
|
|
*buf_end = '\0';
|
|
my_error(ER_FK_COLUMN_CANNOT_DROP_CHILD,
|
|
MYF(0), col_name, foreign->id,
|
|
display_name);
|
|
|
|
return(true);
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/** Determines if an InnoDB FOREIGN KEY constraint depends on a
|
|
column that is being dropped or modified to NOT NULL.
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param altered_table MySQL table that is being altered
|
|
@param old_table MySQL table as it is before the ALTER operation
|
|
@param user_table InnoDB table as it is before the ALTER operation
|
|
@param drop_fk constraints being dropped
|
|
@param n_drop_fk number of constraints that are being dropped
|
|
@retval true Not allowed (will call my_error())
|
|
@retval false Allowed
|
|
*/
|
|
static MY_ATTRIBUTE((pure, nonnull, warn_unused_result))
|
|
bool
|
|
innobase_check_foreigns(
|
|
/*====================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
const TABLE* altered_table,
|
|
const TABLE* old_table,
|
|
const dict_table_t* user_table,
|
|
dict_foreign_t** drop_fk,
|
|
ulint n_drop_fk)
|
|
{
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
|
|
for (Field** fp = old_table->field; *fp; fp++) {
|
|
cf_it.rewind();
|
|
const Create_field* new_field;
|
|
|
|
ut_ad(!(*fp)->real_maybe_null()
|
|
== !!((*fp)->flags & NOT_NULL_FLAG));
|
|
|
|
while ((new_field = cf_it++)) {
|
|
if (new_field->field == *fp) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!new_field || (new_field->flags & NOT_NULL_FLAG)) {
|
|
if (innobase_check_foreigns_low(
|
|
user_table, drop_fk, n_drop_fk,
|
|
(*fp)->field_name, !new_field)) {
|
|
return(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/** Convert a default value for ADD COLUMN.
|
|
|
|
@param heap Memory heap where allocated
|
|
@param dfield InnoDB data field to copy to
|
|
@param field MySQL value for the column
|
|
@param comp nonzero if in compact format */
|
|
static MY_ATTRIBUTE((nonnull))
|
|
void
|
|
innobase_build_col_map_add(
|
|
/*=======================*/
|
|
mem_heap_t* heap,
|
|
dfield_t* dfield,
|
|
const Field* field,
|
|
ulint comp,
|
|
row_prebuilt_t* prebuilt)
|
|
{
|
|
if (field->is_real_null()) {
|
|
dfield_set_null(dfield);
|
|
return;
|
|
}
|
|
|
|
ulint size = field->pack_length();
|
|
|
|
byte* buf = static_cast<byte*>(mem_heap_alloc(heap, size));
|
|
|
|
row_mysql_store_col_in_innobase_format(
|
|
dfield, buf, TRUE, field->ptr, size, comp);
|
|
}
|
|
|
|
/** Construct the translation table for reordering, dropping or
|
|
adding columns.
|
|
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param altered_table MySQL table that is being altered
|
|
@param table MySQL table as it is before the ALTER operation
|
|
@param new_table InnoDB table corresponding to MySQL altered_table
|
|
@param old_table InnoDB table corresponding to MYSQL table
|
|
@param add_cols Default values for ADD COLUMN, or NULL if no ADD COLUMN
|
|
@param heap Memory heap where allocated
|
|
@return array of integers, mapping column numbers in the table
|
|
to column numbers in altered_table */
|
|
static MY_ATTRIBUTE((nonnull(1,2,3,4,5,7), warn_unused_result))
|
|
const ulint*
|
|
innobase_build_col_map(
|
|
/*===================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
const TABLE* altered_table,
|
|
const TABLE* table,
|
|
const dict_table_t* new_table,
|
|
const dict_table_t* old_table,
|
|
dtuple_t* add_cols,
|
|
mem_heap_t* heap,
|
|
row_prebuilt_t* prebuilt)
|
|
{
|
|
uint old_i, old_innobase_i;
|
|
DBUG_ENTER("innobase_build_col_map");
|
|
DBUG_ASSERT(altered_table != table);
|
|
DBUG_ASSERT(new_table != old_table);
|
|
DBUG_ASSERT(dict_table_get_n_cols(new_table)
|
|
>= altered_table->s->stored_fields + DATA_N_SYS_COLS);
|
|
DBUG_ASSERT(dict_table_get_n_cols(old_table)
|
|
>= table->s->stored_fields + DATA_N_SYS_COLS);
|
|
DBUG_ASSERT(!!add_cols == !!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_COLUMN));
|
|
DBUG_ASSERT(!add_cols || dtuple_get_n_fields(add_cols)
|
|
== dict_table_get_n_cols(new_table));
|
|
|
|
ulint* col_map = static_cast<ulint*>(
|
|
mem_heap_alloc(heap, old_table->n_cols * sizeof *col_map));
|
|
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
uint i = 0, sql_idx = 0;
|
|
|
|
/* Any dropped columns will map to ULINT_UNDEFINED. */
|
|
for (old_innobase_i = 0;
|
|
old_innobase_i + DATA_N_SYS_COLS < old_table->n_cols;
|
|
old_innobase_i++) {
|
|
col_map[old_innobase_i] = ULINT_UNDEFINED;
|
|
}
|
|
|
|
while (const Create_field* new_field = cf_it++) {
|
|
if (!new_field->stored_in_db())
|
|
{
|
|
sql_idx++;
|
|
continue;
|
|
}
|
|
for (old_i = 0, old_innobase_i= 0;
|
|
table->field[old_i];
|
|
old_i++) {
|
|
const Field* field = table->field[old_i];
|
|
if (!table->field[old_i]->stored_in_db())
|
|
continue;
|
|
if (new_field->field == field) {
|
|
col_map[old_innobase_i] = i;
|
|
goto found_col;
|
|
}
|
|
old_innobase_i++;
|
|
}
|
|
|
|
innobase_build_col_map_add(
|
|
heap, dtuple_get_nth_field(add_cols, i),
|
|
altered_table->field[sql_idx],
|
|
dict_table_is_comp(new_table), prebuilt);
|
|
found_col:
|
|
i++;
|
|
sql_idx++;
|
|
}
|
|
|
|
DBUG_ASSERT(i == altered_table->s->stored_fields);
|
|
|
|
i = table->s->stored_fields;
|
|
|
|
/* Add the InnoDB hidden FTS_DOC_ID column, if any. */
|
|
if (i + DATA_N_SYS_COLS < old_table->n_cols) {
|
|
/* There should be exactly one extra field,
|
|
the FTS_DOC_ID. */
|
|
DBUG_ASSERT(DICT_TF2_FLAG_IS_SET(old_table,
|
|
DICT_TF2_FTS_HAS_DOC_ID));
|
|
DBUG_ASSERT(i + DATA_N_SYS_COLS + 1 == old_table->n_cols);
|
|
DBUG_ASSERT(!strcmp(dict_table_get_col_name(
|
|
old_table, table->s->stored_fields),
|
|
FTS_DOC_ID_COL_NAME));
|
|
if (altered_table->s->stored_fields + DATA_N_SYS_COLS
|
|
< new_table->n_cols) {
|
|
DBUG_ASSERT(DICT_TF2_FLAG_IS_SET(
|
|
new_table,
|
|
DICT_TF2_FTS_HAS_DOC_ID));
|
|
DBUG_ASSERT(altered_table->s->stored_fields
|
|
+ DATA_N_SYS_COLS + 1
|
|
== new_table->n_cols);
|
|
col_map[i] = altered_table->s->stored_fields;
|
|
} else {
|
|
DBUG_ASSERT(!DICT_TF2_FLAG_IS_SET(
|
|
new_table,
|
|
DICT_TF2_FTS_HAS_DOC_ID));
|
|
col_map[i] = ULINT_UNDEFINED;
|
|
}
|
|
|
|
i++;
|
|
} else {
|
|
DBUG_ASSERT(!DICT_TF2_FLAG_IS_SET(
|
|
old_table,
|
|
DICT_TF2_FTS_HAS_DOC_ID));
|
|
}
|
|
|
|
for (; i < old_table->n_cols; i++) {
|
|
col_map[i] = i + new_table->n_cols - old_table->n_cols;
|
|
}
|
|
|
|
DBUG_RETURN(col_map);
|
|
}
|
|
|
|
/** Drop newly create FTS index related auxiliary table during
|
|
FIC create index process, before fts_add_index is called
|
|
@param table table that was being rebuilt online
|
|
@param trx transaction
|
|
@return DB_SUCCESS if successful, otherwise last error code
|
|
*/
|
|
static
|
|
dberr_t
|
|
innobase_drop_fts_index_table(
|
|
/*==========================*/
|
|
dict_table_t* table,
|
|
trx_t* trx)
|
|
{
|
|
dberr_t ret_err = DB_SUCCESS;
|
|
|
|
for (dict_index_t* index = dict_table_get_first_index(table);
|
|
index != NULL;
|
|
index = dict_table_get_next_index(index)) {
|
|
if (index->type & DICT_FTS) {
|
|
dberr_t err;
|
|
|
|
err = fts_drop_index_tables(trx, index);
|
|
|
|
if (err != DB_SUCCESS) {
|
|
ret_err = err;
|
|
}
|
|
}
|
|
}
|
|
|
|
return(ret_err);
|
|
}
|
|
|
|
/** Get the new column names if any columns were renamed
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param altered_table MySQL table that is being altered
|
|
@param table MySQL table as it is before the ALTER operation
|
|
@param user_table InnoDB table as it is before the ALTER operation
|
|
@param heap Memory heap for the allocation
|
|
@return array of new column names in rebuilt_table, or NULL if not renamed */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
const char**
|
|
innobase_get_col_names(
|
|
Alter_inplace_info* ha_alter_info,
|
|
const TABLE* altered_table,
|
|
const TABLE* table,
|
|
const dict_table_t* user_table,
|
|
mem_heap_t* heap)
|
|
{
|
|
const char** cols;
|
|
uint i;
|
|
|
|
DBUG_ENTER("innobase_get_col_names");
|
|
DBUG_ASSERT(user_table->n_def > table->s->fields);
|
|
DBUG_ASSERT(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME);
|
|
|
|
cols = static_cast<const char**>(
|
|
mem_heap_zalloc(heap, user_table->n_def * sizeof *cols));
|
|
|
|
i = 0;
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
while (const Create_field* new_field = cf_it++) {
|
|
DBUG_ASSERT(i < altered_table->s->fields);
|
|
|
|
for (uint old_i = 0; table->field[old_i]; old_i++) {
|
|
if (new_field->field == table->field[old_i]) {
|
|
cols[old_i] = new_field->field_name;
|
|
break;
|
|
}
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
/* Copy the internal column names. */
|
|
i = table->s->fields;
|
|
cols[i] = dict_table_get_col_name(user_table, i);
|
|
|
|
while (++i < user_table->n_def) {
|
|
cols[i] = cols[i - 1] + strlen(cols[i - 1]) + 1;
|
|
}
|
|
|
|
DBUG_RETURN(cols);
|
|
}
|
|
|
|
/** Update internal structures with concurrent writes blocked,
|
|
while preparing ALTER TABLE.
|
|
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param altered_table MySQL table that is being altered
|
|
@param old_table MySQL table as it is before the ALTER operation
|
|
@param table_name Table name in MySQL
|
|
@param flags Table and tablespace flags
|
|
@param flags2 Additional table flags
|
|
@param fts_doc_id_col The column number of FTS_DOC_ID
|
|
@param add_fts_doc_id Flag: add column FTS_DOC_ID?
|
|
@param add_fts_doc_id_idx Flag: add index FTS_DOC_ID_INDEX (FTS_DOC_ID)?
|
|
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
static MY_ATTRIBUTE((warn_unused_result, nonnull(1,2,3,4)))
|
|
bool
|
|
prepare_inplace_alter_table_dict(
|
|
/*=============================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
const TABLE* altered_table,
|
|
const TABLE* old_table,
|
|
const char* table_name,
|
|
ulint flags,
|
|
ulint flags2,
|
|
ulint fts_doc_id_col,
|
|
bool add_fts_doc_id,
|
|
bool add_fts_doc_id_idx,
|
|
row_prebuilt_t* prebuilt)
|
|
{
|
|
bool dict_locked = false;
|
|
ulint* add_key_nums; /* MySQL key numbers */
|
|
index_def_t* index_defs; /* index definitions */
|
|
dict_table_t* user_table;
|
|
dict_index_t* fts_index = NULL;
|
|
ulint new_clustered = 0;
|
|
dberr_t error;
|
|
ulint num_fts_index;
|
|
ha_innobase_inplace_ctx*ctx;
|
|
uint sql_idx;
|
|
|
|
DBUG_ENTER("prepare_inplace_alter_table_dict");
|
|
|
|
ctx = static_cast<ha_innobase_inplace_ctx*>
|
|
(ha_alter_info->handler_ctx);
|
|
|
|
DBUG_ASSERT((ctx->add_autoinc != ULINT_UNDEFINED)
|
|
== (ctx->sequence.m_max_value > 0));
|
|
DBUG_ASSERT(!ctx->num_to_drop_index == !ctx->drop_index);
|
|
DBUG_ASSERT(!ctx->num_to_drop_fk == !ctx->drop_fk);
|
|
DBUG_ASSERT(!add_fts_doc_id || add_fts_doc_id_idx);
|
|
DBUG_ASSERT(!add_fts_doc_id_idx
|
|
|| innobase_fulltext_exist(altered_table));
|
|
DBUG_ASSERT(!ctx->add_cols);
|
|
DBUG_ASSERT(!ctx->add_index);
|
|
DBUG_ASSERT(!ctx->add_key_numbers);
|
|
DBUG_ASSERT(!ctx->num_to_add_index);
|
|
|
|
user_table = ctx->new_table;
|
|
|
|
trx_start_if_not_started_xa(ctx->prebuilt->trx);
|
|
|
|
/* Create a background transaction for the operations on
|
|
the data dictionary tables. */
|
|
ctx->trx = innobase_trx_allocate(ctx->prebuilt->trx->mysql_thd);
|
|
trx_start_for_ddl(ctx->trx, TRX_DICT_OP_INDEX);
|
|
|
|
DBUG_ASSERT(!ctx->trx->fake_changes);
|
|
|
|
/* Create table containing all indexes to be built in this
|
|
ALTER TABLE ADD INDEX so that they are in the correct order
|
|
in the table. */
|
|
|
|
ctx->num_to_add_index = ha_alter_info->index_add_count;
|
|
|
|
ut_ad(ctx->prebuilt->trx->mysql_thd != NULL);
|
|
const char* path = thd_innodb_tmpdir(
|
|
ctx->prebuilt->trx->mysql_thd);
|
|
|
|
index_defs = innobase_create_key_defs(
|
|
ctx->heap, ha_alter_info, altered_table, ctx->num_to_add_index,
|
|
num_fts_index,
|
|
row_table_got_default_clust_index(ctx->new_table),
|
|
fts_doc_id_col, add_fts_doc_id, add_fts_doc_id_idx,
|
|
old_table);
|
|
|
|
new_clustered = DICT_CLUSTERED & index_defs[0].ind_type;
|
|
|
|
if (num_fts_index > 1) {
|
|
my_error(ER_INNODB_FT_LIMIT, MYF(0));
|
|
goto error_handled;
|
|
}
|
|
|
|
if (!ctx->online) {
|
|
/* This is not an online operation (LOCK=NONE). */
|
|
} else if (ctx->add_autoinc == ULINT_UNDEFINED
|
|
&& num_fts_index == 0
|
|
&& (!innobase_need_rebuild(ha_alter_info, old_table)
|
|
|| !innobase_fulltext_exist(altered_table))) {
|
|
/* InnoDB can perform an online operation (LOCK=NONE). */
|
|
} else {
|
|
/* This should have been blocked in
|
|
check_if_supported_inplace_alter(). */
|
|
ut_ad(0);
|
|
my_error(ER_NOT_SUPPORTED_YET, MYF(0),
|
|
thd_query_string(ctx->prebuilt->trx->mysql_thd)->str);
|
|
goto error_handled;
|
|
}
|
|
|
|
/* The primary index would be rebuilt if a FTS Doc ID
|
|
column is to be added, and the primary index definition
|
|
is just copied from old table and stored in indexdefs[0] */
|
|
DBUG_ASSERT(!add_fts_doc_id || new_clustered);
|
|
DBUG_ASSERT(!!new_clustered ==
|
|
(innobase_need_rebuild(ha_alter_info, old_table)
|
|
|| add_fts_doc_id));
|
|
|
|
/* Allocate memory for dictionary index definitions */
|
|
|
|
ctx->add_index = static_cast<dict_index_t**>(
|
|
mem_heap_alloc(ctx->heap, ctx->num_to_add_index
|
|
* sizeof *ctx->add_index));
|
|
ctx->add_key_numbers = add_key_nums = static_cast<ulint*>(
|
|
mem_heap_alloc(ctx->heap, ctx->num_to_add_index
|
|
* sizeof *ctx->add_key_numbers));
|
|
|
|
/* This transaction should be dictionary operation, so that
|
|
the data dictionary will be locked during crash recovery. */
|
|
|
|
ut_ad(ctx->trx->dict_operation == TRX_DICT_OP_INDEX);
|
|
|
|
/* Acquire a lock on the table before creating any indexes. */
|
|
|
|
if (ctx->online) {
|
|
error = DB_SUCCESS;
|
|
} else {
|
|
error = row_merge_lock_table(
|
|
ctx->prebuilt->trx, ctx->new_table, LOCK_S);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
|
|
goto error_handling;
|
|
}
|
|
}
|
|
|
|
/* Latch the InnoDB data dictionary exclusively so that no deadlocks
|
|
or lock waits can happen in it during an index create operation. */
|
|
|
|
row_mysql_lock_data_dictionary(ctx->trx);
|
|
dict_locked = true;
|
|
|
|
/* Wait for background stats processing to stop using the table that
|
|
we are going to alter. We know bg stats will not start using it again
|
|
until we are holding the data dict locked and we are holding it here
|
|
at least until checking ut_ad(user_table->n_ref_count == 1) below.
|
|
XXX what may happen if bg stats opens the table after we
|
|
have unlocked data dictionary below? */
|
|
dict_stats_wait_bg_to_stop_using_table(user_table, ctx->trx);
|
|
|
|
online_retry_drop_indexes_low(ctx->new_table, ctx->trx);
|
|
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
ctx->new_table, CHECK_ABORTED_OK));
|
|
|
|
/* If a new clustered index is defined for the table we need
|
|
to rebuild the table with a temporary name. */
|
|
|
|
if (new_clustered) {
|
|
fil_space_crypt_t* crypt_data;
|
|
const char* new_table_name
|
|
= dict_mem_create_temporary_tablename(
|
|
ctx->heap,
|
|
ctx->new_table->name,
|
|
ctx->new_table->id);
|
|
ulint n_cols;
|
|
dtuple_t* add_cols;
|
|
ulint key_id = FIL_DEFAULT_ENCRYPTION_KEY;
|
|
fil_encryption_t mode = FIL_ENCRYPTION_DEFAULT;
|
|
|
|
fil_space_t* space = fil_space_acquire(ctx->prebuilt->table->space);
|
|
crypt_data = space->crypt_data;
|
|
fil_space_release(space);
|
|
|
|
if (crypt_data) {
|
|
key_id = crypt_data->key_id;
|
|
mode = crypt_data->encryption;
|
|
}
|
|
|
|
if (innobase_check_foreigns(
|
|
ha_alter_info, altered_table, old_table,
|
|
user_table, ctx->drop_fk, ctx->num_to_drop_fk)) {
|
|
goto new_clustered_failed;
|
|
}
|
|
|
|
n_cols = altered_table->s->stored_fields;
|
|
|
|
if (add_fts_doc_id) {
|
|
n_cols++;
|
|
DBUG_ASSERT(flags2 & DICT_TF2_FTS);
|
|
DBUG_ASSERT(add_fts_doc_id_idx);
|
|
flags2 |= DICT_TF2_FTS_ADD_DOC_ID
|
|
| DICT_TF2_FTS_HAS_DOC_ID
|
|
| DICT_TF2_FTS;
|
|
}
|
|
|
|
DBUG_ASSERT(!add_fts_doc_id_idx || (flags2 & DICT_TF2_FTS));
|
|
|
|
/* Create the table. */
|
|
trx_set_dict_operation(ctx->trx, TRX_DICT_OP_TABLE);
|
|
|
|
if (dict_table_get_low(new_table_name)) {
|
|
my_error(ER_TABLE_EXISTS_ERROR, MYF(0),
|
|
new_table_name);
|
|
goto new_clustered_failed;
|
|
}
|
|
|
|
/* The initial space id 0 may be overridden later. */
|
|
ctx->new_table = dict_mem_table_create(
|
|
new_table_name, 0, n_cols, flags, flags2);
|
|
/* The rebuilt indexed_table will use the renamed
|
|
column names. */
|
|
ctx->col_names = NULL;
|
|
|
|
if (DICT_TF_HAS_DATA_DIR(flags)) {
|
|
ctx->new_table->data_dir_path =
|
|
mem_heap_strdup(ctx->new_table->heap,
|
|
user_table->data_dir_path);
|
|
}
|
|
|
|
sql_idx= 0;
|
|
for (uint i = 0; i < altered_table->s->stored_fields; i++, sql_idx++) {
|
|
const Field* field;
|
|
while (!((field= altered_table->field[sql_idx])->
|
|
stored_in_db()))
|
|
sql_idx++;
|
|
ulint is_unsigned;
|
|
ulint field_type
|
|
= (ulint) field->type();
|
|
ulint col_type
|
|
= get_innobase_type_from_mysql_type(
|
|
&is_unsigned, field);
|
|
ulint charset_no;
|
|
ulint col_len;
|
|
|
|
/* we assume in dtype_form_prtype() that this
|
|
fits in two bytes */
|
|
ut_a(field_type <= MAX_CHAR_COLL_NUM);
|
|
|
|
if (!field->real_maybe_null()) {
|
|
field_type |= DATA_NOT_NULL;
|
|
}
|
|
|
|
if (field->binary()) {
|
|
field_type |= DATA_BINARY_TYPE;
|
|
}
|
|
|
|
if (is_unsigned) {
|
|
field_type |= DATA_UNSIGNED;
|
|
}
|
|
|
|
if (dtype_is_string_type(col_type)) {
|
|
charset_no = (ulint) field->charset()->number;
|
|
|
|
if (charset_no > MAX_CHAR_COLL_NUM) {
|
|
dict_mem_table_free(
|
|
ctx->new_table);
|
|
my_error(ER_WRONG_KEY_COLUMN, MYF(0),
|
|
field->field_name);
|
|
goto new_clustered_failed;
|
|
}
|
|
} else {
|
|
charset_no = 0;
|
|
}
|
|
|
|
col_len = field->pack_length();
|
|
|
|
/* The MySQL pack length contains 1 or 2 bytes
|
|
length field for a true VARCHAR. Let us
|
|
subtract that, so that the InnoDB column
|
|
length in the InnoDB data dictionary is the
|
|
real maximum byte length of the actual data. */
|
|
|
|
if (field->type() == MYSQL_TYPE_VARCHAR) {
|
|
uint32 length_bytes
|
|
= static_cast<const Field_varstring*>(
|
|
field)->length_bytes;
|
|
|
|
col_len -= length_bytes;
|
|
|
|
if (length_bytes == 2) {
|
|
field_type |= DATA_LONG_TRUE_VARCHAR;
|
|
}
|
|
}
|
|
|
|
if (dict_col_name_is_reserved(field->field_name)) {
|
|
dict_mem_table_free(ctx->new_table);
|
|
my_error(ER_WRONG_COLUMN_NAME, MYF(0),
|
|
field->field_name);
|
|
goto new_clustered_failed;
|
|
}
|
|
|
|
dict_mem_table_add_col(
|
|
ctx->new_table, ctx->heap,
|
|
field->field_name,
|
|
col_type,
|
|
dtype_form_prtype(field_type, charset_no),
|
|
col_len);
|
|
}
|
|
|
|
if (add_fts_doc_id) {
|
|
fts_add_doc_id_column(ctx->new_table, ctx->heap);
|
|
ctx->new_table->fts->doc_col = fts_doc_id_col;
|
|
ut_ad(fts_doc_id_col == altered_table->s->stored_fields);
|
|
} else if (ctx->new_table->fts) {
|
|
ctx->new_table->fts->doc_col = fts_doc_id_col;
|
|
}
|
|
|
|
error = row_create_table_for_mysql(
|
|
ctx->new_table, ctx->trx, false, mode, key_id);
|
|
|
|
switch (error) {
|
|
dict_table_t* temp_table;
|
|
case DB_SUCCESS:
|
|
/* We need to bump up the table ref count and
|
|
before we can use it we need to open the
|
|
table. The new_table must be in the data
|
|
dictionary cache, because we are still holding
|
|
the dict_sys->mutex. */
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
temp_table = dict_table_open_on_name(
|
|
ctx->new_table->name, TRUE, FALSE,
|
|
DICT_ERR_IGNORE_NONE);
|
|
ut_a(ctx->new_table == temp_table);
|
|
/* n_ref_count must be 1, because purge cannot
|
|
be executing on this very table as we are
|
|
holding dict_operation_lock X-latch. */
|
|
DBUG_ASSERT(ctx->new_table->n_ref_count == 1);
|
|
break;
|
|
case DB_TABLESPACE_EXISTS:
|
|
my_error(ER_TABLESPACE_EXISTS, MYF(0),
|
|
new_table_name);
|
|
goto new_clustered_failed;
|
|
case DB_DUPLICATE_KEY:
|
|
my_error(HA_ERR_TABLE_EXIST, MYF(0),
|
|
altered_table->s->table_name.str);
|
|
goto new_clustered_failed;
|
|
default:
|
|
my_error_innodb(error, table_name, flags);
|
|
new_clustered_failed:
|
|
DBUG_ASSERT(ctx->trx != ctx->prebuilt->trx);
|
|
trx_rollback_to_savepoint(ctx->trx, NULL);
|
|
|
|
ut_ad(user_table->n_ref_count == 1);
|
|
|
|
online_retry_drop_indexes_with_trx(
|
|
user_table, ctx->trx);
|
|
goto err_exit;
|
|
}
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_COLUMN) {
|
|
add_cols = dtuple_create(
|
|
ctx->heap,
|
|
dict_table_get_n_cols(ctx->new_table));
|
|
|
|
dict_table_copy_types(add_cols, ctx->new_table);
|
|
} else {
|
|
add_cols = NULL;
|
|
}
|
|
|
|
ctx->col_map = innobase_build_col_map(
|
|
ha_alter_info, altered_table, old_table,
|
|
ctx->new_table, user_table,
|
|
add_cols, ctx->heap, prebuilt);
|
|
ctx->add_cols = add_cols;
|
|
} else {
|
|
DBUG_ASSERT(!innobase_need_rebuild(ha_alter_info, old_table));
|
|
DBUG_ASSERT(old_table->s->primary_key
|
|
== altered_table->s->primary_key);
|
|
|
|
if (!ctx->new_table->fts
|
|
&& innobase_fulltext_exist(altered_table)) {
|
|
ctx->new_table->fts = fts_create(
|
|
ctx->new_table);
|
|
ctx->new_table->fts->doc_col = fts_doc_id_col;
|
|
}
|
|
}
|
|
|
|
/* Assign table_id, so that no table id of
|
|
fts_create_index_tables() will be written to the undo logs. */
|
|
DBUG_ASSERT(ctx->new_table->id != 0);
|
|
ctx->trx->table_id = ctx->new_table->id;
|
|
|
|
/* Create the indexes in SYS_INDEXES and load into dictionary. */
|
|
|
|
for (ulint a = 0; a < ctx->num_to_add_index; a++) {
|
|
|
|
ctx->add_index[a] = row_merge_create_index(
|
|
ctx->trx, ctx->new_table,
|
|
&index_defs[a], ctx->col_names);
|
|
|
|
add_key_nums[a] = index_defs[a].key_number;
|
|
|
|
if (!ctx->add_index[a]) {
|
|
error = ctx->trx->error_state;
|
|
DBUG_ASSERT(error != DB_SUCCESS);
|
|
goto error_handling;
|
|
}
|
|
|
|
if (ctx->add_index[a]->type & DICT_FTS) {
|
|
DBUG_ASSERT(num_fts_index);
|
|
DBUG_ASSERT(!fts_index);
|
|
DBUG_ASSERT(ctx->add_index[a]->type == DICT_FTS);
|
|
fts_index = ctx->add_index[a];
|
|
}
|
|
|
|
/* If only online ALTER TABLE operations have been
|
|
requested, allocate a modification log. If the table
|
|
will be locked anyway, the modification
|
|
log is unnecessary. When rebuilding the table
|
|
(new_clustered), we will allocate the log for the
|
|
clustered index of the old table, later. */
|
|
if (new_clustered
|
|
|| !ctx->online
|
|
|| user_table->ibd_file_missing
|
|
|| dict_table_is_discarded(user_table)) {
|
|
/* No need to allocate a modification log. */
|
|
ut_ad(!ctx->add_index[a]->online_log);
|
|
} else if (ctx->add_index[a]->type & DICT_FTS) {
|
|
/* Fulltext indexes are not covered
|
|
by a modification log. */
|
|
} else {
|
|
DBUG_EXECUTE_IF("innodb_OOM_prepare_inplace_alter",
|
|
error = DB_OUT_OF_MEMORY;
|
|
goto error_handling;);
|
|
rw_lock_x_lock(&ctx->add_index[a]->lock);
|
|
|
|
bool ok = row_log_allocate(ctx->add_index[a],
|
|
NULL, true, NULL,
|
|
NULL, path);
|
|
rw_lock_x_unlock(&ctx->add_index[a]->lock);
|
|
|
|
if (!ok) {
|
|
error = DB_OUT_OF_MEMORY;
|
|
goto error_handling;
|
|
}
|
|
}
|
|
}
|
|
|
|
ut_ad(new_clustered == ctx->need_rebuild());
|
|
|
|
DBUG_EXECUTE_IF("innodb_OOM_prepare_inplace_alter",
|
|
error = DB_OUT_OF_MEMORY;
|
|
goto error_handling;);
|
|
|
|
if (new_clustered && ctx->online) {
|
|
/* Allocate a log for online table rebuild. */
|
|
dict_index_t* clust_index = dict_table_get_first_index(
|
|
user_table);
|
|
|
|
rw_lock_x_lock(&clust_index->lock);
|
|
bool ok = row_log_allocate(
|
|
clust_index, ctx->new_table,
|
|
!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_PK_INDEX),
|
|
ctx->add_cols, ctx->col_map, path);
|
|
rw_lock_x_unlock(&clust_index->lock);
|
|
|
|
if (!ok) {
|
|
error = DB_OUT_OF_MEMORY;
|
|
goto error_handling;
|
|
}
|
|
}
|
|
|
|
if (ctx->online) {
|
|
/* Assign a consistent read view for
|
|
row_merge_read_clustered_index(). */
|
|
trx_assign_read_view(ctx->prebuilt->trx);
|
|
}
|
|
|
|
if (fts_index) {
|
|
/* Ensure that the dictionary operation mode will
|
|
not change while creating the auxiliary tables. */
|
|
trx_dict_op_t op = trx_get_dict_operation(ctx->trx);
|
|
|
|
#ifdef UNIV_DEBUG
|
|
switch (op) {
|
|
case TRX_DICT_OP_NONE:
|
|
break;
|
|
case TRX_DICT_OP_TABLE:
|
|
case TRX_DICT_OP_INDEX:
|
|
goto op_ok;
|
|
}
|
|
ut_error;
|
|
op_ok:
|
|
#endif /* UNIV_DEBUG */
|
|
ut_ad(ctx->trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
ut_ad(rw_lock_own(&dict_operation_lock, RW_LOCK_EX));
|
|
#endif /* UNIV_SYNC_DEBUG */
|
|
|
|
DICT_TF2_FLAG_SET(ctx->new_table, DICT_TF2_FTS);
|
|
|
|
/* This function will commit the transaction and reset
|
|
the trx_t::dict_operation flag on success. */
|
|
|
|
error = fts_create_index_tables(ctx->trx, fts_index);
|
|
|
|
DBUG_EXECUTE_IF("innodb_test_fail_after_fts_index_table",
|
|
error = DB_LOCK_WAIT_TIMEOUT;
|
|
goto error_handling;);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
goto error_handling;
|
|
}
|
|
|
|
trx_start_for_ddl(ctx->trx, op);
|
|
|
|
if (!ctx->new_table->fts
|
|
|| ib_vector_size(ctx->new_table->fts->indexes) == 0) {
|
|
error = fts_create_common_tables(
|
|
ctx->trx, ctx->new_table,
|
|
user_table->name, TRUE);
|
|
|
|
DBUG_EXECUTE_IF(
|
|
"innodb_test_fail_after_fts_common_table",
|
|
error = DB_LOCK_WAIT_TIMEOUT;);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
goto error_handling;
|
|
}
|
|
|
|
ctx->new_table->fts->fts_status
|
|
|= TABLE_DICT_LOCKED;
|
|
|
|
error = innobase_fts_load_stopword(
|
|
ctx->new_table, ctx->trx,
|
|
ctx->prebuilt->trx->mysql_thd)
|
|
? DB_SUCCESS : DB_ERROR;
|
|
ctx->new_table->fts->fts_status
|
|
&= ~TABLE_DICT_LOCKED;
|
|
|
|
if (error != DB_SUCCESS) {
|
|
goto error_handling;
|
|
}
|
|
}
|
|
|
|
ut_ad(trx_get_dict_operation(ctx->trx) == op);
|
|
}
|
|
|
|
DBUG_ASSERT(error == DB_SUCCESS);
|
|
|
|
/* Commit the data dictionary transaction in order to release
|
|
the table locks on the system tables. This means that if
|
|
MySQL crashes while creating a new primary key inside
|
|
row_merge_build_indexes(), ctx->new_table will not be dropped
|
|
by trx_rollback_active(). It will have to be recovered or
|
|
dropped by the database administrator. */
|
|
trx_commit_for_mysql(ctx->trx);
|
|
|
|
row_mysql_unlock_data_dictionary(ctx->trx);
|
|
dict_locked = false;
|
|
|
|
ut_a(ctx->trx->lock.n_active_thrs == 0);
|
|
|
|
DBUG_EXECUTE_IF("crash_innodb_add_index_after", DBUG_SUICIDE(););
|
|
|
|
error_handling:
|
|
/* After an error, remove all those index definitions from the
|
|
dictionary which were defined. */
|
|
|
|
switch (error) {
|
|
case DB_SUCCESS:
|
|
ut_a(!dict_locked);
|
|
|
|
ut_d(mutex_enter(&dict_sys->mutex));
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
user_table, CHECK_PARTIAL_OK));
|
|
ut_d(mutex_exit(&dict_sys->mutex));
|
|
DBUG_RETURN(false);
|
|
case DB_TABLESPACE_EXISTS:
|
|
my_error(ER_TABLESPACE_EXISTS, MYF(0), "(unknown)");
|
|
break;
|
|
case DB_DUPLICATE_KEY:
|
|
my_error(ER_DUP_KEY, MYF(0), "SYS_INDEXES");
|
|
break;
|
|
case DB_OUT_OF_FILE_SPACE:
|
|
my_error_innodb(error, table_name, user_table->flags);
|
|
break;
|
|
default:
|
|
my_error_innodb(error, table_name, user_table->flags);
|
|
}
|
|
|
|
error_handled:
|
|
|
|
ctx->prebuilt->trx->error_info = NULL;
|
|
ctx->trx->error_state = DB_SUCCESS;
|
|
|
|
if (!dict_locked) {
|
|
row_mysql_lock_data_dictionary(ctx->trx);
|
|
}
|
|
|
|
if (new_clustered) {
|
|
if (ctx->need_rebuild()) {
|
|
|
|
if (DICT_TF2_FLAG_IS_SET(
|
|
ctx->new_table, DICT_TF2_FTS)) {
|
|
innobase_drop_fts_index_table(
|
|
ctx->new_table, ctx->trx);
|
|
}
|
|
|
|
dict_table_close(ctx->new_table, TRUE, FALSE);
|
|
|
|
#if defined UNIV_DEBUG || defined UNIV_DDL_DEBUG
|
|
/* Nobody should have initialized the stats of the
|
|
newly created table yet. When this is the case, we
|
|
know that it has not been added for background stats
|
|
gathering. */
|
|
ut_a(!ctx->new_table->stat_initialized);
|
|
#endif /* UNIV_DEBUG || UNIV_DDL_DEBUG */
|
|
|
|
row_merge_drop_table(ctx->trx, ctx->new_table);
|
|
|
|
/* Free the log for online table rebuild, if
|
|
one was allocated. */
|
|
|
|
dict_index_t* clust_index = dict_table_get_first_index(
|
|
user_table);
|
|
|
|
rw_lock_x_lock(&clust_index->lock);
|
|
|
|
if (clust_index->online_log) {
|
|
ut_ad(ctx->online);
|
|
row_log_abort_sec(clust_index);
|
|
clust_index->online_status
|
|
= ONLINE_INDEX_COMPLETE;
|
|
}
|
|
|
|
rw_lock_x_unlock(&clust_index->lock);
|
|
}
|
|
|
|
trx_commit_for_mysql(ctx->trx);
|
|
/* n_ref_count must be 1, because purge cannot
|
|
be executing on this very table as we are
|
|
holding dict_operation_lock X-latch. */
|
|
DBUG_ASSERT(user_table->n_ref_count == 1 || ctx->online);
|
|
|
|
online_retry_drop_indexes_with_trx(user_table, ctx->trx);
|
|
} else {
|
|
ut_ad(!ctx->need_rebuild());
|
|
row_merge_drop_indexes(ctx->trx, user_table, TRUE);
|
|
trx_commit_for_mysql(ctx->trx);
|
|
}
|
|
|
|
ut_d(dict_table_check_for_dup_indexes(user_table, CHECK_ALL_COMPLETE));
|
|
ut_ad(!user_table->drop_aborted);
|
|
|
|
err_exit:
|
|
/* Clear the to_be_dropped flag in the data dictionary cache. */
|
|
for (ulint i = 0; i < ctx->num_to_drop_index; i++) {
|
|
DBUG_ASSERT(*ctx->drop_index[i]->name != TEMP_INDEX_PREFIX);
|
|
DBUG_ASSERT(ctx->drop_index[i]->to_be_dropped);
|
|
ctx->drop_index[i]->to_be_dropped = 0;
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(ctx->trx);
|
|
|
|
trx_free_for_mysql(ctx->trx);
|
|
trx_commit_for_mysql(ctx->prebuilt->trx);
|
|
|
|
delete ctx;
|
|
ha_alter_info->handler_ctx = NULL;
|
|
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/* Check whether an index is needed for the foreign key constraint.
|
|
If so, if it is dropped, is there an equivalent index can play its role.
|
|
@return true if the index is needed and can't be dropped */
|
|
static MY_ATTRIBUTE((nonnull(1,2,3,5), warn_unused_result))
|
|
bool
|
|
innobase_check_foreign_key_index(
|
|
/*=============================*/
|
|
Alter_inplace_info* ha_alter_info, /*!< in: Structure describing
|
|
changes to be done by ALTER
|
|
TABLE */
|
|
dict_index_t* index, /*!< in: index to check */
|
|
dict_table_t* indexed_table, /*!< in: table that owns the
|
|
foreign keys */
|
|
const char** col_names, /*!< in: column names, or NULL
|
|
for indexed_table->col_names */
|
|
trx_t* trx, /*!< in/out: transaction */
|
|
dict_foreign_t** drop_fk, /*!< in: Foreign key constraints
|
|
to drop */
|
|
ulint n_drop_fk) /*!< in: Number of foreign keys
|
|
to drop */
|
|
{
|
|
ut_ad(index != NULL);
|
|
ut_ad(indexed_table != NULL);
|
|
|
|
const dict_foreign_set* fks = &indexed_table->referenced_set;
|
|
|
|
/* Check for all FK references from other tables to the index. */
|
|
for (dict_foreign_set::const_iterator it = fks->begin();
|
|
it != fks->end(); ++it) {
|
|
|
|
dict_foreign_t* foreign = *it;
|
|
if (foreign->referenced_index != index) {
|
|
continue;
|
|
}
|
|
ut_ad(indexed_table == foreign->referenced_table);
|
|
|
|
if (NULL == dict_foreign_find_index(
|
|
indexed_table, col_names,
|
|
foreign->referenced_col_names,
|
|
foreign->n_fields, index,
|
|
/*check_charsets=*/TRUE,
|
|
/*check_null=*/FALSE,
|
|
0,0,0)
|
|
&& NULL == innobase_find_equiv_index(
|
|
foreign->referenced_col_names,
|
|
foreign->n_fields,
|
|
ha_alter_info->key_info_buffer,
|
|
ha_alter_info->index_add_buffer,
|
|
ha_alter_info->index_add_count)) {
|
|
|
|
/* Index cannot be dropped. */
|
|
trx->error_info = index;
|
|
return(true);
|
|
}
|
|
}
|
|
|
|
fks = &indexed_table->foreign_set;
|
|
|
|
/* Check for all FK references in current table using the index. */
|
|
for (dict_foreign_set::const_iterator it = fks->begin();
|
|
it != fks->end(); ++it) {
|
|
|
|
dict_foreign_t* foreign = *it;
|
|
if (foreign->foreign_index != index) {
|
|
continue;
|
|
}
|
|
|
|
ut_ad(indexed_table == foreign->foreign_table);
|
|
|
|
if (!innobase_dropping_foreign(
|
|
foreign, drop_fk, n_drop_fk)
|
|
&& NULL == dict_foreign_find_index(
|
|
indexed_table, col_names,
|
|
foreign->foreign_col_names,
|
|
foreign->n_fields, index,
|
|
/*check_charsets=*/TRUE,
|
|
/*check_null=*/FALSE,
|
|
0, 0, 0)
|
|
&& NULL == innobase_find_equiv_index(
|
|
foreign->foreign_col_names,
|
|
foreign->n_fields,
|
|
ha_alter_info->key_info_buffer,
|
|
ha_alter_info->index_add_buffer,
|
|
ha_alter_info->index_add_count)) {
|
|
|
|
/* Index cannot be dropped. */
|
|
trx->error_info = index;
|
|
return(true);
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/** Allows InnoDB to update internal structures with concurrent
|
|
writes blocked (provided that check_if_supported_inplace_alter()
|
|
did not return HA_ALTER_INPLACE_NO_LOCK).
|
|
This will be invoked before inplace_alter_table().
|
|
|
|
@param altered_table TABLE object for new version of table.
|
|
@param ha_alter_info Structure describing changes to be done
|
|
by ALTER TABLE and holding data used during in-place alter.
|
|
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
UNIV_INTERN
|
|
bool
|
|
ha_innobase::prepare_inplace_alter_table(
|
|
/*=====================================*/
|
|
TABLE* altered_table,
|
|
Alter_inplace_info* ha_alter_info)
|
|
{
|
|
dict_index_t** drop_index; /*!< Index to be dropped */
|
|
ulint n_drop_index; /*!< Number of indexes to drop */
|
|
dict_foreign_t**drop_fk; /*!< Foreign key constraints to drop */
|
|
ulint n_drop_fk; /*!< Number of foreign keys to drop */
|
|
dict_foreign_t**add_fk = NULL; /*!< Foreign key constraints to drop */
|
|
ulint n_add_fk; /*!< Number of foreign keys to drop */
|
|
dict_table_t* indexed_table; /*!< Table where indexes are created */
|
|
mem_heap_t* heap;
|
|
const char** col_names;
|
|
int error;
|
|
ulint flags;
|
|
ulint flags2;
|
|
ulint max_col_len;
|
|
ulint add_autoinc_col_no = ULINT_UNDEFINED;
|
|
ulonglong autoinc_col_max_value = 0;
|
|
ulint fts_doc_col_no = ULINT_UNDEFINED;
|
|
bool add_fts_doc_id = false;
|
|
bool add_fts_doc_id_idx = false;
|
|
bool add_fts_idx = false;
|
|
|
|
DBUG_ENTER("prepare_inplace_alter_table");
|
|
DBUG_ASSERT(!ha_alter_info->handler_ctx);
|
|
DBUG_ASSERT(ha_alter_info->create_info);
|
|
DBUG_ASSERT(!srv_read_only_mode);
|
|
|
|
if (UNIV_UNLIKELY(prebuilt->trx->fake_changes)) {
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/* Init online ddl status variables */
|
|
onlineddl_rowlog_rows = 0;
|
|
onlineddl_rowlog_pct_used = 0;
|
|
onlineddl_pct_progress = 0;
|
|
|
|
MONITOR_ATOMIC_INC(MONITOR_PENDING_ALTER_TABLE);
|
|
|
|
#ifdef UNIV_DEBUG
|
|
for (dict_index_t* index = dict_table_get_first_index(prebuilt->table);
|
|
index;
|
|
index = dict_table_get_next_index(index)) {
|
|
ut_ad(!index->to_be_dropped);
|
|
}
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
ut_d(mutex_enter(&dict_sys->mutex));
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
prebuilt->table, CHECK_ABORTED_OK));
|
|
ut_d(mutex_exit(&dict_sys->mutex));
|
|
|
|
if (!(ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE)) {
|
|
/* Nothing to do */
|
|
goto func_exit;
|
|
}
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::CHANGE_CREATE_OPTION) {
|
|
/* Check engine specific table options */
|
|
if (const char* invalid_tbopt = check_table_options(
|
|
user_thd, altered_table,
|
|
ha_alter_info->create_info,
|
|
prebuilt->table->space != 0,
|
|
srv_file_format)) {
|
|
my_error(ER_ILLEGAL_HA_CREATE_OPTION, MYF(0),
|
|
table_type(), invalid_tbopt);
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
if (const char* invalid_opt = create_options_are_invalid(
|
|
user_thd, altered_table,
|
|
ha_alter_info->create_info,
|
|
prebuilt->table->space != 0)) {
|
|
my_error(ER_ILLEGAL_HA_CREATE_OPTION, MYF(0),
|
|
table_type(), invalid_opt);
|
|
goto err_exit_no_heap;
|
|
}
|
|
}
|
|
|
|
/* Check if any index name is reserved. */
|
|
if (innobase_index_name_is_reserved(
|
|
user_thd,
|
|
ha_alter_info->key_info_buffer,
|
|
ha_alter_info->key_count)) {
|
|
err_exit_no_heap:
|
|
DBUG_ASSERT(prebuilt->trx->dict_operation_lock_mode == 0);
|
|
if (ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE) {
|
|
online_retry_drop_indexes(prebuilt->table, user_thd);
|
|
}
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
indexed_table = prebuilt->table;
|
|
|
|
/* Check that index keys are sensible */
|
|
error = innobase_check_index_keys(ha_alter_info, indexed_table);
|
|
|
|
if (error) {
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
/* Prohibit renaming a column to something that the table
|
|
already contains. */
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME) {
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
|
|
for (Field** fp = table->field; *fp; fp++) {
|
|
if (!((*fp)->flags & FIELD_IS_RENAMED)) {
|
|
continue;
|
|
}
|
|
|
|
const char* name = 0;
|
|
|
|
cf_it.rewind();
|
|
while (Create_field* cf = cf_it++) {
|
|
if (cf->field == *fp) {
|
|
name = cf->field_name;
|
|
goto check_if_ok_to_rename;
|
|
}
|
|
}
|
|
|
|
ut_error;
|
|
check_if_ok_to_rename:
|
|
/* Prohibit renaming a column from FTS_DOC_ID
|
|
if full-text indexes exist. */
|
|
if (!my_strcasecmp(system_charset_info,
|
|
(*fp)->field_name,
|
|
FTS_DOC_ID_COL_NAME)
|
|
&& innobase_fulltext_exist(altered_table)) {
|
|
my_error(ER_INNODB_FT_WRONG_DOCID_COLUMN,
|
|
MYF(0), name);
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
/* Prohibit renaming a column to an internal column. */
|
|
const char* s = prebuilt->table->col_names;
|
|
unsigned j;
|
|
/* Skip user columns.
|
|
MySQL should have checked these already.
|
|
We want to allow renaming of c1 to c2, c2 to c1. */
|
|
for (j = 0; j < table->s->fields; j++) {
|
|
s += strlen(s) + 1;
|
|
}
|
|
|
|
for (; j < prebuilt->table->n_def; j++) {
|
|
if (!my_strcasecmp(
|
|
system_charset_info, name, s)) {
|
|
my_error(ER_WRONG_COLUMN_NAME, MYF(0),
|
|
s);
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
s += strlen(s) + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!innobase_table_flags(altered_table,
|
|
ha_alter_info->create_info,
|
|
user_thd,
|
|
srv_file_per_table
|
|
|| indexed_table->space != 0,
|
|
&flags, &flags2)) {
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
max_col_len = DICT_MAX_FIELD_LEN_BY_FORMAT_FLAG(flags);
|
|
|
|
/* Check each index's column length to make sure they do not
|
|
exceed limit */
|
|
for (ulint i = 0; i < ha_alter_info->index_add_count; i++) {
|
|
const KEY* key = &ha_alter_info->key_info_buffer[
|
|
ha_alter_info->index_add_buffer[i]];
|
|
|
|
if (key->flags & HA_FULLTEXT) {
|
|
/* The column length does not matter for
|
|
fulltext search indexes. But, UNIQUE
|
|
fulltext indexes are not supported. */
|
|
DBUG_ASSERT(!(key->flags & HA_NOSAME));
|
|
DBUG_ASSERT(!(key->flags & HA_KEYFLAG_MASK
|
|
& ~(HA_FULLTEXT
|
|
| HA_PACK_KEY
|
|
| HA_BINARY_PACK_KEY)));
|
|
add_fts_idx = true;
|
|
continue;
|
|
}
|
|
|
|
if (innobase_check_column_length(max_col_len, key)) {
|
|
my_error(ER_INDEX_COLUMN_TOO_LONG, MYF(0),
|
|
max_col_len);
|
|
goto err_exit_no_heap;
|
|
}
|
|
}
|
|
|
|
/* We won't be allowed to add fts index to a table with
|
|
fts indexes already but without AUX_HEX_NAME set.
|
|
This means the aux tables of the table failed to
|
|
rename to hex format but new created aux tables
|
|
shall be in hex format, which is contradictory. */
|
|
if (!DICT_TF2_FLAG_IS_SET(indexed_table, DICT_TF2_FTS_AUX_HEX_NAME)
|
|
&& indexed_table->fts != NULL && add_fts_idx) {
|
|
my_error(ER_INNODB_FT_AUX_NOT_HEX_ID, MYF(0));
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
/* Check existing index definitions for too-long column
|
|
prefixes as well, in case max_col_len shrunk. */
|
|
for (const dict_index_t* index
|
|
= dict_table_get_first_index(indexed_table);
|
|
index;
|
|
index = dict_table_get_next_index(index)) {
|
|
if (index->type & DICT_FTS) {
|
|
DBUG_ASSERT(index->type == DICT_FTS
|
|
|| (index->type & DICT_CORRUPT));
|
|
|
|
/* We need to drop any corrupted fts indexes
|
|
before we add a new fts index. */
|
|
if (add_fts_idx && index->type & DICT_CORRUPT) {
|
|
ib_errf(user_thd, IB_LOG_LEVEL_ERROR,
|
|
ER_INNODB_INDEX_CORRUPT,
|
|
"Fulltext index '%s' is corrupt. "
|
|
"you should drop this index first.",
|
|
index->name);
|
|
|
|
goto err_exit_no_heap;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
for (ulint i = 0; i < dict_index_get_n_fields(index); i++) {
|
|
const dict_field_t* field
|
|
= dict_index_get_nth_field(index, i);
|
|
if (field->prefix_len > max_col_len) {
|
|
my_error(ER_INDEX_COLUMN_TOO_LONG, MYF(0),
|
|
max_col_len);
|
|
goto err_exit_no_heap;
|
|
}
|
|
}
|
|
}
|
|
|
|
n_drop_index = 0;
|
|
n_drop_fk = 0;
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& (INNOBASE_ALTER_NOREBUILD | INNOBASE_ALTER_REBUILD)) {
|
|
heap = mem_heap_create(1024);
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME) {
|
|
col_names = innobase_get_col_names(
|
|
ha_alter_info, altered_table, table,
|
|
indexed_table, heap);
|
|
} else {
|
|
col_names = NULL;
|
|
}
|
|
} else {
|
|
heap = NULL;
|
|
col_names = NULL;
|
|
}
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::DROP_FOREIGN_KEY) {
|
|
DBUG_ASSERT(ha_alter_info->alter_info->drop_list.elements > 0);
|
|
|
|
drop_fk = static_cast<dict_foreign_t**>(
|
|
mem_heap_alloc(
|
|
heap,
|
|
ha_alter_info->alter_info->drop_list.elements
|
|
* sizeof(dict_foreign_t*)));
|
|
|
|
List_iterator<Alter_drop> drop_it(
|
|
ha_alter_info->alter_info->drop_list);
|
|
|
|
while (Alter_drop* drop = drop_it++) {
|
|
if (drop->type != Alter_drop::FOREIGN_KEY) {
|
|
continue;
|
|
}
|
|
|
|
for (dict_foreign_set::iterator it
|
|
= prebuilt->table->foreign_set.begin();
|
|
it != prebuilt->table->foreign_set.end();
|
|
++it) {
|
|
|
|
dict_foreign_t* foreign = *it;
|
|
const char* fid = strchr(foreign->id, '/');
|
|
|
|
DBUG_ASSERT(fid);
|
|
/* If no database/ prefix was present in
|
|
the FOREIGN KEY constraint name, compare
|
|
to the full constraint name. */
|
|
fid = fid ? fid + 1 : foreign->id;
|
|
|
|
if (!my_strcasecmp(system_charset_info,
|
|
fid, drop->name)) {
|
|
drop_fk[n_drop_fk++] = foreign;
|
|
goto found_fk;
|
|
}
|
|
}
|
|
|
|
my_error(ER_CANT_DROP_FIELD_OR_KEY, MYF(0),
|
|
drop->type_name(), drop->name);
|
|
goto err_exit;
|
|
found_fk:
|
|
continue;
|
|
}
|
|
|
|
DBUG_ASSERT(n_drop_fk > 0);
|
|
DBUG_ASSERT(n_drop_fk
|
|
== ha_alter_info->alter_info->drop_list.elements);
|
|
} else {
|
|
drop_fk = NULL;
|
|
}
|
|
|
|
if (ha_alter_info->index_drop_count) {
|
|
dict_index_t* drop_primary = NULL;
|
|
|
|
DBUG_ASSERT(ha_alter_info->handler_flags
|
|
& (Alter_inplace_info::DROP_INDEX
|
|
| Alter_inplace_info::DROP_UNIQUE_INDEX
|
|
| Alter_inplace_info::DROP_PK_INDEX));
|
|
/* Check which indexes to drop. */
|
|
drop_index = static_cast<dict_index_t**>(
|
|
mem_heap_alloc(
|
|
heap, (ha_alter_info->index_drop_count + 1)
|
|
* sizeof *drop_index));
|
|
|
|
for (uint i = 0; i < ha_alter_info->index_drop_count; i++) {
|
|
const KEY* key
|
|
= ha_alter_info->index_drop_buffer[i];
|
|
dict_index_t* index
|
|
= dict_table_get_index_on_name_and_min_id(
|
|
indexed_table, key->name);
|
|
|
|
if (!index) {
|
|
push_warning_printf(
|
|
user_thd,
|
|
Sql_condition::WARN_LEVEL_WARN,
|
|
HA_ERR_WRONG_INDEX,
|
|
"InnoDB could not find key "
|
|
"with name %s", key->name);
|
|
} else {
|
|
ut_ad(!index->to_be_dropped);
|
|
if (!dict_index_is_clust(index)) {
|
|
drop_index[n_drop_index++] = index;
|
|
} else {
|
|
drop_primary = index;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* If all FULLTEXT indexes were removed, drop an
|
|
internal FTS_DOC_ID_INDEX as well, unless it exists in
|
|
the table. */
|
|
|
|
if (innobase_fulltext_exist(table)
|
|
&& !innobase_fulltext_exist(altered_table)
|
|
&& !DICT_TF2_FLAG_IS_SET(
|
|
indexed_table, DICT_TF2_FTS_HAS_DOC_ID)) {
|
|
dict_index_t* fts_doc_index
|
|
= dict_table_get_index_on_name(
|
|
indexed_table, FTS_DOC_ID_INDEX_NAME);
|
|
|
|
// Add some fault tolerance for non-debug builds.
|
|
if (fts_doc_index == NULL) {
|
|
goto check_if_can_drop_indexes;
|
|
}
|
|
|
|
DBUG_ASSERT(!fts_doc_index->to_be_dropped);
|
|
|
|
for (uint i = 0; i < table->s->keys; i++) {
|
|
if (!my_strcasecmp(
|
|
system_charset_info,
|
|
FTS_DOC_ID_INDEX_NAME,
|
|
table->key_info[i].name)) {
|
|
/* The index exists in the MySQL
|
|
data dictionary. Do not drop it,
|
|
even though it is no longer needed
|
|
by InnoDB fulltext search. */
|
|
goto check_if_can_drop_indexes;
|
|
}
|
|
}
|
|
|
|
drop_index[n_drop_index++] = fts_doc_index;
|
|
}
|
|
|
|
check_if_can_drop_indexes:
|
|
/* Check if the indexes can be dropped. */
|
|
|
|
/* Prevent a race condition between DROP INDEX and
|
|
CREATE TABLE adding FOREIGN KEY constraints. */
|
|
row_mysql_lock_data_dictionary(prebuilt->trx);
|
|
|
|
if (!n_drop_index) {
|
|
drop_index = NULL;
|
|
} else {
|
|
/* Flag all indexes that are to be dropped. */
|
|
for (ulint i = 0; i < n_drop_index; i++) {
|
|
ut_ad(!drop_index[i]->to_be_dropped);
|
|
drop_index[i]->to_be_dropped = 1;
|
|
}
|
|
}
|
|
|
|
if (prebuilt->trx->check_foreigns) {
|
|
for (uint i = 0; i < n_drop_index; i++) {
|
|
dict_index_t* index = drop_index[i];
|
|
|
|
if (innobase_check_foreign_key_index(
|
|
ha_alter_info, index,
|
|
indexed_table, col_names,
|
|
prebuilt->trx, drop_fk, n_drop_fk)) {
|
|
row_mysql_unlock_data_dictionary(
|
|
prebuilt->trx);
|
|
prebuilt->trx->error_info = index;
|
|
print_error(HA_ERR_DROP_INDEX_FK,
|
|
MYF(0));
|
|
goto err_exit;
|
|
}
|
|
}
|
|
|
|
/* If a primary index is dropped, need to check
|
|
any depending foreign constraints get affected */
|
|
if (drop_primary
|
|
&& innobase_check_foreign_key_index(
|
|
ha_alter_info, drop_primary,
|
|
indexed_table, col_names,
|
|
prebuilt->trx, drop_fk, n_drop_fk)) {
|
|
row_mysql_unlock_data_dictionary(prebuilt->trx);
|
|
print_error(HA_ERR_DROP_INDEX_FK, MYF(0));
|
|
goto err_exit;
|
|
}
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(prebuilt->trx);
|
|
} else {
|
|
drop_index = NULL;
|
|
}
|
|
|
|
/* Check if any of the existing indexes are marked as corruption
|
|
and if they are, refuse adding more indexes. */
|
|
if (ha_alter_info->handler_flags & Alter_inplace_info::ADD_INDEX) {
|
|
for (dict_index_t* index = dict_table_get_first_index(indexed_table);
|
|
index != NULL; index = dict_table_get_next_index(index)) {
|
|
|
|
if (!index->to_be_dropped && dict_index_is_corrupted(index)) {
|
|
char index_name[MAX_FULL_NAME_LEN + 1];
|
|
|
|
innobase_format_name(index_name, sizeof index_name,
|
|
index->name, TRUE);
|
|
|
|
my_error(ER_INDEX_CORRUPT, MYF(0), index_name);
|
|
goto err_exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
n_add_fk = 0;
|
|
|
|
if (ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_FOREIGN_KEY) {
|
|
ut_ad(!prebuilt->trx->check_foreigns);
|
|
|
|
add_fk = static_cast<dict_foreign_t**>(
|
|
mem_heap_zalloc(
|
|
heap,
|
|
ha_alter_info->alter_info->key_list.elements
|
|
* sizeof(dict_foreign_t*)));
|
|
|
|
if (!innobase_get_foreign_key_info(
|
|
ha_alter_info, table_share,
|
|
prebuilt->table, col_names,
|
|
drop_index, n_drop_index,
|
|
add_fk, &n_add_fk, prebuilt->trx)) {
|
|
err_exit:
|
|
if (n_drop_index) {
|
|
row_mysql_lock_data_dictionary(prebuilt->trx);
|
|
|
|
/* Clear the to_be_dropped flags, which might
|
|
have been set at this point. */
|
|
for (ulint i = 0; i < n_drop_index; i++) {
|
|
DBUG_ASSERT(*drop_index[i]->name
|
|
!= TEMP_INDEX_PREFIX);
|
|
drop_index[i]->to_be_dropped = 0;
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(prebuilt->trx);
|
|
}
|
|
|
|
if (heap) {
|
|
mem_heap_free(heap);
|
|
}
|
|
|
|
goto err_exit_no_heap;
|
|
}
|
|
}
|
|
|
|
if (!(ha_alter_info->handler_flags & INNOBASE_ALTER_DATA)
|
|
|| ((ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE)
|
|
== Alter_inplace_info::CHANGE_CREATE_OPTION
|
|
&& !innobase_need_rebuild(ha_alter_info, table))) {
|
|
|
|
if (heap) {
|
|
ha_alter_info->handler_ctx
|
|
= new ha_innobase_inplace_ctx(
|
|
prebuilt,
|
|
drop_index, n_drop_index,
|
|
drop_fk, n_drop_fk,
|
|
add_fk, n_add_fk,
|
|
ha_alter_info->online,
|
|
heap, indexed_table,
|
|
col_names, ULINT_UNDEFINED, 0, 0);
|
|
}
|
|
|
|
func_exit:
|
|
DBUG_ASSERT(prebuilt->trx->dict_operation_lock_mode == 0);
|
|
if (ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE) {
|
|
online_retry_drop_indexes(prebuilt->table, user_thd);
|
|
}
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
/* If we are to build a full-text search index, check whether
|
|
the table already has a DOC ID column. If not, we will need to
|
|
add a Doc ID hidden column and rebuild the primary index */
|
|
if (innobase_fulltext_exist(altered_table)) {
|
|
ulint doc_col_no;
|
|
|
|
if (!innobase_fts_check_doc_id_col(
|
|
prebuilt->table, altered_table, &fts_doc_col_no)) {
|
|
fts_doc_col_no = altered_table->s->stored_fields;
|
|
add_fts_doc_id = true;
|
|
add_fts_doc_id_idx = true;
|
|
|
|
push_warning_printf(
|
|
user_thd,
|
|
Sql_condition::WARN_LEVEL_WARN,
|
|
HA_ERR_WRONG_INDEX,
|
|
"InnoDB rebuilding table to add column "
|
|
FTS_DOC_ID_COL_NAME);
|
|
} else if (fts_doc_col_no == ULINT_UNDEFINED) {
|
|
goto err_exit;
|
|
}
|
|
|
|
switch (innobase_fts_check_doc_id_index(
|
|
prebuilt->table, altered_table, &doc_col_no)) {
|
|
case FTS_NOT_EXIST_DOC_ID_INDEX:
|
|
add_fts_doc_id_idx = true;
|
|
break;
|
|
case FTS_INCORRECT_DOC_ID_INDEX:
|
|
my_error(ER_INNODB_FT_WRONG_DOCID_INDEX, MYF(0),
|
|
FTS_DOC_ID_INDEX_NAME);
|
|
goto err_exit;
|
|
case FTS_EXIST_DOC_ID_INDEX:
|
|
DBUG_ASSERT(doc_col_no == fts_doc_col_no
|
|
|| doc_col_no == ULINT_UNDEFINED
|
|
|| (ha_alter_info->handler_flags
|
|
& (Alter_inplace_info::ALTER_COLUMN_ORDER
|
|
| Alter_inplace_info::DROP_COLUMN
|
|
| Alter_inplace_info::ADD_COLUMN)));
|
|
}
|
|
}
|
|
|
|
/* See if an AUTO_INCREMENT column was added. */
|
|
uint i = 0, innodb_idx= 0;
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
while (const Create_field* new_field = cf_it++) {
|
|
const Field* field;
|
|
if (!new_field->stored_in_db()) {
|
|
i++;
|
|
continue;
|
|
}
|
|
|
|
DBUG_ASSERT(i < altered_table->s->fields);
|
|
DBUG_ASSERT(innodb_idx < altered_table->s->stored_fields);
|
|
|
|
for (uint old_i = 0; table->field[old_i]; old_i++) {
|
|
if (!table->field[old_i]->stored_in_db())
|
|
continue;
|
|
if (new_field->field == table->field[old_i]) {
|
|
goto found_col;
|
|
}
|
|
}
|
|
|
|
/* This is an added column. */
|
|
DBUG_ASSERT(!new_field->field);
|
|
DBUG_ASSERT(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_COLUMN);
|
|
|
|
field = altered_table->field[i];
|
|
|
|
DBUG_ASSERT((MTYP_TYPENR(field->unireg_check)
|
|
== Field::NEXT_NUMBER)
|
|
== !!(field->flags & AUTO_INCREMENT_FLAG));
|
|
|
|
if (field->flags & AUTO_INCREMENT_FLAG) {
|
|
if (add_autoinc_col_no != ULINT_UNDEFINED) {
|
|
/* This should have been blocked earlier. */
|
|
ut_ad(0);
|
|
my_error(ER_WRONG_AUTO_KEY, MYF(0));
|
|
goto err_exit;
|
|
}
|
|
add_autoinc_col_no = innodb_idx;
|
|
|
|
autoinc_col_max_value = innobase_get_int_col_max_value(
|
|
field);
|
|
}
|
|
found_col:
|
|
i++;
|
|
innodb_idx++;
|
|
}
|
|
|
|
DBUG_ASSERT(heap);
|
|
DBUG_ASSERT(user_thd == prebuilt->trx->mysql_thd);
|
|
DBUG_ASSERT(!ha_alter_info->handler_ctx);
|
|
|
|
ha_alter_info->handler_ctx = new ha_innobase_inplace_ctx(
|
|
prebuilt,
|
|
drop_index, n_drop_index,
|
|
drop_fk, n_drop_fk, add_fk, n_add_fk,
|
|
ha_alter_info->online,
|
|
heap, prebuilt->table, col_names,
|
|
add_autoinc_col_no,
|
|
ha_alter_info->create_info->auto_increment_value,
|
|
autoinc_col_max_value);
|
|
|
|
DBUG_RETURN(prepare_inplace_alter_table_dict(
|
|
ha_alter_info, altered_table, table,
|
|
table_share->table_name.str,
|
|
flags, flags2,
|
|
fts_doc_col_no, add_fts_doc_id,
|
|
add_fts_doc_id_idx, prebuilt));
|
|
}
|
|
|
|
/** Get the name of an erroneous key.
|
|
@param[in] error_key_num InnoDB number of the erroneus key
|
|
@param[in] ha_alter_info changes that were being performed
|
|
@param[in] table InnoDB table
|
|
@return the name of the erroneous key */
|
|
static
|
|
const char*
|
|
get_error_key_name(
|
|
ulint error_key_num,
|
|
const Alter_inplace_info* ha_alter_info,
|
|
const dict_table_t* table)
|
|
{
|
|
if (error_key_num == ULINT_UNDEFINED) {
|
|
return(FTS_DOC_ID_INDEX_NAME);
|
|
} else if (ha_alter_info->key_count == 0) {
|
|
return(dict_table_get_first_index(table)->name);
|
|
} else {
|
|
return(ha_alter_info->key_info_buffer[error_key_num].name);
|
|
}
|
|
}
|
|
|
|
/** Alter the table structure in-place with operations
|
|
specified using Alter_inplace_info.
|
|
The level of concurrency allowed during this operation depends
|
|
on the return value from check_if_supported_inplace_alter().
|
|
|
|
@param altered_table TABLE object for new version of table.
|
|
@param ha_alter_info Structure describing changes to be done
|
|
by ALTER TABLE and holding data used during in-place alter.
|
|
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
UNIV_INTERN
|
|
bool
|
|
ha_innobase::inplace_alter_table(
|
|
/*=============================*/
|
|
TABLE* altered_table,
|
|
Alter_inplace_info* ha_alter_info)
|
|
{
|
|
dberr_t error;
|
|
|
|
DBUG_ENTER("inplace_alter_table");
|
|
DBUG_ASSERT(!srv_read_only_mode);
|
|
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
ut_ad(!rw_lock_own(&dict_operation_lock, RW_LOCK_EX));
|
|
ut_ad(!rw_lock_own(&dict_operation_lock, RW_LOCK_SHARED));
|
|
#endif /* UNIV_SYNC_DEBUG */
|
|
|
|
DEBUG_SYNC(user_thd, "innodb_inplace_alter_table_enter");
|
|
|
|
if (!(ha_alter_info->handler_flags & INNOBASE_ALTER_DATA)) {
|
|
ok_exit:
|
|
DEBUG_SYNC(user_thd, "innodb_after_inplace_alter_table");
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
if ((ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE)
|
|
== Alter_inplace_info::CHANGE_CREATE_OPTION
|
|
&& !innobase_need_rebuild(ha_alter_info, table)) {
|
|
goto ok_exit;
|
|
}
|
|
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(ha_alter_info->handler_ctx);
|
|
|
|
DBUG_ASSERT(ctx);
|
|
DBUG_ASSERT(ctx->trx);
|
|
DBUG_ASSERT(ctx->prebuilt == prebuilt);
|
|
|
|
if (prebuilt->table->ibd_file_missing
|
|
|| dict_table_is_discarded(prebuilt->table)) {
|
|
goto all_done;
|
|
}
|
|
|
|
/* Read the clustered index of the table and build
|
|
indexes based on this information using temporary
|
|
files and merge sort. */
|
|
DBUG_EXECUTE_IF("innodb_OOM_inplace_alter",
|
|
error = DB_OUT_OF_MEMORY; goto oom;);
|
|
|
|
error = row_merge_build_indexes(
|
|
prebuilt->trx,
|
|
prebuilt->table, ctx->new_table,
|
|
ctx->online,
|
|
ctx->add_index, ctx->add_key_numbers, ctx->num_to_add_index,
|
|
altered_table, ctx->add_cols, ctx->col_map,
|
|
ctx->add_autoinc, ctx->sequence);
|
|
#ifndef DBUG_OFF
|
|
oom:
|
|
#endif /* !DBUG_OFF */
|
|
if (error == DB_SUCCESS && ctx->online && ctx->need_rebuild()) {
|
|
DEBUG_SYNC_C("row_log_table_apply1_before");
|
|
error = row_log_table_apply(
|
|
ctx->thr, prebuilt->table, altered_table);
|
|
}
|
|
|
|
/* Init online ddl status variables */
|
|
onlineddl_rowlog_rows = 0;
|
|
onlineddl_rowlog_pct_used = 0;
|
|
onlineddl_pct_progress = 0;
|
|
|
|
DEBUG_SYNC_C("inplace_after_index_build");
|
|
|
|
DBUG_EXECUTE_IF("create_index_fail",
|
|
error = DB_DUPLICATE_KEY;
|
|
prebuilt->trx->error_key_num = ULINT_UNDEFINED;);
|
|
|
|
/* After an error, remove all those index definitions
|
|
from the dictionary which were defined. */
|
|
|
|
switch (error) {
|
|
KEY* dup_key;
|
|
all_done:
|
|
case DB_SUCCESS:
|
|
ut_d(mutex_enter(&dict_sys->mutex));
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
prebuilt->table, CHECK_PARTIAL_OK));
|
|
ut_d(mutex_exit(&dict_sys->mutex));
|
|
/* prebuilt->table->n_ref_count can be anything here,
|
|
given that we hold at most a shared lock on the table. */
|
|
goto ok_exit;
|
|
case DB_DUPLICATE_KEY:
|
|
if (prebuilt->trx->error_key_num == ULINT_UNDEFINED
|
|
|| ha_alter_info->key_count == 0) {
|
|
/* This should be the hidden index on
|
|
FTS_DOC_ID, or there is no PRIMARY KEY in the
|
|
table. Either way, we should be seeing and
|
|
reporting a bogus duplicate key error. */
|
|
dup_key = NULL;
|
|
} else {
|
|
DBUG_ASSERT(prebuilt->trx->error_key_num
|
|
< ha_alter_info->key_count);
|
|
dup_key = &ha_alter_info->key_info_buffer[
|
|
prebuilt->trx->error_key_num];
|
|
}
|
|
print_keydup_error(altered_table, dup_key, MYF(0));
|
|
break;
|
|
case DB_ONLINE_LOG_TOO_BIG:
|
|
DBUG_ASSERT(ctx->online);
|
|
my_error(ER_INNODB_ONLINE_LOG_TOO_BIG, MYF(0),
|
|
get_error_key_name(prebuilt->trx->error_key_num,
|
|
ha_alter_info, prebuilt->table));
|
|
break;
|
|
case DB_INDEX_CORRUPT:
|
|
my_error(ER_INDEX_CORRUPT, MYF(0),
|
|
get_error_key_name(prebuilt->trx->error_key_num,
|
|
ha_alter_info, prebuilt->table));
|
|
break;
|
|
case DB_DECRYPTION_FAILED: {
|
|
String str;
|
|
const char* engine= table_type();
|
|
get_error_message(HA_ERR_DECRYPTION_FAILED, &str);
|
|
my_error(ER_GET_ERRMSG, MYF(0), HA_ERR_DECRYPTION_FAILED, str.c_ptr(), engine);
|
|
break;
|
|
}
|
|
default:
|
|
my_error_innodb(error,
|
|
table_share->table_name.str,
|
|
prebuilt->table->flags);
|
|
}
|
|
|
|
/* prebuilt->table->n_ref_count can be anything here, given
|
|
that we hold at most a shared lock on the table. */
|
|
prebuilt->trx->error_info = NULL;
|
|
ctx->trx->error_state = DB_SUCCESS;
|
|
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/** Free the modification log for online table rebuild.
|
|
@param table table that was being rebuilt online */
|
|
static
|
|
void
|
|
innobase_online_rebuild_log_free(
|
|
/*=============================*/
|
|
dict_table_t* table)
|
|
{
|
|
dict_index_t* clust_index = dict_table_get_first_index(table);
|
|
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
ut_ad(rw_lock_own(&dict_operation_lock, RW_LOCK_EX));
|
|
#endif /* UNIV_SYNC_DEBUG */
|
|
|
|
rw_lock_x_lock(&clust_index->lock);
|
|
|
|
if (clust_index->online_log) {
|
|
ut_ad(dict_index_get_online_status(clust_index)
|
|
== ONLINE_INDEX_CREATION);
|
|
clust_index->online_status = ONLINE_INDEX_COMPLETE;
|
|
row_log_free(clust_index->online_log);
|
|
DEBUG_SYNC_C("innodb_online_rebuild_log_free_aborted");
|
|
}
|
|
|
|
DBUG_ASSERT(dict_index_get_online_status(clust_index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
rw_lock_x_unlock(&clust_index->lock);
|
|
}
|
|
|
|
/** Rollback a secondary index creation, drop the indexes with
|
|
temparary index prefix
|
|
@param user_table InnoDB table
|
|
@param table the TABLE
|
|
@param locked TRUE=table locked, FALSE=may need to do a lazy drop
|
|
@param trx the transaction
|
|
*/
|
|
static MY_ATTRIBUTE((nonnull))
|
|
void
|
|
innobase_rollback_sec_index(
|
|
/*========================*/
|
|
dict_table_t* user_table,
|
|
const TABLE* table,
|
|
ibool locked,
|
|
trx_t* trx)
|
|
{
|
|
row_merge_drop_indexes(trx, user_table, locked);
|
|
|
|
/* Free the table->fts only if there is no FTS_DOC_ID
|
|
in the table */
|
|
if (user_table->fts
|
|
&& !DICT_TF2_FLAG_IS_SET(user_table,
|
|
DICT_TF2_FTS_HAS_DOC_ID)
|
|
&& !innobase_fulltext_exist(table)) {
|
|
fts_free(user_table);
|
|
}
|
|
}
|
|
|
|
/** Roll back the changes made during prepare_inplace_alter_table()
|
|
and inplace_alter_table() inside the storage engine. Note that the
|
|
allowed level of concurrency during this operation will be the same as
|
|
for inplace_alter_table() and thus might be higher than during
|
|
prepare_inplace_alter_table(). (E.g concurrent writes were blocked
|
|
during prepare, but might not be during commit).
|
|
|
|
@param ha_alter_info Data used during in-place alter.
|
|
@param table the TABLE
|
|
@param prebuilt the prebuilt struct
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
inline MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
rollback_inplace_alter_table(
|
|
/*=========================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
const TABLE* table,
|
|
row_prebuilt_t* prebuilt)
|
|
{
|
|
bool fail = false;
|
|
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(ha_alter_info->handler_ctx);
|
|
|
|
DBUG_ENTER("rollback_inplace_alter_table");
|
|
|
|
if (!ctx || !ctx->trx) {
|
|
/* If we have not started a transaction yet,
|
|
(almost) nothing has been or needs to be done. */
|
|
goto func_exit;
|
|
}
|
|
|
|
row_mysql_lock_data_dictionary(ctx->trx);
|
|
|
|
if (ctx->need_rebuild()) {
|
|
dberr_t err;
|
|
ulint flags = ctx->new_table->flags;
|
|
|
|
/* DML threads can access ctx->new_table via the
|
|
online rebuild log. Free it first. */
|
|
innobase_online_rebuild_log_free(prebuilt->table);
|
|
|
|
/* Since the FTS index specific auxiliary tables has
|
|
not yet registered with "table->fts" by fts_add_index(),
|
|
we will need explicitly delete them here */
|
|
if (DICT_TF2_FLAG_IS_SET(ctx->new_table, DICT_TF2_FTS)) {
|
|
|
|
err = innobase_drop_fts_index_table(
|
|
ctx->new_table, ctx->trx);
|
|
|
|
if (err != DB_SUCCESS) {
|
|
my_error_innodb(
|
|
err, table->s->table_name.str,
|
|
flags);
|
|
fail = true;
|
|
}
|
|
}
|
|
|
|
/* Drop the table. */
|
|
dict_table_close(ctx->new_table, TRUE, FALSE);
|
|
|
|
#if defined UNIV_DEBUG || defined UNIV_DDL_DEBUG
|
|
/* Nobody should have initialized the stats of the
|
|
newly created table yet. When this is the case, we
|
|
know that it has not been added for background stats
|
|
gathering. */
|
|
ut_a(!ctx->new_table->stat_initialized);
|
|
#endif /* UNIV_DEBUG || UNIV_DDL_DEBUG */
|
|
|
|
err = row_merge_drop_table(ctx->trx, ctx->new_table);
|
|
|
|
switch (err) {
|
|
case DB_SUCCESS:
|
|
break;
|
|
default:
|
|
my_error_innodb(err, table->s->table_name.str,
|
|
flags);
|
|
fail = true;
|
|
}
|
|
} else {
|
|
DBUG_ASSERT(!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ADD_PK_INDEX));
|
|
DBUG_ASSERT(ctx->new_table == prebuilt->table);
|
|
|
|
trx_start_for_ddl(ctx->trx, TRX_DICT_OP_INDEX);
|
|
|
|
innobase_rollback_sec_index(
|
|
prebuilt->table, table, FALSE, ctx->trx);
|
|
}
|
|
|
|
trx_commit_for_mysql(ctx->trx);
|
|
row_mysql_unlock_data_dictionary(ctx->trx);
|
|
trx_free_for_mysql(ctx->trx);
|
|
|
|
func_exit:
|
|
#ifndef DBUG_OFF
|
|
dict_index_t* clust_index = dict_table_get_first_index(
|
|
prebuilt->table);
|
|
DBUG_ASSERT(!clust_index->online_log);
|
|
DBUG_ASSERT(dict_index_get_online_status(clust_index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
#endif /* !DBUG_OFF */
|
|
|
|
if (ctx) {
|
|
DBUG_ASSERT(ctx->prebuilt == prebuilt);
|
|
|
|
if (ctx->num_to_add_fk) {
|
|
for (ulint i = 0; i < ctx->num_to_add_fk; i++) {
|
|
dict_foreign_free(ctx->add_fk[i]);
|
|
}
|
|
}
|
|
|
|
if (ctx->num_to_drop_index) {
|
|
row_mysql_lock_data_dictionary(prebuilt->trx);
|
|
|
|
/* Clear the to_be_dropped flags
|
|
in the data dictionary cache.
|
|
The flags may already have been cleared,
|
|
in case an error was detected in
|
|
commit_inplace_alter_table(). */
|
|
for (ulint i = 0; i < ctx->num_to_drop_index; i++) {
|
|
dict_index_t* index = ctx->drop_index[i];
|
|
DBUG_ASSERT(*index->name != TEMP_INDEX_PREFIX);
|
|
|
|
index->to_be_dropped = 0;
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(prebuilt->trx);
|
|
}
|
|
}
|
|
|
|
trx_commit_for_mysql(prebuilt->trx);
|
|
MONITOR_ATOMIC_DEC(MONITOR_PENDING_ALTER_TABLE);
|
|
DBUG_RETURN(fail);
|
|
}
|
|
|
|
/** Drop a FOREIGN KEY constraint from the data dictionary tables.
|
|
@param trx data dictionary transaction
|
|
@param table_name Table name in MySQL
|
|
@param foreign_id Foreign key constraint identifier
|
|
@retval true Failure
|
|
@retval false Success */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_drop_foreign_try(
|
|
/*======================*/
|
|
trx_t* trx,
|
|
const char* table_name,
|
|
const char* foreign_id)
|
|
{
|
|
DBUG_ENTER("innobase_drop_foreign_try");
|
|
|
|
DBUG_ASSERT(trx_get_dict_operation(trx) == TRX_DICT_OP_INDEX);
|
|
ut_ad(trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
ut_ad(rw_lock_own(&dict_operation_lock, RW_LOCK_EX));
|
|
#endif /* UNIV_SYNC_DEBUG */
|
|
|
|
/* Drop the constraint from the data dictionary. */
|
|
static const char sql[] =
|
|
"PROCEDURE DROP_FOREIGN_PROC () IS\n"
|
|
"BEGIN\n"
|
|
"DELETE FROM SYS_FOREIGN WHERE ID=:id;\n"
|
|
"DELETE FROM SYS_FOREIGN_COLS WHERE ID=:id;\n"
|
|
"END;\n";
|
|
|
|
dberr_t error;
|
|
pars_info_t* info;
|
|
|
|
info = pars_info_create();
|
|
pars_info_add_str_literal(info, "id", foreign_id);
|
|
|
|
trx->op_info = "dropping foreign key constraint from dictionary";
|
|
error = que_eval_sql(info, sql, FALSE, trx);
|
|
trx->op_info = "";
|
|
|
|
DBUG_EXECUTE_IF("ib_drop_foreign_error",
|
|
error = DB_OUT_OF_FILE_SPACE;);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
my_error_innodb(error, table_name, 0);
|
|
trx->error_state = DB_SUCCESS;
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
/** Rename a column in the data dictionary tables.
|
|
@param user_table InnoDB table that was being altered
|
|
@param trx data dictionary transaction
|
|
@param table_name Table name in MySQL
|
|
@param nth_col 0-based index of the column
|
|
@param from old column name
|
|
@param to new column name
|
|
@param new_clustered whether the table has been rebuilt
|
|
@retval true Failure
|
|
@retval false Success */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_rename_column_try(
|
|
/*=======================*/
|
|
const dict_table_t* user_table,
|
|
trx_t* trx,
|
|
const char* table_name,
|
|
ulint nth_col,
|
|
const char* from,
|
|
const char* to,
|
|
bool new_clustered)
|
|
{
|
|
pars_info_t* info;
|
|
dberr_t error;
|
|
|
|
DBUG_ENTER("innobase_rename_column_try");
|
|
|
|
DBUG_ASSERT(trx_get_dict_operation(trx) == TRX_DICT_OP_INDEX);
|
|
ut_ad(trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
ut_ad(mutex_own(&dict_sys->mutex));
|
|
#ifdef UNIV_SYNC_DEBUG
|
|
ut_ad(rw_lock_own(&dict_operation_lock, RW_LOCK_EX));
|
|
#endif /* UNIV_SYNC_DEBUG */
|
|
|
|
if (new_clustered) {
|
|
goto rename_foreign;
|
|
}
|
|
|
|
info = pars_info_create();
|
|
|
|
pars_info_add_ull_literal(info, "tableid", user_table->id);
|
|
pars_info_add_int4_literal(info, "nth", nth_col);
|
|
pars_info_add_str_literal(info, "old", from);
|
|
pars_info_add_str_literal(info, "new", to);
|
|
|
|
trx->op_info = "renaming column in SYS_COLUMNS";
|
|
|
|
error = que_eval_sql(
|
|
info,
|
|
"PROCEDURE RENAME_SYS_COLUMNS_PROC () IS\n"
|
|
"BEGIN\n"
|
|
"UPDATE SYS_COLUMNS SET NAME=:new\n"
|
|
"WHERE TABLE_ID=:tableid AND NAME=:old\n"
|
|
"AND POS=:nth;\n"
|
|
"END;\n",
|
|
FALSE, trx);
|
|
|
|
DBUG_EXECUTE_IF("ib_rename_column_error",
|
|
error = DB_OUT_OF_FILE_SPACE;);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
err_exit:
|
|
my_error_innodb(error, table_name, 0);
|
|
trx->error_state = DB_SUCCESS;
|
|
trx->op_info = "";
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
trx->op_info = "renaming column in SYS_FIELDS";
|
|
|
|
for (const dict_index_t* index = dict_table_get_first_index(
|
|
user_table);
|
|
index != NULL;
|
|
index = dict_table_get_next_index(index)) {
|
|
|
|
for (ulint i = 0; i < dict_index_get_n_fields(index); i++) {
|
|
if (strcmp(dict_index_get_nth_field(index, i)->name,
|
|
from)) {
|
|
continue;
|
|
}
|
|
|
|
info = pars_info_create();
|
|
|
|
pars_info_add_ull_literal(info, "indexid", index->id);
|
|
pars_info_add_int4_literal(info, "nth", i);
|
|
pars_info_add_str_literal(info, "old", from);
|
|
pars_info_add_str_literal(info, "new", to);
|
|
|
|
error = que_eval_sql(
|
|
info,
|
|
"PROCEDURE RENAME_SYS_FIELDS_PROC () IS\n"
|
|
"BEGIN\n"
|
|
|
|
"UPDATE SYS_FIELDS SET COL_NAME=:new\n"
|
|
"WHERE INDEX_ID=:indexid AND COL_NAME=:old\n"
|
|
"AND POS=:nth;\n"
|
|
|
|
/* Try again, in case there is a prefix_len
|
|
encoded in SYS_FIELDS.POS */
|
|
|
|
"UPDATE SYS_FIELDS SET COL_NAME=:new\n"
|
|
"WHERE INDEX_ID=:indexid AND COL_NAME=:old\n"
|
|
"AND POS>=65536*:nth AND POS<65536*(:nth+1);\n"
|
|
|
|
"END;\n",
|
|
FALSE, trx);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
goto err_exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
rename_foreign:
|
|
trx->op_info = "renaming column in SYS_FOREIGN_COLS";
|
|
|
|
std::list<dict_foreign_t*> fk_evict;
|
|
bool foreign_modified;
|
|
|
|
for (dict_foreign_set::const_iterator it = user_table->foreign_set.begin();
|
|
it != user_table->foreign_set.end();
|
|
++it) {
|
|
|
|
dict_foreign_t* foreign = *it;
|
|
foreign_modified = false;
|
|
|
|
for (unsigned i = 0; i < foreign->n_fields; i++) {
|
|
if (strcmp(foreign->foreign_col_names[i], from)) {
|
|
continue;
|
|
}
|
|
|
|
info = pars_info_create();
|
|
|
|
pars_info_add_str_literal(info, "id", foreign->id);
|
|
pars_info_add_int4_literal(info, "nth", i);
|
|
pars_info_add_str_literal(info, "old", from);
|
|
pars_info_add_str_literal(info, "new", to);
|
|
|
|
error = que_eval_sql(
|
|
info,
|
|
"PROCEDURE RENAME_SYS_FOREIGN_F_PROC () IS\n"
|
|
"BEGIN\n"
|
|
"UPDATE SYS_FOREIGN_COLS\n"
|
|
"SET FOR_COL_NAME=:new\n"
|
|
"WHERE ID=:id AND POS=:nth\n"
|
|
"AND FOR_COL_NAME=:old;\n"
|
|
"END;\n",
|
|
FALSE, trx);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
goto err_exit;
|
|
}
|
|
foreign_modified = true;
|
|
}
|
|
|
|
if (foreign_modified) {
|
|
fk_evict.push_back(foreign);
|
|
}
|
|
}
|
|
|
|
for (dict_foreign_set::const_iterator it
|
|
= user_table->referenced_set.begin();
|
|
it != user_table->referenced_set.end();
|
|
++it) {
|
|
|
|
foreign_modified = false;
|
|
dict_foreign_t* foreign = *it;
|
|
|
|
for (unsigned i = 0; i < foreign->n_fields; i++) {
|
|
if (strcmp(foreign->referenced_col_names[i], from)) {
|
|
continue;
|
|
}
|
|
|
|
info = pars_info_create();
|
|
|
|
pars_info_add_str_literal(info, "id", foreign->id);
|
|
pars_info_add_int4_literal(info, "nth", i);
|
|
pars_info_add_str_literal(info, "old", from);
|
|
pars_info_add_str_literal(info, "new", to);
|
|
|
|
error = que_eval_sql(
|
|
info,
|
|
"PROCEDURE RENAME_SYS_FOREIGN_R_PROC () IS\n"
|
|
"BEGIN\n"
|
|
"UPDATE SYS_FOREIGN_COLS\n"
|
|
"SET REF_COL_NAME=:new\n"
|
|
"WHERE ID=:id AND POS=:nth\n"
|
|
"AND REF_COL_NAME=:old;\n"
|
|
"END;\n",
|
|
FALSE, trx);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
goto err_exit;
|
|
}
|
|
foreign_modified = true;
|
|
}
|
|
|
|
if (foreign_modified) {
|
|
fk_evict.push_back(foreign);
|
|
}
|
|
}
|
|
|
|
if (new_clustered) {
|
|
std::for_each(fk_evict.begin(), fk_evict.end(),
|
|
dict_foreign_remove_from_cache);
|
|
}
|
|
|
|
trx->op_info = "";
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
/** Rename columns in the data dictionary tables.
|
|
@param ha_alter_info Data used during in-place alter.
|
|
@param ctx In-place ALTER TABLE context
|
|
@param table the TABLE
|
|
@param trx data dictionary transaction
|
|
@param table_name Table name in MySQL
|
|
@retval true Failure
|
|
@retval false Success */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_rename_columns_try(
|
|
/*========================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
ha_innobase_inplace_ctx*ctx,
|
|
const TABLE* table,
|
|
trx_t* trx,
|
|
const char* table_name)
|
|
{
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
uint i = 0;
|
|
|
|
DBUG_ASSERT(ctx);
|
|
DBUG_ASSERT(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME);
|
|
|
|
for (Field** fp = table->field; *fp; fp++, i++) {
|
|
if (!((*fp)->flags & FIELD_IS_RENAMED) || !((*fp)->stored_in_db())) {
|
|
continue;
|
|
}
|
|
|
|
cf_it.rewind();
|
|
while (Create_field* cf = cf_it++) {
|
|
if (cf->field == *fp) {
|
|
if (innobase_rename_column_try(
|
|
ctx->old_table, trx, table_name, i,
|
|
cf->field->field_name,
|
|
cf->field_name,
|
|
ctx->need_rebuild())) {
|
|
return(true);
|
|
}
|
|
goto processed_field;
|
|
}
|
|
}
|
|
|
|
ut_error;
|
|
processed_field:
|
|
continue;
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/** Rename columns in the data dictionary cache
|
|
as part of commit_cache_norebuild().
|
|
@param ha_alter_info Data used during in-place alter.
|
|
@param table the TABLE
|
|
@param user_table InnoDB table that was being altered */
|
|
static MY_ATTRIBUTE((nonnull))
|
|
void
|
|
innobase_rename_columns_cache(
|
|
/*==========================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
const TABLE* table,
|
|
dict_table_t* user_table)
|
|
{
|
|
if (!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME)) {
|
|
return;
|
|
}
|
|
|
|
List_iterator_fast<Create_field> cf_it(
|
|
ha_alter_info->alter_info->create_list);
|
|
uint i = 0;
|
|
|
|
for (Field** fp = table->field; *fp; fp++, i++) {
|
|
if (!((*fp)->flags & FIELD_IS_RENAMED)) {
|
|
continue;
|
|
}
|
|
|
|
cf_it.rewind();
|
|
while (Create_field* cf = cf_it++) {
|
|
if (cf->field == *fp) {
|
|
dict_mem_table_col_rename(user_table, i,
|
|
cf->field->field_name,
|
|
cf->field_name);
|
|
goto processed_field;
|
|
}
|
|
}
|
|
|
|
ut_error;
|
|
processed_field:
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/** Get the auto-increment value of the table on commit.
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param ctx In-place ALTER TABLE context
|
|
@param altered_table MySQL table that is being altered
|
|
@param old_table MySQL table as it is before the ALTER operation
|
|
@return the next auto-increment value (0 if not present) */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
ulonglong
|
|
commit_get_autoinc(
|
|
/*===============*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
ha_innobase_inplace_ctx*ctx,
|
|
const TABLE* altered_table,
|
|
const TABLE* old_table)
|
|
{
|
|
ulonglong max_autoinc;
|
|
|
|
DBUG_ENTER("commit_get_autoinc");
|
|
|
|
if (!altered_table->found_next_number_field) {
|
|
/* There is no AUTO_INCREMENT column in the table
|
|
after the ALTER operation. */
|
|
max_autoinc = 0;
|
|
} else if (ctx->add_autoinc != ULINT_UNDEFINED) {
|
|
/* An AUTO_INCREMENT column was added. Get the last
|
|
value from the sequence, which may be based on a
|
|
supplied AUTO_INCREMENT value. */
|
|
max_autoinc = ctx->sequence.last();
|
|
} else if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::CHANGE_CREATE_OPTION)
|
|
&& (ha_alter_info->create_info->used_fields
|
|
& HA_CREATE_USED_AUTO)) {
|
|
/* An AUTO_INCREMENT value was supplied, but the table was not
|
|
rebuilt. Get the user-supplied value or the last value from the
|
|
sequence. */
|
|
ib_uint64_t max_value_table;
|
|
dberr_t err;
|
|
|
|
Field* autoinc_field =
|
|
old_table->found_next_number_field;
|
|
KEY* autoinc_key =
|
|
old_table->key_info + old_table->s->next_number_index;
|
|
|
|
dict_index_t* index = dict_table_get_index_on_name(
|
|
ctx->old_table, autoinc_key->name);
|
|
|
|
max_autoinc = ha_alter_info->create_info->auto_increment_value;
|
|
|
|
dict_table_autoinc_lock(ctx->old_table);
|
|
|
|
err = row_search_max_autoinc(
|
|
index, autoinc_field->field_name, &max_value_table);
|
|
|
|
if (err != DB_SUCCESS) {
|
|
ut_ad(0);
|
|
max_autoinc = 0;
|
|
} else if (max_autoinc <= max_value_table) {
|
|
ulonglong col_max_value;
|
|
ulonglong offset;
|
|
|
|
col_max_value = innobase_get_int_col_max_value(
|
|
old_table->found_next_number_field);
|
|
|
|
offset = ctx->prebuilt->autoinc_offset;
|
|
max_autoinc = innobase_next_autoinc(
|
|
max_value_table, 1, 1, offset,
|
|
col_max_value);
|
|
}
|
|
dict_table_autoinc_unlock(ctx->old_table);
|
|
} else {
|
|
/* An AUTO_INCREMENT value was not specified.
|
|
Read the old counter value from the table. */
|
|
ut_ad(old_table->found_next_number_field);
|
|
dict_table_autoinc_lock(ctx->old_table);
|
|
max_autoinc = ctx->old_table->autoinc;
|
|
dict_table_autoinc_unlock(ctx->old_table);
|
|
}
|
|
|
|
DBUG_RETURN(max_autoinc);
|
|
}
|
|
|
|
/** Add or drop foreign key constraints to the data dictionary tables,
|
|
but do not touch the data dictionary cache.
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param ctx In-place ALTER TABLE context
|
|
@param trx Data dictionary transaction
|
|
@param table_name Table name in MySQL
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
innobase_update_foreign_try(
|
|
/*========================*/
|
|
ha_innobase_inplace_ctx*ctx,
|
|
trx_t* trx,
|
|
const char* table_name)
|
|
{
|
|
ulint foreign_id;
|
|
ulint i;
|
|
|
|
DBUG_ENTER("innobase_update_foreign_try");
|
|
DBUG_ASSERT(ctx);
|
|
|
|
foreign_id = dict_table_get_highest_foreign_id(ctx->new_table);
|
|
|
|
foreign_id++;
|
|
|
|
for (i = 0; i < ctx->num_to_add_fk; i++) {
|
|
dict_foreign_t* fk = ctx->add_fk[i];
|
|
|
|
ut_ad(fk->foreign_table == ctx->new_table
|
|
|| fk->foreign_table == ctx->old_table);
|
|
|
|
dberr_t error = dict_create_add_foreign_id(
|
|
&foreign_id, ctx->old_table->name, fk);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
my_error(ER_TOO_LONG_IDENT, MYF(0),
|
|
fk->id);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
if (!fk->foreign_index) {
|
|
fk->foreign_index = dict_foreign_find_index(
|
|
ctx->new_table, ctx->col_names,
|
|
fk->foreign_col_names,
|
|
fk->n_fields, fk->referenced_index, TRUE,
|
|
fk->type
|
|
& (DICT_FOREIGN_ON_DELETE_SET_NULL
|
|
| DICT_FOREIGN_ON_UPDATE_SET_NULL),
|
|
NULL, NULL, NULL);
|
|
if (!fk->foreign_index) {
|
|
my_error(ER_FK_INCORRECT_OPTION,
|
|
MYF(0), table_name, fk->id);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
/* The fk->foreign_col_names[] uses renamed column
|
|
names, while the columns in ctx->old_table have not
|
|
been renamed yet. */
|
|
error = dict_create_add_foreign_to_dictionary(
|
|
(dict_table_t*)ctx->old_table, ctx->old_table->name, fk, trx);
|
|
|
|
DBUG_EXECUTE_IF(
|
|
"innodb_test_cannot_add_fk_system",
|
|
error = DB_ERROR;);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
my_error(ER_FK_FAIL_ADD_SYSTEM, MYF(0),
|
|
fk->id);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < ctx->num_to_drop_fk; i++) {
|
|
dict_foreign_t* fk = ctx->drop_fk[i];
|
|
|
|
DBUG_ASSERT(fk->foreign_table == ctx->old_table);
|
|
|
|
if (innobase_drop_foreign_try(trx, table_name, fk->id)) {
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
/** Update the foreign key constraint definitions in the data dictionary cache
|
|
after the changes to data dictionary tables were committed.
|
|
@param ctx In-place ALTER TABLE context
|
|
@param user_thd MySQL connection
|
|
@return InnoDB error code (should always be DB_SUCCESS) */
|
|
static MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
dberr_t
|
|
innobase_update_foreign_cache(
|
|
/*==========================*/
|
|
ha_innobase_inplace_ctx* ctx,
|
|
THD* user_thd)
|
|
{
|
|
dict_table_t* user_table;
|
|
dberr_t err = DB_SUCCESS;
|
|
|
|
DBUG_ENTER("innobase_update_foreign_cache");
|
|
|
|
user_table = ctx->old_table;
|
|
|
|
/* Discard the added foreign keys, because we will
|
|
load them from the data dictionary. */
|
|
for (ulint i = 0; i < ctx->num_to_add_fk; i++) {
|
|
dict_foreign_t* fk = ctx->add_fk[i];
|
|
dict_foreign_free(fk);
|
|
}
|
|
|
|
if (ctx->need_rebuild()) {
|
|
/* The rebuilt table is already using the renamed
|
|
column names. No need to pass col_names or to drop
|
|
constraints from the data dictionary cache. */
|
|
DBUG_ASSERT(!ctx->col_names);
|
|
DBUG_ASSERT(user_table->foreign_set.empty());
|
|
DBUG_ASSERT(user_table->referenced_set.empty());
|
|
user_table = ctx->new_table;
|
|
} else {
|
|
/* Drop the foreign key constraints if the
|
|
table was not rebuilt. If the table is rebuilt,
|
|
there would not be any foreign key contraints for
|
|
it yet in the data dictionary cache. */
|
|
for (ulint i = 0; i < ctx->num_to_drop_fk; i++) {
|
|
dict_foreign_t* fk = ctx->drop_fk[i];
|
|
dict_foreign_remove_from_cache(fk);
|
|
}
|
|
}
|
|
|
|
/* Load the old or added foreign keys from the data dictionary
|
|
and prevent the table from being evicted from the data
|
|
dictionary cache (work around the lack of WL#6049). */
|
|
err = dict_load_foreigns(user_table->name,
|
|
ctx->col_names, false, true,
|
|
DICT_ERR_IGNORE_NONE);
|
|
|
|
if (err == DB_CANNOT_ADD_CONSTRAINT) {
|
|
/* It is possible there are existing foreign key are
|
|
loaded with "foreign_key checks" off,
|
|
so let's retry the loading with charset_check is off */
|
|
err = dict_load_foreigns(user_table->name,
|
|
ctx->col_names, false, false,
|
|
DICT_ERR_IGNORE_NONE);
|
|
|
|
/* The load with "charset_check" off is successful, warn
|
|
the user that the foreign key has loaded with mis-matched
|
|
charset */
|
|
if (err == DB_SUCCESS) {
|
|
push_warning_printf(
|
|
user_thd,
|
|
Sql_condition::WARN_LEVEL_WARN,
|
|
ER_ALTER_INFO,
|
|
"Foreign key constraints for table '%s'"
|
|
" are loaded with charset check off",
|
|
user_table->name);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(err);
|
|
}
|
|
|
|
/** Commit the changes made during prepare_inplace_alter_table()
|
|
and inplace_alter_table() inside the data dictionary tables,
|
|
when rebuilding the table.
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param ctx In-place ALTER TABLE context
|
|
@param altered_table MySQL table that is being altered
|
|
@param old_table MySQL table as it is before the ALTER operation
|
|
@param trx Data dictionary transaction
|
|
@param table_name Table name in MySQL
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
inline MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
commit_try_rebuild(
|
|
/*===============*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
ha_innobase_inplace_ctx*ctx,
|
|
TABLE* altered_table,
|
|
const TABLE* old_table,
|
|
trx_t* trx,
|
|
const char* table_name)
|
|
{
|
|
dict_table_t* rebuilt_table = ctx->new_table;
|
|
dict_table_t* user_table = ctx->old_table;
|
|
|
|
DBUG_ENTER("commit_try_rebuild");
|
|
DBUG_ASSERT(ctx->need_rebuild());
|
|
DBUG_ASSERT(trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
DBUG_ASSERT(!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::DROP_FOREIGN_KEY)
|
|
|| ctx->num_to_drop_fk > 0);
|
|
DBUG_ASSERT(ctx->num_to_drop_fk
|
|
== ha_alter_info->alter_info->drop_list.elements);
|
|
|
|
for (dict_index_t* index = dict_table_get_first_index(rebuilt_table);
|
|
index;
|
|
index = dict_table_get_next_index(index)) {
|
|
DBUG_ASSERT(dict_index_get_online_status(index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
DBUG_ASSERT(*index->name != TEMP_INDEX_PREFIX);
|
|
if (dict_index_is_corrupted(index)) {
|
|
my_error(ER_INDEX_CORRUPT, MYF(0),
|
|
index->name);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
if (innobase_update_foreign_try(ctx, trx, table_name)) {
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
dberr_t error;
|
|
|
|
/* Clear the to_be_dropped flag in the data dictionary cache
|
|
of user_table. */
|
|
for (ulint i = 0; i < ctx->num_to_drop_index; i++) {
|
|
dict_index_t* index = ctx->drop_index[i];
|
|
DBUG_ASSERT(index->table == user_table);
|
|
DBUG_ASSERT(*index->name != TEMP_INDEX_PREFIX);
|
|
DBUG_ASSERT(index->to_be_dropped);
|
|
index->to_be_dropped = 0;
|
|
}
|
|
|
|
/* We copied the table. Any indexes that were requested to be
|
|
dropped were not created in the copy of the table. Apply any
|
|
last bit of the rebuild log and then rename the tables. */
|
|
|
|
if (ctx->online) {
|
|
DEBUG_SYNC_C("row_log_table_apply2_before");
|
|
error = row_log_table_apply(
|
|
ctx->thr, user_table, altered_table);
|
|
ulint err_key = thr_get_trx(ctx->thr)->error_key_num;
|
|
|
|
switch (error) {
|
|
KEY* dup_key;
|
|
case DB_SUCCESS:
|
|
break;
|
|
case DB_DUPLICATE_KEY:
|
|
if (err_key == ULINT_UNDEFINED) {
|
|
/* This should be the hidden index on
|
|
FTS_DOC_ID. */
|
|
dup_key = NULL;
|
|
} else {
|
|
DBUG_ASSERT(err_key <
|
|
ha_alter_info->key_count);
|
|
dup_key = &ha_alter_info
|
|
->key_info_buffer[err_key];
|
|
}
|
|
print_keydup_error(altered_table, dup_key, MYF(0));
|
|
DBUG_RETURN(true);
|
|
case DB_ONLINE_LOG_TOO_BIG:
|
|
my_error(ER_INNODB_ONLINE_LOG_TOO_BIG, MYF(0),
|
|
get_error_key_name(err_key, ha_alter_info,
|
|
rebuilt_table));
|
|
DBUG_RETURN(true);
|
|
case DB_INDEX_CORRUPT:
|
|
my_error(ER_INDEX_CORRUPT, MYF(0),
|
|
get_error_key_name(err_key, ha_alter_info,
|
|
rebuilt_table));
|
|
DBUG_RETURN(true);
|
|
default:
|
|
my_error_innodb(error, table_name, user_table->flags);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
if ((ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME)
|
|
&& innobase_rename_columns_try(ha_alter_info, ctx, old_table,
|
|
trx, table_name)) {
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
DBUG_EXECUTE_IF("ib_ddl_crash_before_rename", DBUG_SUICIDE(););
|
|
|
|
/* The new table must inherit the flag from the
|
|
"parent" table. */
|
|
if (dict_table_is_discarded(user_table)) {
|
|
rebuilt_table->ibd_file_missing = true;
|
|
rebuilt_table->flags2 |= DICT_TF2_DISCARDED;
|
|
}
|
|
|
|
/* We can now rename the old table as a temporary table,
|
|
rename the new temporary table as the old table and drop the
|
|
old table. First, we only do this in the data dictionary
|
|
tables. The actual renaming will be performed in
|
|
commit_cache_rebuild(), once the data dictionary transaction
|
|
has been successfully committed. */
|
|
|
|
error = row_merge_rename_tables_dict(
|
|
user_table, rebuilt_table, ctx->tmp_name, trx);
|
|
|
|
/* We must be still holding a table handle. */
|
|
DBUG_ASSERT(user_table->n_ref_count >= 1);
|
|
|
|
DBUG_EXECUTE_IF("ib_ddl_crash_after_rename", DBUG_SUICIDE(););
|
|
DBUG_EXECUTE_IF("ib_rebuild_cannot_rename", error = DB_ERROR;);
|
|
|
|
if (user_table->n_ref_count > 1) {
|
|
/* This should only occur when an innodb_memcached
|
|
connection with innodb_api_enable_mdl=off was started
|
|
before commit_inplace_alter_table() locked the data
|
|
dictionary. We must roll back the ALTER TABLE, because
|
|
we cannot drop a table while it is being used. */
|
|
|
|
/* Normally, n_ref_count must be 1, because purge
|
|
cannot be executing on this very table as we are
|
|
holding dict_operation_lock X-latch. */
|
|
|
|
error = DB_LOCK_WAIT_TIMEOUT;
|
|
}
|
|
|
|
switch (error) {
|
|
case DB_SUCCESS:
|
|
DBUG_RETURN(false);
|
|
case DB_TABLESPACE_EXISTS:
|
|
ut_a(rebuilt_table->n_ref_count == 1);
|
|
my_error(ER_TABLESPACE_EXISTS, MYF(0), ctx->tmp_name);
|
|
DBUG_RETURN(true);
|
|
case DB_DUPLICATE_KEY:
|
|
ut_a(rebuilt_table->n_ref_count == 1);
|
|
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), ctx->tmp_name);
|
|
DBUG_RETURN(true);
|
|
default:
|
|
my_error_innodb(error, table_name, user_table->flags);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
/** Apply the changes made during commit_try_rebuild(),
|
|
to the data dictionary cache and the file system.
|
|
@param ctx In-place ALTER TABLE context */
|
|
inline MY_ATTRIBUTE((nonnull))
|
|
void
|
|
commit_cache_rebuild(
|
|
/*=================*/
|
|
ha_innobase_inplace_ctx* ctx)
|
|
{
|
|
dberr_t error;
|
|
|
|
DBUG_ENTER("commit_cache_rebuild");
|
|
DBUG_ASSERT(ctx->need_rebuild());
|
|
DBUG_ASSERT(dict_table_is_discarded(ctx->old_table)
|
|
== dict_table_is_discarded(ctx->new_table));
|
|
|
|
const char* old_name = mem_heap_strdup(
|
|
ctx->heap, ctx->old_table->name);
|
|
|
|
/* We already committed and redo logged the renames,
|
|
so this must succeed. */
|
|
error = dict_table_rename_in_cache(
|
|
ctx->old_table, ctx->tmp_name, FALSE);
|
|
ut_a(error == DB_SUCCESS);
|
|
|
|
DEBUG_SYNC_C("commit_cache_rebuild_middle");
|
|
|
|
error = dict_table_rename_in_cache(
|
|
ctx->new_table, old_name, FALSE);
|
|
ut_a(error == DB_SUCCESS);
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/** Store the column number of the columns in a list belonging
|
|
to indexes which are not being dropped.
|
|
@param[in] ctx In-place ALTER TABLE context
|
|
@param[out] drop_col_list list which will be set, containing columns
|
|
which is part of index being dropped */
|
|
static
|
|
void
|
|
get_col_list_to_be_dropped(
|
|
ha_innobase_inplace_ctx* ctx,
|
|
std::set<ulint>& drop_col_list)
|
|
{
|
|
for (ulint index_count = 0; index_count < ctx->num_to_drop_index;
|
|
index_count++) {
|
|
dict_index_t* index = ctx->drop_index[index_count];
|
|
|
|
for (ulint col = 0; col < index->n_user_defined_cols; col++) {
|
|
ulint col_no = dict_index_get_nth_col_no(index, col);
|
|
drop_col_list.insert(col_no);
|
|
}
|
|
}
|
|
}
|
|
|
|
/** For each column, which is part of an index which is not going to be
|
|
dropped, it checks if the column number of the column is same as col_no
|
|
argument passed.
|
|
@param[in] table table object
|
|
@param[in] col_no column number of the column which is to be checked
|
|
@retval true column exists
|
|
@retval false column does not exist. */
|
|
static
|
|
bool
|
|
check_col_exists_in_indexes(
|
|
const dict_table_t* table,
|
|
ulint col_no)
|
|
{
|
|
for (dict_index_t* index = dict_table_get_first_index(table); index;
|
|
index = dict_table_get_next_index(index)) {
|
|
|
|
if (index->to_be_dropped) {
|
|
continue;
|
|
}
|
|
|
|
for (ulint col = 0; col < index->n_user_defined_cols; col++) {
|
|
|
|
ulint index_col_no = dict_index_get_nth_col_no(
|
|
index, col);
|
|
if (col_no == index_col_no) {
|
|
return(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
return(false);
|
|
}
|
|
|
|
/** Commit the changes made during prepare_inplace_alter_table()
|
|
and inplace_alter_table() inside the data dictionary tables,
|
|
when not rebuilding the table.
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param ctx In-place ALTER TABLE context
|
|
@param old_table MySQL table as it is before the ALTER operation
|
|
@param trx Data dictionary transaction
|
|
@param table_name Table name in MySQL
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
inline MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
commit_try_norebuild(
|
|
/*=================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
ha_innobase_inplace_ctx*ctx,
|
|
const TABLE* old_table,
|
|
trx_t* trx,
|
|
const char* table_name)
|
|
{
|
|
DBUG_ENTER("commit_try_norebuild");
|
|
DBUG_ASSERT(!ctx->need_rebuild());
|
|
DBUG_ASSERT(trx->dict_operation_lock_mode == RW_X_LATCH);
|
|
DBUG_ASSERT(!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::DROP_FOREIGN_KEY)
|
|
|| ctx->num_to_drop_fk > 0);
|
|
DBUG_ASSERT(ctx->num_to_drop_fk
|
|
== ha_alter_info->alter_info->drop_list.elements);
|
|
|
|
for (ulint i = 0; i < ctx->num_to_add_index; i++) {
|
|
dict_index_t* index = ctx->add_index[i];
|
|
DBUG_ASSERT(dict_index_get_online_status(index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
DBUG_ASSERT(*index->name == TEMP_INDEX_PREFIX);
|
|
if (dict_index_is_corrupted(index)) {
|
|
/* Report a duplicate key
|
|
error for the index that was
|
|
flagged corrupted, most likely
|
|
because a duplicate value was
|
|
inserted (directly or by
|
|
rollback) after
|
|
ha_innobase::inplace_alter_table()
|
|
completed.
|
|
TODO: report this as a corruption
|
|
with a detailed reason once
|
|
WL#6379 has been implemented. */
|
|
my_error(ER_DUP_UNKNOWN_IN_INDEX,
|
|
MYF(0), index->name + 1);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
if (innobase_update_foreign_try(ctx, trx, table_name)) {
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
dberr_t error;
|
|
|
|
/* We altered the table in place. */
|
|
/* Lose the TEMP_INDEX_PREFIX. */
|
|
for (ulint i = 0; i < ctx->num_to_add_index; i++) {
|
|
dict_index_t* index = ctx->add_index[i];
|
|
DBUG_ASSERT(dict_index_get_online_status(index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
DBUG_ASSERT(*index->name
|
|
== TEMP_INDEX_PREFIX);
|
|
error = row_merge_rename_index_to_add(
|
|
trx, ctx->new_table->id, index->id);
|
|
if (error != DB_SUCCESS) {
|
|
sql_print_error(
|
|
"InnoDB: rename index to add: %lu\n",
|
|
(ulong) error);
|
|
DBUG_ASSERT(0);
|
|
my_error(ER_INTERNAL_ERROR, MYF(0),
|
|
"rename index to add");
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
/* Drop any indexes that were requested to be dropped.
|
|
Rename them to TEMP_INDEX_PREFIX in the data
|
|
dictionary first. We do not bother to rename
|
|
index->name in the dictionary cache, because the index
|
|
is about to be freed after row_merge_drop_indexes_dict(). */
|
|
|
|
for (ulint i = 0; i < ctx->num_to_drop_index; i++) {
|
|
dict_index_t* index = ctx->drop_index[i];
|
|
DBUG_ASSERT(*index->name != TEMP_INDEX_PREFIX);
|
|
DBUG_ASSERT(index->table == ctx->new_table);
|
|
DBUG_ASSERT(index->to_be_dropped);
|
|
|
|
error = row_merge_rename_index_to_drop(
|
|
trx, index->table->id, index->id);
|
|
if (error != DB_SUCCESS) {
|
|
sql_print_error(
|
|
"InnoDB: rename index to drop: %lu\n",
|
|
(ulong) error);
|
|
DBUG_ASSERT(0);
|
|
my_error(ER_INTERNAL_ERROR, MYF(0),
|
|
"rename index to drop");
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
if (!(ha_alter_info->handler_flags
|
|
& Alter_inplace_info::ALTER_COLUMN_NAME)) {
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
DBUG_RETURN(innobase_rename_columns_try(ha_alter_info, ctx,
|
|
old_table, trx, table_name));
|
|
}
|
|
|
|
/** Commit the changes to the data dictionary cache
|
|
after a successful commit_try_norebuild() call.
|
|
@param ctx In-place ALTER TABLE context
|
|
@param table the TABLE before the ALTER
|
|
@param trx Data dictionary transaction object
|
|
(will be started and committed)
|
|
@return whether all replacements were found for dropped indexes */
|
|
inline MY_ATTRIBUTE((nonnull, warn_unused_result))
|
|
bool
|
|
commit_cache_norebuild(
|
|
/*===================*/
|
|
ha_innobase_inplace_ctx*ctx,
|
|
const TABLE* table,
|
|
trx_t* trx)
|
|
{
|
|
DBUG_ENTER("commit_cache_norebuild");
|
|
|
|
bool found = true;
|
|
|
|
DBUG_ASSERT(!ctx->need_rebuild());
|
|
|
|
std::set<ulint> drop_list;
|
|
std::set<ulint>::const_iterator col_it;
|
|
|
|
/* Check if the column, part of an index to be dropped is part of any
|
|
other index which is not being dropped. If it so, then set the ord_part
|
|
of the column to 0. */
|
|
get_col_list_to_be_dropped(ctx, drop_list);
|
|
|
|
for(col_it = drop_list.begin(); col_it != drop_list.end(); ++col_it) {
|
|
if (!check_col_exists_in_indexes(ctx->new_table, *col_it)) {
|
|
ctx->new_table->cols[*col_it].ord_part = 0;
|
|
}
|
|
}
|
|
|
|
for (ulint i = 0; i < ctx->num_to_add_index; i++) {
|
|
dict_index_t* index = ctx->add_index[i];
|
|
DBUG_ASSERT(dict_index_get_online_status(index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
DBUG_ASSERT(*index->name == TEMP_INDEX_PREFIX);
|
|
index->name++;
|
|
}
|
|
|
|
if (ctx->num_to_drop_index) {
|
|
/* Really drop the indexes that were dropped.
|
|
The transaction had to be committed first
|
|
(after renaming the indexes), so that in the
|
|
event of a crash, crash recovery will drop the
|
|
indexes, because it drops all indexes whose
|
|
names start with TEMP_INDEX_PREFIX. Once we
|
|
have started dropping an index tree, there is
|
|
no way to roll it back. */
|
|
|
|
for (ulint i = 0; i < ctx->num_to_drop_index; i++) {
|
|
dict_index_t* index = ctx->drop_index[i];
|
|
DBUG_ASSERT(*index->name != TEMP_INDEX_PREFIX);
|
|
DBUG_ASSERT(index->table == ctx->new_table);
|
|
DBUG_ASSERT(index->to_be_dropped);
|
|
|
|
/* Replace the indexes in foreign key
|
|
constraints if needed. */
|
|
|
|
if (!dict_foreign_replace_index(
|
|
index->table, ctx->col_names, index)) {
|
|
found = false;
|
|
}
|
|
|
|
/* Mark the index dropped
|
|
in the data dictionary cache. */
|
|
rw_lock_x_lock(dict_index_get_lock(index));
|
|
index->page = FIL_NULL;
|
|
rw_lock_x_unlock(dict_index_get_lock(index));
|
|
}
|
|
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
|
|
row_merge_drop_indexes_dict(trx, ctx->new_table->id);
|
|
|
|
for (ulint i = 0; i < ctx->num_to_drop_index; i++) {
|
|
dict_index_t* index = ctx->drop_index[i];
|
|
DBUG_ASSERT(*index->name != TEMP_INDEX_PREFIX);
|
|
DBUG_ASSERT(index->table == ctx->new_table);
|
|
|
|
if (index->type & DICT_FTS) {
|
|
DBUG_ASSERT(index->type == DICT_FTS
|
|
|| (index->type
|
|
& DICT_CORRUPT));
|
|
DBUG_ASSERT(index->table->fts);
|
|
fts_drop_index(index->table, index, trx);
|
|
}
|
|
|
|
dict_index_remove_from_cache(index->table, index);
|
|
}
|
|
|
|
trx_commit_for_mysql(trx);
|
|
}
|
|
|
|
DBUG_RETURN(found);
|
|
}
|
|
|
|
/** Adjust the persistent statistics after non-rebuilding ALTER TABLE.
|
|
Remove statistics for dropped indexes, add statistics for created indexes
|
|
and rename statistics for renamed indexes.
|
|
@param ha_alter_info Data used during in-place alter
|
|
@param ctx In-place ALTER TABLE context
|
|
@param altered_table MySQL table that is being altered
|
|
@param table_name Table name in MySQL
|
|
@param thd MySQL connection
|
|
*/
|
|
static
|
|
void
|
|
alter_stats_norebuild(
|
|
/*==================*/
|
|
Alter_inplace_info* ha_alter_info,
|
|
ha_innobase_inplace_ctx* ctx,
|
|
TABLE* altered_table,
|
|
const char* table_name,
|
|
THD* thd)
|
|
{
|
|
ulint i;
|
|
|
|
DBUG_ENTER("alter_stats_norebuild");
|
|
DBUG_ASSERT(!ctx->need_rebuild());
|
|
|
|
if (!dict_stats_is_persistent_enabled(ctx->new_table)) {
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/* TODO: This will not drop the (unused) statistics for
|
|
FTS_DOC_ID_INDEX if it was a hidden index, dropped together
|
|
with the last renamining FULLTEXT index. */
|
|
for (i = 0; i < ha_alter_info->index_drop_count; i++) {
|
|
const KEY* key = ha_alter_info->index_drop_buffer[i];
|
|
|
|
if (key->flags & HA_FULLTEXT) {
|
|
/* There are no index cardinality
|
|
statistics for FULLTEXT indexes. */
|
|
continue;
|
|
}
|
|
|
|
char errstr[1024];
|
|
|
|
if (dict_stats_drop_index(
|
|
ctx->new_table->name, key->name,
|
|
errstr, sizeof errstr) != DB_SUCCESS) {
|
|
push_warning(thd,
|
|
Sql_condition::WARN_LEVEL_WARN,
|
|
ER_LOCK_WAIT_TIMEOUT, errstr);
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < ctx->num_to_add_index; i++) {
|
|
dict_index_t* index = ctx->add_index[i];
|
|
DBUG_ASSERT(index->table == ctx->new_table);
|
|
|
|
if (!(index->type & DICT_FTS)) {
|
|
dict_stats_init(ctx->new_table);
|
|
dict_stats_update_for_index(index);
|
|
}
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/** Adjust the persistent statistics after rebuilding ALTER TABLE.
|
|
Remove statistics for dropped indexes, add statistics for created indexes
|
|
and rename statistics for renamed indexes.
|
|
@param table InnoDB table that was rebuilt by ALTER TABLE
|
|
@param table_name Table name in MySQL
|
|
@param thd MySQL connection
|
|
*/
|
|
static
|
|
void
|
|
alter_stats_rebuild(
|
|
/*================*/
|
|
dict_table_t* table,
|
|
const char* table_name,
|
|
THD* thd)
|
|
{
|
|
DBUG_ENTER("alter_stats_rebuild");
|
|
|
|
if (dict_table_is_discarded(table)
|
|
|| !dict_stats_is_persistent_enabled(table)) {
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
dberr_t ret;
|
|
|
|
ret = dict_stats_update(table, DICT_STATS_RECALC_PERSISTENT);
|
|
|
|
if (ret != DB_SUCCESS) {
|
|
push_warning_printf(
|
|
thd,
|
|
Sql_condition::WARN_LEVEL_WARN,
|
|
ER_ALTER_INFO,
|
|
"Error updating stats for table '%s' "
|
|
"after table rebuild: %s",
|
|
table_name, ut_strerr(ret));
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
#ifndef DBUG_OFF
|
|
# define DBUG_INJECT_CRASH(prefix, count) \
|
|
do { \
|
|
char buf[32]; \
|
|
ut_snprintf(buf, sizeof buf, prefix "_%u", count); \
|
|
DBUG_EXECUTE_IF(buf, DBUG_SUICIDE();); \
|
|
} while (0)
|
|
#else
|
|
# define DBUG_INJECT_CRASH(prefix, count)
|
|
#endif
|
|
|
|
/** Commit or rollback the changes made during
|
|
prepare_inplace_alter_table() and inplace_alter_table() inside
|
|
the storage engine. Note that the allowed level of concurrency
|
|
during this operation will be the same as for
|
|
inplace_alter_table() and thus might be higher than during
|
|
prepare_inplace_alter_table(). (E.g concurrent writes were
|
|
blocked during prepare, but might not be during commit).
|
|
@param altered_table TABLE object for new version of table.
|
|
@param ha_alter_info Structure describing changes to be done
|
|
by ALTER TABLE and holding data used during in-place alter.
|
|
@param commit true => Commit, false => Rollback.
|
|
@retval true Failure
|
|
@retval false Success
|
|
*/
|
|
UNIV_INTERN
|
|
bool
|
|
ha_innobase::commit_inplace_alter_table(
|
|
/*====================================*/
|
|
TABLE* altered_table,
|
|
Alter_inplace_info* ha_alter_info,
|
|
bool commit)
|
|
{
|
|
dberr_t error;
|
|
ha_innobase_inplace_ctx* ctx0
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(ha_alter_info->handler_ctx);
|
|
#ifndef DBUG_OFF
|
|
uint crash_inject_count = 1;
|
|
uint crash_fail_inject_count = 1;
|
|
uint failure_inject_count = 1;
|
|
#endif
|
|
|
|
DBUG_ENTER("commit_inplace_alter_table");
|
|
DBUG_ASSERT(!srv_read_only_mode);
|
|
DBUG_ASSERT(!ctx0 || ctx0->prebuilt == prebuilt);
|
|
DBUG_ASSERT(!ctx0 || ctx0->old_table == prebuilt->table);
|
|
|
|
DEBUG_SYNC_C("innodb_commit_inplace_alter_table_enter");
|
|
|
|
DEBUG_SYNC_C("innodb_commit_inplace_alter_table_wait");
|
|
|
|
if (!commit) {
|
|
/* A rollback is being requested. So far we may at
|
|
most have created some indexes. If any indexes were to
|
|
be dropped, they would actually be dropped in this
|
|
method if commit=true. */
|
|
DBUG_RETURN(rollback_inplace_alter_table(
|
|
ha_alter_info, table, prebuilt));
|
|
}
|
|
|
|
if (!(ha_alter_info->handler_flags & ~INNOBASE_INPLACE_IGNORE)) {
|
|
DBUG_ASSERT(!ctx0);
|
|
MONITOR_ATOMIC_DEC(MONITOR_PENDING_ALTER_TABLE);
|
|
ha_alter_info->group_commit_ctx = NULL;
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
DBUG_ASSERT(ctx0);
|
|
|
|
inplace_alter_handler_ctx** ctx_array;
|
|
inplace_alter_handler_ctx* ctx_single[2];
|
|
|
|
if (ha_alter_info->group_commit_ctx) {
|
|
ctx_array = ha_alter_info->group_commit_ctx;
|
|
} else {
|
|
ctx_single[0] = ctx0;
|
|
ctx_single[1] = NULL;
|
|
ctx_array = ctx_single;
|
|
}
|
|
|
|
DBUG_ASSERT(ctx0 == ctx_array[0]);
|
|
ut_ad(prebuilt->table == ctx0->old_table);
|
|
ha_alter_info->group_commit_ctx = NULL;
|
|
|
|
/* Free the ctx->trx of other partitions, if any. We will only
|
|
use the ctx0->trx here. Others may have been allocated in
|
|
the prepare stage. */
|
|
|
|
for (inplace_alter_handler_ctx** pctx = &ctx_array[1]; *pctx;
|
|
pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
|
|
if (ctx->trx) {
|
|
trx_free_for_mysql(ctx->trx);
|
|
ctx->trx = NULL;
|
|
}
|
|
}
|
|
|
|
trx_start_if_not_started_xa(prebuilt->trx);
|
|
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array; *pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
DBUG_ASSERT(ctx->prebuilt->trx == prebuilt->trx);
|
|
|
|
/* Exclusively lock the table, to ensure that no other
|
|
transaction is holding locks on the table while we
|
|
change the table definition. The MySQL meta-data lock
|
|
should normally guarantee that no conflicting locks
|
|
exist. However, FOREIGN KEY constraints checks and any
|
|
transactions collected during crash recovery could be
|
|
holding InnoDB locks only, not MySQL locks. */
|
|
|
|
error = row_merge_lock_table(
|
|
prebuilt->trx, ctx->old_table, LOCK_X);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
my_error_innodb(
|
|
error, table_share->table_name.str, 0);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
DEBUG_SYNC(user_thd, "innodb_alter_commit_after_lock_table");
|
|
|
|
const bool new_clustered = ctx0->need_rebuild();
|
|
trx_t* trx = ctx0->trx;
|
|
bool fail = false;
|
|
|
|
if (new_clustered) {
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
DBUG_ASSERT(ctx->need_rebuild());
|
|
|
|
if (ctx->old_table->fts) {
|
|
ut_ad(!ctx->old_table->fts->add_wq);
|
|
fts_optimize_remove_table(
|
|
ctx->old_table);
|
|
}
|
|
|
|
if (ctx->new_table->fts) {
|
|
ut_ad(!ctx->new_table->fts->add_wq);
|
|
fts_optimize_remove_table(
|
|
ctx->new_table);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!trx) {
|
|
DBUG_ASSERT(!new_clustered);
|
|
trx = innobase_trx_allocate(user_thd);
|
|
}
|
|
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
|
|
/* Latch the InnoDB data dictionary exclusively so that no deadlocks
|
|
or lock waits can happen in it during the data dictionary operation. */
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
/* Prevent the background statistics collection from accessing
|
|
the tables. */
|
|
for (;;) {
|
|
bool retry = false;
|
|
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
|
|
DBUG_ASSERT(new_clustered == ctx->need_rebuild());
|
|
|
|
if (new_clustered
|
|
&& !dict_stats_stop_bg(ctx->old_table)) {
|
|
retry = true;
|
|
}
|
|
|
|
if (!dict_stats_stop_bg(ctx->new_table)) {
|
|
retry = true;
|
|
}
|
|
}
|
|
|
|
if (!retry) {
|
|
break;
|
|
}
|
|
|
|
DICT_STATS_BG_YIELD(trx);
|
|
}
|
|
|
|
/* Apply the changes to the data dictionary tables, for all
|
|
partitions. */
|
|
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx && !fail; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
|
|
DBUG_ASSERT(new_clustered == ctx->need_rebuild());
|
|
|
|
ctx->max_autoinc = commit_get_autoinc(
|
|
ha_alter_info, ctx, altered_table, table);
|
|
|
|
if (ctx->need_rebuild()) {
|
|
ctx->tmp_name = dict_mem_create_temporary_tablename(
|
|
ctx->heap, ctx->new_table->name,
|
|
ctx->new_table->id);
|
|
|
|
fail = commit_try_rebuild(
|
|
ha_alter_info, ctx, altered_table, table,
|
|
trx, table_share->table_name.str);
|
|
} else {
|
|
fail = commit_try_norebuild(
|
|
ha_alter_info, ctx, table, trx,
|
|
table_share->table_name.str);
|
|
}
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash",
|
|
crash_inject_count++);
|
|
#ifndef DBUG_OFF
|
|
{
|
|
/* Generate a dynamic dbug text. */
|
|
char buf[32];
|
|
ut_snprintf(buf, sizeof buf, "ib_commit_inplace_fail_%u",
|
|
failure_inject_count++);
|
|
DBUG_EXECUTE_IF(buf,
|
|
my_error(ER_INTERNAL_ERROR, MYF(0),
|
|
"Injected error!");
|
|
fail = true;
|
|
);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/* Commit or roll back the changes to the data dictionary. */
|
|
|
|
if (fail) {
|
|
trx_rollback_for_mysql(trx);
|
|
} else if (!new_clustered) {
|
|
trx_commit_for_mysql(trx);
|
|
} else {
|
|
mtr_t mtr;
|
|
mtr_start(&mtr);
|
|
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
|
|
DBUG_ASSERT(ctx->need_rebuild());
|
|
/* Check for any possible problems for any
|
|
file operations that will be performed in
|
|
commit_cache_rebuild(), and if none, generate
|
|
the redo log for these operations. */
|
|
error = fil_mtr_rename_log(ctx->old_table,
|
|
ctx->new_table,
|
|
ctx->tmp_name, &mtr);
|
|
if (error != DB_SUCCESS) {
|
|
/* Out of memory or a problem will occur
|
|
when renaming files. */
|
|
fail = true;
|
|
my_error_innodb(error, ctx->old_table->name,
|
|
ctx->old_table->flags);
|
|
}
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash",
|
|
crash_inject_count++);
|
|
}
|
|
|
|
/* Test what happens on crash if the redo logs
|
|
are flushed to disk here. The log records
|
|
about the rename should not be committed, and
|
|
the data dictionary transaction should be
|
|
rolled back, restoring the old table. */
|
|
DBUG_EXECUTE_IF("innodb_alter_commit_crash_before_commit",
|
|
log_buffer_flush_to_disk();
|
|
DBUG_SUICIDE(););
|
|
ut_ad(!trx->fts_trx);
|
|
|
|
if (fail) {
|
|
mtr_set_log_mode(&mtr, MTR_LOG_NO_REDO);
|
|
mtr_commit(&mtr);
|
|
trx_rollback_for_mysql(trx);
|
|
} else {
|
|
/* The following call commits the
|
|
mini-transaction, making the data dictionary
|
|
transaction committed at mtr.end_lsn. The
|
|
transaction becomes 'durable' by the time when
|
|
log_buffer_flush_to_disk() returns. In the
|
|
logical sense the commit in the file-based
|
|
data structures happens here. */
|
|
ut_ad(trx_state_eq(trx, TRX_STATE_ACTIVE));
|
|
ut_ad(trx->insert_undo || trx->update_undo);
|
|
|
|
trx_commit_low(trx, &mtr);
|
|
}
|
|
|
|
/* If server crashes here, the dictionary in
|
|
InnoDB and MySQL will differ. The .ibd files
|
|
and the .frm files must be swapped manually by
|
|
the administrator. No loss of data. */
|
|
DBUG_EXECUTE_IF("innodb_alter_commit_crash_after_commit",
|
|
log_buffer_flush_to_disk();
|
|
DBUG_SUICIDE(););
|
|
}
|
|
|
|
/* Flush the log to reduce probability that the .frm files and
|
|
the InnoDB data dictionary get out-of-sync if the user runs
|
|
with innodb_flush_log_at_trx_commit = 0 */
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
/* At this point, the changes to the persistent storage have
|
|
been committed or rolled back. What remains to be done is to
|
|
update the in-memory structures, close some handles, release
|
|
temporary files, and (unless we rolled back) update persistent
|
|
statistics. */
|
|
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>(*pctx);
|
|
|
|
DBUG_ASSERT(ctx->need_rebuild() == new_clustered);
|
|
|
|
if (new_clustered) {
|
|
innobase_online_rebuild_log_free(ctx->old_table);
|
|
}
|
|
|
|
if (fail) {
|
|
if (new_clustered) {
|
|
dict_table_close(ctx->new_table,
|
|
TRUE, FALSE);
|
|
|
|
#if defined UNIV_DEBUG || defined UNIV_DDL_DEBUG
|
|
/* Nobody should have initialized the
|
|
stats of the newly created table
|
|
yet. When this is the case, we know
|
|
that it has not been added for
|
|
background stats gathering. */
|
|
ut_a(!ctx->new_table->stat_initialized);
|
|
#endif /* UNIV_DEBUG || UNIV_DDL_DEBUG */
|
|
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_TABLE);
|
|
row_merge_drop_table(trx, ctx->new_table);
|
|
trx_commit_for_mysql(trx);
|
|
ctx->new_table = NULL;
|
|
} else {
|
|
/* We failed, but did not rebuild the table.
|
|
Roll back any ADD INDEX, or get rid of garbage
|
|
ADD INDEX that was left over from a previous
|
|
ALTER TABLE statement. */
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_INDEX);
|
|
innobase_rollback_sec_index(
|
|
ctx->new_table, table, TRUE, trx);
|
|
trx_commit_for_mysql(trx);
|
|
}
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash_fail",
|
|
crash_fail_inject_count++);
|
|
|
|
continue;
|
|
}
|
|
|
|
innobase_copy_frm_flags_from_table_share(
|
|
ctx->new_table, altered_table->s);
|
|
|
|
if (new_clustered) {
|
|
/* We will reload and refresh the
|
|
in-memory foreign key constraint
|
|
metadata. This is a rename operation
|
|
in preparing for dropping the old
|
|
table. Set the table to_be_dropped bit
|
|
here, so to make sure DML foreign key
|
|
constraint check does not use the
|
|
stale dict_foreign_t. This is done
|
|
because WL#6049 (FK MDL) has not been
|
|
implemented yet. */
|
|
ctx->old_table->to_be_dropped = true;
|
|
|
|
/* Rename the tablespace files. */
|
|
commit_cache_rebuild(ctx);
|
|
|
|
error = innobase_update_foreign_cache(ctx, user_thd);
|
|
if (error != DB_SUCCESS) {
|
|
goto foreign_fail;
|
|
}
|
|
} else {
|
|
error = innobase_update_foreign_cache(ctx, user_thd);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
foreign_fail:
|
|
/* The data dictionary cache
|
|
should be corrupted now. The
|
|
best solution should be to
|
|
kill and restart the server,
|
|
but the *.frm file has not
|
|
been replaced yet. */
|
|
my_error(ER_CANNOT_ADD_FOREIGN,
|
|
MYF(0));
|
|
sql_print_error(
|
|
"InnoDB: dict_load_foreigns()"
|
|
" returned %u for %s",
|
|
(unsigned) error,
|
|
thd_query_string(user_thd)
|
|
->str);
|
|
ut_ad(0);
|
|
} else {
|
|
if (!commit_cache_norebuild(
|
|
ctx, table, trx)) {
|
|
ut_a(!prebuilt->trx->check_foreigns);
|
|
}
|
|
|
|
innobase_rename_columns_cache(
|
|
ha_alter_info, table,
|
|
ctx->new_table);
|
|
}
|
|
}
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash",
|
|
crash_inject_count++);
|
|
}
|
|
|
|
/* Invalidate the index translation table. In partitioned
|
|
tables, there is one TABLE_SHARE (and also only one TABLE)
|
|
covering all partitions. */
|
|
share->idx_trans_tbl.index_count = 0;
|
|
|
|
if (trx == ctx0->trx) {
|
|
ctx0->trx = NULL;
|
|
}
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
if (fail) {
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(*pctx);
|
|
DBUG_ASSERT(ctx->need_rebuild() == new_clustered);
|
|
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
ctx->old_table,
|
|
CHECK_ABORTED_OK));
|
|
ut_a(fts_check_cached_index(ctx->old_table));
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash_fail",
|
|
crash_fail_inject_count++);
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
trx_free_for_mysql(trx);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/* Release the table locks. */
|
|
trx_commit_for_mysql(prebuilt->trx);
|
|
|
|
DBUG_EXECUTE_IF("ib_ddl_crash_after_user_trx_commit", DBUG_SUICIDE(););
|
|
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(*pctx);
|
|
DBUG_ASSERT(ctx->need_rebuild() == new_clustered);
|
|
|
|
if (altered_table->found_next_number_field) {
|
|
dict_table_t* t = ctx->new_table;
|
|
|
|
dict_table_autoinc_lock(t);
|
|
dict_table_autoinc_initialize(t, ctx->max_autoinc);
|
|
dict_table_autoinc_unlock(t);
|
|
}
|
|
|
|
bool add_fts = false;
|
|
|
|
/* Publish the created fulltext index, if any.
|
|
Note that a fulltext index can be created without
|
|
creating the clustered index, if there already exists
|
|
a suitable FTS_DOC_ID column. If not, one will be
|
|
created, implying new_clustered */
|
|
for (ulint i = 0; i < ctx->num_to_add_index; i++) {
|
|
dict_index_t* index = ctx->add_index[i];
|
|
|
|
if (index->type & DICT_FTS) {
|
|
DBUG_ASSERT(index->type == DICT_FTS);
|
|
/* We reset DICT_TF2_FTS here because the bit
|
|
is left unset when a drop proceeds the add. */
|
|
DICT_TF2_FLAG_SET(ctx->new_table, DICT_TF2_FTS);
|
|
fts_add_index(index, ctx->new_table);
|
|
add_fts = true;
|
|
}
|
|
}
|
|
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
ctx->new_table, CHECK_ALL_COMPLETE));
|
|
|
|
if (add_fts) {
|
|
fts_optimize_add_table(ctx->new_table);
|
|
}
|
|
|
|
ut_d(dict_table_check_for_dup_indexes(
|
|
ctx->new_table, CHECK_ABORTED_OK));
|
|
ut_a(fts_check_cached_index(ctx->new_table));
|
|
|
|
if (new_clustered) {
|
|
/* Since the table has been rebuilt, we remove
|
|
all persistent statistics corresponding to the
|
|
old copy of the table (which was renamed to
|
|
ctx->tmp_name). */
|
|
|
|
char errstr[1024];
|
|
|
|
DBUG_ASSERT(0 == strcmp(ctx->old_table->name,
|
|
ctx->tmp_name));
|
|
|
|
if (dict_stats_drop_table(
|
|
ctx->new_table->name,
|
|
errstr, sizeof(errstr))
|
|
!= DB_SUCCESS) {
|
|
push_warning_printf(
|
|
user_thd,
|
|
Sql_condition::WARN_LEVEL_WARN,
|
|
ER_ALTER_INFO,
|
|
"Deleting persistent statistics"
|
|
" for rebuilt table '%s' in"
|
|
" InnoDB failed: %s",
|
|
table->s->table_name.str,
|
|
errstr);
|
|
}
|
|
|
|
DBUG_EXECUTE_IF("ib_ddl_crash_before_commit",
|
|
DBUG_SUICIDE(););
|
|
|
|
trx_t* const user_trx = prebuilt->trx;
|
|
|
|
row_prebuilt_free(ctx->prebuilt, TRUE);
|
|
|
|
/* Drop the copy of the old table, which was
|
|
renamed to ctx->tmp_name at the atomic DDL
|
|
transaction commit. If the system crashes
|
|
before this is completed, some orphan tables
|
|
with ctx->tmp_name may be recovered. */
|
|
trx_start_for_ddl(trx, TRX_DICT_OP_TABLE);
|
|
row_merge_drop_table(trx, ctx->old_table);
|
|
trx_commit_for_mysql(trx);
|
|
|
|
/* Rebuild the prebuilt object. */
|
|
ctx->prebuilt = row_create_prebuilt(
|
|
ctx->new_table, altered_table->s->reclength);
|
|
trx_start_if_not_started(user_trx);
|
|
user_trx->will_lock++;
|
|
prebuilt->trx = user_trx;
|
|
}
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash",
|
|
crash_inject_count++);
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
trx_free_for_mysql(trx);
|
|
|
|
/* Rebuild index translation table now for temporary tables if we are
|
|
restoring secondary keys, as ha_innobase::open will not be called for
|
|
the next access. */
|
|
if (dict_table_is_temporary(ctx0->new_table)
|
|
&& ctx0->num_to_add_index > 0) {
|
|
ut_ad(!ctx0->num_to_drop_index);
|
|
ut_ad(!ctx0->num_to_drop_fk);
|
|
if (!innobase_build_index_translation(altered_table,
|
|
ctx0->new_table,
|
|
share)) {
|
|
MONITOR_ATOMIC_DEC(MONITOR_PENDING_ALTER_TABLE);
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
|
|
/* TODO: The following code could be executed
|
|
while allowing concurrent access to the table
|
|
(MDL downgrade). */
|
|
|
|
if (new_clustered) {
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(*pctx);
|
|
DBUG_ASSERT(ctx->need_rebuild());
|
|
|
|
alter_stats_rebuild(
|
|
ctx->new_table, table->s->table_name.str,
|
|
user_thd);
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash",
|
|
crash_inject_count++);
|
|
}
|
|
} else {
|
|
for (inplace_alter_handler_ctx** pctx = ctx_array;
|
|
*pctx; pctx++) {
|
|
ha_innobase_inplace_ctx* ctx
|
|
= static_cast<ha_innobase_inplace_ctx*>
|
|
(*pctx);
|
|
DBUG_ASSERT(!ctx->need_rebuild());
|
|
|
|
alter_stats_norebuild(
|
|
ha_alter_info, ctx, altered_table,
|
|
table->s->table_name.str, user_thd);
|
|
DBUG_INJECT_CRASH("ib_commit_inplace_crash",
|
|
crash_inject_count++);
|
|
}
|
|
}
|
|
|
|
/* TODO: Also perform DROP TABLE and DROP INDEX after
|
|
the MDL downgrade. */
|
|
|
|
#ifndef DBUG_OFF
|
|
dict_index_t* clust_index = dict_table_get_first_index(
|
|
prebuilt->table);
|
|
DBUG_ASSERT(!clust_index->online_log);
|
|
DBUG_ASSERT(dict_index_get_online_status(clust_index)
|
|
== ONLINE_INDEX_COMPLETE);
|
|
|
|
for (dict_index_t* index = dict_table_get_first_index(
|
|
prebuilt->table);
|
|
index;
|
|
index = dict_table_get_next_index(index)) {
|
|
DBUG_ASSERT(!index->to_be_dropped);
|
|
}
|
|
#endif /* DBUG_OFF */
|
|
|
|
MONITOR_ATOMIC_DEC(MONITOR_PENDING_ALTER_TABLE);
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
/**
|
|
@param thd - the session
|
|
@param start_value - the lower bound
|
|
@param max_value - the upper bound (inclusive) */
|
|
UNIV_INTERN
|
|
ib_sequence_t::ib_sequence_t(
|
|
THD* thd,
|
|
ulonglong start_value,
|
|
ulonglong max_value)
|
|
:
|
|
m_max_value(max_value),
|
|
m_increment(0),
|
|
m_offset(0),
|
|
m_next_value(start_value),
|
|
m_eof(false)
|
|
{
|
|
if (thd != 0 && m_max_value > 0) {
|
|
|
|
thd_get_autoinc(thd, &m_offset, &m_increment);
|
|
|
|
if (m_increment > 1 || m_offset > 1) {
|
|
|
|
/* If there is an offset or increment specified
|
|
then we need to work out the exact next value. */
|
|
|
|
m_next_value = innobase_next_autoinc(
|
|
start_value, 1,
|
|
m_increment, m_offset, m_max_value);
|
|
|
|
} else if (start_value == 0) {
|
|
/* The next value can never be 0. */
|
|
m_next_value = 1;
|
|
}
|
|
} else {
|
|
m_eof = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Postfix increment
|
|
@return the next value to insert */
|
|
UNIV_INTERN
|
|
ulonglong
|
|
ib_sequence_t::operator++(int) UNIV_NOTHROW
|
|
{
|
|
ulonglong current = m_next_value;
|
|
|
|
ut_ad(!m_eof);
|
|
ut_ad(m_max_value > 0);
|
|
|
|
m_next_value = innobase_next_autoinc(
|
|
current, 1, m_increment, m_offset, m_max_value);
|
|
|
|
if (m_next_value == m_max_value && current == m_next_value) {
|
|
m_eof = true;
|
|
}
|
|
|
|
return(current);
|
|
}
|