|  |  | 
|  | /* | 
|  | * Copyright (C) Igor Sysoev | 
|  | * Copyright (C) Nginx, Inc. | 
|  | */ | 
|  |  | 
|  | #include <ngx_config.h> | 
|  | #include <ngx_core.h> | 
|  |  | 
|  |  | 
|  | /* | 
|  | * All modern pthread mutex implementations try to acquire a lock | 
|  | * atomically in userland before going to sleep in kernel.  Some | 
|  | * spins 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 overridden 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(). | 
|  | *   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. | 
|  | */ | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_thread_mutex_create(ngx_thread_mutex_t *mtx, ngx_log_t *log) | 
|  | { | 
|  | ngx_err_t            err; | 
|  | pthread_mutexattr_t  attr; | 
|  |  | 
|  | err = pthread_mutexattr_init(&attr); | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, err, | 
|  | "pthread_mutexattr_init() failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | err = pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK); | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, err, | 
|  | "pthread_mutexattr_settype" | 
|  | "(PTHREAD_MUTEX_ERRORCHECK) failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | err = pthread_mutex_init(mtx, &attr); | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, err, | 
|  | "pthread_mutex_init() failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | err = pthread_mutexattr_destroy(&attr); | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, log, err, | 
|  | "pthread_mutexattr_destroy() failed"); | 
|  | } | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_thread_mutex_destroy(ngx_thread_mutex_t *mtx, ngx_log_t *log) | 
|  | { | 
|  | ngx_err_t  err; | 
|  |  | 
|  | err = pthread_mutex_destroy(mtx); | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, log, err, | 
|  | "pthread_mutex_destroy() failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_thread_mutex_lock(ngx_thread_mutex_t *mtx, ngx_log_t *log) | 
|  | { | 
|  | ngx_err_t  err; | 
|  |  | 
|  | err = pthread_mutex_lock(mtx); | 
|  | if (err == 0) { | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | ngx_log_error(NGX_LOG_ALERT, log, err, "pthread_mutex_lock() failed"); | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_thread_mutex_unlock(ngx_thread_mutex_t *mtx, ngx_log_t *log) | 
|  | { | 
|  | ngx_err_t  err; | 
|  |  | 
|  | err = pthread_mutex_unlock(mtx); | 
|  |  | 
|  | #if 0 | 
|  | ngx_time_update(); | 
|  | #endif | 
|  |  | 
|  | if (err == 0) { | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | ngx_log_error(NGX_LOG_ALERT, log, err, "pthread_mutex_unlock() failed"); | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } |