2014-02-26 19:11:54 +01:00
|
|
|
/*****************************************************************************
|
|
|
|
|
2016-09-06 08:43:16 +02:00
|
|
|
Copyright (c) 1995, 2016, Oracle and/or its affiliates. All Rights Reserved.
|
2014-02-26 19:11:54 +01:00
|
|
|
Copyright (c) 2008, Google Inc.
|
2018-07-23 12:31:10 +02:00
|
|
|
Copyright (c) 2017, 2018, MariaDB Corporation.
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
Portions of this file contain modifications contributed and copyrighted by
|
|
|
|
Google, Inc. Those modifications are gratefully acknowledged and are described
|
|
|
|
briefly in the InnoDB documentation. The contributions by Google are
|
|
|
|
incorporated with their permission, and subject to the conditions contained in
|
|
|
|
the file COPYING.Google.
|
|
|
|
|
|
|
|
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 sync/sync0rw.cc
|
|
|
|
The read-write lock (for thread synchronization)
|
|
|
|
|
|
|
|
Created 9/11/1995 Heikki Tuuri
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
#include "sync0rw.h"
|
2014-08-19 18:28:35 +02:00
|
|
|
#include "my_cpu.h"
|
2016-08-12 10:17:45 +02:00
|
|
|
#include <my_sys.h>
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
IMPLEMENTATION OF THE RW_LOCK
|
|
|
|
=============================
|
|
|
|
The status of a rw_lock is held in lock_word. The initial value of lock_word is
|
|
|
|
X_LOCK_DECR. lock_word is decremented by 1 for each s-lock and by X_LOCK_DECR
|
2016-08-12 10:17:45 +02:00
|
|
|
or 1 for each x-lock. This describes the lock state for each value of lock_word:
|
|
|
|
|
|
|
|
lock_word == X_LOCK_DECR: Unlocked.
|
|
|
|
X_LOCK_HALF_DECR < lock_word < X_LOCK_DECR:
|
|
|
|
S locked, no waiting writers.
|
|
|
|
(X_LOCK_DECR - lock_word) is the number
|
|
|
|
of S locks.
|
|
|
|
lock_word == X_LOCK_HALF_DECR: SX locked, no waiting writers.
|
|
|
|
0 < lock_word < X_LOCK_HALF_DECR:
|
|
|
|
SX locked AND S locked, no waiting writers.
|
|
|
|
(X_LOCK_HALF_DECR - lock_word) is the number
|
|
|
|
of S locks.
|
|
|
|
lock_word == 0: X locked, no waiting writers.
|
|
|
|
-X_LOCK_HALF_DECR < lock_word < 0:
|
|
|
|
S locked, with a waiting writer.
|
|
|
|
(-lock_word) is the number of S locks.
|
|
|
|
lock_word == -X_LOCK_HALF_DECR: X locked and SX locked, no waiting writers.
|
|
|
|
-X_LOCK_DECR < lock_word < -X_LOCK_HALF_DECR:
|
|
|
|
S locked, with a waiting writer
|
|
|
|
which has SX lock.
|
|
|
|
-(lock_word + X_LOCK_HALF_DECR) is the number
|
|
|
|
of S locks.
|
|
|
|
lock_word == -X_LOCK_DECR: X locked with recursive X lock (2 X locks).
|
|
|
|
-(X_LOCK_DECR + X_LOCK_HALF_DECR) < lock_word < -X_LOCK_DECR:
|
|
|
|
X locked. The number of the X locks is:
|
|
|
|
2 - (lock_word + X_LOCK_DECR)
|
|
|
|
lock_word == -(X_LOCK_DECR + X_LOCK_HALF_DECR):
|
|
|
|
X locked with recursive X lock (2 X locks)
|
|
|
|
and SX locked.
|
|
|
|
lock_word < -(X_LOCK_DECR + X_LOCK_HALF_DECR):
|
|
|
|
X locked and SX locked.
|
|
|
|
The number of the X locks is:
|
|
|
|
2 - (lock_word + X_LOCK_DECR + X_LOCK_HALF_DECR)
|
|
|
|
|
|
|
|
LOCK COMPATIBILITY MATRIX
|
2017-10-12 11:56:20 +02:00
|
|
|
|
|
|
|
| S|SX| X|
|
|
|
|
--+--+--+--+
|
|
|
|
S| +| +| -|
|
|
|
|
--+--+--+--+
|
|
|
|
SX| +| -| -|
|
|
|
|
--+--+--+--+
|
|
|
|
X| -| -| -|
|
|
|
|
--+--+--+--+
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
The lock_word is always read and updated atomically and consistently, so that
|
|
|
|
it always represents the state of the lock, and the state of the lock changes
|
|
|
|
with a single atomic operation. This lock_word holds all of the information
|
|
|
|
that a thread needs in order to determine if it is eligible to gain the lock
|
|
|
|
or if it must spin or sleep. The one exception to this is that writer_thread
|
|
|
|
must be verified before recursive write locks: to solve this scenario, we make
|
2016-08-12 10:17:45 +02:00
|
|
|
writer_thread readable by all threads, but only writeable by the x-lock or
|
|
|
|
sx-lock holder.
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
The other members of the lock obey the following rules to remain consistent:
|
|
|
|
|
2016-11-23 11:07:17 +01:00
|
|
|
writer_thread: Is used only in recursive x-locking or sx-locking.
|
|
|
|
This field is 0 at lock creation time and is updated
|
|
|
|
when x-lock is acquired or when move_ownership is called.
|
|
|
|
A thread is only allowed to set the value of this field to
|
|
|
|
it's thread_id i.e.: a thread cannot set writer_thread to
|
|
|
|
some other thread's id.
|
2014-02-26 19:11:54 +01:00
|
|
|
waiters: May be set to 1 anytime, but to avoid unnecessary wake-up
|
|
|
|
signals, it should only be set to 1 when there are threads
|
|
|
|
waiting on event. Must be 1 when a writer starts waiting to
|
|
|
|
ensure the current x-locking thread sends a wake-up signal
|
|
|
|
during unlock. May only be reset to 0 immediately before a
|
|
|
|
a wake-up signal is sent to event. On most platforms, a
|
|
|
|
memory barrier is required after waiters is set, and before
|
|
|
|
verifying lock_word is still held, to ensure some unlocker
|
|
|
|
really does see the flags new value.
|
|
|
|
event: Threads wait on event for read or writer lock when another
|
|
|
|
thread has an x-lock or an x-lock reservation (wait_ex). A
|
|
|
|
thread may only wait on event after performing the following
|
|
|
|
actions in order:
|
|
|
|
(1) Record the counter value of event (with os_event_reset).
|
|
|
|
(2) Set waiters to 1.
|
|
|
|
(3) Verify lock_word <= 0.
|
|
|
|
(1) must come before (2) to ensure signal is not missed.
|
|
|
|
(2) must come before (3) to ensure a signal is sent.
|
|
|
|
These restrictions force the above ordering.
|
|
|
|
Immediately before sending the wake-up signal, we should:
|
|
|
|
(1) Verify lock_word == X_LOCK_DECR (unlocked)
|
|
|
|
(2) Reset waiters to 0.
|
|
|
|
wait_ex_event: A thread may only wait on the wait_ex_event after it has
|
|
|
|
performed the following actions in order:
|
|
|
|
(1) Decrement lock_word by X_LOCK_DECR.
|
|
|
|
(2) Record counter value of wait_ex_event (os_event_reset,
|
|
|
|
called from sync_array_reserve_cell).
|
|
|
|
(3) Verify that lock_word < 0.
|
|
|
|
(1) must come first to ensures no other threads become reader
|
|
|
|
or next writer, and notifies unlocker that signal must be sent.
|
|
|
|
(2) must come before (3) to ensure the signal is not missed.
|
|
|
|
These restrictions force the above ordering.
|
|
|
|
Immediately before sending the wake-up signal, we should:
|
|
|
|
Verify lock_word == 0 (waiting thread holds x_lock)
|
|
|
|
*/
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats_t rw_lock_stats;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* The global list of rw-locks */
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_list_t rw_lock_list;
|
|
|
|
ib_mutex_t rw_lock_list_mutex;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
#ifdef UNIV_DEBUG
|
2014-02-26 19:11:54 +01:00
|
|
|
/******************************************************************//**
|
|
|
|
Creates a debug info struct. */
|
|
|
|
static
|
|
|
|
rw_lock_debug_t*
|
|
|
|
rw_lock_debug_create(void);
|
|
|
|
/*======================*/
|
|
|
|
/******************************************************************//**
|
|
|
|
Frees a debug info struct. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
rw_lock_debug_free(
|
|
|
|
/*===============*/
|
|
|
|
rw_lock_debug_t* info);
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Creates a debug info struct.
|
2016-08-12 10:17:45 +02:00
|
|
|
@return own: debug info struct */
|
2014-02-26 19:11:54 +01:00
|
|
|
static
|
|
|
|
rw_lock_debug_t*
|
|
|
|
rw_lock_debug_create(void)
|
|
|
|
/*======================*/
|
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
return((rw_lock_debug_t*) ut_malloc_nokey(sizeof(rw_lock_debug_t)));
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Frees a debug info struct. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
rw_lock_debug_free(
|
|
|
|
/*===============*/
|
|
|
|
rw_lock_debug_t* info)
|
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_free(info);
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
#endif /* UNIV_DEBUG */
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Creates, or rather, initializes an rw-lock object in a specified memory
|
|
|
|
location (which must be appropriately aligned). The rw-lock is initialized
|
|
|
|
to the non-locked state. Explicit freeing of the rw-lock with rw_lock_free
|
|
|
|
is necessary only if the memory block containing it is freed. */
|
|
|
|
void
|
|
|
|
rw_lock_create_func(
|
|
|
|
/*================*/
|
|
|
|
rw_lock_t* lock, /*!< in: pointer to memory */
|
|
|
|
#ifdef UNIV_DEBUG
|
2016-08-12 10:17:45 +02:00
|
|
|
latch_level_t level, /*!< in: level */
|
2014-02-26 19:11:54 +01:00
|
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
const char* cfile_name, /*!< in: file name where created */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned cline) /*!< in: file line where created */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
#if defined(UNIV_DEBUG) && !defined(UNIV_PFS_RWLOCK)
|
|
|
|
/* It should have been created in pfs_rw_lock_create_func() */
|
|
|
|
new(lock) rw_lock_t();
|
|
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
/* If this is the very first time a synchronization object is
|
|
|
|
created, then the following call initializes the sync system. */
|
|
|
|
|
|
|
|
lock->lock_word = X_LOCK_DECR;
|
|
|
|
lock->waiters = 0;
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
lock->sx_recursive = 0;
|
2016-11-23 11:07:17 +01:00
|
|
|
lock->writer_thread= 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
lock->m_rw_lock = true;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
UT_LIST_INIT(lock->debug_list, &rw_lock_debug_t::list);
|
|
|
|
|
|
|
|
lock->m_id = sync_latch_get_id(sync_latch_get_name(level));
|
|
|
|
ut_a(lock->m_id != LATCH_ID_NONE);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
lock->level = level;
|
|
|
|
#endif /* UNIV_DEBUG */
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
lock->cfile_name = cfile_name;
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
/* This should hold in practice. If it doesn't then we need to
|
|
|
|
split the source file anyway. Or create the locks on lines
|
|
|
|
less than 8192. cline is unsigned:13. */
|
|
|
|
ut_ad(cline <= 8192);
|
2017-03-01 07:27:39 +01:00
|
|
|
lock->cline = cline;
|
2014-02-26 19:11:54 +01:00
|
|
|
lock->count_os_wait = 0;
|
|
|
|
lock->last_s_file_name = "not yet reserved";
|
|
|
|
lock->last_x_file_name = "not yet reserved";
|
|
|
|
lock->last_s_line = 0;
|
|
|
|
lock->last_x_line = 0;
|
2016-08-12 10:17:45 +02:00
|
|
|
lock->event = os_event_create(0);
|
|
|
|
lock->wait_ex_event = os_event_create(0);
|
|
|
|
|
|
|
|
lock->is_block_lock = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
mutex_enter(&rw_lock_list_mutex);
|
|
|
|
|
|
|
|
ut_ad(UT_LIST_GET_FIRST(rw_lock_list) == NULL
|
|
|
|
|| UT_LIST_GET_FIRST(rw_lock_list)->magic_n == RW_LOCK_MAGIC_N);
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
UT_LIST_ADD_FIRST(rw_lock_list, lock);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
mutex_exit(&rw_lock_list_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Calling this function is obligatory only if the memory buffer containing
|
|
|
|
the rw-lock is freed. Removes an rw-lock object from the global list. The
|
|
|
|
rw-lock is checked to be in the non-locked state. */
|
|
|
|
void
|
|
|
|
rw_lock_free_func(
|
|
|
|
/*==============*/
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_t* lock) /*!< in/out: rw-lock */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
|
|
|
ut_ad(rw_lock_validate(lock));
|
|
|
|
ut_a(lock->lock_word == X_LOCK_DECR);
|
|
|
|
|
|
|
|
mutex_enter(&rw_lock_list_mutex);
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
os_event_destroy(lock->event);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
os_event_destroy(lock->wait_ex_event);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
UT_LIST_REMOVE(rw_lock_list, lock);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
mutex_exit(&rw_lock_list_mutex);
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
/* We did an in-place new in rw_lock_create_func() */
|
|
|
|
ut_d(lock->~rw_lock_t());
|
2017-05-10 13:14:50 +02:00
|
|
|
/* Sometimes (maybe when compiled with GCC -O3) the above call
|
|
|
|
to rw_lock_t::~rw_lock_t() will not actually assign magic_n=0. */
|
|
|
|
ut_d(lock->magic_n = 0);
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Lock an rw-lock in shared mode for the current thread. If the rw-lock is
|
|
|
|
locked in exclusive mode, or there is an exclusive lock request waiting,
|
2016-08-12 10:17:45 +02:00
|
|
|
the function spins a preset time (controlled by srv_n_spin_wait_rounds), waiting
|
2014-02-26 19:11:54 +01:00
|
|
|
for the lock, before suspending the thread. */
|
|
|
|
void
|
|
|
|
rw_lock_s_lock_spin(
|
|
|
|
/*================*/
|
|
|
|
rw_lock_t* lock, /*!< in: pointer to rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value; != 0, if the lock
|
|
|
|
will be passed to another thread to unlock */
|
|
|
|
const char* file_name, /*!< in: file name where lock requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
|
|
|
ulint i = 0; /* spin round count */
|
|
|
|
sync_array_t* sync_arr;
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint spin_count = 0;
|
2016-09-06 08:43:16 +02:00
|
|
|
uint64_t count_os_wait = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* We reuse the thread id to index into the counter, cache
|
|
|
|
it here for efficiency. */
|
|
|
|
|
|
|
|
ut_ad(rw_lock_validate(lock));
|
|
|
|
|
|
|
|
lock_loop:
|
|
|
|
|
|
|
|
/* Spin waiting for the writer field to become free */
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_low();
|
2016-08-12 10:17:45 +02:00
|
|
|
while (i < srv_n_spin_wait_rounds && lock->lock_word <= 0) {
|
2014-02-26 19:11:54 +01:00
|
|
|
if (srv_spin_wait_delay) {
|
|
|
|
ut_delay(ut_rnd_interval(0, srv_spin_wait_delay));
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_medium();
|
2016-08-12 10:17:45 +02:00
|
|
|
if (i >= srv_n_spin_wait_rounds) {
|
2014-02-26 19:11:54 +01:00
|
|
|
os_thread_yield();
|
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
++spin_count;
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
/* We try once again to obtain the lock */
|
2016-08-12 10:17:45 +02:00
|
|
|
if (rw_lock_s_lock_low(lock, pass, file_name, line)) {
|
|
|
|
|
|
|
|
if (count_os_wait > 0) {
|
2016-09-06 08:43:16 +02:00
|
|
|
lock->count_os_wait +=
|
|
|
|
static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_s_os_wait_count.add(count_os_wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
rw_lock_stats.rw_s_spin_round_count.add(spin_count);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
return; /* Success */
|
|
|
|
} else {
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
if (i < srv_n_spin_wait_rounds) {
|
2014-02-26 19:11:54 +01:00
|
|
|
goto lock_loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
++count_os_wait;
|
|
|
|
|
|
|
|
sync_cell_t* cell;
|
|
|
|
|
|
|
|
sync_arr = sync_array_get_and_reserve_cell(
|
|
|
|
lock, RW_LOCK_S, file_name, line, &cell);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* Set waiters before checking lock_word to ensure wake-up
|
|
|
|
signal is sent. This may lead to some unnecessary signals. */
|
2016-11-24 15:18:00 +01:00
|
|
|
my_atomic_fas32((int32*) &lock->waiters, 1);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
if (rw_lock_s_lock_low(lock, pass, file_name, line)) {
|
|
|
|
|
|
|
|
sync_array_free_cell(sync_arr, cell);
|
|
|
|
|
|
|
|
if (count_os_wait > 0) {
|
|
|
|
|
2016-09-06 08:43:16 +02:00
|
|
|
lock->count_os_wait +=
|
|
|
|
static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
rw_lock_stats.rw_s_os_wait_count.add(
|
|
|
|
count_os_wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
rw_lock_stats.rw_s_spin_round_count.add(spin_count);
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
return; /* Success */
|
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
/* see comments in trx_commit_low() to
|
|
|
|
before_trx_state_committed_in_memory explaining
|
|
|
|
this care to invoke the following sync check.*/
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
if (lock->get_level() != SYNC_DICT_OPERATION) {
|
|
|
|
DEBUG_SYNC_C("rw_s_lock_waiting");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
sync_array_wait_event(sync_arr, cell);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
i = 0;
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
goto lock_loop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
This function is used in the insert buffer to move the ownership of an
|
|
|
|
x-latch on a buffer frame to the current thread. The x-latch was set by
|
|
|
|
the buffer read operation and it protected the buffer frame while the
|
|
|
|
read was done. The ownership is moved because we want that the current
|
|
|
|
thread is able to acquire a second x-latch which is stored in an mtr.
|
|
|
|
This, in turn, is needed to pass the debug checks of index page
|
|
|
|
operations. */
|
|
|
|
void
|
|
|
|
rw_lock_x_lock_move_ownership(
|
|
|
|
/*==========================*/
|
|
|
|
rw_lock_t* lock) /*!< in: lock which was x-locked in the
|
|
|
|
buffer read */
|
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_ad(rw_lock_is_locked(lock, RW_LOCK_X));
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-11-23 11:07:17 +01:00
|
|
|
lock->writer_thread = os_thread_get_curr_id();
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Function for the next writer to call. Waits for readers to exit.
|
|
|
|
The caller must have already decremented lock_word by X_LOCK_DECR. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_x_lock_wait_func(
|
|
|
|
/*=====================*/
|
2014-02-26 19:11:54 +01:00
|
|
|
rw_lock_t* lock, /*!< in: pointer to rw-lock */
|
2016-08-12 10:17:45 +02:00
|
|
|
#ifdef UNIV_DEBUG
|
2014-02-26 19:11:54 +01:00
|
|
|
ulint pass, /*!< in: pass value; != 0, if the lock will
|
|
|
|
be passed to another thread to unlock */
|
|
|
|
#endif
|
2016-08-12 10:17:45 +02:00
|
|
|
lint threshold,/*!< in: threshold to wait for */
|
2014-02-26 19:11:54 +01:00
|
|
|
const char* file_name,/*!< in: file name where lock requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
|
|
|
ulint i = 0;
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint n_spins = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
sync_array_t* sync_arr;
|
2016-09-06 08:43:16 +02:00
|
|
|
uint64_t count_os_wait = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_ad(lock->lock_word <= threshold);
|
|
|
|
|
|
|
|
while (lock->lock_word < threshold) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-09-06 08:43:16 +02:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
HMT_low();
|
2014-02-26 19:11:54 +01:00
|
|
|
if (srv_spin_wait_delay) {
|
|
|
|
ut_delay(ut_rnd_interval(0, srv_spin_wait_delay));
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
if (i < srv_n_spin_wait_rounds) {
|
2014-02-26 19:11:54 +01:00
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_medium();
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* If there is still a reader, then go to sleep.*/
|
2016-08-12 10:17:45 +02:00
|
|
|
++n_spins;
|
|
|
|
|
|
|
|
sync_cell_t* cell;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
sync_arr = sync_array_get_and_reserve_cell(
|
|
|
|
lock, RW_LOCK_X_WAIT, file_name, line, &cell);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
i = 0;
|
|
|
|
|
|
|
|
/* Check lock_word to ensure wake-up isn't missed.*/
|
2016-08-12 10:17:45 +02:00
|
|
|
if (lock->lock_word < threshold) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
++count_os_wait;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* Add debug info as it is needed to detect possible
|
|
|
|
deadlock. We must add info for WAIT_EX thread for
|
|
|
|
deadlock detection to work properly. */
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_d(rw_lock_add_debug_info(
|
|
|
|
lock, pass, RW_LOCK_X_WAIT,
|
|
|
|
file_name, line));
|
|
|
|
|
|
|
|
sync_array_wait_event(sync_arr, cell);
|
|
|
|
|
|
|
|
ut_d(rw_lock_remove_debug_info(
|
|
|
|
lock, pass, RW_LOCK_X_WAIT));
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* It is possible to wake when lock_word < 0.
|
|
|
|
We must pass the while-loop check to proceed.*/
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
} else {
|
2016-08-12 10:17:45 +02:00
|
|
|
sync_array_free_cell(sync_arr, cell);
|
|
|
|
break;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_low();
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_medium();
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_x_spin_round_count.add(n_spins);
|
|
|
|
|
|
|
|
if (count_os_wait > 0) {
|
2017-01-25 09:11:37 +01:00
|
|
|
lock->count_os_wait += static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_x_os_wait_count.add(count_os_wait);
|
|
|
|
}
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
# define rw_lock_x_lock_wait(L, P, T, F, O) \
|
|
|
|
rw_lock_x_lock_wait_func(L, P, T, F, O)
|
|
|
|
#else
|
|
|
|
# define rw_lock_x_lock_wait(L, P, T, F, O) \
|
|
|
|
rw_lock_x_lock_wait_func(L, T, F, O)
|
|
|
|
#endif /* UNIV_DBEUG */
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
/******************************************************************//**
|
|
|
|
Low-level function for acquiring an exclusive lock.
|
2016-08-12 10:17:45 +02:00
|
|
|
@return FALSE if did not succeed, TRUE if success. */
|
2014-02-26 19:11:54 +01:00
|
|
|
UNIV_INLINE
|
|
|
|
ibool
|
|
|
|
rw_lock_x_lock_low(
|
|
|
|
/*===============*/
|
|
|
|
rw_lock_t* lock, /*!< in: pointer to rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value; != 0, if the lock will
|
|
|
|
be passed to another thread to unlock */
|
|
|
|
const char* file_name,/*!< in: file name where lock requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
if (rw_lock_lock_word_decr(lock, X_LOCK_DECR, X_LOCK_HALF_DECR)) {
|
2015-03-05 12:30:11 +01:00
|
|
|
|
2016-11-23 11:07:17 +01:00
|
|
|
/* As we are going to write our own thread id in that field it
|
|
|
|
must be that the current writer_thread value is not active. */
|
|
|
|
ut_a(!lock->writer_thread);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* Decrement occurred: we are writer or next-writer. */
|
2016-11-23 11:07:17 +01:00
|
|
|
if (!pass)
|
|
|
|
{
|
|
|
|
lock->writer_thread = os_thread_get_curr_id();
|
|
|
|
}
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_x_lock_wait(lock, pass, 0, file_name, line);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
} else {
|
|
|
|
os_thread_id_t thread_id = os_thread_get_curr_id();
|
|
|
|
|
2016-09-06 08:43:16 +02:00
|
|
|
/* Decrement failed: An X or SX lock is held by either
|
2016-11-23 11:07:17 +01:00
|
|
|
this thread or another. Try to relock. */
|
|
|
|
if (!pass && os_thread_eq(lock->writer_thread, thread_id)) {
|
2016-08-12 10:17:45 +02:00
|
|
|
/* Other s-locks can be allowed. If it is request x
|
|
|
|
recursively while holding sx lock, this x lock should
|
|
|
|
be along with the latching-order. */
|
|
|
|
|
|
|
|
/* The existing X or SX lock is from this thread */
|
|
|
|
if (rw_lock_lock_word_decr(lock, X_LOCK_DECR, 0)) {
|
|
|
|
/* There is at least one SX-lock from this
|
|
|
|
thread, but no X-lock. */
|
|
|
|
|
|
|
|
/* Wait for any the other S-locks to be
|
|
|
|
released. */
|
|
|
|
rw_lock_x_lock_wait(
|
|
|
|
lock, pass, -X_LOCK_HALF_DECR,
|
|
|
|
file_name, line);
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
} else {
|
2016-08-12 10:17:45 +02:00
|
|
|
/* At least one X lock by this thread already
|
|
|
|
exists. Add another. */
|
|
|
|
if (lock->lock_word == 0
|
|
|
|
|| lock->lock_word == -X_LOCK_HALF_DECR) {
|
|
|
|
lock->lock_word -= X_LOCK_DECR;
|
|
|
|
} else {
|
|
|
|
ut_ad(lock->lock_word <= -X_LOCK_DECR);
|
|
|
|
--lock->lock_word;
|
|
|
|
}
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
/* Another thread locked before us */
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
ut_d(rw_lock_add_debug_info(lock, pass, RW_LOCK_X, file_name, line));
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
lock->last_x_file_name = file_name;
|
2017-03-01 07:27:39 +01:00
|
|
|
lock->last_x_line = line;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
/******************************************************************//**
|
|
|
|
Low-level function for acquiring an sx lock.
|
|
|
|
@return FALSE if did not succeed, TRUE if success. */
|
|
|
|
ibool
|
|
|
|
rw_lock_sx_lock_low(
|
|
|
|
/*================*/
|
|
|
|
rw_lock_t* lock, /*!< in: pointer to rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value; != 0, if the lock will
|
|
|
|
be passed to another thread to unlock */
|
|
|
|
const char* file_name,/*!< in: file name where lock requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2016-08-12 10:17:45 +02:00
|
|
|
{
|
|
|
|
if (rw_lock_lock_word_decr(lock, X_LOCK_HALF_DECR, X_LOCK_HALF_DECR)) {
|
|
|
|
|
2016-11-23 11:07:17 +01:00
|
|
|
/* As we are going to write our own thread id in that field it
|
|
|
|
must be that the current writer_thread value is not active. */
|
|
|
|
ut_a(!lock->writer_thread);
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
/* Decrement occurred: we are the SX lock owner. */
|
2016-11-23 11:07:17 +01:00
|
|
|
if (!pass)
|
|
|
|
{
|
|
|
|
lock->writer_thread = os_thread_get_curr_id();
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
lock->sx_recursive = 1;
|
|
|
|
} else {
|
|
|
|
os_thread_id_t thread_id = os_thread_get_curr_id();
|
|
|
|
|
|
|
|
/* Decrement failed: It already has an X or SX lock by this
|
|
|
|
thread or another thread. If it is this thread, relock,
|
2016-11-23 11:07:17 +01:00
|
|
|
else fail. */
|
|
|
|
if (!pass && os_thread_eq(lock->writer_thread, thread_id)) {
|
2016-08-12 10:17:45 +02:00
|
|
|
/* This thread owns an X or SX lock */
|
|
|
|
if (lock->sx_recursive++ == 0) {
|
|
|
|
/* This thread is making first SX-lock request
|
|
|
|
and it must be holding at least one X-lock here
|
|
|
|
because:
|
|
|
|
|
|
|
|
* There can't be a WAIT_EX thread because we are
|
|
|
|
the thread which has it's thread_id written in
|
|
|
|
the writer_thread field and we are not waiting.
|
|
|
|
|
|
|
|
* Any other X-lock thread cannot exist because
|
|
|
|
it must update recursive flag only after
|
|
|
|
updating the thread_id. Had there been
|
|
|
|
a concurrent X-locking thread which succeeded
|
|
|
|
in decrementing the lock_word it must have
|
|
|
|
written it's thread_id before setting the
|
|
|
|
recursive flag. As we cleared the if()
|
|
|
|
condition above therefore we must be the only
|
|
|
|
thread working on this lock and it is safe to
|
|
|
|
read and write to the lock_word. */
|
|
|
|
|
|
|
|
ut_ad((lock->lock_word == 0)
|
|
|
|
|| ((lock->lock_word <= -X_LOCK_DECR)
|
|
|
|
&& (lock->lock_word
|
|
|
|
> -(X_LOCK_DECR
|
|
|
|
+ X_LOCK_HALF_DECR))));
|
|
|
|
lock->lock_word -= X_LOCK_HALF_DECR;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Another thread locked before us */
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ut_d(rw_lock_add_debug_info(lock, pass, RW_LOCK_SX, file_name, line));
|
|
|
|
|
|
|
|
lock->last_x_file_name = file_name;
|
2017-03-01 07:27:39 +01:00
|
|
|
lock->last_x_line = line;
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
/******************************************************************//**
|
|
|
|
NOTE! Use the corresponding macro, not directly this function! Lock an
|
|
|
|
rw-lock in exclusive mode for the current thread. If the rw-lock is locked
|
|
|
|
in shared or exclusive mode, or there is an exclusive lock request waiting,
|
2016-08-12 10:17:45 +02:00
|
|
|
the function spins a preset time (controlled by srv_n_spin_wait_rounds), waiting
|
2014-02-26 19:11:54 +01:00
|
|
|
for the lock before suspending the thread. If the same thread has an x-lock
|
|
|
|
on the rw-lock, locking succeed, with the following exception: if pass != 0,
|
|
|
|
only a single x-lock may be taken on the lock. NOTE: If the same thread has
|
|
|
|
an s-lock, locking does not succeed! */
|
|
|
|
void
|
|
|
|
rw_lock_x_lock_func(
|
|
|
|
/*================*/
|
|
|
|
rw_lock_t* lock, /*!< in: pointer to rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value; != 0, if the lock will
|
|
|
|
be passed to another thread to unlock */
|
|
|
|
const char* file_name,/*!< in: file name where lock requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint i = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
sync_array_t* sync_arr;
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint spin_count = 0;
|
2016-09-06 08:43:16 +02:00
|
|
|
uint64_t count_os_wait = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
ut_ad(rw_lock_validate(lock));
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_ad(!rw_lock_own(lock, RW_LOCK_S));
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
lock_loop:
|
|
|
|
|
|
|
|
if (rw_lock_x_lock_low(lock, pass, file_name, line)) {
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
if (count_os_wait > 0) {
|
2016-09-06 08:43:16 +02:00
|
|
|
lock->count_os_wait +=
|
|
|
|
static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_x_os_wait_count.add(count_os_wait);
|
|
|
|
}
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_x_spin_round_count.add(spin_count);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
/* Locking succeeded */
|
|
|
|
return;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
} else {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* Spin waiting for the lock_word to become free */
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_low();
|
2016-08-12 10:17:45 +02:00
|
|
|
while (i < srv_n_spin_wait_rounds
|
|
|
|
&& lock->lock_word <= X_LOCK_HALF_DECR) {
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
if (srv_spin_wait_delay) {
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_delay(ut_rnd_interval(
|
|
|
|
0, srv_spin_wait_delay));
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-08-19 20:35:14 +02:00
|
|
|
HMT_medium();
|
2016-08-12 10:17:45 +02:00
|
|
|
spin_count += i;
|
|
|
|
|
|
|
|
if (i >= srv_n_spin_wait_rounds) {
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
os_thread_yield();
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
} else {
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
goto lock_loop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
sync_cell_t* cell;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
sync_arr = sync_array_get_and_reserve_cell(
|
|
|
|
lock, RW_LOCK_X, file_name, line, &cell);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* Waiters must be set before checking lock_word, to ensure signal
|
|
|
|
is sent. This could lead to a few unnecessary wake-up signals. */
|
2016-11-24 15:18:00 +01:00
|
|
|
my_atomic_fas32((int32*) &lock->waiters, 1);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
if (rw_lock_x_lock_low(lock, pass, file_name, line)) {
|
2016-08-12 10:17:45 +02:00
|
|
|
sync_array_free_cell(sync_arr, cell);
|
|
|
|
|
|
|
|
if (count_os_wait > 0) {
|
2016-09-06 08:43:16 +02:00
|
|
|
lock->count_os_wait +=
|
|
|
|
static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_x_os_wait_count.add(count_os_wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
rw_lock_stats.rw_x_spin_round_count.add(spin_count);
|
|
|
|
|
|
|
|
/* Locking succeeded */
|
|
|
|
return;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
++count_os_wait;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
sync_array_wait_event(sync_arr, cell);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
i = 0;
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
goto lock_loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
2016-08-12 10:17:45 +02:00
|
|
|
NOTE! Use the corresponding macro, not directly this function! Lock an
|
|
|
|
rw-lock in SX mode for the current thread. If the rw-lock is locked
|
|
|
|
in exclusive mode, or there is an exclusive lock request waiting,
|
|
|
|
the function spins a preset time (controlled by SYNC_SPIN_ROUNDS), waiting
|
|
|
|
for the lock, before suspending the thread. If the same thread has an x-lock
|
|
|
|
on the rw-lock, locking succeed, with the following exception: if pass != 0,
|
|
|
|
only a single sx-lock may be taken on the lock. NOTE: If the same thread has
|
|
|
|
an s-lock, locking does not succeed! */
|
2014-02-26 19:11:54 +01:00
|
|
|
void
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_sx_lock_func(
|
|
|
|
/*=================*/
|
|
|
|
rw_lock_t* lock, /*!< in: pointer to rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value; != 0, if the lock will
|
|
|
|
be passed to another thread to unlock */
|
|
|
|
const char* file_name,/*!< in: file name where lock requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint i = 0;
|
|
|
|
sync_array_t* sync_arr;
|
|
|
|
ulint spin_count = 0;
|
2016-09-06 08:43:16 +02:00
|
|
|
uint64_t count_os_wait = 0;
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint spin_wait_count = 0;
|
|
|
|
|
|
|
|
ut_ad(rw_lock_validate(lock));
|
|
|
|
ut_ad(!rw_lock_own(lock, RW_LOCK_S));
|
|
|
|
|
|
|
|
lock_loop:
|
|
|
|
|
|
|
|
if (rw_lock_sx_lock_low(lock, pass, file_name, line)) {
|
|
|
|
|
|
|
|
if (count_os_wait > 0) {
|
2016-09-06 08:43:16 +02:00
|
|
|
lock->count_os_wait +=
|
|
|
|
static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_sx_os_wait_count.add(count_os_wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
rw_lock_stats.rw_sx_spin_round_count.add(spin_count);
|
|
|
|
rw_lock_stats.rw_sx_spin_wait_count.add(spin_wait_count);
|
|
|
|
|
|
|
|
/* Locking succeeded */
|
|
|
|
return;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
++spin_wait_count;
|
|
|
|
|
|
|
|
/* Spin waiting for the lock_word to become free */
|
|
|
|
while (i < srv_n_spin_wait_rounds
|
|
|
|
&& lock->lock_word <= X_LOCK_HALF_DECR) {
|
|
|
|
|
|
|
|
if (srv_spin_wait_delay) {
|
|
|
|
ut_delay(ut_rnd_interval(
|
|
|
|
0, srv_spin_wait_delay));
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_count += i;
|
|
|
|
|
|
|
|
if (i >= srv_n_spin_wait_rounds) {
|
|
|
|
|
|
|
|
os_thread_yield();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
goto lock_loop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sync_cell_t* cell;
|
|
|
|
|
|
|
|
sync_arr = sync_array_get_and_reserve_cell(
|
|
|
|
lock, RW_LOCK_SX, file_name, line, &cell);
|
|
|
|
|
|
|
|
/* Waiters must be set before checking lock_word, to ensure signal
|
|
|
|
is sent. This could lead to a few unnecessary wake-up signals. */
|
2016-11-24 15:18:00 +01:00
|
|
|
my_atomic_fas32((int32*) &lock->waiters, 1);
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
if (rw_lock_sx_lock_low(lock, pass, file_name, line)) {
|
|
|
|
|
|
|
|
sync_array_free_cell(sync_arr, cell);
|
|
|
|
|
|
|
|
if (count_os_wait > 0) {
|
2016-09-06 08:43:16 +02:00
|
|
|
lock->count_os_wait +=
|
|
|
|
static_cast<uint32_t>(count_os_wait);
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_stats.rw_sx_os_wait_count.add(count_os_wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
rw_lock_stats.rw_sx_spin_round_count.add(spin_count);
|
|
|
|
rw_lock_stats.rw_sx_spin_wait_count.add(spin_wait_count);
|
|
|
|
|
|
|
|
/* Locking succeeded */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
++count_os_wait;
|
|
|
|
|
|
|
|
sync_array_wait_event(sync_arr, cell);
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
|
|
|
|
goto lock_loop;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
/******************************************************************//**
|
2016-08-12 10:17:45 +02:00
|
|
|
Checks that the rw-lock has been initialized and that there are no
|
|
|
|
simultaneous shared and exclusive locks.
|
|
|
|
@return true */
|
|
|
|
bool
|
|
|
|
rw_lock_validate(
|
|
|
|
/*=============*/
|
|
|
|
const rw_lock_t* lock) /*!< in: rw-lock */
|
|
|
|
{
|
|
|
|
lint lock_word;
|
|
|
|
|
|
|
|
ut_ad(lock);
|
|
|
|
|
|
|
|
lock_word = lock->lock_word;
|
|
|
|
|
|
|
|
ut_ad(lock->magic_n == RW_LOCK_MAGIC_N);
|
2016-11-22 11:19:54 +01:00
|
|
|
ut_ad(lock->waiters < 2);
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_ad(lock_word > -(2 * X_LOCK_DECR));
|
|
|
|
ut_ad(lock_word <= X_LOCK_DECR);
|
|
|
|
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Checks if somebody has locked the rw-lock in the specified mode.
|
|
|
|
@return true if locked */
|
|
|
|
bool
|
|
|
|
rw_lock_is_locked(
|
|
|
|
/*==============*/
|
|
|
|
rw_lock_t* lock, /*!< in: rw-lock */
|
|
|
|
ulint lock_type) /*!< in: lock type: RW_LOCK_S,
|
|
|
|
RW_LOCK_X or RW_LOCK_SX */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_ad(rw_lock_validate(lock));
|
|
|
|
|
|
|
|
switch (lock_type) {
|
|
|
|
case RW_LOCK_S:
|
|
|
|
return(rw_lock_get_reader_count(lock) > 0);
|
|
|
|
|
|
|
|
case RW_LOCK_X:
|
|
|
|
return(rw_lock_get_writer(lock) == RW_LOCK_X);
|
|
|
|
|
|
|
|
case RW_LOCK_SX:
|
|
|
|
return(rw_lock_get_sx_lock_count(lock) > 0);
|
|
|
|
|
|
|
|
default:
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
return(false); /* avoid compiler warnings */
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Inserts the debug information for an rw-lock. */
|
|
|
|
void
|
|
|
|
rw_lock_add_debug_info(
|
|
|
|
/*===================*/
|
|
|
|
rw_lock_t* lock, /*!< in: rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value */
|
|
|
|
ulint lock_type, /*!< in: lock type */
|
|
|
|
const char* file_name, /*!< in: file where requested */
|
2017-03-01 07:27:39 +01:00
|
|
|
unsigned line) /*!< in: line where requested */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ut_ad(file_name != NULL);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_debug_t* info = rw_lock_debug_create();
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
rw_lock_debug_mutex_enter();
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
info->pass = pass;
|
2014-02-26 19:11:54 +01:00
|
|
|
info->line = line;
|
|
|
|
info->lock_type = lock_type;
|
2016-08-12 10:17:45 +02:00
|
|
|
info->file_name = file_name;
|
2014-02-26 19:11:54 +01:00
|
|
|
info->thread_id = os_thread_get_curr_id();
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
UT_LIST_ADD_FIRST(lock->debug_list, info);
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
if (pass == 0 && lock_type != RW_LOCK_X_WAIT) {
|
|
|
|
|
|
|
|
/* Recursive x while holding SX
|
|
|
|
(lock_type == RW_LOCK_X && lock_word == -X_LOCK_HALF_DECR)
|
|
|
|
is treated as not-relock (new lock). */
|
|
|
|
|
|
|
|
if ((lock_type == RW_LOCK_X
|
|
|
|
&& lock->lock_word < -X_LOCK_HALF_DECR)
|
|
|
|
|| (lock_type == RW_LOCK_SX
|
|
|
|
&& (lock->lock_word < 0 || lock->sx_recursive == 1))) {
|
|
|
|
|
|
|
|
sync_check_lock_validate(lock);
|
|
|
|
sync_check_lock_granted(lock);
|
|
|
|
} else {
|
|
|
|
sync_check_relock(lock);
|
|
|
|
}
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Removes a debug information struct for an rw-lock. */
|
|
|
|
void
|
|
|
|
rw_lock_remove_debug_info(
|
|
|
|
/*======================*/
|
|
|
|
rw_lock_t* lock, /*!< in: rw-lock */
|
|
|
|
ulint pass, /*!< in: pass value */
|
|
|
|
ulint lock_type) /*!< in: lock type */
|
|
|
|
{
|
|
|
|
rw_lock_debug_t* info;
|
|
|
|
|
|
|
|
ut_ad(lock);
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
if (pass == 0 && lock_type != RW_LOCK_X_WAIT) {
|
|
|
|
sync_check_unlock(lock);
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
rw_lock_debug_mutex_enter();
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
for (info = UT_LIST_GET_FIRST(lock->debug_list);
|
|
|
|
info != 0;
|
|
|
|
info = UT_LIST_GET_NEXT(list, info)) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
if (pass == info->pass
|
|
|
|
&& (pass != 0
|
2014-02-26 19:11:54 +01:00
|
|
|
|| os_thread_eq(info->thread_id,
|
|
|
|
os_thread_get_curr_id()))
|
2016-08-12 10:17:45 +02:00
|
|
|
&& info->lock_type == lock_type) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
/* Found! */
|
2016-08-12 10:17:45 +02:00
|
|
|
UT_LIST_REMOVE(lock->debug_list, info);
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
|
|
|
|
rw_lock_debug_free(info);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ut_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************//**
|
|
|
|
Checks if the thread has locked the rw-lock in the specified mode, with
|
|
|
|
the pass value == 0.
|
2016-08-12 10:17:45 +02:00
|
|
|
@return TRUE if locked */
|
2014-02-26 19:11:54 +01:00
|
|
|
ibool
|
|
|
|
rw_lock_own(
|
|
|
|
/*========*/
|
|
|
|
rw_lock_t* lock, /*!< in: rw-lock */
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint lock_type) /*!< in: lock type: RW_LOCK_S,
|
|
|
|
RW_LOCK_X */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
|
|
|
ut_ad(lock);
|
|
|
|
ut_ad(rw_lock_validate(lock));
|
|
|
|
|
|
|
|
rw_lock_debug_mutex_enter();
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
for (const rw_lock_debug_t* info = UT_LIST_GET_FIRST(lock->debug_list);
|
|
|
|
info != NULL;
|
|
|
|
info = UT_LIST_GET_NEXT(list, info)) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
if (os_thread_eq(info->thread_id, os_thread_get_curr_id())
|
2016-08-12 10:17:45 +02:00
|
|
|
&& info->pass == 0
|
|
|
|
&& info->lock_type == lock_type) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
/* Found! */
|
|
|
|
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
/** Checks if the thread has locked the rw-lock in the specified mode, with
|
|
|
|
the pass value == 0.
|
|
|
|
@param[in] lock rw-lock
|
|
|
|
@param[in] flags specify lock types with OR of the
|
|
|
|
rw_lock_flag_t values
|
|
|
|
@return true if locked */
|
2018-08-03 14:40:13 +02:00
|
|
|
bool rw_lock_own_flagged(const rw_lock_t* lock, rw_lock_flags_t flags)
|
2016-08-12 10:17:45 +02:00
|
|
|
{
|
2018-08-03 14:40:13 +02:00
|
|
|
ut_ad(rw_lock_validate(lock));
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
rw_lock_debug_mutex_enter();
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
for (rw_lock_debug_t* info = UT_LIST_GET_FIRST(lock->debug_list);
|
|
|
|
info != NULL;
|
|
|
|
info = UT_LIST_GET_NEXT(list, info)) {
|
|
|
|
if (!os_thread_eq(info->thread_id, os_thread_get_curr_id())
|
|
|
|
|| info->pass) {
|
2016-08-12 10:17:45 +02:00
|
|
|
continue;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
switch (info->lock_type) {
|
|
|
|
case RW_LOCK_S:
|
2018-08-03 14:40:13 +02:00
|
|
|
if (!(flags & RW_LOCK_FLAG_S)) {
|
|
|
|
continue;
|
2016-08-12 10:17:45 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RW_LOCK_X:
|
2018-08-03 14:40:13 +02:00
|
|
|
if (!(flags & RW_LOCK_FLAG_X)) {
|
|
|
|
continue;
|
2016-08-12 10:17:45 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RW_LOCK_SX:
|
2018-08-03 14:40:13 +02:00
|
|
|
if (!(flags & RW_LOCK_FLAG_SX)) {
|
|
|
|
continue;
|
2016-08-12 10:17:45 +02:00
|
|
|
}
|
2018-08-03 14:40:13 +02:00
|
|
|
break;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
2018-08-03 14:40:13 +02:00
|
|
|
|
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
return true;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
return false;
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************//**
|
|
|
|
Prints debug info of currently locked rw-locks. */
|
|
|
|
void
|
|
|
|
rw_lock_list_print_info(
|
|
|
|
/*====================*/
|
|
|
|
FILE* file) /*!< in: file where to print */
|
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint count = 0;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
mutex_enter(&rw_lock_list_mutex);
|
|
|
|
|
|
|
|
fputs("-------------\n"
|
|
|
|
"RW-LATCH INFO\n"
|
|
|
|
"-------------\n", file);
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
for (const rw_lock_t* lock = UT_LIST_GET_FIRST(rw_lock_list);
|
|
|
|
lock != NULL;
|
|
|
|
lock = UT_LIST_GET_NEXT(list, lock)) {
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
count++;
|
|
|
|
|
|
|
|
if (lock->lock_word != X_LOCK_DECR) {
|
|
|
|
|
|
|
|
fprintf(file, "RW-LOCK: %p ", (void*) lock);
|
|
|
|
|
2016-11-22 11:19:54 +01:00
|
|
|
if (lock->waiters) {
|
2014-02-26 19:11:54 +01:00
|
|
|
fputs(" Waiters for the lock exist\n", file);
|
|
|
|
} else {
|
|
|
|
putc('\n', file);
|
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
rw_lock_debug_t* info;
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
rw_lock_debug_mutex_enter();
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
for (info = UT_LIST_GET_FIRST(lock->debug_list);
|
|
|
|
info != NULL;
|
|
|
|
info = UT_LIST_GET_NEXT(list, info)) {
|
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
rw_lock_debug_print(file, info);
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
rw_lock_debug_mutex_exit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-06 08:43:16 +02:00
|
|
|
fprintf(file, "Total number of rw-locks " ULINTPF "\n", count);
|
2014-02-26 19:11:54 +01:00
|
|
|
mutex_exit(&rw_lock_list_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Prints info of a debug struct. */
|
|
|
|
void
|
|
|
|
rw_lock_debug_print(
|
|
|
|
/*================*/
|
|
|
|
FILE* f, /*!< in: output stream */
|
2016-08-12 10:17:45 +02:00
|
|
|
const rw_lock_debug_t* info) /*!< in: debug struct */
|
2014-02-26 19:11:54 +01:00
|
|
|
{
|
2016-08-12 10:17:45 +02:00
|
|
|
ulint rwt = info->lock_type;
|
2014-02-26 19:11:54 +01:00
|
|
|
|
|
|
|
fprintf(f, "Locked: thread %lu file %s line %lu ",
|
2016-08-12 10:17:45 +02:00
|
|
|
static_cast<ulong>(os_thread_pf(info->thread_id)),
|
|
|
|
sync_basename(info->file_name),
|
|
|
|
static_cast<ulong>(info->line));
|
|
|
|
|
|
|
|
switch (rwt) {
|
|
|
|
case RW_LOCK_S:
|
2014-02-26 19:11:54 +01:00
|
|
|
fputs("S-LOCK", f);
|
2016-08-12 10:17:45 +02:00
|
|
|
break;
|
|
|
|
case RW_LOCK_X:
|
2014-02-26 19:11:54 +01:00
|
|
|
fputs("X-LOCK", f);
|
2016-08-12 10:17:45 +02:00
|
|
|
break;
|
|
|
|
case RW_LOCK_SX:
|
|
|
|
fputs("SX-LOCK", f);
|
|
|
|
break;
|
|
|
|
case RW_LOCK_X_WAIT:
|
2014-02-26 19:11:54 +01:00
|
|
|
fputs("WAIT X-LOCK", f);
|
2016-08-12 10:17:45 +02:00
|
|
|
break;
|
|
|
|
default:
|
2014-02-26 19:11:54 +01:00
|
|
|
ut_error;
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2014-02-26 19:11:54 +01:00
|
|
|
if (info->pass != 0) {
|
|
|
|
fprintf(f, " pass value %lu", (ulong) info->pass);
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
|
|
|
fprintf(f, "\n");
|
2014-02-26 19:11:54 +01:00
|
|
|
}
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
/** Print where it was locked from
|
|
|
|
@return the string representation */
|
|
|
|
std::string
|
|
|
|
rw_lock_t::locked_from() const
|
|
|
|
{
|
|
|
|
/* Note: For X locks it can be locked form multiple places because
|
|
|
|
the same thread can call X lock recursively. */
|
|
|
|
|
|
|
|
std::ostringstream msg;
|
2018-08-03 14:40:13 +02:00
|
|
|
bool written = false;
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
ut_ad(rw_lock_validate(this));
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
rw_lock_debug_mutex_enter();
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
for (rw_lock_debug_t* info = UT_LIST_GET_FIRST(debug_list);
|
|
|
|
info != NULL;
|
|
|
|
info = UT_LIST_GET_NEXT(list, info)) {
|
|
|
|
if (!os_thread_eq(info->thread_id, os_thread_get_curr_id())) {
|
|
|
|
continue;
|
|
|
|
}
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
if (written) {
|
2016-08-12 10:17:45 +02:00
|
|
|
msg << ", ";
|
|
|
|
}
|
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
written = true;
|
|
|
|
|
2016-08-12 10:17:45 +02:00
|
|
|
msg << info->file_name << ":" << info->line;
|
|
|
|
}
|
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
rw_lock_debug_mutex_exit();
|
2016-08-12 10:17:45 +02:00
|
|
|
|
2018-08-03 14:40:13 +02:00
|
|
|
return(msg.str());
|
2016-08-12 10:17:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Print the rw-lock information.
|
|
|
|
@return the string representation */
|
|
|
|
std::string
|
|
|
|
rw_lock_t::to_string() const
|
|
|
|
{
|
|
|
|
std::ostringstream msg;
|
|
|
|
|
|
|
|
msg << "RW-LATCH: "
|
|
|
|
<< "thread id " << os_thread_pf(os_thread_get_curr_id())
|
|
|
|
<< " addr: " << this
|
|
|
|
<< " Locked from: " << locked_from().c_str();
|
|
|
|
|
|
|
|
return(msg.str());
|
|
|
|
}
|
|
|
|
#endif /* UNIV_DEBUG */
|