Commit f99f573d authored by Jan Lindström's avatar Jan Lindström

MDEV-6656: Test wsrep.variables hangs

Analysis: wsrep_applier_thread shutdown signaling does not always work
correctly causing a timing problem where main thread is waiting in a
condition variable a signal that all worker threads to end.
parent 4fb45aa2
...@@ -195,9 +195,6 @@ SET GLOBAL wsrep_slave_threads= 10; ...@@ -195,9 +195,6 @@ SET GLOBAL wsrep_slave_threads= 10;
SHOW STATUS LIKE 'threads_connected'; SHOW STATUS LIKE 'threads_connected';
Variable_name Value Variable_name Value
Threads_connected 1 Threads_connected 1
SHOW STATUS LIKE 'wsrep_thread_count';
Variable_name Value
wsrep_thread_count 11
SET GLOBAL wsrep_slave_threads= @wsrep_slave_threads_saved; SET GLOBAL wsrep_slave_threads= @wsrep_slave_threads_saved;
SET GLOBAL wsrep_provider= none; SET GLOBAL wsrep_provider= none;
SET GLOBAL wsrep_cluster_address= ''; SET GLOBAL wsrep_cluster_address= '';
......
...@@ -120,9 +120,8 @@ SHOW STATUS LIKE 'wsrep_thread_count'; ...@@ -120,9 +120,8 @@ SHOW STATUS LIKE 'wsrep_thread_count';
SET @wsrep_slave_threads_saved= @@global.wsrep_slave_threads; SET @wsrep_slave_threads_saved= @@global.wsrep_slave_threads;
SET GLOBAL wsrep_slave_threads= 10; SET GLOBAL wsrep_slave_threads= 10;
--echo # Wait for applier threads to get created. --echo # Wait for applier threads to get created.
sleep 3; sleep 5;
SHOW STATUS LIKE 'threads_connected'; SHOW STATUS LIKE 'threads_connected';
SHOW STATUS LIKE 'wsrep_thread_count';
# reset (for mtr internal checks) # reset (for mtr internal checks)
SET GLOBAL wsrep_slave_threads= @wsrep_slave_threads_saved; SET GLOBAL wsrep_slave_threads= @wsrep_slave_threads_saved;
......
...@@ -5437,7 +5437,7 @@ static bool abort_replicated(THD *thd) ...@@ -5437,7 +5437,7 @@ static bool abort_replicated(THD *thd)
bool ret_code= false; bool ret_code= false;
if (thd->wsrep_query_state== QUERY_COMMITTING) if (thd->wsrep_query_state== QUERY_COMMITTING)
{ {
if (wsrep_debug) WSREP_INFO("aborting replicated trx: %lu", thd->real_id); WSREP_DEBUG("aborting replicated trx: %lu", thd->real_id);
(void)wsrep_abort_thd(thd, thd, TRUE); (void)wsrep_abort_thd(thd, thd, TRUE);
ret_code= true; ret_code= true;
...@@ -5507,23 +5507,6 @@ static bool have_client_connections() ...@@ -5507,23 +5507,6 @@ static bool have_client_connections()
return false; return false;
} }
/*
returns the number of wsrep appliers running.
However, the caller (thd parameter) is not taken in account
*/
static int have_wsrep_appliers(THD *thd)
{
int ret= 0;
THD *tmp;
I_List_iterator<THD> it(threads);
while ((tmp=it++))
{
ret+= (tmp != thd && tmp->wsrep_applier);
}
return ret;
}
static void wsrep_close_thread(THD *thd) static void wsrep_close_thread(THD *thd)
{ {
thd->killed= KILL_CONNECTION; thd->killed= KILL_CONNECTION;
...@@ -5639,8 +5622,7 @@ void wsrep_close_client_connections(my_bool wait_to_end) ...@@ -5639,8 +5622,7 @@ void wsrep_close_client_connections(my_bool wait_to_end)
} }
DBUG_PRINT("quit",("Waiting for threads to die (count=%u)",thread_count)); DBUG_PRINT("quit",("Waiting for threads to die (count=%u)",thread_count));
if (wsrep_debug) WSREP_DEBUG("waiting for client connections to close: %u", thread_count);
WSREP_INFO("waiting for client connections to close: %u", thread_count);
while (wait_to_end && have_client_connections()) while (wait_to_end && have_client_connections())
{ {
...@@ -5682,33 +5664,11 @@ static void wsrep_close_threads(THD *thd) ...@@ -5682,33 +5664,11 @@ static void wsrep_close_threads(THD *thd)
mysql_mutex_unlock(&LOCK_thread_count); mysql_mutex_unlock(&LOCK_thread_count);
} }
void wsrep_close_applier_threads(int count)
{
THD *tmp;
mysql_mutex_lock(&LOCK_thread_count); // For unlink from list
I_List_iterator<THD> it(threads);
while ((tmp=it++) && count)
{
DBUG_PRINT("quit",("Informing thread %ld that it's time to die",
tmp->thread_id));
/* We skip slave threads & scheduler on this first loop through. */
if (tmp->wsrep_applier)
{
WSREP_DEBUG("closing wsrep applier thread %ld", tmp->thread_id);
tmp->wsrep_applier_closing= TRUE;
count--;
}
}
mysql_mutex_unlock(&LOCK_thread_count);
}
void wsrep_wait_appliers_close(THD *thd) void wsrep_wait_appliers_close(THD *thd)
{ {
/* Wait for wsrep appliers to gracefully exit */ /* Wait for wsrep appliers to gracefully exit */
mysql_mutex_lock(&LOCK_thread_count); mysql_mutex_lock(&LOCK_thread_count);
while (have_wsrep_appliers(thd) > 1) while (wsrep_running_threads > 1)
// 1 is for rollbacker thread which needs to be killed explicitly. // 1 is for rollbacker thread which needs to be killed explicitly.
// This gotta be fixed in a more elegant manner if we gonna have arbitrary // This gotta be fixed in a more elegant manner if we gonna have arbitrary
// number of non-applier wsrep threads. // number of non-applier wsrep threads.
...@@ -5728,7 +5688,7 @@ void wsrep_wait_appliers_close(THD *thd) ...@@ -5728,7 +5688,7 @@ void wsrep_wait_appliers_close(THD *thd)
wsrep_close_threads (thd); wsrep_close_threads (thd);
/* and wait for them to die */ /* and wait for them to die */
mysql_mutex_lock(&LOCK_thread_count); mysql_mutex_lock(&LOCK_thread_count);
while (have_wsrep_appliers(thd) > 0) while (wsrep_running_threads > 0)
{ {
if (thread_handling > SCHEDULER_ONE_THREAD_PER_CONNECTION) if (thread_handling > SCHEDULER_ONE_THREAD_PER_CONNECTION)
{ {
......
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