mirror of
https://github.com/MariaDB/server.git
synced 2025-04-04 06:15:31 +02:00

Under unknown circumstances, the SQL layer may wrongly disregard an invocation of thd_mark_transaction_to_rollback() when an InnoDB transaction had been aborted (rolled back) due to one of the following errors: * HA_ERR_LOCK_DEADLOCK * HA_ERR_RECORD_CHANGED (if innodb_snapshot_isolation=ON) * HA_ERR_LOCK_WAIT_TIMEOUT (if innodb_rollback_on_timeout=ON) Such an error used to cause a crash of InnoDB during transaction commit. These changes aim to catch and report the error earlier, so that not only this crash can be avoided but also the original root cause be found and fixed more easily later. The idea of this fix is from Michael 'Monty' Widenius. HA_ERR_ROLLBACK: A new error code that will be translated into ER_ROLLBACK_ONLY, signalling that the current transaction has been aborted and the only allowed action is ROLLBACK. trx_t::state: Add TRX_STATE_ABORTED that is like TRX_STATE_NOT_STARTED, but noting that the transaction had been rolled back and aborted. trx_t::is_started(): Replaces trx_is_started(). ha_innobase: Check the transaction state in various places. Simplify the logic around SAVEPOINT. ha_innobase::is_valid_trx(): Replaces ha_innobase::is_read_only(). The InnoDB logic around transaction savepoints, commit, and rollback was unnecessarily complex and might have contributed to this inconsistency. So, we are simplifying that logic as well. trx_savept_t: Replace with const undo_no_t*. When we rollback to a savepoint, all we need to know is the number of undo log records that must survive. trx_named_savept_t, DB_NO_SAVEPOINT: Remove. We can store undo_no_t directly in the space allocated at innobase_hton->savepoint_offset. fts_trx_create(): Do not copy previous savepoints. fts_savepoint_rollback(): If a savepoint was not found, roll back everything after the default savepoint of fts_trx_create(). The test innodb_fts.savepoint is extended to cover this code. Reviewed by: Vladislav Lesin Tested by: Matthias Leich
620 lines
17 KiB
C++
620 lines
17 KiB
C++
/*****************************************************************************
|
|
|
|
Copyright (c) 1996, 2017, Oracle and/or its affiliates. All Rights Reserved.
|
|
Copyright (c) 2016, 2022, MariaDB Corporation.
|
|
|
|
This program is free software; you can redistribute it and/or modify it under
|
|
the terms of the GNU General Public License as published by the Free Software
|
|
Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
this program; if not, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
|
|
|
|
*****************************************************************************/
|
|
|
|
/**************************************************//**
|
|
@file trx/trx0roll.cc
|
|
Transaction rollback
|
|
|
|
Created 3/26/1996 Heikki Tuuri
|
|
*******************************************************/
|
|
|
|
#include "trx0roll.h"
|
|
|
|
#include <my_service_manager.h>
|
|
#include <mysql/service_wsrep.h>
|
|
|
|
#include "fsp0fsp.h"
|
|
#include "lock0lock.h"
|
|
#include "mach0data.h"
|
|
#include "pars0pars.h"
|
|
#include "que0que.h"
|
|
#include "row0mysql.h"
|
|
#include "row0undo.h"
|
|
#include "srv0mon.h"
|
|
#include "srv0start.h"
|
|
#include "trx0rec.h"
|
|
#include "trx0rseg.h"
|
|
#include "trx0sys.h"
|
|
#include "trx0trx.h"
|
|
#include "trx0undo.h"
|
|
#include "log.h"
|
|
|
|
#ifdef UNIV_PFS_THREAD
|
|
mysql_pfs_key_t trx_rollback_clean_thread_key;
|
|
#endif
|
|
|
|
/** true if trx_rollback_all_recovered() thread is active */
|
|
bool trx_rollback_is_active;
|
|
|
|
/** In crash recovery, the current trx to be rolled back; NULL otherwise */
|
|
const trx_t* trx_roll_crash_recv_trx;
|
|
|
|
bool trx_t::rollback_finish() noexcept
|
|
{
|
|
mod_tables.clear();
|
|
apply_online_log= false;
|
|
if (UNIV_LIKELY(error_state == DB_SUCCESS))
|
|
{
|
|
commit();
|
|
return true;
|
|
}
|
|
|
|
ut_a(error_state == DB_INTERRUPTED);
|
|
ut_ad(srv_shutdown_state != SRV_SHUTDOWN_NONE);
|
|
ut_a(!srv_undo_sources);
|
|
ut_ad(srv_fast_shutdown);
|
|
ut_d(in_rollback= false);
|
|
if (trx_undo_t *&undo= rsegs.m_redo.undo)
|
|
{
|
|
UT_LIST_REMOVE(rsegs.m_redo.rseg->undo_list, undo);
|
|
ut_free(undo);
|
|
undo= nullptr;
|
|
}
|
|
if (trx_undo_t *&undo= rsegs.m_noredo.undo)
|
|
{
|
|
UT_LIST_REMOVE(rsegs.m_noredo.rseg->undo_list, undo);
|
|
ut_free(undo);
|
|
undo= nullptr;
|
|
}
|
|
commit_low();
|
|
return commit_cleanup();
|
|
}
|
|
|
|
dberr_t trx_t::rollback_low(const undo_no_t *savept) noexcept
|
|
{
|
|
op_info= "rollback";
|
|
mem_heap_t *heap= mem_heap_create(512);
|
|
roll_node_t *roll_node= roll_node_create(heap);
|
|
|
|
roll_node->savept= savept ? *savept : 0;
|
|
|
|
ut_ad(!in_rollback);
|
|
#ifdef UNIV_DEBUG
|
|
if (savept)
|
|
{
|
|
ut_ad(state == TRX_STATE_ACTIVE);
|
|
ut_ad(mysql_thd);
|
|
ut_ad(!is_recovered);
|
|
}
|
|
#endif
|
|
|
|
error_state= DB_SUCCESS;
|
|
|
|
if (has_logged())
|
|
{
|
|
ut_ad(rsegs.m_redo.rseg || rsegs.m_noredo.rseg);
|
|
que_thr_t *thr= pars_complete_graph_for_exec(roll_node, this, heap,
|
|
nullptr);
|
|
ut_a(thr == que_fork_start_command(static_cast<que_fork_t*>
|
|
(que_node_get_parent(thr))));
|
|
que_run_threads(thr);
|
|
que_run_threads(roll_node->undo_thr);
|
|
|
|
/* Free the memory reserved by the undo graph. */
|
|
que_graph_free(static_cast<que_t*>(roll_node->undo_thr->common.parent));
|
|
}
|
|
|
|
if (!savept)
|
|
{
|
|
rollback_finish();
|
|
MONITOR_INC(MONITOR_TRX_ROLLBACK);
|
|
}
|
|
else
|
|
{
|
|
/* There must not be partial rollback if transaction was chosen as deadlock
|
|
victim. Galera transaction abort can be invoked during partial rollback. */
|
|
ut_ad(!(lock.was_chosen_as_deadlock_victim & 1));
|
|
ut_a(error_state == DB_SUCCESS);
|
|
const undo_no_t limit{*savept};
|
|
apply_online_log= false;
|
|
for (trx_mod_tables_t::iterator i= mod_tables.begin();
|
|
i != mod_tables.end(); )
|
|
{
|
|
trx_mod_tables_t::iterator j= i++;
|
|
ut_ad(j->second.valid());
|
|
if (j->second.rollback(limit))
|
|
mod_tables.erase(j);
|
|
else if (!apply_online_log)
|
|
apply_online_log= j->first->is_active_ddl();
|
|
}
|
|
MONITOR_INC(MONITOR_TRX_ROLLBACK_SAVEPOINT);
|
|
}
|
|
|
|
mem_heap_free(heap);
|
|
op_info= "";
|
|
return error_state;
|
|
}
|
|
|
|
dberr_t trx_t::rollback(const undo_no_t *savept) noexcept
|
|
{
|
|
ut_ad(!mutex_is_owner());
|
|
switch (state) {
|
|
case TRX_STATE_ABORTED:
|
|
ut_ad(!savept);
|
|
state= TRX_STATE_NOT_STARTED;
|
|
/* fall through */
|
|
case TRX_STATE_NOT_STARTED:
|
|
error_state= DB_SUCCESS;
|
|
return DB_SUCCESS;
|
|
case TRX_STATE_PREPARED:
|
|
case TRX_STATE_PREPARED_RECOVERED:
|
|
case TRX_STATE_COMMITTED_IN_MEMORY:
|
|
ut_ad("invalid state" == 0);
|
|
/* fall through */
|
|
case TRX_STATE_ACTIVE:
|
|
break;
|
|
}
|
|
#ifdef WITH_WSREP
|
|
if (!savept && is_wsrep() && wsrep_thd_is_SR(mysql_thd))
|
|
wsrep_handle_SR_rollback(nullptr, mysql_thd);
|
|
#endif /* WITH_WSREP */
|
|
return rollback_low(savept);
|
|
}
|
|
|
|
/** Rollback a transaction used in MySQL
|
|
@param[in, out] trx transaction
|
|
@return error code or DB_SUCCESS */
|
|
dberr_t trx_rollback_for_mysql(trx_t* trx)
|
|
{
|
|
/* We are reading trx->state without holding trx->mutex
|
|
here, because the rollback should be invoked for a running
|
|
active MySQL transaction (or recovered prepared transaction)
|
|
that is associated with the current thread. */
|
|
|
|
switch (trx->state) {
|
|
case TRX_STATE_ABORTED:
|
|
trx->state = TRX_STATE_NOT_STARTED;
|
|
/* fall through */
|
|
case TRX_STATE_NOT_STARTED:
|
|
trx->will_lock = false;
|
|
ut_ad(trx->mysql_thd);
|
|
/* Galera transaction abort can be invoked from MDL acquision
|
|
code, so trx->lock.was_chosen_as_deadlock_victim can be set
|
|
even if trx->state is TRX_STATE_NOT_STARTED. */
|
|
ut_ad(!(trx->lock.was_chosen_as_deadlock_victim & 1));
|
|
#ifdef WITH_WSREP
|
|
ut_ad(!trx->is_wsrep());
|
|
trx->lock.was_chosen_as_deadlock_victim= false;
|
|
#endif
|
|
ut_a(UT_LIST_GET_LEN(trx->lock.trx_locks) == 0);
|
|
return(DB_SUCCESS);
|
|
case TRX_STATE_ACTIVE:
|
|
ut_ad(trx->mysql_thd);
|
|
ut_ad(!trx->is_recovered);
|
|
ut_ad(!trx->is_autocommit_non_locking() || trx->read_only);
|
|
return trx->rollback_low();
|
|
|
|
case TRX_STATE_PREPARED:
|
|
case TRX_STATE_PREPARED_RECOVERED:
|
|
ut_ad(!trx->is_autocommit_non_locking());
|
|
if (trx->rsegs.m_redo.undo) {
|
|
/* The XA ROLLBACK of a XA PREPARE transaction
|
|
will consist of multiple mini-transactions.
|
|
|
|
As the very first step of XA ROLLBACK, we must
|
|
change the undo log state back from
|
|
TRX_UNDO_PREPARED to TRX_UNDO_ACTIVE, in order
|
|
to ensure that recovery will complete the
|
|
rollback.
|
|
|
|
Failure to perform this step could cause a
|
|
situation where we would roll back part of
|
|
a XA PREPARE transaction, the server would be
|
|
killed, and finally, the transaction would be
|
|
recovered in XA PREPARE state, with some of
|
|
the actions already having been rolled back. */
|
|
ut_ad(trx->rsegs.m_redo.undo->rseg
|
|
== trx->rsegs.m_redo.rseg);
|
|
mtr_t mtr;
|
|
mtr.start();
|
|
if (trx_undo_t* undo = trx->rsegs.m_redo.undo) {
|
|
trx_undo_set_state_at_prepare(trx, undo, true,
|
|
&mtr);
|
|
}
|
|
/* Write the redo log for the XA ROLLBACK
|
|
state change to the global buffer. It is
|
|
not necessary to flush the redo log. If
|
|
a durable log write of a later mini-transaction
|
|
takes place for whatever reason, then this state
|
|
change will be durable as well. */
|
|
mtr.commit();
|
|
ut_ad(mtr.commit_lsn() > 0);
|
|
}
|
|
return trx->rollback_low();
|
|
|
|
case TRX_STATE_COMMITTED_IN_MEMORY:
|
|
ut_ad(!trx->is_autocommit_non_locking());
|
|
break;
|
|
}
|
|
|
|
ut_error;
|
|
return(DB_CORRUPTION);
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Roll back an active transaction. */
|
|
static
|
|
void
|
|
trx_rollback_active(
|
|
/*================*/
|
|
trx_t* trx) /*!< in/out: transaction */
|
|
{
|
|
mem_heap_t* heap;
|
|
que_fork_t* fork;
|
|
que_thr_t* thr;
|
|
roll_node_t* roll_node;
|
|
const trx_id_t trx_id = trx->id;
|
|
|
|
ut_ad(trx_id);
|
|
|
|
heap = mem_heap_create(512);
|
|
|
|
fork = que_fork_create(heap);
|
|
fork->trx = trx;
|
|
|
|
thr = que_thr_create(fork, heap, NULL);
|
|
|
|
roll_node = roll_node_create(heap);
|
|
|
|
thr->child = roll_node;
|
|
roll_node->common.parent = thr;
|
|
|
|
trx->graph = fork;
|
|
|
|
ut_a(thr == que_fork_start_command(fork));
|
|
|
|
trx_roll_crash_recv_trx = trx;
|
|
|
|
const bool dictionary_locked = trx->dict_operation;
|
|
|
|
if (dictionary_locked) {
|
|
row_mysql_lock_data_dictionary(trx);
|
|
}
|
|
|
|
que_run_threads(thr);
|
|
ut_a(roll_node->undo_thr != NULL);
|
|
|
|
que_run_threads(roll_node->undo_thr);
|
|
|
|
que_graph_free(
|
|
static_cast<que_t*>(roll_node->undo_thr->common.parent));
|
|
|
|
if (UNIV_UNLIKELY(!trx->rollback_finish())) {
|
|
ut_ad(!dictionary_locked);
|
|
} else {
|
|
sql_print_information(
|
|
"InnoDB: Rolled back recovered transaction "
|
|
TRX_ID_FMT, trx_id);
|
|
}
|
|
|
|
if (dictionary_locked) {
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
}
|
|
|
|
mem_heap_free(heap);
|
|
|
|
trx_roll_crash_recv_trx = NULL;
|
|
}
|
|
|
|
|
|
struct trx_roll_count_callback_arg
|
|
{
|
|
uint32_t n_trx;
|
|
uint64_t n_rows;
|
|
trx_roll_count_callback_arg(): n_trx(0), n_rows(0) {}
|
|
};
|
|
|
|
|
|
static my_bool trx_roll_count_callback(rw_trx_hash_element_t *element,
|
|
trx_roll_count_callback_arg *arg)
|
|
{
|
|
element->mutex.wr_lock();
|
|
if (trx_t *trx= element->trx)
|
|
{
|
|
if (trx->is_recovered && trx_state_eq(trx, TRX_STATE_ACTIVE))
|
|
{
|
|
arg->n_trx++;
|
|
arg->n_rows+= trx->undo_no;
|
|
}
|
|
}
|
|
element->mutex.wr_unlock();
|
|
return 0;
|
|
}
|
|
|
|
/** Report progress when rolling back a row of a recovered transaction. */
|
|
void trx_roll_report_progress()
|
|
{
|
|
time_t now = time(NULL);
|
|
mysql_mutex_lock(&recv_sys.mutex);
|
|
bool report = recv_sys.report(now);
|
|
mysql_mutex_unlock(&recv_sys.mutex);
|
|
|
|
if (report) {
|
|
trx_roll_count_callback_arg arg;
|
|
|
|
/* Get number of recovered active transactions and number of
|
|
rows they modified. Numbers must be accurate, because only this
|
|
thread is allowed to touch recovered transactions. */
|
|
trx_sys.rw_trx_hash.iterate_no_dups(
|
|
trx_roll_count_callback, &arg);
|
|
|
|
if (arg.n_rows > 0) {
|
|
service_manager_extend_timeout(
|
|
INNODB_EXTEND_TIMEOUT_INTERVAL,
|
|
"To roll back: " UINT32PF " transactions, "
|
|
UINT64PF " rows", arg.n_trx, arg.n_rows);
|
|
}
|
|
|
|
ib::info() << "To roll back: " << arg.n_trx
|
|
<< " transactions, " << arg.n_rows << " rows";
|
|
|
|
}
|
|
}
|
|
|
|
|
|
static my_bool trx_rollback_recovered_callback(rw_trx_hash_element_t *element,
|
|
std::vector<trx_t*> *trx_list)
|
|
{
|
|
element->mutex.wr_lock();
|
|
if (trx_t *trx= element->trx)
|
|
{
|
|
trx->mutex_lock();
|
|
if (trx_state_eq(trx, TRX_STATE_ACTIVE) && trx->is_recovered)
|
|
trx_list->push_back(trx);
|
|
trx->mutex_unlock();
|
|
}
|
|
element->mutex.wr_unlock();
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
Rollback any incomplete transactions which were encountered in crash recovery.
|
|
|
|
If the transaction already was committed, then we clean up a possible insert
|
|
undo log. If the transaction was not yet committed, then we roll it back.
|
|
|
|
Note: For XA recovered transactions, we rely on MySQL to
|
|
do rollback. They will be in TRX_STATE_PREPARED state. If the server
|
|
is shutdown and they are still lingering in trx_sys_t::trx_list
|
|
then the shutdown will hang.
|
|
|
|
@param[in] all true=roll back all recovered active transactions;
|
|
false=roll back any incomplete dictionary transaction
|
|
*/
|
|
|
|
void trx_rollback_recovered(bool all)
|
|
{
|
|
std::vector<trx_t*> trx_list;
|
|
|
|
ut_a(srv_force_recovery <
|
|
ulong(all ? SRV_FORCE_NO_TRX_UNDO : SRV_FORCE_NO_DDL_UNDO));
|
|
|
|
/*
|
|
Collect list of recovered ACTIVE transaction ids first. Once collected, no
|
|
other thread is allowed to modify or remove these transactions from
|
|
rw_trx_hash.
|
|
*/
|
|
trx_sys.rw_trx_hash.iterate_no_dups(trx_rollback_recovered_callback,
|
|
&trx_list);
|
|
|
|
while (!trx_list.empty())
|
|
{
|
|
trx_t *trx= trx_list.back();
|
|
trx_list.pop_back();
|
|
|
|
ut_ad(trx);
|
|
ut_d(trx->mutex_lock());
|
|
ut_ad(trx->is_recovered);
|
|
ut_ad(trx_state_eq(trx, TRX_STATE_ACTIVE));
|
|
ut_d(trx->mutex_unlock());
|
|
|
|
if (srv_shutdown_state != SRV_SHUTDOWN_NONE && !srv_undo_sources &&
|
|
srv_fast_shutdown)
|
|
goto discard;
|
|
|
|
if (all || trx->dict_operation || trx->has_stats_table_lock())
|
|
{
|
|
trx_rollback_active(trx);
|
|
if (trx->error_state != DB_SUCCESS)
|
|
{
|
|
ut_ad(trx->error_state == DB_INTERRUPTED);
|
|
trx->error_state= DB_SUCCESS;
|
|
ut_ad(!srv_undo_sources);
|
|
ut_ad(srv_fast_shutdown);
|
|
discard:
|
|
/* Note: before kill_server() invoked innobase_end() via
|
|
unireg_end(), it invoked close_connections(), which should initiate
|
|
the rollback of any user transactions via THD::cleanup() in the
|
|
connection threads, and wait for all THD::cleanup() to complete.
|
|
So, no active user transactions should exist at this point.
|
|
|
|
srv_undo_sources=false was cleared early in innobase_end().
|
|
|
|
Generally, the server guarantees that all connections using
|
|
InnoDB must be disconnected by the time we are reaching this code,
|
|
be it during shutdown or UNINSTALL PLUGIN.
|
|
|
|
Because there is no possible race condition with any
|
|
concurrent user transaction, we do not have to invoke
|
|
trx->commit_state() or wait for !trx->is_referenced()
|
|
before trx_sys.deregister_rw(trx). */
|
|
trx_sys.deregister_rw(trx);
|
|
trx_free_at_shutdown(trx);
|
|
}
|
|
else
|
|
trx->free();
|
|
}
|
|
}
|
|
}
|
|
|
|
/*******************************************************************//**
|
|
Rollback or clean up any incomplete transactions which were
|
|
encountered in crash recovery. If the transaction already was
|
|
committed, then we clean up a possible insert undo log. If the
|
|
transaction was not yet committed, then we roll it back.
|
|
Note: this is done in a background thread. */
|
|
void trx_rollback_all_recovered(void*)
|
|
{
|
|
ut_ad(!srv_read_only_mode);
|
|
|
|
if (trx_sys.rw_trx_hash.size()) {
|
|
ib::info() << "Starting in background the rollback of"
|
|
" recovered transactions";
|
|
trx_rollback_recovered(true);
|
|
ib::info() << "Rollback of non-prepared transactions"
|
|
" completed";
|
|
}
|
|
|
|
trx_rollback_is_active = false;
|
|
}
|
|
|
|
/****************************************************************//**
|
|
Builds an undo 'query' graph for a transaction. The actual rollback is
|
|
performed by executing this query graph like a query subprocedure call.
|
|
The reply about the completion of the rollback will be sent by this
|
|
graph.
|
|
@return own: the query graph */
|
|
static
|
|
que_t*
|
|
trx_roll_graph_build(
|
|
/*=================*/
|
|
trx_t* trx) /*!< in/out: transaction */
|
|
{
|
|
mem_heap_t* heap;
|
|
que_fork_t* fork;
|
|
que_thr_t* thr;
|
|
|
|
ut_ad(trx->mutex_is_owner());
|
|
heap = mem_heap_create(512);
|
|
fork = que_fork_create(heap);
|
|
fork->trx = trx;
|
|
|
|
thr = que_thr_create(fork, heap, NULL);
|
|
|
|
thr->child = row_undo_node_create(trx, thr, heap);
|
|
|
|
return(fork);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Starts a rollback operation, creates the UNDO graph that will do the
|
|
actual undo operation.
|
|
@return query graph thread that will perform the UNDO operations. */
|
|
static
|
|
que_thr_t*
|
|
trx_rollback_start(
|
|
/*===============*/
|
|
trx_t* trx, /*!< in: transaction */
|
|
undo_no_t roll_limit) /*!< in: rollback to undo no (for
|
|
partial undo), 0 if we are rolling back
|
|
the entire transaction */
|
|
{
|
|
/* Initialize the rollback field in the transaction */
|
|
|
|
ut_ad(trx->mutex_is_owner());
|
|
ut_ad(!trx->roll_limit);
|
|
ut_ad(!trx->in_rollback);
|
|
|
|
trx->roll_limit = roll_limit;
|
|
trx->in_rollback = true;
|
|
|
|
ut_a(trx->roll_limit <= trx->undo_no);
|
|
|
|
trx->pages_undone = 0;
|
|
|
|
/* Build a 'query' graph which will perform the undo operations */
|
|
|
|
que_t* roll_graph = trx_roll_graph_build(trx);
|
|
|
|
trx->graph = roll_graph;
|
|
|
|
return(que_fork_start_command(roll_graph));
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Creates a rollback command node struct.
|
|
@return own: rollback node struct */
|
|
roll_node_t*
|
|
roll_node_create(
|
|
/*=============*/
|
|
mem_heap_t* heap) /*!< in: mem heap where created */
|
|
{
|
|
roll_node_t* node;
|
|
|
|
node = static_cast<roll_node_t*>(mem_heap_zalloc(heap, sizeof(*node)));
|
|
|
|
node->state = ROLL_NODE_SEND;
|
|
|
|
node->common.type = QUE_NODE_ROLLBACK;
|
|
|
|
return(node);
|
|
}
|
|
|
|
/***********************************************************//**
|
|
Performs an execution step for a rollback command node in a query graph.
|
|
@return query thread to run next, or NULL */
|
|
que_thr_t*
|
|
trx_rollback_step(
|
|
/*==============*/
|
|
que_thr_t* thr) /*!< in: query thread */
|
|
{
|
|
roll_node_t* node;
|
|
|
|
node = static_cast<roll_node_t*>(thr->run_node);
|
|
|
|
ut_ad(que_node_get_type(node) == QUE_NODE_ROLLBACK);
|
|
|
|
if (thr->prev_node == que_node_get_parent(node)) {
|
|
node->state = ROLL_NODE_SEND;
|
|
}
|
|
|
|
if (node->state == ROLL_NODE_SEND) {
|
|
trx_t* trx;
|
|
|
|
trx = thr_get_trx(thr);
|
|
|
|
node->state = ROLL_NODE_WAIT;
|
|
|
|
ut_a(node->undo_thr == NULL);
|
|
|
|
trx->mutex_lock();
|
|
|
|
trx_commit_or_rollback_prepare(trx);
|
|
|
|
node->undo_thr = trx_rollback_start(trx, node->savept);
|
|
|
|
trx->mutex_unlock();
|
|
} else {
|
|
ut_ad(node->state == ROLL_NODE_WAIT);
|
|
|
|
thr->run_node = que_node_get_parent(node);
|
|
}
|
|
|
|
return(thr);
|
|
}
|