Commit 5cec8347 authored by Marko Mäkelä's avatar Marko Mäkelä

MDEV-29896: mariadb-backup --backup --incremental --throttle=... hangs

io_watching_thread(): Declare as a detachable thread, similar to
log_copying_thread().

stop_backup_threads(): Wait for both log_copying_thread and
io_watching_thread to clear their flags. Expect log_sys.mutex
to be held by the caller.

xtrabackup_backup_func(): Initialize log_copying_stop before
creating io_watching_thread. This prevents a race condition
where io_watching_thread() could wait on the condition variable
before it had been fully initialized. This race condition would
cause a hang in the GNU libc implementation of pthread_cond_destroy()
at the end of stop_backup_threads().

This race condition was introduced in
commit 38fd7b7d (MDEV-21452).
parent 5eb545e5
...@@ -190,6 +190,7 @@ struct xb_filter_entry_t{ ...@@ -190,6 +190,7 @@ struct xb_filter_entry_t{
lsn_t checkpoint_lsn_start; lsn_t checkpoint_lsn_start;
lsn_t checkpoint_no_start; lsn_t checkpoint_no_start;
static lsn_t log_copy_scanned_lsn; static lsn_t log_copy_scanned_lsn;
/** whether log_copying_thread() is active; protected by log_sys.mutex */
static bool log_copying_running; static bool log_copying_running;
int xtrabackup_parallel; int xtrabackup_parallel;
...@@ -3092,16 +3093,18 @@ static void log_copying_thread() ...@@ -3092,16 +3093,18 @@ static void log_copying_thread()
my_thread_end(); my_thread_end();
} }
/** whether io_watching_thread() is active; protected by log_sys.mutex */
static bool have_io_watching_thread; static bool have_io_watching_thread;
static pthread_t io_watching_thread_id;
/* io throttle watching (rough) */ /* io throttle watching (rough) */
static void *io_watching_thread(void*) static void io_watching_thread()
{ {
my_thread_init();
/* currently, for --backup only */ /* currently, for --backup only */
ut_a(xtrabackup_backup); ut_ad(xtrabackup_backup);
mysql_mutex_lock(&log_sys.mutex); mysql_mutex_lock(&log_sys.mutex);
ut_ad(have_io_watching_thread);
while (log_copying_running && !metadata_to_lsn) while (log_copying_running && !metadata_to_lsn)
{ {
...@@ -3114,9 +3117,10 @@ static void *io_watching_thread(void*) ...@@ -3114,9 +3117,10 @@ static void *io_watching_thread(void*)
/* stop io throttle */ /* stop io throttle */
xtrabackup_throttle= 0; xtrabackup_throttle= 0;
have_io_watching_thread= false;
mysql_cond_broadcast(&wait_throttle); mysql_cond_broadcast(&wait_throttle);
mysql_mutex_unlock(&log_sys.mutex); mysql_mutex_unlock(&log_sys.mutex);
return nullptr; my_thread_end();
} }
#ifndef DBUG_OFF #ifndef DBUG_OFF
...@@ -4377,27 +4381,29 @@ static ulong xb_set_max_open_files(rlim_t max_file_limit) ...@@ -4377,27 +4381,29 @@ static ulong xb_set_max_open_files(rlim_t max_file_limit)
# define xb_set_max_open_files(x) 0UL # define xb_set_max_open_files(x) 0UL
#endif #endif
static void stop_backup_threads(bool running) static void stop_backup_threads()
{ {
if (running) mysql_cond_broadcast(&log_copying_stop);
if (log_copying_running || have_io_watching_thread)
{ {
mysql_mutex_unlock(&log_sys.mutex);
fputs("mariabackup: Stopping log copying thread", stderr); fputs("mariabackup: Stopping log copying thread", stderr);
fflush(stderr); fflush(stderr);
while (log_copying_running) mysql_mutex_lock(&log_sys.mutex);
while (log_copying_running || have_io_watching_thread)
{ {
mysql_cond_broadcast(&log_copying_stop);
mysql_mutex_unlock(&log_sys.mutex);
putc('.', stderr); putc('.', stderr);
fflush(stderr); fflush(stderr);
std::this_thread::sleep_for(std::chrono::milliseconds(200)); std::this_thread::sleep_for(std::chrono::milliseconds(200));
mysql_mutex_lock(&log_sys.mutex);
} }
putc('\n', stderr); putc('\n', stderr);
mysql_cond_destroy(&log_copying_stop);
} }
if (have_io_watching_thread) mysql_cond_destroy(&log_copying_stop);
{
pthread_join(io_watching_thread_id, nullptr);
mysql_cond_destroy(&wait_throttle);
}
} }
/** Implement the core of --backup /** Implement the core of --backup
...@@ -4427,11 +4433,7 @@ static bool xtrabackup_backup_low() ...@@ -4427,11 +4433,7 @@ static bool xtrabackup_backup_low()
msg("Error: recv_find_max_checkpoint() failed."); msg("Error: recv_find_max_checkpoint() failed.");
} }
mysql_cond_broadcast(&log_copying_stop); stop_backup_threads();
const bool running= log_copying_running;
mysql_mutex_unlock(&log_sys.mutex);
stop_backup_threads(running);
mysql_mutex_lock(&log_sys.mutex);
} }
if (metadata_to_lsn && xtrabackup_copy_logfile(true)) { if (metadata_to_lsn && xtrabackup_copy_logfile(true)) {
...@@ -4532,9 +4534,8 @@ static bool xtrabackup_backup_func() ...@@ -4532,9 +4534,8 @@ static bool xtrabackup_backup_func()
if (log_copying_running) { if (log_copying_running) {
mysql_mutex_lock(&log_sys.mutex); mysql_mutex_lock(&log_sys.mutex);
metadata_to_lsn = 1; metadata_to_lsn = 1;
mysql_cond_broadcast(&log_copying_stop); stop_backup_threads();
mysql_mutex_unlock(&log_sys.mutex); mysql_mutex_unlock(&log_sys.mutex);
stop_backup_threads(true);
} }
log_file_op = NULL; log_file_op = NULL;
...@@ -4695,13 +4696,15 @@ static bool xtrabackup_backup_func() ...@@ -4695,13 +4696,15 @@ static bool xtrabackup_backup_func()
aligned_free(log_hdr_buf); aligned_free(log_hdr_buf);
log_copying_running = true; log_copying_running = true;
mysql_cond_init(0, &log_copying_stop, nullptr);
/* start io throttle */ /* start io throttle */
if(xtrabackup_throttle) { if (xtrabackup_throttle) {
io_ticket = xtrabackup_throttle; io_ticket = xtrabackup_throttle;
have_io_watching_thread = true; have_io_watching_thread = true;
mysql_cond_init(0, &wait_throttle, nullptr); mysql_cond_init(0, &wait_throttle, nullptr);
mysql_thread_create(0, &io_watching_thread_id, nullptr, std::thread(io_watching_thread).detach();
io_watching_thread, nullptr);
} }
/* Populate fil_system with tablespaces to copy */ /* Populate fil_system with tablespaces to copy */
...@@ -4729,7 +4732,6 @@ static bool xtrabackup_backup_func() ...@@ -4729,7 +4732,6 @@ static bool xtrabackup_backup_func()
DBUG_MARIABACKUP_EVENT("before_innodb_log_copy_thread_started", {}); DBUG_MARIABACKUP_EVENT("before_innodb_log_copy_thread_started", {});
mysql_cond_init(0, &log_copying_stop, nullptr);
std::thread(log_copying_thread).detach(); std::thread(log_copying_thread).detach();
/* FLUSH CHANGED_PAGE_BITMAPS call */ /* FLUSH CHANGED_PAGE_BITMAPS call */
......
...@@ -34,7 +34,7 @@ INSERT INTO t VALUES(1); ...@@ -34,7 +34,7 @@ INSERT INTO t VALUES(1);
echo # Create full backup , modify table, then create incremental/differential backup; echo # Create full backup , modify table, then create incremental/differential backup;
--disable_result_log --disable_result_log
exec $XTRABACKUP --defaults-file=$MYSQLTEST_VARDIR/my.cnf --backup --parallel=10 --target-dir=$basedir; exec $XTRABACKUP --defaults-file=$MYSQLTEST_VARDIR/my.cnf --backup --parallel=10 --target-dir=$basedir --throttle=1000;
--enable_result_log --enable_result_log
BEGIN; BEGIN;
INSERT INTO t VALUES(0); INSERT INTO t VALUES(0);
......
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