cregit-Linux how code gets into the kernel

Release 4.7 kernel/kthread.c

Directory: kernel
/* Kernel thread helper functions.
 *   Copyright (C) 2004 IBM Corporation, Rusty Russell.
 *
 * Creation is done via kthreadd, so that we get a clean environment
 * 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>
#include <linux/unistd.h>
#include <linux/file.h>
#include <linux/export.h>
#include <linux/mutex.h>
#include <linux/slab.h>
#include <linux/freezer.h>
#include <linux/ptrace.h>
#include <linux/uaccess.h>
#include <trace/events/sched.h>

static DEFINE_SPINLOCK(kthread_create_lock);
static LIST_HEAD(kthread_create_list);

struct task_struct *kthreadd_task;


struct kthread_create_info
{
	/* Information passed to kthread() from kthreadd. */
	
int (*threadfn)(void *data);
	
void *data;
	
int node;

	/* Result passed back to kthread_create() from kthreadd. */
	
struct task_struct *result;
	
struct completion *done;

	
struct list_head list;
};


struct kthread {
	
unsigned long flags;
	
unsigned int cpu;
	
void *data;
	
struct completion parked;
	
struct completion exited;
};


enum KTHREAD_BITS {
	
KTHREAD_IS_PER_CPU = 0,
	
KTHREAD_SHOULD_STOP,
	
KTHREAD_SHOULD_PARK,
	
KTHREAD_IS_PARKED,
};


#define __to_kthread(vfork)	\
	container_of(vfork, struct kthread, exited)


static inline struct kthread *to_kthread(struct task_struct *k) { return __to_kthread(k->vfork_done); }

Contributors

PersonTokensPropCommitsCommitProp
oleg nesterovoleg nesterov22100.00%1100.00%
Total22100.00%1100.00%


