Commit 2d7c0738 authored by Sergey Vojtovich's avatar Sergey Vojtovich

MDEV-5388 - Reduce usage of LOCK_open: unused_tables

Removed unused_tables, find LRU object by timestamp instead.
parent 98c4f167
...@@ -1026,7 +1026,7 @@ void close_thread_table(THD *thd, TABLE **table_ptr) ...@@ -1026,7 +1026,7 @@ void close_thread_table(THD *thd, TABLE **table_ptr)
if (! table->needs_reopen()) if (! table->needs_reopen())
{ {
/* Avoid having MERGE tables with attached children in unused_tables. */ /* Avoid having MERGE tables with attached children in table cache. */
table->file->extra(HA_EXTRA_DETACH_CHILDREN); table->file->extra(HA_EXTRA_DETACH_CHILDREN);
/* Free memory and reset for next loop. */ /* Free memory and reset for next loop. */
free_field_buffers_larger_than(table, MAX_TDC_BLOB_SIZE); free_field_buffers_larger_than(table, MAX_TDC_BLOB_SIZE);
......
...@@ -20,7 +20,7 @@ ...@@ -20,7 +20,7 @@
#include "sql_priv.h" #include "sql_priv.h"
#include "unireg.h" #include "unireg.h"
#include "sql_test.h" #include "sql_test.h"
#include "sql_base.h" // unused_tables #include "sql_base.h"
#include "sql_show.h" // calc_sum_of_all_status #include "sql_show.h" // calc_sum_of_all_status
#include "sql_select.h" #include "sql_select.h"
#include "keycaches.h" #include "keycaches.h"
...@@ -78,9 +78,8 @@ print_where(COND *cond,const char *info, enum_query_type query_type) ...@@ -78,9 +78,8 @@ print_where(COND *cond,const char *info, enum_query_type query_type)
static void print_cached_tables(void) static void print_cached_tables(void)
{ {
uint count= 0, unused= 0;
TABLE_SHARE *share; TABLE_SHARE *share;
TABLE *start_link, *lnk, *entry; TABLE *entry;
TDC_iterator tdc_it; TDC_iterator tdc_it;
compile_time_assert(TL_WRITE_ONLY+1 == array_elements(lock_descriptions)); compile_time_assert(TL_WRITE_ONLY+1 == array_elements(lock_descriptions));
...@@ -95,42 +94,16 @@ static void print_cached_tables(void) ...@@ -95,42 +94,16 @@ static void print_cached_tables(void)
TABLE_SHARE::All_share_tables_list::Iterator it(share->tdc.all_tables); TABLE_SHARE::All_share_tables_list::Iterator it(share->tdc.all_tables);
while ((entry= it++)) while ((entry= it++))
{ {
if (entry->in_use) printf("%-14.14s %-32s%6ld%8ld%6d %s\n",
{ entry->s->db.str, entry->s->table_name.str, entry->s->version,
printf("%-14.14s %-32s%6ld%8ld%6d %s\n", entry->in_use ? entry->in_use->thread_id : 0,
entry->s->db.str, entry->s->table_name.str, entry->s->version, entry->db_stat ? 1 : 0,
entry->in_use->thread_id, entry->db_stat ? 1 : 0, entry->in_use ? lock_descriptions[(int)entry->reginfo.lock_type] :
lock_descriptions[(int)entry->reginfo.lock_type]); "Not in use");
}
else
{
unused++;
printf("%-14.14s %-32s%6ld%8ld%6d %s\n",
entry->s->db.str, entry->s->table_name.str, entry->s->version,
0L, entry->db_stat ? 1 : 0, "Not in use");
}
}
}
tdc_it.deinit();
if ((start_link=lnk=unused_tables))
{
do
{
if (lnk != lnk->next->prev || lnk != lnk->prev->next)
{
printf("unused_links isn't linked properly\n");
return;
}
} while (count++ < tc_records() && (lnk=lnk->next) != start_link);
if (lnk != start_link)
{
printf("Unused_links aren't connected\n");
} }
} }
mysql_mutex_unlock(&LOCK_open); mysql_mutex_unlock(&LOCK_open);
if (count != unused) tdc_it.deinit();
printf("Unused_links (%d) doesn't match table_def_cache: %d\n", count,
unused);
printf("\nCurrent refresh version: %ld\n", tdc_refresh_version()); printf("\nCurrent refresh version: %ld\n", tdc_refresh_version());
fflush(stdout); fflush(stdout);
/* purecov: end */ /* purecov: end */
......
...@@ -1011,19 +1011,18 @@ struct TABLE ...@@ -1011,19 +1011,18 @@ struct TABLE
private: private:
/** /**
Links for the lists of used/unused TABLE objects for this share. Links for the list of all TABLE objects for this share.
Declared as private to avoid direct manipulation with those objects. Declared as private to avoid direct manipulation with those objects.
One should use methods of I_P_List template instead. One should use methods of I_P_List template instead.
*/ */
TABLE *share_next, **share_prev;
TABLE *share_all_next, **share_all_prev; TABLE *share_all_next, **share_all_prev;
friend struct TABLE_share;
friend struct All_share_tables; friend struct All_share_tables;
public: public:
THD *in_use; /* Which thread uses this */ THD *in_use; /* Which thread uses this */
/* Time when table was released to table cache. Valid for unused tables. */
ulonglong tc_time;
Field **field; /* Pointer to fields */ Field **field; /* Pointer to fields */
uchar *record[2]; /* Pointer to records */ uchar *record[2]; /* Pointer to records */
...@@ -1374,11 +1373,11 @@ struct TABLE_share ...@@ -1374,11 +1373,11 @@ struct TABLE_share
{ {
static inline TABLE **next_ptr(TABLE *l) static inline TABLE **next_ptr(TABLE *l)
{ {
return &l->share_next; return &l->next;
} }
static inline TABLE ***prev_ptr(TABLE *l) static inline TABLE ***prev_ptr(TABLE *l)
{ {
return &l->share_prev; return (TABLE ***) &l->prev;
} }
}; };
......
...@@ -44,7 +44,6 @@ ...@@ -44,7 +44,6 @@
Table cache invariants: Table cache invariants:
- TABLE_SHARE::free_tables shall not contain objects with TABLE::in_use != 0 - TABLE_SHARE::free_tables shall not contain objects with TABLE::in_use != 0
- unused_tables shall not contain objects with TABLE::in_use != 0
*/ */
#include "my_global.h" #include "my_global.h"
...@@ -60,7 +59,6 @@ ulong tc_size; /**< Table cache threshold for LRU eviction. */ ...@@ -60,7 +59,6 @@ ulong tc_size; /**< Table cache threshold for LRU eviction. */
static HASH tdc_hash; /**< Collection of TABLE_SHARE objects. */ static HASH tdc_hash; /**< Collection of TABLE_SHARE objects. */
/** Collection of unused TABLE_SHARE objects. */ /** Collection of unused TABLE_SHARE objects. */
static TABLE_SHARE *oldest_unused_share, end_of_unused_share; static TABLE_SHARE *oldest_unused_share, end_of_unused_share;
TABLE *unused_tables; /**< Collection of unused TABLE objects. */
static int64 tdc_version; /* Increments on each reload */ static int64 tdc_version; /* Increments on each reload */
static int64 last_table_id; static int64 last_table_id;
...@@ -70,15 +68,8 @@ static uint tc_count; /**< Number of TABLE objects in table cache. */ ...@@ -70,15 +68,8 @@ static uint tc_count; /**< Number of TABLE objects in table cache. */
/** /**
Protects used and unused lists in the TABLE_SHARE object, Protects tc_count, TABLE_SHARE::tdc.free_tables, TABLE_SHARE::tdc.all_tables,
LRU lists of used TABLEs. TABLE::in_use.
tc_count
unused_tables
TABLE::next
TABLE::prev
TABLE_SHARE::tdc.free_tables
TABLE_SHARE::tdc.all_tables
*/ */
mysql_mutex_t LOCK_open; mysql_mutex_t LOCK_open;
...@@ -158,7 +149,6 @@ uint tc_records(void) ...@@ -158,7 +149,6 @@ uint tc_records(void)
While locked: While locked:
- remove unused objects from TABLE_SHARE::tdc.free_tables and - remove unused objects from TABLE_SHARE::tdc.free_tables and
TABLE_SHARE::tdc.all_tables TABLE_SHARE::tdc.all_tables
- reset unused_tables
- decrement tc_count - decrement tc_count
While unlocked: While unlocked:
...@@ -170,29 +160,29 @@ uint tc_records(void) ...@@ -170,29 +160,29 @@ uint tc_records(void)
void tc_purge(void) void tc_purge(void)
{ {
TABLE_SHARE *share;
TABLE *table;
TDC_iterator tdc_it;
TABLE_SHARE::TABLE_list purge_tables;
tdc_it.init();
mysql_mutex_lock(&LOCK_open); mysql_mutex_lock(&LOCK_open);
if (unused_tables) while ((share= tdc_it.next()))
{ {
TABLE *table= unused_tables, *next; while ((table= share->tdc.free_tables.pop_front()))
unused_tables->prev->next= 0;
do
{ {
unused_tables->s->tdc.free_tables.remove(unused_tables); share->tdc.all_tables.remove(table);
unused_tables->s->tdc.all_tables.remove(unused_tables); purge_tables.push_front(table);
tc_count--; tc_count--;
} while ((unused_tables= unused_tables->next)); }
mysql_rwlock_rdlock(&LOCK_flush);
mysql_mutex_unlock(&LOCK_open);
do
{
next= table->next;
intern_close_table(table);
} while ((table= next));
mysql_rwlock_unlock(&LOCK_flush);
} }
else tdc_it.deinit();
mysql_mutex_unlock(&LOCK_open); mysql_rwlock_rdlock(&LOCK_flush);
mysql_mutex_unlock(&LOCK_open);
while ((table= purge_tables.pop_front()))
intern_close_table(table);
mysql_rwlock_unlock(&LOCK_flush);
} }
...@@ -203,29 +193,12 @@ void tc_purge(void) ...@@ -203,29 +193,12 @@ void tc_purge(void)
#ifdef EXTRA_DEBUG #ifdef EXTRA_DEBUG
static void check_unused(THD *thd) static void check_unused(THD *thd)
{ {
uint count= 0, open_files= 0; TABLE *entry;
TABLE *cur_link, *start_link, *entry;
TABLE_SHARE *share; TABLE_SHARE *share;
TDC_iterator tdc_it; TDC_iterator tdc_it;
tdc_it.init(); tdc_it.init();
mysql_mutex_lock(&LOCK_open); mysql_mutex_lock(&LOCK_open);
if ((start_link=cur_link=unused_tables))
{
do
{
if (cur_link != cur_link->next->prev || cur_link != cur_link->prev->next)
{
DBUG_PRINT("error",("Unused_links aren't linked properly")); /* purecov: inspected */
return; /* purecov: inspected */
}
} while (count++ < tc_count &&
(cur_link=cur_link->next) != start_link);
if (cur_link != start_link)
{
DBUG_PRINT("error",("Unused_links aren't connected")); /* purecov: inspected */
}
}
while ((share= tdc_it.next())) while ((share= tdc_it.next()))
{ {
TABLE_SHARE::TABLE_list::Iterator it(share->tdc.free_tables); TABLE_SHARE::TABLE_list::Iterator it(share->tdc.free_tables);
...@@ -244,65 +217,16 @@ static void check_unused(THD *thd) ...@@ -244,65 +217,16 @@ static void check_unused(THD *thd)
entry->in_use= thd; entry->in_use= thd;
DBUG_ASSERT(!thd || !entry->file->extra(HA_EXTRA_IS_ATTACHED_CHILDREN)); DBUG_ASSERT(!thd || !entry->file->extra(HA_EXTRA_IS_ATTACHED_CHILDREN));
entry->in_use= 0; entry->in_use= 0;
count--;
open_files++;
}
TABLE_SHARE::All_share_tables_list::Iterator it2(share->tdc.all_tables);
while ((entry= it2++))
{
if (entry->in_use)
open_files++;
} }
} }
mysql_mutex_unlock(&LOCK_open); mysql_mutex_unlock(&LOCK_open);
tdc_it.deinit(); tdc_it.deinit();
if (count != 0)
{
DBUG_PRINT("error",("Unused_links doesn't match open_cache: diff: %d", /* purecov: inspected */
count)); /* purecov: inspected */
}
} }
#else #else
#define check_unused(A) #define check_unused(A)
#endif #endif
/**
Remove unused TABLE object from table cache.
@pre LOCK_open is locked, table is not used.
While locked:
- remove object from TABLE_SHARE::tdc.free_tables and
TABLE_SHARE::tdc.all_tables
- remove object from unused_tables
@note This is helper routine, supposed to be used by table cache
methods only.
*/
static void tc_remove_table(TABLE *table)
{
mysql_mutex_assert_owner(&LOCK_open);
DBUG_ASSERT(!table->in_use);
/* Remove from per-share chain of unused TABLE objects. */
table->s->tdc.free_tables.remove(table);
table->s->tdc.all_tables.remove(table);
/* And global unused chain. */
table->next->prev= table->prev;
table->prev->next= table->next;
if (table == unused_tables)
{
unused_tables= unused_tables->next;
if (table == unused_tables)
unused_tables= 0;
}
tc_count--;
}
/** /**
Add new TABLE object to table cache. Add new TABLE object to table cache.
...@@ -324,20 +248,44 @@ void tc_add_table(THD *thd, TABLE *table) ...@@ -324,20 +248,44 @@ void tc_add_table(THD *thd, TABLE *table)
DBUG_ASSERT(table->in_use == thd); DBUG_ASSERT(table->in_use == thd);
mysql_mutex_lock(&LOCK_open); mysql_mutex_lock(&LOCK_open);
table->s->tdc.all_tables.push_front(table); table->s->tdc.all_tables.push_front(table);
tc_count++;
/* If we have too many TABLE instances around, try to get rid of them */ /* If we have too many TABLE instances around, try to get rid of them */
if (tc_count > tc_size && unused_tables) if (tc_count == tc_size)
{ {
TABLE *purge_table= unused_tables; TDC_iterator tdc_it;
tc_remove_table(purge_table);
mysql_rwlock_rdlock(&LOCK_flush);
mysql_mutex_unlock(&LOCK_open); mysql_mutex_unlock(&LOCK_open);
intern_close_table(purge_table);
mysql_rwlock_unlock(&LOCK_flush); tdc_it.init();
check_unused(thd); mysql_mutex_lock(&LOCK_open);
if (tc_count == tc_size)
{
TABLE *purge_table= 0;
TABLE_SHARE *share;
while ((share= tdc_it.next()))
{
TABLE_SHARE::TABLE_list::Iterator it(share->tdc.free_tables);
TABLE *entry;
while ((entry= it++))
if (!purge_table || entry->tc_time < purge_table->tc_time)
purge_table= entry;
}
if (purge_table)
{
tdc_it.deinit();
purge_table->s->tdc.free_tables.remove(purge_table);
purge_table->s->tdc.all_tables.remove(purge_table);
mysql_rwlock_rdlock(&LOCK_flush);
mysql_mutex_unlock(&LOCK_open);
intern_close_table(purge_table);
mysql_rwlock_unlock(&LOCK_flush);
check_unused(thd);
return;
}
}
tdc_it.deinit();
} }
else /* Nothing to evict, increment tc_count. */
mysql_mutex_unlock(&LOCK_open); tc_count++;
mysql_mutex_unlock(&LOCK_open);
} }
...@@ -350,7 +298,6 @@ void tc_add_table(THD *thd, TABLE *table) ...@@ -350,7 +298,6 @@ void tc_add_table(THD *thd, TABLE *table)
While locked: While locked:
- pop object from TABLE_SHARE::tdc.free_tables() - pop object from TABLE_SHARE::tdc.free_tables()
- remove object from unused_tables
- mark object used by thd - mark object used by thd
@return TABLE object, or NULL if no unused objects. @return TABLE object, or NULL if no unused objects.
...@@ -367,16 +314,6 @@ static TABLE *tc_acquire_table(THD *thd, TABLE_SHARE *share) ...@@ -367,16 +314,6 @@ static TABLE *tc_acquire_table(THD *thd, TABLE_SHARE *share)
return 0; return 0;
} }
DBUG_ASSERT(!table->in_use); DBUG_ASSERT(!table->in_use);
/* Unlink table from global unused tables list. */
if (table == unused_tables)
{ // First unused
unused_tables=unused_tables->next; // Remove from link
if (table == unused_tables)
unused_tables=0;
}
table->prev->next=table->next; /* Remove from unused list */
table->next->prev=table->prev;
table->in_use= thd; table->in_use= thd;
mysql_mutex_unlock(&LOCK_open); mysql_mutex_unlock(&LOCK_open);
...@@ -399,7 +336,6 @@ static TABLE *tc_acquire_table(THD *thd, TABLE_SHARE *share) ...@@ -399,7 +336,6 @@ static TABLE *tc_acquire_table(THD *thd, TABLE_SHARE *share)
- mark object not in use by any thread - mark object not in use by any thread
- if object is marked for purge, decrement tc_count - if object is marked for purge, decrement tc_count
- add object to TABLE_SHARE::tdc.free_tables - add object to TABLE_SHARE::tdc.free_tables
- add object to unused_tables
- evict LRU object from table cache if we reached threshold - evict LRU object from table cache if we reached threshold
While unlocked: While unlocked:
...@@ -421,10 +357,11 @@ bool tc_release_table(TABLE *table) ...@@ -421,10 +357,11 @@ bool tc_release_table(TABLE *table)
DBUG_ASSERT(table->in_use); DBUG_ASSERT(table->in_use);
DBUG_ASSERT(table->file); DBUG_ASSERT(table->file);
table->tc_time= my_interval_timer();
mysql_mutex_lock(&LOCK_open); mysql_mutex_lock(&LOCK_open);
table->in_use= 0; table->in_use= 0;
if (table->s->has_old_version() || table->needs_reopen() || !tdc_size || if (table->s->has_old_version() || table->needs_reopen() || tc_count > tc_size)
tc_count > tc_size)
{ {
tc_count--; tc_count--;
table->s->tdc.all_tables.remove(table); table->s->tdc.all_tables.remove(table);
...@@ -436,16 +373,6 @@ bool tc_release_table(TABLE *table) ...@@ -436,16 +373,6 @@ bool tc_release_table(TABLE *table)
} }
/* Add table to the list of unused TABLE objects for this share. */ /* Add table to the list of unused TABLE objects for this share. */
table->s->tdc.free_tables.push_front(table); table->s->tdc.free_tables.push_front(table);
/* Also link it last in the global list of unused TABLE objects. */
if (unused_tables)
{
table->next=unused_tables;
table->prev=unused_tables->prev;
unused_tables->prev=table;
table->prev->next=table;
}
else
unused_tables=table->next=table->prev=table;
mysql_mutex_unlock(&LOCK_open); mysql_mutex_unlock(&LOCK_open);
check_unused(thd); check_unused(thd);
return false; return false;
...@@ -563,6 +490,7 @@ void tdc_start_shutdown(void) ...@@ -563,6 +490,7 @@ void tdc_start_shutdown(void)
plugins minimal and allows shutdown to proceed smoothly. plugins minimal and allows shutdown to proceed smoothly.
*/ */
tdc_size= 0; tdc_size= 0;
tc_size= 0;
/* Free all cached but unused TABLEs and TABLE_SHAREs. */ /* Free all cached but unused TABLEs and TABLE_SHAREs. */
close_cached_tables(NULL, NULL, FALSE, LONG_TIMEOUT); close_cached_tables(NULL, NULL, FALSE, LONG_TIMEOUT);
} }
...@@ -1031,13 +959,11 @@ bool tdc_remove_table(THD *thd, enum_tdc_remove_table_type remove_type, ...@@ -1031,13 +959,11 @@ bool tdc_remove_table(THD *thd, enum_tdc_remove_table_type remove_type,
if ((share= tdc_delete_share(db, table_name))) if ((share= tdc_delete_share(db, table_name)))
{ {
I_P_List <TABLE, TABLE_share> purge_tables; I_P_List <TABLE, TABLE_share> purge_tables;
purge_tables.empty();
mysql_mutex_lock(&LOCK_open); mysql_mutex_lock(&LOCK_open);
if (kill_delayed_threads) if (kill_delayed_threads)
kill_delayed_threads_for_table(share); kill_delayed_threads_for_table(share);
TABLE_SHARE::TABLE_list::Iterator it(share->tdc.free_tables);
#ifndef DBUG_OFF #ifndef DBUG_OFF
if (remove_type == TDC_RT_REMOVE_NOT_OWN) if (remove_type == TDC_RT_REMOVE_NOT_OWN)
{ {
...@@ -1057,9 +983,10 @@ bool tdc_remove_table(THD *thd, enum_tdc_remove_table_type remove_type, ...@@ -1057,9 +983,10 @@ bool tdc_remove_table(THD *thd, enum_tdc_remove_table_type remove_type,
if (remove_type != TDC_RT_REMOVE_NOT_OWN_KEEP_SHARE) if (remove_type != TDC_RT_REMOVE_NOT_OWN_KEEP_SHARE)
share->version= 0; share->version= 0;
while ((table= it++)) while ((table= share->tdc.free_tables.pop_front()))
{ {
tc_remove_table(table); share->tdc.all_tables.remove(table);
tc_count--;
purge_tables.push_front(table); purge_tables.push_front(table);
} }
mysql_rwlock_rdlock(&LOCK_flush); mysql_rwlock_rdlock(&LOCK_flush);
......
...@@ -26,7 +26,6 @@ enum enum_tdc_remove_table_type ...@@ -26,7 +26,6 @@ enum enum_tdc_remove_table_type
extern ulong tdc_size; extern ulong tdc_size;
extern ulong tc_size; extern ulong tc_size;
extern TABLE *unused_tables; /* FIXME: make private */
extern mysql_mutex_t LOCK_open; /* FIXME: make private */ extern mysql_mutex_t LOCK_open; /* FIXME: make private */
extern int tdc_init(void); extern int tdc_init(void);
......
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