mirror of
https://github.com/MariaDB/server.git
synced 2025-01-21 14:32:34 +01:00
a0b4028fb0
git-svn-id: file:///svn/mysql/tokudb-engine/tokudb-engine@45854 c7de825b-a66e-492c-adef-691d508d4ae1
484 lines
21 KiB
C++
484 lines
21 KiB
C++
#if TOKU_INCLUDE_ALTER_56
|
|
|
|
#include "ha_tokudb_alter_common.cc"
|
|
|
|
void
|
|
ha_tokudb::print_alter_info(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
printf("***are keys of two tables same? %d\n", tables_have_same_keys(table, altered_table, false, false));
|
|
if (ha_alter_info->handler_flags) {
|
|
printf("***alter flags set ***\n");
|
|
for (int i = 0; i < 32; i++) {
|
|
if (ha_alter_info->handler_flags & (1 << i))
|
|
printf("%d\n", i);
|
|
}
|
|
}
|
|
|
|
// everyone calculates data by doing some default_values - record[0], but I do not see why
|
|
// that is necessary
|
|
printf("******\n");
|
|
printf("***orig table***\n");
|
|
for (uint i = 0; i < table->s->fields; i++) {
|
|
//
|
|
// make sure to use table->field, and NOT table->s->field
|
|
//
|
|
Field* curr_field = table->field[i];
|
|
uint null_offset = get_null_offset(table, curr_field);
|
|
printf(
|
|
"name: %s, nullable: %d, null_offset: %d, is_null_field: %d, is_null %d, \n",
|
|
curr_field->field_name,
|
|
curr_field->null_bit,
|
|
null_offset,
|
|
(curr_field->null_ptr != NULL),
|
|
(curr_field->null_ptr != NULL) ? table->s->default_values[null_offset] & curr_field->null_bit : 0xffffffff
|
|
);
|
|
}
|
|
printf("******\n");
|
|
printf("***altered table***\n");
|
|
for (uint i = 0; i < altered_table->s->fields; i++) {
|
|
Field* curr_field = altered_table->field[i];
|
|
uint null_offset = get_null_offset(altered_table, curr_field);
|
|
printf(
|
|
"name: %s, nullable: %d, null_offset: %d, is_null_field: %d, is_null %d, \n",
|
|
curr_field->field_name,
|
|
curr_field->null_bit,
|
|
null_offset,
|
|
(curr_field->null_ptr != NULL),
|
|
(curr_field->null_ptr != NULL) ? altered_table->s->default_values[null_offset] & curr_field->null_bit : 0xffffffff
|
|
);
|
|
}
|
|
printf("******\n");
|
|
}
|
|
|
|
class ha_tokudb_add_index_ctx : public inplace_alter_handler_ctx {
|
|
public:
|
|
ha_tokudb_add_index_ctx(bool _incremented_num_DBs, bool _modified_DBs) : incremented_num_DBs(_incremented_num_DBs), modified_DBs(_modified_DBs) {
|
|
}
|
|
virtual ~ha_tokudb_add_index_ctx() {
|
|
}
|
|
public:
|
|
bool incremented_num_DBs, modified_DBs;
|
|
};
|
|
|
|
// workaround for fill_alter_inplace_info bug (#5193)
|
|
// the function erroneously sets the ADD_INDEX and DROP_INDEX flags for a column addition that does not
|
|
// change the keys. the following code turns the ADD_INDEX and DROP_INDEX flags so that we can do hot
|
|
// column addition later.
|
|
static ulong
|
|
fix_handler_flags(Alter_inplace_info *ha_alter_info, TABLE *table, TABLE *altered_table) {
|
|
ulong handler_flags = ha_alter_info->handler_flags;
|
|
if ((handler_flags & (Alter_inplace_info::ADD_COLUMN + Alter_inplace_info::DROP_COLUMN)) != 0) {
|
|
if ((handler_flags & (Alter_inplace_info::ADD_INDEX + Alter_inplace_info::DROP_INDEX)) != 0) {
|
|
if (tables_have_same_keys(table, altered_table, false, false)) {
|
|
handler_flags &= ~(Alter_inplace_info::ADD_INDEX + Alter_inplace_info::DROP_INDEX);
|
|
}
|
|
}
|
|
}
|
|
return handler_flags;
|
|
}
|
|
|
|
// true if some bit in mask is set and no bit in ~mask is set
|
|
// otherwise false
|
|
static bool
|
|
only_flags(ulong bits, ulong mask) {
|
|
return (bits & mask) != 0 && (bits & ~mask) == 0;
|
|
}
|
|
|
|
enum_alter_inplace_result
|
|
ha_tokudb::check_if_supported_inplace_alter(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
TOKUDB_DBUG_ENTER("check_if_supported_alter");
|
|
|
|
if (tokudb_debug & TOKUDB_DEBUG_ALTER_TABLE_INFO) {
|
|
print_alter_info(altered_table, ha_alter_info);
|
|
}
|
|
|
|
THD *thd = ha_thd();
|
|
enum_alter_inplace_result result = HA_ALTER_INPLACE_NOT_SUPPORTED; // default is NOT inplace
|
|
HA_CREATE_INFO *create_info = ha_alter_info->create_info;
|
|
|
|
ulong handler_flags = fix_handler_flags(ha_alter_info, table, altered_table);
|
|
|
|
// add/drop index
|
|
#if 0
|
|
// turn this on when we support add and drop in the same statement
|
|
if (only_flags(handler_flags, Alter_inplace_info::DROP_INDEX + Alter_inplace_info::DROP_UNIQUE_INDEX +
|
|
Alter_inplace_info::ADD_INDEX + Alter_inplace_info::ADD_UNIQUE_INDEX)) {
|
|
if ((ha_alter_info->index_add_count > 0 || ha_alter_info->index_drop_count > 0) &&
|
|
!tables_have_same_keys(table, altered_table, false, false)) {
|
|
|
|
result = HA_ALTER_INPLACE_SHARED_LOCK;
|
|
|
|
// someday, allow multiple hot indexes via alter table add key. don't forget to change the store_lock function.
|
|
// for now, hot indexing is only supported via session variable with the create index sql command
|
|
if (ha_alter_info->index_add_count == 1 && ha_alter_info->index_drop_count == 0 &&
|
|
get_create_index_online(thd) && thd_sql_command(thd) == SQLCOM_CREATE_INDEX) {
|
|
result = HA_ALTER_INPLACE_NO_LOCK;
|
|
}
|
|
}
|
|
} else
|
|
#else
|
|
if (only_flags(handler_flags, Alter_inplace_info::ADD_INDEX + Alter_inplace_info::ADD_UNIQUE_INDEX)) {
|
|
if ((ha_alter_info->index_add_count > 0) &&
|
|
!tables_have_same_keys(table, altered_table, false, false)) {
|
|
|
|
result = HA_ALTER_INPLACE_SHARED_LOCK;
|
|
|
|
// someday, allow multiple hot indexes via alter table add key. don't forget to change the store_lock function.
|
|
// for now, hot indexing is only supported via session variable with the create index sql command
|
|
if (ha_alter_info->index_add_count == 1 && ha_alter_info->index_drop_count == 0 &&
|
|
get_create_index_online(thd) && thd_sql_command(thd) == SQLCOM_CREATE_INDEX) {
|
|
result = HA_ALTER_INPLACE_NO_LOCK;
|
|
}
|
|
}
|
|
} else
|
|
if (only_flags(handler_flags, Alter_inplace_info::DROP_INDEX + Alter_inplace_info::DROP_UNIQUE_INDEX)) {
|
|
if ((ha_alter_info->index_drop_count > 0) &&
|
|
!tables_have_same_keys(table, altered_table, false, false)) {
|
|
|
|
result = HA_ALTER_INPLACE_SHARED_LOCK;
|
|
}
|
|
} else
|
|
#endif
|
|
// column rename
|
|
if (only_flags(handler_flags, Alter_inplace_info::ALTER_COLUMN_NAME + Alter_inplace_info::ALTER_COLUMN_DEFAULT)) {
|
|
// we have identified a possible column rename,
|
|
// but let's do some more checks
|
|
|
|
// we will only allow an hcr if there are no changes
|
|
// in column positions (ALTER_COLUMN_ORDER is not set)
|
|
|
|
// now need to verify that one and only one column
|
|
// has changed only its name. If we find anything to
|
|
// the contrary, we don't allow it, also check indexes
|
|
bool cr_supported = column_rename_supported(table, altered_table, (ha_alter_info->handler_flags & Alter_inplace_info::ALTER_COLUMN_ORDER) != 0);
|
|
if (cr_supported)
|
|
result = HA_ALTER_INPLACE_NO_LOCK;
|
|
} else
|
|
// add column
|
|
if (only_flags(handler_flags, Alter_inplace_info::ADD_COLUMN + Alter_inplace_info::ALTER_COLUMN_ORDER)) {
|
|
u_int32_t added_columns[altered_table->s->fields];
|
|
u_int32_t num_added_columns = 0;
|
|
int r = find_changed_columns(added_columns, &num_added_columns, table, altered_table);
|
|
if (r == 0) {
|
|
if (tokudb_debug & TOKUDB_DEBUG_ALTER_TABLE_INFO) {
|
|
for (u_int32_t i = 0; i < num_added_columns; i++) {
|
|
u_int32_t curr_added_index = added_columns[i];
|
|
Field* curr_added_field = altered_table->field[curr_added_index];
|
|
printf("Added column: index %d, name %s\n", curr_added_index, curr_added_field->field_name);
|
|
}
|
|
}
|
|
result = HA_ALTER_INPLACE_EXCLUSIVE_LOCK;
|
|
}
|
|
} else
|
|
// drop column
|
|
if (only_flags(handler_flags, Alter_inplace_info::DROP_COLUMN + Alter_inplace_info::ALTER_COLUMN_ORDER)) {
|
|
u_int32_t dropped_columns[table->s->fields];
|
|
u_int32_t num_dropped_columns = 0;
|
|
int r = find_changed_columns(dropped_columns, &num_dropped_columns, altered_table, table);
|
|
if (r == 0) {
|
|
if (tokudb_debug & TOKUDB_DEBUG_ALTER_TABLE_INFO) {
|
|
for (u_int32_t i = 0; i < num_dropped_columns; i++) {
|
|
u_int32_t curr_dropped_index = dropped_columns[i];
|
|
Field* curr_dropped_field = table->field[curr_dropped_index];
|
|
printf("Dropped column: index %d, name %s\n", curr_dropped_index, curr_dropped_field->field_name);
|
|
}
|
|
}
|
|
result = HA_ALTER_INPLACE_EXCLUSIVE_LOCK;
|
|
}
|
|
} else
|
|
if (only_flags(handler_flags, Alter_inplace_info::CHANGE_CREATE_OPTION)) {
|
|
// alter auto_increment
|
|
if (only_flags(create_info->used_fields, HA_CREATE_USED_AUTO)) {
|
|
result = HA_ALTER_INPLACE_EXCLUSIVE_LOCK;
|
|
} else
|
|
// alter row_format
|
|
if (only_flags(create_info->used_fields, HA_CREATE_USED_ROW_FORMAT)) {
|
|
result = HA_ALTER_INPLACE_EXCLUSIVE_LOCK;
|
|
}
|
|
}
|
|
|
|
// turn not supported into error if the slow alter table (copy) is disabled
|
|
if (result == HA_ALTER_INPLACE_NOT_SUPPORTED && get_disable_slow_alter(thd)) {
|
|
print_error(HA_ERR_UNSUPPORTED, MYF(0));
|
|
result = HA_ALTER_ERROR;
|
|
}
|
|
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
// prepare not yet called by the mysql 5.5 patch
|
|
bool
|
|
ha_tokudb::prepare_inplace_alter_table(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
TOKUDB_DBUG_ENTER("prepare_inplace_alter_table");
|
|
|
|
bool result = false; // success
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
bool
|
|
ha_tokudb::inplace_alter_table(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
TOKUDB_DBUG_ENTER("inplace_alter_table");
|
|
|
|
int error = 0;
|
|
HA_CREATE_INFO *create_info = ha_alter_info->create_info;
|
|
ulong handler_flags = fix_handler_flags(ha_alter_info, table, altered_table);
|
|
|
|
if (error == 0 && (handler_flags & (Alter_inplace_info::DROP_INDEX + Alter_inplace_info::DROP_UNIQUE_INDEX))) {
|
|
error = alter_table_drop_index(altered_table, ha_alter_info);
|
|
}
|
|
if (error == 0 && (handler_flags & (Alter_inplace_info::ADD_INDEX + Alter_inplace_info::ADD_UNIQUE_INDEX))) {
|
|
error = alter_table_add_index(altered_table, ha_alter_info);
|
|
}
|
|
if (error == 0 && (handler_flags & (Alter_inplace_info::ADD_COLUMN + Alter_inplace_info::DROP_COLUMN))) {
|
|
error = alter_table_add_or_drop_column(altered_table, ha_alter_info);
|
|
}
|
|
if (error == 0 && (handler_flags & Alter_inplace_info::CHANGE_CREATE_OPTION) && (create_info->used_fields & HA_CREATE_USED_AUTO)) {
|
|
error = write_auto_inc_create(share->status_block, create_info->auto_increment_value, transaction);
|
|
}
|
|
if (error == 0 && (handler_flags & Alter_inplace_info::CHANGE_CREATE_OPTION) && (create_info->used_fields & HA_CREATE_USED_ROW_FORMAT)) {
|
|
enum toku_compression_method method = TOKU_NO_COMPRESSION;
|
|
method = row_type_to_compression_method(create_info->row_type);
|
|
|
|
// Set the new type.
|
|
u_int32_t curr_num_DBs = table->s->keys + test(hidden_primary_key);
|
|
for (u_int32_t i = 0; i < curr_num_DBs; ++i) {
|
|
DB *db = share->key_file[i];
|
|
error = db->change_compression_method(db, method);
|
|
if (error)
|
|
break;
|
|
}
|
|
}
|
|
|
|
bool result = false; // success
|
|
if (error) {
|
|
print_error(error, MYF(0));
|
|
result = true; // failure
|
|
}
|
|
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
int
|
|
ha_tokudb::alter_table_add_index(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
|
|
// sort keys in add index order
|
|
KEY *key_info = (KEY*) my_malloc(sizeof (KEY) * ha_alter_info->index_add_count, MYF(MY_WME));
|
|
for (uint i = 0; i < ha_alter_info->index_add_count; i++) {
|
|
KEY *key = &key_info[i];
|
|
*key = ha_alter_info->key_info_buffer[ha_alter_info->index_add_buffer[i]];
|
|
for (KEY_PART_INFO *key_part= key->key_part; key_part < key->key_part + key->key_parts; key_part++)
|
|
key_part->field = table->field[key_part->fieldnr];
|
|
}
|
|
|
|
bool incremented_num_DBs = false;
|
|
bool modified_DBs = false;
|
|
int error = tokudb_add_index(table, key_info, ha_alter_info->index_add_count, transaction, &incremented_num_DBs, &modified_DBs);
|
|
if (error == HA_ERR_FOUND_DUPP_KEY) {
|
|
// hack for now, in case of duplicate key error,
|
|
// because at the moment we cannot display the right key
|
|
// information to the user, so that he knows potentially what went
|
|
// wrong.
|
|
last_dup_key = MAX_KEY;
|
|
}
|
|
|
|
assert(ha_alter_info->handler_ctx == NULL);
|
|
ha_alter_info->handler_ctx = new ha_tokudb_add_index_ctx(incremented_num_DBs, modified_DBs);
|
|
assert(ha_alter_info->handler_ctx);
|
|
|
|
my_free(key_info);
|
|
|
|
return error;
|
|
}
|
|
|
|
int
|
|
ha_tokudb::alter_table_drop_index(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
// translate KEY pointers to indexes into the key_info array
|
|
uint index_drop_offsets[ha_alter_info->index_drop_count];
|
|
for (uint i = 0; i < ha_alter_info->index_drop_count; i++)
|
|
index_drop_offsets[i] = ha_alter_info->index_drop_buffer[i] - table->key_info;
|
|
|
|
// drop indexes
|
|
int error = drop_indexes(table, index_drop_offsets, ha_alter_info->index_drop_count, transaction);
|
|
|
|
return error;
|
|
}
|
|
|
|
int
|
|
ha_tokudb::alter_table_add_or_drop_column(TABLE *altered_table, Alter_inplace_info *ha_alter_info) {
|
|
int error;
|
|
uchar *column_extra = NULL;
|
|
uchar *row_desc_buff = NULL;
|
|
u_int32_t max_new_desc_size = 0;
|
|
u_int32_t max_column_extra_size;
|
|
u_int32_t num_column_extra;
|
|
u_int32_t num_columns = 0;
|
|
u_int32_t curr_num_DBs = table->s->keys + test(hidden_primary_key);
|
|
|
|
u_int32_t columns[table->s->fields + altered_table->s->fields]; // set size such that we know it is big enough for both cases
|
|
memset(columns, 0, sizeof(columns));
|
|
|
|
KEY_AND_COL_INFO altered_kc_info;
|
|
memset(&altered_kc_info, 0, sizeof(altered_kc_info));
|
|
|
|
error = allocate_key_and_col_info(altered_table->s, &altered_kc_info);
|
|
if (error) { goto cleanup; }
|
|
|
|
max_new_desc_size = get_max_desc_size(&altered_kc_info, altered_table);
|
|
row_desc_buff = (uchar *)my_malloc(max_new_desc_size, MYF(MY_WME));
|
|
if (row_desc_buff == NULL){ error = ENOMEM; goto cleanup;}
|
|
|
|
error = initialize_key_and_col_info(
|
|
altered_table->s,
|
|
altered_table,
|
|
&altered_kc_info,
|
|
hidden_primary_key,
|
|
primary_key
|
|
);
|
|
if (error) { goto cleanup; }
|
|
|
|
// generate the array of columns
|
|
if (ha_alter_info->handler_flags & Alter_inplace_info::DROP_COLUMN) {
|
|
find_changed_columns(
|
|
columns,
|
|
&num_columns,
|
|
altered_table,
|
|
table
|
|
);
|
|
} else
|
|
if (ha_alter_info->handler_flags & Alter_inplace_info::ADD_COLUMN) {
|
|
find_changed_columns(
|
|
columns,
|
|
&num_columns,
|
|
table,
|
|
altered_table
|
|
);
|
|
} else
|
|
assert(0);
|
|
max_column_extra_size =
|
|
STATIC_ROW_MUTATOR_SIZE + //max static row_mutator
|
|
4 + num_columns*(1+1+4+1+1+4) + altered_table->s->reclength + // max dynamic row_mutator
|
|
(4 + share->kc_info.num_blobs) + // max static blob size
|
|
(num_columns*(1+4+1+4)); // max dynamic blob size
|
|
column_extra = (uchar *)my_malloc(max_column_extra_size, MYF(MY_WME));
|
|
if (column_extra == NULL) { error = ENOMEM; goto cleanup; }
|
|
|
|
for (u_int32_t i = 0; i < curr_num_DBs; i++) {
|
|
DBT row_descriptor;
|
|
memset(&row_descriptor, 0, sizeof(row_descriptor));
|
|
KEY* prim_key = (hidden_primary_key) ? NULL : &altered_table->s->key_info[primary_key];
|
|
KEY* key_info = &altered_table->key_info[i];
|
|
if (i == primary_key) {
|
|
row_descriptor.size = create_main_key_descriptor(
|
|
row_desc_buff,
|
|
prim_key,
|
|
hidden_primary_key,
|
|
primary_key,
|
|
altered_table,
|
|
&altered_kc_info
|
|
);
|
|
row_descriptor.data = row_desc_buff;
|
|
}
|
|
else {
|
|
row_descriptor.size = create_secondary_key_descriptor(
|
|
row_desc_buff,
|
|
key_info,
|
|
prim_key,
|
|
hidden_primary_key,
|
|
altered_table,
|
|
primary_key,
|
|
i,
|
|
&altered_kc_info
|
|
);
|
|
row_descriptor.data = row_desc_buff;
|
|
}
|
|
error = share->key_file[i]->change_descriptor(
|
|
share->key_file[i],
|
|
transaction,
|
|
&row_descriptor,
|
|
0
|
|
);
|
|
if (error) { goto cleanup; }
|
|
|
|
if (i == primary_key || table_share->key_info[i].flags & HA_CLUSTERING) {
|
|
num_column_extra = fill_row_mutator(
|
|
column_extra,
|
|
columns,
|
|
num_columns,
|
|
altered_table,
|
|
&altered_kc_info,
|
|
i,
|
|
(ha_alter_info->handler_flags & Alter_inplace_info::ADD_COLUMN) != 0 // true if adding columns, otherwise is a drop
|
|
);
|
|
|
|
DBT column_dbt;
|
|
memset(&column_dbt, 0, sizeof column_dbt);
|
|
column_dbt.data = column_extra;
|
|
column_dbt.size = num_column_extra;
|
|
DBUG_ASSERT(num_column_extra <= max_column_extra_size);
|
|
error = share->key_file[i]->update_broadcast(
|
|
share->key_file[i],
|
|
transaction,
|
|
&column_dbt,
|
|
DB_IS_RESETTING_OP
|
|
);
|
|
if (error) { goto cleanup; }
|
|
}
|
|
}
|
|
|
|
error = 0;
|
|
cleanup:
|
|
free_key_and_col_info(&altered_kc_info);
|
|
my_free(row_desc_buff, MYF(MY_ALLOW_ZERO_PTR));
|
|
my_free(column_extra, MYF(MY_ALLOW_ZERO_PTR));
|
|
return error;
|
|
}
|
|
|
|
bool
|
|
ha_tokudb::commit_inplace_alter_table(TABLE *altered_table, Alter_inplace_info *ha_alter_info, bool commit) {
|
|
TOKUDB_DBUG_ENTER("commit_inplace_alter_table");
|
|
|
|
bool result = false; // success
|
|
ulong handler_flags = fix_handler_flags(ha_alter_info, table, altered_table);
|
|
|
|
if (commit) {
|
|
if (altered_table->part_info == NULL) {
|
|
// read frmdata for the altered table
|
|
uchar *frm_data; size_t frm_len;
|
|
int error = readfrm(altered_table->s->path.str, &frm_data, &frm_len);
|
|
if (error) {
|
|
result = true;
|
|
} else {
|
|
// transactionally write frmdata to status
|
|
assert(transaction);
|
|
error = write_to_status(share->status_block, hatoku_frm_data, (void *)frm_data, (uint)frm_len, transaction);
|
|
if (error) {
|
|
result = true;
|
|
}
|
|
|
|
my_free(frm_data);
|
|
}
|
|
if (error)
|
|
print_error(error, MYF(0));
|
|
}
|
|
}
|
|
|
|
if (!commit || result == true) {
|
|
if (handler_flags & (Alter_inplace_info::ADD_INDEX + Alter_inplace_info::ADD_UNIQUE_INDEX)) {
|
|
ha_tokudb_add_index_ctx *ctx = static_cast<ha_tokudb_add_index_ctx *>(ha_alter_info->handler_ctx);
|
|
assert(ctx);
|
|
restore_add_index(table, ha_alter_info->index_add_count, ctx->incremented_num_DBs, ctx->modified_DBs);
|
|
}
|
|
if (handler_flags & (Alter_inplace_info::DROP_INDEX + Alter_inplace_info::DROP_UNIQUE_INDEX)) {
|
|
// translate KEY pointers to indexes into the key_info array
|
|
uint index_drop_offsets[ha_alter_info->index_drop_count];
|
|
for (uint i = 0; i < ha_alter_info->index_drop_count; i++)
|
|
index_drop_offsets[i] = ha_alter_info->index_drop_buffer[i] - table->key_info;
|
|
|
|
restore_drop_indexes(table, index_drop_offsets, ha_alter_info->index_drop_count);
|
|
}
|
|
// TODO undo auto increment, row format change
|
|
}
|
|
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
#endif
|