mirror of
https://github.com/MariaDB/server.git
synced 2025-01-18 04:53:01 +01:00
50eb40a2a8
MDEV-11581: Mariadb starts InnoDB encryption threads when key has not changed or data scrubbing turned off Background: Key rotation is based on background threads (innodb-encryption-threads) periodically going through all tablespaces on fil_system. For each tablespace current used key version is compared to max key age (innodb-encryption-rotate-key-age). This process naturally takes CPU. Similarly, in same time need for scrubbing is investigated. Currently, key rotation is fully supported on Amazon AWS key management plugin only but InnoDB does not have knowledge what key management plugin is used. This patch re-purposes innodb-encryption-rotate-key-age=0 to disable key rotation and background data scrubbing. All new tables are added to special list for key rotation and key rotation is based on sending a event to background encryption threads instead of using periodic checking (i.e. timeout). fil0fil.cc: Added functions fil_space_acquire_low() to acquire a tablespace when it could be dropped concurrently. This function is used from fil_space_acquire() or fil_space_acquire_silent() that will not print any messages if we try to acquire space that does not exist. fil_space_release() to release a acquired tablespace. fil_space_next() to iterate tablespaces in fil_system using fil_space_acquire() and fil_space_release(). Similarly, fil_space_keyrotation_next() to iterate new list fil_system->rotation_list where new tables. are added if key rotation is disabled. Removed unnecessary functions fil_get_first_space_safe() fil_get_next_space_safe() fil_node_open_file(): After page 0 is read read also crypt_info if it is not yet read. btr_scrub_lock_dict_func() buf_page_check_corrupt() buf_page_encrypt_before_write() buf_merge_or_delete_for_page() lock_print_info_all_transactions() row_fts_psort_info_init() row_truncate_table_for_mysql() row_drop_table_for_mysql() Use fil_space_acquire()/release() to access fil_space_t. buf_page_decrypt_after_read(): Use fil_space_get_crypt_data() because at this point we might not yet have read page 0. fil0crypt.cc/fil0fil.h: Lot of changes. Pass fil_space_t* directly to functions needing it and store fil_space_t* to rotation state. Use fil_space_acquire()/release() when iterating tablespaces and removed unnecessary is_closing from fil_crypt_t. Use fil_space_t::is_stopping() to detect when access to tablespace should be stopped. Removed unnecessary fil_space_get_crypt_data(). fil_space_create(): Inform key rotation that there could be something to do if key rotation is disabled and new table with encryption enabled is created. Remove unnecessary functions fil_get_first_space_safe() and fil_get_next_space_safe(). fil_space_acquire() and fil_space_release() are used instead. Moved fil_space_get_crypt_data() and fil_space_set_crypt_data() to fil0crypt.cc. fsp_header_init(): Acquire fil_space_t*, write crypt_data and release space. check_table_options() Renamed FIL_SPACE_ENCRYPTION_* TO FIL_ENCRYPTION_* i_s.cc: Added ROTATING_OR_FLUSHING field to information_schema.innodb_tablespace_encryption to show current status of key rotation.
6382 lines
178 KiB
C++
6382 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);
|
|
ut_error;
|
|
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->def != 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_FK_DUP_NAME,
|
|
MYF(0),
|
|
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->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);
|
|
}
|