static struct kthread *to_live_kthread(struct task_struct *k) { struct completion *vfork = ACCESS_ONCE(k->vfork_done); if (likely(vfork)) return __to_kthread(vfork); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
oleg nesterovoleg nesterov41100.00%1100.00%
Total41100.00%1100.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(). */
bool kthread_should_stop(void) { return test_bit(KTHREAD_SHOULD_STOP, &to_kthread(current)->flags); }

Contributors

PersonTokensPropCommitsCommitProp
andrew mortonandrew morton942.86%133.33%
thomas gleixnerthomas gleixner838.10%133.33%
oleg nesterovoleg nesterov419.05%133.33%
Total21100.00%3100.00%

EXPORT_SYMBOL(kthread_should_stop); /** * kthread_should_park - should this kthread park now? * * When someone calls kthread_park() on your kthread, it will be woken * and this will return true. You should then do the necessary * cleanup and call kthread_parkme() * * Similar to kthread_should_stop(), but this keeps the thread alive * and in a park position. kthread_unpark() "restarts" the thread and * calls the thread function again. */
bool kthread_should_park(void) { return test_bit(KTHREAD_SHOULD_PARK, &to_kthread(current)->flags); }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner21100.00%1100.00%
Total21100.00%1100.00%

EXPORT_SYMBOL_GPL(kthread_should_park); /** * kthread_freezable_should_stop - should this freezable kthread return now? * @was_frozen: optional out parameter, indicates whether %current was frozen * * kthread_should_stop() for freezable kthreads, which will enter * refrigerator if necessary. This function is safe from kthread_stop() / * freezer deadlock and freezable kthreads should use this function instead * of calling try_to_freeze() directly. */
bool kthread_freezable_should_stop(bool *was_frozen) { bool frozen = false; might_sleep(); if (unlikely(freezing(current))) frozen = __refrigerator(true); if (was_frozen) *was_frozen = frozen; return kthread_should_stop(); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo47100.00%1100.00%
Total47100.00%1100.00%

EXPORT_SYMBOL_GPL(kthread_freezable_should_stop); /** * kthread_data - return data value specified on kthread creation * @task: kthread task in question * * Return the data value specified when kthread @task was created. * The caller is responsible for ensuring the validity of @task when * calling this function. */
void *kthread_data(struct task_struct *task) { return to_kthread(task)->data; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo19100.00%1100.00%
Total19100.00%1100.00%

/** * probe_kthread_data - speculative version of kthread_data() * @task: possible kthread task in question * * @task could be a kthread task. Return the data value specified when it * was created if accessible. If @task isn't a kthread task or its data is * inaccessible for any reason, %NULL is returned. This function requires * that @task itself is safe to dereference. */
void *probe_kthread_data(struct task_struct *task) { struct kthread *kthread = to_kthread(task); void *data = NULL; probe_kernel_read(&data, &kthread->data, sizeof(data)); return data; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo46100.00%1100.00%
Total46100.00%1100.00%


static void __kthread_parkme(struct kthread *self) { __set_current_state(TASK_PARKED); while (test_bit(KTHREAD_SHOULD_PARK, &self->flags)) { if (!test_and_set_bit(KTHREAD_IS_PARKED, &self->flags)) complete(&self->parked); schedule(); __set_current_state(TASK_PARKED); } clear_bit(KTHREAD_IS_PARKED, &self->flags); __set_current_state(TASK_RUNNING); }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner74100.00%2100.00%
Total74100.00%2100.00%


void kthread_parkme(void) { __kthread_parkme(to_kthread(current)); }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner15100.00%1100.00%
Total15100.00%1100.00%

EXPORT_SYMBOL_GPL(kthread_parkme);
static int kthread(void *_create) { /* Copy data: it's on kthread's stack */ struct kthread_create_info *create = _create; int (*threadfn)(void *data) = create->threadfn; void *data = create->data; struct completion *done; struct kthread self; int ret; self.flags = 0; self.data = data; init_completion(&self.exited); init_completion(&self.parked); current->vfork_done = &self.exited; /* If user was SIGKILLed, I release the structure. */ done = xchg(&create->done, NULL); if (!done) { kfree(create); do_exit(-EINTR); } /* OK, tell user we're spawned, wait for stop or wakeup */ __set_current_state(TASK_UNINTERRUPTIBLE); create->result = current; complete(done); schedule(); ret = -EINTR; if (!test_bit(KTHREAD_SHOULD_STOP, &self.flags)) { __kthread_parkme(&self); ret = threadfn(data); } /* we can't just return, we must preserve "self" on stack */ do_exit(ret); }

Contributors

PersonTokensPropCommitsCommitProp
andrew mortonandrew morton6536.11%112.50%
oleg nesterovoleg nesterov4323.89%337.50%
tetsuo handatetsuo handa3620.00%112.50%
thomas gleixnerthomas gleixner2413.33%112.50%
tejun heotejun heo63.33%112.50%
vitaliy gusevvitaliy gusev63.33%112.50%
Total180100.00%8100.00%

/* called from do_fork() to get node information for about to be created task */
int tsk_fork_get_node(struct task_struct *tsk) { #ifdef CONFIG_NUMA if (tsk == kthreadd_task) return tsk->pref_node_fork; #endif return NUMA_NO_NODE; }

Contributors

PersonTokensPropCommitsCommitProp
eric dumazeteric dumazet2896.55%150.00%
nishanth aravamudannishanth aravamudan13.45%150.00%
Total29100.00%2100.00%


static void create_kthread(struct kthread_create_info *create) { int pid; #ifdef CONFIG_NUMA current->pref_node_fork = create->node; #endif /* 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) { /* If user was SIGKILLed, I release the structure. */ struct completion *done = xchg(&create->done, NULL); if (!done) { kfree(create); return; } create->result = ERR_PTR(pid); complete(done); } }

Contributors

PersonTokensPropCommitsCommitProp
andrew mortonandrew morton4648.94%116.67%
tetsuo handatetsuo handa2930.85%116.67%
eric dumazeteric dumazet1313.83%116.67%
eric w. biedermaneric w. biederman33.19%116.67%
oleg nesterovoleg nesterov22.13%116.67%
david howellsdavid howells11.06%116.67%
Total94100.00%6100.00%

/** * kthread_create_on_node - create a kthread. * @threadfn: the function to run until signal_pending(current). * @data: data ptr for @threadfn. * @node: task and thread structures for the thread are allocated on this node * @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(). The new thread has SCHED_NORMAL policy and * is affine to all CPUs. * * If thread is going to be bound on a particular cpu, give its node * in @node, to get NUMA affinity for kthread stack, or else give NUMA_NO_NODE. * 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 no one 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) or ERR_PTR(-EINTR). */
struct task_struct *kthread_create_on_node(int (*threadfn)(void *data), void *data, int node, const char namefmt[], ...) { DECLARE_COMPLETION_ONSTACK(done); struct task_struct *task; struct kthread_create_info *create = kmalloc(sizeof(*create), GFP_KERNEL); if (!create) return ERR_PTR(-ENOMEM); create->threadfn = threadfn; create->data = 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); wake_up_process(kthreadd_task); /* * Wait for completion in killable state, for I might be chosen by * the OOM killer while kthreadd is trying to allocate memory for * new kernel thread. */ if (unlikely(wait_for_completion_killable(&done))) { /* * If I was SIGKILLed before kthreadd (or new kernel thread) * calls complete(), leave the cleanup of this structure to * that thread. */ if (xchg(&create->done, NULL)) return ERR_PTR(-EINTR); /* * kthreadd (or new kernel thread) will call complete() * shortly. */ wait_for_completion(&done); } task = create->result; if (!IS_ERR(task)) { static const struct sched_param param = { .sched_priority = 0 }; va_list args; va_start(args, namefmt); vsnprintf(task->comm, sizeof(task->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(task, SCHED_NORMAL, &param); set_cpus_allowed_ptr(task, cpu_all_mask); } kfree(create); return task; }

Contributors

PersonTokensPropCommitsCommitProp
andrew mortonandrew morton10039.68%110.00%
tetsuo handatetsuo handa9437.30%220.00%
oleg nesterovoleg nesterov2710.71%110.00%
eric dumazeteric dumazet93.57%110.00%
eric w. biedermaneric w. biederman93.57%110.00%
david howellsdavid howells62.38%110.00%
dmitry adamushkodmitry adamushko51.98%110.00%
kosaki motohirokosaki motohiro10.40%110.00%
peter zijlstrapeter zijlstra10.40%110.00%
Total252100.00%10100.00%

EXPORT_SYMBOL(kthread_create_on_node);
static void __kthread_bind_mask(struct task_struct *p, const struct cpumask *mask, long state) { unsigned long flags; if (!wait_task_inactive(p, state)) { WARN_ON(1); return; } /* It's safe because the task is inactive. */ raw_spin_lock_irqsave(&p->pi_lock, flags); do_set_cpus_allowed(p, mask); p->flags |= PF_NO_SETAFFINITY; raw_spin_unlock_irqrestore(&p->pi_lock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner4356.58%250.00%
peter zijlstrapeter zijlstra3242.11%125.00%
tejun heotejun heo11.32%125.00%
Total76100.00%4100.00%


static void __kthread_bind(struct task_struct *p, unsigned int cpu, long state) { __kthread_bind_mask(p, cpumask_of(cpu), state); }

Contributors

PersonTokensPropCommitsCommitProp
peter zijlstrapeter zijlstra30100.00%1100.00%
Total30100.00%1100.00%


void kthread_bind_mask(struct task_struct *p, const struct cpumask *mask) { __kthread_bind_mask(p, mask, TASK_UNINTERRUPTIBLE); }

Contributors

PersonTokensPropCommitsCommitProp
peter zijlstrapeter zijlstra2496.00%150.00%
thomas gleixnerthomas gleixner14.00%150.00%
Total25100.00%2100.00%

/** * 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) { __kthread_bind(p, cpu, TASK_UNINTERRUPTIBLE); }

Contributors

PersonTokensPropCommitsCommitProp
peter zijlstrapeter zijlstra1878.26%125.00%
thomas gleixnerthomas gleixner313.04%250.00%
kosaki motohirokosaki motohiro28.70%125.00%
Total23100.00%4100.00%

EXPORT_SYMBOL(kthread_bind); /** * kthread_create_on_cpu - Create a cpu bound kthread * @threadfn: the function to run until signal_pending(current). * @data: data ptr for @threadfn. * @cpu: The cpu on which the thread should be bound, * @namefmt: printf-style name for the thread. Format is restricted * to "name.*%u". Code fills in cpu number. * * Description: This helper function creates and names a kernel thread * The thread will be woken and put into park mode. */
struct task_struct *kthread_create_on_cpu(int (*threadfn)(void *data), void *data, unsigned int cpu, const char *namefmt) { struct task_struct *p; p = kthread_create_on_node(threadfn, data, cpu_to_node(cpu), namefmt, cpu); if (IS_ERR(p)) return p; set_bit(KTHREAD_IS_PER_CPU, &to_kthread(p)->flags); to_kthread(p)->cpu = cpu; /* Park the thread to get it out of TASK_UNINTERRUPTIBLE state */ kthread_park(p); return p; }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner9397.89%133.33%
andrew mortonandrew morton11.05%133.33%
nishanth aravamudannishanth aravamudan11.05%133.33%
Total95100.00%3100.00%


static void __kthread_unpark(struct task_struct *k, struct kthread *kthread) { clear_bit(KTHREAD_SHOULD_PARK, &kthread->flags); /* * We clear the IS_PARKED bit here as we don't wait * until the task has left the park code. So if we'd * park before that happens we'd see the IS_PARKED bit * which might be about to be cleared. */ if (test_and_clear_bit(KTHREAD_IS_PARKED, &kthread->flags)) { if (test_bit(KTHREAD_IS_PER_CPU, &kthread->flags)) __kthread_bind(k, kthread->cpu, TASK_PARKED); wake_up_state(k, TASK_PARKED); } }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner6997.18%266.67%
oleg nesterovoleg nesterov22.82%133.33%
Total71100.00%3100.00%

/** * kthread_unpark - unpark a thread created by kthread_create(). * @k: thread created by kthread_create(). * * Sets kthread_should_park() for @k to return false, wakes it, and * waits for it to return. If the thread is marked percpu then its * bound to the cpu again. */
void kthread_unpark(struct task_struct *k) { struct kthread *kthread = to_live_kthread(k); if (kthread) __kthread_unpark(k, kthread); }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner3096.77%266.67%
oleg nesterovoleg nesterov13.23%133.33%
Total31100.00%3100.00%

EXPORT_SYMBOL_GPL(kthread_unpark); /** * kthread_park - park a thread created by kthread_create(). * @k: thread created by kthread_create(). * * Sets kthread_should_park() for @k to return true, wakes it, and * waits for it to return. This can also be called after kthread_create() * instead of calling wake_up_process(): the thread will park without * calling threadfn(). * * Returns 0 if the thread is parked, -ENOSYS if the thread exited. * If called by the kthread itself just the park bit is set. */
int kthread_park(struct task_struct *k) { struct kthread *kthread = to_live_kthread(k); int ret = -ENOSYS; if (kthread) { if (!test_bit(KTHREAD_IS_PARKED, &kthread->flags)) { set_bit(KTHREAD_SHOULD_PARK, &kthread->flags); if (k != current) { wake_up_process(k); wait_for_completion(&kthread->parked); } } ret = 0; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner8397.65%133.33%
oleg nesterovoleg nesterov11.18%133.33%
andrew mortonandrew morton11.18%133.33%
Total85100.00%3100.00%

EXPORT_SYMBOL_GPL(kthread_park); /** * 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 * 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. */
int kthread_stop(struct task_struct *k) { struct kthread *kthread; int ret; trace_sched_kthread_stop(k); get_task_struct(k); kthread = to_live_kthread(k); if (kthread) { set_bit(KTHREAD_SHOULD_STOP, &kthread->flags); __kthread_unpark(k, kthread); wake_up_process(k); wait_for_completion(&kthread->exited); } ret = k->exit_code; put_task_struct(k); trace_sched_kthread_stop_ret(ret); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
thomas gleixnerthomas gleixner4347.78%233.33%
andrew mortonandrew morton2123.33%116.67%
oleg nesterovoleg nesterov2123.33%233.33%
mathieu desnoyersmathieu desnoyers55.56%116.67%
Total90100.00%6100.00%

EXPORT_SYMBOL(kthread_stop);
int kthreadd(void *unused) { struct task_struct *tsk = current; /* Setup a clean context for our children to inherit. */ set_task_comm(tsk, "kthreadd"); ignore_signals(tsk); set_cpus_allowed_ptr(tsk, cpu_all_mask); set_mems_allowed(node_states[N_MEMORY]); current->flags |= PF_NOFREEZE; 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); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
eric w. biedermaneric w. biederman11980.41%111.11%
ingo molnaringo molnar128.11%111.11%
miao xiemiao xie74.73%222.22%
satyam sharmasatyam sharma53.38%111.11%
oleg nesterovoleg nesterov21.35%111.11%
mike travismike travis10.68%111.11%
rusty russellrusty russell10.68%111.11%
lai jiangshanlai jiangshan10.68%111.11%
Total148100.00%9100.00%


void __init_kthread_worker(struct kthread_worker *worker, const char *name, struct lock_class_key *key) { spin_lock_init(&worker->lock); lockdep_set_class_and_name(&worker->lock, key, name); INIT_LIST_HEAD(&worker->work_list); worker->task = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
yong zhangyong zhang54100.00%1100.00%
Total54100.00%1100.00%

EXPORT_SYMBOL_GPL(__init_kthread_worker); /** * kthread_worker_fn - kthread function to process kthread_worker * @worker_ptr: pointer to initialized kthread_worker * * This function can be used as @threadfn to kthread_create() or * kthread_run() with @worker_ptr argument pointing to an initialized * kthread_worker. The started kthread will process work_list until * the it is stopped with kthread_stop(). A kthread can also call * this function directly after extra initialization. * * Different kthreads can be used for the same kthread_worker as long * as there's only one kthread attached to it at any given time. A * kthread_worker without an attached kthread simply collects queued * kthread_works. */
int kthread_worker_fn(void *worker_ptr) { struct kthread_worker *worker = worker_ptr; struct kthread_work *work; WARN_ON(worker->task); worker->task = current; repeat: set_current_state(TASK_INTERRUPTIBLE); /* mb paired w/ kthread_stop */ if (kthread_should_stop()) { __set_current_state(TASK_RUNNING); spin_lock_irq(&worker->lock); worker->task = NULL; spin_unlock_irq(&worker->lock); return 0; } work = NULL; spin_lock_irq(&worker->lock); if (!list_empty(&worker->work_list)) { work = list_first_entry(&worker->work_list, struct kthread_work, node); list_del_init(&work->node); } worker->current_work = work; spin_unlock_irq(&worker->lock); if (work) { __set_current_state(TASK_RUNNING); work->func(work); } else if (!freezing(current)) schedule(); try_to_freeze(); goto repeat; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo177100.00%2100.00%
Total177100.00%2100.00%

EXPORT_SYMBOL_GPL(kthread_worker_fn); /* insert @work before @pos in @worker */
static void insert_kthread_work(struct kthread_worker *worker, struct kthread_work *work, struct list_head *pos) { lockdep_assert_held(&worker->lock); list_add_tail(&work->node, pos); work->worker = worker; if (!worker->current_work && likely(worker->task)) wake_up_process(worker->task); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo6192.42%266.67%
lai jiangshanlai jiangshan57.58%133.33%
Total66100.00%3100.00%

/** * queue_kthread_work - queue a kthread_work * @worker: target kthread_worker * @work: kthread_work to queue * * Queue @work to work processor @task for async execution. @task * must have been created with kthread_worker_create(). Returns %true * if @work was successfully queued, %false if it was already pending. */
bool queue_kthread_work(struct kthread_worker *worker, struct kthread_work *work) { bool ret = false; unsigned long flags; spin_lock_irqsave(&worker->lock, flags); if (list_empty(&work->node)) { insert_kthread_work(worker, work, &worker->work_list); ret = true; } spin_unlock_irqrestore(&worker->lock, flags); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo75100.00%2100.00%
Total75100.00%2100.00%

EXPORT_SYMBOL_GPL(queue_kthread_work); struct kthread_flush_work { struct kthread_work work; struct completion done; };
static void kthread_flush_work_fn(struct kthread_work *work) { struct kthread_flush_work *fwork = container_of(work, struct kthread_flush_work, work); complete(&fwork->done); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo34100.00%1100.00%
Total34100.00%1100.00%

/** * flush_kthread_work - flush a kthread_work * @work: work to flush * * If @work is queued or executing, wait for it to finish execution. */
void flush_kthread_work(struct kthread_work *work) { struct kthread_flush_work fwork = { KTHREAD_WORK_INIT(fwork.work, kthread_flush_work_fn), COMPLETION_INITIALIZER_ONSTACK(fwork.done), }; struct kthread_worker *worker; bool noop = false; retry: worker = work->worker; if (!worker) return; spin_lock_irq(&worker->lock); if (work->worker != worker) { spin_unlock_irq(&worker->lock); goto retry; } if (!list_empty(&work->node)) insert_kthread_work(worker, &fwork.work, work->node.next); else if (worker->current_work == work) insert_kthread_work(worker, &fwork.work, worker->work_list.next); else noop = true; spin_unlock_irq(&worker->lock); if (!noop) wait_for_completion(&fwork.done); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo163100.00%2100.00%
Total163100.00%2100.00%

EXPORT_SYMBOL_GPL(flush_kthread_work); /** * flush_kthread_worker - flush all current works on a kthread_worker * @worker: worker to flush * * Wait until all currently executing or pending works on @worker are * finished. */
void flush_kthread_worker(struct kthread_worker *worker) { struct kthread_flush_work fwork = { KTHREAD_WORK_INIT(fwork.work, kthread_flush_work_fn), COMPLETION_INITIALIZER_ONSTACK(fwork.done), }; queue_kthread_work(worker, &fwork.work); wait_for_completion(&fwork.done); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo50100.00%1100.00%
Total50100.00%1100.00%

EXPORT_SYMBOL_GPL(flush_kthread_worker);

Overall Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo73830.75%713.73%
thomas gleixnerthomas gleixner53622.33%23.92%
andrew mortonandrew morton30212.58%35.88%
oleg nesterovoleg nesterov1767.33%713.73%
tetsuo handatetsuo handa1606.67%23.92%
eric w. biedermaneric w. biederman1496.21%11.96%
peter zijlstrapeter zijlstra1114.62%35.88%
yong zhangyong zhang592.46%11.96%
eric dumazeteric dumazet552.29%11.96%
david kershnerdavid kershner200.83%11.96%
ingo molnaringo molnar160.67%11.96%
rusty russellrusty russell160.67%23.92%
miao xiemiao xie100.42%23.92%
david howellsdavid howells90.38%11.96%
mathieu desnoyersmathieu desnoyers70.29%11.96%
vitaliy gusevvitaliy gusev60.25%11.96%
lai jiangshanlai jiangshan60.25%23.92%
satyam sharmasatyam sharma50.21%11.96%
dmitry adamushkodmitry adamushko50.21%11.96%
al viroal viro30.12%11.96%
kosaki motohirokosaki motohiro30.12%23.92%
nishanth aravamudannishanth aravamudan20.08%23.92%
arjan van de venarjan van de ven10.04%11.96%
paul gortmakerpaul gortmaker10.04%11.96%
adrian bunkadrian bunk10.04%11.96%
robert p. j. dayrobert p. j. day10.04%11.96%
steven rostedtsteven rostedt10.04%11.96%
mike travismike travis10.04%11.96%
Total2400100.00%51100.00%
Directory: kernel
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
{% endraw %}