Commit 00cd53d3 authored by Marko Mäkelä's avatar Marko Mäkelä

MDEV-23719: Make lock_sys use page_id_t

Since commit 8ccb3caa it should be
more efficient to use page_id_t rather than two separate variables
for tablespace identifier and page number.

lock_rec_fold(): Replaced with page_id_t::fold().

lock_rec_hash(): Replaced with lock_sys.hash(page_id).

lock_rec_expl_exist_on_page(), lock_rec_get_first_on_page_addr(),
lock_rec_get_first_on_page(): Replaced with lock_sys.get_first().
parent 852771ba
...@@ -5423,8 +5423,7 @@ btr_cur_compress_if_useful( ...@@ -5423,8 +5423,7 @@ btr_cur_compress_if_useful(
const buf_block_t* block = btr_cur_get_block(cursor); const buf_block_t* block = btr_cur_get_block(cursor);
/* Check whether page lock prevents the compression */ /* Check whether page lock prevents the compression */
if (!lock_test_prdt_page_lock(trx, block->page.id().space(), if (!lock_test_prdt_page_lock(trx, block->page.id())) {
block->page.id().page_no())) {
return(false); return(false);
} }
} }
......
...@@ -652,8 +652,7 @@ rtr_adjust_upper_level( ...@@ -652,8 +652,7 @@ rtr_adjust_upper_level(
new_prdt.op = 0; new_prdt.op = 0;
lock_prdt_update_parent(block, new_block, &prdt, &new_prdt, lock_prdt_update_parent(block, new_block, &prdt, &new_prdt,
index->table->space_id, page_cursor->block->page.id());
page_cursor->block->page.id().page_no());
mem_heap_free(heap); mem_heap_free(heap);
...@@ -881,8 +880,6 @@ rtr_page_split_and_insert( ...@@ -881,8 +880,6 @@ rtr_page_split_and_insert(
buf_block_t* block; buf_block_t* block;
page_t* page; page_t* page;
page_t* new_page; page_t* new_page;
ulint page_no;
ulint hint_page_no;
buf_block_t* new_block; buf_block_t* new_block;
page_zip_des_t* page_zip; page_zip_des_t* page_zip;
page_zip_des_t* new_page_zip; page_zip_des_t* new_page_zip;
...@@ -931,7 +928,7 @@ rtr_page_split_and_insert( ...@@ -931,7 +928,7 @@ rtr_page_split_and_insert(
ut_ad(mtr->memo_contains_flagged(block, MTR_MEMO_PAGE_X_FIX)); ut_ad(mtr->memo_contains_flagged(block, MTR_MEMO_PAGE_X_FIX));
ut_ad(page_get_n_recs(page) >= 1); ut_ad(page_get_n_recs(page) >= 1);
page_no = block->page.id().page_no(); const page_id_t page_id(block->page.id());
if (!page_has_prev(page) && !page_is_leaf(page)) { if (!page_has_prev(page) && !page_is_leaf(page)) {
first_rec = page_rec_get_next( first_rec = page_rec_get_next(
...@@ -969,10 +966,9 @@ rtr_page_split_and_insert( ...@@ -969,10 +966,9 @@ rtr_page_split_and_insert(
static_cast<uchar*>(first_rec)); static_cast<uchar*>(first_rec));
/* Allocate a new page to the index */ /* Allocate a new page to the index */
hint_page_no = page_no + 1;
const uint16_t page_level = btr_page_get_level(page); const uint16_t page_level = btr_page_get_level(page);
new_block = btr_page_alloc(cursor->index, hint_page_no, FSP_UP, new_block = btr_page_alloc(cursor->index, page_id.page_no() + 1,
page_level, mtr, mtr); FSP_UP, page_level, mtr, mtr);
if (!new_block) { if (!new_block) {
return NULL; return NULL;
} }
...@@ -1155,8 +1151,7 @@ rtr_page_split_and_insert( ...@@ -1155,8 +1151,7 @@ rtr_page_split_and_insert(
/* Check any predicate locks need to be moved/copied to the /* Check any predicate locks need to be moved/copied to the
new page */ new page */
lock_prdt_update_split(new_block, &prdt, &new_prdt, lock_prdt_update_split(new_block, &prdt, &new_prdt, page_id);
cursor->index->table->space_id, page_no);
/* Adjust the upper level. */ /* Adjust the upper level. */
rtr_adjust_upper_level(cursor, flags, block, new_block, rtr_adjust_upper_level(cursor, flags, block, new_block,
......
...@@ -298,8 +298,9 @@ rtr_pcur_getnext_from_path( ...@@ -298,8 +298,9 @@ rtr_pcur_getnext_from_path(
&& mode != PAGE_CUR_RTREE_LOCATE) { && mode != PAGE_CUR_RTREE_LOCATE) {
ut_ad(rtr_info->thr); ut_ad(rtr_info->thr);
lock_place_prdt_page_lock( lock_place_prdt_page_lock(
index->table->space_id, page_id_t(block->page.id().space(),
next_page_no, index, next_page_no),
index,
rtr_info->thr); rtr_info->thr);
} }
new_split = true; new_split = true;
...@@ -1689,7 +1690,6 @@ rtr_cur_search_with_match( ...@@ -1689,7 +1690,6 @@ rtr_cur_search_with_match(
const rec_t* best_rec; const rec_t* best_rec;
const rec_t* last_match_rec = NULL; const rec_t* last_match_rec = NULL;
bool match_init = false; bool match_init = false;
ulint space = block->page.id().space();
page_cur_mode_t orig_mode = mode; page_cur_mode_t orig_mode = mode;
const rec_t* first_rec = NULL; const rec_t* first_rec = NULL;
...@@ -1870,7 +1870,11 @@ rtr_cur_search_with_match( ...@@ -1870,7 +1870,11 @@ rtr_cur_search_with_match(
/* Lock the page, preventing it /* Lock the page, preventing it
from being shrunk */ from being shrunk */
lock_place_prdt_page_lock( lock_place_prdt_page_lock(
space, page_no, index, page_id_t(block->page
.id()
.space(),
page_no),
index,
rtr_info->thr); rtr_info->thr);
} }
} else { } else {
......
...@@ -699,10 +699,10 @@ fill_innodb_locks_from_cache( ...@@ -699,10 +699,10 @@ fill_innodb_locks_from_cache(
OK(field_store_string(fields[IDX_LOCK_INDEX], OK(field_store_string(fields[IDX_LOCK_INDEX],
row->lock_index)); row->lock_index));
OK(fields[IDX_LOCK_SPACE]->store( OK(fields[IDX_LOCK_SPACE]->store(
row->lock_space, true)); row->lock_page.space(), true));
fields[IDX_LOCK_SPACE]->set_notnull(); fields[IDX_LOCK_SPACE]->set_notnull();
OK(fields[IDX_LOCK_PAGE]->store( OK(fields[IDX_LOCK_PAGE]->store(
row->lock_page, true)); row->lock_page.page_no(), true));
fields[IDX_LOCK_PAGE]->set_notnull(); fields[IDX_LOCK_PAGE]->set_notnull();
OK(fields[IDX_LOCK_REC]->store( OK(fields[IDX_LOCK_REC]->store(
row->lock_rec, true)); row->lock_rec, true));
......
...@@ -3329,18 +3329,22 @@ ibuf_insert_low( ...@@ -3329,18 +3329,22 @@ ibuf_insert_low(
and done mtr_commit(&mtr) to release the latch. */ and done mtr_commit(&mtr) to release the latch. */
ibuf_mtr_start(&bitmap_mtr); ibuf_mtr_start(&bitmap_mtr);
index->set_modified(bitmap_mtr);
bitmap_page = ibuf_bitmap_get_map_page(page_id, zip_size, &bitmap_mtr); bitmap_page = ibuf_bitmap_get_map_page(page_id, zip_size, &bitmap_mtr);
/* We check if the index page is suitable for buffered entries */ /* We check if the index page is suitable for buffered entries */
if (buf_pool.page_hash_contains(page_id) if (buf_pool.page_hash_contains(page_id)) {
|| lock_rec_expl_exist_on_page(page_id.space(), commit_exit:
page_id.page_no())) {
ibuf_mtr_commit(&bitmap_mtr); ibuf_mtr_commit(&bitmap_mtr);
goto fail_exit; goto fail_exit;
} else {
lock_mutex_enter();
const auto lock_exists = lock_sys.get_first(page_id);
lock_mutex_exit();
if (lock_exists) {
goto commit_exit;
}
} }
if (op == IBUF_OP_INSERT) { if (op == IBUF_OP_INSERT) {
...@@ -3378,8 +3382,7 @@ ibuf_insert_low( ...@@ -3378,8 +3382,7 @@ ibuf_insert_low(
dfield_t* field; dfield_t* field;
if (counter == ULINT_UNDEFINED) { if (counter == ULINT_UNDEFINED) {
ibuf_mtr_commit(&bitmap_mtr); goto commit_exit;
goto fail_exit;
} }
field = dtuple_get_nth_field( field = dtuple_get_nth_field(
...@@ -3391,6 +3394,7 @@ ibuf_insert_low( ...@@ -3391,6 +3394,7 @@ ibuf_insert_low(
/* Set the bitmap bit denoting that the insert buffer contains /* Set the bitmap bit denoting that the insert buffer contains
buffered entries for this index page, if the bit is not set yet */ buffered entries for this index page, if the bit is not set yet */
index->set_modified(bitmap_mtr);
ibuf_bitmap_page_set_bits<IBUF_BITMAP_BUFFERED>( ibuf_bitmap_page_set_bits<IBUF_BITMAP_BUFFERED>(
bitmap_page, page_id, physical_size, true, &bitmap_mtr); bitmap_page, page_id, physical_size, true, &bitmap_mtr);
ibuf_mtr_commit(&bitmap_mtr); ibuf_mtr_commit(&bitmap_mtr);
......
...@@ -252,15 +252,6 @@ lock_rec_restore_from_page_infimum( ...@@ -252,15 +252,6 @@ lock_rec_restore_from_page_infimum(
state; lock bits are reset on state; lock bits are reset on
the infimum */ the infimum */
/*********************************************************************//** /*********************************************************************//**
Determines if there are explicit record locks on a page.
@return an explicit record lock on the page, or NULL if there are none */
lock_t*
lock_rec_expl_exist_on_page(
/*========================*/
ulint space, /*!< in: space id */
ulint page_no)/*!< in: page number */
MY_ATTRIBUTE((warn_unused_result));
/*********************************************************************//**
Checks if locks of other transactions prevent an immediate insert of Checks if locks of other transactions prevent an immediate insert of
a record. If they do, first tests if the query thread should anyway a record. If they do, first tests if the query thread should anyway
be suspended for some reason; if not, then puts the transaction and be suspended for some reason; if not, then puts the transaction and
...@@ -487,28 +478,6 @@ lock_rec_unlock( ...@@ -487,28 +478,6 @@ lock_rec_unlock(
and release possible other transactions waiting because of these locks. */ and release possible other transactions waiting because of these locks. */
void lock_release(trx_t* trx); void lock_release(trx_t* trx);
/*********************************************************************//**
Calculates the fold value of a page file address: used in inserting or
searching for a lock in the hash table.
@return folded value */
UNIV_INLINE
ulint
lock_rec_fold(
/*==========*/
ulint space, /*!< in: space */
ulint page_no)/*!< in: page number */
MY_ATTRIBUTE((const));
/*********************************************************************//**
Calculates the hash value of a page file address: used in inserting or
searching for a lock in the hash table.
@return hashed value */
UNIV_INLINE
unsigned
lock_rec_hash(
/*==========*/
ulint space, /*!< in: space */
ulint page_no);/*!< in: page number */
/*************************************************************//** /*************************************************************//**
Get the lock hash table */ Get the lock hash table */
UNIV_INLINE UNIV_INLINE
...@@ -819,6 +788,46 @@ class lock_sys_t ...@@ -819,6 +788,46 @@ class lock_sys_t
/** Closes the lock system at database shutdown. */ /** Closes the lock system at database shutdown. */
void close(); void close();
/** @return the hash value for a page address */
ulint hash(const page_id_t id) const
{ ut_ad(mutex_own(&mutex)); return rec_hash.calc_hash(id.fold()); }
/** Get the first lock on a page.
@param lock_hash hash table to look at
@param id page number
@return first lock
@retval nullptr if none exists */
lock_t *get_first(const hash_table_t &lock_hash, const page_id_t id) const
{
ut_ad(&lock_hash == &rec_hash || &lock_hash == &prdt_hash ||
&lock_hash == &prdt_page_hash);
for (lock_t *lock= static_cast<lock_t*>
(HASH_GET_FIRST(&lock_hash, hash(id)));
lock; lock= static_cast<lock_t*>(HASH_GET_NEXT(hash, lock)))
if (lock->un_member.rec_lock.page_id == id)
return lock;
return nullptr;
}
/** Get the first record lock on a page.
@param id page number
@return first lock
@retval nullptr if none exists */
lock_t *get_first(const page_id_t id) const
{ return get_first(rec_hash, id); }
/** Get the first predicate lock on a SPATIAL INDEX page.
@param id page number
@return first lock
@retval nullptr if none exists */
lock_t *get_first_prdt(const page_id_t id) const
{ return get_first(prdt_hash, id); }
/** Get the first predicate lock on a SPATIAL INDEX page.
@param id page number
@return first lock
@retval nullptr if none exists */
lock_t *get_first_prdt_page(const page_id_t id) const
{ return get_first(prdt_page_hash, id); }
}; };
/*********************************************************************//** /*********************************************************************//**
...@@ -858,8 +867,7 @@ lock_rec_discard( ...@@ -858,8 +867,7 @@ lock_rec_discard(
without checking for deadlocks or conflicts. without checking for deadlocks or conflicts.
@param[in] type_mode lock mode and wait flag; type will be replaced @param[in] type_mode lock mode and wait flag; type will be replaced
with LOCK_REC with LOCK_REC
@param[in] space tablespace id @param[in] page_id index page number
@param[in] page_no index page number
@param[in] page R-tree index page, or NULL @param[in] page R-tree index page, or NULL
@param[in] heap_no record heap number in the index page @param[in] heap_no record heap number in the index page
@param[in] index the index tree @param[in] index the index tree
...@@ -873,8 +881,7 @@ lock_rec_create_low( ...@@ -873,8 +881,7 @@ lock_rec_create_low(
que_thr_t* thr, /*!< thread owning trx */ que_thr_t* thr, /*!< thread owning trx */
#endif #endif
unsigned type_mode, unsigned type_mode,
ulint space, const page_id_t page_id,
ulint page_no,
const page_t* page, const page_t* page,
ulint heap_no, ulint heap_no,
dict_index_t* index, dict_index_t* index,
......
/***************************************************************************** /*****************************************************************************
Copyright (c) 1996, 2015, Oracle and/or its affiliates. All Rights Reserved. Copyright (c) 1996, 2015, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2017, 2018, MariaDB Corporation. Copyright (c) 2017, 2020, MariaDB Corporation.
This program is free software; you can redistribute it and/or modify it under 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 the terms of the GNU General Public License as published by the Free Software
...@@ -28,34 +28,6 @@ Created 5/7/1996 Heikki Tuuri ...@@ -28,34 +28,6 @@ Created 5/7/1996 Heikki Tuuri
#include "buf0buf.h" #include "buf0buf.h"
#include "page0page.h" #include "page0page.h"
/*********************************************************************//**
Calculates the fold value of a page file address: used in inserting or
searching for a lock in the hash table.
@return folded value */
UNIV_INLINE
ulint
lock_rec_fold(
/*==========*/
ulint space, /*!< in: space */
ulint page_no)/*!< in: page number */
{
return page_id_t(space, page_no).fold();
}
/*********************************************************************//**
Calculates the hash value of a page file address: used in inserting or
searching for a lock in the hash table.
@return hashed value */
UNIV_INLINE
unsigned
lock_rec_hash(
/*==========*/
ulint space, /*!< in: space */
ulint page_no)/*!< in: page number */
{
return unsigned(lock_sys.rec_hash.calc_hash(lock_rec_fold(space, page_no)));
}
/*********************************************************************//** /*********************************************************************//**
Gets the heap_no of the smallest user record on a page. Gets the heap_no of the smallest user record on a page.
@return heap_no of smallest user record, or PAGE_HEAP_NO_SUPREMUM */ @return heap_no of smallest user record, or PAGE_HEAP_NO_SUPREMUM */
...@@ -122,12 +94,10 @@ lock_rec_create( ...@@ -122,12 +94,10 @@ lock_rec_create(
trx mutex */ trx mutex */
{ {
btr_assert_not_corrupted(block, index); btr_assert_not_corrupted(block, index);
const page_id_t id(block->page.id());
return lock_rec_create_low( return lock_rec_create_low(
#ifdef WITH_WSREP #ifdef WITH_WSREP
c_lock, thr, c_lock, thr,
#endif #endif
type_mode, id.space(), id.page_no(), type_mode, block->page.id(), block->frame, heap_no,
block->frame, heap_no,
index, trx, caller_owns_trx_mutex); index, trx, caller_owns_trx_mutex);
} }
...@@ -58,9 +58,7 @@ Acquire a "Page" lock on a block ...@@ -58,9 +58,7 @@ Acquire a "Page" lock on a block
@return DB_SUCCESS, DB_LOCK_WAIT, or DB_DEADLOCK */ @return DB_SUCCESS, DB_LOCK_WAIT, or DB_DEADLOCK */
dberr_t dberr_t
lock_place_prdt_page_lock( lock_place_prdt_page_lock(
/*======================*/ const page_id_t page_id, /*!< in: page identifier */
ulint space, /*!< in: space for the page to lock */
ulint pageno, /*!< in: page number */
dict_index_t* index, /*!< in: secondary index */ dict_index_t* index, /*!< in: secondary index */
que_thr_t* thr); /*!< in: query thread */ que_thr_t* thr); /*!< in: query thread */
...@@ -108,8 +106,7 @@ lock_prdt_update_split( ...@@ -108,8 +106,7 @@ lock_prdt_update_split(
buf_block_t* new_block, /*!< in/out: the new half page */ buf_block_t* new_block, /*!< in/out: the new half page */
lock_prdt_t* prdt, /*!< in: MBR on the old page */ lock_prdt_t* prdt, /*!< in: MBR on the old page */
lock_prdt_t* new_prdt, /*!< in: MBR on the new page */ lock_prdt_t* new_prdt, /*!< in: MBR on the new page */
ulint space, /*!< in: space id */ const page_id_t page_id); /*!< in: page number */
ulint page_no); /*!< in: page number */
/**************************************************************//** /**************************************************************//**
Ajust locks from an ancester page of Rtree on the appropriate level . */ Ajust locks from an ancester page of Rtree on the appropriate level . */
...@@ -120,8 +117,7 @@ lock_prdt_update_parent( ...@@ -120,8 +117,7 @@ lock_prdt_update_parent(
buf_block_t* right_block, /*!< in/out: the new half page */ buf_block_t* right_block, /*!< in/out: the new half page */
lock_prdt_t* left_prdt, /*!< in: MBR on the old page */ lock_prdt_t* left_prdt, /*!< in: MBR on the old page */
lock_prdt_t* right_prdt, /*!< in: MBR on the new page */ lock_prdt_t* right_prdt, /*!< in: MBR on the new page */
ulint space, /*!< in: space id */ const page_id_t page_id); /*!< in: parent page */
ulint page_no); /*!< in: page number */
/*********************************************************************//** /*********************************************************************//**
Checks if locks of other transactions prevent an immediate insert of Checks if locks of other transactions prevent an immediate insert of
...@@ -190,17 +186,11 @@ lock_prdt_rec_move( ...@@ -190,17 +186,11 @@ lock_prdt_rec_move(
const buf_block_t* donator); /*!< in: buffer block containing const buf_block_t* donator); /*!< in: buffer block containing
the donating record */ the donating record */
/** Check whether there are R-tree Page lock on a buffer page /** Check whether there are R-tree Page lock on a page
@param[in] trx trx to test the lock @param[in] trx trx to test the lock
@param[in] space space id for the page @param[in] page_id page identifier
@param[in] page_no page number @return true if there is none */
@return true if there is none */ bool lock_test_prdt_page_lock(const trx_t *trx, const page_id_t page_id);
bool
lock_test_prdt_page_lock(
/*=====================*/
const trx_t* trx,
ulint space,
ulint page_no);
/** Removes predicate lock objects set on an index page which is discarded. /** Removes predicate lock objects set on an index page which is discarded.
@param[in] block page to be discarded @param[in] block page to be discarded
......
...@@ -539,29 +539,6 @@ lock_t* ...@@ -539,29 +539,6 @@ lock_t*
lock_rec_get_next_on_page( lock_rec_get_next_on_page(
/*======================*/ /*======================*/
lock_t* lock); /*!< in: a record lock */ lock_t* lock); /*!< in: a record lock */
/*********************************************************************//**
Gets the first record lock on a page, where the page is identified by its
file address.
@return first lock, NULL if none exists */
UNIV_INLINE
lock_t*
lock_rec_get_first_on_page_addr(
/*============================*/
hash_table_t* lock_hash, /* Lock hash table */
ulint space, /*!< in: space */
ulint page_no); /*!< in: page number */
/*********************************************************************//**
Gets the first record lock on a page, where the page is identified by a
pointer to it.
@return first lock, NULL if none exists */
UNIV_INLINE
lock_t*
lock_rec_get_first_on_page(
/*=======================*/
hash_table_t* lock_hash, /*!< in: lock hash table */
const buf_block_t* block); /*!< in: buffer block */
/*********************************************************************//** /*********************************************************************//**
Gets the next explicit lock request on a record. Gets the next explicit lock request on a record.
...@@ -632,20 +609,6 @@ lock_get_wait( ...@@ -632,20 +609,6 @@ lock_get_wait(
/*==========*/ /*==========*/
const lock_t* lock); /*!< in: lock */ const lock_t* lock); /*!< in: lock */
/*********************************************************************//**
Looks for a suitable type record lock struct by the same trx on the same page.
This can be used to save space when a new record lock should be set on a page:
no new struct is needed, if a suitable old is found.
@return lock or NULL */
UNIV_INLINE
lock_t*
lock_rec_find_similar_on_page(
/*==========================*/
ulint type_mode, /*!< in: lock type_mode field */
ulint heap_no, /*!< in: heap number of the record */
lock_t* lock, /*!< in: lock_rec_get_first_on_page() */
const trx_t* trx); /*!< in: transaction */
/*********************************************************************//** /*********************************************************************//**
Checks if a transaction has the specified table lock, or stronger. This Checks if a transaction has the specified table lock, or stronger. This
function should only be called by the thread that owns the transaction. function should only be called by the thread that owns the transaction.
......
...@@ -121,70 +121,6 @@ lock_rec_get_next_on_page( ...@@ -121,70 +121,6 @@ lock_rec_get_next_on_page(
return((lock_t*) lock_rec_get_next_on_page_const(lock)); return((lock_t*) lock_rec_get_next_on_page_const(lock));
} }
/*********************************************************************//**
Gets the first record lock on a page, where the page is identified by its
file address.
@return first lock, NULL if none exists */
UNIV_INLINE
lock_t*
lock_rec_get_first_on_page_addr(
/*============================*/
hash_table_t* lock_hash, /* Lock hash table */
ulint space, /*!< in: space */
ulint page_no) /*!< in: page number */
{
ut_ad(lock_mutex_own());
for (lock_t* lock = static_cast<lock_t*>(
HASH_GET_FIRST(lock_hash,
lock_rec_hash(space, page_no)));
lock != NULL;
lock = static_cast<lock_t*>(HASH_GET_NEXT(hash, lock))) {
if (lock->un_member.rec_lock.space == space
&& lock->un_member.rec_lock.page_no == page_no) {
return(lock);
}
}
return(NULL);
}
/*********************************************************************//**
Gets the first record lock on a page, where the page is identified by a
pointer to it.
@return first lock, NULL if none exists */
UNIV_INLINE
lock_t*
lock_rec_get_first_on_page(
/*=======================*/
hash_table_t* lock_hash, /*!< in: lock hash table */
const buf_block_t* block) /*!< in: buffer block */
{
ut_ad(lock_mutex_own());
const page_id_t page_id(block->page.id());
ulint space = page_id.space();
ulint page_no = page_id.page_no();
for (lock_t* lock = static_cast<lock_t*>(
HASH_GET_FIRST(lock_hash,
lock_rec_hash(space, page_no)));
lock != NULL;
lock = static_cast<lock_t*>(HASH_GET_NEXT(hash, lock))) {
if (lock->un_member.rec_lock.space == space
&& lock->un_member.rec_lock.page_no == page_no) {
return(lock);
}
}
return(NULL);
}
/*********************************************************************//** /*********************************************************************//**
Gets the next explicit lock request on a record. Gets the next explicit lock request on a record.
@return next lock, NULL if none exists or if heap_no == ULINT_UNDEFINED */ @return next lock, NULL if none exists or if heap_no == ULINT_UNDEFINED */
...@@ -229,16 +165,11 @@ lock_rec_get_first( ...@@ -229,16 +165,11 @@ lock_rec_get_first(
const buf_block_t* block, /*!< in: block containing the record */ const buf_block_t* block, /*!< in: block containing the record */
ulint heap_no)/*!< in: heap number of the record */ ulint heap_no)/*!< in: heap number of the record */
{ {
ut_ad(lock_mutex_own()); for (lock_t *lock= lock_sys.get_first(*hash, block->page.id());
lock; lock= lock_rec_get_next_on_page(lock))
for (lock_t* lock = lock_rec_get_first_on_page(hash, block); lock; if (lock_rec_get_nth_bit(lock, heap_no))
lock = lock_rec_get_next_on_page(lock)) { return lock;
if (lock_rec_get_nth_bit(lock, heap_no)) { return nullptr;
return(lock);
}
}
return(NULL);
} }
/*********************************************************************//** /*********************************************************************//**
...@@ -275,23 +206,15 @@ lock_rec_get_next_on_page_const( ...@@ -275,23 +206,15 @@ lock_rec_get_next_on_page_const(
/*============================*/ /*============================*/
const lock_t* lock) /*!< in: a record lock */ const lock_t* lock) /*!< in: a record lock */
{ {
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
ut_ad(lock_get_type_low(lock) == LOCK_REC); ut_ad(lock_get_type_low(lock) == LOCK_REC);
ulint space = lock->un_member.rec_lock.space;
ulint page_no = lock->un_member.rec_lock.page_no;
while ((lock = static_cast<const lock_t*>(HASH_GET_NEXT(hash, lock)))
!= NULL) {
if (lock->un_member.rec_lock.space == space
&& lock->un_member.rec_lock.page_no == page_no) {
return(lock); const page_id_t page_id(lock->un_member.rec_lock.page_id);
}
}
return(NULL); while (!!(lock= static_cast<const lock_t*>(HASH_GET_NEXT(hash, lock))))
if (lock->un_member.rec_lock.page_id == page_id)
break;
return lock;
} }
/*********************************************************************//** /*********************************************************************//**
...@@ -354,37 +277,6 @@ lock_get_wait( ...@@ -354,37 +277,6 @@ lock_get_wait(
return(lock->type_mode & LOCK_WAIT); return(lock->type_mode & LOCK_WAIT);
} }
/*********************************************************************//**
Looks for a suitable type record lock struct by the same trx on the same page.
This can be used to save space when a new record lock should be set on a page:
no new struct is needed, if a suitable old is found.
@return lock or NULL */
UNIV_INLINE
lock_t*
lock_rec_find_similar_on_page(
/*==========================*/
ulint type_mode, /*!< in: lock type_mode field */
ulint heap_no, /*!< in: heap number of the record */
lock_t* lock, /*!< in: lock_rec_get_first_on_page() */
const trx_t* trx) /*!< in: transaction */
{
ut_ad(lock_mutex_own());
for (/* No op */;
lock != NULL;
lock = lock_rec_get_next_on_page(lock)) {
if (lock->trx == trx
&& lock->type_mode == type_mode
&& lock_rec_get_n_bits(lock) > heap_no) {
return(lock);
}
}
return(NULL);
}
/*********************************************************************//** /*********************************************************************//**
Checks if a transaction has the specified table lock, or stronger. This Checks if a transaction has the specified table lock, or stronger. This
function should only be called by the thread that owns the transaction. function should only be called by the thread that owns the transaction.
......
/***************************************************************************** /*****************************************************************************
Copyright (c) 1996, 2015, Oracle and/or its affiliates. All Rights Reserved. Copyright (c) 1996, 2015, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2018, 2019, MariaDB Corporation. Copyright (c) 2018, 2020, MariaDB Corporation.
This program is free software; you can redistribute it and/or modify it under 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 the terms of the GNU General Public License as published by the Free Software
...@@ -25,6 +25,7 @@ Created 5/7/1996 Heikki Tuuri ...@@ -25,6 +25,7 @@ Created 5/7/1996 Heikki Tuuri
*******************************************************/ *******************************************************/
#include "dict0types.h" #include "dict0types.h"
#include "buf0types.h"
#include "ut0lst.h" #include "ut0lst.h"
#ifndef lock0types_h #ifndef lock0types_h
...@@ -89,8 +90,8 @@ struct lock_table_t { ...@@ -89,8 +90,8 @@ struct lock_table_t {
/** Record lock for a page */ /** Record lock for a page */
struct lock_rec_t { struct lock_rec_t {
ib_uint32_t space; /*!< space id */ /** page identifier */
ib_uint32_t page_no; /*!< page number */ page_id_t page_id;
ib_uint32_t n_bits; /*!< number of bits in the lock ib_uint32_t n_bits; /*!< number of bits in the lock
bitmap; NOTE: the lock bitmap is bitmap; NOTE: the lock bitmap is
placed immediately after the placed immediately after the
...@@ -105,12 +106,12 @@ struct lock_rec_t { ...@@ -105,12 +106,12 @@ struct lock_rec_t {
/** Print the record lock into the given output stream /** Print the record lock into the given output stream
@param[in,out] out the output stream @param[in,out] out the output stream
@return the given output stream. */ @return the given output stream. */
inline inline std::ostream &lock_rec_t::print(std::ostream &out) const
std::ostream& lock_rec_t::print(std::ostream& out) const
{ {
out << "[lock_rec_t: space=" << space << ", page_no=" << page_no out << "[lock_rec_t: space=" << page_id.space()
<< ", n_bits=" << n_bits << "]"; << ", page_no=" << page_id.page_no()
return(out); << ", n_bits=" << n_bits << "]";
return out;
} }
inline inline
......
...@@ -31,6 +31,7 @@ Created July 17, 2007 Vasil Dimov ...@@ -31,6 +31,7 @@ Created July 17, 2007 Vasil Dimov
#include "trx0types.h" #include "trx0types.h"
#include "dict0types.h" #include "dict0types.h"
#include "buf0types.h"
/** The maximum amount of memory that can be consumed by innodb_trx, /** The maximum amount of memory that can be consumed by innodb_trx,
innodb_locks and innodb_lock_waits information schema tables. */ innodb_locks and innodb_lock_waits information schema tables. */
...@@ -90,10 +91,8 @@ struct i_s_locks_row_t { ...@@ -90,10 +91,8 @@ struct i_s_locks_row_t {
lock_get_table_name() */ lock_get_table_name() */
/** index name of a record lock; NULL for table locks */ /** index name of a record lock; NULL for table locks */
const char* lock_index; const char* lock_index;
/** tablespace identifier of the record; 0 if !lock_index */ /** page identifier of the record; (0,0) if !lock_index */
uint32_t lock_space; page_id_t lock_page;
/** page number of the record; 0 if !lock_index */
uint32_t lock_page;
/** heap number of the record; 0 if !lock_index */ /** heap number of the record; 0 if !lock_index */
uint16_t lock_rec; uint16_t lock_rec;
/** lock mode corresponding to lock_mode_values_typelib */ /** lock mode corresponding to lock_mode_values_typelib */
......
...@@ -477,13 +477,9 @@ void lock_sys_t::create(ulint n_cells) ...@@ -477,13 +477,9 @@ void lock_sys_t::create(ulint n_cells)
/** Calculates the fold value of a lock: used in migrating the hash table. /** Calculates the fold value of a lock: used in migrating the hash table.
@param[in] lock record lock object @param[in] lock record lock object
@return folded value */ @return folded value */
static static ulint lock_rec_lock_fold(const lock_t *lock)
ulint
lock_rec_lock_fold(
const lock_t* lock)
{ {
return(lock_rec_fold(lock->un_member.rec_lock.space, return lock->un_member.rec_lock.page_id.fold();
lock->un_member.rec_lock.page_no));
} }
...@@ -846,26 +842,6 @@ lock_rec_find_set_bit( ...@@ -846,26 +842,6 @@ lock_rec_find_set_bit(
return(ULINT_UNDEFINED); return(ULINT_UNDEFINED);
} }
/*********************************************************************//**
Determines if there are explicit record locks on a page.
@return an explicit record lock on the page, or NULL if there are none */
lock_t*
lock_rec_expl_exist_on_page(
/*========================*/
ulint space, /*!< in: space id */
ulint page_no)/*!< in: page number */
{
lock_t* lock;
lock_mutex_enter();
/* Only used in ibuf pages, so rec_hash is good enough */
lock = lock_rec_get_first_on_page_addr(&lock_sys.rec_hash,
space, page_no);
lock_mutex_exit();
return(lock);
}
/*********************************************************************//** /*********************************************************************//**
Resets the record lock bitmap to zero. NOTE: does not touch the wait_lock Resets the record lock bitmap to zero. NOTE: does not touch the wait_lock
pointer in the transaction! This function is used in lock object creation pointer in the transaction! This function is used in lock object creation
...@@ -887,7 +863,7 @@ lock_rec_bitmap_reset( ...@@ -887,7 +863,7 @@ lock_rec_bitmap_reset(
ut_ad((lock_rec_get_n_bits(lock) % 8) == 0); ut_ad((lock_rec_get_n_bits(lock) % 8) == 0);
memset(&lock[1], 0, n_bytes); memset(reinterpret_cast<void*>(&lock[1]), 0, n_bytes);
} }
/*********************************************************************//** /*********************************************************************//**
...@@ -919,35 +895,21 @@ lock_rec_get_prev( ...@@ -919,35 +895,21 @@ lock_rec_get_prev(
ulint heap_no)/*!< in: heap number of the record */ ulint heap_no)/*!< in: heap number of the record */
{ {
lock_t* lock; lock_t* lock;
ulint space;
ulint page_no;
lock_t* found_lock = NULL; lock_t* found_lock = NULL;
hash_table_t* hash;
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
ut_ad(lock_get_type_low(in_lock) == LOCK_REC); ut_ad(lock_get_type_low(in_lock) == LOCK_REC);
space = in_lock->un_member.rec_lock.space; for (lock = lock_sys.get_first(*lock_hash_get(in_lock->type_mode),
page_no = in_lock->un_member.rec_lock.page_no; in_lock->un_member.rec_lock.page_id);
lock != in_lock;
hash = lock_hash_get(in_lock->type_mode);
for (lock = lock_rec_get_first_on_page_addr(hash, space, page_no);
/* No op */;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
ut_ad(lock);
if (lock == in_lock) {
return(found_lock);
}
if (lock_rec_get_nth_bit(lock, heap_no)) { if (lock_rec_get_nth_bit(lock, heap_no)) {
found_lock = lock; found_lock = lock;
} }
} }
return found_lock;
} }
/*============= FUNCTIONS FOR ANALYZING RECORD LOCK QUEUE ================*/ /*============= FUNCTIONS FOR ANALYZING RECORD LOCK QUEUE ================*/
...@@ -1285,8 +1247,7 @@ wsrep_print_wait_locks( ...@@ -1285,8 +1247,7 @@ wsrep_print_wait_locks(
without checking for deadlocks or conflicts. without checking for deadlocks or conflicts.
@param[in] type_mode lock mode and wait flag; type will be replaced @param[in] type_mode lock mode and wait flag; type will be replaced
with LOCK_REC with LOCK_REC
@param[in] space tablespace id @param[in] page_id index page number
@param[in] page_no index page number
@param[in] page R-tree index page, or NULL @param[in] page R-tree index page, or NULL
@param[in] heap_no record heap number in the index page @param[in] heap_no record heap number in the index page
@param[in] index the index tree @param[in] index the index tree
...@@ -1300,8 +1261,7 @@ lock_rec_create_low( ...@@ -1300,8 +1261,7 @@ lock_rec_create_low(
que_thr_t* thr, /*!< thread owning trx */ que_thr_t* thr, /*!< thread owning trx */
#endif #endif
unsigned type_mode, unsigned type_mode,
ulint space, const page_id_t page_id,
ulint page_no,
const page_t* page, const page_t* page,
ulint heap_no, ulint heap_no,
dict_index_t* index, dict_index_t* index,
...@@ -1370,8 +1330,7 @@ lock_rec_create_low( ...@@ -1370,8 +1330,7 @@ lock_rec_create_low(
lock->trx = trx; lock->trx = trx;
lock->type_mode = (type_mode & unsigned(~LOCK_TYPE_MASK)) | LOCK_REC; lock->type_mode = (type_mode & unsigned(~LOCK_TYPE_MASK)) | LOCK_REC;
lock->index = index; lock->index = index;
lock->un_member.rec_lock.space = uint32_t(space); lock->un_member.rec_lock.page_id = page_id;
lock->un_member.rec_lock.page_no = uint32_t(page_no);
if (UNIV_LIKELY(!(type_mode & (LOCK_PREDICATE | LOCK_PRDT_PAGE)))) { if (UNIV_LIKELY(!(type_mode & (LOCK_PREDICATE | LOCK_PRDT_PAGE)))) {
lock->un_member.rec_lock.n_bits = uint32_t(n_bytes * 8); lock->un_member.rec_lock.n_bits = uint32_t(n_bytes * 8);
...@@ -1457,10 +1416,10 @@ lock_rec_create_low( ...@@ -1457,10 +1416,10 @@ lock_rec_create_low(
== INNODB_LOCK_SCHEDULE_ALGORITHM_VATS == INNODB_LOCK_SCHEDULE_ALGORITHM_VATS
&& !thd_is_replication_slave_thread(trx->mysql_thd)) { && !thd_is_replication_slave_thread(trx->mysql_thd)) {
HASH_PREPEND(lock_t, hash, &lock_sys.rec_hash, HASH_PREPEND(lock_t, hash, &lock_sys.rec_hash,
lock_rec_fold(space, page_no), lock); page_id.fold(), lock);
} else { } else {
HASH_INSERT(lock_t, hash, lock_hash_get(type_mode), HASH_INSERT(lock_t, hash, lock_hash_get(type_mode),
lock_rec_fold(space, page_no), lock); page_id.fold(), lock);
} }
if (!holds_trx_mutex) { if (!holds_trx_mutex) {
...@@ -1514,19 +1473,14 @@ static ...@@ -1514,19 +1473,14 @@ static
dberr_t dberr_t
lock_rec_insert_by_trx_age( lock_rec_insert_by_trx_age(
lock_t *in_lock) /*!< in: lock to be insert */{ lock_t *in_lock) /*!< in: lock to be insert */{
ulint space;
ulint page_no;
ulint rec_fold;
lock_t* node; lock_t* node;
lock_t* next; lock_t* next;
hash_table_t* hash; hash_table_t* hash;
hash_cell_t* cell; hash_cell_t* cell;
space = in_lock->un_member.rec_lock.space; const page_id_t page_id(in_lock->un_member.rec_lock.page_id);
page_no = in_lock->un_member.rec_lock.page_no;
rec_fold = lock_rec_fold(space, page_no);
hash = lock_hash_get(in_lock->type_mode); hash = lock_hash_get(in_lock->type_mode);
cell = &hash->array[hash->calc_hash(rec_fold)]; cell = &hash->array[hash->calc_hash(page_id.fold())];
node = (lock_t *) cell->node; node = (lock_t *) cell->node;
// If in_lock is not a wait lock, we insert it to the head of the list. // If in_lock is not a wait lock, we insert it to the head of the list.
...@@ -1568,9 +1522,6 @@ bool ...@@ -1568,9 +1522,6 @@ bool
lock_queue_validate( lock_queue_validate(
const lock_t *in_lock) /*!< in: lock whose hash list is to be validated */ const lock_t *in_lock) /*!< in: lock whose hash list is to be validated */
{ {
ulint space;
ulint page_no;
ulint rec_fold;
hash_table_t* hash; hash_table_t* hash;
hash_cell_t* cell; hash_cell_t* cell;
lock_t* next; lock_t* next;
...@@ -1580,11 +1531,9 @@ lock_queue_validate( ...@@ -1580,11 +1531,9 @@ lock_queue_validate(
return true; return true;
} }
space = in_lock->un_member.rec_lock.space; const page_id_t page_id(in_lock->un_member.rec_lock.page_id);
page_no = in_lock->un_member.rec_lock.page_no;
rec_fold = lock_rec_fold(space, page_no);
hash = lock_hash_get(in_lock->type_mode); hash = lock_hash_get(in_lock->type_mode);
cell = &hash->array[hash->calc_hash(rec_fold)]; cell = &hash->array[hash->calc_hash(page_id.fold())];
next = (lock_t *) cell->node; next = (lock_t *) cell->node;
while (next != NULL) { while (next != NULL) {
// If this is a granted lock, check that there's no wait lock before it. // If this is a granted lock, check that there's no wait lock before it.
...@@ -1741,6 +1690,36 @@ lock_rec_enqueue_waiting( ...@@ -1741,6 +1690,36 @@ lock_rec_enqueue_waiting(
return DB_LOCK_WAIT; return DB_LOCK_WAIT;
} }
/*********************************************************************//**
Looks for a suitable type record lock struct by the same trx on the same page.
This can be used to save space when a new record lock should be set on a page:
no new struct is needed, if a suitable old is found.
@return lock or NULL */
static inline
lock_t*
lock_rec_find_similar_on_page(
ulint type_mode, /*!< in: lock type_mode field */
ulint heap_no, /*!< in: heap number of the record */
lock_t* lock, /*!< in: lock_sys.get_first() */
const trx_t* trx) /*!< in: transaction */
{
ut_ad(lock_mutex_own());
for (/* No op */;
lock != NULL;
lock = lock_rec_get_next_on_page(lock)) {
if (lock->trx == trx
&& lock->type_mode == type_mode
&& lock_rec_get_n_bits(lock) > heap_no) {
return(lock);
}
}
return(NULL);
}
/*********************************************************************//** /*********************************************************************//**
Adds a record lock request in the record queue. The request is normally Adds a record lock request in the record queue. The request is normally
added as the last in the queue, but if there are no waiting lock requests added as the last in the queue, but if there are no waiting lock requests
...@@ -1828,11 +1807,11 @@ lock_rec_add_to_queue( ...@@ -1828,11 +1807,11 @@ lock_rec_add_to_queue(
lock_t* lock; lock_t* lock;
lock_t* first_lock; lock_t* first_lock;
hash_table_t* hash = lock_hash_get(type_mode);
/* Look for a waiting lock request on the same record or on a gap */ /* Look for a waiting lock request on the same record or on a gap */
for (first_lock = lock = lock_rec_get_first_on_page(hash, block); for (first_lock = lock = lock_sys.get_first(*lock_hash_get(type_mode),
block->page.id());
lock != NULL; lock != NULL;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
...@@ -1911,7 +1890,7 @@ lock_rec_lock( ...@@ -1911,7 +1890,7 @@ lock_rec_lock(
if (lock_table_has(trx, index->table, if (lock_table_has(trx, index->table,
static_cast<lock_mode>(LOCK_MODE_MASK & mode))); static_cast<lock_mode>(LOCK_MODE_MASK & mode)));
else if (lock_t *lock= lock_rec_get_first_on_page(&lock_sys.rec_hash, block)) else if (lock_t *lock= lock_sys.get_first(block->page.id()))
{ {
trx_mutex_enter(trx); trx_mutex_enter(trx);
if (lock_rec_get_next_on_page(lock) || if (lock_rec_get_next_on_page(lock) ||
...@@ -1992,28 +1971,22 @@ lock_rec_has_to_wait_in_queue( ...@@ -1992,28 +1971,22 @@ lock_rec_has_to_wait_in_queue(
const lock_t* wait_lock) /*!< in: waiting record lock */ const lock_t* wait_lock) /*!< in: waiting record lock */
{ {
const lock_t* lock; const lock_t* lock;
ulint space;
ulint page_no;
ulint heap_no; ulint heap_no;
ulint bit_mask; ulint bit_mask;
ulint bit_offset; ulint bit_offset;
hash_table_t* hash;
ut_ad(wait_lock); ut_ad(wait_lock);
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
ut_ad(lock_get_wait(wait_lock)); ut_ad(lock_get_wait(wait_lock));
ut_ad(lock_get_type_low(wait_lock) == LOCK_REC); ut_ad(lock_get_type_low(wait_lock) == LOCK_REC);
space = wait_lock->un_member.rec_lock.space;
page_no = wait_lock->un_member.rec_lock.page_no;
heap_no = lock_rec_find_set_bit(wait_lock); heap_no = lock_rec_find_set_bit(wait_lock);
bit_offset = heap_no / 8; bit_offset = heap_no / 8;
bit_mask = static_cast<ulint>(1) << (heap_no % 8); bit_mask = static_cast<ulint>(1) << (heap_no % 8);
hash = lock_hash_get(wait_lock->type_mode); for (lock = lock_sys.get_first(*lock_hash_get(wait_lock->type_mode),
wait_lock->un_member.rec_lock.page_id);
for (lock = lock_rec_get_first_on_page_addr(hash, space, page_no);
#ifdef WITH_WSREP #ifdef WITH_WSREP
lock && lock &&
#endif #endif
...@@ -2132,9 +2105,7 @@ lock_rec_cancel( ...@@ -2132,9 +2105,7 @@ lock_rec_cancel(
trx_mutex_exit(lock->trx); trx_mutex_exit(lock->trx);
} }
static static void lock_grant_and_move_on_page(ulint rec_fold, const page_id_t id)
void
lock_grant_and_move_on_page(ulint rec_fold, ulint space, ulint page_no)
{ {
lock_t* lock; lock_t* lock;
lock_t* previous = static_cast<lock_t*>( lock_t* previous = static_cast<lock_t*>(
...@@ -2143,15 +2114,13 @@ lock_grant_and_move_on_page(ulint rec_fold, ulint space, ulint page_no) ...@@ -2143,15 +2114,13 @@ lock_grant_and_move_on_page(ulint rec_fold, ulint space, ulint page_no)
if (previous == NULL) { if (previous == NULL) {
return; return;
} }
if (previous->un_member.rec_lock.space == space && if (previous->un_member.rec_lock.page_id == id) {
previous->un_member.rec_lock.page_no == page_no) {
lock = previous; lock = previous;
} }
else { else {
while (previous->hash && while (previous->hash &&
(previous->hash->un_member.rec_lock.space != space || (previous->hash->un_member.rec_lock.page_id != id)) {
previous->hash->un_member.rec_lock.page_no != page_no)) { previous = previous->hash;
previous = previous->hash;
} }
lock = previous->hash; lock = previous->hash;
} }
...@@ -2162,8 +2131,7 @@ lock_grant_and_move_on_page(ulint rec_fold, ulint space, ulint page_no) ...@@ -2162,8 +2131,7 @@ lock_grant_and_move_on_page(ulint rec_fold, ulint space, ulint page_no)
while (lock) { while (lock) {
/* If the lock is a wait lock on this page, and it does not need to wait. */ /* If the lock is a wait lock on this page, and it does not need to wait. */
if (lock_get_wait(lock) if (lock_get_wait(lock)
&& lock->un_member.rec_lock.space == space && lock->un_member.rec_lock.page_id == id
&& lock->un_member.rec_lock.page_no == page_no
&& !lock_rec_has_to_wait_in_queue(lock)) { && !lock_rec_has_to_wait_in_queue(lock)) {
lock_grant(lock); lock_grant(lock);
...@@ -2190,22 +2158,19 @@ to a lock. NOTE: all record locks contained in in_lock are removed. ...@@ -2190,22 +2158,19 @@ to a lock. NOTE: all record locks contained in in_lock are removed.
@param[in,out] in_lock record lock */ @param[in,out] in_lock record lock */
static void lock_rec_dequeue_from_page(lock_t* in_lock) static void lock_rec_dequeue_from_page(lock_t* in_lock)
{ {
ulint space;
ulint page_no;
hash_table_t* lock_hash; hash_table_t* lock_hash;
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
ut_ad(lock_get_type_low(in_lock) == LOCK_REC); ut_ad(lock_get_type_low(in_lock) == LOCK_REC);
/* We may or may not be holding in_lock->trx->mutex here. */ /* We may or may not be holding in_lock->trx->mutex here. */
space = in_lock->un_member.rec_lock.space; const page_id_t page_id(in_lock->un_member.rec_lock.page_id);
page_no = in_lock->un_member.rec_lock.page_no;
in_lock->index->table->n_rec_locks--; in_lock->index->table->n_rec_locks--;
lock_hash = lock_hash_get(in_lock->type_mode); lock_hash = lock_hash_get(in_lock->type_mode);
ulint rec_fold = lock_rec_fold(space, page_no); const ulint rec_fold = page_id.fold();
HASH_DELETE(lock_t, hash, lock_hash, rec_fold, in_lock); HASH_DELETE(lock_t, hash, lock_hash, rec_fold, in_lock);
UT_LIST_REMOVE(in_lock->trx->lock.trx_locks, in_lock); UT_LIST_REMOVE(in_lock->trx->lock.trx_locks, in_lock);
...@@ -2221,8 +2186,7 @@ static void lock_rec_dequeue_from_page(lock_t* in_lock) ...@@ -2221,8 +2186,7 @@ static void lock_rec_dequeue_from_page(lock_t* in_lock)
grant locks if there are no conflicting locks ahead. Stop at grant locks if there are no conflicting locks ahead. Stop at
the first X lock that is waiting or has been granted. */ the first X lock that is waiting or has been granted. */
for (lock_t* lock = lock_rec_get_first_on_page_addr( for (lock_t* lock = lock_sys.get_first(*lock_hash, page_id);
lock_hash, space, page_no);
lock != NULL; lock != NULL;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
...@@ -2234,7 +2198,7 @@ static void lock_rec_dequeue_from_page(lock_t* in_lock) ...@@ -2234,7 +2198,7 @@ static void lock_rec_dequeue_from_page(lock_t* in_lock)
} }
} }
} else { } else {
lock_grant_and_move_on_page(rec_fold, space, page_no); lock_grant_and_move_on_page(rec_fold, page_id);
} }
} }
...@@ -2247,8 +2211,6 @@ lock_rec_discard( ...@@ -2247,8 +2211,6 @@ lock_rec_discard(
record locks which are contained record locks which are contained
in this lock object are removed */ in this lock object are removed */
{ {
ulint space;
ulint page_no;
trx_lock_t* trx_lock; trx_lock_t* trx_lock;
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
...@@ -2256,13 +2218,10 @@ lock_rec_discard( ...@@ -2256,13 +2218,10 @@ lock_rec_discard(
trx_lock = &in_lock->trx->lock; trx_lock = &in_lock->trx->lock;
space = in_lock->un_member.rec_lock.space;
page_no = in_lock->un_member.rec_lock.page_no;
in_lock->index->table->n_rec_locks--; in_lock->index->table->n_rec_locks--;
HASH_DELETE(lock_t, hash, lock_hash_get(in_lock->type_mode), HASH_DELETE(lock_t, hash, lock_hash_get(in_lock->type_mode),
lock_rec_fold(space, page_no), in_lock); in_lock->un_member.rec_lock.page_id.fold(), in_lock);
UT_LIST_REMOVE(trx_lock->trx_locks, in_lock); UT_LIST_REMOVE(trx_lock->trx_locks, in_lock);
...@@ -2274,29 +2233,19 @@ lock_rec_discard( ...@@ -2274,29 +2233,19 @@ lock_rec_discard(
Removes record lock objects set on an index page which is discarded. This Removes record lock objects set on an index page which is discarded. This
function does not move locks, or check for waiting locks, therefore the function does not move locks, or check for waiting locks, therefore the
lock bitmaps must already be reset when this function is called. */ lock bitmaps must already be reset when this function is called. */
static static void lock_rec_free_all_from_discard_page_low(const page_id_t id,
void hash_table_t *lock_hash)
lock_rec_free_all_from_discard_page_low(
/*====================================*/
ulint space,
ulint page_no,
hash_table_t* lock_hash)
{ {
lock_t* lock; lock_t *lock= lock_sys.get_first(*lock_hash, id);
lock_t* next_lock;
lock = lock_rec_get_first_on_page_addr(lock_hash, space, page_no);
while (lock != NULL) { while (lock)
ut_ad(lock_rec_find_set_bit(lock) == ULINT_UNDEFINED); {
ut_ad(!lock_get_wait(lock)); ut_ad(lock_rec_find_set_bit(lock) == ULINT_UNDEFINED);
ut_ad(!lock_get_wait(lock));
next_lock = lock_rec_get_next_on_page(lock); lock_t *next_lock= lock_rec_get_next_on_page(lock);
lock_rec_discard(lock);
lock_rec_discard(lock); lock= next_lock;
}
lock = next_lock;
}
} }
/*************************************************************//** /*************************************************************//**
...@@ -2308,20 +2257,10 @@ lock_rec_free_all_from_discard_page( ...@@ -2308,20 +2257,10 @@ lock_rec_free_all_from_discard_page(
/*================================*/ /*================================*/
const buf_block_t* block) /*!< in: page to be discarded */ const buf_block_t* block) /*!< in: page to be discarded */
{ {
ulint space; const page_id_t page_id(block->page.id());
ulint page_no; lock_rec_free_all_from_discard_page_low(page_id, &lock_sys.rec_hash);
lock_rec_free_all_from_discard_page_low(page_id, &lock_sys.prdt_hash);
ut_ad(lock_mutex_own()); lock_rec_free_all_from_discard_page_low(page_id, &lock_sys.prdt_page_hash);
space = block->page.id().space();
page_no = block->page.id().page_no();
lock_rec_free_all_from_discard_page_low(
space, page_no, &lock_sys.rec_hash);
lock_rec_free_all_from_discard_page_low(
space, page_no, &lock_sys.prdt_hash);
lock_rec_free_all_from_discard_page_low(
space, page_no, &lock_sys.prdt_page_hash);
} }
/*============= RECORD LOCK MOVING AND INHERITING ===================*/ /*============= RECORD LOCK MOVING AND INHERITING ===================*/
...@@ -2588,7 +2527,7 @@ lock_move_reorganize_page( ...@@ -2588,7 +2527,7 @@ lock_move_reorganize_page(
lock_mutex_enter(); lock_mutex_enter();
/* FIXME: This needs to deal with predicate lock too */ /* FIXME: This needs to deal with predicate lock too */
lock = lock_rec_get_first_on_page(&lock_sys.rec_hash, block); lock = lock_sys.get_first(block->page.id());
if (lock == NULL) { if (lock == NULL) {
lock_mutex_exit(); lock_mutex_exit();
...@@ -2721,7 +2660,7 @@ lock_move_rec_list_end( ...@@ -2721,7 +2660,7 @@ lock_move_rec_list_end(
table to the end of the hash chain, and lock_rec_add_to_queue table to the end of the hash chain, and lock_rec_add_to_queue
does not reuse locks if there are waiters in the queue. */ does not reuse locks if there are waiters in the queue. */
for (lock = lock_rec_get_first_on_page(&lock_sys.rec_hash, block); for (lock = lock_sys.get_first(block->page.id());
lock; lock;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
const rec_t* rec1 = rec; const rec_t* rec1 = rec;
...@@ -2837,7 +2776,7 @@ lock_move_rec_list_start( ...@@ -2837,7 +2776,7 @@ lock_move_rec_list_start(
lock_mutex_enter(); lock_mutex_enter();
for (lock = lock_rec_get_first_on_page(&lock_sys.rec_hash, block); for (lock = lock_sys.get_first(block->page.id());
lock; lock;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
const rec_t* rec1; const rec_t* rec1;
...@@ -2950,7 +2889,7 @@ lock_rtr_move_rec_list( ...@@ -2950,7 +2889,7 @@ lock_rtr_move_rec_list(
lock_mutex_enter(); lock_mutex_enter();
for (lock = lock_rec_get_first_on_page(&lock_sys.rec_hash, block); for (lock = lock_sys.get_first(block->page.id());
lock; lock;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
ulint moved = 0; ulint moved = 0;
...@@ -3067,10 +3006,7 @@ lock_update_merge_right( ...@@ -3067,10 +3006,7 @@ lock_update_merge_right(
/* there should exist no page lock on the left page, /* there should exist no page lock on the left page,
otherwise, it will be blocked from merge */ otherwise, it will be blocked from merge */
ut_ad(!lock_rec_get_first_on_page_addr( ut_ad(!lock_sys.get_first_prdt_page(left_block->page.id()));
&lock_sys.prdt_page_hash,
left_block->page.id().space(),
left_block->page.id().page_no()));
lock_rec_free_all_from_discard_page(left_block); lock_rec_free_all_from_discard_page(left_block);
...@@ -3189,10 +3125,7 @@ lock_update_merge_left( ...@@ -3189,10 +3125,7 @@ lock_update_merge_left(
/* there should exist no page lock on the right page, /* there should exist no page lock on the right page,
otherwise, it will be blocked from merge */ otherwise, it will be blocked from merge */
ut_ad(!lock_rec_get_first_on_page_addr( ut_ad(!lock_sys.get_first_prdt_page(right_block->page.id()));
&lock_sys.prdt_page_hash,
right_block->page.id().space(),
right_block->page.id().page_no()));
lock_rec_free_all_from_discard_page(right_block); lock_rec_free_all_from_discard_page(right_block);
...@@ -3240,13 +3173,13 @@ lock_update_discard( ...@@ -3240,13 +3173,13 @@ lock_update_discard(
const page_t* page = block->frame; const page_t* page = block->frame;
const rec_t* rec; const rec_t* rec;
ulint heap_no; ulint heap_no;
const page_id_t page_id(block->page.id());
lock_mutex_enter(); lock_mutex_enter();
if (lock_rec_get_first_on_page(&lock_sys.rec_hash, block)) { if (lock_sys.get_first(page_id)) {
ut_ad(!lock_rec_get_first_on_page(&lock_sys.prdt_hash, block)); ut_ad(!lock_sys.get_first_prdt(page_id));
ut_ad(!lock_rec_get_first_on_page(&lock_sys.prdt_page_hash, ut_ad(!lock_sys.get_first_prdt_page(page_id));
block));
/* Inherit all the locks on the page to the record and /* Inherit all the locks on the page to the record and
reset all the locks on the page */ reset all the locks on the page */
...@@ -3280,16 +3213,13 @@ lock_update_discard( ...@@ -3280,16 +3213,13 @@ lock_update_discard(
} while (heap_no != PAGE_HEAP_NO_SUPREMUM); } while (heap_no != PAGE_HEAP_NO_SUPREMUM);
} }
lock_rec_free_all_from_discard_page_low( lock_rec_free_all_from_discard_page_low(page_id,
block->page.id().space(), block->page.id().page_no(), &lock_sys.rec_hash);
&lock_sys.rec_hash);
} else { } else {
lock_rec_free_all_from_discard_page_low(page_id,
&lock_sys.prdt_hash);
lock_rec_free_all_from_discard_page_low( lock_rec_free_all_from_discard_page_low(
block->page.id().space(), block->page.id().page_no(), page_id, &lock_sys.prdt_page_hash);
&lock_sys.prdt_hash);
lock_rec_free_all_from_discard_page_low(
block->page.id().space(), block->page.id().page_no(),
&lock_sys.prdt_page_hash);
} }
lock_mutex_exit(); lock_mutex_exit();
...@@ -4061,16 +3991,10 @@ lock_grant_and_move_on_rec( ...@@ -4061,16 +3991,10 @@ lock_grant_and_move_on_rec(
ulint heap_no) ulint heap_no)
{ {
lock_t* lock; lock_t* lock;
ulint space; const page_id_t page_id(first_lock->un_member.rec_lock.page_id);
ulint page_no; const ulint rec_fold= page_id.fold();
ulint rec_fold;
space = first_lock->un_member.rec_lock.space;
page_no = first_lock->un_member.rec_lock.page_no;
rec_fold = lock_rec_fold(space, page_no);
lock_t* previous = static_cast<lock_t*>( lock_t* previous = static_cast<lock_t*>(
lock_sys.rec_hash.array[lock_sys.rec_hash.calc_hash(rec_fold)] lock_sys.rec_hash.array[lock_sys.hash(page_id)]
.node); .node);
if (previous == NULL) { if (previous == NULL) {
return; return;
...@@ -4089,8 +4013,7 @@ lock_grant_and_move_on_rec( ...@@ -4089,8 +4013,7 @@ lock_grant_and_move_on_rec(
for (;lock != NULL;) { for (;lock != NULL;) {
/* If the lock is a wait lock on this page, and it does not need to wait. */ /* If the lock is a wait lock on this page, and it does not need to wait. */
if (lock->un_member.rec_lock.space == space if (lock->un_member.rec_lock.page_id == page_id
&& lock->un_member.rec_lock.page_no == page_no
&& lock_rec_get_nth_bit(lock, heap_no) && lock_rec_get_nth_bit(lock, heap_no)
&& lock_get_wait(lock) && lock_get_wait(lock)
&& !lock_rec_has_to_wait_in_queue(lock)) { && !lock_rec_has_to_wait_in_queue(lock)) {
...@@ -4392,19 +4315,15 @@ lock_table_print(FILE* file, const lock_t* lock) ...@@ -4392,19 +4315,15 @@ lock_table_print(FILE* file, const lock_t* lock)
@param[in,out] mtr mini-transaction for accessing the record */ @param[in,out] mtr mini-transaction for accessing the record */
static void lock_rec_print(FILE* file, const lock_t* lock, mtr_t& mtr) static void lock_rec_print(FILE* file, const lock_t* lock, mtr_t& mtr)
{ {
ulint space;
ulint page_no;
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
ut_a(lock_get_type_low(lock) == LOCK_REC); ut_a(lock_get_type_low(lock) == LOCK_REC);
space = lock->un_member.rec_lock.space; const page_id_t page_id(lock->un_member.rec_lock.page_id);
page_no = lock->un_member.rec_lock.page_no;
fprintf(file, "RECORD LOCKS space id %lu page no %lu n bits %lu " fprintf(file, "RECORD LOCKS space id %u page no %u n bits " ULINTPF
"index %s of table ", " index %s of table ",
(ulong) space, (ulong) page_no, page_id.space(), page_id.page_no(),
(ulong) lock_rec_get_n_bits(lock), lock_rec_get_n_bits(lock),
lock->index->name()); lock->index->name());
ut_print_name(file, lock->trx, lock->index->table->name.m_name); ut_print_name(file, lock->trx, lock->index->table->name.m_name);
fprintf(file, " trx id " TRX_ID_FMT, trx_get_id_for_print(lock->trx)); fprintf(file, " trx id " TRX_ID_FMT, trx_get_id_for_print(lock->trx));
...@@ -4441,8 +4360,7 @@ static void lock_rec_print(FILE* file, const lock_t* lock, mtr_t& mtr) ...@@ -4441,8 +4360,7 @@ static void lock_rec_print(FILE* file, const lock_t* lock, mtr_t& mtr)
rec_offs_init(offsets_); rec_offs_init(offsets_);
mtr.start(); mtr.start();
const buf_block_t* block = buf_page_try_get(page_id_t(space, page_no), const buf_block_t* block = buf_page_try_get(page_id, &mtr);
&mtr);
for (ulint i = 0; i < lock_rec_get_n_bits(lock); ++i) { for (ulint i = 0; i < lock_rec_get_n_bits(lock); ++i) {
...@@ -4941,13 +4859,9 @@ lock_rec_validate_page( ...@@ -4941,13 +4859,9 @@ lock_rec_validate_page(
rec_offs* offsets = offsets_; rec_offs* offsets = offsets_;
rec_offs_init(offsets_); rec_offs_init(offsets_);
ut_ad(!lock_mutex_own());
lock_mutex_enter(); lock_mutex_enter();
loop: loop:
lock = lock_rec_get_first_on_page_addr( lock = lock_sys.get_first(block->page.id());
&lock_sys.rec_hash,
block->page.id().space(), block->page.id().page_no());
if (!lock) { if (!lock) {
goto function_exit; goto function_exit;
...@@ -5019,7 +4933,7 @@ lock_rec_validate( ...@@ -5019,7 +4933,7 @@ lock_rec_validate(
/*==============*/ /*==============*/
ulint start, /*!< in: lock_sys.rec_hash ulint start, /*!< in: lock_sys.rec_hash
bucket */ bucket */
ib_uint64_t* limit) /*!< in/out: upper limit of page_id_t* limit) /*!< in/out: upper limit of
(space, page_no) */ (space, page_no) */
{ {
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
...@@ -5029,14 +4943,10 @@ lock_rec_validate( ...@@ -5029,14 +4943,10 @@ lock_rec_validate(
lock != NULL; lock != NULL;
lock = static_cast<const lock_t*>(HASH_GET_NEXT(hash, lock))) { lock = static_cast<const lock_t*>(HASH_GET_NEXT(hash, lock))) {
ib_uint64_t current;
ut_ad(!trx_is_ac_nl_ro(lock->trx)); ut_ad(!trx_is_ac_nl_ro(lock->trx));
ut_ad(lock_get_type(lock) == LOCK_REC); ut_ad(lock_get_type(lock) == LOCK_REC);
current = ut_ull_create( page_id_t current(lock->un_member.rec_lock.page_id);
lock->un_member.rec_lock.space,
lock->un_member.rec_lock.page_no);
if (current > *limit) { if (current > *limit) {
*limit = current + 1; *limit = current + 1;
...@@ -5049,12 +4959,7 @@ lock_rec_validate( ...@@ -5049,12 +4959,7 @@ lock_rec_validate(
/*********************************************************************//** /*********************************************************************//**
Validate a record lock's block */ Validate a record lock's block */
static static void lock_rec_block_validate(const page_id_t page_id)
void
lock_rec_block_validate(
/*====================*/
ulint space_id,
ulint page_no)
{ {
/* The lock and the block that it is referring to may be freed at /* The lock and the block that it is referring to may be freed at
this point. We pass BUF_GET_POSSIBLY_FREED to skip a debug check. this point. We pass BUF_GET_POSSIBLY_FREED to skip a debug check.
...@@ -5069,12 +4974,12 @@ lock_rec_block_validate( ...@@ -5069,12 +4974,12 @@ lock_rec_block_validate(
discard or rebuild a tablespace do hold an exclusive table discard or rebuild a tablespace do hold an exclusive table
lock, which would conflict with any locks referring to the lock, which would conflict with any locks referring to the
tablespace from other transactions. */ tablespace from other transactions. */
if (fil_space_t* space = fil_space_acquire(space_id)) { if (fil_space_t* space = fil_space_acquire(page_id.space())) {
dberr_t err = DB_SUCCESS; dberr_t err = DB_SUCCESS;
mtr_start(&mtr); mtr_start(&mtr);
block = buf_page_get_gen( block = buf_page_get_gen(
page_id_t(space_id, page_no), page_id,
space->zip_size(), space->zip_size(),
RW_X_LATCH, NULL, RW_X_LATCH, NULL,
BUF_GET_POSSIBLY_FREED, BUF_GET_POSSIBLY_FREED,
...@@ -5083,8 +4988,7 @@ lock_rec_block_validate( ...@@ -5083,8 +4988,7 @@ lock_rec_block_validate(
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
ib::error() << "Lock rec block validate failed for tablespace " ib::error() << "Lock rec block validate failed for tablespace "
<< space->name << space->name
<< " space_id " << space_id << page_id << " err " << err;
<< " page_no " << page_no << " err " << err;
} }
if (block) { if (block) {
...@@ -5128,13 +5032,7 @@ bool ...@@ -5128,13 +5032,7 @@ bool
lock_validate() lock_validate()
/*===========*/ /*===========*/
{ {
typedef std::pair<ulint, ulint> page_addr_t; std::set<page_id_t> pages;
typedef std::set<
page_addr_t,
std::less<page_addr_t>,
ut_allocator<page_addr_t> > page_addr_set;
page_addr_set pages;
lock_mutex_enter(); lock_mutex_enter();
...@@ -5146,24 +5044,21 @@ lock_validate() ...@@ -5146,24 +5044,21 @@ lock_validate()
validation check. */ validation check. */
for (ulint i = 0; i < lock_sys.rec_hash.n_cells; i++) { for (ulint i = 0; i < lock_sys.rec_hash.n_cells; i++) {
ib_uint64_t limit = 0; page_id_t limit(0, 0);
while (const lock_t* lock = lock_rec_validate(i, &limit)) { while (const lock_t* lock = lock_rec_validate(i, &limit)) {
if (lock_rec_find_set_bit(lock) == ULINT_UNDEFINED) { if (lock_rec_find_set_bit(lock) == ULINT_UNDEFINED) {
/* The lock bitmap is empty; ignore it. */ /* The lock bitmap is empty; ignore it. */
continue; continue;
} }
const lock_rec_t& l = lock->un_member.rec_lock; pages.insert(lock->un_member.rec_lock.page_id);
pages.insert(std::make_pair(l.space, l.page_no));
} }
} }
lock_mutex_exit(); lock_mutex_exit();
for (page_addr_set::const_iterator it = pages.begin(); for (const page_id_t page_id : pages) {
it != pages.end(); lock_rec_block_validate(page_id);
++it) {
lock_rec_block_validate((*it).first, (*it).second);
} }
return(true); return(true);
...@@ -6460,10 +6355,6 @@ DeadlockChecker::get_first_lock(ulint* heap_no) const ...@@ -6460,10 +6355,6 @@ DeadlockChecker::get_first_lock(ulint* heap_no) const
const lock_t* lock = m_wait_lock; const lock_t* lock = m_wait_lock;
if (lock_get_type_low(lock) == LOCK_REC) { if (lock_get_type_low(lock) == LOCK_REC) {
hash_table_t* lock_hash = lock->type_mode & LOCK_PREDICATE
? &lock_sys.prdt_hash
: &lock_sys.rec_hash;
/* We are only interested in records that match the heap_no. */ /* We are only interested in records that match the heap_no. */
*heap_no = lock_rec_find_set_bit(lock); *heap_no = lock_rec_find_set_bit(lock);
...@@ -6471,10 +6362,11 @@ DeadlockChecker::get_first_lock(ulint* heap_no) const ...@@ -6471,10 +6362,11 @@ DeadlockChecker::get_first_lock(ulint* heap_no) const
ut_ad(*heap_no != ULINT_UNDEFINED); ut_ad(*heap_no != ULINT_UNDEFINED);
/* Find the locks on the page. */ /* Find the locks on the page. */
lock = lock_rec_get_first_on_page_addr( lock = lock_sys.get_first(
lock_hash, lock->type_mode & LOCK_PREDICATE
lock->un_member.rec_lock.space, ? lock_sys.prdt_hash
lock->un_member.rec_lock.page_no); : lock_sys.rec_hash,
lock->un_member.rec_lock.page_id);
/* Position on the first lock on the physical record.*/ /* Position on the first lock on the physical record.*/
if (!lock_rec_get_nth_bit(lock, *heap_no)) { if (!lock_rec_get_nth_bit(lock, *heap_no)) {
......
...@@ -394,7 +394,8 @@ lock_prdt_find_on_page( ...@@ -394,7 +394,8 @@ lock_prdt_find_on_page(
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
for (lock = lock_rec_get_first_on_page(lock_hash_get(type_mode), block); for (lock = lock_sys.get_first(*lock_hash_get(type_mode),
block->page.id());
lock != NULL; lock != NULL;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
...@@ -457,7 +458,8 @@ lock_prdt_add_to_queue( ...@@ -457,7 +458,8 @@ lock_prdt_add_to_queue(
lock_t* lock; lock_t* lock;
for (lock = lock_rec_get_first_on_page(lock_hash_get(type_mode), block); for (lock = lock_sys.get_first(*lock_hash_get(type_mode),
block->page.id());
lock != NULL; lock != NULL;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
...@@ -619,16 +621,12 @@ lock_prdt_update_parent( ...@@ -619,16 +621,12 @@ lock_prdt_update_parent(
buf_block_t* right_block, /*!< in/out: the new half page */ buf_block_t* right_block, /*!< in/out: the new half page */
lock_prdt_t* left_prdt, /*!< in: MBR on the old page */ lock_prdt_t* left_prdt, /*!< in: MBR on the old page */
lock_prdt_t* right_prdt, /*!< in: MBR on the new page */ lock_prdt_t* right_prdt, /*!< in: MBR on the new page */
ulint space, /*!< in: parent space id */ const page_id_t page_id) /*!< in: parent page */
ulint page_no) /*!< in: parent page number */
{ {
lock_t* lock;
lock_mutex_enter(); lock_mutex_enter();
/* Get all locks in parent */ /* Get all locks in parent */
for (lock = lock_rec_get_first_on_page_addr( for (lock_t *lock = lock_sys.get_first_prdt(page_id);
&lock_sys.prdt_hash, space, page_no);
lock; lock;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
lock_prdt_t* lock_prdt; lock_prdt_t* lock_prdt;
...@@ -675,21 +673,15 @@ lock_prdt_update_split_low( ...@@ -675,21 +673,15 @@ lock_prdt_update_split_low(
buf_block_t* new_block, /*!< in/out: the new half page */ buf_block_t* new_block, /*!< in/out: the new half page */
lock_prdt_t* prdt, /*!< in: MBR on the old page */ lock_prdt_t* prdt, /*!< in: MBR on the old page */
lock_prdt_t* new_prdt, /*!< in: MBR on the new page */ lock_prdt_t* new_prdt, /*!< in: MBR on the new page */
ulint space, /*!< in: space id */ const page_id_t page_id, /*!< in: page number */
ulint page_no, /*!< in: page number */
unsigned type_mode) /*!< in: LOCK_PREDICATE or unsigned type_mode) /*!< in: LOCK_PREDICATE or
LOCK_PRDT_PAGE */ LOCK_PRDT_PAGE */
{ {
lock_t* lock; lock_t* lock;
lock_mutex_enter(); for (lock = lock_sys.get_first(*lock_hash_get(type_mode), page_id);
for (lock = lock_rec_get_first_on_page_addr(
lock_hash_get(type_mode), space, page_no);
lock; lock;
lock = lock_rec_get_next_on_page(lock)) { lock = lock_rec_get_next_on_page(lock)) {
ut_ad(lock);
/* First dealing with Page Lock */ /* First dealing with Page Lock */
if (lock->type_mode & LOCK_PRDT_PAGE) { if (lock->type_mode & LOCK_PRDT_PAGE) {
/* Duplicate the lock to new page */ /* Duplicate the lock to new page */
...@@ -739,8 +731,6 @@ lock_prdt_update_split_low( ...@@ -739,8 +731,6 @@ lock_prdt_update_split_low(
trx_mutex_exit(lock->trx); trx_mutex_exit(lock->trx);
} }
} }
lock_mutex_exit();
} }
/**************************************************************//** /**************************************************************//**
...@@ -751,14 +741,17 @@ lock_prdt_update_split( ...@@ -751,14 +741,17 @@ lock_prdt_update_split(
buf_block_t* new_block, /*!< in/out: the new half page */ buf_block_t* new_block, /*!< in/out: the new half page */
lock_prdt_t* prdt, /*!< in: MBR on the old page */ lock_prdt_t* prdt, /*!< in: MBR on the old page */
lock_prdt_t* new_prdt, /*!< in: MBR on the new page */ lock_prdt_t* new_prdt, /*!< in: MBR on the new page */
ulint space, /*!< in: space id */ const page_id_t page_id) /*!< in: page number */
ulint page_no) /*!< in: page number */
{ {
lock_mutex_enter();
lock_prdt_update_split_low(new_block, prdt, new_prdt, lock_prdt_update_split_low(new_block, prdt, new_prdt,
space, page_no, LOCK_PREDICATE); page_id, LOCK_PREDICATE);
lock_prdt_update_split_low(new_block, NULL, NULL, lock_prdt_update_split_low(new_block, NULL, NULL,
space, page_no, LOCK_PRDT_PAGE); page_id, LOCK_PRDT_PAGE);
lock_mutex_exit();
} }
/*********************************************************************//** /*********************************************************************//**
...@@ -814,9 +807,9 @@ lock_prdt_lock( ...@@ -814,9 +807,9 @@ lock_prdt_lock(
ut_ad(!dict_index_is_online_ddl(index)); ut_ad(!dict_index_is_online_ddl(index));
ut_ad(type_mode & (LOCK_PREDICATE | LOCK_PRDT_PAGE)); ut_ad(type_mode & (LOCK_PREDICATE | LOCK_PRDT_PAGE));
hash_table_t* hash = type_mode == LOCK_PREDICATE const hash_table_t& hash = type_mode == LOCK_PREDICATE
? &lock_sys.prdt_hash ? lock_sys.prdt_hash
: &lock_sys.prdt_page_hash; : lock_sys.prdt_page_hash;
/* Another transaction cannot have an implicit lock on the record, /* Another transaction cannot have an implicit lock on the record,
because when we come here, we already have modified the clustered because when we come here, we already have modified the clustered
...@@ -826,7 +819,7 @@ lock_prdt_lock( ...@@ -826,7 +819,7 @@ lock_prdt_lock(
lock_mutex_enter(); lock_mutex_enter();
const unsigned prdt_mode = type_mode | mode; const unsigned prdt_mode = type_mode | mode;
lock_t* lock = lock_rec_get_first_on_page(hash, block); lock_t* lock = lock_sys.get_first(hash, block->page.id());
if (lock == NULL) { if (lock == NULL) {
lock = lock_rec_create( lock = lock_rec_create(
...@@ -905,9 +898,7 @@ Acquire a "Page" lock on a block ...@@ -905,9 +898,7 @@ Acquire a "Page" lock on a block
@return DB_SUCCESS, DB_LOCK_WAIT, or DB_DEADLOCK */ @return DB_SUCCESS, DB_LOCK_WAIT, or DB_DEADLOCK */
dberr_t dberr_t
lock_place_prdt_page_lock( lock_place_prdt_page_lock(
/*======================*/ const page_id_t page_id, /*!< in: page identifier */
ulint space, /*!< in: space for the page to lock */
ulint page_no, /*!< in: page number */
dict_index_t* index, /*!< in: secondary index */ dict_index_t* index, /*!< in: secondary index */
que_thr_t* thr) /*!< in: query thread */ que_thr_t* thr) /*!< in: query thread */
{ {
...@@ -924,9 +915,7 @@ lock_place_prdt_page_lock( ...@@ -924,9 +915,7 @@ lock_place_prdt_page_lock(
lock_mutex_enter(); lock_mutex_enter();
const lock_t* lock = lock_rec_get_first_on_page_addr( const lock_t* lock = lock_sys.get_first_prdt_page(page_id);
&lock_sys.prdt_page_hash, space, page_no);
const ulint mode = LOCK_S | LOCK_PRDT_PAGE; const ulint mode = LOCK_S | LOCK_PRDT_PAGE;
trx_t* trx = thr_get_trx(thr); trx_t* trx = thr_get_trx(thr);
...@@ -952,7 +941,7 @@ lock_place_prdt_page_lock( ...@@ -952,7 +941,7 @@ lock_place_prdt_page_lock(
#ifdef WITH_WSREP #ifdef WITH_WSREP
NULL, NULL, /* FIXME: replicate SPATIAL INDEX locks */ NULL, NULL, /* FIXME: replicate SPATIAL INDEX locks */
#endif #endif
mode, space, page_no, NULL, PRDT_HEAPNO, mode, page_id, NULL, PRDT_HEAPNO,
index, trx, FALSE); index, trx, FALSE);
#ifdef PRDT_DIAG #ifdef PRDT_DIAG
...@@ -967,25 +956,19 @@ lock_place_prdt_page_lock( ...@@ -967,25 +956,19 @@ lock_place_prdt_page_lock(
/** Check whether there are R-tree Page lock on a page /** Check whether there are R-tree Page lock on a page
@param[in] trx trx to test the lock @param[in] trx trx to test the lock
@param[in] space space id for the page @param[in] page_id page identifier
@param[in] page_no page number
@return true if there is none */ @return true if there is none */
bool bool lock_test_prdt_page_lock(const trx_t *trx, const page_id_t page_id)
lock_test_prdt_page_lock(
const trx_t* trx,
ulint space,
ulint page_no)
{ {
lock_t* lock; lock_t* lock;
lock_mutex_enter(); lock_mutex_enter();
lock = lock_rec_get_first_on_page_addr( lock = lock_sys.get_first_prdt_page(page_id);
&lock_sys.prdt_page_hash, space, page_no);
lock_mutex_exit(); lock_mutex_exit();
return(lock == NULL || trx == lock->trx); return(!lock || trx == lock->trx);
} }
/*************************************************************//** /*************************************************************//**
...@@ -1030,15 +1013,10 @@ lock_prdt_page_free_from_discard( ...@@ -1030,15 +1013,10 @@ lock_prdt_page_free_from_discard(
{ {
lock_t* lock; lock_t* lock;
lock_t* next_lock; lock_t* next_lock;
ulint space;
ulint page_no;
ut_ad(lock_mutex_own()); ut_ad(lock_mutex_own());
space = block->page.id().space(); lock = lock_sys.get_first(*lock_hash, block->page.id());
page_no = block->page.id().page_no();
lock = lock_rec_get_first_on_page_addr(lock_hash, space, page_no);
while (lock != NULL) { while (lock != NULL) {
next_lock = lock_rec_get_next_on_page(lock); next_lock = lock_rec_get_next_on_page(lock);
......
...@@ -499,9 +499,8 @@ row_purge_remove_sec_if_poss_leaf( ...@@ -499,9 +499,8 @@ row_purge_remove_sec_if_poss_leaf(
&& btr_cur->rtr_info->thr && btr_cur->rtr_info->thr
? thr_get_trx( ? thr_get_trx(
btr_cur->rtr_info->thr) btr_cur->rtr_info->thr)
: NULL, : nullptr,
block->page.id().space(), block->page.id())) {
block->page.id().page_no())) {
/* this is the last record on page, /* this is the last record on page,
and it has a "page" lock on it, and it has a "page" lock on it,
which mean search is still depending which mean search is still depending
......
...@@ -394,13 +394,11 @@ i_s_locks_row_validate( ...@@ -394,13 +394,11 @@ i_s_locks_row_validate(
if (!row->lock_index) { if (!row->lock_index) {
/* table lock */ /* table lock */
ut_ad(!row->lock_data); ut_ad(!row->lock_data);
ut_ad(!row->lock_space); ut_ad(row->lock_page == page_id_t(0, 0));
ut_ad(!row->lock_page);
ut_ad(!row->lock_rec); ut_ad(!row->lock_rec);
} else { } else {
/* record lock */ /* record lock */
/* row->lock_data == NULL if buf_page_try_get() == NULL */ /* row->lock_data == NULL if buf_page_try_get() == NULL */
ut_ad(row->lock_page);
} }
return(TRUE); return(TRUE);
...@@ -631,9 +629,7 @@ fill_lock_data( ...@@ -631,9 +629,7 @@ fill_lock_data(
mtr_start(&mtr); mtr_start(&mtr);
block = buf_page_try_get(page_id_t(lock->un_member.rec_lock.space, block = buf_page_try_get(lock->un_member.rec_lock.page_id, &mtr);
lock->un_member.rec_lock.page_no),
&mtr);
if (block == NULL) { if (block == NULL) {
...@@ -754,8 +750,7 @@ static bool fill_locks_row( ...@@ -754,8 +750,7 @@ static bool fill_locks_row(
return false; return false;
} }
row->lock_space = lock->un_member.rec_lock.space; row->lock_page = lock->un_member.rec_lock.page_id;
row->lock_page = lock->un_member.rec_lock.page_no;
row->lock_rec = heap_no; row->lock_rec = heap_no;
if (!fill_lock_data(&row->lock_data, lock, heap_no, cache)) { if (!fill_lock_data(&row->lock_data, lock, heap_no, cache)) {
...@@ -766,8 +761,7 @@ static bool fill_locks_row( ...@@ -766,8 +761,7 @@ static bool fill_locks_row(
} else { } else {
row->lock_index = NULL; row->lock_index = NULL;
row->lock_space = 0; row->lock_page = page_id_t(0, 0);
row->lock_page = 0;
row->lock_rec = 0; row->lock_rec = 0;
row->lock_data = NULL; row->lock_data = NULL;
...@@ -831,13 +825,9 @@ fold_lock( ...@@ -831,13 +825,9 @@ fold_lock(
switch (lock_get_type(lock)) { switch (lock_get_type(lock)) {
case LOCK_REC: case LOCK_REC:
ut_a(heap_no != 0xFFFF); ut_a(heap_no != 0xFFFF);
ret = ut_fold_ulint_pair((ulint) lock->trx->id, ret = ut_fold_ulint_pair((ulint) lock->trx->id,
lock->un_member.rec_lock.space); lock->un_member.rec_lock.page_id.
fold());
ret = ut_fold_ulint_pair(ret,
lock->un_member.rec_lock.page_no);
ret = ut_fold_ulint_pair(ret, heap_no); ret = ut_fold_ulint_pair(ret, heap_no);
break; break;
...@@ -880,8 +870,7 @@ locks_row_eq_lock( ...@@ -880,8 +870,7 @@ locks_row_eq_lock(
ut_a(heap_no != 0xFFFF); ut_a(heap_no != 0xFFFF);
return(row->lock_trx_id == lock->trx->id return(row->lock_trx_id == lock->trx->id
&& row->lock_space == lock->un_member.rec_lock.space && row->lock_page == lock->un_member.rec_lock.page_id
&& row->lock_page == lock->un_member.rec_lock.page_no
&& row->lock_rec == heap_no); && row->lock_rec == heap_no);
case LOCK_TABLE: case LOCK_TABLE:
...@@ -1477,8 +1466,8 @@ trx_i_s_create_lock_id( ...@@ -1477,8 +1466,8 @@ trx_i_s_create_lock_id(
res_len = snprintf(lock_id, lock_id_size, res_len = snprintf(lock_id, lock_id_size,
TRX_ID_FMT TRX_ID_FMT
":%u:%u:%u", ":%u:%u:%u",
row->lock_trx_id, row->lock_space, row->lock_trx_id, row->lock_page.space(),
row->lock_page, row->lock_rec); row->lock_page.page_no(), row->lock_rec);
} else { } else {
/* table lock */ /* table lock */
res_len = snprintf(lock_id, lock_id_size, res_len = snprintf(lock_id, lock_id_size,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment