Run arc_evict thread at higher priority

Run arc_evict thread at higher priority, nice=0, to give it more CPU
time which can improve performance for workload with high ARC evict
activities.

On mixed read/write and sequential read workloads, I've seen between
10-40% better performance.

Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov>
Reviewed-by: George Melikov <mail@gmelikov.ru>
Reviewed-by: Alexander Motin <mav@FreeBSD.org>
Signed-off-by: Tony Nguyen <tony.nguyen@delphix.com>
Closes #12397
This commit is contained in:
Tony Nguyen 2021-08-10 11:36:26 -06:00 committed by GitHub
parent f3678d70ff
commit 6bc61d22c4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 22 additions and 15 deletions

View File

@ -25,10 +25,11 @@ typedef void (zthr_func_t)(void *, zthr_t *);
typedef boolean_t (zthr_checkfunc_t)(void *, zthr_t *); typedef boolean_t (zthr_checkfunc_t)(void *, zthr_t *);
extern zthr_t *zthr_create(const char *zthr_name, extern zthr_t *zthr_create(const char *zthr_name,
zthr_checkfunc_t checkfunc, zthr_func_t *func, void *arg); zthr_checkfunc_t checkfunc, zthr_func_t *func, void *arg,
pri_t pri);
extern zthr_t *zthr_create_timer(const char *zthr_name, extern zthr_t *zthr_create_timer(const char *zthr_name,
zthr_checkfunc_t *checkfunc, zthr_func_t *func, void *arg, zthr_checkfunc_t *checkfunc, zthr_func_t *func, void *arg,
hrtime_t nano_wait); hrtime_t nano_wait, pri_t pri);
extern void zthr_destroy(zthr_t *t); extern void zthr_destroy(zthr_t *t);
extern void zthr_wakeup(zthr_t *t); extern void zthr_wakeup(zthr_t *t);

View File

@ -7950,9 +7950,9 @@ arc_init(void)
} }
arc_evict_zthr = zthr_create("arc_evict", arc_evict_zthr = zthr_create("arc_evict",
arc_evict_cb_check, arc_evict_cb, NULL); arc_evict_cb_check, arc_evict_cb, NULL, defclsyspri);
arc_reap_zthr = zthr_create_timer("arc_reap", arc_reap_zthr = zthr_create_timer("arc_reap",
arc_reap_cb_check, arc_reap_cb, NULL, SEC2NSEC(1)); arc_reap_cb_check, arc_reap_cb, NULL, SEC2NSEC(1), minclsyspri);
arc_warm = B_FALSE; arc_warm = B_FALSE;

View File

