summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sql/sql_base.cc4
-rw-r--r--sql/sql_truncate.cc11
-rw-r--r--sql/table_cache.cc27
-rw-r--r--sql/table_cache.h1
4 files changed, 22 insertions, 21 deletions
diff --git a/sql/sql_base.cc b/sql/sql_base.cc
index 664071a27f5..dfff349eb76 100644
--- a/sql/sql_base.cc
+++ b/sql/sql_base.cc
@@ -2942,9 +2942,7 @@ static bool auto_repair_table(THD *thd, TABLE_LIST *table_list)
result= FALSE;
}
- tdc_release_share(share);
- /* Remove the repaired share from the table cache. */
- tdc_remove_table(thd, table_list->db.str, table_list->table_name.str);
+ tdc_remove_referenced_share(thd, share);
end_free:
my_free(entry);
return result;
diff --git a/sql/sql_truncate.cc b/sql/sql_truncate.cc
index 1d2ec66eae6..a745293ec31 100644
--- a/sql/sql_truncate.cc
+++ b/sql/sql_truncate.cc
@@ -336,7 +336,10 @@ bool Sql_cmd_truncate_table::lock_table(THD *thd, TABLE_LIST *table_ref,
}
#endif
- tdc_release_share(share);
+ if (!versioned)
+ tdc_remove_referenced_share(thd, share);
+ else
+ tdc_release_share(share);
if (hton == view_pseudo_hton)
{
@@ -372,12 +375,6 @@ bool Sql_cmd_truncate_table::lock_table(THD *thd, TABLE_LIST *table_ref,
if (*hton_can_recreate)
close_all_tables_for_name(thd, table->s, HA_EXTRA_NOT_USED, NULL);
}
- else
- {
- /* Table is already locked exclusively. Remove cached instances. */
- tdc_remove_table(thd, table_ref->db.str, table_ref->table_name.str);
- }
-
DBUG_RETURN(FALSE);
}
diff --git a/sql/table_cache.cc b/sql/table_cache.cc
index 82c60359e88..62ccfba7e7d 100644
--- a/sql/table_cache.cc
+++ b/sql/table_cache.cc
@@ -996,6 +996,20 @@ void tdc_release_share(TABLE_SHARE *share)
}
+void tdc_remove_referenced_share(THD *thd, TABLE_SHARE *share)
+{
+ DBUG_ASSERT(thd->mdl_context.is_lock_owner(MDL_key::TABLE, share->db.str,
+ share->table_name.str,
+ MDL_EXCLUSIVE));
+ share->tdc->flush_unused(false);
+ mysql_mutex_lock(&share->tdc->LOCK_table_share);
+ share->tdc->wait_for_refs(1);
+ DBUG_ASSERT(share->tdc->all_tables.is_empty());
+ share->tdc->ref_count--;
+ tdc_delete_share_from_hash(share->tdc);
+}
+
+
/**
Removes all TABLE instances and corresponding TABLE_SHARE
@@ -1009,7 +1023,6 @@ void tdc_release_share(TABLE_SHARE *share)
void tdc_remove_table(THD *thd, const char *db, const char *table_name)
{
- Share_free_tables::List purge_tables;
TDC_element *element;
DBUG_ENTER("tdc_remove_table");
DBUG_PRINT("enter", ("name: %s", table_name));
@@ -1042,18 +1055,10 @@ void tdc_remove_table(THD *thd, const char *db, const char *table_name)
mysql_mutex_unlock(&LOCK_unused_shares);
element->ref_count++;
-
- tc_remove_all_unused_tables(element, &purge_tables);
mysql_mutex_unlock(&element->LOCK_table_share);
- while (auto table= purge_tables.pop_front())
- intern_close_table(table);
-
- mysql_mutex_lock(&element->LOCK_table_share);
- element->wait_for_refs(1);
- DBUG_ASSERT(element->all_tables.is_empty());
- element->ref_count--;
- tdc_delete_share_from_hash(element);
+ /* We have to relock the mutex to avoid code duplication. Sigh. */
+ tdc_remove_referenced_share(thd, element->share);
DBUG_VOID_RETURN;
}
diff --git a/sql/table_cache.h b/sql/table_cache.h
index bffb782849c..433df5e0328 100644
--- a/sql/table_cache.h
+++ b/sql/table_cache.h
@@ -77,6 +77,7 @@ int tdc_share_is_cached(THD *thd, const char *db, const char *table_name);
extern TABLE_SHARE *tdc_acquire_share(THD *thd, TABLE_LIST *tl, uint flags,
TABLE **out_table= 0);
extern void tdc_release_share(TABLE_SHARE *share);
+void tdc_remove_referenced_share(THD *thd, TABLE_SHARE *share);
void tdc_remove_table(THD *thd, const char *db, const char *table_name);
extern int tdc_wait_for_old_version(THD *thd, const char *db,