Commit 438cb8a6 authored by marko's avatar marko

branches/zip: Remove all buf_block_align() calls from lock0lock.c.

Replace all page_t* parameters with buf_block_t*, and replace many
rec_t* parameters with heap_no.  This eliminates also many
rec_get_heap_no() calls, which became more expensive with the
introduction of ROW_FORMAT=COMPACT in MySQL/InnoDB 5.0.3.

page_rec_get_heap_no(), lock_get_min_heap_no(): New functions.
parent 483a5708
...@@ -929,7 +929,7 @@ btr_page_reorganize_low( ...@@ -929,7 +929,7 @@ btr_page_reorganize_low(
if (UNIV_LIKELY(!recovery)) { if (UNIV_LIKELY(!recovery)) {
/* Update the record lock bitmaps */ /* Update the record lock bitmaps */
lock_move_reorganize_page(page, temp_page); lock_move_reorganize_page(block, temp_block);
} }
data_size2 = page_get_data_size(page); data_size2 = page_get_data_size(page);
...@@ -1122,7 +1122,7 @@ btr_root_raise_and_insert( ...@@ -1122,7 +1122,7 @@ btr_root_raise_and_insert(
information of the record to be inserted on the infimum of the information of the record to be inserted on the infimum of the
root page: we cannot discard the lock structs on the root page */ root page: we cannot discard the lock structs on the root page */
lock_update_root_raise(new_page, root); lock_update_root_raise(new_block, root_block);
/* Create a memory heap where the node pointer is stored */ /* Create a memory heap where the node pointer is stored */
heap = mem_heap_create(100); heap = mem_heap_create(100);
...@@ -1851,7 +1851,7 @@ func_start: ...@@ -1851,7 +1851,7 @@ func_start:
left_block = new_block; left_block = new_block;
right_block = block; right_block = block;
lock_update_split_left(page, new_page); lock_update_split_left(right_block, left_block);
} else { } else {
/* fputs("Split right\n", stderr); */ /* fputs("Split right\n", stderr); */
...@@ -1879,7 +1879,7 @@ func_start: ...@@ -1879,7 +1879,7 @@ func_start:
left_block = block; left_block = block;
right_block = new_block; right_block = new_block;
lock_update_split_right(new_page, page); lock_update_split_right(right_block, left_block);
} }
#ifdef UNIV_ZIP_DEBUG #ifdef UNIV_ZIP_DEBUG
...@@ -2197,7 +2197,7 @@ btr_lift_page_up( ...@@ -2197,7 +2197,7 @@ btr_lift_page_up(
page, index, mtr); page, index, mtr);
} }
lock_update_copy_and_discard(father_page, page); lock_update_copy_and_discard(father_block, block);
/* Free the file page */ /* Free the file page */
btr_page_free(index, block, mtr); btr_page_free(index, block, mtr);
...@@ -2367,7 +2367,7 @@ err_exit: ...@@ -2367,7 +2367,7 @@ err_exit:
btr_level_list_remove(page, mtr); btr_level_list_remove(page, mtr);
btr_node_ptr_delete(index, block, mtr); btr_node_ptr_delete(index, block, mtr);
lock_update_merge_left(merge_page, orig_pred, page); lock_update_merge_left(merge_block, orig_pred, block);
} else { } else {
rec_t* orig_succ; rec_t* orig_succ;
#ifdef UNIV_BTR_DEBUG #ifdef UNIV_BTR_DEBUG
...@@ -2426,7 +2426,7 @@ err_exit: ...@@ -2426,7 +2426,7 @@ err_exit:
offsets, right_page_no, mtr); offsets, right_page_no, mtr);
btr_node_ptr_delete(index, merge_block, mtr); btr_node_ptr_delete(index, merge_block, mtr);
lock_update_merge_right(orig_succ, page); lock_update_merge_right(merge_block, orig_succ, block);
} }
mem_heap_free(heap); mem_heap_free(heap);
...@@ -2473,7 +2473,7 @@ btr_discard_only_page_on_level( ...@@ -2473,7 +2473,7 @@ btr_discard_only_page_on_level(
page_level = btr_page_get_level(page, mtr); page_level = btr_page_get_level(page, mtr);
lock_update_discard(page_get_supremum_rec(father_page), page); lock_update_discard(father_block, PAGE_HEAP_NO_SUPREMUM, block);
btr_page_set_level(father_page, father_page_zip, page_level, mtr); btr_page_set_level(father_page, father_page_zip, page_level, mtr);
...@@ -2511,6 +2511,7 @@ btr_discard_page( ...@@ -2511,6 +2511,7 @@ btr_discard_page(
ulint space; ulint space;
ulint left_page_no; ulint left_page_no;
ulint right_page_no; ulint right_page_no;
buf_block_t* merge_block;
page_t* merge_page; page_t* merge_page;
buf_block_t* block; buf_block_t* block;
page_t* page; page_t* page;
...@@ -2531,14 +2532,16 @@ btr_discard_page( ...@@ -2531,14 +2532,16 @@ btr_discard_page(
right_page_no = btr_page_get_next(buf_block_get_frame(block), mtr); right_page_no = btr_page_get_next(buf_block_get_frame(block), mtr);
if (left_page_no != FIL_NULL) { if (left_page_no != FIL_NULL) {
merge_page = btr_page_get(space, left_page_no, RW_X_LATCH, merge_block = btr_block_get(space, left_page_no, RW_X_LATCH,
mtr); mtr);
merge_page = buf_block_get_frame(merge_block);
#ifdef UNIV_BTR_DEBUG #ifdef UNIV_BTR_DEBUG
ut_a(btr_page_get_next(merge_page, mtr) == block->offset); ut_a(btr_page_get_next(merge_page, mtr) == block->offset);
#endif /* UNIV_BTR_DEBUG */ #endif /* UNIV_BTR_DEBUG */
} else if (right_page_no != FIL_NULL) { } else if (right_page_no != FIL_NULL) {
merge_page = btr_page_get(space, right_page_no, RW_X_LATCH, merge_block = btr_block_get(space, right_page_no, RW_X_LATCH,
mtr); mtr);
merge_page = buf_block_get_frame(merge_block);
#ifdef UNIV_BTR_DEBUG #ifdef UNIV_BTR_DEBUG
ut_a(btr_page_get_prev(merge_page, mtr) == block->offset); ut_a(btr_page_get_prev(merge_page, mtr) == block->offset);
#endif /* UNIV_BTR_DEBUG */ #endif /* UNIV_BTR_DEBUG */
...@@ -2575,18 +2578,19 @@ btr_discard_page( ...@@ -2575,18 +2578,19 @@ btr_discard_page(
#ifdef UNIV_ZIP_DEBUG #ifdef UNIV_ZIP_DEBUG
{ {
page_zip_des_t* merge_page_zip page_zip_des_t* merge_page_zip
= buf_frame_get_page_zip(merge_page); = buf_block_get_page_zip(merge_block);
ut_a(!merge_page_zip ut_a(!merge_page_zip
|| page_zip_validate(merge_page_zip, merge_page)); || page_zip_validate(merge_page_zip, merge_page));
} }
#endif /* UNIV_ZIP_DEBUG */ #endif /* UNIV_ZIP_DEBUG */
if (left_page_no != FIL_NULL) { if (left_page_no != FIL_NULL) {
lock_update_discard(page_get_supremum_rec(merge_page), page); lock_update_discard(merge_block, PAGE_HEAP_NO_SUPREMUM,
block);
} else { } else {
lock_update_discard(page_rec_get_next( lock_update_discard(merge_block,
page_get_infimum_rec(merge_page)), lock_get_min_heap_no(merge_block),
page); block);
} }
/* Free the file page */ /* Free the file page */
......
...@@ -1189,7 +1189,7 @@ fail: ...@@ -1189,7 +1189,7 @@ fail:
if (!(flags & BTR_NO_LOCKING_FLAG) && inherit) { if (!(flags & BTR_NO_LOCKING_FLAG) && inherit) {
lock_update_insert(*rec); lock_update_insert(block, *rec);
} }
#if 0 #if 0
...@@ -1389,7 +1389,7 @@ btr_cur_pessimistic_insert( ...@@ -1389,7 +1389,7 @@ btr_cur_pessimistic_insert(
#endif #endif
if (!(flags & BTR_NO_LOCKING_FLAG)) { if (!(flags & BTR_NO_LOCKING_FLAG)) {
lock_update_insert(*rec); lock_update_insert(btr_cur_get_block(cursor), *rec);
} }
err = DB_SUCCESS; err = DB_SUCCESS;
...@@ -1434,7 +1434,7 @@ btr_cur_upd_lock_and_undo( ...@@ -1434,7 +1434,7 @@ btr_cur_upd_lock_and_undo(
/* We do undo logging only when we update a clustered index /* We do undo logging only when we update a clustered index
record */ record */
return(lock_sec_rec_modify_check_and_lock( return(lock_sec_rec_modify_check_and_lock(
flags, rec, btr_cur_get_block(cursor), flags, btr_cur_get_block(cursor), rec,
index, thr)); index, thr));
} }
...@@ -1449,7 +1449,7 @@ btr_cur_upd_lock_and_undo( ...@@ -1449,7 +1449,7 @@ btr_cur_upd_lock_and_undo(
*offsets_ = (sizeof offsets_) / sizeof *offsets_; *offsets_ = (sizeof offsets_) / sizeof *offsets_;
err = lock_clust_rec_modify_check_and_lock( err = lock_clust_rec_modify_check_and_lock(
flags, rec, index, flags, btr_cur_get_block(cursor), rec, index,
rec_get_offsets(rec, index, offsets_, rec_get_offsets(rec, index, offsets_,
ULINT_UNDEFINED, &heap), thr); ULINT_UNDEFINED, &heap), thr);
if (UNIV_LIKELY_NULL(heap)) { if (UNIV_LIKELY_NULL(heap)) {
...@@ -1872,7 +1872,7 @@ btr_cur_optimistic_update( ...@@ -1872,7 +1872,7 @@ btr_cur_optimistic_update(
explicit locks on rec, before deleting rec (see the comment in explicit locks on rec, before deleting rec (see the comment in
.._pessimistic_update). */ .._pessimistic_update). */
lock_rec_store_on_page_infimum(page, rec); lock_rec_store_on_page_infimum(block, rec);
btr_search_update_hash_on_delete(cursor); btr_search_update_hash_on_delete(cursor);
...@@ -1905,7 +1905,7 @@ btr_cur_optimistic_update( ...@@ -1905,7 +1905,7 @@ btr_cur_optimistic_update(
/* Restore the old explicit lock state on the record */ /* Restore the old explicit lock state on the record */
lock_rec_restore_from_page_infimum(rec, page); lock_rec_restore_from_page_infimum(block, rec, block);
page_cur_move_to_next(page_cursor); page_cur_move_to_next(page_cursor);
...@@ -1924,16 +1924,16 @@ static ...@@ -1924,16 +1924,16 @@ static
void void
btr_cur_pess_upd_restore_supremum( btr_cur_pess_upd_restore_supremum(
/*==============================*/ /*==============================*/
rec_t* rec, /* in: updated record */ buf_block_t* block, /* in: buffer block of rec */
mtr_t* mtr) /* in: mtr */ const rec_t* rec, /* in: updated record */
mtr_t* mtr) /* in: mtr */
{ {
page_t* page; page_t* page;
buf_block_t* prev_block; buf_block_t* prev_block;
page_t* prev_page;
ulint space; ulint space;
ulint prev_page_no; ulint prev_page_no;
page = page_align(rec); page = buf_block_get_frame(block);
if (page_rec_get_next(page_get_infimum_rec(page)) != rec) { if (page_rec_get_next(page_get_infimum_rec(page)) != rec) {
/* Updated record is not the first user record on its page */ /* Updated record is not the first user record on its page */
...@@ -1941,22 +1941,22 @@ btr_cur_pess_upd_restore_supremum( ...@@ -1941,22 +1941,22 @@ btr_cur_pess_upd_restore_supremum(
return; return;
} }
space = page_get_space_id(page); space = buf_block_get_space(block);
prev_page_no = btr_page_get_prev(page, mtr); prev_page_no = btr_page_get_prev(page, mtr);
ut_ad(prev_page_no != FIL_NULL); ut_ad(prev_page_no != FIL_NULL);
prev_block = buf_page_get_with_no_latch(space, prev_page_no, mtr); prev_block = buf_page_get_with_no_latch(space, prev_page_no, mtr);
prev_page = buf_block_get_frame(prev_block);
#ifdef UNIV_BTR_DEBUG #ifdef UNIV_BTR_DEBUG
ut_a(btr_page_get_next(prev_page, mtr) ut_a(btr_page_get_next(prev_block->frame, mtr)
== page_get_page_no(page)); == page_get_page_no(page));
#endif /* UNIV_BTR_DEBUG */ #endif /* UNIV_BTR_DEBUG */
/* We must already have an x-latch to prev_page! */ /* We must already have an x-latch on prev_block! */
ut_ad(mtr_memo_contains(mtr, prev_block, MTR_MEMO_PAGE_X_FIX)); ut_ad(mtr_memo_contains(mtr, prev_block, MTR_MEMO_PAGE_X_FIX));
lock_rec_reset_and_inherit_gap_locks(page_get_supremum_rec(prev_page), lock_rec_reset_and_inherit_gap_locks(prev_block, block,
rec); PAGE_HEAP_NO_SUPREMUM,
page_rec_get_heap_no(rec));
} }
/***************************************************************** /*****************************************************************
...@@ -2117,8 +2117,6 @@ btr_cur_pessimistic_update( ...@@ -2117,8 +2117,6 @@ btr_cur_pessimistic_update(
big_rec_vec, &heap); big_rec_vec, &heap);
} }
page_cursor = btr_cur_get_page_cur(cursor);
/* Store state of explicit locks on rec on the page infimum record, /* Store state of explicit locks on rec on the page infimum record,
before deleting rec. The page infimum acts as a dummy carrier of the before deleting rec. The page infimum acts as a dummy carrier of the
locks, taking care also of lock releases, before we can move the locks locks, taking care also of lock releases, before we can move the locks
...@@ -2128,13 +2126,15 @@ btr_cur_pessimistic_update( ...@@ -2128,13 +2126,15 @@ btr_cur_pessimistic_update(
delete the lock structs set on the root page even if the root delete the lock structs set on the root page even if the root
page carries just node pointers. */ page carries just node pointers. */
lock_rec_store_on_page_infimum(page_align(rec), rec); lock_rec_store_on_page_infimum(block, rec);
btr_search_update_hash_on_delete(cursor); btr_search_update_hash_on_delete(cursor);
#ifdef UNIV_ZIP_DEBUG #ifdef UNIV_ZIP_DEBUG
ut_a(!page_zip || page_zip_validate(page_zip, page)); ut_a(!page_zip || page_zip_validate(page_zip, page));
#endif /* UNIV_ZIP_DEBUG */ #endif /* UNIV_ZIP_DEBUG */
page_cursor = btr_cur_get_page_cur(cursor);
page_cur_delete_rec(page_cursor, index, offsets, mtr); page_cur_delete_rec(page_cursor, index, offsets, mtr);
page_cur_move_to_prev(page_cursor); page_cur_move_to_prev(page_cursor);
...@@ -2144,7 +2144,8 @@ btr_cur_pessimistic_update( ...@@ -2144,7 +2144,8 @@ btr_cur_pessimistic_update(
ut_a(rec || optim_err != DB_UNDERFLOW); ut_a(rec || optim_err != DB_UNDERFLOW);
if (rec) { if (rec) {
lock_rec_restore_from_page_infimum(rec, page); lock_rec_restore_from_page_infimum(btr_cur_get_block(cursor),
rec, block);
offsets = rec_get_offsets(rec, index, offsets, offsets = rec_get_offsets(rec, index, offsets,
ULINT_UNDEFINED, &heap); ULINT_UNDEFINED, &heap);
...@@ -2189,7 +2190,8 @@ btr_cur_pessimistic_update( ...@@ -2189,7 +2190,8 @@ btr_cur_pessimistic_update(
rec, index, offsets, mtr); rec, index, offsets, mtr);
} }
lock_rec_restore_from_page_infimum(rec, page); lock_rec_restore_from_page_infimum(btr_cur_get_block(cursor),
rec, block);
/* If necessary, restore also the correct lock state for a new, /* If necessary, restore also the correct lock state for a new,
preceding supremum record created in a page split. While the old preceding supremum record created in a page split. While the old
...@@ -2197,7 +2199,8 @@ btr_cur_pessimistic_update( ...@@ -2197,7 +2199,8 @@ btr_cur_pessimistic_update(
from a wrong record. */ from a wrong record. */
if (!was_first) { if (!was_first) {
btr_cur_pess_upd_restore_supremum(rec, mtr); btr_cur_pess_upd_restore_supremum(btr_cur_get_block(cursor),
rec, mtr);
} }
return_after_reservations: return_after_reservations:
...@@ -2388,6 +2391,7 @@ btr_cur_del_mark_set_clust_rec( ...@@ -2388,6 +2391,7 @@ btr_cur_del_mark_set_clust_rec(
ut_ad(!rec_get_deleted_flag(rec, rec_offs_comp(offsets))); ut_ad(!rec_get_deleted_flag(rec, rec_offs_comp(offsets)));
err = lock_clust_rec_modify_check_and_lock(flags, err = lock_clust_rec_modify_check_and_lock(flags,
btr_cur_get_block(cursor),
rec, index, offsets, thr); rec, index, offsets, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
...@@ -2539,9 +2543,9 @@ btr_cur_del_mark_set_sec_rec( ...@@ -2539,9 +2543,9 @@ btr_cur_del_mark_set_sec_rec(
} }
#endif /* UNIV_DEBUG */ #endif /* UNIV_DEBUG */
err = lock_sec_rec_modify_check_and_lock(flags, rec, err = lock_sec_rec_modify_check_and_lock(flags,
btr_cur_get_block(cursor), btr_cur_get_block(cursor),
cursor->index, thr); rec, cursor->index, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
return(err); return(err);
...@@ -2659,7 +2663,7 @@ btr_cur_optimistic_delete( ...@@ -2659,7 +2663,7 @@ btr_cur_optimistic_delete(
page_zip_des_t* page_zip= buf_block_get_page_zip(block); page_zip_des_t* page_zip= buf_block_get_page_zip(block);
#endif /* UNIV_ZIP_DEBUG */ #endif /* UNIV_ZIP_DEBUG */
lock_update_delete(rec); lock_update_delete(block, rec);
btr_search_update_hash_on_delete(cursor); btr_search_update_hash_on_delete(cursor);
...@@ -2790,7 +2794,7 @@ btr_cur_pessimistic_delete( ...@@ -2790,7 +2794,7 @@ btr_cur_pessimistic_delete(
goto return_after_reservations; goto return_after_reservations;
} }
lock_update_delete(rec); lock_update_delete(block, rec);
level = btr_page_get_level(page, mtr); level = btr_page_get_level(page, mtr);
if (level > 0 if (level > 0
......
...@@ -15,7 +15,6 @@ Created 5/7/1996 Heikki Tuuri ...@@ -15,7 +15,6 @@ Created 5/7/1996 Heikki Tuuri
#include "rem0types.h" #include "rem0types.h"
#include "dict0types.h" #include "dict0types.h"
#include "que0types.h" #include "que0types.h"
#include "page0types.h"
#include "lock0types.h" #include "lock0types.h"
#include "read0types.h" #include "read0types.h"
#include "hash0hash.h" #include "hash0hash.h"
...@@ -49,7 +48,7 @@ lock_sec_rec_some_has_impl_off_kernel( ...@@ -49,7 +48,7 @@ lock_sec_rec_some_has_impl_off_kernel(
/*==================================*/ /*==================================*/
/* out: transaction which has the x-lock, or /* out: transaction which has the x-lock, or
NULL */ NULL */
rec_t* rec, /* in: user record */ const rec_t* rec, /* in: user record */
dict_index_t* index, /* in: secondary index */ dict_index_t* index, /* in: secondary index */
const ulint* offsets);/* in: rec_get_offsets(rec, index) */ const ulint* offsets);/* in: rec_get_offsets(rec, index) */
/************************************************************************* /*************************************************************************
...@@ -61,20 +60,19 @@ lock_clust_rec_some_has_impl( ...@@ -61,20 +60,19 @@ lock_clust_rec_some_has_impl(
/*=========================*/ /*=========================*/
/* out: transaction which has the x-lock, or /* out: transaction which has the x-lock, or
NULL */ NULL */
rec_t* rec, /* in: user record */ const rec_t* rec, /* in: user record */
dict_index_t* index, /* in: clustered index */ dict_index_t* index, /* in: clustered index */
const ulint* offsets);/* in: rec_get_offsets(rec, index) */ const ulint* offsets);/* in: rec_get_offsets(rec, index) */
/***************************************************************** /*************************************************************************
Makes a record to inherit the locks of another record as gap type Gets the heap_no of the smallest user record on a page. */
locks, but does not reset the lock bits of the other record. Also UNIV_INLINE
waiting lock requests on rec are inherited as GRANTED gap locks. */ ulint
lock_get_min_heap_no(
void /*=================*/
lock_rec_inherit_to_gap( /* out: heap_no of smallest
/*====================*/ user record, or
rec_t* heir, /* in: record which inherits */ PAGE_HEAP_NO_SUPREMUM */
rec_t* rec); /* in: record from which inherited; does NOT reset const buf_block_t* block); /* in: buffer block */
the locks on this record */
/***************************************************************** /*****************************************************************
Updates the lock table when we have reorganized a page. NOTE: we copy Updates the lock table when we have reorganized a page. NOTE: we copy
also the locks set on the infimum of the page; the infimum may carry also the locks set on the infimum of the page; the infimum may carry
...@@ -84,8 +82,10 @@ were temporarily stored on the infimum. */ ...@@ -84,8 +82,10 @@ were temporarily stored on the infimum. */
void void
lock_move_reorganize_page( lock_move_reorganize_page(
/*======================*/ /*======================*/
page_t* page, /* in: old index page */ const buf_block_t* block, /* in: old index page, now
page_t* new_page); /* in: reorganized page */ reorganized */
const buf_block_t* oblock);/* in: copy of the old, not
reorganized page */
/***************************************************************** /*****************************************************************
Moves the explicit locks on user records to another page if a record Moves the explicit locks on user records to another page if a record
list end is moved to another page. */ list end is moved to another page. */
...@@ -93,10 +93,10 @@ list end is moved to another page. */ ...@@ -93,10 +93,10 @@ list end is moved to another page. */
void void
lock_move_rec_list_end( lock_move_rec_list_end(
/*===================*/ /*===================*/
page_t* new_page, /* in: index page to move to */ const buf_block_t* new_block, /* in: index page to move to */
page_t* page, /* in: index page */ const buf_block_t* block, /* in: index page */
rec_t* rec); /* in: record on page: this is the const rec_t* rec); /* in: record on page: this
first record moved */ is the first record moved */
/***************************************************************** /*****************************************************************
Moves the explicit locks on user records to another page if a record Moves the explicit locks on user records to another page if a record
list start is moved to another page. */ list start is moved to another page. */
...@@ -104,30 +104,39 @@ list start is moved to another page. */ ...@@ -104,30 +104,39 @@ list start is moved to another page. */
void void
lock_move_rec_list_start( lock_move_rec_list_start(
/*=====================*/ /*=====================*/
page_t* new_page, /* in: index page to move to */ const buf_block_t* new_block, /* in: index page to move to */
page_t* page, /* in: index page */ const buf_block_t* block, /* in: index page */
rec_t* rec, /* in: record on page: this is the const rec_t* rec, /* in: record on page:
first record NOT copied */ this is the first
rec_t* old_end); /* in: old previous-to-last record on record NOT copied */
new_page before the records were copied */ const rec_t* old_end); /* in: old
previous-to-last
record on new_page
before the records
were copied */
/***************************************************************** /*****************************************************************
Updates the lock table when a page is split to the right. */ Updates the lock table when a page is split to the right. */
void void
lock_update_split_right( lock_update_split_right(
/*====================*/ /*====================*/
page_t* right_page, /* in: right page */ const buf_block_t* right_block, /* in: right page */
page_t* left_page); /* in: left page */ const buf_block_t* left_block); /* in: left page */
/***************************************************************** /*****************************************************************
Updates the lock table when a page is merged to the right. */ Updates the lock table when a page is merged to the right. */
void void
lock_update_merge_right( lock_update_merge_right(
/*====================*/ /*====================*/
rec_t* orig_succ, /* in: original successor of infimum const buf_block_t* right_block, /* in: right page to
on the right page before merge */ which merged */
page_t* left_page); /* in: merged index page which will be const rec_t* orig_succ, /* in: original
discarded */ successor of infimum
on the right page
before merge */
const buf_block_t* left_block); /* in: merged index
page which will be
discarded */
/***************************************************************** /*****************************************************************
Updates the lock table when the root page is copied to another in Updates the lock table when the root page is copied to another in
btr_root_raise_and_insert. Note that we leave lock structs on the btr_root_raise_and_insert. Note that we leave lock structs on the
...@@ -139,8 +148,8 @@ to be updated. */ ...@@ -139,8 +148,8 @@ to be updated. */
void void
lock_update_root_raise( lock_update_root_raise(
/*===================*/ /*===================*/
page_t* new_page, /* in: index page to which copied */ const buf_block_t* block, /* in: index page to which copied */
page_t* root); /* in: root page */ const buf_block_t* root); /* in: root page */
/***************************************************************** /*****************************************************************
Updates the lock table when a page is copied to another and the original page Updates the lock table when a page is copied to another and the original page
is removed from the chain of leaf pages, except if page is the root! */ is removed from the chain of leaf pages, except if page is the root! */
...@@ -148,27 +157,31 @@ is removed from the chain of leaf pages, except if page is the root! */ ...@@ -148,27 +157,31 @@ is removed from the chain of leaf pages, except if page is the root! */
void void
lock_update_copy_and_discard( lock_update_copy_and_discard(
/*=========================*/ /*=========================*/
page_t* new_page, /* in: index page to which copied */ const buf_block_t* new_block, /* in: index page to
page_t* page); /* in: index page; NOT the root! */ which copied */
const buf_block_t* block); /* in: index page;
NOT the root! */
/***************************************************************** /*****************************************************************
Updates the lock table when a page is split to the left. */ Updates the lock table when a page is split to the left. */
void void
lock_update_split_left( lock_update_split_left(
/*===================*/ /*===================*/
page_t* right_page, /* in: right page */ const buf_block_t* right_block, /* in: right page */
page_t* left_page); /* in: left page */ const buf_block_t* left_block); /* in: left page */
/***************************************************************** /*****************************************************************
Updates the lock table when a page is merged to the left. */ Updates the lock table when a page is merged to the left. */
void void
lock_update_merge_left( lock_update_merge_left(
/*===================*/ /*===================*/
page_t* left_page, /* in: left page to which merged */ const buf_block_t* left_block, /* in: left page to
rec_t* orig_pred, /* in: original predecessor of supremum which merged */
on the left page before merge */ const rec_t* orig_pred, /* in: original predecessor
page_t* right_page); /* in: merged index page which will be of supremum on the left page
discarded */ before merge */
const buf_block_t* right_block); /* in: merged index page
which will be discarded */
/***************************************************************** /*****************************************************************
Resets the original locks on heir and replaces them with gap type locks Resets the original locks on heir and replaces them with gap type locks
inherited from rec. */ inherited from rec. */
...@@ -176,30 +189,44 @@ inherited from rec. */ ...@@ -176,30 +189,44 @@ inherited from rec. */
void void
lock_rec_reset_and_inherit_gap_locks( lock_rec_reset_and_inherit_gap_locks(
/*=================================*/ /*=================================*/
rec_t* heir, /* in: heir record */ const buf_block_t* heir_block, /* in: block containing the
rec_t* rec); /* in: record */ record which inherits */
const buf_block_t* block, /* in: block containing the
record from which inherited;
does NOT reset the locks on
this record */
ulint heir_heap_no, /* in: heap_no of the
inheriting record */
ulint heap_no); /* in: heap_no of the
donating record */
/***************************************************************** /*****************************************************************
Updates the lock table when a page is discarded. */ Updates the lock table when a page is discarded. */
void void
lock_update_discard( lock_update_discard(
/*================*/ /*================*/
rec_t* heir, /* in: record which will inherit the locks */ const buf_block_t* heir_block, /* in: index page
page_t* page); /* in: index page which will be discarded */ which will inherit the locks */
ulint heir_heap_no, /* in: heap_no of the record
which will inherit the locks */
const buf_block_t* block); /* in: index page
which will be discarded */
/***************************************************************** /*****************************************************************
Updates the lock table when a new user record is inserted. */ Updates the lock table when a new user record is inserted. */
void void
lock_update_insert( lock_update_insert(
/*===============*/ /*===============*/
rec_t* rec); /* in: the inserted record */ const buf_block_t* block, /* in: buffer block containing rec */
const rec_t* rec); /* in: the inserted record */
/***************************************************************** /*****************************************************************
Updates the lock table when a record is removed. */ Updates the lock table when a record is removed. */
void void
lock_update_delete( lock_update_delete(
/*===============*/ /*===============*/
rec_t* rec); /* in: the record to be removed */ const buf_block_t* block, /* in: buffer block containing rec */
const rec_t* rec); /* in: the record to be removed */
/************************************************************************* /*************************************************************************
Stores on the page infimum record the explicit locks of another record. Stores on the page infimum record the explicit locks of another record.
This function is used to store the lock state of a record when it is This function is used to store the lock state of a record when it is
...@@ -211,10 +238,12 @@ actual record is being moved. */ ...@@ -211,10 +238,12 @@ actual record is being moved. */
void void
lock_rec_store_on_page_infimum( lock_rec_store_on_page_infimum(
/*===========================*/ /*===========================*/
page_t* page, /* in: page containing the record */ const buf_block_t* block, /* in: buffer block containing rec */
rec_t* rec); /* in: record whose lock state is stored const rec_t* rec); /* in: record whose lock state
on the infimum record of the same page; lock is stored on the infimum
bits are reset on the record */ record of the same page; lock
bits are reset on the
record */
/************************************************************************* /*************************************************************************
Restores the state of explicit lock requests on a single record, where the Restores the state of explicit lock requests on a single record, where the
state was stored on the infimum of the page. */ state was stored on the infimum of the page. */
...@@ -222,10 +251,17 @@ state was stored on the infimum of the page. */ ...@@ -222,10 +251,17 @@ state was stored on the infimum of the page. */
void void
lock_rec_restore_from_page_infimum( lock_rec_restore_from_page_infimum(
/*===============================*/ /*===============================*/
rec_t* rec, /* in: record whose lock state is restored */ const buf_block_t* block, /* in: buffer block containing rec */
page_t* page); /* in: page (rec is not necessarily on this page) const rec_t* rec, /* in: record whose lock state
whose infimum stored the lock state; lock bits are is stored on the infimum
reset on the infimum */ record of the same page; lock
bits are reset on the
record */
const buf_block_t* donator);/* in: page (rec is not
necessarily on this page)
whose infimum stored the lock
state; lock bits are reset on
the infimum */
/************************************************************************* /*************************************************************************
Returns TRUE if there are explicit record locks on a page. */ Returns TRUE if there are explicit record locks on a page. */
...@@ -248,15 +284,16 @@ lock_rec_insert_check_and_lock( ...@@ -248,15 +284,16 @@ lock_rec_insert_check_and_lock(
/*===========================*/ /*===========================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS, DB_LOCK_WAIT,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is
does nothing */ set, does nothing */
rec_t* rec, /* in: record after which to insert */ rec_t* rec, /* in: record after which to insert */
buf_block_t* block, /* in: buffer block of rec */ buf_block_t* block, /* in/out: buffer block of rec */
dict_index_t* index, /* in: index */ dict_index_t* index, /* in: index */
que_thr_t* thr, /* in: query thread */ que_thr_t* thr, /* in: query thread */
ibool* inherit);/* out: set to TRUE if the new inserted ibool* inherit);/* out: set to TRUE if the new
record maybe should inherit LOCK_GAP type inserted record maybe should inherit
locks from the successor record */ LOCK_GAP type locks from the successor
record */
/************************************************************************* /*************************************************************************
Checks if locks of other transactions prevent an immediate modify (update, Checks if locks of other transactions prevent an immediate modify (update,
delete mark, or delete unmark) of a clustered index record. If they do, delete mark, or delete unmark) of a clustered index record. If they do,
...@@ -268,14 +305,17 @@ lock queue. */ ...@@ -268,14 +305,17 @@ lock queue. */
ulint ulint
lock_clust_rec_modify_check_and_lock( lock_clust_rec_modify_check_and_lock(
/*=================================*/ /*=================================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: record which should be modified */ bit is set, does nothing */
dict_index_t* index, /* in: clustered index */ const buf_block_t* block, /* in: buffer block of rec */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ const rec_t* rec, /* in: record which should be
que_thr_t* thr); /* in: query thread */ modified */
dict_index_t* index, /* in: clustered index */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */
que_thr_t* thr); /* in: query thread */
/************************************************************************* /*************************************************************************
Checks if locks of other transactions prevent an immediate modify Checks if locks of other transactions prevent an immediate modify
(delete mark or delete unmark) of a secondary index record. */ (delete mark or delete unmark) of a secondary index record. */
...@@ -285,13 +325,14 @@ lock_sec_rec_modify_check_and_lock( ...@@ -285,13 +325,14 @@ lock_sec_rec_modify_check_and_lock(
/*===============================*/ /*===============================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS, DB_LOCK_WAIT,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, ulint flags, /* in: if BTR_NO_LOCKING_FLAG
does nothing */ bit is set, does nothing */
rec_t* rec, /* in: record which should be modified; buf_block_t* block, /* in/out: buffer block of rec */
NOTE: as this is a secondary index, we rec_t* rec, /* in: record which should be
always have to modify the clustered index modified; NOTE: as this is a secondary
record first: see the comment below */ index, we always have to modify the
buf_block_t* block, /* in: buffer block of rec */ clustered index record first: see the
comment below */
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 */
/************************************************************************* /*************************************************************************
...@@ -301,21 +342,26 @@ secondary index record. */ ...@@ -301,21 +342,26 @@ secondary index record. */
ulint ulint
lock_sec_rec_read_check_and_lock( lock_sec_rec_read_check_and_lock(
/*=============================*/ /*=============================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: user record or page supremum record bit is set, does nothing */
which should be read or passed over by a read const buf_block_t* block, /* in: buffer block of rec */
cursor */ const rec_t* rec, /* in: user recpord or page
dict_index_t* index, /* in: secondary index */ supremum record which should
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ be read or passed over by a
ulint mode, /* in: mode of the lock which the read cursor read cursor */
should set on records: LOCK_S or LOCK_X; the dict_index_t* index, /* in: secondary index */
latter is possible in SELECT FOR UPDATE */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: mode of the lock which
LOCK_REC_NOT_GAP */ the read cursor should set on
que_thr_t* thr); /* in: query thread */ records: LOCK_S or LOCK_X; the
latter is possible in
SELECT FOR UPDATE */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP */
que_thr_t* thr); /* in: query thread */
/************************************************************************* /*************************************************************************
Checks if locks of other transactions prevent an immediate read, or passing Checks if locks of other transactions prevent an immediate read, or passing
over by a read cursor, of a clustered index record. If they do, first tests over by a read cursor, of a clustered index record. If they do, first tests
...@@ -327,21 +373,26 @@ lock on the record. */ ...@@ -327,21 +373,26 @@ lock on the record. */
ulint ulint
lock_clust_rec_read_check_and_lock( lock_clust_rec_read_check_and_lock(
/*===============================*/ /*===============================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: user record or page supremum record bit is set, does nothing */
which should be read or passed over by a read const buf_block_t* block, /* in: buffer block of rec */
cursor */ const rec_t* rec, /* in: user record or page
dict_index_t* index, /* in: clustered index */ supremum record which should
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ be read or passed over by a
ulint mode, /* in: mode of the lock which the read cursor read cursor */
should set on records: LOCK_S or LOCK_X; the dict_index_t* index, /* in: clustered index */
latter is possible in SELECT FOR UPDATE */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: mode of the lock which
LOCK_REC_NOT_GAP */ the read cursor should set on
que_thr_t* thr); /* in: query thread */ records: LOCK_S or LOCK_X; the
latter is possible in
SELECT FOR UPDATE */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP */
que_thr_t* thr); /* in: query thread */
/************************************************************************* /*************************************************************************
Checks if locks of other transactions prevent an immediate read, or passing Checks if locks of other transactions prevent an immediate read, or passing
over by a read cursor, of a clustered index record. If they do, first tests over by a read cursor, of a clustered index record. If they do, first tests
...@@ -355,20 +406,25 @@ lock_clust_rec_read_check_and_lock() that does not require the parameter ...@@ -355,20 +406,25 @@ lock_clust_rec_read_check_and_lock() that does not require the parameter
ulint ulint
lock_clust_rec_read_check_and_lock_alt( lock_clust_rec_read_check_and_lock_alt(
/*===================================*/ /*===================================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: user record or page supremum record bit is set, does nothing */
which should be read or passed over by a read const buf_block_t* block, /* in: buffer block of rec */
cursor */ const rec_t* rec, /* in: user record or page
dict_index_t* index, /* in: clustered index */ supremum record which should
ulint mode, /* in: mode of the lock which the read cursor be read or passed over by a
should set on records: LOCK_S or LOCK_X; the read cursor */
latter is possible in SELECT FOR UPDATE */ dict_index_t* index, /* in: clustered index */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: mode of the lock which
LOCK_REC_NOT_GAP */ the read cursor should set on
que_thr_t* thr); /* in: query thread */ records: LOCK_S or LOCK_X; the
latter is possible in
SELECT FOR UPDATE */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP */
que_thr_t* thr); /* in: query thread */
/************************************************************************* /*************************************************************************
Checks that a record is seen in a consistent read. */ Checks that a record is seen in a consistent read. */
...@@ -377,7 +433,7 @@ lock_clust_rec_cons_read_sees( ...@@ -377,7 +433,7 @@ lock_clust_rec_cons_read_sees(
/*==========================*/ /*==========================*/
/* out: TRUE if sees, or FALSE if an earlier /* out: TRUE if sees, or FALSE if an earlier
version of the record should be retrieved */ version of the record should be retrieved */
rec_t* rec, /* in: user record which should be read or const rec_t* rec, /* in: user record which should be read or
passed over by a read cursor */ passed over by a read cursor */
dict_index_t* index, /* in: clustered index */ dict_index_t* index, /* in: clustered index */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
...@@ -388,18 +444,23 @@ Checks that a non-clustered index record is seen in a consistent read. */ ...@@ -388,18 +444,23 @@ Checks that a non-clustered index record is seen in a consistent read. */
ulint ulint
lock_sec_rec_cons_read_sees( lock_sec_rec_cons_read_sees(
/*========================*/ /*========================*/
/* out: TRUE if certainly sees, or FALSE if an /* out: TRUE if certainly
earlier version of the clustered index record sees, or FALSE if an earlier
might be needed: NOTE that a non-clustered version of the clustered index
index page contains so little information on record might be needed: NOTE
its modifications that also in the case FALSE, that a non-clustered index
the present version of rec may be the right, page contains so little
but we must check this from the clustered information on its
index record */ modifications that also in the
rec_t* rec, /* in: user record which should be read or case FALSE, the present
passed over by a read cursor */ version of rec may be the
dict_index_t* index, /* in: non-clustered index */ right, but we must check this
read_view_t* view); /* in: consistent read view */ from the clustered index
record */
const rec_t* rec, /* in: user record which
should be read or passed over
by a read cursor */
const read_view_t* view); /* in: consistent read view */
/************************************************************************* /*************************************************************************
Locks the specified database table in the mode given. If the lock cannot Locks the specified database table in the mode given. If the lock cannot
be granted immediately, the query thread is put to wait. */ be granted immediately, the query thread is put to wait. */
...@@ -430,10 +491,11 @@ to a lock. */ ...@@ -430,10 +491,11 @@ to a lock. */
void void
lock_rec_unlock( lock_rec_unlock(
/*============*/ /*============*/
trx_t* trx, /* in: transaction that has set a record trx_t* trx, /* in: transaction that has
lock */ set a record lock */
rec_t* rec, /* in: record */ const buf_block_t* block, /* in: buffer block containing rec */
ulint lock_mode); /* in: LOCK_S or LOCK_X */ const rec_t* rec, /* in: record */
ulint lock_mode);/* in: LOCK_S or LOCK_X */
/************************************************************************* /*************************************************************************
Releases a table lock. Releases a table lock.
Releases possible other transactions waiting for this lock. */ Releases possible other transactions waiting for this lock. */
...@@ -529,22 +591,12 @@ lock_check_trx_id_sanity( ...@@ -529,22 +591,12 @@ lock_check_trx_id_sanity(
/*=====================*/ /*=====================*/
/* out: TRUE if ok */ /* out: TRUE if ok */
dulint trx_id, /* in: trx id */ dulint trx_id, /* in: trx id */
rec_t* rec, /* in: user record */ const rec_t* rec, /* in: user record */
dict_index_t* index, /* in: clustered index */ dict_index_t* index, /* in: clustered index */
const ulint* offsets, /* in: rec_get_offsets(rec, index) */ const ulint* offsets, /* in: rec_get_offsets(rec, index) */
ibool has_kernel_mutex);/* in: TRUE if the caller owns the ibool has_kernel_mutex);/* in: TRUE if the caller owns the
kernel mutex */ kernel mutex */
/************************************************************************* /*************************************************************************
Validates the lock queue on a single record. */
ibool
lock_rec_queue_validate(
/*====================*/
/* out: TRUE if ok */
rec_t* rec, /* in: record to look at */
dict_index_t* index, /* in: index, or NULL if not known */
const ulint* offsets);/* in: rec_get_offsets(rec, index) */
/*************************************************************************
Prints info of a table lock. */ Prints info of a table lock. */
void void
...@@ -575,30 +627,6 @@ lock_print_info_all_transactions( ...@@ -575,30 +627,6 @@ lock_print_info_all_transactions(
/*=============================*/ /*=============================*/
FILE* file); /* in: file where to print */ FILE* file); /* in: file where to print */
/************************************************************************* /*************************************************************************
Validates the lock queue on a table. */
ibool
lock_table_queue_validate(
/*======================*/
/* out: TRUE if ok */
dict_table_t* table); /* in: table */
/*************************************************************************
Validates the record lock queues on a page. */
ibool
lock_rec_validate_page(
/*===================*/
/* out: TRUE if ok */
ulint space, /* in: space id */
ulint page_no);/* in: page number */
/*************************************************************************
Validates the lock system. */
ibool
lock_validate(void);
/*===============*/
/* out: TRUE if ok */
/*************************************************************************
Return approximate number or record locks (bits set in the bitmap) for Return approximate number or record locks (bits set in the bitmap) for
this transaction. Since delete-marked records ma ybe removed, the this transaction. Since delete-marked records ma ybe removed, the
record count will not be precise. */ record count will not be precise. */
......
...@@ -59,7 +59,7 @@ lock_clust_rec_some_has_impl( ...@@ -59,7 +59,7 @@ lock_clust_rec_some_has_impl(
/*=========================*/ /*=========================*/
/* out: transaction which has the x-lock, or /* out: transaction which has the x-lock, or
NULL */ NULL */
rec_t* rec, /* in: user record */ const rec_t* rec, /* in: user record */
dict_index_t* index, /* in: clustered index */ dict_index_t* index, /* in: clustered index */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */ const ulint* offsets)/* in: rec_get_offsets(rec, index) */
{ {
...@@ -81,3 +81,29 @@ lock_clust_rec_some_has_impl( ...@@ -81,3 +81,29 @@ lock_clust_rec_some_has_impl(
return(NULL); return(NULL);
} }
/*************************************************************************
Gets the heap_no of the smallest user record on a page. */
UNIV_INLINE
ulint
lock_get_min_heap_no(
/*=================*/
/* out: heap_no of smallest
user record, or
PAGE_HEAP_NO_SUPREMUM */
const buf_block_t* block) /* in: buffer block */
{
const page_t* page = block->frame;
if (page_is_comp(page)) {
return(rec_get_heap_no_new(
page
+ rec_get_next_offs(page + PAGE_NEW_INFIMUM,
TRUE)));
} else {
return(rec_get_heap_no_old(
page
+ rec_get_next_offs(page + PAGE_OLD_INFIMUM,
FALSE)));
}
}
...@@ -444,6 +444,14 @@ page_rec_is_comp( ...@@ -444,6 +444,14 @@ page_rec_is_comp(
/*=============*/ /*=============*/
/* out: nonzero if in compact format */ /* out: nonzero if in compact format */
const rec_t* rec); /* in: record */ const rec_t* rec); /* in: record */
/*******************************************************************
Returns the heap number of a record. */
UNIV_INLINE
ulint
page_rec_get_heap_no(
/*=================*/
/* out: heap number */
const rec_t* rec); /* in: the physical record */
/**************************************************************** /****************************************************************
Determine whether the page is a B-tree leaf. */ Determine whether the page is a B-tree leaf. */
UNIV_INLINE UNIV_INLINE
......
...@@ -221,6 +221,22 @@ page_rec_is_comp( ...@@ -221,6 +221,22 @@ page_rec_is_comp(
return(page_is_comp(page_align((rec_t*) rec))); return(page_is_comp(page_align((rec_t*) rec)));
} }
/*******************************************************************
Returns the heap number of a record. */
UNIV_INLINE
ulint
page_rec_get_heap_no(
/*=================*/
/* out: heap number */
const rec_t* rec) /* in: the physical record */
{
if (page_rec_is_comp(rec)) {
return(rec_get_heap_no_new(rec));
} else {
return(rec_get_heap_no_old(rec));
}
}
/**************************************************************** /****************************************************************
Determine whether the page is a B-tree leaf. */ Determine whether the page is a B-tree leaf. */
UNIV_INLINE UNIV_INLINE
......
...@@ -314,6 +314,14 @@ locks on the inserted record. */ ...@@ -314,6 +314,14 @@ locks on the inserted record. */
#ifdef UNIV_DEBUG #ifdef UNIV_DEBUG
ibool lock_print_waits = FALSE; ibool lock_print_waits = FALSE;
/*************************************************************************
Validates the lock system. */
static
ibool
lock_validate(void);
/*===============*/
/* out: TRUE if ok */
#endif /* UNIV_DEBUG */ #endif /* UNIV_DEBUG */
/* The lock system */ /* The lock system */
...@@ -457,7 +465,7 @@ lock_check_trx_id_sanity( ...@@ -457,7 +465,7 @@ lock_check_trx_id_sanity(
/*=====================*/ /*=====================*/
/* out: TRUE if ok */ /* out: TRUE if ok */
dulint trx_id, /* in: trx id */ dulint trx_id, /* in: trx id */
rec_t* rec, /* in: user record */ const rec_t* rec, /* in: user record */
dict_index_t* index, /* in: index */ dict_index_t* index, /* in: index */
const ulint* offsets, /* in: rec_get_offsets(rec, index) */ const ulint* offsets, /* in: rec_get_offsets(rec, index) */
ibool has_kernel_mutex)/* in: TRUE if the caller owns the ibool has_kernel_mutex)/* in: TRUE if the caller owns the
...@@ -510,7 +518,7 @@ lock_clust_rec_cons_read_sees( ...@@ -510,7 +518,7 @@ lock_clust_rec_cons_read_sees(
/*==========================*/ /*==========================*/
/* out: TRUE if sees, or FALSE if an earlier /* out: TRUE if sees, or FALSE if an earlier
version of the record should be retrieved */ version of the record should be retrieved */
rec_t* rec, /* in: user record which should be read or const rec_t* rec, /* in: user record which should be read or
passed over by a read cursor */ passed over by a read cursor */
dict_index_t* index, /* in: clustered index */ dict_index_t* index, /* in: clustered index */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
...@@ -537,24 +545,26 @@ Checks that a non-clustered index record is seen in a consistent read. */ ...@@ -537,24 +545,26 @@ Checks that a non-clustered index record is seen in a consistent read. */
ulint ulint
lock_sec_rec_cons_read_sees( lock_sec_rec_cons_read_sees(
/*========================*/ /*========================*/
/* out: TRUE if certainly sees, or FALSE if an /* out: TRUE if certainly
earlier version of the clustered index record sees, or FALSE if an earlier
might be needed: NOTE that a non-clustered version of the clustered index
index page contains so little information on record might be needed: NOTE
its modifications that also in the case FALSE, that a non-clustered index
the present version of rec may be the right, page contains so little
but we must check this from the clustered information on its
index record */ modifications that also in the
rec_t* rec, /* in: user record which should be read or case FALSE, the present
passed over by a read cursor */ version of rec may be the
dict_index_t* index, /* in: non-clustered index */ right, but we must check this
read_view_t* view) /* in: consistent read view */ from the clustered index
record */
const rec_t* rec, /* in: user record which
should be read or passed over
by a read cursor */
const read_view_t* view) /* in: consistent read view */
{ {
dulint max_trx_id; dulint max_trx_id;
UT_NOT_USED(index);
ut_ad(!dict_index_is_clust(index));
ut_ad(page_rec_is_user_rec(rec)); ut_ad(page_rec_is_user_rec(rec));
/* NOTE that we might call this function while holding the search /* NOTE that we might call this function while holding the search
...@@ -566,14 +576,9 @@ lock_sec_rec_cons_read_sees( ...@@ -566,14 +576,9 @@ lock_sec_rec_cons_read_sees(
return(FALSE); return(FALSE);
} }
max_trx_id = page_get_max_trx_id(page_align(rec)); max_trx_id = page_get_max_trx_id(page_align((rec_t*) rec));
if (ut_dulint_cmp(max_trx_id, view->up_limit_id) >= 0) {
return(FALSE);
}
return(TRUE); return(ut_dulint_cmp(max_trx_id, view->up_limit_id) < 0);
} }
/************************************************************************* /*************************************************************************
...@@ -1253,19 +1258,20 @@ UNIV_INLINE ...@@ -1253,19 +1258,20 @@ UNIV_INLINE
lock_t* lock_t*
lock_rec_get_first_on_page( lock_rec_get_first_on_page(
/*=======================*/ /*=======================*/
/* out: first lock, NULL if none exists */ /* out: first lock, NULL if
byte* ptr) /* in: pointer to somewhere on the page */ none exists */
const buf_block_t* block) /* in: buffer block */
{ {
ulint hash; ulint hash;
lock_t* lock; lock_t* lock;
ulint space = page_get_space_id(page_align(ptr)); ulint space = buf_block_get_space(block);
ulint page_no = page_get_page_no(page_align(ptr)); ulint page_no = buf_block_get_page_no(block);
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
hash = buf_block_get_lock_hash_val(buf_block_align(ptr)); hash = buf_block_get_lock_hash_val(block);
lock = HASH_GET_FIRST(lock_sys->rec_hash, hash); lock = HASH_GET_FIRST(lock_sys->rec_hash, hash);
...@@ -1310,9 +1316,10 @@ UNIV_INLINE ...@@ -1310,9 +1316,10 @@ UNIV_INLINE
lock_t* lock_t*
lock_rec_get_first( lock_rec_get_first(
/*===============*/ /*===============*/
/* out: first lock, NULL if none exists */ /* out: first lock, NULL if
rec_t* rec, /* in: record on a page */ none exists */
ulint heap_no)/* in: heap number of the record */ const buf_block_t* block, /* in: block containing the record */
ulint heap_no)/* in: heap number of the record */
{ {
lock_t* lock; lock_t* lock;
...@@ -1320,7 +1327,7 @@ lock_rec_get_first( ...@@ -1320,7 +1327,7 @@ lock_rec_get_first(
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
lock = lock_rec_get_first_on_page(rec); lock = lock_rec_get_first_on_page(block);
if (UNIV_LIKELY_NULL(lock)) { if (UNIV_LIKELY_NULL(lock)) {
while (lock && !lock_rec_get_nth_bit(lock, heap_no)) { while (lock && !lock_rec_get_nth_bit(lock, heap_no)) {
lock = lock_rec_get_next_on_page(lock); lock = lock_rec_get_next_on_page(lock);
...@@ -1480,14 +1487,16 @@ UNIV_INLINE ...@@ -1480,14 +1487,16 @@ UNIV_INLINE
lock_t* lock_t*
lock_rec_has_expl( lock_rec_has_expl(
/*==============*/ /*==============*/
/* out: lock or NULL */ /* out: lock or NULL */
ulint precise_mode,/* in: LOCK_S or LOCK_X possibly ORed to ulint precise_mode,/* in: LOCK_S or LOCK_X
LOCK_GAP or LOCK_REC_NOT_GAP, possibly ORed to LOCK_GAP or
for a supremum record we regard this always a gap LOCK_REC_NOT_GAP, for a
type request */ supremum record we regard this
rec_t* rec, /* in: record */ always a gap type request */
ulint heap_no,/* in: heap number of the record */ const buf_block_t* block, /* in: buffer block containing
trx_t* trx) /* in: transaction */ the record */
ulint heap_no,/* in: heap number of the record */
trx_t* trx) /* in: transaction */
{ {
lock_t* lock; lock_t* lock;
...@@ -1498,7 +1507,7 @@ lock_rec_has_expl( ...@@ -1498,7 +1507,7 @@ lock_rec_has_expl(
|| (precise_mode & LOCK_MODE_MASK) == LOCK_X); || (precise_mode & LOCK_MODE_MASK) == LOCK_X);
ut_ad(!(precise_mode & LOCK_INSERT_INTENTION)); ut_ad(!(precise_mode & LOCK_INSERT_INTENTION));
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
while (lock) { while (lock) {
if (lock->trx == trx if (lock->trx == trx
...@@ -1507,10 +1516,10 @@ lock_rec_has_expl( ...@@ -1507,10 +1516,10 @@ lock_rec_has_expl(
&& !lock_get_wait(lock) && !lock_get_wait(lock)
&& (!lock_rec_get_rec_not_gap(lock) && (!lock_rec_get_rec_not_gap(lock)
|| (precise_mode & LOCK_REC_NOT_GAP) || (precise_mode & LOCK_REC_NOT_GAP)
|| page_rec_is_supremum(rec)) || heap_no == PAGE_HEAP_NO_SUPREMUM)
&& (!lock_rec_get_gap(lock) && (!lock_rec_get_gap(lock)
|| (precise_mode & LOCK_GAP) || (precise_mode & LOCK_GAP)
|| page_rec_is_supremum(rec)) || heap_no == PAGE_HEAP_NO_SUPREMUM)
&& (!lock_rec_get_insert_intention(lock))) { && (!lock_rec_get_insert_intention(lock))) {
return(lock); return(lock);
...@@ -1522,23 +1531,27 @@ lock_rec_has_expl( ...@@ -1522,23 +1531,27 @@ lock_rec_has_expl(
return(NULL); return(NULL);
} }
#ifndef UNIV_HOTBACKUP #ifdef UNIV_DEBUG
# ifndef UNIV_HOTBACKUP
/************************************************************************* /*************************************************************************
Checks if some other transaction has a lock request in the queue. */ Checks if some other transaction has a lock request in the queue. */
static static
lock_t* lock_t*
lock_rec_other_has_expl_req( lock_rec_other_has_expl_req(
/*========================*/ /*========================*/
/* out: lock or NULL */ /* out: lock or NULL */
ulint mode, /* in: LOCK_S or LOCK_X */ ulint mode, /* in: LOCK_S or LOCK_X */
ulint gap, /* in: LOCK_GAP if also gap locks are taken ulint gap, /* in: LOCK_GAP if also gap
into account, or 0 if not */ locks are taken into account,
ulint wait, /* in: LOCK_WAIT if also waiting locks are or 0 if not */
taken into account, or 0 if not */ ulint wait, /* in: LOCK_WAIT if also
rec_t* rec, /* in: record to look at */ waiting locks are taken into
ulint heap_no,/* in: heap number of hte record */ account, or 0 if not */
trx_t* trx) /* in: transaction, or NULL if requests by all const buf_block_t* block, /* in: buffer block containing the record */
transactions are taken into account */ ulint heap_no,/* in: heap number of the record */
trx_t* trx) /* in: transaction, or NULL if
requests by all transactions
are taken into account */
{ {
lock_t* lock; lock_t* lock;
...@@ -1549,13 +1562,13 @@ lock_rec_other_has_expl_req( ...@@ -1549,13 +1562,13 @@ lock_rec_other_has_expl_req(
ut_ad(gap == 0 || gap == LOCK_GAP); ut_ad(gap == 0 || gap == LOCK_GAP);
ut_ad(wait == 0 || wait == LOCK_WAIT); ut_ad(wait == 0 || wait == LOCK_WAIT);
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
while (lock) { while (lock) {
if (lock->trx != trx if (lock->trx != trx
&& (gap && (gap
|| !(lock_rec_get_gap(lock) || !(lock_rec_get_gap(lock)
|| page_rec_is_supremum(rec))) || heap_no == PAGE_HEAP_NO_SUPREMUM))
&& (wait || !lock_get_wait(lock)) && (wait || !lock_get_wait(lock))
&& lock_mode_stronger_or_eq(lock_get_mode(lock), mode)) { && lock_mode_stronger_or_eq(lock_get_mode(lock), mode)) {
...@@ -1567,7 +1580,8 @@ lock_rec_other_has_expl_req( ...@@ -1567,7 +1580,8 @@ lock_rec_other_has_expl_req(
return(NULL); return(NULL);
} }
#endif /* !UNIV_HOTBACKUP */ # endif /* !UNIV_HOTBACKUP */
#endif /* UNIV_DEBUG */
/************************************************************************* /*************************************************************************
Checks if some other transaction has a conflicting explicit lock request Checks if some other transaction has a conflicting explicit lock request
...@@ -1576,23 +1590,25 @@ static ...@@ -1576,23 +1590,25 @@ static
lock_t* lock_t*
lock_rec_other_has_conflicting( lock_rec_other_has_conflicting(
/*===========================*/ /*===========================*/
/* out: lock or NULL */ /* out: lock or NULL */
ulint mode, /* in: LOCK_S or LOCK_X, ulint mode, /* in: LOCK_S or LOCK_X,
possibly ORed to LOCK_GAP or LOC_REC_NOT_GAP, possibly ORed to LOCK_GAP or
LOCK_INSERT_INTENTION */ LOC_REC_NOT_GAP,
rec_t* rec, /* in: record to look at */ LOCK_INSERT_INTENTION */
ulint heap_no,/* in: heap number of the record */ const buf_block_t* block, /* in: buffer block containing
trx_t* trx) /* in: our transaction */ the record */
ulint heap_no,/* in: heap number of the record */
trx_t* trx) /* in: our transaction */
{ {
lock_t* lock; lock_t* lock;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
if (UNIV_LIKELY_NULL(lock)) { if (UNIV_LIKELY_NULL(lock)) {
if (page_rec_is_supremum(rec)) { if (UNIV_UNLIKELY(heap_no == PAGE_HEAP_NO_SUPREMUM)) {
do { do {
if (lock_rec_has_to_wait(trx, mode, lock, if (lock_rec_has_to_wait(trx, mode, lock,
...@@ -1659,11 +1675,11 @@ lock_sec_rec_some_has_impl_off_kernel( ...@@ -1659,11 +1675,11 @@ lock_sec_rec_some_has_impl_off_kernel(
/*==================================*/ /*==================================*/
/* out: transaction which has the x-lock, or /* out: transaction which has the x-lock, or
NULL */ NULL */
rec_t* rec, /* in: user record */ const rec_t* rec, /* in: user record */
dict_index_t* index, /* in: secondary index */ dict_index_t* index, /* in: secondary index */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */ const ulint* offsets)/* in: rec_get_offsets(rec, index) */
{ {
page_t* page; const page_t* page = page_align((rec_t*) rec);
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
...@@ -1672,8 +1688,6 @@ lock_sec_rec_some_has_impl_off_kernel( ...@@ -1672,8 +1688,6 @@ lock_sec_rec_some_has_impl_off_kernel(
ut_ad(page_rec_is_user_rec(rec)); ut_ad(page_rec_is_user_rec(rec));
ut_ad(rec_offs_validate(rec, index, offsets)); ut_ad(rec_offs_validate(rec, index, offsets));
page = page_align(rec);
/* Some transaction may have an implicit x-lock on the record only /* Some transaction may have an implicit x-lock on the record only
if the max trx id for the page >= min trx id for the trx list, or if the max trx id for the page >= min trx id for the trx list, or
database recovery is running. We do not write the changes of a page database recovery is running. We do not write the changes of a page
...@@ -1745,28 +1759,30 @@ static ...@@ -1745,28 +1759,30 @@ static
lock_t* lock_t*
lock_rec_create( lock_rec_create(
/*============*/ /*============*/
/* out: created lock */ /* out: created lock */
ulint type_mode,/* in: lock mode and wait flag, type is ulint type_mode,/* in: lock mode and wait
ignored and replaced by LOCK_REC */ flag, type is ignored and
rec_t* rec, /* in: record on page */ replaced by LOCK_REC */
ulint heap_no,/* in: heap number of the record */ const buf_block_t* block, /* in: buffer block containing
dict_index_t* index, /* in: index of record */ the record */
trx_t* trx) /* in: transaction */ ulint heap_no,/* in: heap number of the record */
dict_index_t* index, /* in: index of record */
trx_t* trx) /* in: transaction */
{ {
page_t* page; lock_t* lock;
lock_t* lock; ulint page_no;
ulint page_no; ulint space;
ulint space; ulint n_bits;
ulint n_bits; ulint n_bytes;
ulint n_bytes; const page_t* page;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
page = page_align(rec); space = buf_block_get_space(block);
space = page_get_space_id(page); page_no = buf_block_get_page_no(block);
page_no = page_get_page_no(page); page = block->frame;
ut_ad(!!page_is_comp(page) == dict_table_is_comp(index->table)); ut_ad(!!page_is_comp(page) == dict_table_is_comp(index->table));
...@@ -1774,7 +1790,7 @@ lock_rec_create( ...@@ -1774,7 +1790,7 @@ lock_rec_create(
LOCK_REC_NOT_GAP bits, as all locks on the supremum are LOCK_REC_NOT_GAP bits, as all locks on the supremum are
automatically of the gap type */ automatically of the gap type */
if (rec == page_get_supremum_rec(page)) { if (UNIV_UNLIKELY(heap_no == PAGE_HEAP_NO_SUPREMUM)) {
ut_ad(!(type_mode & LOCK_REC_NOT_GAP)); ut_ad(!(type_mode & LOCK_REC_NOT_GAP));
type_mode = type_mode & ~(LOCK_GAP | LOCK_REC_NOT_GAP); type_mode = type_mode & ~(LOCK_GAP | LOCK_REC_NOT_GAP);
...@@ -1822,25 +1838,32 @@ static ...@@ -1822,25 +1838,32 @@ static
ulint ulint
lock_rec_enqueue_waiting( lock_rec_enqueue_waiting(
/*=====================*/ /*=====================*/
/* out: DB_LOCK_WAIT, DB_DEADLOCK, or /* out: DB_LOCK_WAIT,
DB_QUE_THR_SUSPENDED, or DB_SUCCESS; DB_DEADLOCK, or
DB_SUCCESS means that there was a deadlock, DB_QUE_THR_SUSPENDED, or
but another transaction was chosen as a DB_SUCCESS; DB_SUCCESS means
victim, and we got the lock immediately: that there was a deadlock, but
no need to wait then */ another transaction was chosen
ulint type_mode,/* in: lock mode this transaction is as a victim, and we got the
requesting: LOCK_S or LOCK_X, possibly ORed lock immediately: no need to
with LOCK_GAP or LOCK_REC_NOT_GAP, ORed wait then */
with LOCK_INSERT_INTENTION if this waiting ulint type_mode,/* in: lock mode this
lock request is set when performing an transaction is requesting:
insert of an index record */ LOCK_S or LOCK_X, possibly
rec_t* rec, /* in: record */ ORed with LOCK_GAP or
dict_index_t* index, /* in: index of record */ LOCK_REC_NOT_GAP, ORed with
que_thr_t* thr) /* in: query thread */ LOCK_INSERT_INTENTION if this
waiting lock request is set
when performing an insert of
an index record */
const buf_block_t* block, /* in: buffer block containing
the record */
ulint heap_no,/* in: heap number of the record */
dict_index_t* index, /* in: index of record */
que_thr_t* thr) /* in: query thread */
{ {
lock_t* lock; lock_t* lock;
trx_t* trx; trx_t* trx;
ulint heap_no;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
...@@ -1871,15 +1894,9 @@ lock_rec_enqueue_waiting( ...@@ -1871,15 +1894,9 @@ lock_rec_enqueue_waiting(
stderr); stderr);
} }
if (page_rec_is_comp(rec)) {
heap_no = rec_get_heap_no_new(rec);
} else {
heap_no = rec_get_heap_no_old(rec);
}
/* Enqueue the lock request that will wait to be granted */ /* Enqueue the lock request that will wait to be granted */
lock = lock_rec_create(type_mode | LOCK_WAIT, rec, lock = lock_rec_create(type_mode | LOCK_WAIT,
heap_no, index, trx); block, heap_no, index, trx);
/* Check if a deadlock occurs: if yes, remove the lock request and /* Check if a deadlock occurs: if yes, remove the lock request and
return an error code */ return an error code */
...@@ -1928,13 +1945,15 @@ static ...@@ -1928,13 +1945,15 @@ static
lock_t* lock_t*
lock_rec_add_to_queue( lock_rec_add_to_queue(
/*==================*/ /*==================*/
/* out: lock where the bit was set */ /* out: lock where the bit was set */
ulint type_mode,/* in: lock mode, wait, gap etc. flags; ulint type_mode,/* in: lock mode, wait, gap
type is ignored and replaced by LOCK_REC */ etc. flags; type is ignored
rec_t* rec, /* in: record on page */ and replaced by LOCK_REC */
ulint heap_no,/* in: heap number of the record */ const buf_block_t* block, /* in: buffer block containing
dict_index_t* index, /* in: index of record */ the record */
trx_t* trx) /* in: transaction */ ulint heap_no,/* in: heap number of the record */
dict_index_t* index, /* in: index of record */
trx_t* trx) /* in: transaction */
{ {
lock_t* lock; lock_t* lock;
...@@ -1945,11 +1964,11 @@ lock_rec_add_to_queue( ...@@ -1945,11 +1964,11 @@ lock_rec_add_to_queue(
ut_ad((type_mode & (LOCK_WAIT | LOCK_GAP)) ut_ad((type_mode & (LOCK_WAIT | LOCK_GAP))
|| ((type_mode & LOCK_MODE_MASK) != LOCK_S) || ((type_mode & LOCK_MODE_MASK) != LOCK_S)
|| !lock_rec_other_has_expl_req(LOCK_X, 0, LOCK_WAIT, || !lock_rec_other_has_expl_req(LOCK_X, 0, LOCK_WAIT,
rec, heap_no, trx)); block, heap_no, trx));
ut_ad((type_mode & (LOCK_WAIT | LOCK_GAP)) ut_ad((type_mode & (LOCK_WAIT | LOCK_GAP))
|| ((type_mode & LOCK_MODE_MASK) != LOCK_X) || ((type_mode & LOCK_MODE_MASK) != LOCK_X)
|| !lock_rec_other_has_expl_req(LOCK_S, 0, LOCK_WAIT, || !lock_rec_other_has_expl_req(LOCK_S, 0, LOCK_WAIT,
rec, heap_no, trx)); block, heap_no, trx));
type_mode = type_mode | LOCK_REC; type_mode = type_mode | LOCK_REC;
...@@ -1958,7 +1977,7 @@ lock_rec_add_to_queue( ...@@ -1958,7 +1977,7 @@ lock_rec_add_to_queue(
try to avoid unnecessary memory consumption of a new record lock try to avoid unnecessary memory consumption of a new record lock
struct for a gap type lock */ struct for a gap type lock */
if (page_rec_is_supremum(rec)) { if (UNIV_UNLIKELY(heap_no == PAGE_HEAP_NO_SUPREMUM)) {
ut_ad(!(type_mode & LOCK_REC_NOT_GAP)); ut_ad(!(type_mode & LOCK_REC_NOT_GAP));
/* There should never be LOCK_REC_NOT_GAP on a supremum /* There should never be LOCK_REC_NOT_GAP on a supremum
...@@ -1969,7 +1988,7 @@ lock_rec_add_to_queue( ...@@ -1969,7 +1988,7 @@ lock_rec_add_to_queue(
/* 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 */
lock = lock_rec_get_first_on_page(rec); lock = lock_rec_get_first_on_page(block);
while (lock != NULL) { while (lock != NULL) {
if (lock_get_wait(lock) if (lock_get_wait(lock)
...@@ -1989,7 +2008,7 @@ lock_rec_add_to_queue( ...@@ -1989,7 +2008,7 @@ lock_rec_add_to_queue(
lock = lock_rec_find_similar_on_page( lock = lock_rec_find_similar_on_page(
type_mode, heap_no, type_mode, heap_no,
lock_rec_get_first_on_page(rec), trx); lock_rec_get_first_on_page(block), trx);
if (lock) { if (lock) {
...@@ -2000,7 +2019,7 @@ lock_rec_add_to_queue( ...@@ -2000,7 +2019,7 @@ lock_rec_add_to_queue(
} }
somebody_waits: somebody_waits:
return(lock_rec_create(type_mode, rec, heap_no, index, trx)); return(lock_rec_create(type_mode, block, heap_no, index, trx));
} }
/************************************************************************* /*************************************************************************
...@@ -2014,16 +2033,19 @@ UNIV_INLINE ...@@ -2014,16 +2033,19 @@ UNIV_INLINE
ibool ibool
lock_rec_lock_fast( lock_rec_lock_fast(
/*===============*/ /*===============*/
/* out: TRUE if locking succeeded */ /* out: TRUE if locking succeeded */
ibool impl, /* in: if TRUE, no lock is set if no wait ibool impl, /* in: if TRUE, no lock is set
is necessary: we assume that the caller will if no wait is necessary: we
set an implicit lock */ assume that the caller will
ulint mode, /* in: lock mode: LOCK_X or LOCK_S possibly set an implicit lock */
ORed to either LOCK_GAP or LOCK_REC_NOT_GAP */ ulint mode, /* in: lock mode: LOCK_X or
rec_t* rec, /* in: record */ LOCK_S possibly ORed to either
ulint heap_no,/* in: heap number of record */ LOCK_GAP or LOCK_REC_NOT_GAP */
dict_index_t* index, /* in: index of record */ const buf_block_t* block, /* in: buffer block containing
que_thr_t* thr) /* in: query thread */ the record */
ulint heap_no,/* in: heap number of record */
dict_index_t* index, /* in: index of record */
que_thr_t* thr) /* in: query thread */
{ {
lock_t* lock; lock_t* lock;
trx_t* trx; trx_t* trx;
...@@ -2041,13 +2063,13 @@ lock_rec_lock_fast( ...@@ -2041,13 +2063,13 @@ lock_rec_lock_fast(
|| mode - (LOCK_MODE_MASK & mode) == 0 || mode - (LOCK_MODE_MASK & mode) == 0
|| mode - (LOCK_MODE_MASK & mode) == LOCK_REC_NOT_GAP); || mode - (LOCK_MODE_MASK & mode) == LOCK_REC_NOT_GAP);
lock = lock_rec_get_first_on_page(rec); lock = lock_rec_get_first_on_page(block);
trx = thr_get_trx(thr); trx = thr_get_trx(thr);
if (lock == NULL) { if (lock == NULL) {
if (!impl) { if (!impl) {
lock_rec_create(mode, rec, heap_no, index, trx); lock_rec_create(mode, block, heap_no, index, trx);
if (srv_locks_unsafe_for_binlog if (srv_locks_unsafe_for_binlog
|| trx->isolation_level || trx->isolation_level
...@@ -2097,17 +2119,20 @@ static ...@@ -2097,17 +2119,20 @@ static
ulint ulint
lock_rec_lock_slow( lock_rec_lock_slow(
/*===============*/ /*===============*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, or error /* out: DB_SUCCESS,
code */ DB_LOCK_WAIT, or error code */
ibool impl, /* in: if TRUE, no lock is set if no wait is ibool impl, /* in: if TRUE, no lock is set
necessary: we assume that the caller will set if no wait is necessary: we
an implicit lock */ assume that the caller will
ulint mode, /* in: lock mode: LOCK_X or LOCK_S possibly set an implicit lock */
ORed to either LOCK_GAP or LOCK_REC_NOT_GAP */ ulint mode, /* in: lock mode: LOCK_X or
rec_t* rec, /* in: record */ LOCK_S possibly ORed to either
ulint heap_no,/* in: heap number of record */ LOCK_GAP or LOCK_REC_NOT_GAP */
dict_index_t* index, /* in: index of record */ const buf_block_t* block, /* in: buffer block containing
que_thr_t* thr) /* in: query thread */ the record */
ulint heap_no,/* in: heap number of record */
dict_index_t* index, /* in: index of record */
que_thr_t* thr) /* in: query thread */
{ {
trx_t* trx; trx_t* trx;
ulint err; ulint err;
...@@ -2127,18 +2152,19 @@ lock_rec_lock_slow( ...@@ -2127,18 +2152,19 @@ lock_rec_lock_slow(
trx = thr_get_trx(thr); trx = thr_get_trx(thr);
if (lock_rec_has_expl(mode, rec, heap_no, trx)) { if (lock_rec_has_expl(mode, block, heap_no, trx)) {
/* The trx already has a strong enough lock on rec: do /* The trx already has a strong enough lock on rec: do
nothing */ nothing */
err = DB_SUCCESS; err = DB_SUCCESS;
} else if (lock_rec_other_has_conflicting(mode, rec, heap_no, trx)) { } else if (lock_rec_other_has_conflicting(mode, block, heap_no, trx)) {
/* If another transaction has a non-gap conflicting request in /* If another transaction has a non-gap conflicting request in
the queue, as this transaction does not have a lock strong the queue, as this transaction does not have a lock strong
enough already granted on the record, we have to wait. */ enough already granted on the record, we have to wait. */
err = lock_rec_enqueue_waiting(mode, rec, index, thr); err = lock_rec_enqueue_waiting(mode, block, heap_no,
index, thr);
if (srv_locks_unsafe_for_binlog if (srv_locks_unsafe_for_binlog
|| trx->isolation_level == TRX_ISO_READ_COMMITTED) { || trx->isolation_level == TRX_ISO_READ_COMMITTED) {
...@@ -2148,7 +2174,7 @@ lock_rec_lock_slow( ...@@ -2148,7 +2174,7 @@ lock_rec_lock_slow(
if (!impl) { if (!impl) {
/* Set the requested lock on the record */ /* Set the requested lock on the record */
lock_rec_add_to_queue(LOCK_REC | mode, rec, lock_rec_add_to_queue(LOCK_REC | mode, block,
heap_no, index, trx); heap_no, index, trx);
if (srv_locks_unsafe_for_binlog if (srv_locks_unsafe_for_binlog
|| trx->isolation_level || trx->isolation_level
...@@ -2173,19 +2199,22 @@ static ...@@ -2173,19 +2199,22 @@ static
ulint ulint
lock_rec_lock( lock_rec_lock(
/*==========*/ /*==========*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, or error /* out: DB_SUCCESS,
code */ DB_LOCK_WAIT, or error code */
ibool impl, /* in: if TRUE, no lock is set if no wait is ibool impl, /* in: if TRUE, no lock is set
necessary: we assume that the caller will set if no wait is necessary: we
an implicit lock */ assume that the caller will
ulint mode, /* in: lock mode: LOCK_X or LOCK_S possibly set an implicit lock */
ORed to either LOCK_GAP or LOCK_REC_NOT_GAP */ ulint mode, /* in: lock mode: LOCK_X or
rec_t* rec, /* in: record */ LOCK_S possibly ORed to either
dict_index_t* index, /* in: index of record */ LOCK_GAP or LOCK_REC_NOT_GAP */
que_thr_t* thr) /* in: query thread */ const buf_block_t* block, /* in: buffer block containing
the record */
ulint heap_no,/* in: heap number of record */
dict_index_t* index, /* in: index of record */
que_thr_t* thr) /* in: query thread */
{ {
ulint err; ulint err;
ulint heap_no;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
...@@ -2200,20 +2229,15 @@ lock_rec_lock( ...@@ -2200,20 +2229,15 @@ lock_rec_lock(
|| mode - (LOCK_MODE_MASK & mode) == LOCK_REC_NOT_GAP || mode - (LOCK_MODE_MASK & mode) == LOCK_REC_NOT_GAP
|| mode - (LOCK_MODE_MASK & mode) == 0); || mode - (LOCK_MODE_MASK & mode) == 0);
if (page_rec_is_comp(rec)) { if (lock_rec_lock_fast(impl, mode, block, heap_no, index, thr)) {
heap_no = rec_get_heap_no_new(rec);
} else {
heap_no = rec_get_heap_no_old(rec);
}
if (lock_rec_lock_fast(impl, mode, rec, heap_no, index, thr)) {
/* We try a simplified and faster subroutine for the most /* We try a simplified and faster subroutine for the most
common cases */ common cases */
err = DB_SUCCESS; err = DB_SUCCESS;
} else { } else {
err = lock_rec_lock_slow(impl, mode, rec, heap_no, index, thr); err = lock_rec_lock_slow(impl, mode, block,
heap_no, index, thr);
} }
return(err); return(err);
...@@ -2419,7 +2443,7 @@ static ...@@ -2419,7 +2443,7 @@ static
void void
lock_rec_free_all_from_discard_page( lock_rec_free_all_from_discard_page(
/*================================*/ /*================================*/
page_t* page) /* in: page to be discarded */ const buf_block_t* block) /* in: page to be discarded */
{ {
ulint space; ulint space;
ulint page_no; ulint page_no;
...@@ -2430,8 +2454,8 @@ lock_rec_free_all_from_discard_page( ...@@ -2430,8 +2454,8 @@ lock_rec_free_all_from_discard_page(
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
space = page_get_space_id(page); space = buf_block_get_space(block);
page_no = page_get_page_no(page); page_no = buf_block_get_page_no(block);
lock = lock_rec_get_first_on_page_addr(space, page_no); lock = lock_rec_get_first_on_page_addr(space, page_no);
...@@ -2456,22 +2480,17 @@ static ...@@ -2456,22 +2480,17 @@ static
void void
lock_rec_reset_and_release_wait( lock_rec_reset_and_release_wait(
/*============================*/ /*============================*/
rec_t* rec) /* in: record whose locks bits should be reset */ const buf_block_t* block, /* in: buffer block containing
the record */
ulint heap_no)/* in: heap number of record */
{ {
lock_t* lock; lock_t* lock;
ulint heap_no;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
if (page_rec_is_comp(rec)) { lock = lock_rec_get_first(block, heap_no);
heap_no = rec_get_heap_no_new(rec);
} else {
heap_no = rec_get_heap_no_old(rec);
}
lock = lock_rec_get_first(rec, heap_no);
while (lock != NULL) { while (lock != NULL) {
if (lock_get_wait(lock)) { if (lock_get_wait(lock)) {
...@@ -2489,30 +2508,27 @@ Makes a record to inherit the locks (except LOCK_INSERT_INTENTION type) ...@@ -2489,30 +2508,27 @@ Makes a record to inherit the locks (except LOCK_INSERT_INTENTION type)
of another record as gap type locks, but does not reset the lock bits of of another record as gap type locks, but does not reset the lock bits of
the other record. Also waiting lock requests on rec are inherited as the other record. Also waiting lock requests on rec are inherited as
GRANTED gap locks. */ GRANTED gap locks. */
static
void void
lock_rec_inherit_to_gap( lock_rec_inherit_to_gap(
/*====================*/ /*====================*/
rec_t* heir, /* in: record which inherits */ const buf_block_t* heir_block, /* in: block containing the
rec_t* rec) /* in: record from which inherited; does NOT reset record which inherits */
the locks on this record */ const buf_block_t* block, /* in: block containing the
record from which inherited;
does NOT reset the locks on
this record */
ulint heir_heap_no, /* in: heap_no of the
inheriting record */
ulint heap_no) /* in: heap_no of the
donating record */
{ {
lock_t* lock; lock_t* lock;
ulint heir_heap_no;
ulint heap_no;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
if (page_rec_is_comp(rec)) { lock = lock_rec_get_first(block, heap_no);
heir_heap_no = rec_get_heap_no_new(heir);
heap_no = rec_get_heap_no_new(rec);
} else {
heir_heap_no = rec_get_heap_no_old(heir);
heap_no = rec_get_heap_no_old(rec);
}
lock = lock_rec_get_first(rec, heap_no);
/* If srv_locks_unsafe_for_binlog is TRUE or session is using /* If srv_locks_unsafe_for_binlog is TRUE or session is using
READ COMMITTED isolation level, we do not want locks set READ COMMITTED isolation level, we do not want locks set
...@@ -2529,7 +2545,7 @@ lock_rec_inherit_to_gap( ...@@ -2529,7 +2545,7 @@ lock_rec_inherit_to_gap(
lock_rec_add_to_queue(LOCK_REC | LOCK_GAP lock_rec_add_to_queue(LOCK_REC | LOCK_GAP
| lock_get_mode(lock), | lock_get_mode(lock),
heir, heir_heap_no, heir_block, heir_heap_no,
lock->index, lock->trx); lock->index, lock->trx);
} }
...@@ -2545,35 +2561,29 @@ static ...@@ -2545,35 +2561,29 @@ static
void void
lock_rec_inherit_to_gap_if_gap_lock( lock_rec_inherit_to_gap_if_gap_lock(
/*================================*/ /*================================*/
rec_t* heir, /* in: record which inherits */ const buf_block_t* block, /* in: buffer block */
rec_t* rec) /* in: record from which inherited; does NOT reset ulint heir_heap_no, /* in: heap_no of
the locks on this record */ record which inherits */
ulint heap_no) /* in: heap_no of record
from which inherited;
does NOT reset the locks
on this record */
{ {
lock_t* lock; lock_t* lock;
ulint heir_heap_no;
ulint heap_no;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
if (page_rec_is_comp(rec)) { lock = lock_rec_get_first(block, heap_no);
heir_heap_no = rec_get_heap_no_new(heir);
heap_no = rec_get_heap_no_new(rec);
} else {
heir_heap_no = rec_get_heap_no_old(heir);
heap_no = rec_get_heap_no_old(rec);
}
lock = lock_rec_get_first(rec, heap_no);
while (lock != NULL) { while (lock != NULL) {
if (!lock_rec_get_insert_intention(lock) if (!lock_rec_get_insert_intention(lock)
&& (page_rec_is_supremum(rec) && (heap_no == PAGE_HEAP_NO_SUPREMUM
|| !lock_rec_get_rec_not_gap(lock))) { || !lock_rec_get_rec_not_gap(lock))) {
lock_rec_add_to_queue(LOCK_REC | LOCK_GAP lock_rec_add_to_queue(LOCK_REC | LOCK_GAP
| lock_get_mode(lock), | lock_get_mode(lock),
heir, heir_heap_no, block, heir_heap_no,
lock->index, lock->trx); lock->index, lock->trx);
} }
...@@ -2588,28 +2598,24 @@ static ...@@ -2588,28 +2598,24 @@ static
void void
lock_rec_move( lock_rec_move(
/*==========*/ /*==========*/
rec_t* receiver, /* in: record which gets locks; this record const buf_block_t* receiver, /* in: buffer block containing
must have no lock requests on it! */ the receiving record */
rec_t* donator, /* in: record which gives locks */ const buf_block_t* donator, /* in: buffer block containing
ulint comp) /* in: nonzero=compact page format */ the donating record */
ulint receiver_heap_no,/* in: heap_no of the record
which gets the locks; there
must be no lock requests
on it! */
ulint donator_heap_no)/* in: heap_no of the record
which gives the locks */
{ {
lock_t* lock; lock_t* lock;
ulint receiver_heap_no;
ulint donator_heap_no;
ulint type_mode; ulint type_mode;
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
ut_ad(mutex_own(&kernel_mutex)); ut_ad(mutex_own(&kernel_mutex));
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
if (UNIV_LIKELY(comp)) {
receiver_heap_no = rec_get_heap_no_new(receiver);
donator_heap_no = rec_get_heap_no_new(donator);
} else {
receiver_heap_no = rec_get_heap_no_old(receiver);
donator_heap_no = rec_get_heap_no_old(donator);
}
lock = lock_rec_get_first(donator, donator_heap_no); lock = lock_rec_get_first(donator, donator_heap_no);
ut_ad(lock_rec_get_first(receiver, receiver_heap_no) == NULL); ut_ad(lock_rec_get_first(receiver, receiver_heap_no) == NULL);
...@@ -2643,8 +2649,10 @@ were temporarily stored on the infimum. */ ...@@ -2643,8 +2649,10 @@ were temporarily stored on the infimum. */
void void
lock_move_reorganize_page( lock_move_reorganize_page(
/*======================*/ /*======================*/
page_t* page, /* in: old index page, now reorganized */ const buf_block_t* block, /* in: old index page, now
page_t* old_page) /* in: copy of the old, not reorganized page */ reorganized */
const buf_block_t* oblock) /* in: copy of the old, not
reorganized page */
{ {
lock_t* lock; lock_t* lock;
lock_t* old_lock; lock_t* old_lock;
...@@ -2658,7 +2666,7 @@ lock_move_reorganize_page( ...@@ -2658,7 +2666,7 @@ lock_move_reorganize_page(
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
lock = lock_rec_get_first_on_page(page); lock = lock_rec_get_first_on_page(block);
if (lock == NULL) { if (lock == NULL) {
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
...@@ -2693,8 +2701,8 @@ lock_move_reorganize_page( ...@@ -2693,8 +2701,8 @@ lock_move_reorganize_page(
lock = UT_LIST_GET_FIRST(old_locks); lock = UT_LIST_GET_FIRST(old_locks);
comp = page_is_comp(page); comp = page_is_comp(block->frame);
ut_ad(comp == page_is_comp(old_page)); ut_ad(comp == page_is_comp(oblock->frame));
while (lock) { while (lock) {
/* NOTE: we copy also the locks set on the infimum and /* NOTE: we copy also the locks set on the infimum and
...@@ -2702,8 +2710,8 @@ lock_move_reorganize_page( ...@@ -2702,8 +2710,8 @@ lock_move_reorganize_page(
update of a record is occurring on the page, and its locks update of a record is occurring on the page, and its locks
were temporarily stored on the infimum */ were temporarily stored on the infimum */
page_cur_set_before_first(buf_block_align(page), &cur1); page_cur_set_before_first((buf_block_t*) block, &cur1);
page_cur_set_before_first(buf_block_align(old_page), &cur2); page_cur_set_before_first((buf_block_t*) oblock, &cur2);
/* Set locks according to old locks */ /* Set locks according to old locks */
for (;;) { for (;;) {
...@@ -2729,12 +2737,11 @@ lock_move_reorganize_page( ...@@ -2729,12 +2737,11 @@ lock_move_reorganize_page(
/* NOTE that the old lock bitmap could be too /* NOTE that the old lock bitmap could be too
small for the new heap number! */ small for the new heap number! */
lock_rec_add_to_queue(lock->type_mode, lock_rec_add_to_queue(lock->type_mode, block,
page_cur_get_rec(&cur1),
new_heap_no, new_heap_no,
lock->index, lock->trx); lock->index, lock->trx);
/* if ((page_cur_is_after_last(&cur1)) /* if (new_heap_no == PAGE_HEAP_NO_SUPREMUM
&& lock_get_wait(lock)) { && lock_get_wait(lock)) {
fprintf(stderr, fprintf(stderr,
"---\n--\n!!!Lock reorg: supr type %lu\n", "---\n--\n!!!Lock reorg: supr type %lu\n",
...@@ -2742,7 +2749,8 @@ lock_move_reorganize_page( ...@@ -2742,7 +2749,8 @@ lock_move_reorganize_page(
} */ } */
} }
if (page_cur_is_after_last(&cur1)) { if (UNIV_UNLIKELY
(new_heap_no == PAGE_HEAP_NO_SUPREMUM)) {
break; break;
} }
...@@ -2773,18 +2781,17 @@ list end is moved to another page. */ ...@@ -2773,18 +2781,17 @@ list end is moved to another page. */
void void
lock_move_rec_list_end( lock_move_rec_list_end(
/*===================*/ /*===================*/
page_t* new_page, /* in: index page to move to */ const buf_block_t* new_block, /* in: index page to move to */
page_t* page, /* in: index page */ const buf_block_t* block, /* in: index page */
rec_t* rec) /* in: record on page: this is the const rec_t* rec) /* in: record on page: this
first record moved */ is the first record moved */
{ {
lock_t* lock; lock_t* lock;
page_cur_t cur1; page_cur_t cur1;
page_cur_t cur2; page_cur_t cur2;
ulint heap_no; ulint heap_no;
ulint type_mode; ulint type_mode;
ut_ad(page_is_comp(page) == page_is_comp(new_page)); const ulint comp = page_rec_is_comp(rec);
ut_ad(page == page_align(rec));
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
...@@ -2794,29 +2801,29 @@ lock_move_rec_list_end( ...@@ -2794,29 +2801,29 @@ 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. */
lock = lock_rec_get_first_on_page(page); lock = lock_rec_get_first_on_page(block);
while (lock != NULL) { while (lock != NULL) {
page_cur_position(rec, buf_block_align(rec), &cur1); page_cur_position((rec_t*) rec, (buf_block_t*) block, &cur1);
if (page_cur_is_before_first(&cur1)) { if (page_cur_is_before_first(&cur1)) {
page_cur_move_to_next(&cur1); page_cur_move_to_next(&cur1);
} }
page_cur_set_before_first(buf_block_align(new_page), &cur2); page_cur_set_before_first((buf_block_t*) new_block, &cur2);
page_cur_move_to_next(&cur2); page_cur_move_to_next(&cur2);
/* Copy lock requests on user records to new page and /* Copy lock requests on user records to new page and
reset the lock bits on the old */ reset the lock bits on the old */
while (!page_cur_is_after_last(&cur1)) { while (!page_cur_is_after_last(&cur1)) {
ut_ad(page_is_comp(page) ut_ad(comp
|| !memcmp(page_cur_get_rec(&cur1), || !memcmp(page_cur_get_rec(&cur1),
page_cur_get_rec(&cur2), page_cur_get_rec(&cur2),
rec_get_data_size_old( rec_get_data_size_old(
page_cur_get_rec(&cur2)))); page_cur_get_rec(&cur2))));
if (page_is_comp(page)) { if (comp) {
heap_no = rec_get_heap_no_new( heap_no = rec_get_heap_no_new(
page_cur_get_rec(&cur1)); page_cur_get_rec(&cur1));
} else { } else {
...@@ -2833,7 +2840,7 @@ lock_move_rec_list_end( ...@@ -2833,7 +2840,7 @@ lock_move_rec_list_end(
lock_reset_lock_and_trx_wait(lock); lock_reset_lock_and_trx_wait(lock);
} }
if (page_is_comp(page)) { if (comp) {
heap_no = rec_get_heap_no_new( heap_no = rec_get_heap_no_new(
page_cur_get_rec(&cur2)); page_cur_get_rec(&cur2));
} else { } else {
...@@ -2842,8 +2849,7 @@ lock_move_rec_list_end( ...@@ -2842,8 +2849,7 @@ lock_move_rec_list_end(
} }
lock_rec_add_to_queue(type_mode, lock_rec_add_to_queue(type_mode,
page_cur_get_rec(&cur2), new_block, heap_no,
heap_no,
lock->index, lock->trx); lock->index, lock->trx);
} }
...@@ -2857,10 +2863,10 @@ lock_move_rec_list_end( ...@@ -2857,10 +2863,10 @@ lock_move_rec_list_end(
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
#if 0 #if 0
ut_ad(lock_rec_validate_page(page_get_space_id(page), ut_ad(lock_rec_validate_page(buf_block_get_space(block),
page_get_page_no(page))); buf_block_get_page_no(block)));
ut_ad(lock_rec_validate_page(page_get_space_id(new_page), ut_ad(lock_rec_validate_page(buf_block_get_space(new_block),
page_get_page_no(new_page))); buf_block_get_page_no(new_block)));
#endif #endif
} }
...@@ -2871,45 +2877,50 @@ list start is moved to another page. */ ...@@ -2871,45 +2877,50 @@ list start is moved to another page. */
void void
lock_move_rec_list_start( lock_move_rec_list_start(
/*=====================*/ /*=====================*/
page_t* new_page, /* in: index page to move to */ const buf_block_t* new_block, /* in: index page to move to */
page_t* page, /* in: index page */ const buf_block_t* block, /* in: index page */
rec_t* rec, /* in: record on page: this is the const rec_t* rec, /* in: record on page:
first record NOT copied */ this is the first
rec_t* old_end) /* in: old previous-to-last record on record NOT copied */
new_page before the records were copied */ const rec_t* old_end) /* in: old
previous-to-last
record on new_page
before the records
were copied */
{ {
lock_t* lock; lock_t* lock;
page_cur_t cur1; page_cur_t cur1;
page_cur_t cur2; page_cur_t cur2;
ulint heap_no; ulint heap_no;
ulint type_mode; ulint type_mode;
const ulint comp = page_rec_is_comp(rec);
ut_a(new_page); ut_ad(block->frame == page_align((rec_t*) rec));
ut_ad(new_block->frame == page_align((rec_t*) old_end));
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
lock = lock_rec_get_first_on_page(page); lock = lock_rec_get_first_on_page(block);
ut_ad(page_is_comp(page) == page_is_comp(new_page));
ut_ad(page == page_align(rec));
while (lock != NULL) { while (lock != NULL) {
page_cur_set_before_first(buf_block_align(page), &cur1); page_cur_set_before_first((buf_block_t*) block, &cur1);
page_cur_move_to_next(&cur1); page_cur_move_to_next(&cur1);
page_cur_position(old_end, buf_block_align(old_end), &cur2); page_cur_position((rec_t*) old_end, (buf_block_t*) new_block,
&cur2);
page_cur_move_to_next(&cur2); page_cur_move_to_next(&cur2);
/* Copy lock requests on user records to new page and /* Copy lock requests on user records to new page and
reset the lock bits on the old */ reset the lock bits on the old */
while (page_cur_get_rec(&cur1) != rec) { while (page_cur_get_rec(&cur1) != rec) {
ut_ad(page_is_comp(page) ut_ad(comp
|| !memcmp(page_cur_get_rec(&cur1), || !memcmp(page_cur_get_rec(&cur1),
page_cur_get_rec(&cur2), page_cur_get_rec(&cur2),
rec_get_data_size_old( rec_get_data_size_old(
page_cur_get_rec(&cur2)))); page_cur_get_rec(&cur2))));
if (page_is_comp(page)) { if (comp) {
heap_no = rec_get_heap_no_new( heap_no = rec_get_heap_no_new(
page_cur_get_rec(&cur1)); page_cur_get_rec(&cur1));
} else { } else {
...@@ -2926,7 +2937,7 @@ lock_move_rec_list_start( ...@@ -2926,7 +2937,7 @@ lock_move_rec_list_start(
lock_reset_lock_and_trx_wait(lock); lock_reset_lock_and_trx_wait(lock);
} }
if (page_is_comp(page)) { if (comp) {
heap_no = rec_get_heap_no_new( heap_no = rec_get_heap_no_new(
page_cur_get_rec(&cur2)); page_cur_get_rec(&cur2));
} else { } else {
...@@ -2935,8 +2946,7 @@ lock_move_rec_list_start( ...@@ -2935,8 +2946,7 @@ lock_move_rec_list_start(
} }
lock_rec_add_to_queue(type_mode, lock_rec_add_to_queue(type_mode,
page_cur_get_rec(&cur2), new_block, heap_no,
heap_no,
lock->index, lock->trx); lock->index, lock->trx);
} }
...@@ -2949,10 +2959,8 @@ lock_move_rec_list_start( ...@@ -2949,10 +2959,8 @@ lock_move_rec_list_start(
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
#if 0 #if 0
ut_ad(lock_rec_validate_page(page_get_space_id(page), ut_ad(lock_rec_validate_page(buf_block_get_space(block),
page_get_page_no(page))); buf_block_get_page_no(block)));
ut_ad(lock_rec_validate_page(page_get_space_id(new_page),
page_get_page_no(new_page)));
#endif #endif
} }
...@@ -2962,25 +2970,24 @@ Updates the lock table when a page is split to the right. */ ...@@ -2962,25 +2970,24 @@ Updates the lock table when a page is split to the right. */
void void
lock_update_split_right( lock_update_split_right(
/*====================*/ /*====================*/
page_t* right_page, /* in: right page */ const buf_block_t* right_block, /* in: right page */
page_t* left_page) /* in: left page */ const buf_block_t* left_block) /* in: left page */
{ {
ulint heap_no = lock_get_min_heap_no(right_block);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
ut_ad(page_is_comp(left_page) == page_is_comp(right_page));
/* Move the locks on the supremum of the left page to the supremum /* Move the locks on the supremum of the left page to the supremum
of the right page */ of the right page */
lock_rec_move(page_get_supremum_rec(right_page), lock_rec_move(right_block, left_block,
page_get_supremum_rec(left_page), PAGE_HEAP_NO_SUPREMUM, PAGE_HEAP_NO_SUPREMUM);
page_is_comp(left_page));
/* Inherit the locks to the supremum of left page from the successor /* Inherit the locks to the supremum of left page from the successor
of the infimum on right page */ of the infimum on right page */
lock_rec_inherit_to_gap(page_get_supremum_rec(left_page), lock_rec_inherit_to_gap(left_block, right_block,
page_rec_get_next( PAGE_HEAP_NO_SUPREMUM, heap_no);
page_get_infimum_rec(right_page)));
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -2991,10 +2998,15 @@ Updates the lock table when a page is merged to the right. */ ...@@ -2991,10 +2998,15 @@ Updates the lock table when a page is merged to the right. */
void void
lock_update_merge_right( lock_update_merge_right(
/*====================*/ /*====================*/
rec_t* orig_succ, /* in: original successor of infimum const buf_block_t* right_block, /* in: right page to
on the right page before merge */ which merged */
page_t* left_page) /* in: merged index page which will be const rec_t* orig_succ, /* in: original
discarded */ successor of infimum
on the right page
before merge */
const buf_block_t* left_block) /* in: merged index
page which will be
discarded */
{ {
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
...@@ -3002,14 +3014,17 @@ lock_update_merge_right( ...@@ -3002,14 +3014,17 @@ lock_update_merge_right(
original successor of infimum on the right page, to which the left original successor of infimum on the right page, to which the left
page was merged */ page was merged */
lock_rec_inherit_to_gap(orig_succ, page_get_supremum_rec(left_page)); lock_rec_inherit_to_gap(right_block, left_block,
page_rec_get_heap_no(orig_succ),
PAGE_HEAP_NO_SUPREMUM);
/* Reset the locks on the supremum of the left page, releasing /* Reset the locks on the supremum of the left page, releasing
waiting transactions */ waiting transactions */
lock_rec_reset_and_release_wait(page_get_supremum_rec(left_page)); lock_rec_reset_and_release_wait(left_block,
PAGE_HEAP_NO_SUPREMUM);
lock_rec_free_all_from_discard_page(left_page); lock_rec_free_all_from_discard_page(left_block);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3025,19 +3040,16 @@ to be updated. */ ...@@ -3025,19 +3040,16 @@ to be updated. */
void void
lock_update_root_raise( lock_update_root_raise(
/*===================*/ /*===================*/
page_t* new_page, /* in: index page to which copied */ const buf_block_t* block, /* in: index page to which copied */
page_t* root) /* in: root page */ const buf_block_t* root) /* in: root page */
{ {
ulint comp;
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
comp = page_is_comp(root);
ut_ad(comp == page_is_comp(new_page));
/* Move the locks on the supremum of the root to the supremum /* Move the locks on the supremum of the root to the supremum
of new_page */ of block */
lock_rec_move(page_get_supremum_rec(new_page), lock_rec_move(block, root,
page_get_supremum_rec(root), comp); PAGE_HEAP_NO_SUPREMUM, PAGE_HEAP_NO_SUPREMUM);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3048,20 +3060,19 @@ is removed from the chain of leaf pages, except if page is the root! */ ...@@ -3048,20 +3060,19 @@ is removed from the chain of leaf pages, except if page is the root! */
void void
lock_update_copy_and_discard( lock_update_copy_and_discard(
/*=========================*/ /*=========================*/
page_t* new_page, /* in: index page to which copied */ const buf_block_t* new_block, /* in: index page to
page_t* page) /* in: index page; NOT the root! */ which copied */
const buf_block_t* block) /* in: index page;
NOT the root! */
{ {
ulint comp;
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
comp = page_is_comp(page);
ut_ad(comp == page_is_comp(new_page));
/* Move the locks on the supremum of the old page to the supremum /* Move the locks on the supremum of the old page to the supremum
of new_page */ of new_page */
lock_rec_move(page_get_supremum_rec(new_page), lock_rec_move(new_block, block,
page_get_supremum_rec(page), comp); PAGE_HEAP_NO_SUPREMUM, PAGE_HEAP_NO_SUPREMUM);
lock_rec_free_all_from_discard_page(page); lock_rec_free_all_from_discard_page(block);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3072,17 +3083,18 @@ Updates the lock table when a page is split to the left. */ ...@@ -3072,17 +3083,18 @@ Updates the lock table when a page is split to the left. */
void void
lock_update_split_left( lock_update_split_left(
/*===================*/ /*===================*/
page_t* right_page, /* in: right page */ const buf_block_t* right_block, /* in: right page */
page_t* left_page) /* in: left page */ const buf_block_t* left_block) /* in: left page */
{ {
ulint heap_no = lock_get_min_heap_no(right_block);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
/* Inherit the locks to the supremum of the left page from the /* Inherit the locks to the supremum of the left page from the
successor of the infimum on the right page */ successor of the infimum on the right page */
lock_rec_inherit_to_gap(page_get_supremum_rec(left_page), lock_rec_inherit_to_gap(left_block, right_block,
page_rec_get_next( PAGE_HEAP_NO_SUPREMUM, heap_no);
page_get_infimum_rec(right_page)));
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3093,42 +3105,45 @@ Updates the lock table when a page is merged to the left. */ ...@@ -3093,42 +3105,45 @@ Updates the lock table when a page is merged to the left. */
void void
lock_update_merge_left( lock_update_merge_left(
/*===================*/ /*===================*/
page_t* left_page, /* in: left page to which merged */ const buf_block_t* left_block, /* in: left page to
rec_t* orig_pred, /* in: original predecessor of supremum which merged */
on the left page before merge */ const rec_t* orig_pred, /* in: original predecessor
page_t* right_page) /* in: merged index page which will be of supremum on the left page
discarded */ before merge */
const buf_block_t* right_block) /* in: merged index page
which will be discarded */
{ {
rec_t* left_next_rec; const rec_t* left_next_rec;
rec_t* left_supremum;
ulint comp; ut_ad(left_block->frame == page_align((rec_t*) orig_pred));
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
comp = page_is_comp(left_page);
ut_ad(comp == page_is_comp(right_page));
ut_ad(left_page == page_align(orig_pred));
left_next_rec = page_rec_get_next(orig_pred); left_next_rec = page_rec_get_next((rec_t*) orig_pred);
left_supremum = page_get_supremum_rec(left_page);
if (UNIV_LIKELY(left_next_rec != left_supremum)) { if (!page_rec_is_supremum(left_next_rec)) {
/* Inherit the locks on the supremum of the left page to the /* Inherit the locks on the supremum of the left page to the
first record which was moved from the right page */ first record which was moved from the right page */
lock_rec_inherit_to_gap(left_next_rec, left_supremum); lock_rec_inherit_to_gap(left_block, left_block,
page_rec_get_heap_no(left_next_rec),
PAGE_HEAP_NO_SUPREMUM);
/* Reset the locks on the supremum of the left page, /* Reset the locks on the supremum of the left page,
releasing waiting transactions */ releasing waiting transactions */
lock_rec_reset_and_release_wait(left_supremum); lock_rec_reset_and_release_wait(left_block,
PAGE_HEAP_NO_SUPREMUM);
} }
/* Move the locks from the supremum of right page to the supremum /* Move the locks from the supremum of right page to the supremum
of the left page */ of the left page */
lock_rec_move(left_supremum, page_get_supremum_rec(right_page), comp); lock_rec_move(left_block, right_block,
PAGE_HEAP_NO_SUPREMUM, PAGE_HEAP_NO_SUPREMUM);
lock_rec_free_all_from_discard_page(right_page); lock_rec_free_all_from_discard_page(right_block);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3140,14 +3155,22 @@ inherited from rec. */ ...@@ -3140,14 +3155,22 @@ inherited from rec. */
void void
lock_rec_reset_and_inherit_gap_locks( lock_rec_reset_and_inherit_gap_locks(
/*=================================*/ /*=================================*/
rec_t* heir, /* in: heir record */ const buf_block_t* heir_block, /* in: block containing the
rec_t* rec) /* in: record */ record which inherits */
const buf_block_t* block, /* in: block containing the
record from which inherited;
does NOT reset the locks on
this record */
ulint heir_heap_no, /* in: heap_no of the
inheriting record */
ulint heap_no) /* in: heap_no of the
donating record */
{ {
mutex_enter(&kernel_mutex); mutex_enter(&kernel_mutex);
lock_rec_reset_and_release_wait(heir); lock_rec_reset_and_release_wait(heir_block, heir_heap_no);
lock_rec_inherit_to_gap(heir, rec); lock_rec_inherit_to_gap(heir_block, block, heir_heap_no, heap_no);
mutex_exit(&kernel_mutex); mutex_exit(&kernel_mutex);
} }
...@@ -3158,14 +3181,20 @@ Updates the lock table when a page is discarded. */ ...@@ -3158,14 +3181,20 @@ Updates the lock table when a page is discarded. */
void void
lock_update_discard( lock_update_discard(
/*================*/ /*================*/
rec_t* heir, /* in: record which will inherit the locks */ const buf_block_t* heir_block, /* in: index page
page_t* page) /* in: index page which will be discarded */ which will inherit the locks */
ulint heir_heap_no, /* in: heap_no of the record
which will inherit the locks */
const buf_block_t* block) /* in: index page
which will be discarded */
{ {
rec_t* rec; const page_t* page = block->frame;
const rec_t* rec;
ulint heap_no;
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
if (NULL == lock_rec_get_first_on_page(page)) { if (!lock_rec_get_first_on_page(block)) {
/* No locks exist on page, nothing to do */ /* No locks exist on page, nothing to do */
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
...@@ -3176,24 +3205,35 @@ lock_update_discard( ...@@ -3176,24 +3205,35 @@ lock_update_discard(
/* Inherit all the locks on the page to the record and reset all /* Inherit all the locks on the page to the record and reset all
the locks on the page */ the locks on the page */
rec = page_get_infimum_rec(page); if (page_is_comp(page)) {
rec = page + PAGE_NEW_INFIMUM;
for (;;) { do {
lock_rec_inherit_to_gap(heir, rec); heap_no = rec_get_heap_no_new(rec);
/* Reset the locks on rec, releasing waiting transactions */ lock_rec_inherit_to_gap(heir_block, block,
heir_heap_no, heap_no);
lock_rec_reset_and_release_wait(rec); lock_rec_reset_and_release_wait(block, heap_no);
if (page_rec_is_supremum(rec)) { rec = page + rec_get_next_offs(rec, TRUE);
} while (heap_no != PAGE_NEW_SUPREMUM);
} else {
rec = page + PAGE_OLD_INFIMUM;
break; do {
} heap_no = rec_get_heap_no_old(rec);
lock_rec_inherit_to_gap(heir_block, block,
heir_heap_no, heap_no);
rec = page_rec_get_next(rec); lock_rec_reset_and_release_wait(block, heap_no);
rec = page + rec_get_next_offs(rec, FALSE);
} while (heap_no != PAGE_OLD_SUPREMUM);
} }
lock_rec_free_all_from_discard_page(page); lock_rec_free_all_from_discard_page(block);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3204,15 +3244,30 @@ Updates the lock table when a new user record is inserted. */ ...@@ -3204,15 +3244,30 @@ Updates the lock table when a new user record is inserted. */
void void
lock_update_insert( lock_update_insert(
/*===============*/ /*===============*/
rec_t* rec) /* in: the inserted record */ const buf_block_t* block, /* in: buffer block containing rec */
const rec_t* rec) /* in: the inserted record */
{ {
lock_mutex_enter_kernel(); ulint receiver_heap_no;
ulint donator_heap_no;
ut_ad(block->frame == page_align((rec_t*) rec));
/* Inherit the gap-locking locks for rec, in gap mode, from the next /* Inherit the gap-locking locks for rec, in gap mode, from the next
record */ record */
lock_rec_inherit_to_gap_if_gap_lock(rec, page_rec_get_next(rec)); if (page_rec_is_comp(rec)) {
receiver_heap_no = rec_get_heap_no_new(rec);
donator_heap_no = rec_get_heap_no_new(
page_rec_get_next_low((rec_t*) rec, TRUE));
} else {
receiver_heap_no = rec_get_heap_no_old(rec);
donator_heap_no = rec_get_heap_no_old(
page_rec_get_next_low((rec_t*) rec, FALSE));
}
lock_mutex_enter_kernel();
lock_rec_inherit_to_gap_if_gap_lock(block,
receiver_heap_no, donator_heap_no);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3222,17 +3277,34 @@ Updates the lock table when a record is removed. */ ...@@ -3222,17 +3277,34 @@ Updates the lock table when a record is removed. */
void void
lock_update_delete( lock_update_delete(
/*===============*/ /*===============*/
rec_t* rec) /* in: the record to be removed */ const buf_block_t* block, /* in: buffer block containing rec */
const rec_t* rec) /* in: the record to be removed */
{ {
const page_t* page = block->frame;
ulint heap_no;
ulint next_heap_no;
if (page_is_comp(page)) {
heap_no = rec_get_heap_no_new(rec);
next_heap_no = rec_get_heap_no_new(page
+ rec_get_next_offs(rec,
TRUE));
} else {
heap_no = rec_get_heap_no_old(rec);
next_heap_no = rec_get_heap_no_old(page
+ rec_get_next_offs(rec,
FALSE));
}
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
/* Let the next record inherit the locks from rec, in gap mode */ /* Let the next record inherit the locks from rec, in gap mode */
lock_rec_inherit_to_gap(page_rec_get_next(rec), rec); lock_rec_inherit_to_gap(block, block, next_heap_no, heap_no);
/* Reset the lock bits on rec and release waiting transactions */ /* Reset the lock bits on rec and release waiting transactions */
lock_rec_reset_and_release_wait(rec); lock_rec_reset_and_release_wait(block, heap_no);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3248,16 +3320,20 @@ actual record is being moved. */ ...@@ -3248,16 +3320,20 @@ actual record is being moved. */
void void
lock_rec_store_on_page_infimum( lock_rec_store_on_page_infimum(
/*===========================*/ /*===========================*/
page_t* page, /* in: page containing the record */ const buf_block_t* block, /* in: buffer block containing rec */
rec_t* rec) /* in: record whose lock state is stored const rec_t* rec) /* in: record whose lock state
on the infimum record of the same page; lock is stored on the infimum
bits are reset on the record */ record of the same page; lock
bits are reset on the
record */
{ {
ut_ad(page == page_align(rec)); ulint heap_no = page_rec_get_heap_no(rec);
ut_ad(block->frame == page_align((rec_t*) rec));
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
lock_rec_move(page_get_infimum_rec(page), rec, page_is_comp(page)); lock_rec_move(block, block, PAGE_HEAP_NO_INFIMUM, heap_no);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3269,17 +3345,23 @@ state was stored on the infimum of the page. */ ...@@ -3269,17 +3345,23 @@ state was stored on the infimum of the page. */
void void
lock_rec_restore_from_page_infimum( lock_rec_restore_from_page_infimum(
/*===============================*/ /*===============================*/
rec_t* rec, /* in: record whose lock state is restored */ const buf_block_t* block, /* in: buffer block containing rec */
page_t* page) /* in: page (rec is not necessarily on this page) const rec_t* rec, /* in: record whose lock state
whose infimum stored the lock state; lock bits are is stored on the infimum
reset on the infimum */ record of the same page; lock
bits are reset on the
record */
const buf_block_t* donator)/* in: page (rec is not
necessarily on this page)
whose infimum stored the lock
state; lock bits are reset on
the infimum */
{ {
ulint comp; ulint heap_no = page_rec_get_heap_no(rec);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
comp = page_is_comp(page);
ut_ad(!comp == !page_rec_is_comp(rec));
lock_rec_move(rec, page_get_infimum_rec(page), comp); lock_rec_move(block, donator, heap_no, PAGE_HEAP_NO_INFIMUM);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
} }
...@@ -3899,26 +3981,24 @@ to a lock. */ ...@@ -3899,26 +3981,24 @@ to a lock. */
void void
lock_rec_unlock( lock_rec_unlock(
/*============*/ /*============*/
trx_t* trx, /* in: transaction that has set a record trx_t* trx, /* in: transaction that has
lock */ set a record lock */
rec_t* rec, /* in: record */ const buf_block_t* block, /* in: buffer block containing rec */
ulint lock_mode) /* in: LOCK_S or LOCK_X */ const rec_t* rec, /* in: record */
ulint lock_mode)/* in: LOCK_S or LOCK_X */
{ {
lock_t* lock; lock_t* lock;
lock_t* release_lock = NULL; lock_t* release_lock = NULL;
ulint heap_no; ulint heap_no;
ut_ad(trx && rec); ut_ad(trx && rec);
ut_ad(block->frame == page_align((rec_t*) rec));
mutex_enter(&kernel_mutex); heap_no = page_rec_get_heap_no(rec);
if (page_rec_is_comp(rec)) { mutex_enter(&kernel_mutex);
heap_no = rec_get_heap_no_new(rec);
} else {
heap_no = rec_get_heap_no_old(rec);
}
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
/* Find the last lock with the same lock_mode and transaction /* Find the last lock with the same lock_mode and transaction
from the record. */ from the record. */
...@@ -3949,7 +4029,7 @@ lock_rec_unlock( ...@@ -3949,7 +4029,7 @@ lock_rec_unlock(
/* Check if we can now grant waiting lock requests */ /* Check if we can now grant waiting lock requests */
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
while (lock != NULL) { while (lock != NULL) {
if (lock_get_wait(lock) if (lock_get_wait(lock)
...@@ -4565,9 +4645,10 @@ loop: ...@@ -4565,9 +4645,10 @@ loop:
goto loop; goto loop;
} }
# ifdef UNIV_DEBUG
/************************************************************************* /*************************************************************************
Validates the lock queue on a table. */ Validates the lock queue on a table. */
static
ibool ibool
lock_table_queue_validate( lock_table_queue_validate(
/*======================*/ /*======================*/
...@@ -4611,34 +4692,32 @@ lock_table_queue_validate( ...@@ -4611,34 +4692,32 @@ lock_table_queue_validate(
/************************************************************************* /*************************************************************************
Validates the lock queue on a single record. */ Validates the lock queue on a single record. */
static
ibool ibool
lock_rec_queue_validate( lock_rec_queue_validate(
/*====================*/ /*====================*/
/* out: TRUE if ok */ /* out: TRUE if ok */
rec_t* rec, /* in: record to look at */ const buf_block_t* block, /* in: buffer block containing rec */
dict_index_t* index, /* in: index, or NULL if not known */ const rec_t* rec, /* in: record to look at */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */ dict_index_t* index, /* in: index, or NULL if not known */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */
{ {
trx_t* impl_trx; trx_t* impl_trx;
lock_t* lock; lock_t* lock;
ulint heap_no; ulint heap_no;
ut_a(rec); ut_a(rec);
ut_a(block->frame == page_align((rec_t*) rec));
ut_ad(rec_offs_validate(rec, index, offsets)); ut_ad(rec_offs_validate(rec, index, offsets));
ut_ad(!page_rec_is_comp(rec) == !rec_offs_comp(offsets)); ut_ad(!page_rec_is_comp(rec) == !rec_offs_comp(offsets));
lock_mutex_enter_kernel(); heap_no = page_rec_get_heap_no(rec);
if (page_rec_is_comp(rec)) { lock_mutex_enter_kernel();
heap_no = rec_get_heap_no_new(rec);
} else {
heap_no = rec_get_heap_no_old(rec);
}
if (!page_rec_is_user_rec(rec)) { if (!page_rec_is_user_rec(rec)) {
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
while (lock) { while (lock) {
switch(lock->trx->conc_state) { switch(lock->trx->conc_state) {
...@@ -4673,11 +4752,12 @@ lock_rec_queue_validate( ...@@ -4673,11 +4752,12 @@ lock_rec_queue_validate(
impl_trx = lock_clust_rec_some_has_impl(rec, index, offsets); impl_trx = lock_clust_rec_some_has_impl(rec, index, offsets);
if (impl_trx && lock_rec_other_has_expl_req( if (impl_trx
LOCK_S, 0, LOCK_WAIT, rec, heap_no, impl_trx)) { && lock_rec_other_has_expl_req(LOCK_S, 0, LOCK_WAIT,
block, heap_no, impl_trx)) {
ut_a(lock_rec_has_expl(LOCK_X | LOCK_REC_NOT_GAP, ut_a(lock_rec_has_expl(LOCK_X | LOCK_REC_NOT_GAP,
rec, heap_no, impl_trx)); block, heap_no, impl_trx));
} }
} else { } else {
...@@ -4688,15 +4768,16 @@ lock_rec_queue_validate( ...@@ -4688,15 +4768,16 @@ lock_rec_queue_validate(
impl_trx = lock_sec_rec_some_has_impl_off_kernel( impl_trx = lock_sec_rec_some_has_impl_off_kernel(
rec, index, offsets); rec, index, offsets);
if (impl_trx && lock_rec_other_has_expl_req( if (impl_trx
LOCK_S, 0, LOCK_WAIT, rec, heap_no, impl_trx)) { && lock_rec_other_has_expl_req(LOCK_S, 0, LOCK_WAIT,
block, heap_no, impl_trx)) {
ut_a(lock_rec_has_expl(LOCK_X | LOCK_REC_NOT_GAP, ut_a(lock_rec_has_expl(LOCK_X | LOCK_REC_NOT_GAP,
rec, heap_no, impl_trx)); block, heap_no, impl_trx));
} }
} }
lock = lock_rec_get_first(rec, heap_no); lock = lock_rec_get_first(block, heap_no);
while (lock) { while (lock) {
ut_a(lock->trx->conc_state == TRX_ACTIVE ut_a(lock->trx->conc_state == TRX_ACTIVE
...@@ -4718,7 +4799,7 @@ lock_rec_queue_validate( ...@@ -4718,7 +4799,7 @@ lock_rec_queue_validate(
mode = LOCK_S; mode = LOCK_S;
} }
ut_a(!lock_rec_other_has_expl_req( ut_a(!lock_rec_other_has_expl_req(
mode, 0, 0, rec, heap_no, lock->trx)); mode, 0, 0, block, heap_no, lock->trx));
} else if (lock_get_wait(lock) && !lock_rec_get_gap(lock)) { } else if (lock_get_wait(lock) && !lock_rec_get_gap(lock)) {
...@@ -4735,7 +4816,7 @@ lock_rec_queue_validate( ...@@ -4735,7 +4816,7 @@ lock_rec_queue_validate(
/************************************************************************* /*************************************************************************
Validates the record lock queues on a page. */ Validates the record lock queues on a page. */
static
ibool ibool
lock_rec_validate_page( lock_rec_validate_page(
/*===================*/ /*===================*/
...@@ -4745,9 +4826,9 @@ lock_rec_validate_page( ...@@ -4745,9 +4826,9 @@ lock_rec_validate_page(
{ {
dict_index_t* index; dict_index_t* index;
buf_block_t* block; buf_block_t* block;
page_t* page; const page_t* page;
lock_t* lock; lock_t* lock;
rec_t* rec; const rec_t* rec;
ulint nth_lock = 0; ulint nth_lock = 0;
ulint nth_bit = 0; ulint nth_bit = 0;
ulint i; ulint i;
...@@ -4767,7 +4848,7 @@ lock_rec_validate_page( ...@@ -4767,7 +4848,7 @@ lock_rec_validate_page(
#ifdef UNIV_SYNC_DEBUG #ifdef UNIV_SYNC_DEBUG
buf_block_dbg_add_level(block, SYNC_NO_ORDER_CHECK); buf_block_dbg_add_level(block, SYNC_NO_ORDER_CHECK);
#endif /* UNIV_SYNC_DEBUG */ #endif /* UNIV_SYNC_DEBUG */
page = buf_block_get_frame(block); page = block->frame;
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
loop: loop:
...@@ -4806,7 +4887,7 @@ loop: ...@@ -4806,7 +4887,7 @@ loop:
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
lock_rec_queue_validate(rec, index, offsets); lock_rec_queue_validate(block, rec, index, offsets);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
...@@ -4834,7 +4915,7 @@ function_exit: ...@@ -4834,7 +4915,7 @@ function_exit:
/************************************************************************* /*************************************************************************
Validates the lock system. */ Validates the lock system. */
static
ibool ibool
lock_validate(void) lock_validate(void)
/*===============*/ /*===============*/
...@@ -4908,6 +4989,7 @@ lock_validate(void) ...@@ -4908,6 +4989,7 @@ lock_validate(void)
return(TRUE); return(TRUE);
} }
# endif /* UNIV_DEBUG */
#endif /* !UNIV_HOTBACKUP */ #endif /* !UNIV_HOTBACKUP */
/*============ RECORD LOCK CHECKS FOR ROW OPERATIONS ====================*/ /*============ RECORD LOCK CHECKS FOR ROW OPERATIONS ====================*/
...@@ -4923,21 +5005,22 @@ lock_rec_insert_check_and_lock( ...@@ -4923,21 +5005,22 @@ lock_rec_insert_check_and_lock(
/*===========================*/ /*===========================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS, DB_LOCK_WAIT,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is
does nothing */ set, does nothing */
rec_t* rec, /* in: record after which to insert */ rec_t* rec, /* in: record after which to insert */
buf_block_t* block, /* in: buffer block of rec */ buf_block_t* block, /* in/out: buffer block of rec */
dict_index_t* index, /* in: index */ dict_index_t* index, /* in: index */
que_thr_t* thr, /* in: query thread */ que_thr_t* thr, /* in: query thread */
ibool* inherit)/* out: set to TRUE if the new inserted ibool* inherit)/* out: set to TRUE if the new
record maybe should inherit LOCK_GAP type inserted record maybe should inherit
locks from the successor record */ LOCK_GAP type locks from the successor
record */
{ {
rec_t* next_rec; const rec_t* next_rec;
trx_t* trx; trx_t* trx;
lock_t* lock; lock_t* lock;
ulint err; ulint err;
ulint next_rec_heap_no; ulint next_rec_heap_no;
if (flags & BTR_NO_LOCKING_FLAG) { if (flags & BTR_NO_LOCKING_FLAG) {
...@@ -4953,13 +5036,9 @@ lock_rec_insert_check_and_lock( ...@@ -4953,13 +5036,9 @@ lock_rec_insert_check_and_lock(
ut_ad(lock_table_has(thr_get_trx(thr), index->table, LOCK_IX)); ut_ad(lock_table_has(thr_get_trx(thr), index->table, LOCK_IX));
if (page_rec_is_comp(next_rec)) { next_rec_heap_no = page_rec_get_heap_no(next_rec);
next_rec_heap_no = rec_get_heap_no_new(next_rec);
} else {
next_rec_heap_no = rec_get_heap_no_old(next_rec);
}
lock = lock_rec_get_first(next_rec, next_rec_heap_no); lock = lock_rec_get_first(block, next_rec_heap_no);
if (UNIV_LIKELY(lock == NULL)) { if (UNIV_LIKELY(lock == NULL)) {
/* We optimize CPU time usage in the simplest case */ /* We optimize CPU time usage in the simplest case */
...@@ -4992,12 +5071,13 @@ lock_rec_insert_check_and_lock( ...@@ -4992,12 +5071,13 @@ lock_rec_insert_check_and_lock(
if (lock_rec_other_has_conflicting( if (lock_rec_other_has_conflicting(
LOCK_X | LOCK_GAP | LOCK_INSERT_INTENTION, LOCK_X | LOCK_GAP | LOCK_INSERT_INTENTION,
next_rec, next_rec_heap_no, trx)) { block, next_rec_heap_no, trx)) {
/* Note that we may get DB_SUCCESS also here! */ /* Note that we may get DB_SUCCESS also here! */
err = lock_rec_enqueue_waiting(LOCK_X | LOCK_GAP err = lock_rec_enqueue_waiting(LOCK_X | LOCK_GAP
| LOCK_INSERT_INTENTION, | LOCK_INSERT_INTENTION,
next_rec, index, thr); block, next_rec_heap_no,
index, thr);
} else { } else {
err = DB_SUCCESS; err = DB_SUCCESS;
} }
...@@ -5020,7 +5100,8 @@ lock_rec_insert_check_and_lock( ...@@ -5020,7 +5100,8 @@ lock_rec_insert_check_and_lock(
offsets = rec_get_offsets(next_rec, index, offsets_, offsets = rec_get_offsets(next_rec, index, offsets_,
ULINT_UNDEFINED, &heap); ULINT_UNDEFINED, &heap);
ut_ad(lock_rec_queue_validate(next_rec, index, offsets)); ut_ad(lock_rec_queue_validate(block,
next_rec, index, offsets));
if (UNIV_LIKELY_NULL(heap)) { if (UNIV_LIKELY_NULL(heap)) {
mem_heap_free(heap); mem_heap_free(heap);
} }
...@@ -5038,9 +5119,10 @@ static ...@@ -5038,9 +5119,10 @@ static
void void
lock_rec_convert_impl_to_expl( lock_rec_convert_impl_to_expl(
/*==========================*/ /*==========================*/
rec_t* rec, /* in: user record on page */ const buf_block_t* block, /* in: buffer block of rec */
dict_index_t* index, /* in: index of record */ const rec_t* rec, /* in: user record on page */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */ dict_index_t* index, /* in: index of record */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */
{ {
trx_t* impl_trx; trx_t* impl_trx;
...@@ -5059,23 +5141,17 @@ lock_rec_convert_impl_to_expl( ...@@ -5059,23 +5141,17 @@ lock_rec_convert_impl_to_expl(
} }
if (impl_trx) { if (impl_trx) {
ulint heap_no = page_rec_get_heap_no(rec);
/* If the transaction has no explicit x-lock set on the /* If the transaction has no explicit x-lock set on the
record, set one for it */ record, set one for it */
ulint heap_no; if (!lock_rec_has_expl(LOCK_X | LOCK_REC_NOT_GAP, block,
if (page_rec_is_comp(rec)) {
heap_no = rec_get_heap_no_new(rec);
} else {
heap_no = rec_get_heap_no_old(rec);
}
if (!lock_rec_has_expl(LOCK_X | LOCK_REC_NOT_GAP, rec,
heap_no, impl_trx)) { heap_no, impl_trx)) {
lock_rec_add_to_queue( lock_rec_add_to_queue(
LOCK_REC | LOCK_X | LOCK_REC_NOT_GAP, LOCK_REC | LOCK_X | LOCK_REC_NOT_GAP,
rec, heap_no, index, impl_trx); block, heap_no, index, impl_trx);
} }
} }
} }
...@@ -5091,16 +5167,20 @@ lock queue. */ ...@@ -5091,16 +5167,20 @@ lock queue. */
ulint ulint
lock_clust_rec_modify_check_and_lock( lock_clust_rec_modify_check_and_lock(
/*=================================*/ /*=================================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: record which should be modified */ bit is set, does nothing */
dict_index_t* index, /* in: clustered index */ const buf_block_t* block, /* in: buffer block of rec */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ const rec_t* rec, /* in: record which should be
que_thr_t* thr) /* in: query thread */ modified */
dict_index_t* index, /* in: clustered index */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */
que_thr_t* thr) /* in: query thread */
{ {
ulint err; ulint err;
ulint heap_no;
ut_ad(rec_offs_validate(rec, index, offsets)); ut_ad(rec_offs_validate(rec, index, offsets));
ut_ad(dict_index_is_clust(index)); ut_ad(dict_index_is_clust(index));
...@@ -5110,6 +5190,10 @@ lock_clust_rec_modify_check_and_lock( ...@@ -5110,6 +5190,10 @@ lock_clust_rec_modify_check_and_lock(
return(DB_SUCCESS); return(DB_SUCCESS);
} }
heap_no = rec_offs_comp(offsets)
? rec_get_heap_no_new(rec)
: rec_get_heap_no_old(rec);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
ut_ad(lock_table_has(thr_get_trx(thr), index->table, LOCK_IX)); ut_ad(lock_table_has(thr_get_trx(thr), index->table, LOCK_IX));
...@@ -5117,13 +5201,14 @@ lock_clust_rec_modify_check_and_lock( ...@@ -5117,13 +5201,14 @@ lock_clust_rec_modify_check_and_lock(
/* If a transaction has no explicit x-lock set on the record, set one /* If a transaction has no explicit x-lock set on the record, set one
for it */ for it */
lock_rec_convert_impl_to_expl(rec, index, offsets); lock_rec_convert_impl_to_expl(block, rec, index, offsets);
err = lock_rec_lock(TRUE, LOCK_X | LOCK_REC_NOT_GAP, rec, index, thr); err = lock_rec_lock(TRUE, LOCK_X | LOCK_REC_NOT_GAP,
block, heap_no, index, thr);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
ut_ad(lock_rec_queue_validate(rec, index, offsets)); ut_ad(lock_rec_queue_validate(block, rec, index, offsets));
return(err); return(err);
} }
...@@ -5137,17 +5222,19 @@ lock_sec_rec_modify_check_and_lock( ...@@ -5137,17 +5222,19 @@ lock_sec_rec_modify_check_and_lock(
/*===============================*/ /*===============================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS, DB_LOCK_WAIT,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, ulint flags, /* in: if BTR_NO_LOCKING_FLAG
does nothing */ bit is set, does nothing */
rec_t* rec, /* in: record which should be modified; buf_block_t* block, /* in/out: buffer block of rec */
NOTE: as this is a secondary index, we rec_t* rec, /* in: record which should be
always have to modify the clustered index modified; NOTE: as this is a secondary
record first: see the comment below */ index, we always have to modify the
buf_block_t* block, /* in: buffer block of rec */ clustered index record first: see the
comment below */
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 */
{ {
ulint err; ulint err;
ulint heap_no;
if (flags & BTR_NO_LOCKING_FLAG) { if (flags & BTR_NO_LOCKING_FLAG) {
...@@ -5156,6 +5243,8 @@ lock_sec_rec_modify_check_and_lock( ...@@ -5156,6 +5243,8 @@ lock_sec_rec_modify_check_and_lock(
ut_ad(!dict_index_is_clust(index)); ut_ad(!dict_index_is_clust(index));
heap_no = page_rec_get_heap_no(rec);
/* 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
index record, and this would not have been possible if another active index record, and this would not have been possible if another active
...@@ -5165,7 +5254,8 @@ lock_sec_rec_modify_check_and_lock( ...@@ -5165,7 +5254,8 @@ lock_sec_rec_modify_check_and_lock(
ut_ad(lock_table_has(thr_get_trx(thr), index->table, LOCK_IX)); ut_ad(lock_table_has(thr_get_trx(thr), index->table, LOCK_IX));
err = lock_rec_lock(TRUE, LOCK_X | LOCK_REC_NOT_GAP, rec, index, thr); err = lock_rec_lock(TRUE, LOCK_X | LOCK_REC_NOT_GAP,
block, heap_no, index, thr);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
...@@ -5178,7 +5268,7 @@ lock_sec_rec_modify_check_and_lock( ...@@ -5178,7 +5268,7 @@ lock_sec_rec_modify_check_and_lock(
offsets = rec_get_offsets(rec, index, offsets_, offsets = rec_get_offsets(rec, index, offsets_,
ULINT_UNDEFINED, &heap); ULINT_UNDEFINED, &heap);
ut_ad(lock_rec_queue_validate(rec, index, offsets)); ut_ad(lock_rec_queue_validate(block, rec, index, offsets));
if (UNIV_LIKELY_NULL(heap)) { if (UNIV_LIKELY_NULL(heap)) {
mem_heap_free(heap); mem_heap_free(heap);
} }
...@@ -5202,23 +5292,29 @@ secondary index record. */ ...@@ -5202,23 +5292,29 @@ secondary index record. */
ulint ulint
lock_sec_rec_read_check_and_lock( lock_sec_rec_read_check_and_lock(
/*=============================*/ /*=============================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: user record or page supremum record bit is set, does nothing */
which should be read or passed over by a read const buf_block_t* block, /* in: buffer block of rec */
cursor */ const rec_t* rec, /* in: user recpord or page
dict_index_t* index, /* in: secondary index */ supremum record which should
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ be read or passed over by a
ulint mode, /* in: mode of the lock which the read cursor read cursor */
should set on records: LOCK_S or LOCK_X; the dict_index_t* index, /* in: secondary index */
latter is possible in SELECT FOR UPDATE */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: mode of the lock which
LOCK_REC_NOT_GAP */ the read cursor should set on
que_thr_t* thr) /* in: query thread */ records: LOCK_S or LOCK_X; the
latter is possible in
SELECT FOR UPDATE */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP */
que_thr_t* thr) /* in: query thread */
{ {
ulint err; ulint err;
ulint heap_no;
ut_ad(!dict_index_is_clust(index)); ut_ad(!dict_index_is_clust(index));
ut_ad(page_rec_is_user_rec(rec) || page_rec_is_supremum(rec)); ut_ad(page_rec_is_user_rec(rec) || page_rec_is_supremum(rec));
...@@ -5229,6 +5325,8 @@ lock_sec_rec_read_check_and_lock( ...@@ -5229,6 +5325,8 @@ lock_sec_rec_read_check_and_lock(
return(DB_SUCCESS); return(DB_SUCCESS);
} }
heap_no = page_rec_get_heap_no(rec);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
ut_ad(mode != LOCK_X ut_ad(mode != LOCK_X
...@@ -5240,19 +5338,20 @@ lock_sec_rec_read_check_and_lock( ...@@ -5240,19 +5338,20 @@ lock_sec_rec_read_check_and_lock(
if the max trx id for the page >= min trx id for the trx list or a if the max trx id for the page >= min trx id for the trx list or a
database recovery is running. */ database recovery is running. */
if (((ut_dulint_cmp(page_get_max_trx_id(page_align(rec)), if (((ut_dulint_cmp(page_get_max_trx_id(block->frame),
trx_list_get_min_trx_id()) >= 0) trx_list_get_min_trx_id()) >= 0)
|| recv_recovery_is_on()) || recv_recovery_is_on())
&& !page_rec_is_supremum(rec)) { && !page_rec_is_supremum(rec)) {
lock_rec_convert_impl_to_expl(rec, index, offsets); lock_rec_convert_impl_to_expl(block, rec, index, offsets);
} }
err = lock_rec_lock(FALSE, mode | gap_mode, rec, index, thr); err = lock_rec_lock(FALSE, mode | gap_mode,
block, heap_no, index, thr);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
ut_ad(lock_rec_queue_validate(rec, index, offsets)); ut_ad(lock_rec_queue_validate(block, rec, index, offsets));
return(err); return(err);
} }
...@@ -5268,23 +5367,29 @@ lock on the record. */ ...@@ -5268,23 +5367,29 @@ lock on the record. */
ulint ulint
lock_clust_rec_read_check_and_lock( lock_clust_rec_read_check_and_lock(
/*===============================*/ /*===============================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: user record or page supremum record bit is set, does nothing */
which should be read or passed over by a read const buf_block_t* block, /* in: buffer block of rec */
cursor */ const rec_t* rec, /* in: user record or page
dict_index_t* index, /* in: clustered index */ supremum record which should
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ be read or passed over by a
ulint mode, /* in: mode of the lock which the read cursor read cursor */
should set on records: LOCK_S or LOCK_X; the dict_index_t* index, /* in: clustered index */
latter is possible in SELECT FOR UPDATE */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: mode of the lock which
LOCK_REC_NOT_GAP */ the read cursor should set on
que_thr_t* thr) /* in: query thread */ records: LOCK_S or LOCK_X; the
latter is possible in
SELECT FOR UPDATE */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP */
que_thr_t* thr) /* in: query thread */
{ {
ulint err; ulint err;
ulint heap_no;
ut_ad(dict_index_is_clust(index)); ut_ad(dict_index_is_clust(index));
ut_ad(page_rec_is_user_rec(rec) || page_rec_is_supremum(rec)); ut_ad(page_rec_is_user_rec(rec) || page_rec_is_supremum(rec));
...@@ -5297,6 +5402,8 @@ lock_clust_rec_read_check_and_lock( ...@@ -5297,6 +5402,8 @@ lock_clust_rec_read_check_and_lock(
return(DB_SUCCESS); return(DB_SUCCESS);
} }
heap_no = page_rec_get_heap_no(rec);
lock_mutex_enter_kernel(); lock_mutex_enter_kernel();
ut_ad(mode != LOCK_X ut_ad(mode != LOCK_X
...@@ -5304,16 +5411,17 @@ lock_clust_rec_read_check_and_lock( ...@@ -5304,16 +5411,17 @@ lock_clust_rec_read_check_and_lock(
ut_ad(mode != LOCK_S ut_ad(mode != LOCK_S
|| lock_table_has(thr_get_trx(thr), index->table, LOCK_IS)); || lock_table_has(thr_get_trx(thr), index->table, LOCK_IS));
if (!page_rec_is_supremum(rec)) { if (UNIV_LIKELY(heap_no != PAGE_HEAP_NO_SUPREMUM)) {
lock_rec_convert_impl_to_expl(rec, index, offsets); lock_rec_convert_impl_to_expl(block, rec, index, offsets);
} }
err = lock_rec_lock(FALSE, mode | gap_mode, rec, index, thr); err = lock_rec_lock(FALSE, mode | gap_mode,
block, heap_no, index, thr);
lock_mutex_exit_kernel(); lock_mutex_exit_kernel();
ut_ad(lock_rec_queue_validate(rec, index, offsets)); ut_ad(lock_rec_queue_validate(block, rec, index, offsets));
return(err); return(err);
} }
...@@ -5330,20 +5438,25 @@ lock_clust_rec_read_check_and_lock() that does not require the parameter ...@@ -5330,20 +5438,25 @@ lock_clust_rec_read_check_and_lock() that does not require the parameter
ulint ulint
lock_clust_rec_read_check_and_lock_alt( lock_clust_rec_read_check_and_lock_alt(
/*===================================*/ /*===================================*/
/* out: DB_SUCCESS, DB_LOCK_WAIT, /* out: DB_SUCCESS,
DB_DEADLOCK, or DB_QUE_THR_SUSPENDED */ DB_LOCK_WAIT, DB_DEADLOCK, or
ulint flags, /* in: if BTR_NO_LOCKING_FLAG bit is set, DB_QUE_THR_SUSPENDED */
does nothing */ ulint flags, /* in: if BTR_NO_LOCKING_FLAG
rec_t* rec, /* in: user record or page supremum record bit is set, does nothing */
which should be read or passed over by a read const buf_block_t* block, /* in: buffer block of rec */
cursor */ const rec_t* rec, /* in: user record or page
dict_index_t* index, /* in: clustered index */ supremum record which should
ulint mode, /* in: mode of the lock which the read cursor be read or passed over by a
should set on records: LOCK_S or LOCK_X; the read cursor */
latter is possible in SELECT FOR UPDATE */ dict_index_t* index, /* in: clustered index */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: mode of the lock which
LOCK_REC_NOT_GAP */ the read cursor should set on
que_thr_t* thr) /* in: query thread */ records: LOCK_S or LOCK_X; the
latter is possible in
SELECT FOR UPDATE */
ulint gap_mode,/* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP */
que_thr_t* thr) /* in: query thread */
{ {
mem_heap_t* tmp_heap = NULL; mem_heap_t* tmp_heap = NULL;
ulint offsets_[REC_OFFS_NORMAL_SIZE]; ulint offsets_[REC_OFFS_NORMAL_SIZE];
...@@ -5353,7 +5466,7 @@ lock_clust_rec_read_check_and_lock_alt( ...@@ -5353,7 +5466,7 @@ lock_clust_rec_read_check_and_lock_alt(
offsets = rec_get_offsets(rec, index, offsets, offsets = rec_get_offsets(rec, index, offsets,
ULINT_UNDEFINED, &tmp_heap); ULINT_UNDEFINED, &tmp_heap);
ret = lock_clust_rec_read_check_and_lock(flags, rec, index, ret = lock_clust_rec_read_check_and_lock(flags, block, rec, index,
offsets, mode, gap_mode, thr); offsets, mode, gap_mode, thr);
if (tmp_heap) { if (tmp_heap) {
mem_heap_free(tmp_heap); mem_heap_free(tmp_heap);
......
...@@ -645,7 +645,7 @@ page_copy_rec_list_end( ...@@ -645,7 +645,7 @@ page_copy_rec_list_end(
/* Update the lock table, MAX_TRX_ID, and possible hash index */ /* Update the lock table, MAX_TRX_ID, and possible hash index */
lock_move_rec_list_end(new_page, page, rec); lock_move_rec_list_end(new_block, block, rec);
page_update_max_trx_id(new_block, new_page_zip, page_update_max_trx_id(new_block, new_page_zip,
page_get_max_trx_id(page)); page_get_max_trx_id(page));
...@@ -753,7 +753,7 @@ page_copy_rec_list_start( ...@@ -753,7 +753,7 @@ page_copy_rec_list_start(
page_update_max_trx_id(new_block, new_page_zip, page_update_max_trx_id(new_block, new_page_zip,
page_get_max_trx_id(page_align(rec))); page_get_max_trx_id(page_align(rec)));
lock_move_rec_list_start(new_page, page_align(rec), rec, ret); lock_move_rec_list_start(new_block, block, rec, ret);
btr_search_move_or_delete_hash_entries(new_block, block, index); btr_search_move_or_delete_hash_entries(new_block, block, index);
......
...@@ -3558,7 +3558,7 @@ page_zip_reorganize( ...@@ -3558,7 +3558,7 @@ page_zip_reorganize(
return(FALSE); return(FALSE);
} }
lock_move_reorganize_page(page, temp_page); lock_move_reorganize_page(block, temp_block);
btr_search_drop_page_hash_index(block); btr_search_drop_page_hash_index(block);
buf_block_free(temp_block); buf_block_free(temp_block);
......
...@@ -969,7 +969,8 @@ row_ins_foreign_check_on_constraint( ...@@ -969,7 +969,8 @@ row_ins_foreign_check_on_constraint(
gap if the search criterion was not unique */ gap if the search criterion was not unique */
err = lock_clust_rec_read_check_and_lock_alt( err = lock_clust_rec_read_check_and_lock_alt(
0, clust_rec, clust_index, LOCK_X, LOCK_REC_NOT_GAP, 0, btr_pcur_get_block(pcur),
clust_rec, clust_index, LOCK_X, LOCK_REC_NOT_GAP,
thr); thr);
} }
...@@ -1128,13 +1129,14 @@ static ...@@ -1128,13 +1129,14 @@ static
ulint ulint
row_ins_set_shared_rec_lock( row_ins_set_shared_rec_lock(
/*========================*/ /*========================*/
/* out: DB_SUCCESS or error code */ /* out: DB_SUCCESS or error code */
ulint type, /* in: LOCK_ORDINARY, LOCK_GAP, or ulint type, /* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP type lock */ LOCK_REC_NOT_GAP type lock */
rec_t* rec, /* in: record */ const buf_block_t* block, /* in: buffer block of rec */
dict_index_t* index, /* in: index */ const rec_t* rec, /* in: record */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ dict_index_t* index, /* in: index */
que_thr_t* thr) /* in: query thread */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
que_thr_t* thr) /* in: query thread */
{ {
ulint err; ulint err;
...@@ -1142,10 +1144,10 @@ row_ins_set_shared_rec_lock( ...@@ -1142,10 +1144,10 @@ row_ins_set_shared_rec_lock(
if (dict_index_is_clust(index)) { if (dict_index_is_clust(index)) {
err = lock_clust_rec_read_check_and_lock( err = lock_clust_rec_read_check_and_lock(
0, rec, index, offsets, LOCK_S, type, thr); 0, block, rec, index, offsets, LOCK_S, type, thr);
} else { } else {
err = lock_sec_rec_read_check_and_lock( err = lock_sec_rec_read_check_and_lock(
0, rec, index, offsets, LOCK_S, type, thr); 0, block, rec, index, offsets, LOCK_S, type, thr);
} }
return(err); return(err);
...@@ -1159,13 +1161,14 @@ static ...@@ -1159,13 +1161,14 @@ static
ulint ulint
row_ins_set_exclusive_rec_lock( row_ins_set_exclusive_rec_lock(
/*===========================*/ /*===========================*/
/* out: DB_SUCCESS or error code */ /* out: DB_SUCCESS or error code */
ulint type, /* in: LOCK_ORDINARY, LOCK_GAP, or ulint type, /* in: LOCK_ORDINARY, LOCK_GAP, or
LOCK_REC_NOT_GAP type lock */ LOCK_REC_NOT_GAP type lock */
rec_t* rec, /* in: record */ const buf_block_t* block, /* in: buffer block of rec */
dict_index_t* index, /* in: index */ const rec_t* rec, /* in: record */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ dict_index_t* index, /* in: index */
que_thr_t* thr) /* in: query thread */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
que_thr_t* thr) /* in: query thread */
{ {
ulint err; ulint err;
...@@ -1173,10 +1176,10 @@ row_ins_set_exclusive_rec_lock( ...@@ -1173,10 +1176,10 @@ row_ins_set_exclusive_rec_lock(
if (dict_index_is_clust(index)) { if (dict_index_is_clust(index)) {
err = lock_clust_rec_read_check_and_lock( err = lock_clust_rec_read_check_and_lock(
0, rec, index, offsets, LOCK_X, type, thr); 0, block, rec, index, offsets, LOCK_X, type, thr);
} else { } else {
err = lock_sec_rec_read_check_and_lock( err = lock_sec_rec_read_check_and_lock(
0, rec, index, offsets, LOCK_X, type, thr); 0, block, rec, index, offsets, LOCK_X, type, thr);
} }
return(err); return(err);
...@@ -1209,6 +1212,7 @@ row_ins_check_foreign_constraint( ...@@ -1209,6 +1212,7 @@ row_ins_check_foreign_constraint(
dict_table_t* check_table; dict_table_t* check_table;
dict_index_t* check_index; dict_index_t* check_index;
ulint n_fields_cmp; ulint n_fields_cmp;
buf_block_t* block;
rec_t* rec; rec_t* rec;
btr_pcur_t pcur; btr_pcur_t pcur;
ibool moved; ibool moved;
...@@ -1338,6 +1342,7 @@ run_again: ...@@ -1338,6 +1342,7 @@ run_again:
btr_pcur_open(check_index, entry, PAGE_CUR_GE, btr_pcur_open(check_index, entry, PAGE_CUR_GE,
BTR_SEARCH_LEAF, &pcur, &mtr); BTR_SEARCH_LEAF, &pcur, &mtr);
block = btr_pcur_get_block(&pcur);
/* Scan index records and check if there is a matching record */ /* Scan index records and check if there is a matching record */
...@@ -1354,8 +1359,9 @@ run_again: ...@@ -1354,8 +1359,9 @@ run_again:
if (page_rec_is_supremum(rec)) { if (page_rec_is_supremum(rec)) {
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(LOCK_ORDINARY, block,
LOCK_ORDINARY, rec, check_index, offsets, thr); rec, check_index,
offsets, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
break; break;
...@@ -1370,8 +1376,8 @@ run_again: ...@@ -1370,8 +1376,8 @@ run_again:
if (rec_get_deleted_flag(rec, if (rec_get_deleted_flag(rec,
rec_offs_comp(offsets))) { rec_offs_comp(offsets))) {
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(
LOCK_ORDINARY, rec, check_index, LOCK_ORDINARY, block,
offsets, thr); rec, check_index, offsets, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
break; break;
...@@ -1382,8 +1388,8 @@ run_again: ...@@ -1382,8 +1388,8 @@ run_again:
into gaps */ into gaps */
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(
LOCK_REC_NOT_GAP, rec, check_index, LOCK_REC_NOT_GAP, block,
offsets, thr); rec, check_index, offsets, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
...@@ -1434,7 +1440,8 @@ run_again: ...@@ -1434,7 +1440,8 @@ run_again:
if (cmp < 0) { if (cmp < 0) {
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(
LOCK_GAP, rec, check_index, offsets, thr); LOCK_GAP, block,
rec, check_index, offsets, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
break; break;
...@@ -1589,10 +1596,10 @@ ibool ...@@ -1589,10 +1596,10 @@ ibool
row_ins_dupl_error_with_rec( row_ins_dupl_error_with_rec(
/*========================*/ /*========================*/
/* out: TRUE if error */ /* out: TRUE if error */
rec_t* rec, /* in: user record; NOTE that we assume const rec_t* rec, /* in: user record; NOTE that we assume
that the caller already has a record lock on that the caller already has a record lock on
the record! */ the record! */
dtuple_t* entry, /* in: entry to insert */ const dtuple_t* entry, /* in: entry to insert */
dict_index_t* index, /* in: index */ dict_index_t* index, /* in: index */
const ulint* offsets)/* in: rec_get_offsets(rec, index) */ const ulint* offsets)/* in: rec_get_offsets(rec, index) */
{ {
...@@ -1653,7 +1660,8 @@ row_ins_scan_sec_index_for_duplicate( ...@@ -1653,7 +1660,8 @@ row_ins_scan_sec_index_for_duplicate(
ulint i; ulint i;
int cmp; int cmp;
ulint n_fields_cmp; ulint n_fields_cmp;
rec_t* rec; buf_block_t* block;
const rec_t* rec;
btr_pcur_t pcur; btr_pcur_t pcur;
ulint err = DB_SUCCESS; ulint err = DB_SUCCESS;
ibool moved; ibool moved;
...@@ -1687,6 +1695,8 @@ row_ins_scan_sec_index_for_duplicate( ...@@ -1687,6 +1695,8 @@ row_ins_scan_sec_index_for_duplicate(
btr_pcur_open(index, entry, PAGE_CUR_GE, BTR_SEARCH_LEAF, &pcur, &mtr); btr_pcur_open(index, entry, PAGE_CUR_GE, BTR_SEARCH_LEAF, &pcur, &mtr);
block = btr_pcur_get_block(&pcur);
/* Scan index records and check if there is a duplicate */ /* Scan index records and check if there is a duplicate */
for (;;) { for (;;) {
...@@ -1708,11 +1718,13 @@ row_ins_scan_sec_index_for_duplicate( ...@@ -1708,11 +1718,13 @@ row_ins_scan_sec_index_for_duplicate(
INSERT ON DUPLICATE KEY UPDATE). */ INSERT ON DUPLICATE KEY UPDATE). */
err = row_ins_set_exclusive_rec_lock( err = row_ins_set_exclusive_rec_lock(
LOCK_ORDINARY, rec, index, offsets, thr); LOCK_ORDINARY, block,
rec, index, offsets, thr);
} else { } else {
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(
LOCK_ORDINARY, rec, index, offsets, thr); LOCK_ORDINARY, block,
rec, index, offsets, thr);
} }
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
...@@ -1837,12 +1849,14 @@ row_ins_duplicate_error_in_clust( ...@@ -1837,12 +1849,14 @@ row_ins_duplicate_error_in_clust(
INSERT ON DUPLICATE KEY UPDATE). */ INSERT ON DUPLICATE KEY UPDATE). */
err = row_ins_set_exclusive_rec_lock( err = row_ins_set_exclusive_rec_lock(
LOCK_REC_NOT_GAP, rec, LOCK_REC_NOT_GAP,
cursor->index, offsets, thr); btr_cur_get_block(cursor),
rec, cursor->index, offsets, thr);
} else { } else {
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(
LOCK_REC_NOT_GAP, rec, LOCK_REC_NOT_GAP,
btr_cur_get_block(cursor), rec,
cursor->index, offsets, thr); cursor->index, offsets, thr);
} }
...@@ -1875,13 +1889,15 @@ row_ins_duplicate_error_in_clust( ...@@ -1875,13 +1889,15 @@ row_ins_duplicate_error_in_clust(
INSERT ON DUPLICATE KEY UPDATE). */ INSERT ON DUPLICATE KEY UPDATE). */
err = row_ins_set_exclusive_rec_lock( err = row_ins_set_exclusive_rec_lock(
LOCK_REC_NOT_GAP, rec, LOCK_REC_NOT_GAP,
cursor->index, offsets, thr); btr_cur_get_block(cursor),
rec, cursor->index, offsets, thr);
} else { } else {
err = row_ins_set_shared_rec_lock( err = row_ins_set_shared_rec_lock(
LOCK_REC_NOT_GAP, rec, LOCK_REC_NOT_GAP,
cursor->index, offsets, thr); btr_cur_get_block(cursor),
rec, cursor->index, offsets, thr);
} }
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
......
...@@ -1488,7 +1488,8 @@ row_unlock_for_mysql( ...@@ -1488,7 +1488,8 @@ row_unlock_for_mysql(
rec = btr_pcur_get_rec(pcur); rec = btr_pcur_get_rec(pcur);
lock_rec_unlock(trx, rec, prebuilt->select_lock_type); lock_rec_unlock(trx, btr_pcur_get_block(pcur),
rec, prebuilt->select_lock_type);
mtr_commit(&mtr); mtr_commit(&mtr);
...@@ -1518,7 +1519,8 @@ row_unlock_for_mysql( ...@@ -1518,7 +1519,8 @@ row_unlock_for_mysql(
rec = btr_pcur_get_rec(clust_pcur); rec = btr_pcur_get_rec(clust_pcur);
lock_rec_unlock(trx, rec, prebuilt->select_lock_type); lock_rec_unlock(trx, btr_pcur_get_block(clust_pcur),
rec, prebuilt->select_lock_type);
mtr_commit(&mtr); mtr_commit(&mtr);
} }
......
...@@ -812,7 +812,8 @@ row_sel_get_clust_rec( ...@@ -812,7 +812,8 @@ row_sel_get_clust_rec(
} }
err = lock_clust_rec_read_check_and_lock( err = lock_clust_rec_read_check_and_lock(
0, clust_rec, index, offsets, 0, btr_pcur_get_block(&plan->clust_pcur),
clust_rec, index, offsets,
node->row_lock_mode, lock_type, thr); node->row_lock_mode, lock_type, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
...@@ -885,14 +886,15 @@ UNIV_INLINE ...@@ -885,14 +886,15 @@ UNIV_INLINE
ulint ulint
sel_set_rec_lock( sel_set_rec_lock(
/*=============*/ /*=============*/
/* out: DB_SUCCESS or error code */ /* out: DB_SUCCESS or error code */
rec_t* rec, /* in: record */ const buf_block_t* block, /* in: buffer block of rec */
dict_index_t* index, /* in: index */ const rec_t* rec, /* in: record */
const ulint* offsets,/* in: rec_get_offsets(rec, index) */ dict_index_t* index, /* in: index */
ulint mode, /* in: lock mode */ const ulint* offsets,/* in: rec_get_offsets(rec, index) */
ulint type, /* in: LOCK_ORDINARY, LOCK_GAP, or ulint mode, /* in: lock mode */
LOC_REC_NOT_GAP */ ulint type, /* in: LOCK_ORDINARY, LOCK_GAP, or
que_thr_t* thr) /* in: query thread */ LOC_REC_NOT_GAP */
que_thr_t* thr) /* in: query thread */
{ {
trx_t* trx; trx_t* trx;
ulint err; ulint err;
...@@ -908,10 +910,10 @@ sel_set_rec_lock( ...@@ -908,10 +910,10 @@ sel_set_rec_lock(
if (dict_index_is_clust(index)) { if (dict_index_is_clust(index)) {
err = lock_clust_rec_read_check_and_lock( err = lock_clust_rec_read_check_and_lock(
0, rec, index, offsets, mode, type, thr); 0, block, rec, index, offsets, mode, type, thr);
} else { } else {
err = lock_sec_rec_read_check_and_lock( err = lock_sec_rec_read_check_and_lock(
0, rec, index, offsets, mode, type, thr); 0, block, rec, index, offsets, mode, type, thr);
} }
return(err); return(err);
...@@ -1165,7 +1167,7 @@ row_sel_try_search_shortcut( ...@@ -1165,7 +1167,7 @@ row_sel_try_search_shortcut(
ret = SEL_RETRY; ret = SEL_RETRY;
goto func_exit; goto func_exit;
} }
} else if (!lock_sec_rec_cons_read_sees(rec, index, node->read_view)) { } else if (!lock_sec_rec_cons_read_sees(rec, node->read_view)) {
ret = SEL_RETRY; ret = SEL_RETRY;
goto func_exit; goto func_exit;
...@@ -1429,7 +1431,8 @@ rec_loop: ...@@ -1429,7 +1431,8 @@ rec_loop:
lock_type = LOCK_ORDINARY; lock_type = LOCK_ORDINARY;
} }
err = sel_set_rec_lock(next_rec, index, offsets, err = sel_set_rec_lock(btr_pcur_get_block(&plan->pcur),
next_rec, index, offsets,
node->row_lock_mode, node->row_lock_mode,
lock_type, thr); lock_type, thr);
...@@ -1485,7 +1488,8 @@ skip_lock: ...@@ -1485,7 +1488,8 @@ skip_lock:
lock_type = LOCK_ORDINARY; lock_type = LOCK_ORDINARY;
} }
err = sel_set_rec_lock(rec, index, offsets, err = sel_set_rec_lock(btr_pcur_get_block(&plan->pcur),
rec, index, offsets,
node->row_lock_mode, lock_type, thr); node->row_lock_mode, lock_type, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
...@@ -1581,7 +1585,7 @@ skip_lock: ...@@ -1581,7 +1585,7 @@ skip_lock:
rec = old_vers; rec = old_vers;
} }
} else if (!lock_sec_rec_cons_read_sees(rec, index, } else if (!lock_sec_rec_cons_read_sees(rec,
node->read_view)) { node->read_view)) {
cons_read_requires_clust_rec = TRUE; cons_read_requires_clust_rec = TRUE;
} }
...@@ -2899,7 +2903,8 @@ row_sel_get_clust_rec_for_mysql( ...@@ -2899,7 +2903,8 @@ row_sel_get_clust_rec_for_mysql(
we set a LOCK_REC_NOT_GAP type lock */ we set a LOCK_REC_NOT_GAP type lock */
err = lock_clust_rec_read_check_and_lock( err = lock_clust_rec_read_check_and_lock(
0, clust_rec, clust_index, *offsets, 0, btr_pcur_get_block(prebuilt->clust_pcur),
clust_rec, clust_index, *offsets,
prebuilt->select_lock_type, LOCK_REC_NOT_GAP, thr); prebuilt->select_lock_type, LOCK_REC_NOT_GAP, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
...@@ -3749,7 +3754,8 @@ rec_loop: ...@@ -3749,7 +3754,8 @@ rec_loop:
offsets = rec_get_offsets(rec, index, offsets, offsets = rec_get_offsets(rec, index, offsets,
ULINT_UNDEFINED, &heap); ULINT_UNDEFINED, &heap);
err = sel_set_rec_lock(rec, index, offsets, err = sel_set_rec_lock(btr_pcur_get_block(pcur),
rec, index, offsets,
prebuilt->select_lock_type, prebuilt->select_lock_type,
LOCK_ORDINARY, thr); LOCK_ORDINARY, thr);
...@@ -3883,6 +3889,7 @@ wrong_offs: ...@@ -3883,6 +3889,7 @@ wrong_offs:
using a READ COMMITTED isolation level. */ using a READ COMMITTED isolation level. */
err = sel_set_rec_lock( err = sel_set_rec_lock(
btr_pcur_get_block(pcur),
rec, index, offsets, rec, index, offsets,
prebuilt->select_lock_type, LOCK_GAP, prebuilt->select_lock_type, LOCK_GAP,
thr); thr);
...@@ -3918,6 +3925,7 @@ wrong_offs: ...@@ -3918,6 +3925,7 @@ wrong_offs:
using a READ COMMITTED isolation level. */ using a READ COMMITTED isolation level. */
err = sel_set_rec_lock( err = sel_set_rec_lock(
btr_pcur_get_block(pcur),
rec, index, offsets, rec, index, offsets,
prebuilt->select_lock_type, LOCK_GAP, prebuilt->select_lock_type, LOCK_GAP,
thr); thr);
...@@ -3986,7 +3994,8 @@ no_gap_lock: ...@@ -3986,7 +3994,8 @@ no_gap_lock:
lock_type = LOCK_REC_NOT_GAP; lock_type = LOCK_REC_NOT_GAP;
} }
err = sel_set_rec_lock(rec, index, offsets, err = sel_set_rec_lock(btr_pcur_get_block(pcur),
rec, index, offsets,
prebuilt->select_lock_type, prebuilt->select_lock_type,
lock_type, thr); lock_type, thr);
...@@ -4093,8 +4102,7 @@ no_gap_lock: ...@@ -4093,8 +4102,7 @@ no_gap_lock:
rec = old_vers; rec = old_vers;
} }
} else if (!lock_sec_rec_cons_read_sees(rec, index, } else if (!lock_sec_rec_cons_read_sees(rec, trx->read_view)) {
trx->read_view)) {
/* We are looking into a non-clustered index, /* We are looking into a non-clustered index,
and to get the right version of the record we and to get the right version of the record we
have to look also into the clustered index: this have to look also into the clustered index: this
......
...@@ -1764,7 +1764,8 @@ row_upd_clust_step( ...@@ -1764,7 +1764,8 @@ row_upd_clust_step(
if (!node->has_clust_rec_x_lock) { if (!node->has_clust_rec_x_lock) {
err = lock_clust_rec_modify_check_and_lock( err = lock_clust_rec_modify_check_and_lock(
0, rec, index, offsets, thr); 0, btr_pcur_get_block(pcur),
rec, index, offsets, thr);
if (err != DB_SUCCESS) { if (err != DB_SUCCESS) {
mtr_commit(mtr); mtr_commit(mtr);
goto exit_func; goto exit_func;
......
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