@ -2610,7 +2610,8 @@ spa_start_livelist_destroy_thread(spa_t *spa)
ASSERT3P(spa->spa_livelist_delete_zthr, ==, NULL); ASSERT3P(spa->spa_livelist_delete_zthr, ==, NULL);
spa->spa_livelist_delete_zthr = spa->spa_livelist_delete_zthr =
zthr_create("z_livelist_destroy", zthr_create("z_livelist_destroy",
spa_livelist_delete_cb_check, spa_livelist_delete_cb, spa); spa_livelist_delete_cb_check, spa_livelist_delete_cb, spa,
minclsyspri);
} }
typedef struct livelist_new_arg { typedef struct livelist_new_arg {
@ -2820,7 +2821,7 @@ spa_start_livelist_condensing_thread(spa_t *spa)
spa->spa_livelist_condense_zthr = spa->spa_livelist_condense_zthr =
zthr_create("z_livelist_condense", zthr_create("z_livelist_condense",
spa_livelist_condense_cb_check, spa_livelist_condense_cb_check,
spa_livelist_condense_cb, spa); spa_livelist_condense_cb, spa, minclsyspri);
} }
static void static void
@ -2838,7 +2839,7 @@ spa_spawn_aux_threads(spa_t *spa)
spa->spa_checkpoint_discard_zthr = spa->spa_checkpoint_discard_zthr =
zthr_create("z_checkpoint_discard", zthr_create("z_checkpoint_discard",
spa_checkpoint_discard_thread_check, spa_checkpoint_discard_thread_check,
spa_checkpoint_discard_thread, spa); spa_checkpoint_discard_thread, spa, minclsyspri);
} }
/* /*

View File

@ -885,7 +885,7 @@ spa_start_indirect_condensing_thread(spa_t *spa)
ASSERT3P(spa->spa_condense_zthr, ==, NULL); ASSERT3P(spa->spa_condense_zthr, ==, NULL);
spa->spa_condense_zthr = zthr_create("z_indirect_condense", spa->spa_condense_zthr = zthr_create("z_indirect_condense",
spa_condense_indirect_thread_check, spa_condense_indirect_thread_check,
spa_condense_indirect_thread, spa); spa_condense_indirect_thread, spa, minclsyspri);
} }
/* /*

View File

@ -83,10 +83,11 @@
* can be cancelled while doing work and not while checking for work. * can be cancelled while doing work and not while checking for work.
* *
* To start a zthr: * To start a zthr:
* zthr_t *zthr_pointer = zthr_create(checkfunc, func, args); * zthr_t *zthr_pointer = zthr_create(checkfunc, func, args,
* pri);
* or * or
* zthr_t *zthr_pointer = zthr_create_timer(checkfunc, func, * zthr_t *zthr_pointer = zthr_create_timer(checkfunc, func,
* args, max_sleep); * args, max_sleep, pri);
* *
* After that you should be able to wakeup, cancel, and resume the * After that you should be able to wakeup, cancel, and resume the
* zthr from another thread using the zthr_pointer. * zthr from another thread using the zthr_pointer.
@ -220,6 +221,9 @@ struct zthr {
*/ */
hrtime_t zthr_sleep_timeout; hrtime_t zthr_sleep_timeout;
/* Thread priority */
pri_t zthr_pri;
/* consumer-provided callbacks & data */ /* consumer-provided callbacks & data */
zthr_checkfunc_t *zthr_checkfunc; zthr_checkfunc_t *zthr_checkfunc;
zthr_func_t *zthr_func; zthr_func_t *zthr_func;
@ -269,10 +273,10 @@ zthr_procedure(void *arg)
zthr_t * zthr_t *
zthr_create(const char *zthr_name, zthr_checkfunc_t *checkfunc, zthr_create(const char *zthr_name, zthr_checkfunc_t *checkfunc,
zthr_func_t *func, void *arg) zthr_func_t *func, void *arg, pri_t pri)
{ {
return (zthr_create_timer(zthr_name, checkfunc, return (zthr_create_timer(zthr_name, checkfunc,
func, arg, (hrtime_t)0)); func, arg, (hrtime_t)0, pri));
} }
/* /*
@ -282,7 +286,7 @@ zthr_create(const char *zthr_name, zthr_checkfunc_t *checkfunc,
*/ */
zthr_t * zthr_t *
zthr_create_timer(const char *zthr_name, zthr_checkfunc_t *checkfunc, zthr_create_timer(const char *zthr_name, zthr_checkfunc_t *checkfunc,
zthr_func_t *func, void *arg, hrtime_t max_sleep) zthr_func_t *func, void *arg, hrtime_t max_sleep, pri_t pri)
{ {
zthr_t *t = kmem_zalloc(sizeof (*t), KM_SLEEP); zthr_t *t = kmem_zalloc(sizeof (*t), KM_SLEEP);
mutex_init(&t->zthr_state_lock, NULL, MUTEX_DEFAULT, NULL); mutex_init(&t->zthr_state_lock, NULL, MUTEX_DEFAULT, NULL);
@ -296,9 +300,10 @@ zthr_create_timer(const char *zthr_name, zthr_checkfunc_t *checkfunc,
t->zthr_arg = arg; t->zthr_arg = arg;
t->zthr_sleep_timeout = max_sleep; t->zthr_sleep_timeout = max_sleep;
t->zthr_name = zthr_name; t->zthr_name = zthr_name;
t->zthr_pri = pri;
t->zthr_thread = thread_create_named(zthr_name, NULL, 0, t->zthr_thread = thread_create_named(zthr_name, NULL, 0,
zthr_procedure, t, 0, &p0, TS_RUN, minclsyspri); zthr_procedure, t, 0, &p0, TS_RUN, pri);
mutex_exit(&t->zthr_state_lock); mutex_exit(&t->zthr_state_lock);
@ -423,7 +428,7 @@ zthr_resume(zthr_t *t)
*/ */
if (t->zthr_thread == NULL) { if (t->zthr_thread == NULL) {
t->zthr_thread = thread_create_named(t->zthr_name, NULL, 0, t->zthr_thread = thread_create_named(t->zthr_name, NULL, 0,
zthr_procedure, t, 0, &p0, TS_RUN, minclsyspri); zthr_procedure, t, 0, &p0, TS_RUN, t->zthr_pri);
} }
mutex_exit(&t->zthr_state_lock); mutex_exit(&t->zthr_state_lock);