⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ngx_freebsd_rfork_thread.c

📁 nginx 反向代理0.7.1版本 用于实现反向代理
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * Copyright (C) Igor Sysoev */#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 determinated 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 * mutecies 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 _spinunlockvoid_spinunlock(ngx_atomic_t *lock){    *lock = 0;}#endifngx_err_tngx_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_tngx_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_tngx_thread_self(){    ngx_int_t  tid;    tid = ngx_gettid();    if (tids == NULL) {        return ngx_pid;    }    return tids[tid];}ngx_err_tngx_thread_key_create(ngx_tls_key_t *key){    if (nkeys >= NGX_THREAD_KEYS_MAX) {        return NGX_ENOMEM;    }    *key = nkeys++;    return 0;}ngx_err_tngx_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;}voidngx_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_tngx_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) {

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -