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. */ ...@@ -309,15 +309,21 @@ Decrements the count of open MySQL handles to a table. */
void void
dict_table_decrement_handle_count( 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); ut_a(table->n_mysql_handles_opened > 0);
table->n_mysql_handles_opened--; 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( ...@@ -2473,7 +2473,7 @@ ha_innobase::open(
my_free(upd_buff, MYF(0)); my_free(upd_buff, MYF(0));
my_errno = ENOENT; 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); DBUG_RETURN(HA_ERR_NO_SUCH_TABLE);
} }
...@@ -2574,7 +2574,7 @@ ha_innobase::close(void) ...@@ -2574,7 +2574,7 @@ ha_innobase::close(void)
innobase_release_temporary_latches(ht, thd); innobase_release_temporary_latches(ht, thd);
} }
row_prebuilt_free(prebuilt); row_prebuilt_free(prebuilt, FALSE);
my_free(upd_buff, MYF(0)); my_free(upd_buff, MYF(0));
free_share(share); free_share(share);
...@@ -5446,7 +5446,8 @@ innobase_rename_table( ...@@ -5446,7 +5446,8 @@ innobase_rename_table(
trx_t* trx, /* in: transaction */ trx_t* trx, /* in: transaction */
const char* from, /* in: old name of the table */ const char* from, /* in: old name of the table */
const char* to, /* in: new 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; int error;
char* norm_to; char* norm_to;
...@@ -5477,10 +5478,12 @@ innobase_rename_table( ...@@ -5477,10 +5478,12 @@ innobase_rename_table(
/* Serialize data dictionary operations with dictionary mutex: /* Serialize data dictionary operations with dictionary mutex:
no deadlocks can occur then in these operations */ 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( 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) { if (error != DB_SUCCESS) {
FILE* ef = dict_foreign_err_file; FILE* ef = dict_foreign_err_file;
...@@ -5492,16 +5495,15 @@ innobase_rename_table( ...@@ -5492,16 +5495,15 @@ innobase_rename_table(
fputs(" failed!\n", ef); fputs(" failed!\n", ef);
} }
row_mysql_unlock_data_dictionary(trx); if (lock_and_commit) {
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 */
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 log_buffer_flush_to_disk();
utility threads: */ }
my_free(norm_to, MYF(0)); my_free(norm_to, MYF(0));
my_free(norm_from, MYF(0)); my_free(norm_from, MYF(0));
...@@ -8414,9 +8416,9 @@ ha_innobase::add_index( ...@@ -8414,9 +8416,9 @@ ha_innobase::add_index(
Drop table etc. do this latching in row0mysql.c. */ Drop table etc. do this latching in row0mysql.c. */
row_mysql_lock_data_dictionary(trx); row_mysql_lock_data_dictionary(trx);
dict_locked = TRUE;
num_created = 0; num_created = 0;
dict_locked = TRUE;
/* Create the indexes in SYS_INDEXES and load into dictionary.*/ /* Create the indexes in SYS_INDEXES and load into dictionary.*/
...@@ -8438,48 +8440,47 @@ ha_innobase::add_index( ...@@ -8438,48 +8440,47 @@ ha_innobase::add_index(
num_created++; num_created++;
} }
if (error == DB_SUCCESS) { ut_ad(error == DB_SUCCESS);
/* Raise version number of the table to track this table's
definition changes.*/
indexed_table->version_number++; /* Raise version number of the table to track this table's
definition changes. */
row_mysql_unlock_data_dictionary(trx); indexed_table->version_number++;
dict_locked = FALSE;
mem_heap_empty(heap); row_mysql_unlock_data_dictionary(trx);
dict_locked = FALSE;
ut_a(trx->n_active_thrs == 0); mem_heap_empty(heap);
ut_a(UT_LIST_GET_LEN(trx->signals) == 0);
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 goto error_handling;
key is to be built.*/ }
if (error == DB_SUCCESS && new_primary) {
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); ut_ad(indexed_table != innodb_table);
error = row_lock_table_for_merge(trx, indexed_table, LOCK_X); 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 goto error_handling;
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);
} }
} }
/* 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 #ifdef UNIV_DEBUG
/* TODO: At the moment we can't handle the following statement /* TODO: At the moment we can't handle the following statement
in our debugging code below: in our debugging code below:
...@@ -8491,27 +8492,26 @@ ha_innobase::add_index( ...@@ -8491,27 +8492,26 @@ ha_innobase::add_index(
ignore that in the dup index check.*/ ignore that in the dup index check.*/
//dict_table_check_for_dup_indexes(prebuilt->table); //dict_table_check_for_dup_indexes(prebuilt->table);
#endif #endif
error_handling:
if (dict_locked) {
row_mysql_unlock_data_dictionary(trx);
}
/* After an error, remove all those index definitions from the /* After an error, remove all those index definitions from the
dictionary which were defined. */ dictionary which were defined. */
switch (error) { switch (error) {
case DB_SUCCESS: case DB_SUCCESS:
ut_ad(!dict_locked);
break; break;
case DB_DUPLICATE_KEY: case DB_DUPLICATE_KEY:
prebuilt->trx->error_info = NULL; prebuilt->trx->error_info = NULL;
prebuilt->trx->error_key_num = trx->error_key_num; prebuilt->trx->error_key_num = trx->error_key_num;
/* fall through */ /* fall through */
default: default:
row_merge_drop_indexes( if (new_primary) {
trx, indexed_table, index, num_created);
if (indexed_table != innodb_table) {
row_merge_drop_table(trx, indexed_table); row_merge_drop_table(trx, indexed_table);
} else {
row_merge_drop_indexes(trx, indexed_table,
index, num_created);
} }
goto func_exit; goto func_exit;
} }
...@@ -8528,6 +8528,11 @@ ha_innobase::add_index( ...@@ -8528,6 +8528,11 @@ ha_innobase::add_index(
trx_start_if_not_started(trx); 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 */ /* Write entry for UNDO */
error = row_undo_report_rename_table_dict_operation( error = row_undo_report_rename_table_dict_operation(
trx, old_name, indexed_table->name, tmp_table_name); trx, old_name, indexed_table->name, tmp_table_name);
...@@ -8555,24 +8560,26 @@ ha_innobase::add_index( ...@@ -8555,24 +8560,26 @@ ha_innobase::add_index(
goto func_exit; goto func_exit;
} }
row_prebuilt_free(prebuilt); row_prebuilt_free(prebuilt, TRUE);
prebuilt = row_create_prebuilt(indexed_table); prebuilt = row_create_prebuilt(indexed_table);
row_mysql_lock_data_dictionary(trx);
prebuilt->table->n_mysql_handles_opened++; prebuilt->table->n_mysql_handles_opened++;
row_mysql_unlock_data_dictionary(trx);
/* Drop the old table if there are no open views /* Drop the old table if there are no open views
referring to it. If there such views, we will drop referring to it. If there are such views, we will
the table when we free the prebuilts and there are no drop the table when we free the prebuilts and there
more references to it. */ are no more references to it. */
error = row_merge_drop_table(trx, innodb_table); error = row_merge_drop_table(trx, innodb_table);
} }
func_exit: func_exit:
mem_heap_free(heap); 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.*/ /* There might be work for utility threads.*/
srv_active_wake_master_thread(); srv_active_wake_master_thread();
......
...@@ -68,7 +68,8 @@ Decrements the count of open MySQL handles to a table. */ ...@@ -68,7 +68,8 @@ Decrements the count of open MySQL handles to a table. */
void void
dict_table_decrement_handle_count( 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. */ Inits the data dictionary module. */
......
...@@ -141,7 +141,8 @@ Free a prebuilt struct for a MySQL table handle. */ ...@@ -141,7 +141,8 @@ Free a prebuilt struct for a MySQL table handle. */
void void
row_prebuilt_free( 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 Updates the transaction pointers in query graphs stored in the prebuilt
struct. */ struct. */
......
...@@ -668,7 +668,8 @@ Free a prebuilt struct for a MySQL table handle. */ ...@@ -668,7 +668,8 @@ Free a prebuilt struct for a MySQL table handle. */
void void
row_prebuilt_free( 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; ulint i;
...@@ -742,7 +743,7 @@ row_prebuilt_free( ...@@ -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 /* 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 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