misc: introduce uv_get|set_threadpool_size
This patch adds the posibility to modify the threadpool size at runtime Fixes: https://github.com/libuv/libuv/issues/4401 Signed-off-by: Juan José Arboleda <soyjuanarbol@gmail.com>
This commit is contained in:
parent
e59e2a9e49
commit
e5471e5bd3
@ -670,6 +670,7 @@ if(LIBUV_BUILD_TESTS)
|
|||||||
test/test-thread-name.c
|
test/test-thread-name.c
|
||||||
test/test-thread-priority.c
|
test/test-thread-priority.c
|
||||||
test/test-threadpool-cancel.c
|
test/test-threadpool-cancel.c
|
||||||
|
test/test-threadpool-size.c
|
||||||
test/test-threadpool.c
|
test/test-threadpool.c
|
||||||
test/test-timer-again.c
|
test/test-timer-again.c
|
||||||
test/test-timer-from-check.c
|
test/test-timer-from-check.c
|
||||||
|
|||||||
@ -297,6 +297,7 @@ test_run_tests_SOURCES = test/blackhole-server.c \
|
|||||||
test/test-thread-name.c \
|
test/test-thread-name.c \
|
||||||
test/test-thread-priority.c \
|
test/test-thread-priority.c \
|
||||||
test/test-threadpool-cancel.c \
|
test/test-threadpool-cancel.c \
|
||||||
|
test/test-threadpool-size.c \
|
||||||
test/test-threadpool.c \
|
test/test-threadpool.c \
|
||||||
test/test-timer-again.c \
|
test/test-timer-again.c \
|
||||||
test/test-timer-from-check.c \
|
test/test-timer-from-check.c \
|
||||||
|
|||||||
@ -1337,6 +1337,9 @@ enum {
|
|||||||
UV_EXTERN int uv_thread_getpriority(uv_thread_t tid, int* priority);
|
UV_EXTERN int uv_thread_getpriority(uv_thread_t tid, int* priority);
|
||||||
UV_EXTERN int uv_thread_setpriority(uv_thread_t tid, int priority);
|
UV_EXTERN int uv_thread_setpriority(uv_thread_t tid, int priority);
|
||||||
|
|
||||||
|
UV_EXTERN unsigned int uv_get_threadpool_size(void);
|
||||||
|
UV_EXTERN unsigned int uv_set_threadpool_size(unsigned int n);
|
||||||
|
|
||||||
UV_EXTERN unsigned int uv_available_parallelism(void);
|
UV_EXTERN unsigned int uv_available_parallelism(void);
|
||||||
UV_EXTERN int uv_cpu_info(uv_cpu_info_t** cpu_infos, int* count);
|
UV_EXTERN int uv_cpu_info(uv_cpu_info_t** cpu_infos, int* count);
|
||||||
UV_EXTERN void uv_free_cpu_info(uv_cpu_info_t* cpu_infos, int count);
|
UV_EXTERN void uv_free_cpu_info(uv_cpu_info_t* cpu_infos, int count);
|
||||||
|
|||||||
159
src/threadpool.c
159
src/threadpool.c
@ -31,16 +31,20 @@
|
|||||||
|
|
||||||
static uv_once_t once = UV_ONCE_INIT;
|
static uv_once_t once = UV_ONCE_INIT;
|
||||||
static uv_cond_t cond;
|
static uv_cond_t cond;
|
||||||
|
static uv_cond_t cond_kill;
|
||||||
static uv_mutex_t mutex;
|
static uv_mutex_t mutex;
|
||||||
static unsigned int idle_threads;
|
static unsigned int idle_threads;
|
||||||
static unsigned int slow_io_work_running;
|
static unsigned int slow_io_work_running;
|
||||||
static unsigned int nthreads;
|
static unsigned int nthreads;
|
||||||
static uv_thread_t* threads;
|
static uv_thread_t* threads;
|
||||||
static uv_thread_t default_threads[4];
|
static uv_thread_t default_threads[4];
|
||||||
|
static uv_thread_t tid_to_join;
|
||||||
static struct uv__queue exit_message;
|
static struct uv__queue exit_message;
|
||||||
|
static struct uv__queue kill_thread_message;
|
||||||
static struct uv__queue wq;
|
static struct uv__queue wq;
|
||||||
static struct uv__queue run_slow_work_message;
|
static struct uv__queue run_slow_work_message;
|
||||||
static struct uv__queue slow_io_pending_wq;
|
static struct uv__queue slow_io_pending_wq;
|
||||||
|
static uv_sem_t sem;
|
||||||
|
|
||||||
static unsigned int slow_work_thread_threshold(void) {
|
static unsigned int slow_work_thread_threshold(void) {
|
||||||
return (nthreads + 1) / 2;
|
return (nthreads + 1) / 2;
|
||||||
@ -85,6 +89,16 @@ static void worker(void* arg) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Allow the main thread to kill targeted thread. */
|
||||||
|
if (q == &kill_thread_message) {
|
||||||
|
tid_to_join = uv_thread_self();
|
||||||
|
uv__queue_remove(q); /* Remove the kill message. */
|
||||||
|
uv__queue_init(q);
|
||||||
|
uv_cond_signal(&cond_kill);
|
||||||
|
uv_mutex_unlock(&mutex);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
uv__queue_remove(q);
|
uv__queue_remove(q);
|
||||||
uv__queue_init(q); /* Signal uv_cancel() that the work req is executing. */
|
uv__queue_init(q); /* Signal uv_cancel() that the work req is executing. */
|
||||||
|
|
||||||
@ -185,6 +199,7 @@ void uv__threadpool_cleanup(void) {
|
|||||||
|
|
||||||
uv_mutex_destroy(&mutex);
|
uv_mutex_destroy(&mutex);
|
||||||
uv_cond_destroy(&cond);
|
uv_cond_destroy(&cond);
|
||||||
|
uv_cond_destroy(&cond_kill);
|
||||||
|
|
||||||
threads = NULL;
|
threads = NULL;
|
||||||
nthreads = 0;
|
nthreads = 0;
|
||||||
@ -195,7 +210,6 @@ static void init_threads(void) {
|
|||||||
uv_thread_options_t config;
|
uv_thread_options_t config;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
const char* val;
|
const char* val;
|
||||||
uv_sem_t sem;
|
|
||||||
|
|
||||||
nthreads = ARRAY_SIZE(default_threads);
|
nthreads = ARRAY_SIZE(default_threads);
|
||||||
val = getenv("UV_THREADPOOL_SIZE");
|
val = getenv("UV_THREADPOOL_SIZE");
|
||||||
@ -218,6 +232,9 @@ static void init_threads(void) {
|
|||||||
if (uv_cond_init(&cond))
|
if (uv_cond_init(&cond))
|
||||||
abort();
|
abort();
|
||||||
|
|
||||||
|
if (uv_cond_init(&cond_kill))
|
||||||
|
abort();
|
||||||
|
|
||||||
if (uv_mutex_init(&mutex))
|
if (uv_mutex_init(&mutex))
|
||||||
abort();
|
abort();
|
||||||
|
|
||||||
@ -263,6 +280,146 @@ static void init_once(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int uv__threads_spin(unsigned int n) {
|
||||||
|
uv_thread_options_t config;
|
||||||
|
int r;
|
||||||
|
size_t i;
|
||||||
|
uv_thread_t* threads_tmp;
|
||||||
|
|
||||||
|
assert(n > nthreads);
|
||||||
|
|
||||||
|
config.flags = UV_THREAD_HAS_STACK_SIZE;
|
||||||
|
config.stack_size = 8u << 20; /* 8 MB */
|
||||||
|
|
||||||
|
threads_tmp = uv__malloc(n * sizeof(threads[0]));
|
||||||
|
if (threads_tmp == NULL)
|
||||||
|
return UV_ENOMEM;
|
||||||
|
|
||||||
|
|
||||||
|
/* Copy all threads into the new list. */
|
||||||
|
for (i = 0; i < nthreads; i++)
|
||||||
|
threads_tmp[i] = threads[i];
|
||||||
|
|
||||||
|
/* Free the old list. And make it points to the new one */
|
||||||
|
if (threads != default_threads)
|
||||||
|
uv__free(threads);
|
||||||
|
|
||||||
|
threads = threads_tmp;
|
||||||
|
|
||||||
|
r = uv_sem_init(&sem, 0);
|
||||||
|
if (r)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
while (nthreads < n) {
|
||||||
|
/* Wire up the error code to the return value. */
|
||||||
|
r = uv_thread_create_ex(threads + nthreads, &config, worker, &sem);
|
||||||
|
if (r)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
uv_sem_wait(&sem);
|
||||||
|
nthreads++;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(nthreads == n);
|
||||||
|
|
||||||
|
out:
|
||||||
|
uv_sem_destroy(&sem);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int uv__threads_join(unsigned int n) {
|
||||||
|
int r;
|
||||||
|
size_t i;
|
||||||
|
size_t j;
|
||||||
|
uv_thread_t* threads_tmp;
|
||||||
|
|
||||||
|
assert(nthreads > n);
|
||||||
|
while (nthreads != n) {
|
||||||
|
/* Request any thread to kill */
|
||||||
|
uv__queue_insert_tail(&wq, &kill_thread_message);
|
||||||
|
|
||||||
|
if (idle_threads > 0)
|
||||||
|
uv_cond_signal(&cond);
|
||||||
|
|
||||||
|
uv_cond_wait(&cond_kill, &mutex);
|
||||||
|
r = uv_thread_join(&tid_to_join);
|
||||||
|
if (r)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
/* Alloc for one thread less */
|
||||||
|
threads_tmp = uv__malloc((nthreads - 1) * sizeof(threads[0]));
|
||||||
|
if (threads_tmp == NULL)
|
||||||
|
return UV_ENOMEM;
|
||||||
|
|
||||||
|
|
||||||
|
/* Copy all threads into the new list except the joined one. */
|
||||||
|
j = 0;
|
||||||
|
for (i = 0; i < nthreads; i++) {
|
||||||
|
if (tid_to_join != threads[i]) {
|
||||||
|
if (j >= nthreads) abort();
|
||||||
|
threads_tmp[j] = threads[i];
|
||||||
|
j++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Free the old list (if malloc'd) */
|
||||||
|
if (threads != default_threads)
|
||||||
|
uv__free(threads);
|
||||||
|
|
||||||
|
/* Decrease the number of threads */
|
||||||
|
nthreads--;
|
||||||
|
threads = threads_tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(nthreads == n);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int uv_get_threadpool_size(void) {
|
||||||
|
int r;
|
||||||
|
|
||||||
|
uv_once(&once, init_once);
|
||||||
|
|
||||||
|
uv_mutex_lock(&mutex);
|
||||||
|
r = nthreads;
|
||||||
|
uv_mutex_unlock(&mutex);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
unsigned int uv_set_threadpool_size(unsigned int n) {
|
||||||
|
int r;
|
||||||
|
uv_once(&once, init_once);
|
||||||
|
|
||||||
|
/* For now, NO-threadpool is not supported. */
|
||||||
|
if (n > MAX_THREADPOOL_SIZE || n < 1)
|
||||||
|
return UV_EINVAL;
|
||||||
|
|
||||||
|
uv_mutex_lock(&mutex);
|
||||||
|
|
||||||
|
/* No-op */
|
||||||
|
if (n == nthreads) {
|
||||||
|
uv_mutex_unlock(&mutex);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Shrink the threadpool request. */
|
||||||
|
if (n < nthreads) {
|
||||||
|
r = uv__threads_join(n);
|
||||||
|
uv_mutex_unlock(&mutex);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Grow the threadpool. */
|
||||||
|
r = uv__threads_spin(n);
|
||||||
|
uv_mutex_unlock(&mutex);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void uv__work_submit(uv_loop_t* loop,
|
void uv__work_submit(uv_loop_t* loop,
|
||||||
struct uv__work* w,
|
struct uv__work* w,
|
||||||
enum uv__work_kind kind,
|
enum uv__work_kind kind,
|
||||||
|
|||||||
@ -456,6 +456,7 @@ TEST_DECLARE (fs_wtf)
|
|||||||
TEST_DECLARE (fs_get_system_error)
|
TEST_DECLARE (fs_get_system_error)
|
||||||
TEST_DECLARE (strscpy)
|
TEST_DECLARE (strscpy)
|
||||||
TEST_DECLARE (strtok)
|
TEST_DECLARE (strtok)
|
||||||
|
TEST_DECLARE (threadpool_size)
|
||||||
TEST_DECLARE (threadpool_queue_work_simple)
|
TEST_DECLARE (threadpool_queue_work_simple)
|
||||||
TEST_DECLARE (threadpool_queue_work_einval)
|
TEST_DECLARE (threadpool_queue_work_einval)
|
||||||
TEST_DECLARE (threadpool_multiple_event_loops)
|
TEST_DECLARE (threadpool_multiple_event_loops)
|
||||||
@ -1175,6 +1176,7 @@ TASK_LIST_START
|
|||||||
TEST_ENTRY (open_osfhandle_valid_handle)
|
TEST_ENTRY (open_osfhandle_valid_handle)
|
||||||
TEST_ENTRY (strscpy)
|
TEST_ENTRY (strscpy)
|
||||||
TEST_ENTRY (strtok)
|
TEST_ENTRY (strtok)
|
||||||
|
TEST_ENTRY (threadpool_size)
|
||||||
TEST_ENTRY (threadpool_queue_work_simple)
|
TEST_ENTRY (threadpool_queue_work_simple)
|
||||||
TEST_ENTRY (threadpool_queue_work_einval)
|
TEST_ENTRY (threadpool_queue_work_einval)
|
||||||
TEST_ENTRY_CUSTOM (threadpool_multiple_event_loops, 0, 0, 60000)
|
TEST_ENTRY_CUSTOM (threadpool_multiple_event_loops, 0, 0, 60000)
|
||||||
|
|||||||
45
test/test-threadpool-size.c
Normal file
45
test/test-threadpool-size.c
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
/* Copyright libuv project contributors. All rights reserved.
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
* deal in the Software without restriction, including without limitation the
|
||||||
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
||||||
|
* sell copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
||||||
|
* IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "uv.h"
|
||||||
|
#include "task.h"
|
||||||
|
|
||||||
|
TEST_IMPL(threadpool_size) {
|
||||||
|
/* The default thread pool size is 4 */
|
||||||
|
ASSERT_EQ(uv_get_threadpool_size(), 4);
|
||||||
|
|
||||||
|
/* Normal use case (increase the size of the thread pool) */
|
||||||
|
ASSERT_OK(uv_set_threadpool_size(5));
|
||||||
|
ASSERT_EQ(uv_get_threadpool_size(), 5);
|
||||||
|
|
||||||
|
/* Shrink the thread pool */
|
||||||
|
ASSERT_OK(uv_set_threadpool_size(4));
|
||||||
|
ASSERT_EQ(uv_get_threadpool_size(), 4);
|
||||||
|
|
||||||
|
/* Grow the thread pool again */
|
||||||
|
ASSERT_OK(uv_set_threadpool_size(5));
|
||||||
|
ASSERT_EQ(uv_get_threadpool_size(), 5);
|
||||||
|
|
||||||
|
uv_run(uv_default_loop(), UV_RUN_DEFAULT);
|
||||||
|
|
||||||
|
MAKE_VALGRIND_HAPPY(uv_default_loop());
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue
Block a user