|  |  | 
|  | /* | 
|  | * Copyright (C) Igor Sysoev | 
|  | * Copyright (C) Nginx, Inc. | 
|  | */ | 
|  |  | 
|  |  | 
|  | #include <ngx_config.h> | 
|  | #include <ngx_core.h> | 
|  |  | 
|  |  | 
|  | static ngx_uint_t   nthreads; | 
|  | static ngx_uint_t   max_threads; | 
|  |  | 
|  |  | 
|  | static pthread_attr_t  thr_attr; | 
|  |  | 
|  |  | 
|  | ngx_err_t | 
|  | ngx_create_thread(ngx_tid_t *tid, ngx_thread_value_t (*func)(void *arg), | 
|  | void *arg, ngx_log_t *log) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | if (nthreads >= max_threads) { | 
|  | ngx_log_error(NGX_LOG_CRIT, log, 0, | 
|  | "no more than %ui threads can be created", max_threads); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | err = pthread_create(tid, &thr_attr, func, arg); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, log, err, "pthread_create() failed"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_CORE, log, 0, | 
|  | "thread is created: " NGX_TID_T_FMT, *tid); | 
|  |  | 
|  | nthreads++; | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_init_threads(int n, size_t size, ngx_cycle_t *cycle) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | max_threads = n; | 
|  |  | 
|  | err = pthread_attr_init(&thr_attr); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, err, | 
|  | "pthread_attr_init() failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | err = pthread_attr_setstacksize(&thr_attr, size); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, err, | 
|  | "pthread_attr_setstacksize() failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | ngx_threaded = 1; | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_mutex_t * | 
|  | ngx_mutex_init(ngx_log_t *log, ngx_uint_t flags) | 
|  | { | 
|  | int           err; | 
|  | ngx_mutex_t  *m; | 
|  |  | 
|  | m = ngx_alloc(sizeof(ngx_mutex_t), log); | 
|  | if (m == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | m->log = log; | 
|  |  | 
|  | err = pthread_mutex_init(&m->mutex, NULL); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, m->log, err, | 
|  | "pthread_mutex_init() failed"); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | return m; | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_mutex_destroy(ngx_mutex_t *m) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | err = pthread_mutex_destroy(&m->mutex); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, m->log, err, | 
|  | "pthread_mutex_destroy(%p) failed", m); | 
|  | } | 
|  |  | 
|  | ngx_free(m); | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_mutex_lock(ngx_mutex_t *m) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | if (!ngx_threaded) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "lock mutex %p", m); | 
|  |  | 
|  | err = pthread_mutex_lock(&m->mutex); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, m->log, err, | 
|  | "pthread_mutex_lock(%p) failed", m); | 
|  | ngx_abort(); | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "mutex %p is locked", m); | 
|  |  | 
|  | return; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_mutex_trylock(ngx_mutex_t *m) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | if (!ngx_threaded) { | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "try lock mutex %p", m); | 
|  |  | 
|  | err = pthread_mutex_trylock(&m->mutex); | 
|  |  | 
|  | if (err == NGX_EBUSY) { | 
|  | return NGX_AGAIN; | 
|  | } | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, m->log, err, | 
|  | "pthread_mutex_trylock(%p) failed", m); | 
|  | ngx_abort(); | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "mutex %p is locked", m); | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_mutex_unlock(ngx_mutex_t *m) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | if (!ngx_threaded) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "unlock mutex %p", m); | 
|  |  | 
|  | err = pthread_mutex_unlock(&m->mutex); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, m->log, err, | 
|  | "pthread_mutex_unlock(%p) failed", m); | 
|  | ngx_abort(); | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "mutex %p is unlocked", m); | 
|  |  | 
|  | return; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_cond_t * | 
|  | ngx_cond_init(ngx_log_t *log) | 
|  | { | 
|  | int          err; | 
|  | ngx_cond_t  *cv; | 
|  |  | 
|  | cv = ngx_alloc(sizeof(ngx_cond_t), log); | 
|  | if (cv == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | cv->log = log; | 
|  |  | 
|  | err = pthread_cond_init(&cv->cond, NULL); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cv->log, err, | 
|  | "pthread_cond_init() failed"); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | return cv; | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_cond_destroy(ngx_cond_t *cv) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | err = pthread_cond_destroy(&cv->cond); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cv->log, err, | 
|  | "pthread_cond_destroy(%p) failed", cv); | 
|  | } | 
|  |  | 
|  | ngx_free(cv); | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_cond_wait(ngx_cond_t *cv, ngx_mutex_t *m) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_CORE, cv->log, 0, "cv %p wait", cv); | 
|  |  | 
|  | err = pthread_cond_wait(&cv->cond, &m->mutex); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cv->log, err, | 
|  | "pthread_cond_wait(%p) failed", cv); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_CORE, cv->log, 0, "cv %p is waked up", cv); | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, "mutex %p is locked", m); | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_cond_signal(ngx_cond_t *cv) | 
|  | { | 
|  | int  err; | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_CORE, cv->log, 0, "cv %p to signal", cv); | 
|  |  | 
|  | err = pthread_cond_signal(&cv->cond); | 
|  |  | 
|  | if (err != 0) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cv->log, err, | 
|  | "pthread_cond_signal(%p) failed", cv); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_CORE, cv->log, 0, "cv %p is signaled", cv); | 
|  |  | 
|  | return NGX_OK; | 
|  | } |