Received: by 2002:a25:1985:0:0:0:0:0 with SMTP id 127csp818382ybz; Fri, 1 May 2020 09:05:38 -0700 (PDT) X-Google-Smtp-Source: APiQypIocZGozQn2JjDe1YlgWDz0+ScKZ5bsqJfEwTYK1HrvtMz0rPw8t47DTcTfrhlCsBGCMjJA X-Received: by 2002:a17:906:2988:: with SMTP id x8mr3806968eje.16.1588349137974; Fri, 01 May 2020 09:05:37 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1588349137; cv=none; d=google.com; s=arc-20160816; b=N4zUgzsRSnGpgh3U3lXCiLUGAA9wAUFvsUddDuiiF3aXLOaDX7bQ9L6FPJU4DzZdxi CDgGD+XSsF0+XyQE9Yosjqlyd2SceGs3/363bkndlJhvM6o0n8wurYLU1sw5pt/Ro5Up Z87/hzA6A5nlR0BG+9R2nItIY7KEwuf7NgDD3ICEo/qY/7KTJbuY+IaFdRJeSP3agb51 P6YUD1uJMe/04bF0vewnNU05qSYOC0hPmSZOCNKpltuj07vvy6AveBHrAkf3p3VUw0Ma TPGFmOsFHafgxtaosgEHvDPhFWgqYVt5sGMu8DOmXpNpfJbIStYfCEmoG3oe2/AzVO2n GLVQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:references:in-reply-to:message-id:date :subject:cc:to:from; bh=8SdK4Qf65q5YPfW4POxIS0h0q/XHj5vTpycLpcyLMcY=; b=eS3+MmFmtBNKOHq9uq+i7chqCFlaAgXbuJKX3ueKxVVR/XJtvZrVwUBEVqJ0df6qWF +NdWT+Dnas3dM31CemMZ/BPk25UIpKTbTZ19eO4TRP/9DTT4+8JatQTmKRD7mQz+I6ja J7H+1tB+q6Vj6Fe2G3VYT+uBCH8cJ1R5CESXoUAwQB8Ago1m53sWZxZGayZObaw+j3du ZKH34oZ6GStSnzKm00agSGhXV/GwUcgKNQ03YwKYkP9L6XuntB9dULHZ45en8XypqDGb xdRNDEQp5sp/AsEfa4+zOUH7mIwKEptL1o1qTgm2jRa1KuCGXrkCHT8PIroLP6cTPgcE 5uTA== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=redhat.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id i14si2118072eja.57.2020.05.01.09.05.14; Fri, 01 May 2020 09:05:37 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=redhat.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1730130AbgEAQCH (ORCPT + 99 others); Fri, 1 May 2020 12:02:07 -0400 Received: from fieldses.org ([173.255.197.46]:40344 "EHLO fieldses.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1729859AbgEAQBy (ORCPT ); Fri, 1 May 2020 12:01:54 -0400 Received: by fieldses.org (Postfix, from userid 2815) id F159D5F28; Fri, 1 May 2020 12:01:53 -0400 (EDT) From: "J. Bruce Fields" To: Linus Torvalds Cc: linux-nfs@vger.kernel.org, Jeff Layton , David Howells , Tejun Heo , Shaohua Li , Oleg Nesterov , linux-kernel@vger.kernel.org, "J. Bruce Fields" Subject: [PATCH 3/4] kthreads: allow multiple kthreadd's Date: Fri, 1 May 2020 12:01:51 -0400 Message-Id: <1588348912-24781-4-git-send-email-bfields@redhat.com> X-Mailer: git-send-email 1.8.3.1 In-Reply-To: <1588348912-24781-1-git-send-email-bfields@redhat.com> References: <1588348912-24781-1-git-send-email-bfields@redhat.com> Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: "J. Bruce Fields" Allow subsystems to run their own kthreadd's. I'm experimenting with this to allow nfsd to put its threads into its own thread group to make it easy for the vfs to tell when nfsd is breaking one of its own leases. Signed-off-by: J. Bruce Fields --- include/linux/kthread.h | 20 ++++++- init/main.c | 4 +- kernel/kthread.c | 113 ++++++++++++++++++++++++++++++---------- 3 files changed, 107 insertions(+), 30 deletions(-) diff --git a/include/linux/kthread.h b/include/linux/kthread.h index 8bbcaad7ef0f..a7ffdf96a3b2 100644 --- a/include/linux/kthread.h +++ b/include/linux/kthread.h @@ -5,6 +5,24 @@ #include #include +struct kthread_group { + char *name; + spinlock_t create_lock; + struct list_head create_list; + struct task_struct *task; +}; + +extern struct kthread_group kthreadd_default; + +struct kthread_group *kthread_start_group(char *); +void kthread_stop_group(struct kthread_group *); + +struct task_struct *kthread_group_create_on_node(struct kthread_group *, + int (*threadfn)(void *data), + void *data, + int node, + const char namefmt[], ...); + __printf(4, 5) struct task_struct *kthread_create_on_node(int (*threadfn)(void *data), void *data, @@ -63,7 +81,7 @@ int kthread_park(struct task_struct *k); void kthread_unpark(struct task_struct *k); void kthread_parkme(void); -int kthreadd(void *unused); +int kthreadd(void *); extern struct task_struct *kthreadd_task; extern int tsk_fork_get_node(struct task_struct *tsk); diff --git a/init/main.c b/init/main.c index a48617f2e5e5..5256071b0e05 100644 --- a/init/main.c +++ b/init/main.c @@ -641,9 +641,9 @@ noinline void __ref rest_init(void) rcu_read_unlock(); numa_default_policy(); - pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES); + pid = kernel_thread(kthreadd, &kthreadd_default, CLONE_FS | CLONE_FILES); rcu_read_lock(); - kthreadd_task = find_task_by_pid_ns(pid, &init_pid_ns); + kthreadd_default.task = find_task_by_pid_ns(pid, &init_pid_ns); rcu_read_unlock(); /* diff --git a/kernel/kthread.c b/kernel/kthread.c index 483bee57a9c8..5232f6f597b5 100644 --- a/kernel/kthread.c +++ b/kernel/kthread.c @@ -25,9 +25,44 @@ #include #include -static DEFINE_SPINLOCK(kthread_create_lock); -static LIST_HEAD(kthread_create_list); -struct task_struct *kthreadd_task; +struct kthread_group kthreadd_default = { + .name = "kthreadd", + .create_lock = __SPIN_LOCK_UNLOCKED(kthreadd_default.create_lock), + .create_list = LIST_HEAD_INIT(kthreadd_default.create_list), +}; + +void wake_kthreadd(struct kthread_group *kg) +{ + wake_up_process(kg->task); +} + +struct kthread_group *kthread_start_group(char *name) +{ + struct kthread_group *new; + struct task_struct *task; + + new = kmalloc(sizeof(struct kthread_group), GFP_KERNEL); + if (!new) + return ERR_PTR(-ENOMEM); + spin_lock_init(&new->create_lock); + INIT_LIST_HEAD(&new->create_list); + new->name = name; + task = kthread_run(kthreadd, new, name); + if (IS_ERR(task)) { + kfree(new); + return ERR_CAST(task); + } + new->task = task; + return new; +} +EXPORT_SYMBOL_GPL(kthread_start_group); + +void kthread_stop_group(struct kthread_group *kg) +{ + kthread_stop(kg->task); + kfree(kg); +} +EXPORT_SYMBOL_GPL(kthread_stop_group); struct kthread_create_info { @@ -301,11 +336,13 @@ static void create_kthread(struct kthread_create_info *create) } } -static __printf(4, 0) -struct task_struct *__kthread_create_on_node(int (*threadfn)(void *data), - void *data, int node, - const char namefmt[], - va_list args) + +static __printf(5, 0) +struct task_struct *__kthread_group_create_on_node(struct kthread_group *kg, + int (*threadfn)(void *data), + void *data, int node, + const char namefmt[], + va_list args) { DECLARE_COMPLETION_ONSTACK(done); struct task_struct *task; @@ -319,11 +356,11 @@ struct task_struct *__kthread_create_on_node(int (*threadfn)(void *data), create->node = node; create->done = &done; - spin_lock(&kthread_create_lock); - list_add_tail(&create->list, &kthread_create_list); - spin_unlock(&kthread_create_lock); + spin_lock(&kg->create_lock); + list_add_tail(&create->list, &kg->create_list); + spin_unlock(&kg->create_lock); - wake_up_process(kthreadd_task); + wake_kthreadd(kg); /* * Wait for completion in killable state, for I might be chosen by * the OOM killer while kthreadd is trying to allocate memory for @@ -365,6 +402,25 @@ struct task_struct *__kthread_create_on_node(int (*threadfn)(void *data), return task; } +__printf(5, 0) +struct task_struct *kthread_group_create_on_node(struct kthread_group *kg, + int (*threadfn)(void *data), + void *data, int node, + const char namefmt[], + ...) +{ + struct task_struct *task; + va_list args; + + va_start(args, namefmt); + task = __kthread_group_create_on_node(kg, threadfn, + data, node, namefmt, args); + va_end(args); + + return task; +} +EXPORT_SYMBOL_GPL(kthread_group_create_on_node); + /** * kthread_create_on_node - create a kthread. * @threadfn: the function to run until signal_pending(current). @@ -397,7 +453,8 @@ struct task_struct *kthread_create_on_node(int (*threadfn)(void *data), va_list args; va_start(args, namefmt); - task = __kthread_create_on_node(threadfn, data, node, namefmt, args); + task = __kthread_group_create_on_node(&kthreadd_default, threadfn, + data, node, namefmt, args); va_end(args); return task; @@ -577,30 +634,31 @@ int kthread_stop(struct task_struct *k) } EXPORT_SYMBOL(kthread_stop); -void kthread_do_work(void) +void kthread_do_work(struct kthread_group *kg) { - spin_lock(&kthread_create_lock); - while (!list_empty(&kthread_create_list)) { + spin_lock(&kg->create_lock); + while (!list_empty(&kg->create_list)) { struct kthread_create_info *create; - create = list_entry(kthread_create_list.next, + create = list_entry(kg->create_list.next, struct kthread_create_info, list); list_del_init(&create->list); - spin_unlock(&kthread_create_lock); + spin_unlock(&kg->create_lock); create_kthread(create); - spin_lock(&kthread_create_lock); + spin_lock(&kg->create_lock); } - spin_unlock(&kthread_create_lock); + spin_unlock(&kg->create_lock); } -int kthreadd(void *unused) +int kthreadd(void *data) { + struct kthread_group *kg = data; struct task_struct *tsk = current; /* Setup a clean context for our children to inherit. */ - set_task_comm(tsk, "kthreadd"); + set_task_comm(tsk, kg->name); ignore_signals(tsk); set_cpus_allowed_ptr(tsk, cpu_all_mask); set_mems_allowed(node_states[N_MEMORY]); @@ -608,13 +666,13 @@ int kthreadd(void *unused) current->flags |= PF_NOFREEZE; cgroup_init_kthreadd(); - for (;;) { + while (current == kthreadd_default.task || !kthread_should_stop()) { set_current_state(TASK_INTERRUPTIBLE); - if (list_empty(&kthread_create_list)) + if (list_empty(&kg->create_list)) schedule(); __set_current_state(TASK_RUNNING); - kthread_do_work(); + kthread_do_work(kg); } return 0; @@ -712,8 +770,9 @@ __kthread_create_worker(int cpu, unsigned int flags, if (cpu >= 0) node = cpu_to_node(cpu); - task = __kthread_create_on_node(kthread_worker_fn, worker, - node, namefmt, args); + task = __kthread_group_create_on_node(&kthreadd_default, + kthread_worker_fn, + worker, node, namefmt, args); if (IS_ERR(task)) goto fail_task; -- 2.26.2