Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
MariaDB
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
MariaDB
Commits
2e64513f
Commit
2e64513f
authored
Feb 05, 2021
by
Marko Mäkelä
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
MDEV-20612 preparation: Fewer calls to buf_page_t::id()
parent
b19ec884
Changes
14
Show whitespace changes
Inline
Side-by-side
Showing
14 changed files
with
598 additions
and
767 deletions
+598
-767
storage/innobase/btr/btr0btr.cc
storage/innobase/btr/btr0btr.cc
+21
-18
storage/innobase/btr/btr0cur.cc
storage/innobase/btr/btr0cur.cc
+16
-21
storage/innobase/btr/btr0defragment.cc
storage/innobase/btr/btr0defragment.cc
+2
-2
storage/innobase/gis/gis0rtree.cc
storage/innobase/gis/gis0rtree.cc
+2
-2
storage/innobase/gis/gis0sea.cc
storage/innobase/gis/gis0sea.cc
+8
-10
storage/innobase/ibuf/ibuf0ibuf.cc
storage/innobase/ibuf/ibuf0ibuf.cc
+2
-1
storage/innobase/include/lock0lock.h
storage/innobase/include/lock0lock.h
+34
-65
storage/innobase/include/lock0prdt.h
storage/innobase/include/lock0prdt.h
+3
-7
storage/innobase/include/lock0priv.h
storage/innobase/include/lock0priv.h
+15
-10
storage/innobase/include/lock0priv.ic
storage/innobase/include/lock0priv.ic
+2
-20
storage/innobase/lock/lock0lock.cc
storage/innobase/lock/lock0lock.cc
+470
-586
storage/innobase/lock/lock0prdt.cc
storage/innobase/lock/lock0prdt.cc
+17
-20
storage/innobase/row/row0mysql.cc
storage/innobase/row/row0mysql.cc
+3
-2
storage/innobase/row/row0upd.cc
storage/innobase/row/row0upd.cc
+3
-3
No files found.
storage/innobase/btr/btr0btr.cc
View file @
2e64513f
...
@@ -1873,6 +1873,8 @@ btr_root_raise_and_insert(
...
@@ -1873,6 +1873,8 @@ btr_root_raise_and_insert(
ut_a
(
!
root_page_zip
||
page_zip_validate
(
root_page_zip
,
root
->
frame
,
ut_a
(
!
root_page_zip
||
page_zip_validate
(
root_page_zip
,
root
->
frame
,
index
));
index
));
#endif
/* UNIV_ZIP_DEBUG */
#endif
/* UNIV_ZIP_DEBUG */
const
page_id_t
root_id
{
root
->
page
.
id
()};
#ifdef UNIV_BTR_DEBUG
#ifdef UNIV_BTR_DEBUG
if
(
!
dict_index_is_ibuf
(
index
))
{
if
(
!
dict_index_is_ibuf
(
index
))
{
ulint
space
=
index
->
table
->
space_id
;
ulint
space
=
index
->
table
->
space_id
;
...
@@ -1883,7 +1885,7 @@ btr_root_raise_and_insert(
...
@@ -1883,7 +1885,7 @@ btr_root_raise_and_insert(
+
root
->
frame
,
space
));
+
root
->
frame
,
space
));
}
}
ut_a
(
dict_index_get_page
(
index
)
==
root
->
page
.
id
()
.
page_no
());
ut_a
(
dict_index_get_page
(
index
)
==
root
_id
.
page_no
());
#endif
/* UNIV_BTR_DEBUG */
#endif
/* UNIV_BTR_DEBUG */
ut_ad
(
mtr
->
memo_contains_flagged
(
&
index
->
lock
,
MTR_MEMO_X_LOCK
ut_ad
(
mtr
->
memo_contains_flagged
(
&
index
->
lock
,
MTR_MEMO_X_LOCK
|
MTR_MEMO_SX_LOCK
));
|
MTR_MEMO_SX_LOCK
));
...
@@ -1941,7 +1943,7 @@ btr_root_raise_and_insert(
...
@@ -1941,7 +1943,7 @@ btr_root_raise_and_insert(
/* Move any existing predicate locks */
/* Move any existing predicate locks */
if
(
dict_index_is_spatial
(
index
))
{
if
(
dict_index_is_spatial
(
index
))
{
lock_prdt_rec_move
(
new_block
,
root
);
lock_prdt_rec_move
(
new_block
,
root
_id
);
}
else
{
}
else
{
btr_search_move_or_delete_hash_entries
(
btr_search_move_or_delete_hash_entries
(
new_block
,
root
);
new_block
,
root
);
...
@@ -1986,7 +1988,7 @@ btr_root_raise_and_insert(
...
@@ -1986,7 +1988,7 @@ btr_root_raise_and_insert(
root page: we cannot discard the lock structs on the root page */
root page: we cannot discard the lock structs on the root page */
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
lock_update_root_raise
(
new_block
,
root
);
lock_update_root_raise
(
*
new_block
,
root_id
);
}
}
/* Create a memory heap where the node pointer is stored */
/* Create a memory heap where the node pointer is stored */
...
@@ -3342,7 +3344,7 @@ btr_lift_page_up(
...
@@ -3342,7 +3344,7 @@ btr_lift_page_up(
/* Also update the predicate locks */
/* Also update the predicate locks */
if
(
dict_index_is_spatial
(
index
))
{
if
(
dict_index_is_spatial
(
index
))
{
lock_prdt_rec_move
(
father_block
,
block
);
lock_prdt_rec_move
(
father_block
,
block
->
page
.
id
()
);
}
else
{
}
else
{
btr_search_move_or_delete_hash_entries
(
btr_search_move_or_delete_hash_entries
(
father_block
,
block
);
father_block
,
block
);
...
@@ -3350,14 +3352,14 @@ btr_lift_page_up(
...
@@ -3350,14 +3352,14 @@ btr_lift_page_up(
}
}
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
const
page_id_t
id
{
block
->
page
.
id
()};
/* Free predicate page locks on the block */
/* Free predicate page locks on the block */
if
(
dict_index_is_spatial
(
index
))
{
if
(
index
->
is_spatial
(
))
{
lock_sys
.
mutex_lock
()
;
LockMutexGuard
g
;
lock_prdt_page_free_from_discard
(
lock_prdt_page_free_from_discard
(
block
,
&
lock_sys
.
prdt_page_hash
);
id
,
&
lock_sys
.
prdt_page_hash
);
lock_sys
.
mutex_unlock
();
}
}
lock_update_copy_and_discard
(
father_block
,
block
);
lock_update_copy_and_discard
(
*
father_block
,
id
);
}
}
/* Go upward to root page, decrementing levels by one. */
/* Go upward to root page, decrementing levels by one. */
...
@@ -3576,6 +3578,8 @@ btr_compress(
...
@@ -3576,6 +3578,8 @@ btr_compress(
/* Remove the page from the level list */
/* Remove the page from the level list */
btr_level_list_remove
(
*
block
,
*
index
,
mtr
);
btr_level_list_remove
(
*
block
,
*
index
,
mtr
);
const
page_id_t
id
{
block
->
page
.
id
()};
if
(
dict_index_is_spatial
(
index
))
{
if
(
dict_index_is_spatial
(
index
))
{
rec_t
*
my_rec
=
father_cursor
.
page_cur
.
rec
;
rec_t
*
my_rec
=
father_cursor
.
page_cur
.
rec
;
...
@@ -3605,16 +3609,15 @@ btr_compress(
...
@@ -3605,16 +3609,15 @@ btr_compress(
}
}
/* No GAP lock needs to be worrying about */
/* No GAP lock needs to be worrying about */
lock_sys
.
mutex_lock
()
;
LockMutexGuard
g
;
lock_prdt_page_free_from_discard
(
lock_prdt_page_free_from_discard
(
block
,
&
lock_sys
.
prdt_page_hash
);
id
,
&
lock_sys
.
prdt_page_hash
);
lock_rec_free_all_from_discard_page
(
block
);
lock_rec_free_all_from_discard_page
(
id
);
lock_sys
.
mutex_unlock
();
}
else
{
}
else
{
btr_cur_node_ptr_delete
(
&
father_cursor
,
mtr
);
btr_cur_node_ptr_delete
(
&
father_cursor
,
mtr
);
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
lock_update_merge_left
(
lock_update_merge_left
(
merge_block
,
orig_pred
,
block
);
*
merge_block
,
orig_pred
,
id
);
}
}
}
}
...
@@ -3758,11 +3761,11 @@ btr_compress(
...
@@ -3758,11 +3761,11 @@ btr_compress(
offsets2
,
offsets
,
offsets2
,
offsets
,
merge_page
,
mtr
);
merge_page
,
mtr
);
}
}
lock_sys
.
mutex_lock
();
const
page_id_t
id
{
block
->
page
.
id
()};
LockMutexGuard
g
;
lock_prdt_page_free_from_discard
(
lock_prdt_page_free_from_discard
(
block
,
&
lock_sys
.
prdt_page_hash
);
id
,
&
lock_sys
.
prdt_page_hash
);
lock_rec_free_all_from_discard_page
(
block
);
lock_rec_free_all_from_discard_page
(
id
);
lock_sys
.
mutex_unlock
();
}
else
{
}
else
{
compressed
=
btr_cur_pessimistic_delete
(
&
err
,
TRUE
,
compressed
=
btr_cur_pessimistic_delete
(
&
err
,
TRUE
,
...
...
storage/innobase/btr/btr0cur.cc
View file @
2e64513f
...
@@ -3311,15 +3311,15 @@ static void btr_cur_prefetch_siblings(const buf_block_t *block,
...
@@ -3311,15 +3311,15 @@ static void btr_cur_prefetch_siblings(const buf_block_t *block,
uint32_t
prev
=
mach_read_from_4
(
my_assume_aligned
<
4
>
(
page
+
FIL_PAGE_PREV
));
uint32_t
prev
=
mach_read_from_4
(
my_assume_aligned
<
4
>
(
page
+
FIL_PAGE_PREV
));
uint32_t
next
=
mach_read_from_4
(
my_assume_aligned
<
4
>
(
page
+
FIL_PAGE_NEXT
));
uint32_t
next
=
mach_read_from_4
(
my_assume_aligned
<
4
>
(
page
+
FIL_PAGE_NEXT
));
fil_space_t
*
space
=
index
->
table
->
space
;
if
(
prev
==
FIL_NULL
);
if
(
prev
==
FIL_NULL
);
else
if
(
index
->
table
->
space
->
acquire
())
else
if
(
space
->
acquire
())
buf_read_page_background
(
index
->
table
->
space
,
buf_read_page_background
(
space
,
page_id_t
(
space
->
id
,
prev
),
page_id_t
(
block
->
page
.
id
().
space
(),
prev
),
block
->
zip_size
(),
false
);
block
->
zip_size
(),
false
);
if
(
next
==
FIL_NULL
);
if
(
next
==
FIL_NULL
);
else
if
(
index
->
table
->
space
->
acquire
())
else
if
(
space
->
acquire
())
buf_read_page_background
(
index
->
table
->
space
,
buf_read_page_background
(
space
,
page_id_t
(
space
->
id
,
next
),
page_id_t
(
block
->
page
.
id
().
space
(),
next
),
block
->
zip_size
(),
false
);
block
->
zip_size
(),
false
);
}
}
...
@@ -3859,7 +3859,7 @@ btr_cur_upd_lock_and_undo(
...
@@ -3859,7 +3859,7 @@ btr_cur_upd_lock_and_undo(
if
(
!
(
flags
&
BTR_NO_LOCKING_FLAG
))
{
if
(
!
(
flags
&
BTR_NO_LOCKING_FLAG
))
{
err
=
lock_clust_rec_modify_check_and_lock
(
err
=
lock_clust_rec_modify_check_and_lock
(
flags
,
btr_cur_get_block
(
cursor
),
rec
,
index
,
btr_cur_get_block
(
cursor
),
rec
,
index
,
offsets
,
thr
);
offsets
,
thr
);
if
(
err
!=
DB_SUCCESS
)
{
if
(
err
!=
DB_SUCCESS
)
{
return
(
err
);
return
(
err
);
...
@@ -4751,7 +4751,8 @@ btr_cur_optimistic_update(
...
@@ -4751,7 +4751,8 @@ btr_cur_optimistic_update(
btr_page_reorganize
(
page_cursor
,
index
,
mtr
);
btr_page_reorganize
(
page_cursor
,
index
,
mtr
);
}
else
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
}
else
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
/* Restore the old explicit lock state on the record */
/* Restore the old explicit lock state on the record */
lock_rec_restore_from_page_infimum
(
block
,
rec
,
block
);
lock_rec_restore_from_page_infimum
(
*
block
,
rec
,
block
->
page
.
id
());
}
}
page_cur_move_to_next
(
page_cursor
);
page_cur_move_to_next
(
page_cursor
);
...
@@ -4805,10 +4806,11 @@ btr_cur_pess_upd_restore_supremum(
...
@@ -4805,10 +4806,11 @@ btr_cur_pess_upd_restore_supremum(
const
uint32_t
prev_page_no
=
btr_page_get_prev
(
page
);
const
uint32_t
prev_page_no
=
btr_page_get_prev
(
page
);
const
page_id_t
page_id
(
block
->
page
.
id
().
space
(),
prev_page_no
);
const
page_id_t
block_id
{
block
->
page
.
id
()};
const
page_id_t
prev_id
(
block_id
.
space
(),
prev_page_no
);
ut_ad
(
prev_page_no
!=
FIL_NULL
);
ut_ad
(
prev_page_no
!=
FIL_NULL
);
prev_block
=
buf_page_get_with_no_latch
(
p
age
_id
,
block
->
zip_size
(),
prev_block
=
buf_page_get_with_no_latch
(
p
rev
_id
,
block
->
zip_size
(),
mtr
);
mtr
);
#ifdef UNIV_BTR_DEBUG
#ifdef UNIV_BTR_DEBUG
ut_a
(
btr_page_get_next
(
prev_block
->
frame
)
ut_a
(
btr_page_get_next
(
prev_block
->
frame
)
...
@@ -4818,7 +4820,7 @@ btr_cur_pess_upd_restore_supremum(
...
@@ -4818,7 +4820,7 @@ btr_cur_pess_upd_restore_supremum(
/* We must already have an x-latch on prev_block! */
/* We must already have an x-latch on prev_block! */
ut_ad
(
mtr
->
memo_contains_flagged
(
prev_block
,
MTR_MEMO_PAGE_X_FIX
));
ut_ad
(
mtr
->
memo_contains_flagged
(
prev_block
,
MTR_MEMO_PAGE_X_FIX
));
lock_rec_reset_and_inherit_gap_locks
(
prev_block
,
block
,
lock_rec_reset_and_inherit_gap_locks
(
*
prev_block
,
block_id
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
,
page_rec_get_heap_no
(
rec
));
page_rec_get_heap_no
(
rec
));
}
}
...
@@ -5106,7 +5108,8 @@ btr_cur_pessimistic_update(
...
@@ -5106,7 +5108,8 @@ btr_cur_pessimistic_update(
}
}
}
else
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
}
else
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
lock_rec_restore_from_page_infimum
(
lock_rec_restore_from_page_infimum
(
btr_cur_get_block
(
cursor
),
rec
,
block
);
*
btr_cur_get_block
(
cursor
),
rec
,
block
->
page
.
id
());
}
}
if
(
!
rec_get_deleted_flag
(
rec
,
rec_offs_comp
(
*
offsets
))
if
(
!
rec_get_deleted_flag
(
rec
,
rec_offs_comp
(
*
offsets
))
...
@@ -5251,7 +5254,7 @@ btr_cur_pessimistic_update(
...
@@ -5251,7 +5254,7 @@ btr_cur_pessimistic_update(
rec
=
page_cursor
->
rec
;
rec
=
page_cursor
->
rec
;
}
else
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
}
else
if
(
!
dict_table_is_locking_disabled
(
index
->
table
))
{
lock_rec_restore_from_page_infimum
(
lock_rec_restore_from_page_infimum
(
btr_cur_get_block
(
cursor
),
rec
,
block
);
*
btr_cur_get_block
(
cursor
),
rec
,
block
->
page
.
id
()
);
}
}
/* If necessary, restore also the correct lock state for a new,
/* If necessary, restore also the correct lock state for a new,
...
@@ -5351,14 +5354,6 @@ btr_cur_del_mark_set_clust_rec(
...
@@ -5351,14 +5354,6 @@ btr_cur_del_mark_set_clust_rec(
return
(
DB_SUCCESS
);
return
(
DB_SUCCESS
);
}
}
err
=
lock_clust_rec_modify_check_and_lock
(
BTR_NO_LOCKING_FLAG
,
block
,
rec
,
index
,
offsets
,
thr
);
if
(
err
!=
DB_SUCCESS
)
{
return
(
err
);
}
err
=
trx_undo_report_row_operation
(
thr
,
index
,
err
=
trx_undo_report_row_operation
(
thr
,
index
,
entry
,
NULL
,
0
,
rec
,
offsets
,
entry
,
NULL
,
0
,
rec
,
offsets
,
&
roll_ptr
);
&
roll_ptr
);
...
...
storage/innobase/btr/btr0defragment.cc
View file @
2e64513f
...
@@ -397,8 +397,8 @@ btr_defragment_merge_pages(
...
@@ -397,8 +397,8 @@ btr_defragment_merge_pages(
if
(
n_recs_to_move
==
n_recs
)
{
if
(
n_recs_to_move
==
n_recs
)
{
/* The whole page is merged with the previous page,
/* The whole page is merged with the previous page,
free it. */
free it. */
lock_update_merge_left
(
to_block
,
orig_pred
,
const
page_id_t
from
{
from_block
->
page
.
id
()};
from_block
);
lock_update_merge_left
(
*
to_block
,
orig_pred
,
from
);
btr_search_drop_page_hash_index
(
from_block
);
btr_search_drop_page_hash_index
(
from_block
);
btr_level_list_remove
(
*
from_block
,
*
index
,
mtr
);
btr_level_list_remove
(
*
from_block
,
*
index
,
mtr
);
btr_page_get_father
(
index
,
from_block
,
mtr
,
&
parent
);
btr_page_get_father
(
index
,
from_block
,
mtr
,
&
parent
);
...
...
storage/innobase/gis/gis0rtree.cc
View file @
2e64513f
/*****************************************************************************
/*****************************************************************************
Copyright (c) 2016, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2016, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2018, 202
0
, MariaDB Corporation.
Copyright (c) 2018, 202
1
, MariaDB Corporation.
This program is free software; you can redistribute it and/or modify it under
This program is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free Software
the terms of the GNU General Public License as published by the Free Software
...
@@ -769,7 +769,7 @@ rtr_split_page_move_rec_list(
...
@@ -769,7 +769,7 @@ rtr_split_page_move_rec_list(
ut_a
(
rec
);
ut_a
(
rec
);
lock_rec_restore_from_page_infimum
(
lock_rec_restore_from_page_infimum
(
new_block
,
rec
,
block
);
*
new_block
,
rec
,
block
->
page
.
id
()
);
page_cur_move_to_next
(
&
new_page_cursor
);
page_cur_move_to_next
(
&
new_page_cursor
);
...
...
storage/innobase/gis/gis0sea.cc
View file @
2e64513f
/*****************************************************************************
/*****************************************************************************
Copyright (c) 2016, 2018, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2016, 2018, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2017, 202
0
, MariaDB Corporation.
Copyright (c) 2017, 202
1
, MariaDB Corporation.
This program is free software; you can redistribute it and/or modify it under
This program is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free Software
the terms of the GNU General Public License as published by the Free Software
...
@@ -1164,7 +1164,7 @@ rtr_check_discard_page(
...
@@ -1164,7 +1164,7 @@ rtr_check_discard_page(
the root page */
the root page */
buf_block_t
*
block
)
/*!< in: block of page to be discarded */
buf_block_t
*
block
)
/*!< in: block of page to be discarded */
{
{
const
ulint
pageno
=
block
->
page
.
id
().
page_no
()
;
const
page_id_t
id
{
block
->
page
.
id
()}
;
mysql_mutex_lock
(
&
index
->
rtr_track
->
rtr_active_mutex
);
mysql_mutex_lock
(
&
index
->
rtr_track
->
rtr_active_mutex
);
...
@@ -1175,8 +1175,8 @@ rtr_check_discard_page(
...
@@ -1175,8 +1175,8 @@ rtr_check_discard_page(
mysql_mutex_lock
(
&
rtr_info
->
rtr_path_mutex
);
mysql_mutex_lock
(
&
rtr_info
->
rtr_path_mutex
);
for
(
const
node_visit_t
&
node
:
*
rtr_info
->
path
)
{
for
(
const
node_visit_t
&
node
:
*
rtr_info
->
path
)
{
if
(
node
.
page_no
==
pageno
)
{
if
(
node
.
page_no
==
id
.
page_no
()
)
{
rtr_rebuild_path
(
rtr_info
,
page
no
);
rtr_rebuild_path
(
rtr_info
,
node
.
page_
no
);
break
;
break
;
}
}
}
}
...
@@ -1185,8 +1185,7 @@ rtr_check_discard_page(
...
@@ -1185,8 +1185,7 @@ rtr_check_discard_page(
if
(
rtr_info
->
matches
)
{
if
(
rtr_info
->
matches
)
{
mysql_mutex_lock
(
&
rtr_info
->
matches
->
rtr_match_mutex
);
mysql_mutex_lock
(
&
rtr_info
->
matches
->
rtr_match_mutex
);
if
((
&
rtr_info
->
matches
->
block
)
->
page
.
id
().
page_no
()
if
((
&
rtr_info
->
matches
->
block
)
->
page
.
id
()
==
id
)
{
==
pageno
)
{
if
(
!
rtr_info
->
matches
->
matched_recs
->
empty
())
{
if
(
!
rtr_info
->
matches
->
matched_recs
->
empty
())
{
rtr_info
->
matches
->
matched_recs
->
clear
();
rtr_info
->
matches
->
matched_recs
->
clear
();
}
}
...
@@ -1200,10 +1199,9 @@ rtr_check_discard_page(
...
@@ -1200,10 +1199,9 @@ rtr_check_discard_page(
mysql_mutex_unlock
(
&
index
->
rtr_track
->
rtr_active_mutex
);
mysql_mutex_unlock
(
&
index
->
rtr_track
->
rtr_active_mutex
);
lock_sys
.
mutex_lock
();
LockMutexGuard
g
;
lock_prdt_page_free_from_discard
(
block
,
&
lock_sys
.
prdt_hash
);
lock_prdt_page_free_from_discard
(
id
,
&
lock_sys
.
prdt_hash
);
lock_prdt_page_free_from_discard
(
block
,
&
lock_sys
.
prdt_page_hash
);
lock_prdt_page_free_from_discard
(
id
,
&
lock_sys
.
prdt_page_hash
);
lock_sys
.
mutex_unlock
();
}
}
/** Structure acts as functor to get the optimistic access of the page.
/** Structure acts as functor to get the optimistic access of the page.
...
...
storage/innobase/ibuf/ibuf0ibuf.cc
View file @
2e64513f
...
@@ -3790,7 +3790,8 @@ ibuf_insert_to_index_page(
...
@@ -3790,7 +3790,8 @@ ibuf_insert_to_index_page(
&
page_cur
);
&
page_cur
);
ut_ad
(
!
cmp_dtuple_rec
(
entry
,
rec
,
offsets
));
ut_ad
(
!
cmp_dtuple_rec
(
entry
,
rec
,
offsets
));
lock_rec_restore_from_page_infimum
(
block
,
rec
,
block
);
lock_rec_restore_from_page_infimum
(
*
block
,
rec
,
block
->
page
.
id
());
}
else
{
}
else
{
offsets
=
NULL
;
offsets
=
NULL
;
ibuf_insert_to_index_page_low
(
entry
,
block
,
index
,
ibuf_insert_to_index_page_low
(
entry
,
block
,
index
,
...
...
storage/innobase/include/lock0lock.h
View file @
2e64513f
...
@@ -112,28 +112,17 @@ lock_update_merge_right(
...
@@ -112,28 +112,17 @@ lock_update_merge_right(
const
buf_block_t
*
left_block
);
/*!< in: merged index
const
buf_block_t
*
left_block
);
/*!< in: merged index
page which will be
page which will be
discarded */
discarded */
/*************************************************************//**
/** Update locks 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
root page, even though they do not make sense on other than leaf
root page, even though they do not make sense on other than leaf
pages: the reason is that in a pessimistic update the infimum record
pages: the reason is that in a pessimistic update the infimum record
of the root page will act as a dummy carrier of the locks of the record
of the root page will act as a dummy carrier of the locks of the record
to be updated. */
to be updated. */
void
void
lock_update_root_raise
(
const
buf_block_t
&
block
,
const
page_id_t
root
);
lock_update_root_raise
(
/** Update the lock table when a page is copied to another.
/*===================*/
@param new_block the target page
const
buf_block_t
*
block
,
/*!< in: index page to which copied */
@param old old page (not index root page) */
const
buf_block_t
*
root
);
/*!< in: root page */
void
lock_update_copy_and_discard
(
const
buf_block_t
&
new_block
,
page_id_t
old
);
/*************************************************************//**
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! */
void
lock_update_copy_and_discard
(
/*=========================*/
const
buf_block_t
*
new_block
,
/*!< in: index page to
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
...
@@ -141,18 +130,12 @@ lock_update_split_left(
...
@@ -141,18 +130,12 @@ lock_update_split_left(
/*===================*/
/*===================*/
const
buf_block_t
*
right_block
,
/*!< in: right page */
const
buf_block_t
*
right_block
,
/*!< in: right page */
const
buf_block_t
*
left_block
);
/*!< in: left page */
const
buf_block_t
*
left_block
);
/*!< in: left page */
/*************************************************************//**
/** Update the lock table when a page is merged to the left.
Updates the lock table when a page is merged to the left. */
@param left left page
void
@param orig_pred original predecessor of supremum on the left page before merge
lock_update_merge_left
(
@param right merged, to-be-discarded right page */
/*===================*/
void
lock_update_merge_left
(
const
buf_block_t
&
left
,
const
rec_t
*
orig_pred
,
const
buf_block_t
*
left_block
,
/*!< in: left page to
const
page_id_t
right
);
which merged */
const
rec_t
*
orig_pred
,
/*!< in: original predecessor
of supremum on the left page
before merge */
const
buf_block_t
*
right_block
);
/*!< in: merged index page
which will be discarded */
/*************************************************************//**
/*************************************************************//**
Updates the lock table when a page is split and merged to
Updates the lock table when a page is split and merged to
two pages. */
two pages. */
...
@@ -169,9 +152,9 @@ inherited from rec. */
...
@@ -169,9 +152,9 @@ inherited from rec. */
void
void
lock_rec_reset_and_inherit_gap_locks
(
lock_rec_reset_and_inherit_gap_locks
(
/*=================================*/
/*=================================*/
const
buf_block_t
*
heir_block
,
/*!< in: block containing the
const
buf_block_t
&
heir_block
,
/*!< in: block containing the
record which inherits */
record which inherits */
const
buf_block_t
*
block
,
/*!< in: block
containing the
const
page_id_t
donor
,
/*!< in: page
containing the
record from which inherited;
record from which inherited;
does NOT reset the locks on
does NOT reset the locks on
this record */
this record */
...
@@ -220,20 +203,14 @@ lock_rec_store_on_page_infimum(
...
@@ -220,20 +203,14 @@ lock_rec_store_on_page_infimum(
record of the same page; lock
record of the same page; lock
bits are reset on the
bits are reset on the
record */
record */
/*********************************************************************//**
/** Restore the 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 a page.
state was stored on the infimum of the page. */
@param block buffer block containing rec
void
@param rec record whose lock state is restored
lock_rec_restore_from_page_infimum
(
@param donator page (rec is not necessarily on this page)
/*===============================*/
whose infimum stored the lock state; lock bits are reset on the infimum */
const
buf_block_t
*
block
,
/*!< in: buffer block containing rec */
void
lock_rec_restore_from_page_infimum
(
const
buf_block_t
&
block
,
const
rec_t
*
rec
,
/*!< in: record whose lock state
const
rec_t
*
rec
,
page_id_t
donator
);
is restored */
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 */
/*********************************************************************//**
/*********************************************************************//**
Checks if locks of other transactions prevent an immediate insert of
Checks if locks of other transactions prevent an immediate insert of
a record. If they do, first tests if the query thread should anyway
a record. If they do, first tests if the query thread should anyway
...
@@ -266,8 +243,6 @@ lock queue.
...
@@ -266,8 +243,6 @@ lock queue.
dberr_t
dberr_t
lock_clust_rec_modify_check_and_lock
(
lock_clust_rec_modify_check_and_lock
(
/*=================================*/
/*=================================*/
ulint
flags
,
/*!< in: if BTR_NO_LOCKING_FLAG
bit is set, does nothing */
const
buf_block_t
*
block
,
/*!< in: buffer block of rec */
const
buf_block_t
*
block
,
/*!< in: buffer block of rec */
const
rec_t
*
rec
,
/*!< in: record which should be
const
rec_t
*
rec
,
/*!< in: record which should be
modified */
modified */
...
@@ -423,7 +398,7 @@ lock_rec_unlock(
...
@@ -423,7 +398,7 @@ lock_rec_unlock(
/*============*/
/*============*/
trx_t
*
trx
,
/*!< in/out: transaction that has
trx_t
*
trx
,
/*!< in/out: transaction that has
set a record lock */
set a record lock */
const
buf_block_t
*
block
,
/*!< in: buffer block
containing rec */
const
page_id_t
id
,
/*!< in: page
containing rec */
const
rec_t
*
rec
,
/*!< in: record */
const
rec_t
*
rec
,
/*!< in: record */
lock_mode
lock_mode
);
/*!< in: LOCK_S or LOCK_X */
lock_mode
lock_mode
);
/*!< in: LOCK_S or LOCK_X */
...
@@ -589,16 +564,11 @@ lock_trx_has_sys_table_locks(
...
@@ -589,16 +564,11 @@ lock_trx_has_sys_table_locks(
/** Check if the transaction holds an explicit exclusive lock on a record.
/** Check if the transaction holds an explicit exclusive lock on a record.
@param[in] trx transaction
@param[in] trx transaction
@param[in] table table
@param[in] table table
@param[in]
block leaf page
@param[in]
id leaf page identifier
@param[in] heap_no heap number identifying the record
@param[in] heap_no heap number identifying the record
@return whether an explicit X-lock is held */
@return whether an explicit X-lock is held */
bool
bool
lock_trx_has_expl_x_lock
(
const
trx_t
&
trx
,
const
dict_table_t
&
table
,
lock_trx_has_expl_x_lock
(
page_id_t
id
,
ulint
heap_no
);
const
trx_t
*
trx
,
/*!< in: transaction to check */
const
dict_table_t
*
table
,
/*!< in: table to check */
const
buf_block_t
*
block
,
/*!< in: buffer block of the record */
ulint
heap_no
)
/*!< in: record heap number */
MY_ATTRIBUTE
((
nonnull
,
warn_unused_result
));
#endif
/* UNIV_DEBUG */
#endif
/* UNIV_DEBUG */
/** Lock operation struct */
/** Lock operation struct */
...
@@ -811,6 +781,7 @@ lock_rec_create_low(
...
@@ -811,6 +781,7 @@ lock_rec_create_low(
dict_index_t
*
index
,
dict_index_t
*
index
,
trx_t
*
trx
,
trx_t
*
trx
,
bool
holds_trx_mutex
);
bool
holds_trx_mutex
);
/** Enqueue a waiting request for a lock which cannot be granted immediately.
/** Enqueue a waiting request for a lock which cannot be granted immediately.
Check for deadlocks.
Check for deadlocks.
@param[in] type_mode the requested lock mode (LOCK_S or LOCK_X)
@param[in] type_mode the requested lock mode (LOCK_S or LOCK_X)
...
@@ -820,7 +791,8 @@ Check for deadlocks.
...
@@ -820,7 +791,8 @@ Check for deadlocks.
waiting lock request is set
waiting lock request is set
when performing an insert of
when performing an insert of
an index record
an index record
@param[in] block leaf page in the index
@param[in] id page identifier
@param[in] page leaf page in the index
@param[in] heap_no record heap number in the block
@param[in] heap_no record heap number in the block
@param[in] index index tree
@param[in] index index tree
@param[in,out] thr query thread
@param[in,out] thr query thread
...
@@ -833,7 +805,8 @@ lock_rec_enqueue_waiting(
...
@@ -833,7 +805,8 @@ lock_rec_enqueue_waiting(
lock_t
*
c_lock
,
/*!< conflicting lock */
lock_t
*
c_lock
,
/*!< conflicting lock */
#endif
#endif
unsigned
type_mode
,
unsigned
type_mode
,
const
buf_block_t
*
block
,
const
page_id_t
id
,
const
page_t
*
page
,
ulint
heap_no
,
ulint
heap_no
,
dict_index_t
*
index
,
dict_index_t
*
index
,
que_thr_t
*
thr
,
que_thr_t
*
thr
,
...
@@ -851,14 +824,10 @@ lock_rtr_move_rec_list(
...
@@ -851,14 +824,10 @@ lock_rtr_move_rec_list(
moved */
moved */
ulint
num_move
);
/*!< in: num of rec to move */
ulint
num_move
);
/*!< in: num of rec to move */
/*************************************************************//**
/** Remove record locks for an index page which is discarded. This
Removes record lock objects set on an index page which is discarded. This
function does not move locks, or check for waiting locks, therefore the
function does not move locks, or check for waiting locks, therefore the
lock bitmaps must already be reset when this function is called. */
lock bitmaps must already be reset when this function is called. */
void
void
lock_rec_free_all_from_discard_page
(
const
page_id_t
page_id
);
lock_rec_free_all_from_discard_page
(
/*================================*/
const
buf_block_t
*
block
);
/*!< in: page to be discarded */
/** Cancel a waiting lock request and release possibly waiting transactions */
/** Cancel a waiting lock request and release possibly waiting transactions */
void
lock_cancel_waiting_and_release
(
lock_t
*
lock
);
void
lock_cancel_waiting_and_release
(
lock_t
*
lock
);
...
...
storage/innobase/include/lock0prdt.h
View file @
2e64513f
...
@@ -181,8 +181,7 @@ lock_prdt_rec_move(
...
@@ -181,8 +181,7 @@ lock_prdt_rec_move(
/*===============*/
/*===============*/
const
buf_block_t
*
receiver
,
/*!< in: buffer block containing
const
buf_block_t
*
receiver
,
/*!< in: buffer block containing
the receiving record */
the receiving record */
const
buf_block_t
*
donator
);
/*!< in: buffer block containing
const
page_id_t
donator
);
/*!< in: target page */
the donating record */
/** Check whether there are R-tree Page lock on a page
/** Check whether there are R-tree Page lock on a page
@param[in] trx trx to test the lock
@param[in] trx trx to test the lock
...
@@ -191,12 +190,9 @@ lock_prdt_rec_move(
...
@@ -191,12 +190,9 @@ lock_prdt_rec_move(
bool
lock_test_prdt_page_lock
(
const
trx_t
*
trx
,
const
page_id_t
page_id
);
bool
lock_test_prdt_page_lock
(
const
trx_t
*
trx
,
const
page_id_t
page_id
);
/** Removes predicate lock objects set on an index page which is discarded.
/** Removes predicate lock objects set on an index page which is discarded.
@param[in]
block
page to be discarded
@param[in]
id
page to be discarded
@param[in] lock_hash lock hash */
@param[in] lock_hash lock hash */
void
void
lock_prdt_page_free_from_discard
(
lock_prdt_page_free_from_discard
(
const
page_id_t
id
,
hash_table_t
*
lock_hash
);
/*=============================*/
const
buf_block_t
*
block
,
hash_table_t
*
lock_hash
);
#endif
#endif
storage/innobase/include/lock0priv.h
View file @
2e64513f
...
@@ -527,16 +527,21 @@ lock_rec_get_next_const(
...
@@ -527,16 +527,21 @@ lock_rec_get_next_const(
ulint
heap_no
,
/*!< in: heap number of the record */
ulint
heap_no
,
/*!< in: heap number of the record */
const
lock_t
*
lock
);
/*!< in: lock */
const
lock_t
*
lock
);
/*!< in: lock */
/*********************************************************************//**
/** Get the first explicit lock request on a record.
Gets the first explicit lock request on a record.
@param hash lock hash table
@return first lock, NULL if none exists */
@param id page identifier
UNIV_INLINE
@param heap_no record identifier in page
lock_t
*
@return first lock
lock_rec_get_first
(
@retval nullptr if none exists */
/*===============*/
inline
lock_t
*
hash_table_t
*
hash
,
/*!< in: hash chain the lock on */
lock_rec_get_first
(
hash_table_t
*
hash
,
const
page_id_t
id
,
ulint
heap_no
)
const
buf_block_t
*
block
,
/*!< in: block containing the record */
{
ulint
heap_no
);
/*!< in: heap number of the record */
for
(
lock_t
*
lock
=
lock_sys
.
get_first
(
*
hash
,
id
);
lock
;
lock
=
lock_rec_get_next_on_page
(
lock
))
if
(
lock_rec_get_nth_bit
(
lock
,
heap_no
))
return
lock
;
return
nullptr
;
}
/*********************************************************************//**
/*********************************************************************//**
Calculates if lock mode 1 is compatible with lock mode 2.
Calculates if lock mode 1 is compatible with lock mode 2.
...
...
storage/innobase/include/lock0priv.ic
View file @
2e64513f
...
@@ -136,25 +136,7 @@ lock_rec_get_next_const(
...
@@ -136,25 +136,7 @@ lock_rec_get_next_const(
ulint heap_no,/*!< in: heap number of the record */
ulint heap_no,/*!< in: heap number of the record */
const lock_t* lock) /*!< in: lock */
const lock_t* lock) /*!< in: lock */
{
{
return(lock_rec_get_next(heap_no, (lock_t*) lock));
return lock_rec_get_next(heap_no, const_cast<lock_t*>(lock));
}
/*********************************************************************//**
Gets the first explicit lock request on a record.
@return first lock, NULL if none exists */
UNIV_INLINE
lock_t*
lock_rec_get_first(
/*===============*/
hash_table_t* hash, /*!< in: hash chain the lock on */
const buf_block_t* block, /*!< in: block containing the record */
ulint heap_no)/*!< in: heap number of the record */
{
for (lock_t *lock= lock_sys.get_first(*hash, block->page.id());
lock; lock= lock_rec_get_next_on_page(lock))
if (lock_rec_get_nth_bit(lock, heap_no))
return lock;
return nullptr;
}
}
/*********************************************************************//**
/*********************************************************************//**
...
@@ -192,7 +174,7 @@ lock_rec_get_next_on_page_const(
...
@@ -192,7 +174,7 @@ lock_rec_get_next_on_page_const(
{
{
ut_ad(!lock->is_table());
ut_ad(!lock->is_table());
const page_id_t page_id
(lock->un_member.rec_lock.page_id)
;
const page_id_t page_id
{lock->un_member.rec_lock.page_id}
;
lock_sys.mutex_assert_locked();
lock_sys.mutex_assert_locked();
while (!!(lock= static_cast<const lock_t*>(HASH_GET_NEXT(hash, lock))))
while (!!(lock= static_cast<const lock_t*>(HASH_GET_NEXT(hash, lock))))
...
...
storage/innobase/lock/lock0lock.cc
View file @
2e64513f
...
@@ -793,17 +793,15 @@ lock_rec_has_expl(
...
@@ -793,17 +793,15 @@ lock_rec_has_expl(
LOCK_REC_NOT_GAP, for a
LOCK_REC_NOT_GAP, for a
supremum record we regard this
supremum record we regard this
always a gap type request */
always a gap type request */
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
the record */
ulint
heap_no
,
/*!< in: heap number of the record */
ulint
heap_no
,
/*!< in: heap number of the record */
const
trx_t
*
trx
)
/*!< in: transaction */
const
trx_t
*
trx
)
/*!< in: transaction */
{
{
lock_sys
.
mutex_assert_locked
();
ut_ad
((
precise_mode
&
LOCK_MODE_MASK
)
==
LOCK_S
ut_ad
((
precise_mode
&
LOCK_MODE_MASK
)
==
LOCK_S
||
(
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
));
for
(
lock_t
*
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
for
(
lock_t
*
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
lock
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
lock
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
if
(
lock
->
trx
==
trx
&&
if
(
lock
->
trx
==
trx
&&
!
(
lock
->
type_mode
&
(
LOCK_WAIT
|
LOCK_INSERT_INTENTION
))
&&
!
(
lock
->
type_mode
&
(
LOCK_WAIT
|
LOCK_INSERT_INTENTION
))
&&
...
@@ -826,8 +824,7 @@ lock_t*
...
@@ -826,8 +824,7 @@ lock_t*
lock_rec_other_has_expl_req
(
lock_rec_other_has_expl_req
(
/*========================*/
/*========================*/
lock_mode
mode
,
/*!< in: LOCK_S or LOCK_X */
lock_mode
mode
,
/*!< in: LOCK_S or LOCK_X */
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
the record */
bool
wait
,
/*!< in: whether also waiting locks
bool
wait
,
/*!< in: whether also waiting locks
are taken into account */
are taken into account */
ulint
heap_no
,
/*!< in: heap number of the record */
ulint
heap_no
,
/*!< in: heap number of the record */
...
@@ -845,8 +842,7 @@ lock_rec_other_has_expl_req(
...
@@ -845,8 +842,7 @@ lock_rec_other_has_expl_req(
return
(
NULL
);
return
(
NULL
);
}
}
for
(
lock_t
*
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
for
(
lock_t
*
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
block
,
heap_no
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
...
@@ -934,8 +930,7 @@ lock_rec_other_has_conflicting(
...
@@ -934,8 +930,7 @@ lock_rec_other_has_conflicting(
possibly ORed to LOCK_GAP or
possibly ORed to LOCK_GAP or
LOC_REC_NOT_GAP,
LOC_REC_NOT_GAP,
LOCK_INSERT_INTENTION */
LOCK_INSERT_INTENTION */
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
the record */
ulint
heap_no
,
/*!< in: heap number of the record */
ulint
heap_no
,
/*!< in: heap number of the record */
const
trx_t
*
trx
)
/*!< in: our transaction */
const
trx_t
*
trx
)
/*!< in: our transaction */
{
{
...
@@ -945,7 +940,7 @@ lock_rec_other_has_conflicting(
...
@@ -945,7 +940,7 @@ lock_rec_other_has_conflicting(
bool
is_supremum
=
(
heap_no
==
PAGE_HEAP_NO_SUPREMUM
);
bool
is_supremum
=
(
heap_no
==
PAGE_HEAP_NO_SUPREMUM
);
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
...
@@ -1301,7 +1296,8 @@ Check for deadlocks.
...
@@ -1301,7 +1296,8 @@ Check for deadlocks.
waiting lock request is set
waiting lock request is set
when performing an insert of
when performing an insert of
an index record
an index record
@param[in] block leaf page in the index
@param[in] id page identifier
@param[in] page leaf page in the index
@param[in] heap_no record heap number in the block
@param[in] heap_no record heap number in the block
@param[in] index index tree
@param[in] index index tree
@param[in,out] thr query thread
@param[in,out] thr query thread
...
@@ -1314,7 +1310,8 @@ lock_rec_enqueue_waiting(
...
@@ -1314,7 +1310,8 @@ lock_rec_enqueue_waiting(
lock_t
*
c_lock
,
/*!< conflicting lock */
lock_t
*
c_lock
,
/*!< conflicting lock */
#endif
#endif
unsigned
type_mode
,
unsigned
type_mode
,
const
buf_block_t
*
block
,
const
page_id_t
id
,
const
page_t
*
page
,
ulint
heap_no
,
ulint
heap_no
,
dict_index_t
*
index
,
dict_index_t
*
index
,
que_thr_t
*
thr
,
que_thr_t
*
thr
,
...
@@ -1348,11 +1345,11 @@ lock_rec_enqueue_waiting(
...
@@ -1348,11 +1345,11 @@ lock_rec_enqueue_waiting(
/* Enqueue the lock request that will wait to be granted, note that
/* Enqueue the lock request that will wait to be granted, note that
we already own the trx mutex. */
we already own the trx mutex. */
lock_t
*
lock
=
lock_rec_create
(
lock_t
*
lock
=
lock_rec_create
_low
(
#ifdef WITH_WSREP
#ifdef WITH_WSREP
c_lock
,
thr
,
c_lock
,
thr
,
#endif
#endif
type_mode
|
LOCK_WAIT
,
block
,
heap_no
,
index
,
trx
,
TRUE
);
type_mode
|
LOCK_WAIT
,
id
,
page
,
heap_no
,
index
,
trx
,
true
);
if
(
prdt
&&
type_mode
&
LOCK_PREDICATE
)
{
if
(
prdt
&&
type_mode
&
LOCK_PREDICATE
)
{
lock_prdt_set_prdt
(
lock
,
prdt
);
lock_prdt_set_prdt
(
lock
,
prdt
);
...
@@ -1415,7 +1412,8 @@ lock_rec_add_to_queue(
...
@@ -1415,7 +1412,8 @@ lock_rec_add_to_queue(
/*==================*/
/*==================*/
unsigned
type_mode
,
/*!< in: lock mode, wait, gap
unsigned
type_mode
,
/*!< in: lock mode, wait, gap
etc. flags */
etc. flags */
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
const
page_t
*
page
,
/*!< in: buffer block containing
the record */
the record */
ulint
heap_no
,
/*!< in: heap number of the record */
ulint
heap_no
,
/*!< in: heap number of the record */
dict_index_t
*
index
,
/*!< in: index of record */
dict_index_t
*
index
,
/*!< in: index of record */
...
@@ -1444,7 +1442,7 @@ lock_rec_add_to_queue(
...
@@ -1444,7 +1442,7 @@ lock_rec_add_to_queue(
:
LOCK_S
;
:
LOCK_S
;
const
lock_t
*
other_lock
const
lock_t
*
other_lock
=
lock_rec_other_has_expl_req
(
=
lock_rec_other_has_expl_req
(
mode
,
block
,
false
,
heap_no
,
trx
);
mode
,
id
,
false
,
heap_no
,
trx
);
#ifdef WITH_WSREP
#ifdef WITH_WSREP
if
(
UNIV_LIKELY_NULL
(
other_lock
)
&&
trx
->
is_wsrep
())
{
if
(
UNIV_LIKELY_NULL
(
other_lock
)
&&
trx
->
is_wsrep
())
{
/* Only BF transaction may be granted lock
/* Only BF transaction may be granted lock
...
@@ -1476,46 +1474,37 @@ lock_rec_add_to_queue(
...
@@ -1476,46 +1474,37 @@ lock_rec_add_to_queue(
type_mode
&=
~
(
LOCK_GAP
|
LOCK_REC_NOT_GAP
);
type_mode
&=
~
(
LOCK_GAP
|
LOCK_REC_NOT_GAP
);
}
}
lock_t
*
lock
;
if
(
type_mode
&
LOCK_WAIT
)
{
lock_t
*
first_lock
;
goto
create
;
}
else
if
(
lock_t
*
first_lock
=
/* Look for a waiting lock request on the same record or on a gap */
lock_sys
.
get_first
(
*
lock_hash_get
(
type_mode
),
id
))
{
for
(
lock_t
*
lock
=
first_lock
;;)
{
for
(
first_lock
=
lock
=
lock_sys
.
get_first
(
*
lock_hash_get
(
type_mode
),
block
->
page
.
id
());
lock
!=
NULL
;
lock
=
lock_rec_get_next_on_page
(
lock
))
{
if
(
lock
->
is_waiting
()
if
(
lock
->
is_waiting
()
&&
lock_rec_get_nth_bit
(
lock
,
heap_no
))
{
&&
lock_rec_get_nth_bit
(
lock
,
heap_no
))
{
goto
create
;
goto
create
;
}
}
if
(
!
(
lock
=
lock_rec_get_next_on_page
(
lock
)))
{
break
;
}
}
}
if
(
first_lock
&&
!
(
type_mode
&
LOCK_WAIT
))
{
/* Look for a similar record lock on the same page:
/* Look for a similar record lock on the same page:
if one is found and there are no waiting lock requests,
if one is found and there are no waiting lock requests,
we can just set the bit */
we can just set the bit */
if
(
lock_t
*
lock
=
lock_rec_find_similar_on_page
(
lock
=
lock_rec_find_similar_on_page
(
type_mode
,
heap_no
,
first_lock
,
trx
))
{
type_mode
,
heap_no
,
first_lock
,
trx
);
if
(
lock
!=
NULL
)
{
lock_rec_set_nth_bit
(
lock
,
heap_no
);
lock_rec_set_nth_bit
(
lock
,
heap_no
);
return
;
return
;
}
}
}
}
create
:
create
:
lock_rec_create
(
lock_rec_create
_low
(
#ifdef WITH_WSREP
#ifdef WITH_WSREP
NULL
,
NULL
,
NULL
,
NULL
,
#endif
#endif
type_mode
,
block
,
heap_no
,
index
,
trx
,
caller_owns_trx_mutex
);
type_mode
,
id
,
page
,
heap_no
,
index
,
trx
,
caller_owns_trx_mutex
);
}
}
/*********************************************************************//**
/*********************************************************************//**
...
@@ -1574,13 +1563,13 @@ lock_rec_lock(
...
@@ -1574,13 +1563,13 @@ lock_rec_lock(
lock_rec_get_n_bits
(
lock
)
<=
heap_no
)
lock_rec_get_n_bits
(
lock
)
<=
heap_no
)
{
{
/* Do nothing if the trx already has a strong enough lock on rec */
/* Do nothing if the trx already has a strong enough lock on rec */
if
(
!
lock_rec_has_expl
(
mode
,
block
,
heap_no
,
trx
))
if
(
!
lock_rec_has_expl
(
mode
,
id
,
heap_no
,
trx
))
{
{
if
(
if
(
#ifdef WITH_WSREP
#ifdef WITH_WSREP
lock_t
*
c_lock
=
lock_t
*
c_lock
=
#endif
#endif
lock_rec_other_has_conflicting
(
mode
,
block
,
heap_no
,
trx
))
lock_rec_other_has_conflicting
(
mode
,
id
,
heap_no
,
trx
))
{
{
/*
/*
If another transaction has a non-gap conflicting
If another transaction has a non-gap conflicting
...
@@ -1591,12 +1580,13 @@ lock_rec_lock(
...
@@ -1591,12 +1580,13 @@ lock_rec_lock(
#ifdef WITH_WSREP
#ifdef WITH_WSREP
c_lock
,
c_lock
,
#endif
/* WITH_WSREP */
#endif
/* WITH_WSREP */
mode
,
block
,
heap_no
,
index
,
thr
,
NULL
);
mode
,
id
,
block
->
frame
,
heap_no
,
index
,
thr
,
nullptr
);
}
}
else
if
(
!
impl
)
else
if
(
!
impl
)
{
{
/* Set the requested lock on the record. */
/* Set the requested lock on the record. */
lock_rec_add_to_queue
(
mode
,
block
,
heap_no
,
index
,
trx
,
true
);
lock_rec_add_to_queue
(
mode
,
id
,
block
->
frame
,
heap_no
,
index
,
trx
,
true
);
err
=
DB_SUCCESS_LOCKED_REC
;
err
=
DB_SUCCESS_LOCKED_REC
;
}
}
}
}
...
@@ -2025,16 +2015,11 @@ static void lock_rec_free_all_from_discard_page_low(const page_id_t id,
...
@@ -2025,16 +2015,11 @@ static void lock_rec_free_all_from_discard_page_low(const page_id_t id,
}
}
}
}
/*************************************************************//**
/** Remove record locks for an index page which is discarded. This
Removes record lock objects set on an index page which is discarded. This
function does not move locks, or check for waiting locks, therefore the
function does not move locks, or check for waiting locks, therefore the
lock bitmaps must already be reset when this function is called. */
lock bitmaps must already be reset when this function is called. */
void
void
lock_rec_free_all_from_discard_page
(
const
page_id_t
page_id
)
lock_rec_free_all_from_discard_page
(
/*================================*/
const
buf_block_t
*
block
)
/*!< in: page to be discarded */
{
{
const
page_id_t
page_id
(
block
->
page
.
id
());
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
rec_hash
);
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
rec_hash
);
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
prdt_hash
);
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
prdt_hash
);
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
prdt_page_hash
);
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
prdt_page_hash
);
...
@@ -2050,11 +2035,10 @@ void
...
@@ -2050,11 +2035,10 @@ void
lock_rec_reset_and_release_wait_low
(
lock_rec_reset_and_release_wait_low
(
/*================================*/
/*================================*/
hash_table_t
*
hash
,
/*!< in: hash table */
hash_table_t
*
hash
,
/*!< in: hash table */
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
the record */
ulint
heap_no
)
/*!< in: heap number of record */
ulint
heap_no
)
/*!< in: heap number of record */
{
{
for
(
lock_t
*
lock
=
lock_rec_get_first
(
hash
,
block
,
heap_no
);
lock
;
for
(
lock_t
*
lock
=
lock_rec_get_first
(
hash
,
id
,
heap_no
);
lock
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
if
(
lock
->
is_waiting
())
if
(
lock
->
is_waiting
())
lock_rec_cancel
(
lock
);
lock_rec_cancel
(
lock
);
...
@@ -2069,17 +2053,16 @@ static
...
@@ -2069,17 +2053,16 @@ static
void
void
lock_rec_reset_and_release_wait
(
lock_rec_reset_and_release_wait
(
/*============================*/
/*============================*/
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
the record */
ulint
heap_no
)
/*!< in: heap number of record */
ulint
heap_no
)
/*!< in: heap number of record */
{
{
lock_rec_reset_and_release_wait_low
(
lock_rec_reset_and_release_wait_low
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
lock_rec_reset_and_release_wait_low
(
lock_rec_reset_and_release_wait_low
(
&
lock_sys
.
prdt_hash
,
block
,
PAGE_HEAP_NO_INFIMUM
);
&
lock_sys
.
prdt_hash
,
id
,
PAGE_HEAP_NO_INFIMUM
);
lock_rec_reset_and_release_wait_low
(
lock_rec_reset_and_release_wait_low
(
&
lock_sys
.
prdt_page_hash
,
block
,
PAGE_HEAP_NO_INFIMUM
);
&
lock_sys
.
prdt_page_hash
,
id
,
PAGE_HEAP_NO_INFIMUM
);
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -2091,19 +2074,18 @@ static
...
@@ -2091,19 +2074,18 @@ static
void
void
lock_rec_inherit_to_gap
(
lock_rec_inherit_to_gap
(
/*====================*/
/*====================*/
const
buf_block_t
*
heir_block
,
/*!< in: block
containing the
const
page_id_t
heir
,
/*!< in: page
containing the
record which inherits */
record which inherits */
const
buf_block_t
*
block
,
/*!< in: block
containing the
const
page_id_t
id
,
/*!< in: page
containing the
record from which inherited;
record from which inherited;
does NOT reset the locks on
does NOT reset the locks on
this record */
this record */
const
page_t
*
heir_page
,
/*!< in: heir page frame */
ulint
heir_heap_no
,
/*!< in: heap_no of the
ulint
heir_heap_no
,
/*!< in: heap_no of the
inheriting record */
inheriting record */
ulint
heap_no
)
/*!< in: heap_no of the
ulint
heap_no
)
/*!< in: heap_no of the
donating record */
donating record */
{
{
lock_t
*
lock
;
lock_sys
.
mutex_assert_locked
();
lock_sys
.
mutex_assert_locked
();
/* At READ UNCOMMITTED or READ COMMITTED isolation level,
/* At READ UNCOMMITTED or READ COMMITTED isolation level,
...
@@ -2112,7 +2094,7 @@ lock_rec_inherit_to_gap(
...
@@ -2112,7 +2094,7 @@ lock_rec_inherit_to_gap(
DO want S-locks/X-locks(taken for replace) set by a consistency
DO want S-locks/X-locks(taken for replace) set by a consistency
constraint to be inherited also then. */
constraint to be inherited also then. */
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
for
(
lock
_t
*
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
...
@@ -2121,7 +2103,8 @@ lock_rec_inherit_to_gap(
...
@@ -2121,7 +2103,8 @@ lock_rec_inherit_to_gap(
||
lock
->
mode
()
!=
||
lock
->
mode
()
!=
(
lock
->
trx
->
duplicates
?
LOCK_S
:
LOCK_X
)))
{
(
lock
->
trx
->
duplicates
?
LOCK_S
:
LOCK_X
)))
{
lock_rec_add_to_queue
(
LOCK_GAP
|
lock
->
mode
(),
lock_rec_add_to_queue
(
LOCK_GAP
|
lock
->
mode
(),
heir_block
,
heir_heap_no
,
heir
,
heir_page
,
heir_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
lock
->
index
,
lock
->
trx
,
false
);
}
}
}
}
...
@@ -2143,26 +2126,16 @@ lock_rec_inherit_to_gap_if_gap_lock(
...
@@ -2143,26 +2126,16 @@ lock_rec_inherit_to_gap_if_gap_lock(
does NOT reset the locks
does NOT reset the locks
on this record */
on this record */
{
{
lock_t
*
lock
;
const
page_id_t
id
{
block
->
page
.
id
()};
LockMutexGuard
g
;
lock_sys
.
mutex_lock
();
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
{
if
(
!
lock
->
is_insert_intention
()
&&
(
heap_no
==
PAGE_HEAP_NO_SUPREMUM
||
!
lock
->
is_record_not_gap
())
&&
!
lock_table_has
(
lock
->
trx
,
lock
->
index
->
table
,
LOCK_X
))
{
lock_rec_add_to_queue
(
LOCK_GAP
|
lock
->
mode
(),
block
,
heir_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
}
}
lock_sys
.
mutex_unlock
();
for
(
lock_t
*
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
lock
;
lock
=
lock_rec_get_next
(
heap_no
,
lock
))
if
(
!
lock
->
is_insert_intention
()
&&
(
heap_no
==
PAGE_HEAP_NO_SUPREMUM
||
!
lock
->
is_record_not_gap
())
&&
!
lock_table_has
(
lock
->
trx
,
lock
->
index
->
table
,
LOCK_X
))
lock_rec_add_to_queue
(
LOCK_GAP
|
lock
->
mode
(),
id
,
block
->
frame
,
heir_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -2173,9 +2146,9 @@ void
...
@@ -2173,9 +2146,9 @@ void
lock_rec_move_low
(
lock_rec_move_low
(
/*==============*/
/*==============*/
hash_table_t
*
lock_hash
,
/*!< in: hash table to use */
hash_table_t
*
lock_hash
,
/*!< in: hash table to use */
const
buf_block_t
*
receiver
,
/*!< in: buffer block containing
const
buf_block_t
&
receiver
,
/*!< in: buffer block containing
the receiving record */
the receiving record */
const
buf_block_t
*
donator
,
/*!< in: buffer block containing
const
page_id_t
donator_id
,
/*!< in: page identifier of
the donating record */
the donating record */
ulint
receiver_heap_no
,
/*!< in: heap_no of the record
ulint
receiver_heap_no
,
/*!< in: heap_no of the record
which gets the locks; there
which gets the locks; there
...
@@ -2184,18 +2157,17 @@ lock_rec_move_low(
...
@@ -2184,18 +2157,17 @@ lock_rec_move_low(
ulint
donator_heap_no
)
/*!< in: heap_no of the record
ulint
donator_heap_no
)
/*!< in: heap_no of the record
which gives the locks */
which gives the locks */
{
{
lock_t
*
lock
;
const
page_id_t
receiver_id
{
receiver
.
page
.
id
()}
;
lock_sys
.
mutex_assert_locked
();
lock_sys
.
mutex_assert_locked
();
/* If the lock is predicate lock, it resides on INFIMUM record */
/* If the lock is predicate lock, it resides on INFIMUM record */
ut_ad
(
lock_rec_get_first
(
ut_ad
(
!
lock_rec_get_first
(
lock_hash
,
receiver_id
,
receiver_heap_no
)
lock_hash
,
receiver
,
receiver_heap_no
)
==
NULL
||
lock_hash
==
&
lock_sys
.
prdt_hash
||
lock_hash
==
&
lock_sys
.
prdt_hash
||
lock_hash
==
&
lock_sys
.
prdt_page_hash
);
||
lock_hash
==
&
lock_sys
.
prdt_page_hash
);
for
(
lock
=
lock_rec_get_first
(
lock_hash
,
for
(
lock
_t
*
lock
=
donator
,
donator_heap_no
);
lock_rec_get_first
(
lock_hash
,
donator_id
,
donator_heap_no
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
donator_heap_no
,
lock
))
{
lock
=
lock_rec_get_next
(
donator_heap_no
,
lock
))
{
...
@@ -2208,14 +2180,15 @@ lock_rec_move_low(
...
@@ -2208,14 +2180,15 @@ lock_rec_move_low(
}
}
/* Note that we FIRST reset the bit, and then set the lock:
/* Note that we FIRST reset the bit, and then set the lock:
the function works also if donator
== receiver
*/
the function works also if donator
_id == receiver_id
*/
lock_rec_add_to_queue
(
type_mode
,
receiver
,
receiver_heap_no
,
lock_rec_add_to_queue
(
type_mode
,
receiver_id
,
receiver
.
frame
,
receiver_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
lock
->
index
,
lock
->
trx
,
false
);
}
}
ut_ad
(
!
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
ut_ad
(
!
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
donator
,
donator_heap_no
));
donator
_id
,
donator_heap_no
));
}
}
/** Move all the granted locks to the front of the given lock list.
/** Move all the granted locks to the front of the given lock list.
...
@@ -2258,9 +2231,9 @@ UNIV_INLINE
...
@@ -2258,9 +2231,9 @@ UNIV_INLINE
void
void
lock_rec_move
(
lock_rec_move
(
/*==========*/
/*==========*/
const
buf_block_t
*
receiver
,
/*!< in: buffer block containing
const
buf_block_t
&
receiver
,
/*!< in: buffer block containing
the receiving record */
the receiving record */
const
buf_block_t
*
donator
,
/*!< in: buffer block containing
const
page_id_t
donator_id
,
/*!< in: page identifier of
the donating record */
the donating record */
ulint
receiver_heap_no
,
/*!< in: heap_no of the record
ulint
receiver_heap_no
,
/*!< in: heap_no of the record
which gets the locks; there
which gets the locks; there
...
@@ -2269,7 +2242,7 @@ lock_rec_move(
...
@@ -2269,7 +2242,7 @@ lock_rec_move(
ulint
donator_heap_no
)
/*!< in: heap_no of the record
ulint
donator_heap_no
)
/*!< in: heap_no of the record
which gives the locks */
which gives the locks */
{
{
lock_rec_move_low
(
&
lock_sys
.
rec_hash
,
receiver
,
donator
,
lock_rec_move_low
(
&
lock_sys
.
rec_hash
,
receiver
,
donator_id
,
receiver_heap_no
,
donator_heap_no
);
receiver_heap_no
,
donator_heap_no
);
}
}
...
@@ -2290,11 +2263,12 @@ lock_move_reorganize_page(
...
@@ -2290,11 +2263,12 @@ lock_move_reorganize_page(
UT_LIST_BASE_NODE_T
(
lock_t
)
old_locks
;
UT_LIST_BASE_NODE_T
(
lock_t
)
old_locks
;
mem_heap_t
*
heap
=
NULL
;
mem_heap_t
*
heap
=
NULL
;
ulint
comp
;
ulint
comp
;
const
page_id_t
id
{
block
->
page
.
id
()};
lock_sys
.
mutex_lock
();
lock_sys
.
mutex_lock
();
/* FIXME: This needs to deal with predicate lock too */
/* FIXME: This needs to deal with predicate lock too */
lock
=
lock_sys
.
get_first
(
block
->
page
.
id
()
);
lock
=
lock_sys
.
get_first
(
id
);
if
(
lock
==
NULL
)
{
if
(
lock
==
NULL
)
{
lock_sys
.
mutex_unlock
();
lock_sys
.
mutex_unlock
();
...
@@ -2380,7 +2354,8 @@ lock_move_reorganize_page(
...
@@ -2380,7 +2354,8 @@ lock_move_reorganize_page(
small for the new heap number! */
small for the new heap number! */
lock_rec_add_to_queue
(
lock_rec_add_to_queue
(
lock
->
type_mode
,
block
,
new_heap_no
,
lock
->
type_mode
,
id
,
block
->
frame
,
new_heap_no
,
lock
->
index
,
lock
->
trx
,
FALSE
);
lock
->
index
,
lock
->
trx
,
FALSE
);
}
}
...
@@ -2416,105 +2391,97 @@ lock_move_rec_list_end(
...
@@ -2416,105 +2391,97 @@ lock_move_rec_list_end(
const
rec_t
*
rec
)
/*!< in: record on page: this
const
rec_t
*
rec
)
/*!< in: record on page: this
is the first record moved */
is the first record moved */
{
{
lock_t
*
lock
;
const
ulint
comp
=
page_rec_is_comp
(
rec
);
const
ulint
comp
=
page_rec_is_comp
(
rec
);
ut_ad
(
buf_block_get_frame
(
block
)
==
page_align
(
rec
));
ut_ad
(
block
->
frame
==
page_align
(
rec
));
ut_ad
(
comp
==
page_is_comp
(
buf_block_get_frame
(
new_block
)
));
ut_ad
(
comp
==
page_is_comp
(
new_block
->
frame
));
lock_sys
.
mutex_lock
();
const
page_id_t
id
{
block
->
page
.
id
()};
const
page_id_t
new_id
{
new_block
->
page
.
id
()};
{
LockMutexGuard
g
;
/* Note: when we move locks from record to record, waiting locks
/* Note: when we move locks from record to record, waiting locks
and possible granted gap type locks behind them are enqueued in
and possible granted gap type locks behind them are enqueued in
the original order, because new elements are inserted to a hash
the original order, because new elements are inserted to a hash
table to the end of the hash chain, and lock_rec_add_to_queue
table to the end of the hash chain, and lock_rec_add_to_queue
does not reuse locks if there are waiters in the queue. */
does not reuse locks if there are waiters in the queue. */
for
(
lock_t
*
lock
=
lock_sys
.
get_first
(
id
);
lock
;
lock
=
lock_rec_get_next_on_page
(
lock
))
{
const
rec_t
*
rec1
=
rec
;
const
rec_t
*
rec2
;
const
auto
type_mode
=
lock
->
type_mode
;
for
(
lock
=
lock_sys
.
get_first
(
block
->
page
.
id
());
if
(
comp
)
lock
;
{
lock
=
lock_rec_get_next_on_page
(
lock
))
{
if
(
page_offset
(
rec1
)
==
PAGE_NEW_INFIMUM
)
const
rec_t
*
rec1
=
rec
;
rec1
=
page_rec_get_next_low
(
rec1
,
TRUE
);
const
rec_t
*
rec2
;
rec2
=
page_rec_get_next_low
(
new_block
->
frame
+
PAGE_NEW_INFIMUM
,
TRUE
);
const
auto
type_mode
=
lock
->
type_mode
;
if
(
comp
)
{
if
(
page_offset
(
rec1
)
==
PAGE_NEW_INFIMUM
)
{
rec1
=
page_rec_get_next_low
(
rec1
,
TRUE
);
}
rec2
=
page_rec_get_next_low
(
buf_block_get_frame
(
new_block
)
+
PAGE_NEW_INFIMUM
,
TRUE
);
}
else
{
if
(
page_offset
(
rec1
)
==
PAGE_OLD_INFIMUM
)
{
rec1
=
page_rec_get_next_low
(
rec1
,
FALSE
);
}
}
else
rec2
=
page_rec_get_next_low
(
{
buf_block_get_frame
(
new_block
)
if
(
page_offset
(
rec1
)
==
PAGE_OLD_INFIMUM
)
+
PAGE_OLD_INFIMUM
,
FALSE
);
rec1
=
page_rec_get_next_low
(
rec1
,
FALSE
);
rec2
=
page_rec_get_next_low
(
new_block
->
frame
+
PAGE_OLD_INFIMUM
,
FALSE
);
}
}
/* 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 */
for
(;;)
for
(;;)
{
{
ut_ad
(
page_rec_is_metadata
(
rec1
)
ut_ad
(
page_rec_is_metadata
(
rec1
)
==
page_rec_is_metadata
(
rec2
));
==
page_rec_is_metadata
(
rec2
));
ut_d
(
const
rec_t
*
const
orec
=
rec1
);
ut_d
(
const
rec_t
*
const
orec
=
rec1
);
ulint
rec1_heap_no
;
ulint
rec1_heap_no
;
ulint
rec2_heap_no
;
ulint
rec2_heap_no
;
if
(
comp
)
{
if
(
comp
)
rec1_heap_no
=
rec_get_heap_no_new
(
rec1
);
{
rec1_heap_no
=
rec_get_heap_no_new
(
rec1
);
if
(
rec1_heap_no
==
PAGE_HEAP_NO_SUPREMUM
)
{
if
(
rec1_heap_no
==
PAGE_HEAP_NO_SUPREMUM
)
break
;
break
;
}
rec2_heap_no
=
rec_get_heap_no_new
(
rec2
);
rec2_heap_no
=
rec_get_heap_no_new
(
rec2
);
rec1
=
page_rec_get_next_low
(
rec1
,
TRUE
);
rec1
=
page_rec_get_next_low
(
rec1
,
TRUE
);
rec2
=
page_rec_get_next_low
(
rec2
,
TRUE
);
rec2
=
page_rec_get_next_low
(
rec2
,
TRUE
);
}
else
{
rec1_heap_no
=
rec_get_heap_no_old
(
rec1
);
if
(
rec1_heap_no
==
PAGE_HEAP_NO_SUPREMUM
)
{
break
;
}
}
else
{
rec1_heap_no
=
rec_get_heap_no_old
(
rec1
);
rec2_heap_no
=
rec_get_heap_no_old
(
rec2
);
if
(
rec1_heap_no
==
PAGE_HEAP_NO_SUPREMUM
)
break
;
ut_ad
(
rec_get_data_size_old
(
rec1
)
rec2_heap_no
=
rec_get_heap_no_old
(
rec2
);
==
rec_get_data_size_old
(
rec2
));
ut_ad
(
!
memcmp
(
rec1
,
rec2
,
ut_ad
(
rec_get_data_size_old
(
rec1
)
==
rec_get_data_size_old
(
rec2
));
rec_get_data_size_old
(
rec1
)));
ut_ad
(
!
memcmp
(
rec1
,
rec2
,
rec_get_data_size_old
(
rec1
)));
rec1
=
page_rec_get_next_low
(
rec1
,
FALSE
);
rec1
=
page_rec_get_next_low
(
rec1
,
FALSE
);
rec2
=
page_rec_get_next_low
(
rec2
,
FALSE
);
rec2
=
page_rec_get_next_low
(
rec2
,
FALSE
);
}
}
if
(
rec1_heap_no
<
lock
->
un_member
.
rec_lock
.
n_bits
if
(
rec1_heap_no
<
lock
->
un_member
.
rec_lock
.
n_bits
&&
&&
lock_rec_reset_nth_bit
(
lock
,
rec1_heap_no
))
{
lock_rec_reset_nth_bit
(
lock
,
rec1_heap_no
))
{
ut_ad
(
!
page_rec_is_metadata
(
orec
));
ut_ad
(
!
page_rec_is_metadata
(
orec
));
if
(
type_mode
&
LOCK_WAIT
)
{
if
(
type_mode
&
LOCK_WAIT
)
ut_ad
(
lock
->
trx
->
lock
.
wait_lock
==
lock
);
{
lock
->
type_mode
&=
~
LOCK_WAIT
;
ut_ad
(
lock
->
trx
->
lock
.
wait_lock
==
lock
);
lock
->
type_mode
&=
~
LOCK_WAIT
;
}
}
lock_rec_add_to_queue
(
lock_rec_add_to_queue
(
type_mode
,
new_id
,
new_block
->
frame
,
type_mode
,
new_block
,
rec2_heap_no
,
rec2_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
lock
->
index
,
lock
->
trx
,
FALSE
);
}
}
}
}
}
}
}
lock_sys
.
mutex_unlock
();
#ifdef UNIV_DEBUG_LOCK_VALIDATE
#ifdef UNIV_DEBUG_LOCK_VALIDATE
if
(
fil_space_t
*
space
=
fil_space_t
::
get
(
page_id
.
space
()))
{
if
(
fil_space_t
*
space
=
fil_space_t
::
get
(
id
.
space
()))
{
const
bool
is_latched
{
space
->
is_latched
()};
const
bool
is_latched
{
space
->
is_latched
()};
ut_ad
(
lock_rec_validate_page
(
block
,
is_latched
));
ut_ad
(
lock_rec_validate_page
(
block
,
is_latched
));
ut_ad
(
lock_rec_validate_page
(
new_block
,
is_latched
));
ut_ad
(
lock_rec_validate_page
(
new_block
,
is_latched
));
...
@@ -2541,96 +2508,89 @@ lock_move_rec_list_start(
...
@@ -2541,96 +2508,89 @@ lock_move_rec_list_start(
before the records
before the records
were copied */
were copied */
{
{
lock_t
*
lock
;
const
ulint
comp
=
page_rec_is_comp
(
rec
);
const
ulint
comp
=
page_rec_is_comp
(
rec
);
ut_ad
(
block
->
frame
==
page_align
(
rec
));
ut_ad
(
block
->
frame
==
page_align
(
rec
));
ut_ad
(
comp
==
page_is_comp
(
new_block
->
frame
));
ut_ad
(
new_block
->
frame
==
page_align
(
old_end
));
ut_ad
(
new_block
->
frame
==
page_align
(
old_end
));
ut_ad
(
comp
==
page_rec_is_comp
(
old_end
));
ut_ad
(
!
page_rec_is_metadata
(
rec
));
ut_ad
(
!
page_rec_is_metadata
(
rec
));
const
page_id_t
id
{
block
->
page
.
id
()};
const
page_id_t
new_id
{
new_block
->
page
.
id
()};
lock_sys
.
mutex_lock
();
{
LockMutexGuard
g
;
for
(
lock
=
lock_sys
.
get_first
(
block
->
page
.
id
())
;
for
(
lock_t
*
lock
=
lock_sys
.
get_first
(
id
);
lock
;
lock
;
lock
=
lock_rec_get_next_on_page
(
lock
))
lock
=
lock_rec_get_next_on_page
(
lock
))
{
{
const
rec_t
*
rec1
;
const
rec_t
*
rec1
;
const
rec_t
*
rec2
;
const
rec_t
*
rec2
;
const
auto
type_mode
=
lock
->
type_mode
;
const
auto
type_mode
=
lock
->
type_mode
;
if
(
comp
)
{
if
(
comp
)
rec1
=
page_rec_get_next_low
(
{
buf_block_get_frame
(
block
)
rec1
=
page_rec_get_next_low
(
block
->
frame
+
PAGE_NEW_INFIMUM
,
TRUE
);
+
PAGE_NEW_INFIMUM
,
TRUE
);
rec2
=
page_rec_get_next_low
(
old_end
,
TRUE
);
rec2
=
page_rec_get_next_low
(
old_end
,
TRUE
);
}
}
else
{
else
rec1
=
page_rec_get_next_low
(
{
buf_block_get_frame
(
block
)
rec1
=
page_rec_get_next_low
(
block
->
frame
+
PAGE_OLD_INFIMUM
,
FALSE
);
+
PAGE_OLD_INFIMUM
,
FALSE
);
rec2
=
page_rec_get_next_low
(
old_end
,
FALSE
);
rec2
=
page_rec_get_next_low
(
old_end
,
FALSE
);
}
}
/* 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
(
rec1
!=
rec
)
{
while
(
rec1
!=
rec
)
ut_ad
(
page_rec_is_metadata
(
rec1
)
{
==
page_rec_is_metadata
(
rec2
));
ut_ad
(
page_rec_is_metadata
(
rec1
)
==
page_rec_is_metadata
(
rec2
));
ut_d
(
const
rec_t
*
const
prev
=
rec1
);
ut_d
(
const
rec_t
*
const
prev
=
rec1
);
ulint
rec1_heap_no
;
ulint
rec1_heap_no
;
ulint
rec2_heap_no
;
ulint
rec2_heap_no
;
if
(
comp
)
{
if
(
comp
)
rec1_heap_no
=
rec_get_heap_no_new
(
rec1
);
{
rec2_heap_no
=
rec_get_heap_no_new
(
rec2
);
rec1_heap_no
=
rec_get_heap_no_new
(
rec1
);
rec2_heap_no
=
rec_get_heap_no_new
(
rec2
);
rec1
=
page_rec_get_next_low
(
rec1
,
TRUE
);
rec1
=
page_rec_get_next_low
(
rec1
,
TRUE
);
rec2
=
page_rec_get_next_low
(
rec2
,
TRUE
);
rec2
=
page_rec_get_next_low
(
rec2
,
TRUE
);
}
else
{
}
rec1_heap_no
=
rec_get_heap_no_old
(
rec1
);
else
rec2_heap_no
=
rec_get_heap_no_old
(
rec2
);
{
rec1_heap_no
=
rec_get_heap_no_old
(
rec1
);
rec2_heap_no
=
rec_get_heap_no_old
(
rec2
);
ut_ad
(
!
memcmp
(
rec1
,
rec2
,
ut_ad
(
!
memcmp
(
rec1
,
rec2
,
rec_get_data_size_old
(
rec2
)));
rec_get_data_size_old
(
rec2
)));
rec1
=
page_rec_get_next_low
(
rec1
,
FALSE
);
rec1
=
page_rec_get_next_low
(
rec1
,
FALSE
);
rec2
=
page_rec_get_next_low
(
rec2
,
FALSE
);
rec2
=
page_rec_get_next_low
(
rec2
,
FALSE
);
}
}
if
(
rec1_heap_no
<
lock
->
un_member
.
rec_lock
.
n_bits
if
(
rec1_heap_no
<
lock
->
un_member
.
rec_lock
.
n_bits
&&
&&
lock_rec_reset_nth_bit
(
lock
,
rec1_heap_no
))
{
lock_rec_reset_nth_bit
(
lock
,
rec1_heap_no
))
{
ut_ad
(
!
page_rec_is_metadata
(
prev
));
ut_ad
(
!
page_rec_is_metadata
(
prev
));
if
(
type_mode
&
LOCK_WAIT
)
{
if
(
type_mode
&
LOCK_WAIT
)
ut_ad
(
lock
->
trx
->
lock
.
wait_lock
==
lock
);
{
lock
->
type_mode
&=
~
LOCK_WAIT
;
ut_ad
(
lock
->
trx
->
lock
.
wait_lock
==
lock
);
lock
->
type_mode
&=
~
LOCK_WAIT
;
}
}
lock_rec_add_to_queue
(
lock_rec_add_to_queue
(
type_mode
,
new_id
,
new_block
->
frame
,
type_mode
,
new_block
,
rec2_heap_no
,
rec2_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
lock
->
index
,
lock
->
trx
,
FALSE
);
}
}
}
}
#ifdef UNIV_DEBUG
#ifdef UNIV_DEBUG
if
(
page_rec_is_supremum
(
rec
))
{
if
(
page_rec_is_supremum
(
rec
))
ulint
i
;
for
(
auto
i
=
lock_rec_get_n_bits
(
lock
);
--
i
>
PAGE_HEAP_NO_USER_LOW
;
)
ut_ad
(
!
lock_rec_get_nth_bit
(
lock
,
i
));
for
(
i
=
PAGE_HEAP_NO_USER_LOW
;
i
<
lock_rec_get_n_bits
(
lock
);
i
++
)
{
if
(
lock_rec_get_nth_bit
(
lock
,
i
))
{
ib
::
fatal
()
<<
"lock_move_rec_list_start():"
<<
i
<<
" not moved in "
<<
(
void
*
)
lock
;
}
}
}
#endif
/* UNIV_DEBUG */
#endif
/* UNIV_DEBUG */
}
}
}
lock_sys
.
mutex_unlock
();
#ifdef UNIV_DEBUG_LOCK_VALIDATE
#ifdef UNIV_DEBUG_LOCK_VALIDATE
ut_ad
(
lock_rec_validate_page
(
block
));
ut_ad
(
lock_rec_validate_page
(
block
));
...
@@ -2650,72 +2610,70 @@ lock_rtr_move_rec_list(
...
@@ -2650,72 +2610,70 @@ lock_rtr_move_rec_list(
moved */
moved */
ulint
num_move
)
/*!< in: num of rec to move */
ulint
num_move
)
/*!< in: num of rec to move */
{
{
lock_t
*
lock
;
if
(
!
num_move
)
ulint
comp
;
if
(
!
num_move
)
{
return
;
return
;
}
comp
=
page_rec_is_comp
(
rec_move
[
0
].
old_rec
);
const
ulint
comp
=
page_rec_is_comp
(
rec_move
[
0
].
old_rec
);
ut_ad
(
block
->
frame
==
page_align
(
rec_move
[
0
].
old_rec
));
ut_ad
(
block
->
frame
==
page_align
(
rec_move
[
0
].
old_rec
));
ut_ad
(
new_block
->
frame
==
page_align
(
rec_move
[
0
].
new_rec
));
ut_ad
(
new_block
->
frame
==
page_align
(
rec_move
[
0
].
new_rec
));
ut_ad
(
comp
==
page_rec_is_comp
(
rec_move
[
0
].
new_rec
));
ut_ad
(
comp
==
page_rec_is_comp
(
rec_move
[
0
].
new_rec
));
const
page_id_t
id
{
block
->
page
.
id
()};
const
page_id_t
new_id
{
new_block
->
page
.
id
()};
lock_sys
.
mutex_lock
();
{
LockMutexGuard
g
;
for
(
lock
=
lock_sys
.
get_first
(
block
->
page
.
id
());
for
(
lock_t
*
lock
=
lock_sys
.
get_first
(
id
);
lock
;
lock
;
lock
=
lock_rec_get_next_on_page
(
lock
))
lock
=
lock_rec_get_next_on_page
(
lock
))
{
{
ulint
moved
=
0
;
const
rec_t
*
rec1
;
const
rec_t
*
rec1
;
const
rec_t
*
rec2
;
const
rec_t
*
rec2
;
const
auto
type_mode
=
lock
->
type_mode
;
const
auto
type_mode
=
lock
->
type_mode
;
/* 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
(
moved
<
num_move
)
{
for
(
ulint
moved
=
0
;
moved
<
num_move
;
moved
++
)
{
ulint
rec1_heap_no
;
ulint
rec1_heap_no
;
ulint
rec2_heap_no
;
ulint
rec2_heap_no
;
rec1
=
rec_move
[
moved
].
old_rec
;
rec1
=
rec_move
[
moved
].
old_rec
;
rec2
=
rec_move
[
moved
].
new_rec
;
rec2
=
rec_move
[
moved
].
new_rec
;
ut_ad
(
!
page_rec_is_metadata
(
rec1
));
ut_ad
(
!
page_rec_is_metadata
(
rec1
));
ut_ad
(
!
page_rec_is_metadata
(
rec2
));
ut_ad
(
!
page_rec_is_metadata
(
rec2
));
if
(
comp
)
{
if
(
comp
)
rec1_heap_no
=
rec_get_heap_no_new
(
rec1
);
{
rec2_heap_no
=
rec_get_heap_no_new
(
rec2
);
rec1_heap_no
=
rec_get_heap_no_new
(
rec1
);
rec2_heap_no
=
rec_get_heap_no_new
(
rec2
);
}
else
{
}
rec1_heap_no
=
rec_get_heap_no_old
(
rec1
);
else
rec2_heap_no
=
rec_get_heap_no_old
(
rec2
);
{
rec1_heap_no
=
rec_get_heap_no_old
(
rec1
);
rec2_heap_no
=
rec_get_heap_no_old
(
rec2
);
ut_ad
(
!
memcmp
(
rec1
,
rec2
,
ut_ad
(
!
memcmp
(
rec1
,
rec2
,
rec_get_data_size_old
(
rec2
)));
rec_get_data_size_old
(
rec2
)));
}
}
if
(
rec1_heap_no
<
lock
->
un_member
.
rec_lock
.
n_bits
if
(
rec1_heap_no
<
lock
->
un_member
.
rec_lock
.
n_bits
&&
&&
lock_rec_reset_nth_bit
(
lock
,
rec1_heap_no
))
{
lock_rec_reset_nth_bit
(
lock
,
rec1_heap_no
))
if
(
type_mode
&
LOCK_WAIT
)
{
{
ut_ad
(
lock
->
trx
->
lock
.
wait_lock
==
lock
);
if
(
type_mode
&
LOCK_WAIT
)
lock
->
type_mode
&=
~
LOCK_WAIT
;
{
ut_ad
(
lock
->
trx
->
lock
.
wait_lock
==
lock
);
lock
->
type_mode
&=
~
LOCK_WAIT
;
}
}
lock_rec_add_to_queue
(
lock_rec_add_to_queue
(
type_mode
,
new_id
,
new_block
->
frame
,
type_mode
,
new_block
,
rec2_heap_no
,
rec2_heap_no
,
lock
->
index
,
lock
->
trx
,
false
);
lock
->
index
,
lock
->
trx
,
FALSE
);
rec_move
[
moved
].
moved
=
true
;
rec_move
[
moved
].
moved
=
true
;
}
}
}
moved
++
;
}
}
}
}
lock_sys
.
mutex_unlock
();
#ifdef UNIV_DEBUG_LOCK_VALIDATE
#ifdef UNIV_DEBUG_LOCK_VALIDATE
ut_ad
(
lock_rec_validate_page
(
block
));
ut_ad
(
lock_rec_validate_page
(
block
));
...
@@ -2729,23 +2687,20 @@ lock_update_split_right(
...
@@ -2729,23 +2687,20 @@ lock_update_split_right(
const
buf_block_t
*
right_block
,
/*!< in: right page */
const
buf_block_t
*
right_block
,
/*!< in: right page */
const
buf_block_t
*
left_block
)
/*!< in: left page */
const
buf_block_t
*
left_block
)
/*!< in: left page */
{
{
ulint
heap_no
=
lock_get_min_heap_no
(
right_block
);
const
ulint
h
=
lock_get_min_heap_no
(
right_block
);
const
page_id_t
l
{
left_block
->
page
.
id
()};
const
page_id_t
r
{
right_block
->
page
.
id
()};
lock_sys
.
mutex_lock
()
;
LockMutexGuard
g
;
/* 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
(
right_block
,
left_block
,
lock_rec_move
(
*
right_block
,
l
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
/* 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
(
l
,
r
,
left_block
->
frame
,
PAGE_HEAP_NO_SUPREMUM
,
h
);
lock_rec_inherit_to_gap
(
left_block
,
right_block
,
PAGE_HEAP_NO_SUPREMUM
,
heap_no
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -2765,13 +2720,15 @@ lock_update_merge_right(
...
@@ -2765,13 +2720,15 @@ lock_update_merge_right(
{
{
ut_ad
(
!
page_rec_is_metadata
(
orig_succ
));
ut_ad
(
!
page_rec_is_metadata
(
orig_succ
));
lock_sys
.
mutex_lock
();
const
page_id_t
l
{
left_block
->
page
.
id
()};
const
page_id_t
r
{
right_block
->
page
.
id
()};
LockMutexGuard
g
;
/* Inherit the locks from the supremum of the left page to the
/* Inherit the locks from the supremum of the left page to the
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
(
r
ight_block
,
left_block
,
lock_rec_inherit_to_gap
(
r
,
l
,
right_block
->
frame
,
page_rec_get_heap_no
(
orig_succ
),
page_rec_get_heap_no
(
orig_succ
),
PAGE_HEAP_NO_SUPREMUM
);
PAGE_HEAP_NO_SUPREMUM
);
...
@@ -2779,61 +2736,37 @@ lock_update_merge_right(
...
@@ -2779,61 +2736,37 @@ lock_update_merge_right(
waiting transactions */
waiting transactions */
lock_rec_reset_and_release_wait_low
(
lock_rec_reset_and_release_wait_low
(
&
lock_sys
.
rec_hash
,
l
eft_block
,
PAGE_HEAP_NO_SUPREMUM
);
&
lock_sys
.
rec_hash
,
l
,
PAGE_HEAP_NO_SUPREMUM
);
/* there should exist no page lock on the left page,
/* there should exist no page lock on the left page,
otherwise, it will be blocked from merge */
otherwise, it will be blocked from merge */
ut_ad
(
!
lock_sys
.
get_first_prdt_page
(
l
eft_block
->
page
.
id
()
));
ut_ad
(
!
lock_sys
.
get_first_prdt_page
(
l
));
lock_rec_free_all_from_discard_page
(
left_block
);
lock_rec_free_all_from_discard_page
(
l
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/** Update locks 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
root page, even though they do not make sense on other than leaf
root page, even though they do not make sense on other than leaf
pages: the reason is that in a pessimistic update the infimum record
pages: the reason is that in a pessimistic update the infimum record
of the root page will act as a dummy carrier of the locks of the record
of the root page will act as a dummy carrier of the locks of the record
to be updated. */
to be updated. */
void
void
lock_update_root_raise
(
const
buf_block_t
&
block
,
const
page_id_t
root
)
lock_update_root_raise
(
/*===================*/
const
buf_block_t
*
block
,
/*!< in: index page to which copied */
const
buf_block_t
*
root
)
/*!< in: root page */
{
{
lock_sys
.
mutex_lock
();
LockMutexGuard
g
;
/* Move the locks on the supremum of the root to the supremum of block */
/* Move the locks on the supremum of the root to the supremum
lock_rec_move
(
block
,
root
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
of block */
lock_rec_move
(
block
,
root
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/** Update the lock table when a page is copied to another.
Updates the lock table when a page is copied to another and the original page
@param new_block the target page
is removed from the chain of leaf pages, except if page is the root! */
@param old old page (not index root page) */
void
void
lock_update_copy_and_discard
(
const
buf_block_t
&
new_block
,
page_id_t
old
)
lock_update_copy_and_discard
(
/*=========================*/
const
buf_block_t
*
new_block
,
/*!< in: index page to
which copied */
const
buf_block_t
*
block
)
/*!< in: index page;
NOT the root! */
{
{
lock_sys
.
mutex_lock
();
LockMutexGuard
g
;
/* Move the locks on the supremum of the old page to the supremum of new */
/* Move the locks on the supremum of the old page to the supremum
lock_rec_move
(
new_block
,
old
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
of new_page */
lock_rec_free_all_from_discard_page
(
old
);
lock_rec_move
(
new_block
,
block
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
lock_rec_free_all_from_discard_page
(
block
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -2844,69 +2777,51 @@ lock_update_split_left(
...
@@ -2844,69 +2777,51 @@ lock_update_split_left(
const
buf_block_t
*
right_block
,
/*!< in: right page */
const
buf_block_t
*
right_block
,
/*!< in: right page */
const
buf_block_t
*
left_block
)
/*!< in: left page */
const
buf_block_t
*
left_block
)
/*!< in: left page */
{
{
ulint
heap_no
=
lock_get_min_heap_no
(
right_block
);
ulint
h
=
lock_get_min_heap_no
(
right_block
);
const
page_id_t
l
{
left_block
->
page
.
id
()};
lock_sys
.
mutex_lock
()
;
const
page_id_t
r
{
right_block
->
page
.
id
()}
;
LockMutexGuard
g
;
/* 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
(
l
,
r
,
left_block
->
frame
,
PAGE_HEAP_NO_SUPREMUM
,
h
);
lock_rec_inherit_to_gap
(
left_block
,
right_block
,
PAGE_HEAP_NO_SUPREMUM
,
heap_no
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/** Update the lock table when a page is merged to the left.
Updates the lock table when a page is merged to the left. */
@param left left page
void
@param orig_pred original predecessor of supremum on the left page before merge
lock_update_merge_left
(
@param right merged, to-be-discarded right page */
/*===================*/
void
lock_update_merge_left
(
const
buf_block_t
&
left
,
const
rec_t
*
orig_pred
,
const
buf_block_t
*
left_block
,
/*!< in: left page to
const
page_id_t
right
)
which merged */
const
rec_t
*
orig_pred
,
/*!< in: original predecessor
of supremum on the left page
before merge */
const
buf_block_t
*
right_block
)
/*!< in: merged index page
which will be discarded */
{
{
const
rec_t
*
left_next_rec
;
ut_ad
(
left
.
frame
==
page_align
(
orig_pred
));
ut_ad
(
left_block
->
frame
==
page_align
(
orig_pred
));
lock_sys
.
mutex_lock
();
left_next_rec
=
page_rec_get_next_const
(
orig_pred
)
;
const
page_id_t
l
{
left
.
page
.
id
()}
;
if
(
!
page_rec_is_supremum
(
left_next_rec
))
{
LockMutexGuard
g
;
const
rec_t
*
left_next_rec
=
page_rec_get_next_const
(
orig_pred
);
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
(
l
,
l
,
left
.
frame
,
lock_rec_inherit_to_gap
(
left_block
,
left_block
,
page_rec_get_heap_no
(
left_next_rec
),
page_rec_get_heap_no
(
left_next_rec
),
PAGE_HEAP_NO_SUPREMUM
);
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_low
(
&
lock_sys
.
rec_hash
,
l
,
lock_rec_reset_and_release_wait_low
(
PAGE_HEAP_NO_SUPREMUM
);
&
lock_sys
.
rec_hash
,
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
,
right
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
lock_rec_move
(
left_block
,
right_block
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
);
/* there should exist no page lock on the right page,
/* there should exist no page lock on the right page,
otherwise, it will be blocked from merge */
otherwise, it will be blocked from merge */
ut_ad
(
!
lock_sys
.
get_first_prdt_page
(
right_block
->
page
.
id
()));
ut_ad
(
!
lock_sys
.
get_first_prdt_page
(
right
));
lock_rec_free_all_from_discard_page
(
right
);
lock_rec_free_all_from_discard_page
(
right_block
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -2915,9 +2830,9 @@ inherited from rec. */
...
@@ -2915,9 +2830,9 @@ inherited from rec. */
void
void
lock_rec_reset_and_inherit_gap_locks
(
lock_rec_reset_and_inherit_gap_locks
(
/*=================================*/
/*=================================*/
const
buf_block_t
*
heir_block
,
/*!< in: block containing the
const
buf_block_t
&
heir_block
,
/*!< in: block containing the
record which inherits */
record which inherits */
const
buf_block_t
*
block
,
/*!< in: block
containing the
const
page_id_t
donor
,
/*!< in: page
containing the
record from which inherited;
record from which inherited;
does NOT reset the locks on
does NOT reset the locks on
this record */
this record */
...
@@ -2926,13 +2841,10 @@ lock_rec_reset_and_inherit_gap_locks(
...
@@ -2926,13 +2841,10 @@ lock_rec_reset_and_inherit_gap_locks(
ulint
heap_no
)
/*!< in: heap_no of the
ulint
heap_no
)
/*!< in: heap_no of the
donating record */
donating record */
{
{
lock_sys
.
mutex_lock
();
const
page_id_t
heir
{
heir_block
.
page
.
id
()};
LockMutexGuard
g
;
lock_rec_reset_and_release_wait
(
heir_block
,
heir_heap_no
);
lock_rec_reset_and_release_wait
(
heir
,
heir_heap_no
);
lock_rec_inherit_to_gap
(
heir
,
donor
,
heir_block
.
frame
,
heir_heap_no
,
heap_no
);
lock_rec_inherit_to_gap
(
heir_block
,
block
,
heir_heap_no
,
heap_no
);
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -2950,9 +2862,10 @@ lock_update_discard(
...
@@ -2950,9 +2862,10 @@ lock_update_discard(
const
page_t
*
page
=
block
->
frame
;
const
page_t
*
page
=
block
->
frame
;
const
rec_t
*
rec
;
const
rec_t
*
rec
;
ulint
heap_no
;
ulint
heap_no
;
const
page_id_t
heir
(
heir_block
->
page
.
id
());
const
page_id_t
page_id
(
block
->
page
.
id
());
const
page_id_t
page_id
(
block
->
page
.
id
());
lock_sys
.
mutex_lock
()
;
LockMutexGuard
g
;
if
(
lock_sys
.
get_first
(
page_id
))
{
if
(
lock_sys
.
get_first
(
page_id
))
{
ut_ad
(
!
lock_sys
.
get_first_prdt
(
page_id
));
ut_ad
(
!
lock_sys
.
get_first_prdt
(
page_id
));
...
@@ -2966,11 +2879,12 @@ lock_update_discard(
...
@@ -2966,11 +2879,12 @@ lock_update_discard(
do
{
do
{
heap_no
=
rec_get_heap_no_new
(
rec
);
heap_no
=
rec_get_heap_no_new
(
rec
);
lock_rec_inherit_to_gap
(
heir_block
,
block
,
lock_rec_inherit_to_gap
(
heir
,
page_id
,
heir_block
->
frame
,
heir_heap_no
,
heap_no
);
heir_heap_no
,
heap_no
);
lock_rec_reset_and_release_wait
(
lock_rec_reset_and_release_wait
(
block
,
heap_no
);
page_id
,
heap_no
);
rec
=
page
+
rec_get_next_offs
(
rec
,
TRUE
);
rec
=
page
+
rec_get_next_offs
(
rec
,
TRUE
);
}
while
(
heap_no
!=
PAGE_HEAP_NO_SUPREMUM
);
}
while
(
heap_no
!=
PAGE_HEAP_NO_SUPREMUM
);
...
@@ -2980,11 +2894,12 @@ lock_update_discard(
...
@@ -2980,11 +2894,12 @@ lock_update_discard(
do
{
do
{
heap_no
=
rec_get_heap_no_old
(
rec
);
heap_no
=
rec_get_heap_no_old
(
rec
);
lock_rec_inherit_to_gap
(
heir_block
,
block
,
lock_rec_inherit_to_gap
(
heir
,
page_id
,
heir_block
->
frame
,
heir_heap_no
,
heap_no
);
heir_heap_no
,
heap_no
);
lock_rec_reset_and_release_wait
(
lock_rec_reset_and_release_wait
(
block
,
heap_no
);
page_id
,
heap_no
);
rec
=
page
+
rec_get_next_offs
(
rec
,
FALSE
);
rec
=
page
+
rec_get_next_offs
(
rec
,
FALSE
);
}
while
(
heap_no
!=
PAGE_HEAP_NO_SUPREMUM
);
}
while
(
heap_no
!=
PAGE_HEAP_NO_SUPREMUM
);
...
@@ -2998,8 +2913,6 @@ lock_update_discard(
...
@@ -2998,8 +2913,6 @@ lock_update_discard(
lock_rec_free_all_from_discard_page_low
(
lock_rec_free_all_from_discard_page_low
(
page_id
,
&
lock_sys
.
prdt_page_hash
);
page_id
,
&
lock_sys
.
prdt_page_hash
);
}
}
lock_sys
.
mutex_unlock
();
}
}
/*************************************************************//**
/*************************************************************//**
...
@@ -3060,17 +2973,16 @@ lock_update_delete(
...
@@ -3060,17 +2973,16 @@ lock_update_delete(
FALSE
));
FALSE
));
}
}
lock_sys
.
mutex_lock
();
const
page_id_t
id
{
block
->
page
.
id
()};
LockMutexGuard
g
;
/* 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
(
block
,
block
,
next_heap_no
,
heap_no
);
lock_rec_inherit_to_gap
(
id
,
id
,
block
->
frame
,
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
(
block
,
heap_no
);
lock_rec_reset_and_release_wait
(
id
,
heap_no
);
lock_sys
.
mutex_unlock
();
}
}
/*********************************************************************//**
/*********************************************************************//**
...
@@ -3090,39 +3002,27 @@ lock_rec_store_on_page_infimum(
...
@@ -3090,39 +3002,27 @@ lock_rec_store_on_page_infimum(
bits are reset on the
bits are reset on the
record */
record */
{
{
ulint
heap_no
=
page_rec_get_heap_no
(
rec
);
const
ulint
heap_no
=
page_rec_get_heap_no
(
rec
);
ut_ad
(
block
->
frame
==
page_align
(
rec
));
ut_ad
(
block
->
frame
==
page_align
(
rec
));
const
page_id_t
id
{
block
->
page
.
id
()};
lock_sys
.
mutex_lock
();
LockMutexGuard
g
;
lock_rec_move
(
*
block
,
id
,
PAGE_HEAP_NO_INFIMUM
,
heap_no
);
lock_rec_move
(
block
,
block
,
PAGE_HEAP_NO_INFIMUM
,
heap_no
);
lock_sys
.
mutex_unlock
();
}
}
/*********************************************************************//**
/** Restore the 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 a page.
state was stored on the infimum of the page. */
@param block buffer block containing rec
void
@param rec record whose lock state is restored
lock_rec_restore_from_page_infimum
(
@param donator page (rec is not necessarily on this page)
/*===============================*/
whose infimum stored the lock state; lock bits are reset on the infimum */
const
buf_block_t
*
block
,
/*!< in: buffer block containing rec */
void
lock_rec_restore_from_page_infimum
(
const
buf_block_t
&
block
,
const
rec_t
*
rec
,
/*!< in: record whose lock state
const
rec_t
*
rec
,
page_id_t
donator
)
is restored */
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
heap_no
=
page_rec_get_heap_no
(
rec
);
const
ulint
heap_no
=
page_rec_get_heap_no
(
rec
);
LockMutexGuard
g
;
lock_sys
.
mutex_lock
();
lock_rec_move
(
block
,
donator
,
heap_no
,
PAGE_HEAP_NO_INFIMUM
);
lock_rec_move
(
block
,
donator
,
heap_no
,
PAGE_HEAP_NO_INFIMUM
);
lock_sys
.
mutex_unlock
();
}
}
/*========================= TABLE LOCKS ==============================*/
/*========================= TABLE LOCKS ==============================*/
...
@@ -3765,7 +3665,7 @@ lock_rec_unlock(
...
@@ -3765,7 +3665,7 @@ lock_rec_unlock(
/*============*/
/*============*/
trx_t
*
trx
,
/*!< in/out: transaction that has
trx_t
*
trx
,
/*!< in/out: transaction that has
set a record lock */
set a record lock */
const
buf_block_t
*
block
,
/*!< in: buffer block
containing rec */
const
page_id_t
id
,
/*!< in: page
containing rec */
const
rec_t
*
rec
,
/*!< in: record */
const
rec_t
*
rec
,
/*!< in: record */
lock_mode
lock_mode
)
/*!< in: LOCK_S or LOCK_X */
lock_mode
lock_mode
)
/*!< in: LOCK_S or LOCK_X */
{
{
...
@@ -3775,7 +3675,6 @@ lock_rec_unlock(
...
@@ -3775,7 +3675,6 @@ lock_rec_unlock(
ut_ad
(
trx
);
ut_ad
(
trx
);
ut_ad
(
rec
);
ut_ad
(
rec
);
ut_ad
(
block
->
frame
==
page_align
(
rec
));
ut_ad
(
!
trx
->
lock
.
wait_lock
);
ut_ad
(
!
trx
->
lock
.
wait_lock
);
ut_ad
(
trx_state_eq
(
trx
,
TRX_STATE_ACTIVE
));
ut_ad
(
trx_state_eq
(
trx
,
TRX_STATE_ACTIVE
));
ut_ad
(
!
page_rec_is_metadata
(
rec
));
ut_ad
(
!
page_rec_is_metadata
(
rec
));
...
@@ -3784,7 +3683,7 @@ lock_rec_unlock(
...
@@ -3784,7 +3683,7 @@ lock_rec_unlock(
lock_sys
.
mutex_lock
();
lock_sys
.
mutex_lock
();
first_lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
first_lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
/* Find the last lock with the same lock_mode and transaction
/* Find the last lock with the same lock_mode and transaction
on the record. */
on the record. */
...
@@ -4390,7 +4289,7 @@ lock_rec_queue_validate(
...
@@ -4390,7 +4289,7 @@ lock_rec_queue_validate(
/*!< in: if the caller holds
/*!< in: if the caller holds
both the lock mutex and
both the lock mutex and
trx_sys_t->lock. */
trx_sys_t->lock. */
const
buf_block_t
*
block
,
/*!< in: buffer block containing rec
*/
const
page_id_t
id
,
/*!< in: page identifier
*/
const
rec_t
*
rec
,
/*!< in: record to look at */
const
rec_t
*
rec
,
/*!< in: record to look at */
const
dict_index_t
*
index
,
/*!< in: index, or NULL if not known */
const
dict_index_t
*
index
,
/*!< in: index, or NULL if not known */
const
rec_offs
*
offsets
)
/*!< in: rec_get_offsets(rec, index) */
const
rec_offs
*
offsets
)
/*!< in: rec_get_offsets(rec, index) */
...
@@ -4399,7 +4298,6 @@ lock_rec_queue_validate(
...
@@ -4399,7 +4298,6 @@ lock_rec_queue_validate(
ulint
heap_no
;
ulint
heap_no
;
ut_a
(
rec
);
ut_a
(
rec
);
ut_a
(
block
->
frame
==
page_align
(
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
));
ut_ad
(
page_rec_is_leaf
(
rec
));
ut_ad
(
page_rec_is_leaf
(
rec
));
...
@@ -4417,7 +4315,7 @@ lock_rec_queue_validate(
...
@@ -4417,7 +4315,7 @@ lock_rec_queue_validate(
if
(
!
page_rec_is_user_rec
(
rec
))
{
if
(
!
page_rec_is_user_rec
(
rec
))
{
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
id
,
heap_no
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next_const
(
heap_no
,
lock
))
{
lock
=
lock_rec_get_next_const
(
heap_no
,
lock
))
{
...
@@ -4458,8 +4356,7 @@ lock_rec_queue_validate(
...
@@ -4458,8 +4356,7 @@ lock_rec_queue_validate(
if
(
impl_trx
->
state
==
TRX_STATE_COMMITTED_IN_MEMORY
)
{
if
(
impl_trx
->
state
==
TRX_STATE_COMMITTED_IN_MEMORY
)
{
}
else
if
(
const
lock_t
*
other_lock
}
else
if
(
const
lock_t
*
other_lock
=
lock_rec_other_has_expl_req
(
=
lock_rec_other_has_expl_req
(
LOCK_S
,
block
,
true
,
heap_no
,
LOCK_S
,
id
,
true
,
heap_no
,
impl_trx
))
{
impl_trx
))
{
/* The impl_trx is holding an implicit lock on the
/* The impl_trx is holding an implicit lock on the
given record 'rec'. So there cannot be another
given record 'rec'. So there cannot be another
explicit granted lock. Also, there can be another
explicit granted lock. Also, there can be another
...
@@ -4491,7 +4388,7 @@ lock_rec_queue_validate(
...
@@ -4491,7 +4388,7 @@ lock_rec_queue_validate(
wsrep_report_bf_lock_wait
(
other_lock
->
trx
->
mysql_thd
,
other_lock
->
trx
->
id
);
wsrep_report_bf_lock_wait
(
other_lock
->
trx
->
mysql_thd
,
other_lock
->
trx
->
id
);
if
(
!
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
if
(
!
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
block
,
heap_no
,
id
,
heap_no
,
impl_trx
))
{
impl_trx
))
{
ib
::
info
()
<<
"WSREP impl BF lock conflict"
;
ib
::
info
()
<<
"WSREP impl BF lock conflict"
;
}
}
...
@@ -4500,14 +4397,14 @@ lock_rec_queue_validate(
...
@@ -4500,14 +4397,14 @@ lock_rec_queue_validate(
{
{
ut_ad
(
other_lock
->
is_waiting
());
ut_ad
(
other_lock
->
is_waiting
());
ut_ad
(
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
ut_ad
(
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
block
,
heap_no
,
impl_trx
));
id
,
heap_no
,
impl_trx
));
}
}
}
}
impl_trx
->
mutex_unlock
();
impl_trx
->
mutex_unlock
();
}
}
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
for
(
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next_const
(
heap_no
,
lock
))
{
lock
=
lock_rec_get_next_const
(
heap_no
,
lock
))
{
...
@@ -4527,8 +4424,7 @@ lock_rec_queue_validate(
...
@@ -4527,8 +4424,7 @@ lock_rec_queue_validate(
const
lock_t
*
other_lock
const
lock_t
*
other_lock
=
lock_rec_other_has_expl_req
(
=
lock_rec_other_has_expl_req
(
mode
,
block
,
false
,
heap_no
,
mode
,
id
,
false
,
heap_no
,
lock
->
trx
);
lock
->
trx
);
#ifdef WITH_WSREP
#ifdef WITH_WSREP
if
(
UNIV_UNLIKELY
(
other_lock
&&
lock
->
trx
->
is_wsrep
()))
{
if
(
UNIV_UNLIKELY
(
other_lock
&&
lock
->
trx
->
is_wsrep
()))
{
/* Only BF transaction may be granted
/* Only BF transaction may be granted
...
@@ -4566,9 +4462,11 @@ static bool lock_rec_validate_page(const buf_block_t *block, bool latched)
...
@@ -4566,9 +4462,11 @@ static bool lock_rec_validate_page(const buf_block_t *block, bool latched)
rec_offs
*
offsets
=
offsets_
;
rec_offs
*
offsets
=
offsets_
;
rec_offs_init
(
offsets_
);
rec_offs_init
(
offsets_
);
lock_sys
.
mutex_lock
();
const
page_id_t
id
{
block
->
page
.
id
()};
LockMutexGuard
g
;
loop:
loop:
lock
=
lock_sys
.
get_first
(
block
->
page
.
id
()
);
lock
=
lock_sys
.
get_first
(
id
);
if
(
!
lock
)
{
if
(
!
lock
)
{
goto
function_exit
;
goto
function_exit
;
...
@@ -4609,7 +4507,7 @@ static bool lock_rec_validate_page(const buf_block_t *block, bool latched)
...
@@ -4609,7 +4507,7 @@ static bool lock_rec_validate_page(const buf_block_t *block, bool latched)
cause a deadlock of threads. */
cause a deadlock of threads. */
lock_rec_queue_validate
(
lock_rec_queue_validate
(
TRUE
,
block
,
rec
,
lock
->
index
,
offsets
);
true
,
id
,
rec
,
lock
->
index
,
offsets
);
nth_bit
=
i
+
1
;
nth_bit
=
i
+
1
;
...
@@ -4623,8 +4521,6 @@ static bool lock_rec_validate_page(const buf_block_t *block, bool latched)
...
@@ -4623,8 +4521,6 @@ static bool lock_rec_validate_page(const buf_block_t *block, bool latched)
goto
loop
;
goto
loop
;
function_exit:
function_exit:
lock_sys
.
mutex_unlock
();
if
(
heap
!=
NULL
)
{
if
(
heap
!=
NULL
)
{
mem_heap_free
(
heap
);
mem_heap_free
(
heap
);
}
}
...
@@ -4803,6 +4699,8 @@ lock_rec_insert_check_and_lock(
...
@@ -4803,6 +4699,8 @@ lock_rec_insert_check_and_lock(
ulint
heap_no
=
page_rec_get_heap_no
(
next_rec
);
ulint
heap_no
=
page_rec_get_heap_no
(
next_rec
);
ut_ad
(
!
rec_is_metadata
(
next_rec
,
*
index
));
ut_ad
(
!
rec_is_metadata
(
next_rec
,
*
index
));
const
page_id_t
id
{
block
->
page
.
id
()};
lock_sys
.
mutex_lock
();
lock_sys
.
mutex_lock
();
/* Because this code is invoked for a running transaction by
/* Because this code is invoked for a running transaction by
the thread that is serving the transaction, it is not necessary
the thread that is serving the transaction, it is not necessary
...
@@ -4813,7 +4711,7 @@ lock_rec_insert_check_and_lock(
...
@@ -4813,7 +4711,7 @@ lock_rec_insert_check_and_lock(
BTR_NO_LOCKING_FLAG and skip the locking altogether. */
BTR_NO_LOCKING_FLAG and skip the locking altogether. */
ut_ad
(
lock_table_has
(
trx
,
index
->
table
,
LOCK_IX
));
ut_ad
(
lock_table_has
(
trx
,
index
->
table
,
LOCK_IX
));
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
block
,
heap_no
);
lock
=
lock_rec_get_first
(
&
lock_sys
.
rec_hash
,
id
,
heap_no
);
if
(
lock
==
NULL
)
{
if
(
lock
==
NULL
)
{
/* We optimize CPU time usage in the simplest case */
/* We optimize CPU time usage in the simplest case */
...
@@ -4856,14 +4754,15 @@ lock_rec_insert_check_and_lock(
...
@@ -4856,14 +4754,15 @@ lock_rec_insert_check_and_lock(
#ifdef WITH_WSREP
#ifdef WITH_WSREP
lock_t
*
c_lock
=
lock_t
*
c_lock
=
#endif
/* WITH_WSREP */
#endif
/* WITH_WSREP */
lock_rec_other_has_conflicting
(
type_mode
,
block
,
heap_no
,
trx
))
{
lock_rec_other_has_conflicting
(
type_mode
,
id
,
heap_no
,
trx
))
{
trx
->
mutex_lock
();
trx
->
mutex_lock
();
err
=
lock_rec_enqueue_waiting
(
err
=
lock_rec_enqueue_waiting
(
#ifdef WITH_WSREP
#ifdef WITH_WSREP
c_lock
,
c_lock
,
#endif
/* WITH_WSREP */
#endif
/* WITH_WSREP */
type_mode
,
block
,
heap_no
,
index
,
thr
,
NULL
);
type_mode
,
id
,
block
->
frame
,
heap_no
,
index
,
thr
,
nullptr
);
trx
->
mutex_unlock
();
trx
->
mutex_unlock
();
}
else
{
}
else
{
...
@@ -4899,8 +4798,7 @@ lock_rec_insert_check_and_lock(
...
@@ -4899,8 +4798,7 @@ lock_rec_insert_check_and_lock(
offsets
=
rec_get_offsets
(
next_rec
,
index
,
offsets_
,
true
,
offsets
=
rec_get_offsets
(
next_rec
,
index
,
offsets_
,
true
,
ULINT_UNDEFINED
,
&
heap
);
ULINT_UNDEFINED
,
&
heap
);
ut_ad
(
lock_rec_queue_validate
(
ut_ad
(
lock_rec_queue_validate
(
false
,
id
,
next_rec
,
index
,
offsets
));
FALSE
,
block
,
next_rec
,
index
,
offsets
));
if
(
heap
!=
NULL
)
{
if
(
heap
!=
NULL
)
{
mem_heap_free
(
heap
);
mem_heap_free
(
heap
);
...
@@ -4920,7 +4818,7 @@ static
...
@@ -4920,7 +4818,7 @@ static
void
void
lock_rec_convert_impl_to_expl_for_trx
(
lock_rec_convert_impl_to_expl_for_trx
(
/*==================================*/
/*==================================*/
const
buf_block_t
*
block
,
/*!< in: buffer block of rec
*/
const
page_id_t
id
,
/*!< in: page identifier
*/
const
rec_t
*
rec
,
/*!< in: user record on page */
const
rec_t
*
rec
,
/*!< in: user record on page */
dict_index_t
*
index
,
/*!< in: index of record */
dict_index_t
*
index
,
/*!< in: index of record */
trx_t
*
trx
,
/*!< in/out: active transaction */
trx_t
*
trx
,
/*!< in/out: active transaction */
...
@@ -4931,18 +4829,17 @@ lock_rec_convert_impl_to_expl_for_trx(
...
@@ -4931,18 +4829,17 @@ lock_rec_convert_impl_to_expl_for_trx(
ut_ad
(
!
rec_is_metadata
(
rec
,
*
index
));
ut_ad
(
!
rec_is_metadata
(
rec
,
*
index
));
DEBUG_SYNC_C
(
"before_lock_rec_convert_impl_to_expl_for_trx"
);
DEBUG_SYNC_C
(
"before_lock_rec_convert_impl_to_expl_for_trx"
);
lock_sys
.
mutex_lock
();
{
LockMutexGuard
g
;
trx
->
mutex_lock
();
trx
->
mutex_lock
();
ut_ad
(
!
trx_state_eq
(
trx
,
TRX_STATE_NOT_STARTED
));
ut_ad
(
!
trx_state_eq
(
trx
,
TRX_STATE_NOT_STARTED
));
if
(
!
trx_state_eq
(
trx
,
TRX_STATE_COMMITTED_IN_MEMORY
)
if
(
!
trx_state_eq
(
trx
,
TRX_STATE_COMMITTED_IN_MEMORY
)
&&
&&
!
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
!
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
id
,
heap_no
,
trx
))
block
,
heap_no
,
trx
))
{
lock_rec_add_to_queue
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
id
,
page_align
(
rec
),
lock_rec_add_to_queue
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
heap_no
,
index
,
trx
,
true
);
block
,
heap_no
,
index
,
trx
,
true
);
}
}
lock_sys
.
mutex_unlock
();
trx
->
mutex_unlock
();
trx
->
mutex_unlock
();
trx
->
release_reference
();
trx
->
release_reference
();
...
@@ -4954,8 +4851,8 @@ lock_rec_convert_impl_to_expl_for_trx(
...
@@ -4954,8 +4851,8 @@ lock_rec_convert_impl_to_expl_for_trx(
struct
lock_rec_other_trx_holds_expl_arg
struct
lock_rec_other_trx_holds_expl_arg
{
{
const
ulint
heap_no
;
const
ulint
heap_no
;
const
buf_block_t
*
const
block
;
const
page_id_t
id
;
const
trx_t
*
impl_trx
;
const
trx_t
&
impl_trx
;
};
};
...
@@ -4969,13 +4866,14 @@ static my_bool lock_rec_other_trx_holds_expl_callback(
...
@@ -4969,13 +4866,14 @@ static my_bool lock_rec_other_trx_holds_expl_callback(
element
->
trx
->
mutex_lock
();
element
->
trx
->
mutex_lock
();
ut_ad
(
element
->
trx
->
state
!=
TRX_STATE_NOT_STARTED
);
ut_ad
(
element
->
trx
->
state
!=
TRX_STATE_NOT_STARTED
);
lock_t
*
expl_lock
=
element
->
trx
->
state
==
TRX_STATE_COMMITTED_IN_MEMORY
lock_t
*
expl_lock
=
element
->
trx
->
state
==
TRX_STATE_COMMITTED_IN_MEMORY
?
NULL
:
lock_rec_has_expl
(
LOCK_S
|
LOCK_REC_NOT_GAP
,
arg
->
block
,
?
nullptr
arg
->
heap_no
,
element
->
trx
);
:
lock_rec_has_expl
(
LOCK_S
|
LOCK_REC_NOT_GAP
,
arg
->
id
,
arg
->
heap_no
,
element
->
trx
);
/*
/*
An explicit lock is held by trx other than the trx holding the implicit
An explicit lock is held by trx other than the trx holding the implicit
lock.
lock.
*/
*/
ut_ad
(
!
expl_lock
||
expl_lock
->
trx
==
arg
->
impl_trx
);
ut_ad
(
!
expl_lock
||
expl_lock
->
trx
==
&
arg
->
impl_trx
);
element
->
trx
->
mutex_unlock
();
element
->
trx
->
mutex_unlock
();
}
}
mysql_mutex_unlock
(
&
element
->
mutex
);
mysql_mutex_unlock
(
&
element
->
mutex
);
...
@@ -4994,31 +4892,28 @@ static my_bool lock_rec_other_trx_holds_expl_callback(
...
@@ -4994,31 +4892,28 @@ static my_bool lock_rec_other_trx_holds_expl_callback(
@param caller_trx trx of current thread
@param caller_trx trx of current thread
@param[in] trx trx holding implicit lock on rec
@param[in] trx trx holding implicit lock on rec
@param[in] rec user record
@param[in] rec user record
@param[in]
block buffer block containing the record
@param[in]
id page identifier
*/
*/
static
void
lock_rec_other_trx_holds_expl
(
trx_t
*
caller_trx
,
trx_t
*
trx
,
static
void
lock_rec_other_trx_holds_expl
(
trx_t
*
caller_trx
,
trx_t
*
trx
,
const
rec_t
*
rec
,
const
rec_t
*
rec
,
const
buf_block_t
*
block
)
const
page_id_t
id
)
{
{
if
(
trx
)
if
(
trx
)
{
{
ut_ad
(
!
page_rec_is_metadata
(
rec
));
ut_ad
(
!
page_rec_is_metadata
(
rec
));
lock_sys
.
mutex_lock
()
;
LockMutexGuard
g
;
ut_ad
(
trx
->
is_referenced
());
ut_ad
(
trx
->
is_referenced
());
const
trx_state_t
state
{
trx
->
state
};
const
trx_state_t
state
{
trx
->
state
};
ut_ad
(
state
!=
TRX_STATE_NOT_STARTED
);
ut_ad
(
state
!=
TRX_STATE_NOT_STARTED
);
if
(
state
==
TRX_STATE_COMMITTED_IN_MEMORY
)
if
(
state
==
TRX_STATE_COMMITTED_IN_MEMORY
)
{
/* The transaction was committed before we acquired LockMutexGuard. */
/* The transaction was committed before our lock_sys.mutex_lock(). */
lock_sys
.
mutex_unlock
();
return
;
return
;
}
lock_rec_other_trx_holds_expl_arg
arg
=
{
page_rec_get_heap_no
(
rec
),
block
,
lock_rec_other_trx_holds_expl_arg
arg
=
{
page_rec_get_heap_no
(
rec
),
id
,
trx
};
*
trx
};
trx_sys
.
rw_trx_hash
.
iterate
(
caller_trx
,
trx_sys
.
rw_trx_hash
.
iterate
(
caller_trx
,
lock_rec_other_trx_holds_expl_callback
,
&
arg
);
lock_rec_other_trx_holds_expl_callback
,
&
arg
);
lock_sys
.
mutex_unlock
();
}
}
}
}
#endif
/* UNIV_DEBUG */
#endif
/* UNIV_DEBUG */
...
@@ -5036,7 +4931,7 @@ an implicit exclusive lock on the record. In this case, no explicit lock
...
@@ -5036,7 +4931,7 @@ an implicit exclusive lock on the record. In this case, no explicit lock
should be created.
should be created.
@param[in,out] caller_trx current transaction
@param[in,out] caller_trx current transaction
@param[in]
block index tree leaf page
@param[in]
id index tree leaf page identifier
@param[in] rec record on the leaf page
@param[in] rec record on the leaf page
@param[in] index the index of the record
@param[in] index the index of the record
@param[in] offsets rec_get_offsets(rec,index)
@param[in] offsets rec_get_offsets(rec,index)
...
@@ -5045,7 +4940,7 @@ static
...
@@ -5045,7 +4940,7 @@ static
bool
bool
lock_rec_convert_impl_to_expl
(
lock_rec_convert_impl_to_expl
(
trx_t
*
caller_trx
,
trx_t
*
caller_trx
,
const
buf_block_t
*
block
,
page_id_t
id
,
const
rec_t
*
rec
,
const
rec_t
*
rec
,
dict_index_t
*
index
,
dict_index_t
*
index
,
const
rec_offs
*
offsets
)
const
rec_offs
*
offsets
)
...
@@ -5082,11 +4977,10 @@ lock_rec_convert_impl_to_expl(
...
@@ -5082,11 +4977,10 @@ lock_rec_convert_impl_to_expl(
return
true
;
return
true
;
}
}
ut_d
(
lock_rec_other_trx_holds_expl
(
caller_trx
,
trx
,
rec
,
ut_d
(
lock_rec_other_trx_holds_expl
(
caller_trx
,
trx
,
rec
,
id
));
block
));
}
}
if
(
trx
!=
0
)
{
if
(
trx
)
{
ulint
heap_no
=
page_rec_get_heap_no
(
rec
);
ulint
heap_no
=
page_rec_get_heap_no
(
rec
);
ut_ad
(
trx
->
is_referenced
());
ut_ad
(
trx
->
is_referenced
());
...
@@ -5096,7 +4990,7 @@ lock_rec_convert_impl_to_expl(
...
@@ -5096,7 +4990,7 @@ lock_rec_convert_impl_to_expl(
trx cannot be committed until the ref count is zero. */
trx cannot be committed until the ref count is zero. */
lock_rec_convert_impl_to_expl_for_trx
(
lock_rec_convert_impl_to_expl_for_trx
(
block
,
rec
,
index
,
trx
,
heap_no
);
id
,
rec
,
index
,
trx
,
heap_no
);
}
}
return
false
;
return
false
;
...
@@ -5113,8 +5007,6 @@ lock queue.
...
@@ -5113,8 +5007,6 @@ lock queue.
dberr_t
dberr_t
lock_clust_rec_modify_check_and_lock
(
lock_clust_rec_modify_check_and_lock
(
/*=================================*/
/*=================================*/
ulint
flags
,
/*!< in: if BTR_NO_LOCKING_FLAG
bit is set, does nothing */
const
buf_block_t
*
block
,
/*!< in: buffer block of rec */
const
buf_block_t
*
block
,
/*!< in: buffer block of rec */
const
rec_t
*
rec
,
/*!< in: record which should be
const
rec_t
*
rec
,
/*!< in: record which should be
modified */
modified */
...
@@ -5130,10 +5022,6 @@ lock_clust_rec_modify_check_and_lock(
...
@@ -5130,10 +5022,6 @@ lock_clust_rec_modify_check_and_lock(
ut_ad
(
dict_index_is_clust
(
index
));
ut_ad
(
dict_index_is_clust
(
index
));
ut_ad
(
block
->
frame
==
page_align
(
rec
));
ut_ad
(
block
->
frame
==
page_align
(
rec
));
if
(
flags
&
BTR_NO_LOCKING_FLAG
)
{
return
(
DB_SUCCESS
);
}
ut_ad
(
!
rec_is_metadata
(
rec
,
*
index
));
ut_ad
(
!
rec_is_metadata
(
rec
,
*
index
));
ut_ad
(
!
index
->
table
->
is_temporary
());
ut_ad
(
!
index
->
table
->
is_temporary
());
...
@@ -5144,16 +5032,17 @@ lock_clust_rec_modify_check_and_lock(
...
@@ -5144,16 +5032,17 @@ 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 */
if
(
lock_rec_convert_impl_to_expl
(
thr_get_trx
(
thr
),
block
,
rec
,
index
,
if
(
lock_rec_convert_impl_to_expl
(
thr_get_trx
(
thr
),
block
->
page
.
id
()
,
offsets
))
{
rec
,
index
,
offsets
))
{
/* We already hold an implicit exclusive lock. */
/* We already hold an implicit exclusive lock. */
return
DB_SUCCESS
;
return
DB_SUCCESS
;
}
}
err
=
lock_rec_lock
(
TRUE
,
LOCK_X
|
LOCK_REC_NOT_GAP
,
err
=
lock_rec_lock
(
true
,
LOCK_X
|
LOCK_REC_NOT_GAP
,
block
,
heap_no
,
index
,
thr
);
block
,
heap_no
,
index
,
thr
);
ut_ad
(
lock_rec_queue_validate
(
FALSE
,
block
,
rec
,
index
,
offsets
));
ut_ad
(
lock_rec_queue_validate
(
false
,
block
->
page
.
id
(),
rec
,
index
,
offsets
));
if
(
err
==
DB_SUCCESS_LOCKED_REC
)
{
if
(
err
==
DB_SUCCESS_LOCKED_REC
)
{
err
=
DB_SUCCESS
;
err
=
DB_SUCCESS
;
...
@@ -5205,7 +5094,7 @@ lock_sec_rec_modify_check_and_lock(
...
@@ -5205,7 +5094,7 @@ lock_sec_rec_modify_check_and_lock(
index record, and this would not have been possible if another active
index record, and this would not have been possible if another active
transaction had modified this secondary index record. */
transaction had modified this secondary index record. */
err
=
lock_rec_lock
(
TRUE
,
LOCK_X
|
LOCK_REC_NOT_GAP
,
err
=
lock_rec_lock
(
true
,
LOCK_X
|
LOCK_REC_NOT_GAP
,
block
,
heap_no
,
index
,
thr
);
block
,
heap_no
,
index
,
thr
);
#ifdef UNIV_DEBUG
#ifdef UNIV_DEBUG
...
@@ -5219,7 +5108,7 @@ lock_sec_rec_modify_check_and_lock(
...
@@ -5219,7 +5108,7 @@ lock_sec_rec_modify_check_and_lock(
ULINT_UNDEFINED
,
&
heap
);
ULINT_UNDEFINED
,
&
heap
);
ut_ad
(
lock_rec_queue_validate
(
ut_ad
(
lock_rec_queue_validate
(
FALSE
,
block
,
rec
,
index
,
offsets
));
false
,
block
->
page
.
id
()
,
rec
,
index
,
offsets
));
if
(
heap
!=
NULL
)
{
if
(
heap
!=
NULL
)
{
mem_heap_free
(
heap
);
mem_heap_free
(
heap
);
...
@@ -5284,6 +5173,8 @@ lock_sec_rec_read_check_and_lock(
...
@@ -5284,6 +5173,8 @@ lock_sec_rec_read_check_and_lock(
return
(
DB_SUCCESS
);
return
(
DB_SUCCESS
);
}
}
const
page_id_t
id
{
block
->
page
.
id
()};
ut_ad
(
!
rec_is_metadata
(
rec
,
*
index
));
ut_ad
(
!
rec_is_metadata
(
rec
,
*
index
));
heap_no
=
page_rec_get_heap_no
(
rec
);
heap_no
=
page_rec_get_heap_no
(
rec
);
...
@@ -5293,16 +5184,16 @@ lock_sec_rec_read_check_and_lock(
...
@@ -5293,16 +5184,16 @@ lock_sec_rec_read_check_and_lock(
if
(
!
page_rec_is_supremum
(
rec
)
if
(
!
page_rec_is_supremum
(
rec
)
&&
page_get_max_trx_id
(
block
->
frame
)
>=
trx_sys
.
get_min_trx_id
()
&&
page_get_max_trx_id
(
block
->
frame
)
>=
trx_sys
.
get_min_trx_id
()
&&
lock_rec_convert_impl_to_expl
(
thr_get_trx
(
thr
),
block
,
rec
,
&&
lock_rec_convert_impl_to_expl
(
thr_get_trx
(
thr
),
id
,
rec
,
index
,
offsets
))
{
index
,
offsets
))
{
/* We already hold an implicit exclusive lock. */
/* We already hold an implicit exclusive lock. */
return
DB_SUCCESS
;
return
DB_SUCCESS
;
}
}
err
=
lock_rec_lock
(
FALSE
,
gap_mode
|
mode
,
err
=
lock_rec_lock
(
false
,
gap_mode
|
mode
,
block
,
heap_no
,
index
,
thr
);
block
,
heap_no
,
index
,
thr
);
ut_ad
(
lock_rec_queue_validate
(
FALSE
,
block
,
rec
,
index
,
offsets
));
ut_ad
(
lock_rec_queue_validate
(
false
,
id
,
rec
,
index
,
offsets
));
return
(
err
);
return
(
err
);
}
}
...
@@ -5355,19 +5246,21 @@ lock_clust_rec_read_check_and_lock(
...
@@ -5355,19 +5246,21 @@ lock_clust_rec_read_check_and_lock(
return
(
DB_SUCCESS
);
return
(
DB_SUCCESS
);
}
}
const
page_id_t
id
{
block
->
page
.
id
()};
heap_no
=
page_rec_get_heap_no
(
rec
);
heap_no
=
page_rec_get_heap_no
(
rec
);
if
(
heap_no
!=
PAGE_HEAP_NO_SUPREMUM
if
(
heap_no
!=
PAGE_HEAP_NO_SUPREMUM
&&
lock_rec_convert_impl_to_expl
(
thr_get_trx
(
thr
),
block
,
rec
,
&&
lock_rec_convert_impl_to_expl
(
thr_get_trx
(
thr
),
id
,
rec
,
index
,
offsets
))
{
index
,
offsets
))
{
/* We already hold an implicit exclusive lock. */
/* We already hold an implicit exclusive lock. */
return
DB_SUCCESS
;
return
DB_SUCCESS
;
}
}
err
=
lock_rec_lock
(
FALSE
,
gap_mode
|
mode
,
err
=
lock_rec_lock
(
false
,
gap_mode
|
mode
,
block
,
heap_no
,
index
,
thr
);
block
,
heap_no
,
index
,
thr
);
ut_ad
(
lock_rec_queue_validate
(
FALSE
,
block
,
rec
,
index
,
offsets
));
ut_ad
(
lock_rec_queue_validate
(
false
,
id
,
rec
,
index
,
offsets
));
DEBUG_SYNC_C
(
"after_lock_clust_rec_read_check_and_lock"
);
DEBUG_SYNC_C
(
"after_lock_clust_rec_read_check_and_lock"
);
...
@@ -5767,25 +5660,18 @@ lock_trx_has_sys_table_locks(
...
@@ -5767,25 +5660,18 @@ lock_trx_has_sys_table_locks(
/** Check if the transaction holds an explicit exclusive lock on a record.
/** Check if the transaction holds an explicit exclusive lock on a record.
@param[in] trx transaction
@param[in] trx transaction
@param[in] table table
@param[in] table table
@param[in]
block leaf page
@param[in]
id leaf page identifier
@param[in] heap_no heap number identifying the record
@param[in] heap_no heap number identifying the record
@return whether an explicit X-lock is held */
@return whether an explicit X-lock is held */
bool
bool
lock_trx_has_expl_x_lock
(
const
trx_t
&
trx
,
const
dict_table_t
&
table
,
lock_trx_has_expl_x_lock
(
page_id_t
id
,
ulint
heap_no
)
const
trx_t
*
trx
,
/*!< in: transaction to check */
const
dict_table_t
*
table
,
/*!< in: table to check */
const
buf_block_t
*
block
,
/*!< in: buffer block of the record */
ulint
heap_no
)
/*!< in: record heap number */
{
{
ut_ad
(
heap_no
>
PAGE_HEAP_NO_SUPREMUM
);
ut_ad
(
heap_no
>
PAGE_HEAP_NO_SUPREMUM
);
LockMutexGuard
g
;
lock_sys
.
mutex_lock
();
ut_ad
(
lock_table_has
(
&
trx
,
&
table
,
LOCK_IX
));
ut_ad
(
lock_table_has
(
trx
,
table
,
LOCK_IX
));
ut_ad
(
lock_table_has
(
&
trx
,
&
table
,
LOCK_X
)
||
ut_ad
(
lock_table_has
(
trx
,
table
,
LOCK_X
)
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
id
,
heap_no
,
&
trx
));
||
lock_rec_has_expl
(
LOCK_X
|
LOCK_REC_NOT_GAP
,
block
,
heap_no
,
return
true
;
trx
));
lock_sys
.
mutex_unlock
();
return
(
true
);
}
}
#endif
/* UNIV_DEBUG */
#endif
/* UNIV_DEBUG */
...
@@ -6273,28 +6159,26 @@ lock_update_split_and_merge(
...
@@ -6273,28 +6159,26 @@ lock_update_split_and_merge(
ut_ad
(
page_is_leaf
(
right_block
->
frame
));
ut_ad
(
page_is_leaf
(
right_block
->
frame
));
ut_ad
(
page_align
(
orig_pred
)
==
left_block
->
frame
);
ut_ad
(
page_align
(
orig_pred
)
==
left_block
->
frame
);
lock_sys
.
mutex_lock
();
const
page_id_t
l
{
left_block
->
page
.
id
()};
const
page_id_t
r
{
right_block
->
page
.
id
()};
LockMutexGuard
g
;
left_next_rec
=
page_rec_get_next_const
(
orig_pred
);
left_next_rec
=
page_rec_get_next_const
(
orig_pred
);
ut_ad
(
!
page_rec_is_metadata
(
left_next_rec
));
ut_ad
(
!
page_rec_is_metadata
(
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
(
lock_rec_inherit_to_gap
(
l
,
l
,
left_block
->
frame
,
left_block
,
left_block
,
page_rec_get_heap_no
(
left_next_rec
),
page_rec_get_heap_no
(
left_next_rec
),
PAGE_HEAP_NO_SUPREMUM
);
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_block
,
lock_rec_reset_and_release_wait
(
l
,
PAGE_HEAP_NO_SUPREMUM
);
PAGE_HEAP_NO_SUPREMUM
);
/* 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
(
l
eft_block
,
right_block
,
lock_rec_inherit_to_gap
(
l
,
r
,
left_block
->
frame
,
PAGE_HEAP_NO_SUPREMUM
,
PAGE_HEAP_NO_SUPREMUM
,
lock_get_min_heap_no
(
right_block
));
lock_get_min_heap_no
(
right_block
));
lock_sys
.
mutex_unlock
();
}
}
storage/innobase/lock/lock0prdt.cc
View file @
2e64513f
...
@@ -229,8 +229,7 @@ lock_prdt_has_lock(
...
@@ -229,8 +229,7 @@ lock_prdt_has_lock(
/*===============*/
/*===============*/
ulint
precise_mode
,
/*!< in: LOCK_S or LOCK_X */
ulint
precise_mode
,
/*!< in: LOCK_S or LOCK_X */
unsigned
type_mode
,
/*!< in: LOCK_PREDICATE etc. */
unsigned
type_mode
,
/*!< in: LOCK_PREDICATE etc. */
const
buf_block_t
*
block
,
/*!< in: buffer block
const
page_id_t
id
,
/*!< in: page identifier */
containing the record */
lock_prdt_t
*
prdt
,
/*!< in: The predicate to be
lock_prdt_t
*
prdt
,
/*!< in: The predicate to be
attached to the new lock */
attached to the new lock */
const
trx_t
*
trx
)
/*!< in: transaction */
const
trx_t
*
trx
)
/*!< in: transaction */
...
@@ -243,7 +242,7 @@ lock_prdt_has_lock(
...
@@ -243,7 +242,7 @@ lock_prdt_has_lock(
ut_ad
(
!
(
precise_mode
&
LOCK_INSERT_INTENTION
));
ut_ad
(
!
(
precise_mode
&
LOCK_INSERT_INTENTION
));
for
(
lock
=
lock_rec_get_first
(
for
(
lock
=
lock_rec_get_first
(
lock_hash_get
(
type_mode
),
block
,
PRDT_HEAPNO
);
lock_hash_get
(
type_mode
),
id
,
PRDT_HEAPNO
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
PRDT_HEAPNO
,
lock
))
{
lock
=
lock_rec_get_next
(
PRDT_HEAPNO
,
lock
))
{
ut_ad
(
lock
->
type_mode
&
(
LOCK_PREDICATE
|
LOCK_PRDT_PAGE
));
ut_ad
(
lock
->
type_mode
&
(
LOCK_PREDICATE
|
LOCK_PRDT_PAGE
));
...
@@ -286,8 +285,7 @@ lock_prdt_other_has_conflicting(
...
@@ -286,8 +285,7 @@ lock_prdt_other_has_conflicting(
unsigned
mode
,
/*!< in: LOCK_S or LOCK_X,
unsigned
mode
,
/*!< in: LOCK_S or LOCK_X,
possibly ORed to LOCK_PREDICATE or
possibly ORed to LOCK_PREDICATE or
LOCK_PRDT_PAGE, LOCK_INSERT_INTENTION */
LOCK_PRDT_PAGE, LOCK_INSERT_INTENTION */
const
buf_block_t
*
block
,
/*!< in: buffer block containing
const
page_id_t
id
,
/*!< in: page identifier */
the record */
lock_prdt_t
*
prdt
,
/*!< in: Predicates (currently)
lock_prdt_t
*
prdt
,
/*!< in: Predicates (currently)
the Minimum Bounding Rectangle)
the Minimum Bounding Rectangle)
the new lock will be on */
the new lock will be on */
...
@@ -296,7 +294,7 @@ lock_prdt_other_has_conflicting(
...
@@ -296,7 +294,7 @@ lock_prdt_other_has_conflicting(
lock_sys
.
mutex_assert_locked
();
lock_sys
.
mutex_assert_locked
();
for
(
lock_t
*
lock
=
lock_rec_get_first
(
for
(
lock_t
*
lock
=
lock_rec_get_first
(
lock_hash_get
(
mode
),
block
,
PRDT_HEAPNO
);
lock_hash_get
(
mode
),
id
,
PRDT_HEAPNO
);
lock
!=
NULL
;
lock
!=
NULL
;
lock
=
lock_rec_get_next
(
PRDT_HEAPNO
,
lock
))
{
lock
=
lock_rec_get_next
(
PRDT_HEAPNO
,
lock
))
{
...
@@ -509,6 +507,7 @@ lock_prdt_insert_check_and_lock(
...
@@ -509,6 +507,7 @@ lock_prdt_insert_check_and_lock(
ut_ad
(
index
->
is_spatial
());
ut_ad
(
index
->
is_spatial
());
trx_t
*
trx
=
thr_get_trx
(
thr
);
trx_t
*
trx
=
thr_get_trx
(
thr
);
const
page_id_t
id
{
block
->
page
.
id
()};
lock_sys
.
mutex_lock
();
lock_sys
.
mutex_lock
();
...
@@ -521,7 +520,7 @@ lock_prdt_insert_check_and_lock(
...
@@ -521,7 +520,7 @@ lock_prdt_insert_check_and_lock(
lock_t
*
lock
;
lock_t
*
lock
;
/* Only need to check locks on prdt_hash */
/* Only need to check locks on prdt_hash */
lock
=
lock_rec_get_first
(
&
lock_sys
.
prdt_hash
,
block
,
PRDT_HEAPNO
);
lock
=
lock_rec_get_first
(
&
lock_sys
.
prdt_hash
,
id
,
PRDT_HEAPNO
);
if
(
lock
==
NULL
)
{
if
(
lock
==
NULL
)
{
lock_sys
.
mutex_unlock
();
lock_sys
.
mutex_unlock
();
...
@@ -547,7 +546,7 @@ lock_prdt_insert_check_and_lock(
...
@@ -547,7 +546,7 @@ lock_prdt_insert_check_and_lock(
const
ulint
mode
=
LOCK_X
|
LOCK_PREDICATE
|
LOCK_INSERT_INTENTION
;
const
ulint
mode
=
LOCK_X
|
LOCK_PREDICATE
|
LOCK_INSERT_INTENTION
;
const
lock_t
*
wait_for
=
lock_prdt_other_has_conflicting
(
const
lock_t
*
wait_for
=
lock_prdt_other_has_conflicting
(
mode
,
block
,
prdt
,
trx
);
mode
,
id
,
prdt
,
trx
);
if
(
wait_for
!=
NULL
)
{
if
(
wait_for
!=
NULL
)
{
rtr_mbr_t
*
mbr
=
prdt_get_mbr_from_prdt
(
prdt
);
rtr_mbr_t
*
mbr
=
prdt_get_mbr_from_prdt
(
prdt
);
...
@@ -563,7 +562,7 @@ lock_prdt_insert_check_and_lock(
...
@@ -563,7 +562,7 @@ lock_prdt_insert_check_and_lock(
NULL
,
/* FIXME: replicate SPATIAL INDEX locks */
NULL
,
/* FIXME: replicate SPATIAL INDEX locks */
#endif
#endif
LOCK_X
|
LOCK_PREDICATE
|
LOCK_INSERT_INTENTION
,
LOCK_X
|
LOCK_PREDICATE
|
LOCK_INSERT_INTENTION
,
block
,
PRDT_HEAPNO
,
index
,
thr
,
prdt
);
id
,
block
->
frame
,
PRDT_HEAPNO
,
index
,
thr
,
prdt
);
trx
->
mutex_unlock
();
trx
->
mutex_unlock
();
}
else
{
}
else
{
...
@@ -762,6 +761,7 @@ lock_prdt_lock(
...
@@ -762,6 +761,7 @@ lock_prdt_lock(
const
hash_table_t
&
hash
=
type_mode
==
LOCK_PREDICATE
const
hash_table_t
&
hash
=
type_mode
==
LOCK_PREDICATE
?
lock_sys
.
prdt_hash
?
lock_sys
.
prdt_hash
:
lock_sys
.
prdt_page_hash
;
:
lock_sys
.
prdt_page_hash
;
const
page_id_t
id
{
block
->
page
.
id
()};
/* 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
...
@@ -771,7 +771,7 @@ lock_prdt_lock(
...
@@ -771,7 +771,7 @@ lock_prdt_lock(
lock_sys
.
mutex_lock
();
lock_sys
.
mutex_lock
();
const
unsigned
prdt_mode
=
type_mode
|
mode
;
const
unsigned
prdt_mode
=
type_mode
|
mode
;
lock_t
*
lock
=
lock_sys
.
get_first
(
hash
,
block
->
page
.
id
()
);
lock_t
*
lock
=
lock_sys
.
get_first
(
hash
,
id
);
if
(
lock
==
NULL
)
{
if
(
lock
==
NULL
)
{
lock
=
lock_rec_create
(
lock
=
lock_rec_create
(
...
@@ -793,14 +793,14 @@ lock_prdt_lock(
...
@@ -793,14 +793,14 @@ lock_prdt_lock(
trx
->
mutex_lock
();
trx
->
mutex_lock
();
lock
=
lock_prdt_has_lock
(
lock
=
lock_prdt_has_lock
(
mode
,
type_mode
,
block
,
prdt
,
trx
);
mode
,
type_mode
,
id
,
prdt
,
trx
);
if
(
lock
==
NULL
)
{
if
(
lock
==
NULL
)
{
lock_t
*
wait_for
;
lock_t
*
wait_for
;
wait_for
=
lock_prdt_other_has_conflicting
(
wait_for
=
lock_prdt_other_has_conflicting
(
prdt_mode
,
block
,
prdt
,
trx
);
prdt_mode
,
id
,
prdt
,
trx
);
if
(
wait_for
!=
NULL
)
{
if
(
wait_for
!=
NULL
)
{
...
@@ -810,7 +810,7 @@ lock_prdt_lock(
...
@@ -810,7 +810,7 @@ lock_prdt_lock(
SPATIAL INDEX locks */
SPATIAL INDEX locks */
#endif
#endif
prdt_mode
,
prdt_mode
,
block
,
PRDT_HEAPNO
,
id
,
block
->
frame
,
PRDT_HEAPNO
,
index
,
thr
,
prdt
);
index
,
thr
,
prdt
);
}
else
{
}
else
{
...
@@ -921,8 +921,7 @@ lock_prdt_rec_move(
...
@@ -921,8 +921,7 @@ lock_prdt_rec_move(
/*===============*/
/*===============*/
const
buf_block_t
*
receiver
,
/*!< in: buffer block containing
const
buf_block_t
*
receiver
,
/*!< in: buffer block containing
the receiving record */
the receiving record */
const
buf_block_t
*
donator
)
/*!< in: buffer block containing
const
page_id_t
donator
)
/*!< in: target page */
the donating record */
{
{
lock_sys
.
mutex_lock
();
lock_sys
.
mutex_lock
();
...
@@ -948,19 +947,17 @@ lock_prdt_rec_move(
...
@@ -948,19 +947,17 @@ lock_prdt_rec_move(
}
}
/** Removes predicate lock objects set on an index page which is discarded.
/** Removes predicate lock objects set on an index page which is discarded.
@param[in]
block
page to be discarded
@param[in]
id
page to be discarded
@param[in] lock_hash lock hash */
@param[in] lock_hash lock hash */
void
void
lock_prdt_page_free_from_discard
(
lock_prdt_page_free_from_discard
(
const
page_id_t
id
,
hash_table_t
*
lock_hash
)
const
buf_block_t
*
block
,
hash_table_t
*
lock_hash
)
{
{
lock_t
*
lock
;
lock_t
*
lock
;
lock_t
*
next_lock
;
lock_t
*
next_lock
;
lock_sys
.
mutex_assert_locked
();
lock_sys
.
mutex_assert_locked
();
lock
=
lock_sys
.
get_first
(
*
lock_hash
,
block
->
page
.
id
()
);
lock
=
lock_sys
.
get_first
(
*
lock_hash
,
id
);
while
(
lock
!=
NULL
)
{
while
(
lock
!=
NULL
)
{
next_lock
=
lock_rec_get_next_on_page
(
lock
);
next_lock
=
lock_rec_get_next_on_page
(
lock
);
...
...
storage/innobase/row/row0mysql.cc
View file @
2e64513f
...
@@ -1988,7 +1988,7 @@ row_unlock_for_mysql(
...
@@ -1988,7 +1988,7 @@ row_unlock_for_mysql(
lock_rec_unlock
(
lock_rec_unlock
(
trx
,
trx
,
btr_pcur_get_block
(
pcur
),
btr_pcur_get_block
(
pcur
)
->
page
.
id
()
,
rec
,
rec
,
static_cast
<
enum
lock_mode
>
(
static_cast
<
enum
lock_mode
>
(
prebuilt
->
select_lock_type
));
prebuilt
->
select_lock_type
));
...
@@ -1998,7 +1998,8 @@ row_unlock_for_mysql(
...
@@ -1998,7 +1998,8 @@ row_unlock_for_mysql(
lock_rec_unlock
(
lock_rec_unlock
(
trx
,
trx
,
btr_pcur_get_block
(
clust_pcur
),
btr_pcur_get_block
(
clust_pcur
)
->
page
.
id
(),
rec
,
rec
,
static_cast
<
enum
lock_mode
>
(
static_cast
<
enum
lock_mode
>
(
prebuilt
->
select_lock_type
));
prebuilt
->
select_lock_type
));
...
...
storage/innobase/row/row0upd.cc
View file @
2e64513f
...
@@ -2840,7 +2840,7 @@ row_upd_clust_step(
...
@@ -2840,7 +2840,7 @@ row_upd_clust_step(
if
(
!
flags
&&
!
node
->
has_clust_rec_x_lock
)
{
if
(
!
flags
&&
!
node
->
has_clust_rec_x_lock
)
{
err
=
lock_clust_rec_modify_check_and_lock
(
err
=
lock_clust_rec_modify_check_and_lock
(
0
,
btr_pcur_get_block
(
pcur
),
btr_pcur_get_block
(
pcur
),
rec
,
index
,
offsets
,
thr
);
rec
,
index
,
offsets
,
thr
);
if
(
err
!=
DB_SUCCESS
)
{
if
(
err
!=
DB_SUCCESS
)
{
mtr
.
commit
();
mtr
.
commit
();
...
@@ -2850,8 +2850,8 @@ row_upd_clust_step(
...
@@ -2850,8 +2850,8 @@ row_upd_clust_step(
ut_ad
(
index
->
table
->
no_rollback
()
||
index
->
table
->
is_temporary
()
ut_ad
(
index
->
table
->
no_rollback
()
||
index
->
table
->
is_temporary
()
||
row_get_rec_trx_id
(
rec
,
index
,
offsets
)
==
trx
->
id
||
row_get_rec_trx_id
(
rec
,
index
,
offsets
)
==
trx
->
id
||
lock_trx_has_expl_x_lock
(
trx
,
index
->
table
,
||
lock_trx_has_expl_x_lock
(
*
trx
,
*
index
->
table
,
btr_pcur_get_block
(
pcur
),
btr_pcur_get_block
(
pcur
)
->
page
.
id
()
,
page_rec_get_heap_no
(
rec
)));
page_rec_get_heap_no
(
rec
)));
/* NOTE: the following function calls will also commit mtr */
/* NOTE: the following function calls will also commit mtr */
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment