Blame src/threadpool.c

Packit Service 7c31a4
/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit Service 7c31a4
 * of this software and associated documentation files (the "Software"), to
Packit Service 7c31a4
 * deal in the Software without restriction, including without limitation the
Packit Service 7c31a4
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit Service 7c31a4
 * sell copies of the Software, and to permit persons to whom the Software is
Packit Service 7c31a4
 * furnished to do so, subject to the following conditions:
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * The above copyright notice and this permission notice shall be included in
Packit Service 7c31a4
 * all copies or substantial portions of the Software.
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit Service 7c31a4
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit Service 7c31a4
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit Service 7c31a4
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit Service 7c31a4
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit Service 7c31a4
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit Service 7c31a4
 * IN THE SOFTWARE.
Packit Service 7c31a4
 */
Packit Service 7c31a4
Packit Service 7c31a4
#include "uv-common.h"
Packit Service 7c31a4
Packit Service 7c31a4
#if !defined(_WIN32)
Packit Service 7c31a4
# include "unix/internal.h"
Packit Service 7c31a4
#endif
Packit Service 7c31a4
Packit Service 7c31a4
#include <stdlib.h>
Packit Service 7c31a4
Packit Service 7c31a4
#define MAX_THREADPOOL_SIZE 1024
Packit Service 7c31a4
Packit Service 7c31a4
static uv_once_t once = UV_ONCE_INIT;
Packit Service 7c31a4
static uv_cond_t cond;
Packit Service 7c31a4
static uv_mutex_t mutex;
Packit Service 7c31a4
static unsigned int idle_threads;
Packit Service 7c31a4
static unsigned int slow_io_work_running;
Packit Service 7c31a4
static unsigned int nthreads;
Packit Service 7c31a4
static uv_thread_t* threads;
Packit Service 7c31a4
static uv_thread_t default_threads[4];
Packit Service 7c31a4
static QUEUE exit_message;
Packit Service 7c31a4
static QUEUE wq;
Packit Service 7c31a4
static QUEUE run_slow_work_message;
Packit Service 7c31a4
static QUEUE slow_io_pending_wq;
Packit Service 7c31a4
Packit Service 7c31a4
static unsigned int slow_work_thread_threshold(void) {
Packit Service 7c31a4
  return (nthreads + 1) / 2;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
static void uv__cancelled(struct uv__work* w) {
Packit Service 7c31a4
  abort();
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
/* To avoid deadlock with uv_cancel() it's crucial that the worker
Packit Service 7c31a4
 * never holds the global mutex and the loop-local mutex at the same time.
Packit Service 7c31a4
 */
Packit Service 7c31a4
static void worker(void* arg) {
Packit Service 7c31a4
  struct uv__work* w;
Packit Service 7c31a4
  QUEUE* q;
Packit Service 7c31a4
  int is_slow_work;
Packit Service 7c31a4
Packit Service 7c31a4
  uv_sem_post((uv_sem_t*) arg);
Packit Service 7c31a4
  arg = NULL;
Packit Service 7c31a4
Packit Service 7c31a4
  uv_mutex_lock(&mutex);
Packit Service 7c31a4
  for (;;) {
Packit Service 7c31a4
    /* `mutex` should always be locked at this point. */
Packit Service 7c31a4
Packit Service 7c31a4
    /* Keep waiting while either no work is present or only slow I/O
Packit Service 7c31a4
       and we're at the threshold for that. */
Packit Service 7c31a4
    while (QUEUE_EMPTY(&wq) ||
Packit Service 7c31a4
           (QUEUE_HEAD(&wq) == &run_slow_work_message &&
Packit Service 7c31a4
            QUEUE_NEXT(&run_slow_work_message) == &wq &&
Packit Service 7c31a4
            slow_io_work_running >= slow_work_thread_threshold())) {
Packit Service 7c31a4
      idle_threads += 1;
Packit Service 7c31a4
      uv_cond_wait(&cond, &mutex);
Packit Service 7c31a4
      idle_threads -= 1;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    q = QUEUE_HEAD(&wq;;
Packit Service 7c31a4
    if (q == &exit_message) {
Packit Service 7c31a4
      uv_cond_signal(&cond;;
Packit Service 7c31a4
      uv_mutex_unlock(&mutex);
Packit Service 7c31a4
      break;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    QUEUE_REMOVE(q);
Packit Service 7c31a4
    QUEUE_INIT(q);  /* Signal uv_cancel() that the work req is executing. */
Packit Service 7c31a4
Packit Service 7c31a4
    is_slow_work = 0;
Packit Service 7c31a4
    if (q == &run_slow_work_message) {
Packit Service 7c31a4
      /* If we're at the slow I/O threshold, re-schedule until after all
Packit Service 7c31a4
         other work in the queue is done. */
Packit Service 7c31a4
      if (slow_io_work_running >= slow_work_thread_threshold()) {
Packit Service 7c31a4
        QUEUE_INSERT_TAIL(&wq, q);
Packit Service 7c31a4
        continue;
Packit Service 7c31a4
      }
Packit Service 7c31a4
Packit Service 7c31a4
      /* If we encountered a request to run slow I/O work but there is none
Packit Service 7c31a4
         to run, that means it's cancelled => Start over. */
Packit Service 7c31a4
      if (QUEUE_EMPTY(&slow_io_pending_wq))
Packit Service 7c31a4
        continue;
Packit Service 7c31a4
Packit Service 7c31a4
      is_slow_work = 1;
Packit Service 7c31a4
      slow_io_work_running++;
Packit Service 7c31a4
Packit Service 7c31a4
      q = QUEUE_HEAD(&slow_io_pending_wq);
Packit Service 7c31a4
      QUEUE_REMOVE(q);
Packit Service 7c31a4
      QUEUE_INIT(q);
Packit Service 7c31a4
Packit Service 7c31a4
      /* If there is more slow I/O work, schedule it to be run as well. */
Packit Service 7c31a4
      if (!QUEUE_EMPTY(&slow_io_pending_wq)) {
Packit Service 7c31a4
        QUEUE_INSERT_TAIL(&wq, &run_slow_work_message);
Packit Service 7c31a4
        if (idle_threads > 0)
Packit Service 7c31a4
          uv_cond_signal(&cond;;
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    uv_mutex_unlock(&mutex);
Packit Service 7c31a4
Packit Service 7c31a4
    w = QUEUE_DATA(q, struct uv__work, wq);
Packit Service 7c31a4
    w->work(w);
Packit Service 7c31a4
Packit Service 7c31a4
    uv_mutex_lock(&w->loop->wq_mutex);
Packit Service 7c31a4
    w->work = NULL;  /* Signal uv_cancel() that the work req is done
Packit Service 7c31a4
                        executing. */
Packit Service 7c31a4
    QUEUE_INSERT_TAIL(&w->loop->wq, &w->wq);
Packit Service 7c31a4
    uv_async_send(&w->loop->wq_async);
Packit Service 7c31a4
    uv_mutex_unlock(&w->loop->wq_mutex);
Packit Service 7c31a4
Packit Service 7c31a4
    /* Lock `mutex` since that is expected at the start of the next
Packit Service 7c31a4
     * iteration. */
Packit Service 7c31a4
    uv_mutex_lock(&mutex);
Packit Service 7c31a4
    if (is_slow_work) {
Packit Service 7c31a4
      /* `slow_io_work_running` is protected by `mutex`. */
Packit Service 7c31a4
      slow_io_work_running--;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void post(QUEUE* q, enum uv__work_kind kind) {
Packit Service 7c31a4
  uv_mutex_lock(&mutex);
Packit Service 7c31a4
  if (kind == UV__WORK_SLOW_IO) {
Packit Service 7c31a4
    /* Insert into a separate queue. */
Packit Service 7c31a4
    QUEUE_INSERT_TAIL(&slow_io_pending_wq, q);
Packit Service 7c31a4
    if (!QUEUE_EMPTY(&run_slow_work_message)) {
Packit Service 7c31a4
      /* Running slow I/O tasks is already scheduled => Nothing to do here.
Packit Service 7c31a4
         The worker that runs said other task will schedule this one as well. */
Packit Service 7c31a4
      uv_mutex_unlock(&mutex);
Packit Service 7c31a4
      return;
Packit Service 7c31a4
    }
Packit Service 7c31a4
    q = &run_slow_work_message;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  QUEUE_INSERT_TAIL(&wq, q);
Packit Service 7c31a4
  if (idle_threads > 0)
Packit Service 7c31a4
    uv_cond_signal(&cond;;
Packit Service 7c31a4
  uv_mutex_unlock(&mutex);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv__threadpool_cleanup(void) {
Packit Service 7c31a4
#ifndef _WIN32
Packit Service 7c31a4
  unsigned int i;
Packit Service 7c31a4
Packit Service 7c31a4
  if (nthreads == 0)
Packit Service 7c31a4
    return;
Packit Service 7c31a4
Packit Service 7c31a4
  post(&exit_message, UV__WORK_CPU);
Packit Service 7c31a4
Packit Service 7c31a4
  for (i = 0; i < nthreads; i++)
Packit Service 7c31a4
    if (uv_thread_join(threads + i))
Packit Service 7c31a4
      abort();
Packit Service 7c31a4
Packit Service 7c31a4
  if (threads != default_threads)
Packit Service 7c31a4
    uv__free(threads);
Packit Service 7c31a4
Packit Service 7c31a4
  uv_mutex_destroy(&mutex);
Packit Service 7c31a4
  uv_cond_destroy(&cond;;
Packit Service 7c31a4
Packit Service 7c31a4
  threads = NULL;
Packit Service 7c31a4
  nthreads = 0;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void init_threads(void) {
Packit Service 7c31a4
  unsigned int i;
Packit Service 7c31a4
  const char* val;
Packit Service 7c31a4
  uv_sem_t sem;
Packit Service 7c31a4
Packit Service 7c31a4
  nthreads = ARRAY_SIZE(default_threads);
Packit Service 7c31a4
  val = getenv("UV_THREADPOOL_SIZE");
Packit Service 7c31a4
  if (val != NULL)
Packit Service 7c31a4
    nthreads = atoi(val);
Packit Service 7c31a4
  if (nthreads == 0)
Packit Service 7c31a4
    nthreads = 1;
Packit Service 7c31a4
  if (nthreads > MAX_THREADPOOL_SIZE)
Packit Service 7c31a4
    nthreads = MAX_THREADPOOL_SIZE;
Packit Service 7c31a4
Packit Service 7c31a4
  threads = default_threads;
Packit Service 7c31a4
  if (nthreads > ARRAY_SIZE(default_threads)) {
Packit Service 7c31a4
    threads = uv__malloc(nthreads * sizeof(threads[0]));
Packit Service 7c31a4
    if (threads == NULL) {
Packit Service 7c31a4
      nthreads = ARRAY_SIZE(default_threads);
Packit Service 7c31a4
      threads = default_threads;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (uv_cond_init(&cond))
Packit Service 7c31a4
    abort();
Packit Service 7c31a4
Packit Service 7c31a4
  if (uv_mutex_init(&mutex))
Packit Service 7c31a4
    abort();
Packit Service 7c31a4
Packit Service 7c31a4
  QUEUE_INIT(&wq;;
Packit Service 7c31a4
  QUEUE_INIT(&slow_io_pending_wq);
Packit Service 7c31a4
  QUEUE_INIT(&run_slow_work_message);
Packit Service 7c31a4
Packit Service 7c31a4
  if (uv_sem_init(&sem, 0))
Packit Service 7c31a4
    abort();
Packit Service 7c31a4
Packit Service 7c31a4
  for (i = 0; i < nthreads; i++)
Packit Service 7c31a4
    if (uv_thread_create(threads + i, worker, &sem))
Packit Service 7c31a4
      abort();
Packit Service 7c31a4
Packit Service 7c31a4
  for (i = 0; i < nthreads; i++)
Packit Service 7c31a4
    uv_sem_wait(&sem);
Packit Service 7c31a4
Packit Service 7c31a4
  uv_sem_destroy(&sem);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
#ifndef _WIN32
Packit Service 7c31a4
static void reset_once(void) {
Packit Service 7c31a4
  uv_once_t child_once = UV_ONCE_INIT;
Packit Service 7c31a4
  memcpy(&once, &child_once, sizeof(child_once));
Packit Service 7c31a4
}
Packit Service 7c31a4
#endif
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void init_once(void) {
Packit Service 7c31a4
#ifndef _WIN32
Packit Service 7c31a4
  /* Re-initialize the threadpool after fork.
Packit Service 7c31a4
   * Note that this discards the global mutex and condition as well
Packit Service 7c31a4
   * as the work queue.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  if (pthread_atfork(NULL, NULL, &reset_once))
Packit Service 7c31a4
    abort();
Packit Service 7c31a4
#endif
Packit Service 7c31a4
  init_threads();
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv__work_submit(uv_loop_t* loop,
Packit Service 7c31a4
                     struct uv__work* w,
Packit Service 7c31a4
                     enum uv__work_kind kind,
Packit Service 7c31a4
                     void (*work)(struct uv__work* w),
Packit Service 7c31a4
                     void (*done)(struct uv__work* w, int status)) {
Packit Service 7c31a4
  uv_once(&once, init_once);
Packit Service 7c31a4
  w->loop = loop;
Packit Service 7c31a4
  w->work = work;
Packit Service 7c31a4
  w->done = done;
Packit Service 7c31a4
  post(&w->wq, kind);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static int uv__work_cancel(uv_loop_t* loop, uv_req_t* req, struct uv__work* w) {
Packit Service 7c31a4
  int cancelled;
Packit Service 7c31a4
Packit Service 7c31a4
  uv_mutex_lock(&mutex);
Packit Service 7c31a4
  uv_mutex_lock(&w->loop->wq_mutex);
Packit Service 7c31a4
Packit Service 7c31a4
  cancelled = !QUEUE_EMPTY(&w->wq) && w->work != NULL;
Packit Service 7c31a4
  if (cancelled)
Packit Service 7c31a4
    QUEUE_REMOVE(&w->wq);
Packit Service 7c31a4
Packit Service 7c31a4
  uv_mutex_unlock(&w->loop->wq_mutex);
Packit Service 7c31a4
  uv_mutex_unlock(&mutex);
Packit Service 7c31a4
Packit Service 7c31a4
  if (!cancelled)
Packit Service 7c31a4
    return UV_EBUSY;
Packit Service 7c31a4
Packit Service 7c31a4
  w->work = uv__cancelled;
Packit Service 7c31a4
  uv_mutex_lock(&loop->wq_mutex);
Packit Service 7c31a4
  QUEUE_INSERT_TAIL(&loop->wq, &w->wq);
Packit Service 7c31a4
  uv_async_send(&loop->wq_async);
Packit Service 7c31a4
  uv_mutex_unlock(&loop->wq_mutex);
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv__work_done(uv_async_t* handle) {
Packit Service 7c31a4
  struct uv__work* w;
Packit Service 7c31a4
  uv_loop_t* loop;
Packit Service 7c31a4
  QUEUE* q;
Packit Service 7c31a4
  QUEUE wq;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  loop = container_of(handle, uv_loop_t, wq_async);
Packit Service 7c31a4
  uv_mutex_lock(&loop->wq_mutex);
Packit Service 7c31a4
  QUEUE_MOVE(&loop->wq, &wq;;
Packit Service 7c31a4
  uv_mutex_unlock(&loop->wq_mutex);
Packit Service 7c31a4
Packit Service 7c31a4
  while (!QUEUE_EMPTY(&wq)) {
Packit Service 7c31a4
    q = QUEUE_HEAD(&wq;;
Packit Service 7c31a4
    QUEUE_REMOVE(q);
Packit Service 7c31a4
Packit Service 7c31a4
    w = container_of(q, struct uv__work, wq);
Packit Service 7c31a4
    err = (w->work == uv__cancelled) ? UV_ECANCELED : 0;
Packit Service 7c31a4
    w->done(w, err);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv__queue_work(struct uv__work* w) {
Packit Service 7c31a4
  uv_work_t* req = container_of(w, uv_work_t, work_req);
Packit Service 7c31a4
Packit Service 7c31a4
  req->work_cb(req);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv__queue_done(struct uv__work* w, int err) {
Packit Service 7c31a4
  uv_work_t* req;
Packit Service 7c31a4
Packit Service 7c31a4
  req = container_of(w, uv_work_t, work_req);
Packit Service 7c31a4
  uv__req_unregister(req->loop, req);
Packit Service 7c31a4
Packit Service 7c31a4
  if (req->after_work_cb == NULL)
Packit Service 7c31a4
    return;
Packit Service 7c31a4
Packit Service 7c31a4
  req->after_work_cb(req, err);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_queue_work(uv_loop_t* loop,
Packit Service 7c31a4
                  uv_work_t* req,
Packit Service 7c31a4
                  uv_work_cb work_cb,
Packit Service 7c31a4
                  uv_after_work_cb after_work_cb) {
Packit Service 7c31a4
  if (work_cb == NULL)
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
Packit Service 7c31a4
  uv__req_init(loop, req, UV_WORK);
Packit Service 7c31a4
  req->loop = loop;
Packit Service 7c31a4
  req->work_cb = work_cb;
Packit Service 7c31a4
  req->after_work_cb = after_work_cb;
Packit Service 7c31a4
  uv__work_submit(loop,
Packit Service 7c31a4
                  &req->work_req,
Packit Service 7c31a4
                  UV__WORK_CPU,
Packit Service 7c31a4
                  uv__queue_work,
Packit Service 7c31a4
                  uv__queue_done);
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_cancel(uv_req_t* req) {
Packit Service 7c31a4
  struct uv__work* wreq;
Packit Service 7c31a4
  uv_loop_t* loop;
Packit Service 7c31a4
Packit Service 7c31a4
  switch (req->type) {
Packit Service 7c31a4
  case UV_FS:
Packit Service 7c31a4
    loop =  ((uv_fs_t*) req)->loop;
Packit Service 7c31a4
    wreq = &((uv_fs_t*) req)->work_req;
Packit Service 7c31a4
    break;
Packit Service 7c31a4
  case UV_GETADDRINFO:
Packit Service 7c31a4
    loop =  ((uv_getaddrinfo_t*) req)->loop;
Packit Service 7c31a4
    wreq = &((uv_getaddrinfo_t*) req)->work_req;
Packit Service 7c31a4
    break;
Packit Service 7c31a4
  case UV_GETNAMEINFO:
Packit Service 7c31a4
    loop = ((uv_getnameinfo_t*) req)->loop;
Packit Service 7c31a4
    wreq = &((uv_getnameinfo_t*) req)->work_req;
Packit Service 7c31a4
    break;
Packit Service 7c31a4
  case UV_RANDOM:
Packit Service 7c31a4
    loop = ((uv_random_t*) req)->loop;
Packit Service 7c31a4
    wreq = &((uv_random_t*) req)->work_req;
Packit Service 7c31a4
    break;
Packit Service 7c31a4
  case UV_WORK:
Packit Service 7c31a4
    loop =  ((uv_work_t*) req)->loop;
Packit Service 7c31a4
    wreq = &((uv_work_t*) req)->work_req;
Packit Service 7c31a4
    break;
Packit Service 7c31a4
  default:
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return uv__work_cancel(loop, req, wreq);
Packit Service 7c31a4
}