mirror of
https://github.com/MariaDB/server.git
synced 2026-05-14 19:07:15 +02:00
We replace the old lock_sys.mutex (which was renamed to lock_sys.latch)
with a combination of a global lock_sys.latch and table or page hash lock
mutexes.
The global lock_sys.latch can be acquired in exclusive mode, or
it can be acquired in shared mode and another mutex will be acquired
to protect the locks for a particular page or a table.
This is inspired by
mysql/mysql-server@1d259b87a6
but the optimization of lock_release() will be done in the next commit.
Also, we will interleave mutexes with the hash table elements, similar
to how buf_pool.page_hash was optimized
in commit 5155a300fa (MDEV-22871).
dict_table_t::autoinc_trx: Use Atomic_relaxed.
dict_table_t::autoinc_mutex: Use srw_mutex in order to reduce the
memory footprint. On 64-bit Linux or OpenBSD, both this and the new
dict_table_t::lock_mutex should be 32 bits and be stored in the same
64-bit word. On Microsoft Windows, the underlying SRWLOCK is 32 or 64
bits, and on other systems, sizeof(pthread_mutex_t) can be much larger.
ib_lock_t::trx_locks, trx_lock_t::trx_locks: Document the new rules.
Writers must assert lock_sys.is_writer() || trx->mutex_is_owner().
LockGuard: A RAII wrapper for acquiring a page hash table lock.
LockGGuard: Like LockGuard, but when Galera Write-Set Replication
is enabled, we must acquire all shards, for updating arbitrary trx_locks.
LockMultiGuard: A RAII wrapper for acquiring two page hash table locks.
lock_rec_create_wsrep(), lock_table_create_wsrep(): Special
Galera conflict resolution in non-inlined functions in order
to keep the common code paths shorter.
lock_sys_t::prdt_page_free_from_discard(): Refactored from
lock_prdt_page_free_from_discard() and
lock_rec_free_all_from_discard_page().
trx_t::commit_tables(): Replaces trx_update_mod_tables_timestamp().
lock_release(): Let trx_t::commit_tables() invalidate the query cache
for those tables that were actually modified by the transaction.
Merge lock_check_dict_lock() to lock_release().
We must never release lock_sys.latch while holding any
lock_sys_t::hash_latch. Failure to do that could lead to
memory corruption if the buffer pool is resized between
the time lock_sys.latch is released and the hash_latch is released.
210 lines
6.1 KiB
C
210 lines
6.1 KiB
C
/*****************************************************************************
|
|
|
|
Copyright (c) 1997, 2016, Oracle and/or its affiliates. All Rights Reserved.
|
|
Copyright (c) 2018, 2021, 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 include/hash0hash.h
|
|
The simple hash table utility
|
|
|
|
Created 5/20/1997 Heikki Tuuri
|
|
*******************************************************/
|
|
|
|
#pragma once
|
|
#include "ut0rnd.h"
|
|
#include "ut0new.h"
|
|
|
|
struct hash_table_t;
|
|
struct hash_cell_t{
|
|
void* node; /*!< hash chain node, NULL if none */
|
|
};
|
|
typedef void* hash_node_t;
|
|
|
|
/*******************************************************************//**
|
|
Inserts a struct to a hash table. */
|
|
|
|
#define HASH_INSERT(TYPE, NAME, TABLE, FOLD, DATA)\
|
|
do {\
|
|
hash_cell_t* cell3333;\
|
|
TYPE* struct3333;\
|
|
\
|
|
(DATA)->NAME = NULL;\
|
|
\
|
|
cell3333 = &(TABLE)->array[(TABLE)->calc_hash(FOLD)]; \
|
|
\
|
|
if (cell3333->node == NULL) {\
|
|
cell3333->node = DATA;\
|
|
} else {\
|
|
struct3333 = (TYPE*) cell3333->node;\
|
|
\
|
|
while (struct3333->NAME != NULL) {\
|
|
\
|
|
struct3333 = (TYPE*) struct3333->NAME;\
|
|
}\
|
|
\
|
|
struct3333->NAME = DATA;\
|
|
}\
|
|
} while (0)
|
|
|
|
/*******************************************************************//**
|
|
Inserts a struct to the head of hash table. */
|
|
|
|
#define HASH_PREPEND(TYPE, NAME, TABLE, FOLD, DATA) \
|
|
do { \
|
|
hash_cell_t* cell3333; \
|
|
TYPE* struct3333; \
|
|
\
|
|
(DATA)->NAME = NULL; \
|
|
\
|
|
cell3333 = &(TABLE)->array[(TABLE)->calc_hash(FOLD)]; \
|
|
\
|
|
if (cell3333->node == NULL) { \
|
|
cell3333->node = DATA; \
|
|
DATA->NAME = NULL; \
|
|
} else { \
|
|
struct3333 = (TYPE*) cell3333->node; \
|
|
\
|
|
DATA->NAME = struct3333; \
|
|
\
|
|
cell3333->node = DATA; \
|
|
} \
|
|
} while (0)
|
|
#ifdef UNIV_HASH_DEBUG
|
|
# define HASH_ASSERT_VALID(DATA) ut_a((void*) (DATA) != (void*) -1)
|
|
# define HASH_INVALIDATE(DATA, NAME) *(void**) (&DATA->NAME) = (void*) -1
|
|
#else
|
|
# define HASH_ASSERT_VALID(DATA) do {} while (0)
|
|
# define HASH_INVALIDATE(DATA, NAME) do {} while (0)
|
|
#endif
|
|
|
|
/*******************************************************************//**
|
|
Deletes a struct from a hash table. */
|
|
|
|
#define HASH_DELETE(TYPE, NAME, TABLE, FOLD, DATA)\
|
|
do {\
|
|
hash_cell_t* cell3333;\
|
|
TYPE* struct3333;\
|
|
\
|
|
cell3333 = &(TABLE)->array[(TABLE)->calc_hash(FOLD)]; \
|
|
\
|
|
if (cell3333->node == DATA) {\
|
|
HASH_ASSERT_VALID(DATA->NAME);\
|
|
cell3333->node = DATA->NAME;\
|
|
} else {\
|
|
struct3333 = (TYPE*) cell3333->node;\
|
|
\
|
|
while (struct3333->NAME != DATA) {\
|
|
\
|
|
struct3333 = (TYPE*) struct3333->NAME;\
|
|
ut_a(struct3333);\
|
|
}\
|
|
\
|
|
struct3333->NAME = DATA->NAME;\
|
|
}\
|
|
HASH_INVALIDATE(DATA, NAME);\
|
|
} while (0)
|
|
|
|
#define HASH_REPLACE(TYPE, NAME, TABLE, FOLD, DATA_OLD, DATA_NEW) \
|
|
do { \
|
|
(DATA_NEW)->NAME = (DATA_OLD)->NAME; \
|
|
\
|
|
hash_cell_t& cell3333 \
|
|
= (TABLE)->array[(TABLE)->calc_hash(FOLD)]; \
|
|
TYPE** struct3333 = (TYPE**)&cell3333.node; \
|
|
while (*struct3333 != DATA_OLD) { \
|
|
struct3333 = &((*struct3333)->NAME); \
|
|
} \
|
|
*struct3333 = DATA_NEW; \
|
|
} while (0)
|
|
/*******************************************************************//**
|
|
Gets the first struct in a hash chain, NULL if none. */
|
|
|
|
#define HASH_GET_FIRST(TABLE, HASH_VAL) (TABLE)->array[HASH_VAL].node
|
|
|
|
/*******************************************************************//**
|
|
Gets the next struct in a hash chain, NULL if none. */
|
|
|
|
#define HASH_GET_NEXT(NAME, DATA) ((DATA)->NAME)
|
|
|
|
/********************************************************************//**
|
|
Looks for a struct in a hash table. */
|
|
#define HASH_SEARCH(NAME, TABLE, FOLD, TYPE, DATA, ASSERTION, TEST)\
|
|
{\
|
|
(DATA) = (TYPE) HASH_GET_FIRST(TABLE, (TABLE)->calc_hash(FOLD)); \
|
|
HASH_ASSERT_VALID(DATA);\
|
|
\
|
|
while ((DATA) != NULL) {\
|
|
ASSERTION;\
|
|
if (TEST) {\
|
|
break;\
|
|
} else {\
|
|
HASH_ASSERT_VALID(HASH_GET_NEXT(NAME, DATA));\
|
|
(DATA) = (TYPE) HASH_GET_NEXT(NAME, DATA);\
|
|
}\
|
|
}\
|
|
}
|
|
|
|
/********************************************************************//**
|
|
Looks for an item in all hash buckets. */
|
|
#define HASH_SEARCH_ALL(NAME, TABLE, TYPE, DATA, ASSERTION, TEST) \
|
|
do { \
|
|
ulint i3333; \
|
|
\
|
|
for (i3333 = (TABLE)->n_cells; i3333--; ) { \
|
|
(DATA) = (TYPE) HASH_GET_FIRST(TABLE, i3333); \
|
|
\
|
|
while ((DATA) != NULL) { \
|
|
HASH_ASSERT_VALID(DATA); \
|
|
ASSERTION; \
|
|
\
|
|
if (TEST) { \
|
|
break; \
|
|
} \
|
|
\
|
|
(DATA) = (TYPE) HASH_GET_NEXT(NAME, DATA); \
|
|
} \
|
|
\
|
|
if ((DATA) != NULL) { \
|
|
break; \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
/** Hash table with singly-linked overflow lists */
|
|
struct hash_table_t
|
|
{
|
|
/** number of elements in array (a prime number) */
|
|
ulint n_cells;
|
|
/** the hash array */
|
|
hash_cell_t *array;
|
|
|
|
/** Create the hash table.
|
|
@param n the lower bound of n_cells */
|
|
void create(ulint n)
|
|
{
|
|
n_cells= ut_find_prime(n);
|
|
array= static_cast<hash_cell_t*>(ut_zalloc_nokey(n_cells * sizeof *array));
|
|
}
|
|
|
|
/** Clear the hash table. */
|
|
void clear() { memset(array, 0, n_cells * sizeof *array); }
|
|
|
|
/** Free the hash table. */
|
|
void free() { ut_free(array); array= nullptr; }
|
|
|
|
ulint calc_hash(ulint fold) const { return ut_hash_ulint(fold, n_cells); }
|
|
};
|