| |
| /* |
| * Copyright (C) Igor Sysoev |
| * Copyright (C) Nginx, Inc. |
| */ |
| |
| |
| #include <ngx_config.h> |
| #include <ngx_core.h> |
| |
| /* |
| * The threads implementation uses the rfork(RFPROC|RFTHREAD|RFMEM) syscall |
| * to create threads. All threads use the stacks of the same size mmap()ed |
| * below the main stack. Thus the current thread id is determined via |
| * the stack pointer value. |
| * |
| * The mutex implementation uses the ngx_atomic_cmp_set() operation |
| * to acquire a mutex and the SysV semaphore to wait on a mutex and to wake up |
| * the waiting threads. The light mutex does not use semaphore, so after |
| * spinning in the lock the thread calls sched_yield(). However the light |
| * mutexes are intended to be used with the "trylock" operation only. |
| * The SysV semop() is a cheap syscall, particularly if it has little sembuf's |
| * and does not use SEM_UNDO. |
| * |
| * The condition variable implementation uses the signal #64. |
| * The signal handler is SIG_IGN so the kill() is a cheap syscall. |
| * The thread waits a signal in kevent(). The use of the EVFILT_SIGNAL |
| * is safe since FreeBSD 4.10-STABLE. |
| * |
| * This threads implementation currently works on i386 (486+) and amd64 |
| * platforms only. |
| */ |
| |
| |
| char *ngx_freebsd_kern_usrstack; |
| size_t ngx_thread_stack_size; |
| |
| |
| static size_t rz_size; |
| static size_t usable_stack_size; |
| static char *last_stack; |
| |
| static ngx_uint_t nthreads; |
| static ngx_uint_t max_threads; |
| |
| static ngx_uint_t nkeys; |
| static ngx_tid_t *tids; /* the threads tids array */ |
| void **ngx_tls; /* the threads tls's array */ |
| |
| /* the thread-safe libc errno */ |
| |
| static int errno0; /* the main thread's errno */ |
| static int *errnos; /* the threads errno's array */ |
| |
| int * |
| __error() |
| { |
| int tid; |
| |
| tid = ngx_gettid(); |
| |
| return tid ? &errnos[tid - 1] : &errno0; |
| } |
| |
| |
| /* |
| * __isthreaded enables the spinlocks in some libc functions, i.e. in malloc() |
| * and some other places. Nevertheless we protect our malloc()/free() calls |
| * by own mutex that is more efficient than the spinlock. |
| * |
| * _spinlock() is a weak referenced stub in src/lib/libc/gen/_spinlock_stub.c |
| * that does nothing. |
| */ |
| |
| extern int __isthreaded; |
| |
| void |
| _spinlock(ngx_atomic_t *lock) |
| { |
| ngx_int_t tries; |
| |
| tries = 0; |
| |
| for ( ;; ) { |
| |
| if (*lock) { |
| if (ngx_ncpu > 1 && tries++ < 1000) { |
| continue; |
| } |
| |
| sched_yield(); |
| tries = 0; |
| |
| } else { |
| if (ngx_atomic_cmp_set(lock, 0, 1)) { |
| return; |
| } |
| } |
| } |
| } |
| |
| |
| /* |
| * Before FreeBSD 5.1 _spinunlock() is a simple #define in |
| * src/lib/libc/include/spinlock.h that zeroes lock. |
| * |
| * Since FreeBSD 5.1 _spinunlock() is a weak referenced stub in |
| * src/lib/libc/gen/_spinlock_stub.c that does nothing. |
| */ |
| |
| #ifndef _spinunlock |
| |
| void |
| _spinunlock(ngx_atomic_t *lock) |
| { |
| *lock = 0; |
| } |
| |
| #endif |
| |
| |
| ngx_err_t |
| ngx_create_thread(ngx_tid_t *tid, ngx_thread_value_t (*func)(void *arg), |
| void *arg, ngx_log_t *log) |
| { |
| ngx_pid_t id; |
| ngx_err_t err; |
| char *stack, *stack_top; |
| |
| 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; |
| } |
| |
| last_stack -= ngx_thread_stack_size; |
| |
| stack = mmap(last_stack, usable_stack_size, PROT_READ|PROT_WRITE, |
| MAP_STACK, -1, 0); |
| |
| if (stack == MAP_FAILED) { |
| ngx_log_error(NGX_LOG_ALERT, log, ngx_errno, |
| "mmap(%p:%uz, MAP_STACK) thread stack failed", |
| last_stack, usable_stack_size); |
| return NGX_ERROR; |
| } |
| |
| if (stack != last_stack) { |
| ngx_log_error(NGX_LOG_ALERT, log, 0, |
| "stack %p address was changed to %p", last_stack, stack); |
| return NGX_ERROR; |
| } |
| |
| stack_top = stack + usable_stack_size; |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_CORE, log, 0, |
| "thread stack: %p-%p", stack, stack_top); |
| |
| ngx_set_errno(0); |
| |
| id = rfork_thread(RFPROC|RFTHREAD|RFMEM, stack_top, |
| (ngx_rfork_thread_func_pt) func, arg); |
| |
| err = ngx_errno; |
| |
| if (id == -1) { |
| ngx_log_error(NGX_LOG_ALERT, log, err, "rfork() failed"); |
| |
| } else { |
| *tid = id; |
| nthreads = (ngx_freebsd_kern_usrstack - stack_top) |
| / ngx_thread_stack_size; |
| tids[nthreads] = id; |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, log, 0, "rfork()ed thread: %P", id); |
| } |
| |
| return err; |
| } |
| |
| |
| ngx_int_t |
| ngx_init_threads(int n, size_t size, ngx_cycle_t *cycle) |
| { |
| char *red_zone, *zone; |
| size_t len; |
| ngx_int_t i; |
| struct sigaction sa; |
| |
| max_threads = n + 1; |
| |
| for (i = 0; i < n; i++) { |
| ngx_memzero(&sa, sizeof(struct sigaction)); |
| sa.sa_handler = SIG_IGN; |
| sigemptyset(&sa.sa_mask); |
| if (sigaction(NGX_CV_SIGNAL, &sa, NULL) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno, |
| "sigaction(%d, SIG_IGN) failed", NGX_CV_SIGNAL); |
| return NGX_ERROR; |
| } |
| } |
| |
| len = sizeof(ngx_freebsd_kern_usrstack); |
| if (sysctlbyname("kern.usrstack", &ngx_freebsd_kern_usrstack, &len, |
| NULL, 0) == -1) |
| { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno, |
| "sysctlbyname(kern.usrstack) failed"); |
| return NGX_ERROR; |
| } |
| |
| /* the main thread stack red zone */ |
| rz_size = ngx_pagesize; |
| red_zone = ngx_freebsd_kern_usrstack - (size + rz_size); |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_CORE, cycle->log, 0, |
| "usrstack: %p red zone: %p", |
| ngx_freebsd_kern_usrstack, red_zone); |
| |
| zone = mmap(red_zone, rz_size, PROT_NONE, MAP_ANON, -1, 0); |
| if (zone == MAP_FAILED) { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno, |
| "mmap(%p:%uz, PROT_NONE, MAP_ANON) red zone failed", |
| red_zone, rz_size); |
| return NGX_ERROR; |
| } |
| |
| if (zone != red_zone) { |
| ngx_log_error(NGX_LOG_ALERT, cycle->log, 0, |
| "red zone %p address was changed to %p", red_zone, zone); |
| return NGX_ERROR; |
| } |
| |
| /* create the thread errno' array */ |
| |
| errnos = ngx_calloc(n * sizeof(int), cycle->log); |
| if (errnos == NULL) { |
| return NGX_ERROR; |
| } |
| |
| /* create the thread tids array */ |
| |
| tids = ngx_calloc((n + 1) * sizeof(ngx_tid_t), cycle->log); |
| if (tids == NULL) { |
| return NGX_ERROR; |
| } |
| |
| tids[0] = ngx_pid; |
| |
| /* create the thread tls' array */ |
| |
| ngx_tls = ngx_calloc(NGX_THREAD_KEYS_MAX * (n + 1) * sizeof(void *), |
| cycle->log); |
| if (ngx_tls == NULL) { |
| return NGX_ERROR; |
| } |
| |
| nthreads = 1; |
| |
| last_stack = zone + rz_size; |
| usable_stack_size = size; |
| ngx_thread_stack_size = size + rz_size; |
| |
| /* allow the spinlock in libc malloc() */ |
| __isthreaded = 1; |
| |
| ngx_threaded = 1; |
| |
| return NGX_OK; |
| } |
| |
| |
| ngx_tid_t |
| ngx_thread_self(void) |
| { |
| ngx_int_t tid; |
| |
| tid = ngx_gettid(); |
| |
| if (tids == NULL) { |
| return ngx_pid; |
| } |
| |
| return tids[tid]; |
| } |
| |
| |
| ngx_err_t |
| ngx_thread_key_create(ngx_tls_key_t *key) |
| { |
| if (nkeys >= NGX_THREAD_KEYS_MAX) { |
| return NGX_ENOMEM; |
| } |
| |
| *key = nkeys++; |
| |
| return 0; |
| } |
| |
| |
| ngx_err_t |
| ngx_thread_set_tls(ngx_tls_key_t key, void *value) |
| { |
| if (key >= NGX_THREAD_KEYS_MAX) { |
| return NGX_EINVAL; |
| } |
| |
| ngx_tls[key * NGX_THREAD_KEYS_MAX + ngx_gettid()] = value; |
| return 0; |
| } |
| |
| |
| ngx_mutex_t * |
| ngx_mutex_init(ngx_log_t *log, ngx_uint_t flags) |
| { |
| ngx_mutex_t *m; |
| union semun op; |
| |
| m = ngx_alloc(sizeof(ngx_mutex_t), log); |
| if (m == NULL) { |
| return NULL; |
| } |
| |
| m->lock = 0; |
| m->log = log; |
| |
| if (flags & NGX_MUTEX_LIGHT) { |
| m->semid = -1; |
| return m; |
| } |
| |
| m->semid = semget(IPC_PRIVATE, 1, SEM_R|SEM_A); |
| if (m->semid == -1) { |
| ngx_log_error(NGX_LOG_ALERT, log, ngx_errno, "semget() failed"); |
| return NULL; |
| } |
| |
| op.val = 0; |
| |
| if (semctl(m->semid, 0, SETVAL, op) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, log, ngx_errno, "semctl(SETVAL) failed"); |
| |
| if (semctl(m->semid, 0, IPC_RMID) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, log, ngx_errno, |
| "semctl(IPC_RMID) failed"); |
| } |
| |
| return NULL; |
| } |
| |
| return m; |
| } |
| |
| |
| void |
| ngx_mutex_destroy(ngx_mutex_t *m) |
| { |
| if (semctl(m->semid, 0, IPC_RMID) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, m->log, ngx_errno, |
| "semctl(IPC_RMID) failed"); |
| } |
| |
| ngx_free((void *) m); |
| } |
| |
| |
| ngx_int_t |
| ngx_mutex_dolock(ngx_mutex_t *m, ngx_int_t try) |
| { |
| uint32_t lock, old; |
| ngx_uint_t tries; |
| struct sembuf op; |
| |
| if (!ngx_threaded) { |
| return NGX_OK; |
| } |
| |
| #if (NGX_DEBUG) |
| if (try) { |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "try lock mutex %p lock:%XD", m, m->lock); |
| } else { |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "lock mutex %p lock:%XD", m, m->lock); |
| } |
| #endif |
| |
| old = m->lock; |
| tries = 0; |
| |
| for ( ;; ) { |
| if (old & NGX_MUTEX_LOCK_BUSY) { |
| |
| if (try) { |
| return NGX_AGAIN; |
| } |
| |
| if (ngx_ncpu > 1 && tries++ < 1000) { |
| |
| /* the spinlock is used only on the SMP system */ |
| |
| old = m->lock; |
| continue; |
| } |
| |
| if (m->semid == -1) { |
| sched_yield(); |
| |
| tries = 0; |
| old = m->lock; |
| continue; |
| } |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "mutex %p lock:%XD", m, m->lock); |
| |
| /* |
| * The mutex is locked so we increase a number |
| * of the threads that are waiting on the mutex |
| */ |
| |
| lock = old + 1; |
| |
| if ((lock & ~NGX_MUTEX_LOCK_BUSY) > nthreads) { |
| ngx_log_error(NGX_LOG_ALERT, m->log, ngx_errno, |
| "%D threads wait for mutex %p, " |
| "while only %ui threads are available", |
| lock & ~NGX_MUTEX_LOCK_BUSY, m, nthreads); |
| ngx_abort(); |
| } |
| |
| if (ngx_atomic_cmp_set(&m->lock, old, lock)) { |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "wait mutex %p lock:%XD", m, m->lock); |
| |
| /* |
| * The number of the waiting threads has been increased |
| * and we would wait on the SysV semaphore. |
| * A semaphore should wake up us more efficiently than |
| * a simple sched_yield() or usleep(). |
| */ |
| |
| op.sem_num = 0; |
| op.sem_op = -1; |
| op.sem_flg = 0; |
| |
| if (semop(m->semid, &op, 1) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, m->log, ngx_errno, |
| "semop() failed while waiting on mutex %p", m); |
| ngx_abort(); |
| } |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "mutex waked up %p lock:%XD", m, m->lock); |
| |
| tries = 0; |
| old = m->lock; |
| continue; |
| } |
| |
| old = m->lock; |
| |
| } else { |
| lock = old | NGX_MUTEX_LOCK_BUSY; |
| |
| if (ngx_atomic_cmp_set(&m->lock, old, lock)) { |
| |
| /* we locked the mutex */ |
| |
| break; |
| } |
| |
| old = m->lock; |
| } |
| |
| if (tries++ > 1000) { |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "mutex %p is contested", m); |
| |
| /* the mutex is probably contested so we are giving up now */ |
| |
| sched_yield(); |
| |
| tries = 0; |
| old = m->lock; |
| } |
| } |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "mutex %p is locked, lock:%XD", m, m->lock); |
| |
| return NGX_OK; |
| } |
| |
| |
| void |
| ngx_mutex_unlock(ngx_mutex_t *m) |
| { |
| uint32_t lock, old; |
| struct sembuf op; |
| |
| if (!ngx_threaded) { |
| return; |
| } |
| |
| old = m->lock; |
| |
| if (!(old & NGX_MUTEX_LOCK_BUSY)) { |
| ngx_log_error(NGX_LOG_ALERT, m->log, 0, |
| "trying to unlock the free mutex %p", m); |
| ngx_abort(); |
| } |
| |
| /* free the mutex */ |
| |
| #if 0 |
| ngx_log_debug2(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "unlock mutex %p lock:%XD", m, old); |
| #endif |
| |
| for ( ;; ) { |
| lock = old & ~NGX_MUTEX_LOCK_BUSY; |
| |
| if (ngx_atomic_cmp_set(&m->lock, old, lock)) { |
| break; |
| } |
| |
| old = m->lock; |
| } |
| |
| if (m->semid == -1) { |
| ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "mutex %p is unlocked", m); |
| |
| return; |
| } |
| |
| /* check whether we need to wake up a waiting thread */ |
| |
| old = m->lock; |
| |
| for ( ;; ) { |
| if (old & NGX_MUTEX_LOCK_BUSY) { |
| |
| /* the mutex is just locked by another thread */ |
| |
| break; |
| } |
| |
| if (old == 0) { |
| break; |
| } |
| |
| /* there are the waiting threads */ |
| |
| lock = old - 1; |
| |
| if (ngx_atomic_cmp_set(&m->lock, old, lock)) { |
| |
| /* wake up the thread that waits on semaphore */ |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_MUTEX, m->log, 0, |
| "wake up mutex %p", m); |
| |
| op.sem_num = 0; |
| op.sem_op = 1; |
| op.sem_flg = 0; |
| |
| if (semop(m->semid, &op, 1) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, m->log, ngx_errno, |
| "semop() failed while waking up on mutex %p", m); |
| ngx_abort(); |
| } |
| |
| break; |
| } |
| |
| old = m->lock; |
| } |
| |
| 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) |
| { |
| ngx_cond_t *cv; |
| |
| cv = ngx_alloc(sizeof(ngx_cond_t), log); |
| if (cv == NULL) { |
| return NULL; |
| } |
| |
| cv->signo = NGX_CV_SIGNAL; |
| cv->tid = -1; |
| cv->log = log; |
| cv->kq = -1; |
| |
| return cv; |
| } |
| |
| |
| void |
| ngx_cond_destroy(ngx_cond_t *cv) |
| { |
| if (close(cv->kq) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, cv->log, ngx_errno, |
| "kqueue close() failed"); |
| } |
| |
| ngx_free(cv); |
| } |
| |
| |
| ngx_int_t |
| ngx_cond_wait(ngx_cond_t *cv, ngx_mutex_t *m) |
| { |
| int n; |
| ngx_err_t err; |
| struct kevent kev; |
| struct timespec ts; |
| |
| if (cv->kq == -1) { |
| |
| /* |
| * We have to add the EVFILT_SIGNAL filter in the rfork()ed thread. |
| * Otherwise the thread would not get a signal event. |
| * |
| * However, we have not to open the kqueue in the thread, |
| * it is simply handy do it together. |
| */ |
| |
| cv->kq = kqueue(); |
| if (cv->kq == -1) { |
| ngx_log_error(NGX_LOG_ALERT, cv->log, ngx_errno, "kqueue() failed"); |
| return NGX_ERROR; |
| } |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_CORE, cv->log, 0, |
| "cv kq:%d signo:%d", cv->kq, cv->signo); |
| |
| kev.ident = cv->signo; |
| kev.filter = EVFILT_SIGNAL; |
| kev.flags = EV_ADD; |
| kev.fflags = 0; |
| kev.data = 0; |
| kev.udata = NULL; |
| |
| ts.tv_sec = 0; |
| ts.tv_nsec = 0; |
| |
| if (kevent(cv->kq, &kev, 1, NULL, 0, &ts) == -1) { |
| ngx_log_error(NGX_LOG_ALERT, cv->log, ngx_errno, "kevent() failed"); |
| return NGX_ERROR; |
| } |
| |
| cv->tid = ngx_thread_self(); |
| } |
| |
| ngx_mutex_unlock(m); |
| |
| ngx_log_debug3(NGX_LOG_DEBUG_CORE, cv->log, 0, |
| "cv %p wait, kq:%d, signo:%d", cv, cv->kq, cv->signo); |
| |
| for ( ;; ) { |
| n = kevent(cv->kq, NULL, 0, &kev, 1, NULL); |
| |
| ngx_log_debug2(NGX_LOG_DEBUG_CORE, cv->log, 0, |
| "cv %p kevent: %d", cv, n); |
| |
| if (n == -1) { |
| err = ngx_errno; |
| ngx_log_error((err == NGX_EINTR) ? NGX_LOG_INFO : NGX_LOG_ALERT, |
| cv->log, ngx_errno, |
| "kevent() failed while waiting condition variable %p", |
| cv); |
| |
| if (err == NGX_EINTR) { |
| break; |
| } |
| |
| return NGX_ERROR; |
| } |
| |
| if (n == 0) { |
| ngx_log_error(NGX_LOG_ALERT, cv->log, 0, |
| "kevent() returned no events " |
| "while waiting condition variable %p", |
| cv); |
| continue; |
| } |
| |
| if (kev.filter != EVFILT_SIGNAL) { |
| ngx_log_error(NGX_LOG_ALERT, cv->log, 0, |
| "kevent() returned unexpected events: %d " |
| "while waiting condition variable %p", |
| kev.filter, cv); |
| continue; |
| } |
| |
| if (kev.ident != (uintptr_t) cv->signo) { |
| ngx_log_error(NGX_LOG_ALERT, cv->log, 0, |
| "kevent() returned unexpected signal: %d ", |
| "while waiting condition variable %p", |
| kev.ident, cv); |
| continue; |
| } |
| |
| break; |
| } |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, cv->log, 0, "cv %p is waked up", cv); |
| |
| ngx_mutex_lock(m); |
| |
| return NGX_OK; |
| } |
| |
| |
| ngx_int_t |
| ngx_cond_signal(ngx_cond_t *cv) |
| { |
| ngx_err_t err; |
| |
| ngx_log_debug3(NGX_LOG_DEBUG_CORE, cv->log, 0, |
| "cv %p to signal %P %d", |
| cv, cv->tid, cv->signo); |
| |
| if (cv->tid == -1) { |
| return NGX_OK; |
| } |
| |
| if (kill(cv->tid, cv->signo) == -1) { |
| |
| err = ngx_errno; |
| |
| ngx_log_error(NGX_LOG_ALERT, cv->log, err, |
| "kill() failed while signaling condition variable %p", cv); |
| |
| if (err == NGX_ESRCH) { |
| cv->tid = -1; |
| } |
| |
| return NGX_ERROR; |
| } |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, cv->log, 0, "cv %p is signaled", cv); |
| |
| return NGX_OK; |
| } |