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
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
fa5529ee
Commit
fa5529ee
authored
Dec 05, 2008
by
Mikael Ronstrom
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Added memory barriers to the portability layer in InnoDB
Removed dependency on pthread being an int
parent
2ed1383f
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
61 additions
and
31 deletions
+61
-31
storage/innobase/include/os0sync.ic
storage/innobase/include/os0sync.ic
+40
-0
storage/innobase/include/sync0rw.ic
storage/innobase/include/sync0rw.ic
+11
-8
storage/innobase/sync/sync0rw.c
storage/innobase/sync/sync0rw.c
+10
-23
No files found.
storage/innobase/include/os0sync.ic
View file @
fa5529ee
...
...
@@ -69,6 +69,46 @@ os_compare_and_swap(
#endif
}
/**************************************************************
Memory barrier for load */
UNIV_INLINE
void
os_memory_barrier_load()
{
#ifdef HAVE_GCC_ATOMIC_BUILTINS
__sync_synchronize();
#elif HAVE_SOLARIS_ATOMIC
membar_consumer();
#endif
}
/**************************************************************
Memory barrier for store */
UNIV_INLINE
void
os_memory_barrier_store()
{
#ifdef HAVE_GCC_ATOMIC_BUILTINS
__sync_synchronize();
#elif HAVE_SOLARIS_ATOMIC
membar_producer();
#endif
}
/**************************************************************
Memory barrier */
UNIV_INLINE
void
os_memory_barrier()
{
#ifdef HAVE_GCC_ATOMIC_BUILTINS
__sync_synchronize();
#elif HAVE_SOLARIS_ATOMIC
membar_enter();
#endif
}
/**************************************************************
Atomic increment for InnoDB. Currently requires GCC atomic builtins. */
UNIV_INLINE
...
...
storage/innobase/include/sync0rw.ic
View file @
fa5529ee
...
...
@@ -254,6 +254,7 @@ rw_lock_s_lock_low(
return(TRUE); /* locking succeeded */
}
/* TODO: The "direct" functions are not used. Remove them? */
/**********************************************************************
Low-level function which locks an rw-lock in s-mode when we know that it
is possible and none else is currently accessing the rw-lock structure.
...
...
@@ -279,6 +280,7 @@ rw_lock_s_lock_direct(
#endif
}
/* TODO: The "direct" functions are not used. Remove them? */
/**********************************************************************
Low-level function which locks an rw-lock in x-mode when we know that it
is not locked and none else is currently accessing the rw-lock structure.
...
...
@@ -451,6 +453,7 @@ rw_lock_s_unlock_func(
#endif
}
/* TODO: The "direct" functions are not used. Remove them? */
/**********************************************************************
Releases a shared mode lock when we know there are no waiters and none
else will access the lock during the time this function is executed. */
...
...
@@ -489,17 +492,16 @@ rw_lock_x_unlock_func(
#endif
)
{
os_thread_id_t local_writer_thread
;
uint local_pass
;
ut_ad((lock->lock_word % X_LOCK_DECR) == 0);
/*
Must reset
writer_thread
while we still have the lock.
Must reset
pass
while we still have the lock.
If we are not the last unlocker, we correct it later in the function,
which is harmless since we still hold the lock.
TODO: are there any risks of a thread id == -1 on any platform?
*/
local_
writer_thread = lock->writer_thread
;
lock->writer_thread = -
1;
local_
pass = lock->pass
;
lock->pass =
1;
#ifdef UNIV_SYNC_DEBUG
rw_lock_remove_debug_info(lock, pass, RW_LOCK_EX);
...
...
@@ -516,8 +518,8 @@ rw_lock_x_unlock_func(
}
} else {
/* We still hold x-lock, so we correct
writer_thread
. */
lock->
writer_thread = local_writer_thread
;
/* We still hold x-lock, so we correct
pass
. */
lock->
pass = local_pass
;
}
ut_ad(rw_lock_validate(lock));
...
...
@@ -527,6 +529,7 @@ rw_lock_x_unlock_func(
#endif
}
/* TODO: The "direct" functions are not used. Remove them? */
/**********************************************************************
Releases an exclusive mode lock when we know there are no waiters, and
none else will access the lock durint the time this function is executed. */
...
...
@@ -544,7 +547,7 @@ rw_lock_x_unlock_direct(
#ifdef UNIV_SYNC_DEBUG
rw_lock_remove_debug_info(lock, 0, RW_LOCK_EX);
#endif
lock->pass = 1;
lock->lock_word += X_LOCK_DECR;
ut_ad(!rw_lock_get_waiters(lock));
...
...
storage/innobase/sync/sync0rw.c
View file @
fa5529ee
...
...
@@ -50,17 +50,16 @@ The other members of the lock obey the following rules to remain consistent:
pass: This is only set to 1 to prevent recursive x-locks. It must
be set as specified by x_lock caller after the lock_word
indicates that the thread holds the lock, but before that
thread resumes execution. It must
be reset to 0
during the
thread resumes execution. It must
also be set to 1
during the
final x_unlock, but before the lock_word status is updated.
When an x_lock or move_ownership call wishes to change
pass, it must first update the writer_thread appropriately.
writer_thread: Must be set to the writers thread_id after the lock_word
indicates that the thread holds the lock, but before that
thread resumes execution. It must be reset to -1 during the
final x_unlock, but before the lock_word status is updated.
This ensures that when the lock_word indicates that an x_lock
is held, the only legitimate values for writer_thread are -1
(x_lock function hasn't completed) or the writer's thread_id.
thread resumes execution. writer_thread may be invalid and
should not be read when pass == 1. A thread trying to become
writer never reads its own stale writer_thread, since it sets
pass during its previous unlock call.
waiters: May be set to 1 anytime, but to avoid unnecessary wake-up
signals, it should only be set to 1 when there are threads
waiting on event. Must be 1 when a writer starts waiting to
...
...
@@ -210,8 +209,8 @@ rw_lock_create_func(
lock
->
lock_word
=
X_LOCK_DECR
;
lock
->
waiters
=
0
;
lock
->
writer_thread
=
-
1
;
lock
->
pass
=
0
;
lock
->
pass
=
1
;
/* We do not have to initialize writer_thread until pass == 0 */
#ifdef UNIV_SYNC_DEBUG
UT_LIST_INIT
(
lock
->
debug_list
);
...
...
@@ -413,19 +412,8 @@ rw_lock_x_lock_move_ownership(
ut_ad
(
rw_lock_is_locked
(
lock
,
RW_LOCK_EX
));
#ifdef UNIV_SYNC_ATOMIC
os_thread_id_t
local_writer_thread
=
lock
->
writer_thread
;
os_thread_id_t
new_writer_thread
=
os_thread_get_curr_id
();
while
(
TRUE
)
{
if
((
int
)
local_writer_thread
!=
-
1
)
{
if
(
os_compare_and_swap
(
(
volatile
lint
*
)
&
(
lock
->
writer_thread
),
local_writer_thread
,
new_writer_thread
))
{
break
;
}
}
local_writer_thread
=
lock
->
writer_thread
;
}
lock
->
writer_thread
=
os_thread_get_curr_id
();
os_memory_barrier_store
();
lock
->
pass
=
0
;
#else
/* UNIV_SYNC_ATOMIC */
mutex_enter
(
&
(
lock
->
mutex
));
...
...
@@ -518,10 +506,9 @@ rw_lock_x_lock_low(
ulint
line
)
/* in: line where requested */
{
os_thread_id_t
curr_thread
=
os_thread_get_curr_id
();
ut_ad
(
curr_thread
!=
-
1
);
/* We use -1 as the unlocked value. */
if
(
rw_lock_lock_word_decr
(
lock
,
X_LOCK_DECR
))
{
ut_ad
(
lock
->
writer_thread
==
-
1
);
ut_ad
(
lock
->
pass
);
/* Decrement occurred: we are writer or next-writer. */
lock
->
writer_thread
=
curr_thread
;
...
...
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