Files
linux-apfs/kernel/kthread.c
T

250 lines
6.6 KiB
C
Raw Normal View History

2005-04-16 15:20:36 -07:00
/* Kernel thread helper functions.
* Copyright (C) 2004 IBM Corporation, Rusty Russell.
*
2007-05-09 02:34:32 -07:00
* Creation is done via kthreadd, so that we get a clean environment
2005-04-16 15:20:36 -07:00
* even if we're invoked from userspace (think modprobe, hotplug cpu,
* etc.).
*/
#include <linux/sched.h>
#include <linux/kthread.h>
#include <linux/completion.h>
#include <linux/err.h>
#include <linux/cpuset.h>
2005-04-16 15:20:36 -07:00
#include <linux/unistd.h>
#include <linux/file.h>
#include <linux/module.h>
2006-03-23 03:00:24 -08:00
#include <linux/mutex.h>
#include <trace/events/sched.h>
2005-04-16 15:20:36 -07:00
2007-05-09 02:34:32 -07:00
static DEFINE_SPINLOCK(kthread_create_lock);
static LIST_HEAD(kthread_create_list);
struct task_struct *kthreadd_task;
2005-04-16 15:20:36 -07:00
struct kthread_create_info
{
2007-05-09 02:34:32 -07:00
/* Information passed to kthread() from kthreadd. */
2005-04-16 15:20:36 -07:00
int (*threadfn)(void *data);
void *data;
2007-05-09 02:34:32 -07:00
/* Result passed back to kthread_create() from kthreadd. */
2005-04-16 15:20:36 -07:00
struct task_struct *result;
struct completion done;
2007-05-09 02:34:32 -07:00
struct list_head list;
2005-04-16 15:20:36 -07:00
};
2009-06-17 16:27:45 -07:00
struct kthread {
int should_stop;
struct completion exited;
2005-04-16 15:20:36 -07:00
};
2009-06-17 16:27:45 -07:00
#define to_kthread(tsk) \
container_of((tsk)->vfork_done, struct kthread, exited)
2005-04-16 15:20:36 -07:00
/**
* kthread_should_stop - should this kthread return now?
*
* When someone calls kthread_stop() on your kthread, it will be woken
* and this will return true. You should then return, and your return
* value will be passed through to kthread_stop().
*/
2005-04-16 15:20:36 -07:00
int kthread_should_stop(void)
{
2009-06-17 16:27:45 -07:00
return to_kthread(current)->should_stop;
2005-04-16 15:20:36 -07:00
}
EXPORT_SYMBOL(kthread_should_stop);
static int kthread(void *_create)
{
2007-05-09 02:34:32 -07:00
/* Copy data: it's on kthread's stack */
2009-06-17 16:27:45 -07:00
struct kthread_create_info *create = _create;
int (*threadfn)(void *data) = create->threadfn;
void *data = create->data;
struct kthread self;
int ret;
self.should_stop = 0;
init_completion(&self.exited);
current->vfork_done = &self.exited;
2005-04-16 15:20:36 -07:00
/* OK, tell user we're spawned, wait for stop or wakeup */
__set_current_state(TASK_UNINTERRUPTIBLE);
create->result = current;
complete(&create->done);
2005-04-16 15:20:36 -07:00
schedule();
2009-06-17 16:27:45 -07:00
ret = -EINTR;
if (!self.should_stop)
2005-04-16 15:20:36 -07:00
ret = threadfn(data);
2009-06-17 16:27:45 -07:00
/* we can't just return, we must preserve "self" on stack */
do_exit(ret);
2005-04-16 15:20:36 -07:00
}
2007-05-09 02:34:32 -07:00
static void create_kthread(struct kthread_create_info *create)
2005-04-16 15:20:36 -07:00
{
int pid;
/* We want our own signal handler (we take no signals by default). */
pid = kernel_thread(kthread, create, CLONE_FS | CLONE_FILES | SIGCHLD);
if (pid < 0) {
2005-04-16 15:20:36 -07:00
create->result = ERR_PTR(pid);
complete(&create->done);
}
2005-04-16 15:20:36 -07:00
}
/**
* kthread_create - create a kthread.
* @threadfn: the function to run until signal_pending(current).
* @data: data ptr for @threadfn.
* @namefmt: printf-style name for the thread.
*
* Description: This helper function creates and names a kernel
* thread. The thread will be stopped: use wake_up_process() to start
* it. See also kthread_run().
*
* When woken, the thread will run @threadfn() with @data as its
* argument. @threadfn() can either call do_exit() directly if it is a
* standalone thread for which noone will call kthread_stop(), or
* return when 'kthread_should_stop()' is true (which means
* kthread_stop() has been called). The return value should be zero
* or a negative error number; it will be passed to kthread_stop().
*
* Returns a task_struct or ERR_PTR(-ENOMEM).
*/
2005-04-16 15:20:36 -07:00
struct task_struct *kthread_create(int (*threadfn)(void *data),
void *data,
const char namefmt[],
...)
{
struct kthread_create_info create;
create.threadfn = threadfn;
create.data = data;
init_completion(&create.done);
2007-05-09 02:34:32 -07:00
spin_lock(&kthread_create_lock);
list_add_tail(&create.list, &kthread_create_list);
spin_unlock(&kthread_create_lock);
wake_up_process(kthreadd_task);
2007-05-09 02:34:32 -07:00
wait_for_completion(&create.done);
2005-04-16 15:20:36 -07:00
if (!IS_ERR(create.result)) {
struct sched_param param = { .sched_priority = 0 };
2005-04-16 15:20:36 -07:00
va_list args;
2005-04-16 15:20:36 -07:00
va_start(args, namefmt);
vsnprintf(create.result->comm, sizeof(create.result->comm),
namefmt, args);
va_end(args);
/*
* root may have changed our (kthreadd's) priority or CPU mask.
* The kernel thread should not inherit these properties.
*/
sched_setscheduler_nocheck(create.result, SCHED_NORMAL, &param);
set_cpus_allowed_ptr(create.result, cpu_all_mask);
2005-04-16 15:20:36 -07:00
}
return create.result;
}
EXPORT_SYMBOL(kthread_create);
/**
* kthread_bind - bind a just-created kthread to a cpu.
* @p: thread created by kthread_create().
* @cpu: cpu (might not be online, must be possible) for @k to run on.
*
* Description: This function is equivalent to set_cpus_allowed(),
* except that @cpu doesn't need to be online, and the thread must be
* stopped (i.e., just returned from kthread_create()).
*/
void kthread_bind(struct task_struct *p, unsigned int cpu)
{
/* Must have done schedule() in kthread() before we set_task_cpu */
if (!wait_task_inactive(p, TASK_UNINTERRUPTIBLE)) {
WARN_ON(1);
return;
}
p->cpus_allowed = cpumask_of_cpu(cpu);
p->rt.nr_cpus_allowed = 1;
p->flags |= PF_THREAD_BOUND;
}
EXPORT_SYMBOL(kthread_bind);
/**
* kthread_stop - stop a thread created by kthread_create().
* @k: thread created by kthread_create().
*
* Sets kthread_should_stop() for @k to return true, wakes it, and
2009-06-19 02:51:13 +02:00
* waits for it to exit. This can also be called after kthread_create()
* instead of calling wake_up_process(): the thread will exit without
* calling threadfn().
*
* If threadfn() may call do_exit() itself, the caller must ensure
* task_struct can't go away.
*
* Returns the result of threadfn(), or %-EINTR if wake_up_process()
* was never called.
*/
2005-04-16 15:20:36 -07:00
int kthread_stop(struct task_struct *k)
{
2009-06-17 16:27:45 -07:00
struct kthread *kthread;
2005-04-16 15:20:36 -07:00
int ret;
2009-06-17 16:27:45 -07:00
trace_sched_kthread_stop(k);
2005-04-16 15:20:36 -07:00
get_task_struct(k);
2009-06-17 16:27:45 -07:00
kthread = to_kthread(k);
barrier(); /* it might have exited */
if (k->vfork_done != NULL) {
kthread->should_stop = 1;
wake_up_process(k);
wait_for_completion(&kthread->exited);
}
ret = k->exit_code;
2005-04-16 15:20:36 -07:00
put_task_struct(k);
trace_sched_kthread_stop_ret(ret);
2005-04-16 15:20:36 -07:00
return ret;
}
EXPORT_SYMBOL(kthread_stop);
2005-04-16 15:20:36 -07:00
int kthreadd(void *unused)
2005-04-16 15:20:36 -07:00
{
2007-05-09 02:34:32 -07:00
struct task_struct *tsk = current;
/* Setup a clean context for our children to inherit. */
2007-05-09 02:34:32 -07:00
set_task_comm(tsk, "kthreadd");
ignore_signals(tsk);
set_cpus_allowed_ptr(tsk, cpu_all_mask);
set_mems_allowed(node_possible_map);
2007-05-09 02:34:32 -07:00
2008-06-11 22:04:29 +02:00
current->flags |= PF_NOFREEZE | PF_FREEZER_NOSIG;
2007-05-09 02:34:32 -07:00
for (;;) {
set_current_state(TASK_INTERRUPTIBLE);
if (list_empty(&kthread_create_list))
schedule();
__set_current_state(TASK_RUNNING);
spin_lock(&kthread_create_lock);
while (!list_empty(&kthread_create_list)) {
struct kthread_create_info *create;
create = list_entry(kthread_create_list.next,
struct kthread_create_info, list);
list_del_init(&create->list);
spin_unlock(&kthread_create_lock);
create_kthread(create);
spin_lock(&kthread_create_lock);
}
spin_unlock(&kthread_create_lock);
}
2005-04-16 15:20:36 -07:00
return 0;
}