Commit 1a122dc2 authored by Konstantin Osipov's avatar Konstantin Osipov

Partial backport of:

----------------------------------------------------
2736.2.10 Michael Widenius	2008-10-22
Fix for bug#39395 Maria: ma_extra.c:286: maria_extra: 
Assertion `share->reopen == 1' failed


sql/sql_base.cc:
  Race condition in wait_while_table_is_used() where a table used 
  by another connection could be forced closed, but there was no protection against the other thread re-opening the table and trying to lock it 
  again before the table was name locked by original thread.
parent b89c882c
...@@ -2118,13 +2118,15 @@ bool wait_while_table_is_used(THD *thd, TABLE *table, ...@@ -2118,13 +2118,15 @@ bool wait_while_table_is_used(THD *thd, TABLE *table,
enum ha_extra_function function) enum ha_extra_function function)
{ {
enum thr_lock_type old_lock_type; enum thr_lock_type old_lock_type;
DBUG_ENTER("wait_while_table_is_used"); DBUG_ENTER("wait_while_table_is_used");
DBUG_PRINT("enter", ("table: '%s' share: 0x%lx db_stat: %u version: %lu", DBUG_PRINT("enter", ("table: '%s' share: 0x%lx db_stat: %u version: %lu",
table->s->table_name.str, (ulong) table->s, table->s->table_name.str, (ulong) table->s,
table->db_stat, table->s->version)); table->db_stat, table->s->version));
(void) table->file->extra(function); /* Ensure no one can reopen table before it's removed */
pthread_mutex_lock(&LOCK_open);
table->s->version= 0;
pthread_mutex_unlock(&LOCK_open);
old_lock_type= table->reginfo.lock_type; old_lock_type= table->reginfo.lock_type;
mysql_lock_abort(thd, table, TRUE); /* end threads waiting on lock */ mysql_lock_abort(thd, table, TRUE); /* end threads waiting on lock */
...@@ -2139,6 +2141,8 @@ bool wait_while_table_is_used(THD *thd, TABLE *table, ...@@ -2139,6 +2141,8 @@ bool wait_while_table_is_used(THD *thd, TABLE *table,
tdc_remove_table(thd, TDC_RT_REMOVE_NOT_OWN, tdc_remove_table(thd, TDC_RT_REMOVE_NOT_OWN,
table->s->db.str, table->s->table_name.str); table->s->db.str, table->s->table_name.str);
pthread_mutex_unlock(&LOCK_open); pthread_mutex_unlock(&LOCK_open);
/* extra() call must come only after all instances above are closed */
(void) table->file->extra(function);
DBUG_RETURN(FALSE); DBUG_RETURN(FALSE);
} }
...@@ -2165,6 +2169,7 @@ bool wait_while_table_is_used(THD *thd, TABLE *table, ...@@ -2165,6 +2169,7 @@ bool wait_while_table_is_used(THD *thd, TABLE *table,
void drop_open_table(THD *thd, TABLE *table, const char *db_name, void drop_open_table(THD *thd, TABLE *table, const char *db_name,
const char *table_name) const char *table_name)
{ {
DBUG_ENTER("drop_open_table");
if (table->s->tmp_table) if (table->s->tmp_table)
close_temporary_table(thd, table, 1, 1); close_temporary_table(thd, table, 1, 1);
else else
...@@ -2176,10 +2181,12 @@ void drop_open_table(THD *thd, TABLE *table, const char *db_name, ...@@ -2176,10 +2181,12 @@ void drop_open_table(THD *thd, TABLE *table, const char *db_name,
table->s->version= 0; table->s->version= 0;
pthread_mutex_lock(&LOCK_open); pthread_mutex_lock(&LOCK_open);
table->file->extra(HA_EXTRA_PREPARE_FOR_DROP);
close_thread_table(thd, &thd->open_tables); close_thread_table(thd, &thd->open_tables);
quick_rm_table(table_type, db_name, table_name, 0); quick_rm_table(table_type, db_name, table_name, 0);
pthread_mutex_unlock(&LOCK_open); pthread_mutex_unlock(&LOCK_open);
} }
DBUG_VOID_RETURN;
} }
...@@ -3141,7 +3148,7 @@ Locked_tables_list::unlock_locked_tables(THD *thd) ...@@ -3141,7 +3148,7 @@ Locked_tables_list::unlock_locked_tables(THD *thd)
the locked tables list. Passing a TABLE_LIST the locked tables list. Passing a TABLE_LIST
instance that is not part of locked tables instance that is not part of locked tables
list will lead to a crash. list will lead to a crash.
@parma remove_from_locked_tables @param remove_from_locked_tables
TRUE if the table is removed from the list TRUE if the table is removed from the list
permanently. permanently.
...@@ -3443,7 +3450,6 @@ check_and_update_table_version(THD *thd, ...@@ -3443,7 +3450,6 @@ check_and_update_table_version(THD *thd,
} }
DBUG_EXECUTE_IF("reprepare_each_statement", return inject_reprepare(thd);); DBUG_EXECUTE_IF("reprepare_each_statement", return inject_reprepare(thd););
return FALSE; return FALSE;
} }
...@@ -4119,6 +4125,7 @@ bool open_tables(THD *thd, TABLE_LIST **start, uint *counter, uint flags, ...@@ -4119,6 +4125,7 @@ bool open_tables(THD *thd, TABLE_LIST **start, uint *counter, uint flags,
MEM_ROOT new_frm_mem; MEM_ROOT new_frm_mem;
bool has_prelocking_list= thd->lex->requires_prelocking(); bool has_prelocking_list= thd->lex->requires_prelocking();
DBUG_ENTER("open_tables"); DBUG_ENTER("open_tables");
/* /*
Close HANDLER tables which are marked for flush or against which there Close HANDLER tables which are marked for flush or against which there
are pending exclusive metadata locks. Note that we do this not to avoid are pending exclusive metadata locks. Note that we do this not to avoid
......
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