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
b56fce41
Commit
b56fce41
authored
Dec 01, 2020
by
Marko Mäkelä
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
SHOW ENGINE INNODB MUTEX and INFORMATION_SCHEMA.INNODB_MUTEXES
parent
6490e99e
Changes
8
Show whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
176 additions
and
209 deletions
+176
-209
mysql-test/suite/innodb/r/innodb_skip_innodb_is_tables.result
...l-test/suite/innodb/r/innodb_skip_innodb_is_tables.result
+1
-1
mysql-test/suite/innodb_i_s/innodb_mutexes.result
mysql-test/suite/innodb_i_s/innodb_mutexes.result
+0
-2
storage/innobase/handler/ha_innodb.cc
storage/innobase/handler/ha_innodb.cc
+24
-90
storage/innobase/handler/i_s.cc
storage/innobase/handler/i_s.cc
+25
-79
storage/innobase/include/buf0buf.h
storage/innobase/include/buf0buf.h
+47
-2
storage/innobase/include/dict0dict.h
storage/innobase/include/dict0dict.h
+4
-3
storage/innobase/include/srw_lock.h
storage/innobase/include/srw_lock.h
+48
-22
storage/innobase/include/sux_lock.h
storage/innobase/include/sux_lock.h
+27
-10
No files found.
mysql-test/suite/innodb/r/innodb_skip_innodb_is_tables.result
View file @
b56fce41
...
@@ -364,7 +364,7 @@ SPACE NAME ENCRYPTION_SCHEME KEYSERVER_REQUESTS MIN_KEY_VERSION CURRENT_KEY_VERS
...
@@ -364,7 +364,7 @@ SPACE NAME ENCRYPTION_SCHEME KEYSERVER_REQUESTS MIN_KEY_VERSION CURRENT_KEY_VERS
Warnings:
Warnings:
Warning 1012 InnoDB: SELECTing from INFORMATION_SCHEMA.innodb_tablespaces_encryption but the InnoDB storage engine is not installed
Warning 1012 InnoDB: SELECTing from INFORMATION_SCHEMA.innodb_tablespaces_encryption but the InnoDB storage engine is not installed
select * from information_schema.innodb_mutexes;
select * from information_schema.innodb_mutexes;
NAME
CREATE_FILE CREATE_LINE
OS_WAITS
NAME OS_WAITS
Warnings:
Warnings:
Warning 1012 InnoDB: SELECTing from INFORMATION_SCHEMA.innodb_mutexes but the InnoDB storage engine is not installed
Warning 1012 InnoDB: SELECTing from INFORMATION_SCHEMA.innodb_mutexes but the InnoDB storage engine is not installed
select * from information_schema.innodb_sys_semaphore_waits;
select * from information_schema.innodb_sys_semaphore_waits;
...
...
mysql-test/suite/innodb_i_s/innodb_mutexes.result
View file @
b56fce41
...
@@ -2,7 +2,5 @@ SHOW CREATE TABLE INFORMATION_SCHEMA.INNODB_MUTEXES;
...
@@ -2,7 +2,5 @@ SHOW CREATE TABLE INFORMATION_SCHEMA.INNODB_MUTEXES;
Table Create Table
Table Create Table
INNODB_MUTEXES CREATE TEMPORARY TABLE `INNODB_MUTEXES` (
INNODB_MUTEXES CREATE TEMPORARY TABLE `INNODB_MUTEXES` (
`NAME` varchar(4000) NOT NULL DEFAULT '',
`NAME` varchar(4000) NOT NULL DEFAULT '',
`CREATE_FILE` varchar(4000) NOT NULL DEFAULT '',
`CREATE_LINE` int(11) unsigned NOT NULL DEFAULT 0,
`OS_WAITS` bigint(21) unsigned NOT NULL DEFAULT 0
`OS_WAITS` bigint(21) unsigned NOT NULL DEFAULT 0
) ENGINE=MEMORY DEFAULT CHARSET=utf8
) ENGINE=MEMORY DEFAULT CHARSET=utf8
storage/innobase/handler/ha_innodb.cc
View file @
b56fce41
...
@@ -15838,103 +15838,37 @@ innodb_show_mutex_status(
...
@@ -15838,103 +15838,37 @@ innodb_show_mutex_status(
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
/** Implement
s the SHOW MUTEX STATUS command
.
/** Implement
SHOW ENGINE INNODB MUTEX for rw-locks
.
@param
[in,out] hton
the innodb handlerton
@param
hton
the innodb handlerton
@param
[in,out] thd the MySQL query thread of the caller
@param
thd connection
@param
[in,out] stat_print
function for printing statistics
@param
fn
function for printing statistics
@return 0 on success. */
@return 0 on success. */
static
static
int
int
innodb_show_rwlock_status
(
innodb_show_rwlock_status
(
handlerton
*
ut_d
(
hton
),
THD
*
thd
,
stat_print_fn
*
fn
)
handlerton
*
#ifdef DBUG_ASSERT_EXISTS
hton
#endif
,
THD
*
thd
,
stat_print_fn
*
stat_print
)
{
{
DBUG_ENTER
(
"innodb_show_rwlock_status"
);
DBUG_ENTER
(
"innodb_show_rwlock_status"
);
ut_ad
(
hton
==
innodb_hton_ptr
);
DBUG_ASSERT
(
hton
==
innodb_hton_ptr
);
constexpr
size_t
prefix_len
=
sizeof
"waits="
-
1
;
#if 0 // FIXME
char
waits
[
prefix_len
+
20
+
1
];
const block_lock* block_rwlock= nullptr;
snprintf
(
waits
,
sizeof
waits
,
"waits="
UINT64PF
,
buf_pool
.
waited
());
ulint block_rwlock_oswait_count = 0;
uint hton_name_len = (uint) strlen(innobase_hton_name);
mutex_enter(&dict_sys.mutex);
for (const block_lock& rw_lock : rw_lock_list) {
if (rw_lock.count_os_wait == 0) {
continue;
}
int buf1len;
char buf1[IO_SIZE];
if (rw_lock.is_block_lock) {
block_rwlock = &rw_lock;
block_rwlock_oswait_count += rw_lock.count_os_wait;
continue;
}
buf1len = snprintf(
buf1, sizeof buf1, "rwlock: %s:%u",
innobase_basename(rw_lock.cfile_name),
rw_lock.cline);
int buf2len;
char buf2[IO_SIZE];
buf2len = snprintf(
buf2, sizeof buf2, "waits=%u",
rw_lock.count_os_wait);
if (stat_print(thd, innobase_hton_name,
hton_name_len,
buf1, static_cast<uint>(buf1len),
buf2, static_cast<uint>(buf2len))) {
mutex_exit(&dict_sys.mutex);
DBUG_RETURN(1);
}
}
if (block_rwlock != NULL) {
int buf1len;
char buf1[IO_SIZE];
buf1len = snprintf(
buf1, sizeof buf1, "sum rwlock: %s:%u",
innobase_basename(block_rwlock->cfile_name),
block_rwlock->cline);
int buf2len;
char buf2[IO_SIZE];
buf2len = snprintf(
buf2, sizeof buf2, "waits=" ULINTPF,
block_rwlock_oswait_count);
if (stat_print(thd, innobase_hton_name,
hton_name_len,
buf1, static_cast<uint>(buf1len),
buf2, static_cast<uint>(buf2len))) {
mutex_exit(&dict_sys.mutex);
if
(
fn
(
thd
,
STRING_WITH_LEN
(
innobase_hton_name
),
STRING_WITH_LEN
(
"buf_block_t::lock"
),
waits
,
strlen
(
waits
)))
DBUG_RETURN
(
1
);
DBUG_RETURN
(
1
);
}
}
mutex_exit(&dict_sys.mutex);
DBUG_RETURN
(
!
dict_sys
.
for_each_index
([
&
](
const
dict_index_t
&
i
)
#endif
{
DBUG_RETURN
(
0
);
uint32_t
waited
=
i
.
lock
.
waited
();
if
(
!
waited
)
return
true
;
snprintf
(
waits
+
prefix_len
,
sizeof
waits
-
prefix_len
,
"%u"
,
waited
);
std
::
ostringstream
s
;
s
<<
i
.
name
<<
'('
<<
i
.
table
->
name
<<
')'
;
return
!
fn
(
thd
,
STRING_WITH_LEN
(
innobase_hton_name
),
s
.
str
().
data
(),
s
.
str
().
size
(),
waits
,
strlen
(
waits
));
}));
}
}
/** Implements the SHOW MUTEX STATUS command.
/** Implements the SHOW MUTEX STATUS command.
...
...
storage/innobase/handler/i_s.cc
View file @
b56fce41
...
@@ -6885,26 +6885,15 @@ namespace Show {
...
@@ -6885,26 +6885,15 @@ namespace Show {
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_MUTEXES */
/* Fields of the dynamic table INFORMATION_SCHEMA.INNODB_MUTEXES */
static
ST_FIELD_INFO
innodb_mutexes_fields_info
[]
=
static
ST_FIELD_INFO
innodb_mutexes_fields_info
[]
=
{
{
#define MUTEXES_NAME 0
Column
(
"NAME"
,
Varchar
(
OS_FILE_MAX_PATH
),
NOT_NULL
),
Column
(
"NAME"
,
Varchar
(
OS_FILE_MAX_PATH
),
NOT_NULL
),
#define MUTEXES_CREATE_FILE 1
Column
(
"CREATE_FILE"
,
Varchar
(
OS_FILE_MAX_PATH
),
NOT_NULL
),
#define MUTEXES_CREATE_LINE 2
Column
(
"CREATE_LINE"
,
ULong
(),
NOT_NULL
),
#define MUTEXES_OS_WAITS 3
Column
(
"OS_WAITS"
,
ULonglong
(),
NOT_NULL
),
Column
(
"OS_WAITS"
,
ULonglong
(),
NOT_NULL
),
CEnd
()
CEnd
()
};
};
}
// namespace Show
}
// namespace Show
/*******************************************************************//**
/*******************************************************************//**
Function to populate INFORMATION_SCHEMA.INNODB_MUTEXES table.
Function to populate INFORMATION_SCHEMA.INNODB_MUTEXES table.
Loop through each record in mutex and rw_lock lists, and extract the column
@see innodb_show_rwlock_status
information and fill the INFORMATION_SCHEMA.INNODB_MUTEXES table.
@return 0 on success */
@return 0 on success */
static
static
int
int
...
@@ -6917,74 +6906,31 @@ i_s_innodb_mutexes_fill_table(
...
@@ -6917,74 +6906,31 @@ i_s_innodb_mutexes_fill_table(
DBUG_ENTER
(
"i_s_innodb_mutexes_fill_table"
);
DBUG_ENTER
(
"i_s_innodb_mutexes_fill_table"
);
RETURN_IF_INNODB_NOT_STARTED
(
tables
->
schema_table_name
.
str
);
RETURN_IF_INNODB_NOT_STARTED
(
tables
->
schema_table_name
.
str
);
/* deny access to user without PROCESS_ACL privilege */
if
(
check_global_access
(
thd
,
PROCESS_ACL
))
if
(
check_global_access
(
thd
,
PROCESS_ACL
))
{
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
#if 0 // FIXME
ulint block_lock_oswait_count = 0;
const block_lock* block_lock= nullptr;
Field** fields = tables->table->field;
struct Locking
{
Locking() { mutex_enter(&dict_sys.mutex); }
~Locking() { mutex_exit(&dict_sys.mutex); }
} locking;
char lock_name[sizeof "buf0dump.cc:12345"];
for (const block_lock& lock : rw_lock_list) {
if (lock.count_os_wait == 0) {
continue;
}
if (buf_pool.is_block_lock(&lock)) {
Field
**
fields
=
tables
->
table
->
field
;
block_lock = &lock;
OK
(
fields
[
0
]
->
store
(
STRING_WITH_LEN
(
"buf_block_t::lock"
),
block_lock_oswait_count += lock.count_os_wait;
system_charset_info
));
continue;
OK
(
fields
[
1
]
->
store
(
buf_pool
.
waited
(),
true
));
}
fields
[
0
]
->
set_notnull
();
fields
[
1
]
->
set_notnull
();
const char* basename = innobase_basename(
lock.cfile_name);
snprintf(lock_name, sizeof lock_name, "%s:%u",
basename, lock.cline);
OK(field_store_string(fields[MUTEXES_NAME],
lock_name));
OK(field_store_string(fields[MUTEXES_CREATE_FILE],
basename));
OK(fields[MUTEXES_CREATE_LINE]->store(lock.cline,
true));
fields[MUTEXES_CREATE_LINE]->set_notnull();
OK(fields[MUTEXES_OS_WAITS]->store(lock.count_os_wait,
true));
fields[MUTEXES_OS_WAITS]->set_notnull();
OK(schema_table_store_record(thd, tables->table));
}
if (block_lock) {
char buf1[IO_SIZE];
snprintf(buf1, sizeof buf1, "combined %s",
innobase_basename(block_lock->cfile_name));
OK(field_store_string(fields[MUTEXES_NAME],
"buf_block_t::lock"));
OK(field_store_string(fields[MUTEXES_CREATE_FILE],
buf1));
OK(fields[MUTEXES_CREATE_LINE]->store(block_lock->cline,
true));
fields[MUTEXES_CREATE_LINE]->set_notnull();
OK(fields[MUTEXES_OS_WAITS]->store(
block_lock_oswait_count, true));
fields[MUTEXES_OS_WAITS]->set_notnull();
OK
(
schema_table_store_record
(
thd
,
tables
->
table
));
OK
(
schema_table_store_record
(
thd
,
tables
->
table
));
}
#endif
DBUG_RETURN
(
0
);
DBUG_RETURN
(
!
dict_sys
.
for_each_index
([
&
](
const
dict_index_t
&
i
)
{
uint32_t
waited
=
i
.
lock
.
waited
();
if
(
!
waited
)
return
true
;
if
(
fields
[
1
]
->
store
(
waited
,
true
))
return
false
;
std
::
ostringstream
s
;
s
<<
i
.
name
<<
'('
<<
i
.
table
->
name
<<
')'
;
return
!
fields
[
0
]
->
store
(
s
.
str
().
data
(),
s
.
str
().
size
(),
system_charset_info
)
&&
!
schema_table_store_record
(
thd
,
tables
->
table
);
}));
}
}
/*******************************************************************//**
/*******************************************************************//**
...
...
storage/innobase/include/buf0buf.h
View file @
b56fce41
...
@@ -1326,6 +1326,16 @@ class buf_pool_t
...
@@ -1326,6 +1326,16 @@ class buf_pool_t
@return whether the allocation succeeded */
@return whether the allocation succeeded */
inline
bool
create
(
size_t
bytes
);
inline
bool
create
(
size_t
bytes
);
/** @return sum of buf_block_t::lock::waited() */
uint64_t
waited
()
const
{
uint64_t
total_waited
=
0
;
for
(
const
buf_block_t
*
block
=
blocks
,
*
const
end
=
blocks
+
size
;
block
!=
end
;
block
++
)
total_waited
+=
block
->
lock
.
waited
();
return
total_waited
;
}
#ifdef UNIV_DEBUG
#ifdef UNIV_DEBUG
/** Find a block that points to a ROW_FORMAT=COMPRESSED page
/** Find a block that points to a ROW_FORMAT=COMPRESSED page
@param data pointer to the start of a ROW_FORMAT=COMPRESSED page frame
@param data pointer to the start of a ROW_FORMAT=COMPRESSED page frame
...
@@ -1403,6 +1413,19 @@ class buf_pool_t
...
@@ -1403,6 +1413,19 @@ class buf_pool_t
return
size
;
return
size
;
}
}
/** @return sum of buf_block_t::lock::waited() */
uint64_t
waited
()
{
ut_ad
(
is_initialised
());
uint64_t
waited_count
=
0
;
page_hash
.
read_lock_all
();
/* prevent any race with resize() */
for
(
const
chunk_t
*
chunk
=
chunks
,
*
const
end
=
chunks
+
n_chunks
;
chunk
!=
end
;
chunk
++
)
waited_count
+=
chunks
->
waited
();
page_hash
.
read_unlock_all
();
return
waited_count
;
}
/** Determine whether a frame is intended to be withdrawn during resize().
/** Determine whether a frame is intended to be withdrawn during resize().
@param ptr pointer within a buf_block_t::frame
@param ptr pointer within a buf_block_t::frame
@return whether the frame will be withdrawn */
@return whether the frame will be withdrawn */
...
@@ -1410,7 +1433,7 @@ class buf_pool_t
...
@@ -1410,7 +1433,7 @@ class buf_pool_t
{
{
ut_ad
(
curr_size
<
old_size
);
ut_ad
(
curr_size
<
old_size
);
#ifdef SAFE_MUTEX
#ifdef SAFE_MUTEX
if
(
resiz
ing
.
load
(
std
::
memory_order_relaxed
))
if
(
resiz
e_in_progress
(
))
mysql_mutex_assert_owner
(
&
mutex
);
mysql_mutex_assert_owner
(
&
mutex
);
#endif
/* SAFE_MUTEX */
#endif
/* SAFE_MUTEX */
...
@@ -1430,7 +1453,7 @@ class buf_pool_t
...
@@ -1430,7 +1453,7 @@ class buf_pool_t
{
{
ut_ad
(
curr_size
<
old_size
);
ut_ad
(
curr_size
<
old_size
);
#ifdef SAFE_MUTEX
#ifdef SAFE_MUTEX
if
(
resiz
ing
.
load
(
std
::
memory_order_relaxed
))
if
(
resiz
e_in_progress
(
))
mysql_mutex_assert_owner
(
&
mutex
);
mysql_mutex_assert_owner
(
&
mutex
);
#endif
/* SAFE_MUTEX */
#endif
/* SAFE_MUTEX */
...
@@ -1804,6 +1827,28 @@ class buf_pool_t
...
@@ -1804,6 +1827,28 @@ class buf_pool_t
}
}
}
}
/** Acquire all latches in shared mode */
void
read_lock_all
()
{
for
(
auto
n
=
pad
(
n_cells
)
&
~
ELEMENTS_PER_LATCH
;;
n
-=
ELEMENTS_PER_LATCH
+
1
)
{
reinterpret_cast
<
page_hash_latch
&>
(
array
[
n
]).
read_lock
();
if
(
!
n
)
break
;
}
}
/** Release all latches in shared mode */
void
read_unlock_all
()
{
for
(
auto
n
=
pad
(
n_cells
)
&
~
ELEMENTS_PER_LATCH
;;
n
-=
ELEMENTS_PER_LATCH
+
1
)
{
reinterpret_cast
<
page_hash_latch
&>
(
array
[
n
]).
read_unlock
();
if
(
!
n
)
break
;
}
}
/** Exclusively aqcuire all latches */
/** Exclusively aqcuire all latches */
inline
void
write_lock_all
();
inline
void
write_lock_all
();
...
...
storage/innobase/include/dict0dict.h
View file @
b56fce41
...
@@ -1537,7 +1537,8 @@ class dict_sys_t
...
@@ -1537,7 +1537,8 @@ class dict_sys_t
@param t table
@param t table
@retval false if f returned false
@retval false if f returned false
@retval true if f never returned false */
@retval true if f never returned false */
template
<
typename
F
>
inline
bool
for_each_index
(
const
F
&
f
,
dict_table_t
*
t
);
template
<
typename
F
>
inline
bool
for_each_index
(
const
F
&
f
,
const
dict_table_t
*
t
);
public:
public:
/** Invoke f on each index of each persistent table, until it returns false
/** Invoke f on each index of each persistent table, until it returns false
@retval false if f returned false
@retval false if f returned false
...
@@ -1611,9 +1612,9 @@ extern dict_sys_t dict_sys;
...
@@ -1611,9 +1612,9 @@ extern dict_sys_t dict_sys;
#define dict_sys_unlock() dict_sys.unlock()
#define dict_sys_unlock() dict_sys.unlock()
template
<
typename
F
>
template
<
typename
F
>
inline
bool
dict_sys_t
::
for_each_index
(
const
F
&
f
,
dict_table_t
*
t
)
inline
bool
dict_sys_t
::
for_each_index
(
const
F
&
f
,
const
dict_table_t
*
t
)
{
{
dict_index_t
*
i
=
UT_LIST_GET_FIRST
(
t
->
indexes
);
const
dict_index_t
*
i
=
UT_LIST_GET_FIRST
(
t
->
indexes
);
do
do
{
{
if
(
!
i
->
is_corrupted
()
&&
!
f
(
*
i
))
if
(
!
i
->
is_corrupted
()
&&
!
f
(
*
i
))
...
...
storage/innobase/include/srw_lock.h
View file @
b56fce41
...
@@ -78,13 +78,39 @@ class srw_lock_low final : private rw_lock
...
@@ -78,13 +78,39 @@ class srw_lock_low final : private rw_lock
#endif
#endif
bool
rd_lock_try
()
{
uint32_t
l
;
return
read_trylock
(
l
);
}
bool
rd_lock_try
()
{
uint32_t
l
;
return
read_trylock
(
l
);
}
bool
wr_lock_try
()
{
return
write_trylock
();
}
bool
wr_lock_try
()
{
return
write_trylock
();
}
template
<
bool
update
=
false
>
template
<
bool
update
=
false
>
void
rd_lock
()
{
uint32_t
l
;
if
(
!
read_trylock
(
l
))
read_lock
(
l
);
}
bool
rd_lock
()
void
u_lock
()
{
uint32_t
l
;
if
(
!
update_trylock
(
l
))
update_lock
(
l
);
}
{
uint32_t
l
;
if
(
read_trylock
(
l
))
return
true
;
read_lock
(
l
);
return
false
;
}
bool
u_lock
()
{
uint32_t
l
;
if
(
update_trylock
(
l
))
return
true
;
update_lock
(
l
);
return
false
;
}
bool
u_lock_try
()
{
uint32_t
l
;
return
update_trylock
(
l
);
}
bool
u_lock_try
()
{
uint32_t
l
;
return
update_trylock
(
l
);
}
void
u_wr_upgrade
()
{
if
(
!
upgrade_trylock
())
write_lock
(
true
);
}
bool
u_wr_upgrade
()
template
<
bool
update
=
false
>
{
void
wr_lock
()
{
if
(
!
write_trylock
())
write_lock
(
false
);
}
if
(
upgrade_trylock
())
return
true
;
write_lock
(
true
);
return
false
;
}
template
<
bool
update
=
false
>
bool
wr_lock
()
{
if
(
write_trylock
())
return
true
;
write_lock
(
false
);
return
false
;
}
void
rd_unlock
();
void
rd_unlock
();
void
u_unlock
();
void
u_unlock
();
void
wr_unlock
();
void
wr_unlock
();
...
@@ -122,7 +148,7 @@ class srw_lock
...
@@ -122,7 +148,7 @@ class srw_lock
lock
.
destroy
();
lock
.
destroy
();
}
}
template
<
bool
update
=
false
>
template
<
bool
update
=
false
>
void
rd_lock
(
const
char
*
file
,
unsigned
line
)
bool
rd_lock
(
const
char
*
file
,
unsigned
line
)
{
{
if
(
pfs_psi
)
if
(
pfs_psi
)
{
{
...
@@ -130,12 +156,12 @@ class srw_lock
...
@@ -130,12 +156,12 @@ class srw_lock
PSI_rwlock_locker
*
locker
=
PSI_RWLOCK_CALL
(
start_rwlock_rdwait
)
PSI_rwlock_locker
*
locker
=
PSI_RWLOCK_CALL
(
start_rwlock_rdwait
)
(
&
state
,
pfs_psi
,
update
?
PSI_RWLOCK_SHAREDLOCK
:
PSI_RWLOCK_READLOCK
,
(
&
state
,
pfs_psi
,
update
?
PSI_RWLOCK_SHAREDLOCK
:
PSI_RWLOCK_READLOCK
,
file
,
line
);
file
,
line
);
lock
.
rd_lock
();
bool
no_wait
=
lock
.
rd_lock
();
if
(
locker
)
if
(
locker
)
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
return
;
return
no_wait
;
}
}
lock
.
rd_lock
();
return
lock
.
rd_lock
();
}
}
void
rd_unlock
()
void
rd_unlock
()
{
{
...
@@ -143,19 +169,19 @@ class srw_lock
...
@@ -143,19 +169,19 @@ class srw_lock
PSI_RWLOCK_CALL
(
unlock_rwlock
)(
pfs_psi
);
PSI_RWLOCK_CALL
(
unlock_rwlock
)(
pfs_psi
);
lock
.
rd_unlock
();
lock
.
rd_unlock
();
}
}
void
u_lock
(
const
char
*
file
,
unsigned
line
)
bool
u_lock
(
const
char
*
file
,
unsigned
line
)
{
{
if
(
pfs_psi
)
if
(
pfs_psi
)
{
{
PSI_rwlock_locker_state
state
;
PSI_rwlock_locker_state
state
;
PSI_rwlock_locker
*
locker
=
PSI_RWLOCK_CALL
(
start_rwlock_wrwait
)
PSI_rwlock_locker
*
locker
=
PSI_RWLOCK_CALL
(
start_rwlock_wrwait
)
(
&
state
,
pfs_psi
,
PSI_RWLOCK_SHAREDEXCLUSIVELOCK
,
file
,
line
);
(
&
state
,
pfs_psi
,
PSI_RWLOCK_SHAREDEXCLUSIVELOCK
,
file
,
line
);
lock
.
u_lock
();
bool
no_wait
=
lock
.
u_lock
();
if
(
locker
)
if
(
locker
)
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
return
;
return
no_wait
;
}
}
lock
.
u_lock
();
return
lock
.
u_lock
();
}
}
void
u_unlock
()
void
u_unlock
()
{
{
...
@@ -164,7 +190,7 @@ class srw_lock
...
@@ -164,7 +190,7 @@ class srw_lock
lock
.
u_unlock
();
lock
.
u_unlock
();
}
}
template
<
bool
update
=
false
>
template
<
bool
update
=
false
>
void
wr_lock
(
const
char
*
file
,
unsigned
line
)
bool
wr_lock
(
const
char
*
file
,
unsigned
line
)
{
{
if
(
pfs_psi
)
if
(
pfs_psi
)
{
{
...
@@ -173,12 +199,12 @@ class srw_lock
...
@@ -173,12 +199,12 @@ class srw_lock
(
&
state
,
pfs_psi
,
(
&
state
,
pfs_psi
,
update
?
PSI_RWLOCK_EXCLUSIVELOCK
:
PSI_RWLOCK_WRITELOCK
,
update
?
PSI_RWLOCK_EXCLUSIVELOCK
:
PSI_RWLOCK_WRITELOCK
,
file
,
line
);
file
,
line
);
lock
.
wr_lock
();
bool
no_wait
=
lock
.
wr_lock
();
if
(
locker
)
if
(
locker
)
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
return
;
return
no_wait
;
}
}
lock
.
wr_lock
();
return
lock
.
wr_lock
();
}
}
void
wr_unlock
()
void
wr_unlock
()
{
{
...
@@ -186,19 +212,19 @@ class srw_lock
...
@@ -186,19 +212,19 @@ class srw_lock
PSI_RWLOCK_CALL
(
unlock_rwlock
)(
pfs_psi
);
PSI_RWLOCK_CALL
(
unlock_rwlock
)(
pfs_psi
);
lock
.
wr_unlock
();
lock
.
wr_unlock
();
}
}
void
u_wr_upgrade
(
const
char
*
file
,
unsigned
line
)
bool
u_wr_upgrade
(
const
char
*
file
,
unsigned
line
)
{
{
if
(
pfs_psi
)
if
(
pfs_psi
)
{
{
PSI_rwlock_locker_state
state
;
PSI_rwlock_locker_state
state
;
PSI_rwlock_locker
*
locker
=
PSI_RWLOCK_CALL
(
start_rwlock_wrwait
)
PSI_rwlock_locker
*
locker
=
PSI_RWLOCK_CALL
(
start_rwlock_wrwait
)
(
&
state
,
pfs_psi
,
PSI_RWLOCK_EXCLUSIVELOCK
,
file
,
line
);
(
&
state
,
pfs_psi
,
PSI_RWLOCK_EXCLUSIVELOCK
,
file
,
line
);
lock
.
u_wr_upgrade
();
bool
no_wait
=
lock
.
u_wr_upgrade
();
if
(
locker
)
if
(
locker
)
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
PSI_RWLOCK_CALL
(
end_rwlock_rdwait
)(
locker
,
0
);
return
;
return
no_wait
;
}
}
lock
.
u_wr_upgrade
();
return
lock
.
u_wr_upgrade
();
}
}
bool
rd_lock_try
()
{
return
lock
.
rd_lock_try
();
}
bool
rd_lock_try
()
{
return
lock
.
rd_lock_try
();
}
bool
u_lock_try
()
{
return
lock
.
u_lock_try
();
}
bool
u_lock_try
()
{
return
lock
.
u_lock_try
();
}
...
...
storage/innobase/include/sux_lock.h
View file @
b56fce41
...
@@ -43,6 +43,8 @@ class sux_lock final
...
@@ -43,6 +43,8 @@ class sux_lock final
#endif
#endif
/** Numbers of U and X locks. Protected by lock. */
/** Numbers of U and X locks. Protected by lock. */
uint32_t
recursive
;
uint32_t
recursive
;
/** Number of blocking waits */
std
::
atomic
<
uint32_t
>
waits
;
#ifdef UNIV_DEBUG
#ifdef UNIV_DEBUG
/** Protects readers */
/** Protects readers */
mutable
srw_mutex
readers_lock
;
mutable
srw_mutex
readers_lock
;
...
@@ -67,6 +69,7 @@ class sux_lock final
...
@@ -67,6 +69,7 @@ class sux_lock final
lock
.
SRW_LOCK_INIT
(
key
);
lock
.
SRW_LOCK_INIT
(
key
);
ut_ad
(
!
writer
.
load
(
std
::
memory_order_relaxed
));
ut_ad
(
!
writer
.
load
(
std
::
memory_order_relaxed
));
ut_ad
(
!
recursive
);
ut_ad
(
!
recursive
);
ut_ad
(
!
waits
.
load
(
std
::
memory_order_relaxed
));
ut_d
(
readers_lock
.
init
());
ut_d
(
readers_lock
.
init
());
ut_ad
(
!
readers
.
load
(
std
::
memory_order_relaxed
));
ut_ad
(
!
readers
.
load
(
std
::
memory_order_relaxed
));
}
}
...
@@ -88,8 +91,11 @@ class sux_lock final
...
@@ -88,8 +91,11 @@ class sux_lock final
lock
.
destroy
();
lock
.
destroy
();
}
}
/** @return number of blocking waits */
uint32_t
waited
()
const
{
return
waits
.
load
(
std
::
memory_order_relaxed
);
}
/** needed for dict_index_t::clone() */
/** needed for dict_index_t::clone() */
void
operator
=
(
const
sux_lock
&
)
{}
inline
void
operator
=
(
const
sux_lock
&
);
#ifdef UNIV_DEBUG
#ifdef UNIV_DEBUG
/** @return whether no recursive locks are being held */
/** @return whether no recursive locks are being held */
...
@@ -281,6 +287,9 @@ class sux_lock final
...
@@ -281,6 +287,9 @@ class sux_lock final
};
};
typedef
sux_lock
<
srw_lock_low
>
block_lock
;
typedef
sux_lock
<
srw_lock_low
>
block_lock
;
/** needed for dict_index_t::clone() */
template
<
>
inline
void
sux_lock
<
srw_lock
>::
operator
=
(
const
sux_lock
&
)
{}
#ifndef UNIV_PFS_RWLOCK
#ifndef UNIV_PFS_RWLOCK
typedef
block_lock
index_lock
;
typedef
block_lock
index_lock
;
#else
#else
...
@@ -291,6 +300,7 @@ template<> inline void sux_lock<srw_lock_low>::init()
...
@@ -291,6 +300,7 @@ template<> inline void sux_lock<srw_lock_low>::init()
lock
.
init
();
lock
.
init
();
ut_ad
(
!
writer
.
load
(
std
::
memory_order_relaxed
));
ut_ad
(
!
writer
.
load
(
std
::
memory_order_relaxed
));
ut_ad
(
!
recursive
);
ut_ad
(
!
recursive
);
ut_ad
(
!
waits
.
load
(
std
::
memory_order_relaxed
));
ut_d
(
readers_lock
.
init
());
ut_d
(
readers_lock
.
init
());
ut_ad
(
!
readers
.
load
(
std
::
memory_order_relaxed
));
ut_ad
(
!
readers
.
load
(
std
::
memory_order_relaxed
));
}
}
...
@@ -300,7 +310,8 @@ inline void sux_lock<srw_lock>::s_lock(const char *file, unsigned line)
...
@@ -300,7 +310,8 @@ inline void sux_lock<srw_lock>::s_lock(const char *file, unsigned line)
{
{
ut_ad
(
!
have_x
());
ut_ad
(
!
have_x
());
ut_ad
(
!
have_s
());
ut_ad
(
!
have_s
());
lock
.
template
rd_lock
<
true
>(
file
,
line
);
if
(
!
lock
.
template
rd_lock
<
true
>(
file
,
line
))
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
ut_d
(
s_lock_register
());
ut_d
(
s_lock_register
());
}
}
...
@@ -312,7 +323,8 @@ inline void sux_lock<srw_lock>::u_lock(const char *file, unsigned line)
...
@@ -312,7 +323,8 @@ inline void sux_lock<srw_lock>::u_lock(const char *file, unsigned line)
writer_recurse
<
true
>
();
writer_recurse
<
true
>
();
else
else
{
{
lock
.
u_lock
(
file
,
line
);
if
(
!
lock
.
u_lock
(
file
,
line
))
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
ut_ad
(
!
recursive
);
ut_ad
(
!
recursive
);
recursive
=
RECURSIVE_U
;
recursive
=
RECURSIVE_U
;
set_first_owner
(
id
);
set_first_owner
(
id
);
...
@@ -327,7 +339,8 @@ inline void sux_lock<srw_lock>::x_lock(const char *file, unsigned line)
...
@@ -327,7 +339,8 @@ inline void sux_lock<srw_lock>::x_lock(const char *file, unsigned line)
writer_recurse
<
false
>
();
writer_recurse
<
false
>
();
else
else
{
{
lock
.
template
wr_lock
<
true
>(
file
,
line
);
if
(
!
lock
.
template
wr_lock
<
true
>(
file
,
line
))
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
ut_ad
(
!
recursive
);
ut_ad
(
!
recursive
);
recursive
=
RECURSIVE_X
;
recursive
=
RECURSIVE_X
;
set_first_owner
(
id
);
set_first_owner
(
id
);
...
@@ -338,7 +351,8 @@ template<>
...
@@ -338,7 +351,8 @@ template<>
inline
void
sux_lock
<
srw_lock
>::
u_x_upgrade
(
const
char
*
file
,
unsigned
line
)
inline
void
sux_lock
<
srw_lock
>::
u_x_upgrade
(
const
char
*
file
,
unsigned
line
)
{
{
ut_ad
(
have_u_not_x
());
ut_ad
(
have_u_not_x
());
lock
.
u_wr_upgrade
(
file
,
line
);
if
(
!
lock
.
u_wr_upgrade
(
file
,
line
))
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
recursive
/=
RECURSIVE_U
;
recursive
/=
RECURSIVE_U
;
}
}
#endif
#endif
...
@@ -348,11 +362,11 @@ inline void sux_lock<srw_lock_low>::s_lock()
...
@@ -348,11 +362,11 @@ inline void sux_lock<srw_lock_low>::s_lock()
{
{
ut_ad
(
!
have_x
());
ut_ad
(
!
have_x
());
ut_ad
(
!
have_s
());
ut_ad
(
!
have_s
());
lock
.
template
rd_lock
<
true
>();
if
(
!
lock
.
template
rd_lock
<
true
>())
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
ut_d
(
s_lock_register
());
ut_d
(
s_lock_register
());
}
}
template
<
>
template
<
>
inline
void
sux_lock
<
srw_lock_low
>::
u_lock
()
inline
void
sux_lock
<
srw_lock_low
>::
u_lock
()
{
{
...
@@ -361,7 +375,8 @@ inline void sux_lock<srw_lock_low>::u_lock()
...
@@ -361,7 +375,8 @@ inline void sux_lock<srw_lock_low>::u_lock()
writer_recurse
<
true
>
();
writer_recurse
<
true
>
();
else
else
{
{
lock
.
u_lock
();
if
(
!
lock
.
u_lock
())
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
ut_ad
(
!
recursive
);
ut_ad
(
!
recursive
);
recursive
=
RECURSIVE_U
;
recursive
=
RECURSIVE_U
;
set_first_owner
(
id
);
set_first_owner
(
id
);
...
@@ -379,7 +394,8 @@ inline void sux_lock<srw_lock_low>::x_lock(bool for_io)
...
@@ -379,7 +394,8 @@ inline void sux_lock<srw_lock_low>::x_lock(bool for_io)
}
}
else
else
{
{
lock
.
template
wr_lock
<
true
>();
if
(
!
lock
.
template
wr_lock
<
true
>())
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
ut_ad
(
!
recursive
);
ut_ad
(
!
recursive
);
recursive
=
RECURSIVE_X
;
recursive
=
RECURSIVE_X
;
set_first_owner
(
for_io
?
FOR_IO
:
id
);
set_first_owner
(
for_io
?
FOR_IO
:
id
);
...
@@ -390,7 +406,8 @@ template<>
...
@@ -390,7 +406,8 @@ template<>
inline
void
sux_lock
<
srw_lock_low
>::
u_x_upgrade
()
inline
void
sux_lock
<
srw_lock_low
>::
u_x_upgrade
()
{
{
ut_ad
(
have_u_not_x
());
ut_ad
(
have_u_not_x
());
lock
.
u_wr_upgrade
();
if
(
!
lock
.
u_wr_upgrade
())
waits
.
fetch_add
(
1
,
std
::
memory_order_relaxed
);
recursive
/=
RECURSIVE_U
;
recursive
/=
RECURSIVE_U
;
}
}
...
...
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