Commit e937a64d authored by Kristian Nielsen's avatar Kristian Nielsen

MDEV-10356: rpl.rpl_parallel_temptable failure due to incorrect commit optimization of temptables

The problem was that parallel replication of temporary tables using
statement-based binlogging could overlap the COMMIT in one thread with a DML
or DROP TEMPORARY TABLE in another thread using the same temporary table.
Temporary tables are not safe for concurrent access, so this caused
reference to freed memory and possibly other nastiness.

The fix is to disable the optimisation with overlapping commits of one
transaction with the start of a later transaction, when temporary tables are
in use. Then the following event groups will be blocked from starting until
the one using temporary tables is completed.

This also fixes occasional test failures of rpl.rpl_parallel_temptable seen
in Buildbot.
Signed-off-by: default avatarKristian Nielsen <knielsen@knielsen-hq.org>
parent d762e9d9
...@@ -218,6 +218,7 @@ finish_event_group(rpl_parallel_thread *rpt, uint64 sub_id, ...@@ -218,6 +218,7 @@ finish_event_group(rpl_parallel_thread *rpt, uint64 sub_id,
waiting for this). In most cases (normal DML), it will be a no-op. waiting for this). In most cases (normal DML), it will be a no-op.
*/ */
rgi->mark_start_commit_no_lock(); rgi->mark_start_commit_no_lock();
rgi->commit_orderer.wakeup_blocked= false;
if (entry->last_committed_sub_id < sub_id) if (entry->last_committed_sub_id < sub_id)
{ {
...@@ -1425,7 +1426,22 @@ handle_rpl_parallel_thread(void *arg) ...@@ -1425,7 +1426,22 @@ handle_rpl_parallel_thread(void *arg)
if (!thd->killed) if (!thd->killed)
{ {
DEBUG_SYNC(thd, "rpl_parallel_before_mark_start_commit"); DEBUG_SYNC(thd, "rpl_parallel_before_mark_start_commit");
rgi->mark_start_commit(); if (thd->lex->stmt_accessed_temp_table())
{
/*
Temporary tables are special, they require strict
single-threaded use as they have no locks protecting concurrent
access. Therefore, we cannot safely use the optimization of
overlapping the commit of this transaction with the start of the
following.
So we skip the early mark_start_commit() and also block any
wakeup_subsequent_commits() until this event group is fully
done, inside finish_event_group().
*/
rgi->commit_orderer.wakeup_blocked= true;
}
else
rgi->mark_start_commit();
DEBUG_SYNC(thd, "rpl_parallel_after_mark_start_commit"); DEBUG_SYNC(thd, "rpl_parallel_after_mark_start_commit");
} }
} }
......
...@@ -7536,6 +7536,7 @@ wait_for_commit::reinit() ...@@ -7536,6 +7536,7 @@ wait_for_commit::reinit()
wakeup_error= 0; wakeup_error= 0;
wakeup_subsequent_commits_running= false; wakeup_subsequent_commits_running= false;
commit_started= false; commit_started= false;
wakeup_blocked= false;
#ifdef SAFE_MUTEX #ifdef SAFE_MUTEX
/* /*
When using SAFE_MUTEX, the ordering between taking the LOCK_wait_commit When using SAFE_MUTEX, the ordering between taking the LOCK_wait_commit
...@@ -7808,6 +7809,9 @@ wait_for_commit::wakeup_subsequent_commits2(int wakeup_error) ...@@ -7808,6 +7809,9 @@ wait_for_commit::wakeup_subsequent_commits2(int wakeup_error)
{ {
wait_for_commit *waiter; wait_for_commit *waiter;
if (unlikely(wakeup_blocked))
return;
mysql_mutex_lock(&LOCK_wait_commit); mysql_mutex_lock(&LOCK_wait_commit);
wakeup_subsequent_commits_running= true; wakeup_subsequent_commits_running= true;
waiter= subsequent_commits_list; waiter= subsequent_commits_list;
......
...@@ -2142,6 +2142,19 @@ struct wait_for_commit ...@@ -2142,6 +2142,19 @@ struct wait_for_commit
group commit as T1. group commit as T1.
*/ */
bool commit_started; bool commit_started;
/*
Set to temporarily ignore calls to wakeup_subsequent_commits(). The
caller must arrange that another wakeup_subsequent_commits() gets called
later after wakeup_blocked has been set back to false.
This is used for parallel replication with temporary tables.
Temporary tables require strict single-threaded operation. The normal
optimization, of doing wakeup_subsequent_commits early and overlapping
part of the commit with the following transaction, is not safe. Thus
when temporary tables are replicated, wakeup is blocked until the
event group is fully done.
*/
bool wakeup_blocked;
void register_wait_for_prior_commit(wait_for_commit *waitee); void register_wait_for_prior_commit(wait_for_commit *waitee);
int wait_for_prior_commit(THD *thd, bool allow_kill=true) int wait_for_prior_commit(THD *thd, bool allow_kill=true)
......
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