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
afca52bb
Commit
afca52bb
authored
Nov 20, 2014
by
Sergei Golubchik
Browse files
Options
Browse Files
Download
Plain Diff
5.5 merge
parents
3495801e
6ea41f1e
Changes
12
Hide whitespace changes
Inline
Side-by-side
Showing
12 changed files
with
219 additions
and
76 deletions
+219
-76
storage/innobase/include/os0sync.h
storage/innobase/include/os0sync.h
+52
-23
storage/innobase/include/os0sync.ic
storage/innobase/include/os0sync.ic
+32
-0
storage/innobase/include/sync0sync.h
storage/innobase/include/sync0sync.h
+0
-2
storage/innobase/include/sync0sync.ic
storage/innobase/include/sync0sync.ic
+7
-11
storage/innobase/os/os0sync.cc
storage/innobase/os/os0sync.cc
+19
-0
storage/innobase/sync/sync0sync.cc
storage/innobase/sync/sync0sync.cc
+0
-2
storage/xtradb/include/os0sync.h
storage/xtradb/include/os0sync.h
+52
-23
storage/xtradb/include/os0sync.ic
storage/xtradb/include/os0sync.ic
+31
-0
storage/xtradb/include/sync0sync.h
storage/xtradb/include/sync0sync.h
+0
-2
storage/xtradb/include/sync0sync.ic
storage/xtradb/include/sync0sync.ic
+7
-11
storage/xtradb/os/os0sync.cc
storage/xtradb/os/os0sync.cc
+19
-0
storage/xtradb/sync/sync0sync.cc
storage/xtradb/sync/sync0sync.cc
+0
-2
No files found.
storage/innobase/include/os0sync.h
View file @
afca52bb
...
@@ -321,6 +321,15 @@ pfs_os_fast_mutex_unlock(
...
@@ -321,6 +321,15 @@ pfs_os_fast_mutex_unlock(
os_fast_mutex_free_func(&((os_fast_mutex_t*)(M))->mutex)
os_fast_mutex_free_func(&((os_fast_mutex_t*)(M))->mutex)
#endif
/* UNIV_PFS_MUTEX */
#endif
/* UNIV_PFS_MUTEX */
/**********************************************************//**
Acquires ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required.
@return 0 if success, != 0 if was reserved by another thread */
UNIV_INLINE
ulint
os_fast_mutex_trylock_full_barrier
(
/*==================*/
os_fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to acquire */
/**********************************************************//**
/**********************************************************//**
Releases ownership of a fast mutex. */
Releases ownership of a fast mutex. */
UNIV_INTERN
UNIV_INTERN
...
@@ -328,6 +337,14 @@ void
...
@@ -328,6 +337,14 @@ void
os_fast_mutex_unlock_func
(
os_fast_mutex_unlock_func
(
/*======================*/
/*======================*/
fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to release */
fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to release */
/**********************************************************//**
Releases ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required. */
UNIV_INTERN
void
os_fast_mutex_unlock_full_barrier
(
/*=================*/
os_fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to release */
/*********************************************************//**
/*********************************************************//**
Initializes an operating system fast mutex semaphore. */
Initializes an operating system fast mutex semaphore. */
UNIV_INTERN
UNIV_INTERN
...
@@ -432,14 +449,31 @@ amount to decrement. */
...
@@ -432,14 +449,31 @@ amount to decrement. */
/**********************************************************//**
/**********************************************************//**
Returns the old value of *ptr, atomically sets *ptr to new_val */
Returns the old value of *ptr, atomically sets *ptr to new_val */
# define os_atomic_test_and_set_byte(ptr, new_val) \
__sync_lock_test_and_set(ptr, (byte) new_val)
# define os_atomic_test_and_set_ulint(ptr, new_val) \
# define os_atomic_test_and_set_ulint(ptr, new_val) \
__sync_lock_test_and_set(ptr, new_val)
__sync_lock_test_and_set(ptr, new_val)
# define os_atomic_lock_release_byte(ptr) \
#ifdef __powerpc__
__sync_lock_release(ptr)
/*
os_atomic_test_and_set_byte_release() should imply a release barrier before
setting, and a full barrier after. But __sync_lock_test_and_set() is only
documented as an aquire barrier. So on PowerPC we need to add the full
barrier explicitly. */
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
do { __sync_lock_release(ptr); \
__sync_synchronize(); } while (0)
#else
/*
On x86, __sync_lock_test_and_set() happens to be full barrier, due to
LOCK prefix.
*/
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
__sync_lock_test_and_set(ptr, (byte) new_val)
#endif
/*
os_atomic_test_and_set_byte_acquire() is a full memory barrier on x86. But
in general, just an aquire barrier should be sufficient. */
# define os_atomic_test_and_set_byte_acquire(ptr, new_val) \
__sync_lock_test_and_set(ptr, (byte) new_val)
#elif defined(HAVE_IB_SOLARIS_ATOMICS)
#elif defined(HAVE_IB_SOLARIS_ATOMICS)
...
@@ -517,14 +551,14 @@ amount to decrement. */
...
@@ -517,14 +551,14 @@ amount to decrement. */
/**********************************************************//**
/**********************************************************//**
Returns the old value of *ptr, atomically sets *ptr to new_val */
Returns the old value of *ptr, atomically sets *ptr to new_val */
# define os_atomic_test_and_set_byte(ptr, new_val) \
atomic_swap_uchar(ptr, new_val)
# define os_atomic_test_and_set_ulint(ptr, new_val) \
# define os_atomic_test_and_set_ulint(ptr, new_val) \
atomic_swap_ulong(ptr, new_val)
atomic_swap_ulong(ptr, new_val)
# define os_atomic_lock_release_byte(ptr) \
# define os_atomic_test_and_set_byte_acquire(ptr, new_val) \
(void) atomic_swap_uchar(ptr, 0)
atomic_swap_uchar(ptr, new_val)
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
atomic_swap_uchar(ptr, new_val)
#elif defined(HAVE_WINDOWS_ATOMICS)
#elif defined(HAVE_WINDOWS_ATOMICS)
...
@@ -644,7 +678,9 @@ Returns the old value of *ptr, atomically sets *ptr to new_val.
...
@@ -644,7 +678,9 @@ Returns the old value of *ptr, atomically sets *ptr to new_val.
InterlockedExchange() operates on LONG, and the LONG will be
InterlockedExchange() operates on LONG, and the LONG will be
clobbered */
clobbered */
# define os_atomic_test_and_set_byte(ptr, new_val) \
# define os_atomic_test_and_set_byte_acquire(ptr, new_val) \
((byte) InterlockedExchange(ptr, new_val))
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
((byte) InterlockedExchange(ptr, new_val))
((byte) InterlockedExchange(ptr, new_val))
# define os_atomic_test_and_set_ulong(ptr, new_val) \
# define os_atomic_test_and_set_ulong(ptr, new_val) \
...
@@ -713,11 +749,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -713,11 +749,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
# define HAVE_MEMORY_BARRIER
# define HAVE_MEMORY_BARRIER
# define os_rmb __atomic_thread_fence(__ATOMIC_ACQUIRE)
# define os_rmb __atomic_thread_fence(__ATOMIC_ACQUIRE)
# define os_wmb __atomic_thread_fence(__ATOMIC_RELEASE)
# define os_wmb __atomic_thread_fence(__ATOMIC_RELEASE)
#ifdef __powerpc__
# define os_mb __atomic_thread_fence(__ATOMIC_SEQ_CST)
# define os_isync __asm __volatile ("isync":::"memory")
#else
#define os_isync do { } while(0)
#endif
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"GCC builtin __atomic_thread_fence() is used for memory barrier"
"GCC builtin __atomic_thread_fence() is used for memory barrier"
...
@@ -726,7 +758,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -726,7 +758,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
# define HAVE_MEMORY_BARRIER
# define HAVE_MEMORY_BARRIER
# define os_rmb __sync_synchronize()
# define os_rmb __sync_synchronize()
# define os_wmb __sync_synchronize()
# define os_wmb __sync_synchronize()
# define os_
isync
__sync_synchronize()
# define os_
mb
__sync_synchronize()
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"GCC builtin __sync_synchronize() is used for memory barrier"
"GCC builtin __sync_synchronize() is used for memory barrier"
...
@@ -735,7 +767,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -735,7 +767,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
# include <mbarrier.h>
# include <mbarrier.h>
# define os_rmb __machine_r_barrier()
# define os_rmb __machine_r_barrier()
# define os_wmb __machine_w_barrier()
# define os_wmb __machine_w_barrier()
# define os_
isync os_rmb; os_wmb
# define os_
mb __machine_rw_barrier()
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"Solaris memory ordering functions are used for memory barrier"
"Solaris memory ordering functions are used for memory barrier"
...
@@ -744,17 +776,14 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -744,17 +776,14 @@ architecture. Disable memory barrier for Intel architecture for now. */
# include <intrin.h>
# include <intrin.h>
# define os_rmb _mm_lfence()
# define os_rmb _mm_lfence()
# define os_wmb _mm_sfence()
# define os_wmb _mm_sfence()
# define os_
isync os_rmb; os_wmb
# define os_
mb _mm_mfence()
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"_mm_lfence() and _mm_sfence() are used for memory barrier"
"_mm_lfence() and _mm_sfence() are used for memory barrier"
# define os_atomic_lock_release_byte(ptr) \
(void) InterlockedExchange(ptr, 0)
#else
#else
# define os_rmb do { } while(0)
# define os_rmb do { } while(0)
# define os_wmb do { } while(0)
# define os_wmb do { } while(0)
# define os_
isync
do { } while(0)
# define os_
mb
do { } while(0)
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"Memory barrier is not used"
"Memory barrier is not used"
#endif
#endif
...
...
storage/innobase/include/os0sync.ic
View file @
afca52bb
...
@@ -232,3 +232,35 @@ win_cmp_and_xchg_dword(
...
@@ -232,3 +232,35 @@ win_cmp_and_xchg_dword(
#endif /* HAVE_WINDOWS_ATOMICS */
#endif /* HAVE_WINDOWS_ATOMICS */
/**********************************************************//**
Acquires ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required.
@return 0 if success, != 0 if was reserved by another thread */
UNIV_INLINE
ulint
os_fast_mutex_trylock_full_barrier(
/*==================*/
os_fast_mutex_t* fast_mutex) /*!< in: mutex to acquire */
{
#ifdef __WIN__
if (TryEnterCriticalSection(&fast_mutex->mutex)) {
return(0);
} else {
return(1);
}
#else
/* NOTE that the MySQL my_pthread.h redefines pthread_mutex_trylock
so that it returns 0 on success. In the operating system
libraries, HP-UX-10.20 follows the old Posix 1003.4a Draft 4 and
returns 1 on success (but MySQL remaps that to 0), while Linux,
FreeBSD, Solaris, AIX, Tru64 Unix, HP-UX-11.0 return 0 on success. */
#ifdef __powerpc__
os_mb;
#endif
return((ulint) pthread_mutex_trylock(&fast_mutex->mutex));
#endif
}
storage/innobase/include/sync0sync.h
View file @
afca52bb
...
@@ -49,8 +49,6 @@ extern "C" my_bool timed_mutexes;
...
@@ -49,8 +49,6 @@ extern "C" my_bool timed_mutexes;
#ifdef HAVE_WINDOWS_ATOMICS
#ifdef HAVE_WINDOWS_ATOMICS
typedef
LONG
lock_word_t
;
/*!< On Windows, InterlockedExchange operates
typedef
LONG
lock_word_t
;
/*!< On Windows, InterlockedExchange operates
on LONG variable */
on LONG variable */
#elif defined(HAVE_ATOMIC_BUILTINS) && !defined(HAVE_ATOMIC_BUILTINS_BYTE)
typedef
ulint
lock_word_t
;
#else
#else
typedef
byte
lock_word_t
;
typedef
byte
lock_word_t
;
#endif
#endif
...
...
storage/innobase/include/sync0sync.ic
View file @
afca52bb
...
@@ -80,15 +80,11 @@ ib_mutex_test_and_set(
...
@@ -80,15 +80,11 @@ ib_mutex_test_and_set(
ib_mutex_t* mutex) /*!< in: mutex */
ib_mutex_t* mutex) /*!< in: mutex */
{
{
#if defined(HAVE_ATOMIC_BUILTINS)
#if defined(HAVE_ATOMIC_BUILTINS)
# if defined(HAVE_ATOMIC_BUILTINS_BYTE)
return(os_atomic_test_and_set_byte_acquire(&mutex->lock_word, 1));
return(os_atomic_test_and_set_byte(&mutex->lock_word, 1));
# else
return(os_atomic_test_and_set_ulint(&mutex->lock_word, 1));
# endif
#else
#else
ibool ret;
ibool ret;
ret = os_fast_mutex_trylock(&(mutex->os_fast_mutex));
ret = os_fast_mutex_trylock
_full_barrier
(&(mutex->os_fast_mutex));
if (ret == 0) {
if (ret == 0) {
/* We check that os_fast_mutex_trylock does not leak
/* We check that os_fast_mutex_trylock does not leak
...
@@ -96,7 +92,6 @@ ib_mutex_test_and_set(
...
@@ -96,7 +92,6 @@ ib_mutex_test_and_set(
ut_a(mutex->lock_word == 0);
ut_a(mutex->lock_word == 0);
mutex->lock_word = 1;
mutex->lock_word = 1;
os_wmb;
}
}
return((byte) ret);
return((byte) ret);
...
@@ -113,11 +108,14 @@ mutex_reset_lock_word(
...
@@ -113,11 +108,14 @@ mutex_reset_lock_word(
ib_mutex_t* mutex) /*!< in: mutex */
ib_mutex_t* mutex) /*!< in: mutex */
{
{
#if defined(HAVE_ATOMIC_BUILTINS)
#if defined(HAVE_ATOMIC_BUILTINS)
os_atomic_lock_release_byte(&mutex->lock_word);
/* In theory __sync_lock_release should be used to release the lock.
Unfortunately, it does not work properly alone. The workaround is
that more conservative __sync_lock_test_and_set is used instead. */
os_atomic_test_and_set_byte_release(&mutex->lock_word, 0);
#else
#else
mutex->lock_word = 0;
mutex->lock_word = 0;
os_fast_mutex_unlock(&(mutex->os_fast_mutex));
os_fast_mutex_unlock
_full_barrier
(&(mutex->os_fast_mutex));
#endif
#endif
}
}
...
@@ -149,7 +147,6 @@ mutex_get_waiters(
...
@@ -149,7 +147,6 @@ mutex_get_waiters(
ptr = &(mutex->waiters);
ptr = &(mutex->waiters);
os_rmb;
return(*ptr); /* Here we assume that the read of a single
return(*ptr); /* Here we assume that the read of a single
word from memory is atomic */
word from memory is atomic */
}
}
...
@@ -184,7 +181,6 @@ mutex_exit_func(
...
@@ -184,7 +181,6 @@ mutex_exit_func(
to wake up possible hanging threads if
to wake up possible hanging threads if
they are missed in mutex_signal_object. */
they are missed in mutex_signal_object. */
os_isync;
if (mutex_get_waiters(mutex) != 0) {
if (mutex_get_waiters(mutex) != 0) {
mutex_signal_object(mutex);
mutex_signal_object(mutex);
...
...
storage/innobase/os/os0sync.cc
View file @
afca52bb
...
@@ -889,6 +889,25 @@ os_fast_mutex_unlock_func(
...
@@ -889,6 +889,25 @@ os_fast_mutex_unlock_func(
#endif
#endif
}
}
/**********************************************************//**
Releases ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required. */
UNIV_INTERN
void
os_fast_mutex_unlock_full_barrier
(
/*=================*/
os_fast_mutex_t
*
fast_mutex
)
/*!< in: mutex to release */
{
#ifdef __WIN__
LeaveCriticalSection
(
&
fast_mutex
->
mutex
);
#else
pthread_mutex_unlock
(
&
fast_mutex
->
mutex
);
#ifdef __powerpc__
os_mb
;
#endif
#endif
}
/**********************************************************//**
/**********************************************************//**
Frees a mutex object. */
Frees a mutex object. */
UNIV_INTERN
UNIV_INTERN
...
...
storage/innobase/sync/sync0sync.cc
View file @
afca52bb
...
@@ -457,8 +457,6 @@ mutex_set_waiters(
...
@@ -457,8 +457,6 @@ mutex_set_waiters(
ptr
=
&
(
mutex
->
waiters
);
ptr
=
&
(
mutex
->
waiters
);
os_wmb
;
*
ptr
=
n
;
/* Here we assume that the write of a single
*
ptr
=
n
;
/* Here we assume that the write of a single
word in memory is atomic */
word in memory is atomic */
}
}
...
...
storage/xtradb/include/os0sync.h
View file @
afca52bb
...
@@ -321,6 +321,15 @@ pfs_os_fast_mutex_unlock(
...
@@ -321,6 +321,15 @@ pfs_os_fast_mutex_unlock(
os_fast_mutex_free_func(&((os_fast_mutex_t*)(M))->mutex)
os_fast_mutex_free_func(&((os_fast_mutex_t*)(M))->mutex)
#endif
/* UNIV_PFS_MUTEX */
#endif
/* UNIV_PFS_MUTEX */
/**********************************************************//**
Acquires ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required.
@return 0 if success, != 0 if was reserved by another thread */
UNIV_INLINE
ulint
os_fast_mutex_trylock_full_barrier
(
/*==================*/
os_fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to acquire */
/**********************************************************//**
/**********************************************************//**
Releases ownership of a fast mutex. */
Releases ownership of a fast mutex. */
UNIV_INTERN
UNIV_INTERN
...
@@ -328,6 +337,14 @@ void
...
@@ -328,6 +337,14 @@ void
os_fast_mutex_unlock_func
(
os_fast_mutex_unlock_func
(
/*======================*/
/*======================*/
fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to release */
fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to release */
/**********************************************************//**
Releases ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required. */
UNIV_INTERN
void
os_fast_mutex_unlock_full_barrier
(
/*=================*/
os_fast_mutex_t
*
fast_mutex
);
/*!< in: mutex to release */
/*********************************************************//**
/*********************************************************//**
Initializes an operating system fast mutex semaphore. */
Initializes an operating system fast mutex semaphore. */
UNIV_INTERN
UNIV_INTERN
...
@@ -432,14 +449,31 @@ amount to decrement. */
...
@@ -432,14 +449,31 @@ amount to decrement. */
/**********************************************************//**
/**********************************************************//**
Returns the old value of *ptr, atomically sets *ptr to new_val */
Returns the old value of *ptr, atomically sets *ptr to new_val */
# define os_atomic_test_and_set_byte(ptr, new_val) \
__sync_lock_test_and_set(ptr, (byte) new_val)
# define os_atomic_test_and_set_ulint(ptr, new_val) \
# define os_atomic_test_and_set_ulint(ptr, new_val) \
__sync_lock_test_and_set(ptr, new_val)
__sync_lock_test_and_set(ptr, new_val)
# define os_atomic_lock_release_byte(ptr) \
#ifdef __powerpc__
__sync_lock_release(ptr)
/*
os_atomic_test_and_set_byte_release() should imply a release barrier before
setting, and a full barrier after. But __sync_lock_test_and_set() is only
documented as an aquire barrier. So on PowerPC we need to add the full
barrier explicitly. */
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
do { __sync_lock_release(ptr); \
__sync_synchronize(); } while (0)
#else
/*
On x86, __sync_lock_test_and_set() happens to be full barrier, due to
LOCK prefix.
*/
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
__sync_lock_test_and_set(ptr, (byte) new_val)
#endif
/*
os_atomic_test_and_set_byte_acquire() is a full memory barrier on x86. But
in general, just an aquire barrier should be sufficient. */
# define os_atomic_test_and_set_byte_acquire(ptr, new_val) \
__sync_lock_test_and_set(ptr, (byte) new_val)
#elif defined(HAVE_IB_SOLARIS_ATOMICS)
#elif defined(HAVE_IB_SOLARIS_ATOMICS)
...
@@ -517,14 +551,14 @@ amount to decrement. */
...
@@ -517,14 +551,14 @@ amount to decrement. */
/**********************************************************//**
/**********************************************************//**
Returns the old value of *ptr, atomically sets *ptr to new_val */
Returns the old value of *ptr, atomically sets *ptr to new_val */
# define os_atomic_test_and_set_byte(ptr, new_val) \
atomic_swap_uchar(ptr, new_val)
# define os_atomic_test_and_set_ulint(ptr, new_val) \
# define os_atomic_test_and_set_ulint(ptr, new_val) \
atomic_swap_ulong(ptr, new_val)
atomic_swap_ulong(ptr, new_val)
# define os_atomic_lock_release_byte(ptr) \
# define os_atomic_test_and_set_byte_acquire(ptr, new_val) \
(void) atomic_swap_uchar(ptr, 0)
atomic_swap_uchar(ptr, new_val)
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
atomic_swap_uchar(ptr, new_val)
#elif defined(HAVE_WINDOWS_ATOMICS)
#elif defined(HAVE_WINDOWS_ATOMICS)
...
@@ -644,7 +678,9 @@ Returns the old value of *ptr, atomically sets *ptr to new_val.
...
@@ -644,7 +678,9 @@ Returns the old value of *ptr, atomically sets *ptr to new_val.
InterlockedExchange() operates on LONG, and the LONG will be
InterlockedExchange() operates on LONG, and the LONG will be
clobbered */
clobbered */
# define os_atomic_test_and_set_byte(ptr, new_val) \
# define os_atomic_test_and_set_byte_acquire(ptr, new_val) \
((byte) InterlockedExchange(ptr, new_val))
# define os_atomic_test_and_set_byte_release(ptr, new_val) \
((byte) InterlockedExchange(ptr, new_val))
((byte) InterlockedExchange(ptr, new_val))
# define os_atomic_test_and_set_ulong(ptr, new_val) \
# define os_atomic_test_and_set_ulong(ptr, new_val) \
...
@@ -713,11 +749,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -713,11 +749,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
# define HAVE_MEMORY_BARRIER
# define HAVE_MEMORY_BARRIER
# define os_rmb __atomic_thread_fence(__ATOMIC_ACQUIRE)
# define os_rmb __atomic_thread_fence(__ATOMIC_ACQUIRE)
# define os_wmb __atomic_thread_fence(__ATOMIC_RELEASE)
# define os_wmb __atomic_thread_fence(__ATOMIC_RELEASE)
#ifdef __powerpc__
# define os_mb __atomic_thread_fence(__ATOMIC_SEQ_CST)
# define os_isync __asm __volatile ("isync":::"memory")
#else
#define os_isync do { } while(0)
#endif
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"GCC builtin __atomic_thread_fence() is used for memory barrier"
"GCC builtin __atomic_thread_fence() is used for memory barrier"
...
@@ -726,7 +758,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -726,7 +758,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
# define HAVE_MEMORY_BARRIER
# define HAVE_MEMORY_BARRIER
# define os_rmb __sync_synchronize()
# define os_rmb __sync_synchronize()
# define os_wmb __sync_synchronize()
# define os_wmb __sync_synchronize()
# define os_
isync
__sync_synchronize()
# define os_
mb
__sync_synchronize()
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"GCC builtin __sync_synchronize() is used for memory barrier"
"GCC builtin __sync_synchronize() is used for memory barrier"
...
@@ -735,7 +767,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -735,7 +767,7 @@ architecture. Disable memory barrier for Intel architecture for now. */
# include <mbarrier.h>
# include <mbarrier.h>
# define os_rmb __machine_r_barrier()
# define os_rmb __machine_r_barrier()
# define os_wmb __machine_w_barrier()
# define os_wmb __machine_w_barrier()
# define os_
isync os_rmb; os_wmb
# define os_
mb __machine_rw_barrier()
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"Solaris memory ordering functions are used for memory barrier"
"Solaris memory ordering functions are used for memory barrier"
...
@@ -744,17 +776,14 @@ architecture. Disable memory barrier for Intel architecture for now. */
...
@@ -744,17 +776,14 @@ architecture. Disable memory barrier for Intel architecture for now. */
# include <intrin.h>
# include <intrin.h>
# define os_rmb _mm_lfence()
# define os_rmb _mm_lfence()
# define os_wmb _mm_sfence()
# define os_wmb _mm_sfence()
# define os_
isync os_rmb; os_wmb
# define os_
mb _mm_mfence()
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"_mm_lfence() and _mm_sfence() are used for memory barrier"
"_mm_lfence() and _mm_sfence() are used for memory barrier"
# define os_atomic_lock_release_byte(ptr) \
(void) InterlockedExchange(ptr, 0)
#else
#else
# define os_rmb do { } while(0)
# define os_rmb do { } while(0)
# define os_wmb do { } while(0)
# define os_wmb do { } while(0)
# define os_
isync
do { } while(0)
# define os_
mb
do { } while(0)
# define IB_MEMORY_BARRIER_STARTUP_MSG \
# define IB_MEMORY_BARRIER_STARTUP_MSG \
"Memory barrier is not used"
"Memory barrier is not used"
#endif
#endif
...
...
storage/xtradb/include/os0sync.ic
View file @
afca52bb
...
@@ -232,3 +232,34 @@ win_cmp_and_xchg_dword(
...
@@ -232,3 +232,34 @@ win_cmp_and_xchg_dword(
#endif /* HAVE_WINDOWS_ATOMICS */
#endif /* HAVE_WINDOWS_ATOMICS */
/**********************************************************//**
Acquires ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required.
@return 0 if success, != 0 if was reserved by another thread */
UNIV_INLINE
ulint
os_fast_mutex_trylock_full_barrier(
/*==================*/
os_fast_mutex_t* fast_mutex) /*!< in: mutex to acquire */
{
#ifdef __WIN__
if (TryEnterCriticalSection(&fast_mutex->mutex)) {
return(0);
} else {
return(1);
}
#else
/* NOTE that the MySQL my_pthread.h redefines pthread_mutex_trylock
so that it returns 0 on success. In the operating system
libraries, HP-UX-10.20 follows the old Posix 1003.4a Draft 4 and
returns 1 on success (but MySQL remaps that to 0), while Linux,
FreeBSD, Solaris, AIX, Tru64 Unix, HP-UX-11.0 return 0 on success. */
#ifdef __powerpc__
os_mb;
#endif
return((ulint) pthread_mutex_trylock(&fast_mutex->mutex));
#endif
}
storage/xtradb/include/sync0sync.h
View file @
afca52bb
...
@@ -50,8 +50,6 @@ extern "C" my_bool timed_mutexes;
...
@@ -50,8 +50,6 @@ extern "C" my_bool timed_mutexes;
#ifdef _WIN32
#ifdef _WIN32
typedef
LONG
lock_word_t
;
/*!< On Windows, InterlockedExchange operates
typedef
LONG
lock_word_t
;
/*!< On Windows, InterlockedExchange operates
on LONG variable */
on LONG variable */
#elif defined(HAVE_ATOMIC_BUILTINS) && !defined(HAVE_ATOMIC_BUILTINS_BYTE)
typedef
ulint
lock_word_t
;
#else
#else
typedef
byte
lock_word_t
;
typedef
byte
lock_word_t
;
#endif
#endif
...
...
storage/xtradb/include/sync0sync.ic
View file @
afca52bb
...
@@ -83,15 +83,11 @@ ib_mutex_test_and_set(
...
@@ -83,15 +83,11 @@ ib_mutex_test_and_set(
ib_mutex_t* mutex) /*!< in: mutex */
ib_mutex_t* mutex) /*!< in: mutex */
{
{
#if defined(HAVE_ATOMIC_BUILTINS)
#if defined(HAVE_ATOMIC_BUILTINS)
# if defined(HAVE_ATOMIC_BUILTINS_BYTE)
return(os_atomic_test_and_set_byte_acquire(&mutex->lock_word, 1));
return(os_atomic_test_and_set_byte(&mutex->lock_word, 1));
# else
return(os_atomic_test_and_set_ulint(&mutex->lock_word, 1));
# endif
#else
#else
ibool ret;
ibool ret;
ret = os_fast_mutex_trylock(&(mutex->os_fast_mutex));
ret = os_fast_mutex_trylock
_full_barrier
(&(mutex->os_fast_mutex));
if (ret == 0) {
if (ret == 0) {
/* We check that os_fast_mutex_trylock does not leak
/* We check that os_fast_mutex_trylock does not leak
...
@@ -99,7 +95,6 @@ ib_mutex_test_and_set(
...
@@ -99,7 +95,6 @@ ib_mutex_test_and_set(
ut_a(mutex->lock_word == 0);
ut_a(mutex->lock_word == 0);
mutex->lock_word = 1;
mutex->lock_word = 1;
os_wmb;
}
}
return((byte) ret);
return((byte) ret);
...
@@ -116,11 +111,14 @@ mutex_reset_lock_word(
...
@@ -116,11 +111,14 @@ mutex_reset_lock_word(
ib_mutex_t* mutex) /*!< in: mutex */
ib_mutex_t* mutex) /*!< in: mutex */
{
{
#if defined(HAVE_ATOMIC_BUILTINS)
#if defined(HAVE_ATOMIC_BUILTINS)
os_atomic_lock_release_byte(&mutex->lock_word);
/* In theory __sync_lock_release should be used to release the lock.
Unfortunately, it does not work properly alone. The workaround is
that more conservative __sync_lock_test_and_set is used instead. */
os_atomic_test_and_set_byte_release(&mutex->lock_word, 0);
#else
#else
mutex->lock_word = 0;
mutex->lock_word = 0;
os_fast_mutex_unlock(&(mutex->os_fast_mutex));
os_fast_mutex_unlock
_full_barrier
(&(mutex->os_fast_mutex));
#endif
#endif
}
}
...
@@ -152,7 +150,6 @@ mutex_get_waiters(
...
@@ -152,7 +150,6 @@ mutex_get_waiters(
ptr = &(mutex->waiters);
ptr = &(mutex->waiters);
os_rmb;
return(*ptr); /* Here we assume that the read of a single
return(*ptr); /* Here we assume that the read of a single
word from memory is atomic */
word from memory is atomic */
}
}
...
@@ -187,7 +184,6 @@ mutex_exit_func(
...
@@ -187,7 +184,6 @@ mutex_exit_func(
to wake up possible hanging threads if
to wake up possible hanging threads if
they are missed in mutex_signal_object. */
they are missed in mutex_signal_object. */
os_isync;
if (mutex_get_waiters(mutex) != 0) {
if (mutex_get_waiters(mutex) != 0) {
mutex_signal_object(mutex);
mutex_signal_object(mutex);
...
...
storage/xtradb/os/os0sync.cc
View file @
afca52bb
...
@@ -889,6 +889,25 @@ os_fast_mutex_unlock_func(
...
@@ -889,6 +889,25 @@ os_fast_mutex_unlock_func(
#endif
#endif
}
}
/**********************************************************//**
Releases ownership of a fast mutex. Implies a full memory barrier even on
platforms such as PowerPC where this is not normally required. */
UNIV_INTERN
void
os_fast_mutex_unlock_full_barrier
(
/*=================*/
os_fast_mutex_t
*
fast_mutex
)
/*!< in: mutex to release */
{
#ifdef __WIN__
LeaveCriticalSection
(
&
fast_mutex
->
mutex
);
#else
pthread_mutex_unlock
(
&
fast_mutex
->
mutex
);
#ifdef __powerpc__
os_mb
;
#endif
#endif
}
/**********************************************************//**
/**********************************************************//**
Frees a mutex object. */
Frees a mutex object. */
UNIV_INTERN
UNIV_INTERN
...
...
storage/xtradb/sync/sync0sync.cc
View file @
afca52bb
...
@@ -536,8 +536,6 @@ mutex_set_waiters(
...
@@ -536,8 +536,6 @@ mutex_set_waiters(
ptr
=
&
(
mutex
->
waiters
);
ptr
=
&
(
mutex
->
waiters
);
os_wmb
;
*
ptr
=
n
;
/* Here we assume that the write of a single
*
ptr
=
n
;
/* Here we assume that the write of a single
word in memory is atomic */
word in memory is atomic */
}
}
...
...
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