2012-08-01 17:27:34 +03:00
|
|
|
/*****************************************************************************
|
|
|
|
|
2016-06-21 14:21:03 +02:00
|
|
|
Copyright (c) 1996, 2016, Oracle and/or its affiliates. All Rights Reserved.
|
2017-01-05 10:48:03 +02:00
|
|
|
Copyright (c) 2014, 2017, MariaDB Corporation. All Rights Reserved.
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
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, Suite 500, Boston, MA 02110-1335 USA
|
|
|
|
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
/**************************************************//**
|
|
|
|
@file lock/lock0wait.cc
|
|
|
|
The transaction lock system
|
|
|
|
|
|
|
|
Created 25/5/2010 Sunny Bains
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
#define LOCK_MODULE_IMPLEMENTATION
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
#include "ha_prototypes.h"
|
|
|
|
#include <mysql/service_thd_wait.h>
|
2016-09-10 16:04:44 +02:00
|
|
|
#include <mysql/service_wsrep.h>
|
2016-08-12 11:17:45 +03:00
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
#include "srv0mon.h"
|
|
|
|
#include "que0que.h"
|
|
|
|
#include "lock0lock.h"
|
|
|
|
#include "row0mysql.h"
|
|
|
|
#include "srv0start.h"
|
|
|
|
#include "lock0priv.h"
|
2016-02-24 01:21:40 -05:00
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
/*********************************************************************//**
|
|
|
|
Print the contents of the lock_sys_t::waiting_threads array. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
lock_wait_table_print(void)
|
|
|
|
/*=======================*/
|
|
|
|
{
|
|
|
|
ut_ad(lock_wait_mutex_own());
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
const srv_slot_t* slot = lock_sys->waiting_threads;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
for (ulint i = 0; i < OS_THREAD_MAX_N; i++, ++slot) {
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
fprintf(stderr,
|
|
|
|
"Slot %lu: thread type %lu,"
|
|
|
|
" in use %lu, susp %lu, timeout %lu, time %lu\n",
|
|
|
|
(ulong) i,
|
|
|
|
(ulong) slot->type,
|
|
|
|
(ulong) slot->in_use,
|
|
|
|
(ulong) slot->suspended,
|
|
|
|
slot->wait_timeout,
|
|
|
|
(ulong) difftime(ut_time(), slot->suspend_time));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Release a slot in the lock_sys_t::waiting_threads. Adjust the array last pointer
|
|
|
|
if there are empty slots towards the end of the table. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
lock_wait_table_release_slot(
|
|
|
|
/*=========================*/
|
|
|
|
srv_slot_t* slot) /*!< in: slot to release */
|
|
|
|
{
|
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
srv_slot_t* upper = lock_sys->waiting_threads + OS_THREAD_MAX_N;
|
|
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
|
|
|
|
lock_wait_mutex_enter();
|
|
|
|
|
|
|
|
ut_ad(slot->in_use);
|
|
|
|
ut_ad(slot->thr != NULL);
|
|
|
|
ut_ad(slot->thr->slot != NULL);
|
|
|
|
ut_ad(slot->thr->slot == slot);
|
|
|
|
|
|
|
|
/* Must be within the array boundaries. */
|
|
|
|
ut_ad(slot >= lock_sys->waiting_threads);
|
|
|
|
ut_ad(slot < upper);
|
|
|
|
|
|
|
|
/* Note: When we reserve the slot we use the trx_t::mutex to update
|
|
|
|
the slot values to change the state to reserved. Here we are using the
|
|
|
|
lock mutex to change the state of the slot to free. This is by design,
|
|
|
|
because when we query the slot state we always hold both the lock and
|
|
|
|
trx_t::mutex. To reduce contention on the lock mutex when reserving the
|
|
|
|
slot we avoid acquiring the lock mutex. */
|
|
|
|
|
|
|
|
lock_mutex_enter();
|
|
|
|
|
|
|
|
slot->thr->slot = NULL;
|
|
|
|
slot->thr = NULL;
|
|
|
|
slot->in_use = FALSE;
|
|
|
|
|
|
|
|
lock_mutex_exit();
|
|
|
|
|
|
|
|
/* Scan backwards and adjust the last free slot pointer. */
|
|
|
|
for (slot = lock_sys->last_slot;
|
|
|
|
slot > lock_sys->waiting_threads && !slot->in_use;
|
|
|
|
--slot) {
|
|
|
|
/* No op */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Either the array is empty or the last scanned slot is in use. */
|
|
|
|
ut_ad(slot->in_use || slot == lock_sys->waiting_threads);
|
|
|
|
|
|
|
|
lock_sys->last_slot = slot + 1;
|
|
|
|
|
|
|
|
/* The last slot is either outside of the array boundary or it's
|
|
|
|
on an empty slot. */
|
|
|
|
ut_ad(lock_sys->last_slot == upper || !lock_sys->last_slot->in_use);
|
|
|
|
|
|
|
|
ut_ad(lock_sys->last_slot >= lock_sys->waiting_threads);
|
|
|
|
ut_ad(lock_sys->last_slot <= upper);
|
|
|
|
|
|
|
|
lock_wait_mutex_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Reserves a slot in the thread table for the current user OS thread.
|
2016-08-12 11:17:45 +03:00
|
|
|
@return reserved slot */
|
2012-08-01 17:27:34 +03:00
|
|
|
static
|
|
|
|
srv_slot_t*
|
|
|
|
lock_wait_table_reserve_slot(
|
|
|
|
/*=========================*/
|
|
|
|
que_thr_t* thr, /*!< in: query thread associated
|
|
|
|
with the user OS thread */
|
|
|
|
ulong wait_timeout) /*!< in: lock wait timeout value */
|
|
|
|
{
|
|
|
|
ulint i;
|
|
|
|
srv_slot_t* slot;
|
|
|
|
|
|
|
|
ut_ad(lock_wait_mutex_own());
|
|
|
|
ut_ad(trx_mutex_own(thr_get_trx(thr)));
|
|
|
|
|
|
|
|
slot = lock_sys->waiting_threads;
|
|
|
|
|
|
|
|
for (i = OS_THREAD_MAX_N; i--; ++slot) {
|
|
|
|
if (!slot->in_use) {
|
|
|
|
slot->in_use = TRUE;
|
|
|
|
slot->thr = thr;
|
|
|
|
slot->thr->slot = slot;
|
|
|
|
|
|
|
|
if (slot->event == NULL) {
|
2016-08-12 11:17:45 +03:00
|
|
|
slot->event = os_event_create(0);
|
2012-08-01 17:27:34 +03:00
|
|
|
ut_a(slot->event);
|
|
|
|
}
|
|
|
|
|
|
|
|
os_event_reset(slot->event);
|
|
|
|
slot->suspended = TRUE;
|
|
|
|
slot->suspend_time = ut_time();
|
|
|
|
slot->wait_timeout = wait_timeout;
|
|
|
|
|
|
|
|
if (slot == lock_sys->last_slot) {
|
|
|
|
++lock_sys->last_slot;
|
|
|
|
}
|
|
|
|
|
|
|
|
ut_ad(lock_sys->last_slot
|
|
|
|
<= lock_sys->waiting_threads + OS_THREAD_MAX_N);
|
|
|
|
|
|
|
|
return(slot);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
ib::error() << "There appear to be " << OS_THREAD_MAX_N << " user"
|
|
|
|
" threads currently waiting inside InnoDB, which is the upper"
|
|
|
|
" limit. Cannot continue operation. Before aborting, we print"
|
|
|
|
" a list of waiting threads.";
|
2012-08-01 17:27:34 +03:00
|
|
|
lock_wait_table_print();
|
|
|
|
|
|
|
|
ut_error;
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
2014-03-25 23:13:30 -04:00
|
|
|
#ifdef WITH_WSREP
|
|
|
|
/*********************************************************************//**
|
2016-08-12 11:17:45 +03:00
|
|
|
check if lock timeout was for priority thread,
|
2014-03-25 23:13:30 -04:00
|
|
|
as a side effect trigger lock monitor
|
2016-08-12 11:17:45 +03:00
|
|
|
@return false for regular lock timeout */
|
2014-03-25 23:13:30 -04:00
|
|
|
static ibool
|
|
|
|
wsrep_is_BF_lock_timeout(
|
|
|
|
/*====================*/
|
|
|
|
trx_t* trx) /* in: trx to check for lock priority */
|
|
|
|
{
|
2016-08-12 11:17:45 +03:00
|
|
|
if (wsrep_on(trx->mysql_thd) &&
|
|
|
|
wsrep_thd_is_BF(trx->mysql_thd, FALSE)) {
|
|
|
|
fprintf(stderr, "WSREP: BF lock wait long\n");
|
2016-09-06 09:43:16 +03:00
|
|
|
srv_print_innodb_monitor = TRUE;
|
|
|
|
srv_print_innodb_lock_monitor = TRUE;
|
|
|
|
os_event_set(srv_monitor_event);
|
|
|
|
return TRUE;
|
2016-08-12 11:17:45 +03:00
|
|
|
}
|
|
|
|
return FALSE;
|
2014-03-25 23:13:30 -04:00
|
|
|
}
|
|
|
|
#endif /* WITH_WSREP */
|
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
/***************************************************************//**
|
|
|
|
Puts a user OS thread to wait for a lock to be released. If an error
|
|
|
|
occurs during the wait trx->error_state associated with thr is
|
|
|
|
!= DB_SUCCESS when we return. DB_LOCK_WAIT_TIMEOUT and DB_DEADLOCK
|
|
|
|
are possible errors. DB_DEADLOCK is returned if selective deadlock
|
|
|
|
resolution chose this transaction as a victim. */
|
|
|
|
void
|
|
|
|
lock_wait_suspend_thread(
|
|
|
|
/*=====================*/
|
|
|
|
que_thr_t* thr) /*!< in: query thread associated with the
|
|
|
|
user OS thread */
|
|
|
|
{
|
|
|
|
srv_slot_t* slot;
|
|
|
|
double wait_time;
|
|
|
|
trx_t* trx;
|
|
|
|
ibool was_declared_inside_innodb;
|
2016-08-12 11:17:45 +03:00
|
|
|
int64_t start_time = 0;
|
|
|
|
int64_t finish_time;
|
2012-08-01 17:27:34 +03:00
|
|
|
ulint sec;
|
|
|
|
ulint ms;
|
|
|
|
ulong lock_wait_timeout;
|
|
|
|
|
|
|
|
trx = thr_get_trx(thr);
|
|
|
|
|
|
|
|
if (trx->mysql_thd != 0) {
|
|
|
|
DEBUG_SYNC_C("lock_wait_suspend_thread_enter");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* InnoDB system transactions (such as the purge, and
|
|
|
|
incomplete transactions that are being rolled back after crash
|
|
|
|
recovery) will use the global value of
|
|
|
|
innodb_lock_wait_timeout, because trx->mysql_thd == NULL. */
|
|
|
|
lock_wait_timeout = trx_lock_wait_timeout_get(trx);
|
|
|
|
|
|
|
|
lock_wait_mutex_enter();
|
|
|
|
|
|
|
|
trx_mutex_enter(trx);
|
|
|
|
|
|
|
|
trx->error_state = DB_SUCCESS;
|
|
|
|
|
|
|
|
if (thr->state == QUE_THR_RUNNING) {
|
|
|
|
|
|
|
|
ut_ad(thr->is_active);
|
|
|
|
|
|
|
|
/* The lock has already been released or this transaction
|
|
|
|
was chosen as a deadlock victim: no need to suspend */
|
|
|
|
|
|
|
|
if (trx->lock.was_chosen_as_deadlock_victim) {
|
|
|
|
|
|
|
|
trx->error_state = DB_DEADLOCK;
|
2016-08-12 11:17:45 +03:00
|
|
|
trx->lock.was_chosen_as_deadlock_victim = false;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
lock_wait_mutex_exit();
|
|
|
|
trx_mutex_exit(trx);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ut_ad(!thr->is_active);
|
|
|
|
|
|
|
|
slot = lock_wait_table_reserve_slot(thr, lock_wait_timeout);
|
|
|
|
|
|
|
|
if (thr->lock_state == QUE_THR_LOCK_ROW) {
|
2013-03-26 00:03:13 +02:00
|
|
|
srv_stats.n_lock_wait_count.inc();
|
|
|
|
srv_stats.n_lock_wait_current_count.inc();
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
if (ut_usectime(&sec, &ms) == -1) {
|
|
|
|
start_time = -1;
|
|
|
|
} else {
|
2016-08-12 11:17:45 +03:00
|
|
|
start_time = static_cast<int64_t>(sec) * 1000000 + ms;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
/* Wake the lock timeout monitor thread, if it is suspended */
|
|
|
|
|
|
|
|
os_event_set(lock_sys->timeout_event);
|
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
lock_wait_mutex_exit();
|
|
|
|
trx_mutex_exit(trx);
|
|
|
|
|
2014-02-01 09:33:26 +01:00
|
|
|
ulint lock_type = ULINT_UNDEFINED;
|
|
|
|
|
|
|
|
lock_mutex_enter();
|
|
|
|
|
|
|
|
if (const lock_t* wait_lock = trx->lock.wait_lock) {
|
|
|
|
lock_type = lock_get_type_low(wait_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
lock_mutex_exit();
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
ulint had_dict_lock = trx->dict_operation_lock_mode;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
switch (had_dict_lock) {
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case RW_S_LATCH:
|
|
|
|
/* Release foreign key check latch */
|
|
|
|
row_mysql_unfreeze_data_dictionary(trx);
|
2013-03-26 00:03:13 +02:00
|
|
|
|
|
|
|
DEBUG_SYNC_C("lock_wait_release_s_latch_before_sleep");
|
2012-08-01 17:27:34 +03:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* There should never be a lock wait when the
|
|
|
|
dictionary latch is reserved in X mode. Dictionary
|
|
|
|
transactions should only acquire locks on dictionary
|
|
|
|
tables, not other tables. All access to dictionary
|
|
|
|
tables should be covered by dictionary
|
|
|
|
transactions. */
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ut_a(trx->dict_operation_lock_mode == 0);
|
|
|
|
|
|
|
|
/* Suspend this thread and wait for the event. */
|
|
|
|
|
|
|
|
was_declared_inside_innodb = trx->declared_to_be_inside_innodb;
|
|
|
|
|
|
|
|
if (was_declared_inside_innodb) {
|
|
|
|
/* We must declare this OS thread to exit InnoDB, since a
|
|
|
|
possible other thread holding a lock which this thread waits
|
|
|
|
for must be allowed to enter, sooner or later */
|
|
|
|
|
|
|
|
srv_conc_force_exit_innodb(trx);
|
|
|
|
}
|
|
|
|
|
2014-02-01 09:33:26 +01:00
|
|
|
/* Unknown is also treated like a record lock */
|
|
|
|
if (lock_type == ULINT_UNDEFINED || lock_type == LOCK_REC) {
|
|
|
|
thd_wait_begin(trx->mysql_thd, THD_WAIT_ROW_LOCK);
|
|
|
|
} else {
|
|
|
|
ut_ad(lock_type == LOCK_TABLE);
|
|
|
|
thd_wait_begin(trx->mysql_thd, THD_WAIT_TABLE_LOCK);
|
|
|
|
}
|
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
os_event_wait(slot->event);
|
|
|
|
|
2014-02-01 09:33:26 +01:00
|
|
|
thd_wait_end(trx->mysql_thd);
|
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
/* After resuming, reacquire the data dictionary latch if
|
|
|
|
necessary. */
|
|
|
|
|
|
|
|
if (was_declared_inside_innodb) {
|
|
|
|
|
|
|
|
/* Return back inside InnoDB */
|
|
|
|
|
|
|
|
srv_conc_force_enter_innodb(trx);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (had_dict_lock) {
|
|
|
|
|
|
|
|
row_mysql_freeze_data_dictionary(trx);
|
|
|
|
}
|
|
|
|
|
|
|
|
wait_time = ut_difftime(ut_time(), slot->suspend_time);
|
|
|
|
|
|
|
|
/* Release the slot for others to use */
|
|
|
|
|
|
|
|
lock_wait_table_release_slot(slot);
|
|
|
|
|
|
|
|
if (thr->lock_state == QUE_THR_LOCK_ROW) {
|
|
|
|
ulint diff_time;
|
|
|
|
|
|
|
|
if (ut_usectime(&sec, &ms) == -1) {
|
|
|
|
finish_time = -1;
|
|
|
|
} else {
|
2016-08-12 11:17:45 +03:00
|
|
|
finish_time = static_cast<int64_t>(sec) * 1000000 + ms;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
|
2014-02-01 09:33:26 +01:00
|
|
|
diff_time = (finish_time > start_time) ?
|
|
|
|
(ulint) (finish_time - start_time) : 0;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2013-03-26 00:03:13 +02:00
|
|
|
srv_stats.n_lock_wait_current_count.dec();
|
|
|
|
srv_stats.n_lock_wait_time.add(diff_time);
|
2012-08-01 17:27:34 +03:00
|
|
|
|
2013-03-26 00:03:13 +02:00
|
|
|
/* Only update the variable if we successfully
|
|
|
|
retrieved the start and finish times. See Bug#36819. */
|
|
|
|
if (diff_time > lock_sys->n_lock_max_wait_time
|
|
|
|
&& start_time != -1
|
|
|
|
&& finish_time != -1) {
|
|
|
|
|
|
|
|
lock_sys->n_lock_max_wait_time = diff_time;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
2014-02-01 09:33:26 +01:00
|
|
|
|
|
|
|
/* Record the lock wait time for this thread */
|
|
|
|
thd_set_lock_wait_time(trx->mysql_thd, diff_time);
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
DBUG_EXECUTE_IF("lock_instrument_slow_query_log",
|
|
|
|
os_thread_sleep(1000););
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The transaction is chosen as deadlock victim during sleep. */
|
|
|
|
if (trx->error_state == DB_DEADLOCK) {
|
|
|
|
return;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lock_wait_timeout < 100000000
|
2016-08-12 11:17:45 +03:00
|
|
|
&& wait_time > (double) lock_wait_timeout
|
2014-03-25 23:13:30 -04:00
|
|
|
#ifdef WITH_WSREP
|
2016-09-06 09:43:16 +03:00
|
|
|
&& (!wsrep_on(trx->mysql_thd) ||
|
|
|
|
(!wsrep_is_BF_lock_timeout(trx) && trx->error_state != DB_DEADLOCK))
|
2014-03-25 23:13:30 -04:00
|
|
|
#endif /* WITH_WSREP */
|
2016-08-12 11:17:45 +03:00
|
|
|
&& !trx_is_high_priority(trx)) {
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
trx->error_state = DB_LOCK_WAIT_TIMEOUT;
|
|
|
|
|
|
|
|
MONITOR_INC(MONITOR_TIMEOUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trx_is_interrupted(trx)) {
|
|
|
|
|
|
|
|
trx->error_state = DB_INTERRUPTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************//**
|
|
|
|
Releases a user OS thread waiting for a lock to be released, if the
|
|
|
|
thread is already suspended. */
|
|
|
|
void
|
|
|
|
lock_wait_release_thread_if_suspended(
|
|
|
|
/*==================================*/
|
|
|
|
que_thr_t* thr) /*!< in: query thread associated with the
|
|
|
|
user OS thread */
|
|
|
|
{
|
|
|
|
ut_ad(lock_mutex_own());
|
|
|
|
ut_ad(trx_mutex_own(thr_get_trx(thr)));
|
|
|
|
|
|
|
|
/* We own both the lock mutex and the trx_t::mutex but not the
|
|
|
|
lock wait mutex. This is OK because other threads will see the state
|
|
|
|
of this slot as being in use and no other thread can change the state
|
|
|
|
of the slot to free unless that thread also owns the lock mutex. */
|
|
|
|
|
|
|
|
if (thr->slot != NULL && thr->slot->in_use && thr->slot->thr == thr) {
|
|
|
|
trx_t* trx = thr_get_trx(thr);
|
|
|
|
|
|
|
|
if (trx->lock.was_chosen_as_deadlock_victim) {
|
|
|
|
|
|
|
|
trx->error_state = DB_DEADLOCK;
|
2016-08-12 11:17:45 +03:00
|
|
|
trx->lock.was_chosen_as_deadlock_victim = false;
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
os_event_set(thr->slot->event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Check if the thread lock wait has timed out. Release its locks if the
|
|
|
|
wait has actually timed out. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
lock_wait_check_and_cancel(
|
|
|
|
/*=======================*/
|
|
|
|
const srv_slot_t* slot) /*!< in: slot reserved by a user
|
|
|
|
thread when the wait started */
|
|
|
|
{
|
|
|
|
trx_t* trx;
|
|
|
|
double wait_time;
|
|
|
|
ib_time_t suspend_time = slot->suspend_time;
|
|
|
|
|
|
|
|
ut_ad(lock_wait_mutex_own());
|
|
|
|
|
|
|
|
ut_ad(slot->in_use);
|
|
|
|
|
|
|
|
ut_ad(slot->suspended);
|
|
|
|
|
|
|
|
wait_time = ut_difftime(ut_time(), suspend_time);
|
|
|
|
|
|
|
|
trx = thr_get_trx(slot->thr);
|
|
|
|
|
|
|
|
if (trx_is_interrupted(trx)
|
|
|
|
|| (slot->wait_timeout < 100000000
|
|
|
|
&& (wait_time > (double) slot->wait_timeout
|
|
|
|
|| wait_time < 0))) {
|
|
|
|
|
|
|
|
/* Timeout exceeded or a wrap-around in system
|
|
|
|
time counter: cancel the lock request queued
|
|
|
|
by the transaction and release possible
|
|
|
|
other transactions waiting behind; it is
|
|
|
|
possible that the lock has already been
|
|
|
|
granted: in that case do nothing */
|
|
|
|
|
|
|
|
lock_mutex_enter();
|
|
|
|
|
|
|
|
trx_mutex_enter(trx);
|
|
|
|
|
2016-08-12 11:17:45 +03:00
|
|
|
if (trx->lock.wait_lock != NULL && !trx_is_high_priority(trx)) {
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
ut_a(trx->lock.que_state == TRX_QUE_LOCK_WAIT);
|
2016-08-12 11:17:45 +03:00
|
|
|
|
2014-03-25 23:13:30 -04:00
|
|
|
#ifdef WITH_WSREP
|
|
|
|
if (!wsrep_is_BF_lock_timeout(trx)) {
|
|
|
|
#endif /* WITH_WSREP */
|
2016-08-12 11:17:45 +03:00
|
|
|
lock_cancel_waiting_and_release(trx->lock.wait_lock);
|
2014-03-25 23:13:30 -04:00
|
|
|
#ifdef WITH_WSREP
|
|
|
|
}
|
|
|
|
#endif /* WITH_WSREP */
|
2012-08-01 17:27:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
lock_mutex_exit();
|
|
|
|
|
|
|
|
trx_mutex_exit(trx);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
A thread which wakes up threads whose lock wait may have lasted too long.
|
2016-08-12 11:17:45 +03:00
|
|
|
@return a dummy parameter */
|
|
|
|
extern "C"
|
2012-08-01 17:27:34 +03:00
|
|
|
os_thread_ret_t
|
MDEV-11638 Encryption causes race conditions in InnoDB shutdown
InnoDB shutdown failed to properly take fil_crypt_thread() into account.
The encryption threads were signalled to shut down together with other
non-critical tasks. This could be much too early in case of slow shutdown,
which could need minutes to complete the purge. Furthermore, InnoDB
failed to wait for the fil_crypt_thread() to actually exit before
proceeding to the final steps of shutdown, causing the race conditions.
Furthermore, the log_scrub_thread() was shut down way too early.
Also it should remain until the SRV_SHUTDOWN_FLUSH_PHASE.
fil_crypt_threads_end(): Remove. This would cause the threads to
be terminated way too early.
srv_buf_dump_thread_active, srv_dict_stats_thread_active,
lock_sys->timeout_thread_active, log_scrub_thread_active,
srv_monitor_active, srv_error_monitor_active: Remove a race condition
between startup and shutdown, by setting these in the startup thread
that creates threads, not in each created thread. In this way, once the
flag is cleared, it will remain cleared during shutdown.
srv_n_fil_crypt_threads_started, fil_crypt_threads_event: Declare in
global rather than static scope.
log_scrub_event, srv_log_scrub_thread_active, log_scrub_thread():
Declare in static rather than global scope. Let these be created by
log_init() and freed by log_shutdown().
rotate_thread_t::should_shutdown(): Do not shut down before the
SRV_SHUTDOWN_FLUSH_PHASE.
srv_any_background_threads_are_active(): Remove. These checks now
exist in logs_empty_and_mark_files_at_shutdown().
logs_empty_and_mark_files_at_shutdown(): Shut down the threads in
the proper order. Keep fil_crypt_thread() and log_scrub_thread() alive
until SRV_SHUTDOWN_FLUSH_PHASE, and check that they actually terminate.
2017-01-04 18:43:32 +02:00
|
|
|
DECLARE_THREAD(lock_wait_timeout_thread)(void*)
|
2012-08-01 17:27:34 +03:00
|
|
|
{
|
2016-08-12 11:17:45 +03:00
|
|
|
int64_t sig_count = 0;
|
2013-03-26 00:03:13 +02:00
|
|
|
os_event_t event = lock_sys->timeout_event;
|
|
|
|
|
|
|
|
ut_ad(!srv_read_only_mode);
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
#ifdef UNIV_PFS_THREAD
|
2016-09-11 10:57:05 +02:00
|
|
|
pfs_register_thread(srv_lock_timeout_thread_key);
|
2013-03-26 00:03:13 +02:00
|
|
|
#endif /* UNIV_PFS_THREAD */
|
|
|
|
|
2012-08-01 17:27:34 +03:00
|
|
|
do {
|
|
|
|
srv_slot_t* slot;
|
|
|
|
|
|
|
|
/* When someone is waiting for a lock, we wake up every second
|
|
|
|
and check if a timeout has passed for a lock wait */
|
|
|
|
|
2013-03-26 00:03:13 +02:00
|
|
|
os_event_wait_time_low(event, 1000000, sig_count);
|
|
|
|
sig_count = os_event_reset(event);
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
if (srv_shutdown_state >= SRV_SHUTDOWN_CLEANUP) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
lock_wait_mutex_enter();
|
|
|
|
|
|
|
|
/* Check all slots for user threads that are waiting
|
|
|
|
on locks, and if they have exceeded the time limit. */
|
|
|
|
|
|
|
|
for (slot = lock_sys->waiting_threads;
|
|
|
|
slot < lock_sys->last_slot;
|
|
|
|
++slot) {
|
|
|
|
|
|
|
|
/* We are doing a read without the lock mutex
|
|
|
|
and/or the trx mutex. This is OK because a slot
|
|
|
|
can't be freed or reserved without the lock wait
|
|
|
|
mutex. */
|
|
|
|
|
|
|
|
if (slot->in_use) {
|
|
|
|
lock_wait_check_and_cancel(slot);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-26 00:03:13 +02:00
|
|
|
sig_count = os_event_reset(event);
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
lock_wait_mutex_exit();
|
|
|
|
|
|
|
|
} while (srv_shutdown_state < SRV_SHUTDOWN_CLEANUP);
|
|
|
|
|
2013-03-26 00:03:13 +02:00
|
|
|
lock_sys->timeout_thread_active = false;
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
/* We count the number of threads in os_thread_exit(). A created
|
|
|
|
thread should always use that to exit and not use return() to exit. */
|
|
|
|
|
2016-09-06 09:43:16 +03:00
|
|
|
os_thread_exit();
|
2012-08-01 17:27:34 +03:00
|
|
|
|
|
|
|
OS_THREAD_DUMMY_RETURN;
|
|
|
|
}
|
2016-08-12 11:17:45 +03:00
|
|
|
|