diff options
Diffstat (limited to '')
-rw-r--r-- | src/nxt_thread_mutex.c | 192 |
1 files changed, 192 insertions, 0 deletions
diff --git a/src/nxt_thread_mutex.c b/src/nxt_thread_mutex.c new file mode 100644 index 00000000..10c08c2a --- /dev/null +++ b/src/nxt_thread_mutex.c @@ -0,0 +1,192 @@ + +/* + * Copyright (C) Igor Sysoev + * Copyright (C) NGINX, Inc. + */ + +#include <nxt_main.h> + + +/* + * All modern pthread mutex implementations try to acquire a lock atomically + * in userland before going to sleep in kernel. Some spins on SMP systems + * before the sleeping. + * + * In Solaris since version 8 all mutex types spin before sleeping. + * The default spin count is 1000. It can be overridden using + * _THREAD_ADAPTIVE_SPIN=100 environment variable. + * + * In MacOSX all mutex types spin to acquire a lock protecting a mutex's + * internals. If the mutex is busy, thread calls Mach semaphore_wait(). + * + * + * PTHREAD_MUTEX_NORMAL lacks deadlock detection and is the fastest + * mutex type. + * + * Linux: No spinning. The internal name PTHREAD_MUTEX_TIMED_NP + * remains from the times when pthread_mutex_timedlock() was + * non-standard extension. Alias name: PTHREAD_MUTEX_FAST_NP. + * FreeBSD: No spinning. + * + * + * PTHREAD_MUTEX_ERRORCHECK is usually as fast as PTHREAD_MUTEX_NORMAL + * yet has lightweight deadlock detection. + * + * Linux: No spinning. The internal name: PTHREAD_MUTEX_ERRORCHECK_NP. + * FreeBSD: No spinning. + * + * + * PTHREAD_MUTEX_RECURSIVE allows recursive locking. + * + * Linux: No spinning. The internal name: PTHREAD_MUTEX_RECURSIVE_NP. + * FreeBSD: No spinning. + * + * + * PTHREAD_MUTEX_ADAPTIVE_NP spins on SMP systems before sleeping. + * + * Linux: No deadlock detection. Dynamically changes a spin count + * for each mutex from 10 to 100 based on spin count taken + * previously. + * + * FreeBSD: Deadlock detection. The default spin count is 2000. + * It can be overriden using LIBPTHREAD_SPINLOOPS environment + * variable or by pthread_mutex_setspinloops_np(). If a lock + * is still busy, sched_yield() can be called on both UP and + * SMP systems. The default yield loop count is zero, but it + * can be set by LIBPTHREAD_YIELDLOOPS environment variable or + * by pthread_mutex_setyieldloops_np(). sched_yield() moves + * a thread to the end of CPU scheduler run queue and this is + * cheaper than removing the thread from the queue and sleeping. + * + * Solaris: No PTHREAD_MUTEX_ADAPTIVE_NP . + * MacOSX: No PTHREAD_MUTEX_ADAPTIVE_NP. + * + * + * PTHREAD_MUTEX_ELISION_NP is a Linux extension to elide locks using + * Intel Restricted Transactional Memory. It is the most suitable for + * rwlock pattern access because it allows simultaneous reads without lock. + * Supported since glibc 2.18. + * + * + * PTHREAD_MUTEX_DEFAULT is default mutex type. + * + * Linux: PTHREAD_MUTEX_NORMAL. + * FreeBSD: PTHREAD_MUTEX_ERRORCHECK. + * Solaris: PTHREAD_MUTEX_NORMAL. + * MacOSX: PTHREAD_MUTEX_NORMAL. + */ + + +nxt_int_t +nxt_thread_mutex_create(nxt_thread_mutex_t *mtx) +{ + nxt_err_t err; + pthread_mutexattr_t attr; + + err = pthread_mutexattr_init(&attr); + if (err != 0) { + nxt_thread_log_emerg("pthread_mutexattr_init() failed %E", err); + return NXT_ERROR; + } + + err = pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK); + if (err != 0) { + nxt_thread_log_emerg("pthread_mutexattr_settype" + "(PTHREAD_MUTEX_ERRORCHECK) failed %E", err); + return NXT_ERROR; + } + + err = pthread_mutex_init(mtx, &attr); + if (err != 0) { + nxt_thread_log_emerg("pthread_mutex_init() failed %E", err); + return NXT_ERROR; + } + + err = pthread_mutexattr_destroy(&attr); + if (err != 0) { + nxt_thread_log_alert("pthread_mutexattr_destroy() failed %E", err); + } + + nxt_thread_log_debug("pthread_mutex_init(%p)", mtx); + + return NXT_OK; +} + + +void +nxt_thread_mutex_destroy(nxt_thread_mutex_t *mtx) +{ + nxt_err_t err; + + err = pthread_mutex_destroy(mtx); + if (nxt_slow_path(err != 0)) { + nxt_thread_log_alert("pthread_mutex_destroy() failed %E", err); + } + + nxt_thread_log_debug("pthread_mutex_destroy(%p)", mtx); +} + + +nxt_int_t +nxt_thread_mutex_lock(nxt_thread_mutex_t *mtx) +{ + nxt_err_t err; + + nxt_thread_log_debug("pthread_mutex_lock(%p) enter", mtx); + + err = pthread_mutex_lock(mtx); + if (nxt_fast_path(err == 0)) { + return NXT_OK; + } + + nxt_thread_log_alert("pthread_mutex_lock() failed %E", err); + + return NXT_ERROR; +} + + +nxt_bool_t +nxt_thread_mutex_trylock(nxt_thread_mutex_t *mtx) +{ + nxt_err_t err; + + nxt_thread_debug(thr); + + nxt_log_debug(thr->log, "pthread_mutex_trylock(%p) enter", mtx); + + err = pthread_mutex_trylock(mtx); + if (nxt_fast_path(err == 0)) { + return 1; + } + + if (err == NXT_EBUSY) { + nxt_log_debug(thr->log, "pthread_mutex_trylock(%p) failed", mtx); + + } else { + nxt_thread_log_alert("pthread_mutex_trylock() failed %E", err); + } + + return 0; +} + + +nxt_int_t +nxt_thread_mutex_unlock(nxt_thread_mutex_t *mtx) +{ + nxt_err_t err; + nxt_thread_t *thr; + + err = pthread_mutex_unlock(mtx); + + thr = nxt_thread(); + nxt_thread_time_update(thr); + + if (nxt_fast_path(err == 0)) { + nxt_log_debug(thr->log, "pthread_mutex_unlock(%p) exit", mtx); + return NXT_OK; + } + + nxt_log_alert(thr->log, "pthread_mutex_unlock() failed %E", err); + + return NXT_ERROR; +} |