Commit 11665dc3 authored by marko's avatar marko

branches/zip: Ensure that the dictionary updates are atomic by keeping

the data dictionary locked across the operations.

dict_table_decrement_handle_count(), row_prebuilt_free(): Add the flag
dict_locked, to prevent the acquisition of dict_sys->mutex.

innobase_rename_table(): Rename commit_flag to lock_and_commit,
and do not acquire dict_sys->mutex or flush the log unless the flag is set.
Remove bogus comment about utility threads, because the threads will
be waken up by the upper-level function ha_innobase::rename_table().

ha_innobase::add_index(): After creating a primary key, hold dict_sys->mutex
across all dictionary operations.
parent fd6f7e4d
......@@ -309,15 +309,21 @@ Decrements the count of open MySQL handles to a table. */
void
dict_table_decrement_handle_count(
/*==============================*/
dict_table_t* table) /* in: table */
dict_table_t* table, /* in: table */
ibool dict_locked) /* in: TRUE=data dictionary locked */
{
mutex_enter(&(dict_sys->mutex));
if (!dict_locked) {
mutex_enter(&dict_sys->mutex);
}
ut_ad(mutex_own(&dict_sys->mutex));
ut_a(table->n_mysql_handles_opened > 0);
table->n_mysql_handles_opened--;
mutex_exit(&(dict_sys->mutex));
if (!dict_locked) {
mutex_exit(&dict_sys->mutex);
}
}
/**************************************************************************
......
......@@ -2473,7 +2473,7 @@ ha_innobase::open(
my_free(upd_buff, MYF(0));
my_errno = ENOENT;
dict_table_decrement_handle_count(ib_table);
dict_table_decrement_handle_count(ib_table, FALSE);
DBUG_RETURN(HA_ERR_NO_SUCH_TABLE);
}
......@@ -2574,7 +2574,7 @@ ha_innobase::close(void)
innobase_release_temporary_latches(ht, thd);
}
row_prebuilt_free(prebuilt);
row_prebuilt_free(prebuilt, FALSE);
my_free(upd_buff, MYF(0));
free_share(share);
......@@ -5446,7 +5446,8 @@ innobase_rename_table(
trx_t* trx, /* in: transaction */
const char* from, /* in: old name of the table */
const char* to, /* in: new name of the table */
ibool commit_flag) /* in: if TRUE then commit */
ibool lock_and_commit)
/* in: TRUE=lock data dictionary and commit */
{
int error;
char* norm_to;
......@@ -5477,10 +5478,12 @@ innobase_rename_table(
/* Serialize data dictionary operations with dictionary mutex:
no deadlocks can occur then in these operations */
row_mysql_lock_data_dictionary(trx);
if (lock_and_commit) {
row_mysql_lock_data_dictionary(trx);
}
error = row_rename_table_for_mysql(
norm_from, norm_to, trx, commit_flag);
norm_from, norm_to, trx, lock_and_commit);
if (error != DB_SUCCESS) {
FILE* ef = dict_foreign_err_file;
......@@ -5492,16 +5495,15 @@ innobase_rename_table(
fputs(" failed!\n", ef);
}
row_mysql_unlock_data_dictionary(trx);
/* 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 */
if (lock_and_commit) {
row_mysql_unlock_data_dictionary(trx);
log_buffer_flush_to_disk();
/* 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 */
/* Tell the InnoDB server that there might be work for
utility threads: */
log_buffer_flush_to_disk();
}
my_free(norm_to, MYF(0));
my_free(norm_from, MYF(0));
......@@ -8414,9 +8416,9 @@ err_exit:
Drop table etc. do this latching in row0mysql.c. */
row_mysql_lock_data_dictionary(trx);
dict_locked = TRUE;
num_created = 0;
dict_locked = TRUE;
/* Create the indexes in SYS_INDEXES and load into dictionary.*/
......@@ -8438,48 +8440,47 @@ err_exit:
num_created++;
}
if (error == DB_SUCCESS) {
/* Raise version number of the table to track this table's
definition changes.*/
ut_ad(error == DB_SUCCESS);
indexed_table->version_number++;
/* Raise version number of the table to track this table's
definition changes. */
row_mysql_unlock_data_dictionary(trx);
dict_locked = FALSE;
indexed_table->version_number++;
mem_heap_empty(heap);
row_mysql_unlock_data_dictionary(trx);
dict_locked = FALSE;
ut_a(trx->n_active_thrs == 0);
ut_a(UT_LIST_GET_LEN(trx->signals) == 0);
mem_heap_empty(heap);
error = row_lock_table_for_merge(trx, innodb_table, LOCK_X);
}
ut_a(trx->n_active_thrs == 0);
ut_a(UT_LIST_GET_LEN(trx->signals) == 0);
error = row_lock_table_for_merge(trx, innodb_table, LOCK_X);
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
/* Acquire an exclusive table lock on the new table if a primary
key is to be built.*/
if (error == DB_SUCCESS && new_primary) {
goto error_handling;
}
if (UNIV_UNLIKELY(new_primary)) {
/* A primary key is to be built. Acquire an exclusive
table lock also on the table that is being created. */
ut_ad(indexed_table != innodb_table);
error = row_lock_table_for_merge(trx, indexed_table, LOCK_X);
}
if (error == DB_SUCCESS) {
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
/* Read clustered index of the table and build indexes
based on this information using temporary files and merge
sort.*/
error = row_merge_build_indexes(
trx, innodb_table, indexed_table, index,
num_of_idx);
if (error == DB_SUCCESS && new_primary) {
row_mysql_lock_data_dictionary(trx);
row_prebuilt_table_obsolete(innodb_table);
row_mysql_unlock_data_dictionary(trx);
goto error_handling;
}
}
/* Read the clustered index of the table and build indexes
based on this information using temporary files and merge sort. */
error = row_merge_build_indexes(trx, innodb_table, indexed_table,
index, num_of_idx);
error_handling:
#ifdef UNIV_DEBUG
/* TODO: At the moment we can't handle the following statement
in our debugging code below:
......@@ -8491,27 +8492,26 @@ err_exit:
ignore that in the dup index check.*/
//dict_table_check_for_dup_indexes(prebuilt->table);
#endif
error_handling:
if (dict_locked) {
row_mysql_unlock_data_dictionary(trx);
}
/* After an error, remove all those index definitions from the
dictionary which were defined. */
switch (error) {
case DB_SUCCESS:
ut_ad(!dict_locked);
break;
case DB_DUPLICATE_KEY:
prebuilt->trx->error_info = NULL;
prebuilt->trx->error_key_num = trx->error_key_num;
/* fall through */
default:
row_merge_drop_indexes(
trx, indexed_table, index, num_created);
if (indexed_table != innodb_table) {
if (new_primary) {
row_merge_drop_table(trx, indexed_table);
} else {
row_merge_drop_indexes(trx, indexed_table,
index, num_created);
}
goto func_exit;
}
......@@ -8528,6 +8528,11 @@ error_handling:
trx_start_if_not_started(trx);
row_mysql_lock_data_dictionary(trx);
dict_locked = TRUE;
row_prebuilt_table_obsolete(innodb_table);
/* Write entry for UNDO */
error = row_undo_report_rename_table_dict_operation(
trx, old_name, indexed_table->name, tmp_table_name);
......@@ -8555,24 +8560,26 @@ error_handling:
goto func_exit;
}
row_prebuilt_free(prebuilt);
row_prebuilt_free(prebuilt, TRUE);
prebuilt = row_create_prebuilt(indexed_table);
row_mysql_lock_data_dictionary(trx);
prebuilt->table->n_mysql_handles_opened++;
row_mysql_unlock_data_dictionary(trx);
/* Drop the old table if there are no open views
referring to it. If there such views, we will drop
the table when we free the prebuilts and there are no
more references to it. */
referring to it. If there are such views, we will
drop the table when we free the prebuilts and there
are no more references to it. */
error = row_merge_drop_table(trx, innodb_table);
}
func_exit:
mem_heap_free(heap);
innobase_commit_low(trx);/* work around a bug in mysql_alter_table() */
innobase_commit_low(trx);
if (dict_locked) {
row_mysql_unlock_data_dictionary(trx);
}
/* There might be work for utility threads.*/
srv_active_wake_master_thread();
......
......@@ -68,7 +68,8 @@ Decrements the count of open MySQL handles to a table. */
void
dict_table_decrement_handle_count(
/*==============================*/
dict_table_t* table); /* in: table */
dict_table_t* table, /* in: table */
ibool dict_locked); /* in: TRUE=data dictionary locked */
/**************************************************************************
Inits the data dictionary module. */
......
......@@ -141,7 +141,8 @@ Free a prebuilt struct for a MySQL table handle. */
void
row_prebuilt_free(
/*==============*/
row_prebuilt_t* prebuilt); /* in, own: prebuilt struct */
row_prebuilt_t* prebuilt, /* in, own: prebuilt struct */
ibool dict_locked); /* in: TRUE=data dictionary locked */
/*************************************************************************
Updates the transaction pointers in query graphs stored in the prebuilt
struct. */
......
......@@ -668,7 +668,8 @@ Free a prebuilt struct for a MySQL table handle. */
void
row_prebuilt_free(
/*==============*/
row_prebuilt_t* prebuilt) /* in, own: prebuilt struct */
row_prebuilt_t* prebuilt, /* in, own: prebuilt struct */
ibool dict_locked) /* in: TRUE=data dictionary locked */
{
ulint i;
......@@ -742,7 +743,7 @@ row_prebuilt_free(
}
}
dict_table_decrement_handle_count(prebuilt->table);
dict_table_decrement_handle_count(prebuilt->table, dict_locked);
/* If there were references to this table when a primary index on
this table was created then we drop it here since there are no
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment