| /* | 
 |  * Copyright (C) 2007 Oracle.  All rights reserved. | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or | 
 |  * modify it under the terms of the GNU General Public | 
 |  * License v2 as published by the Free Software Foundation. | 
 |  * | 
 |  * This program is distributed in the hope that it will be useful, | 
 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |  * General Public License for more details. | 
 |  * | 
 |  * You should have received a copy of the GNU General Public | 
 |  * License along with this program; if not, write to the | 
 |  * Free Software Foundation, Inc., 59 Temple Place - Suite 330, | 
 |  * Boston, MA 021110-1307, USA. | 
 |  */ | 
 |  | 
 | #include <linux/kthread.h> | 
 | #include <linux/slab.h> | 
 | #include <linux/list.h> | 
 | #include <linux/spinlock.h> | 
 | #include <linux/freezer.h> | 
 | #include "async-thread.h" | 
 |  | 
 | #define WORK_QUEUED_BIT 0 | 
 | #define WORK_DONE_BIT 1 | 
 | #define WORK_ORDER_DONE_BIT 2 | 
 | #define WORK_HIGH_PRIO_BIT 3 | 
 |  | 
 | /* | 
 |  * container for the kthread task pointer and the list of pending work | 
 |  * One of these is allocated per thread. | 
 |  */ | 
 | struct btrfs_worker_thread { | 
 | 	/* pool we belong to */ | 
 | 	struct btrfs_workers *workers; | 
 |  | 
 | 	/* list of struct btrfs_work that are waiting for service */ | 
 | 	struct list_head pending; | 
 | 	struct list_head prio_pending; | 
 |  | 
 | 	/* list of worker threads from struct btrfs_workers */ | 
 | 	struct list_head worker_list; | 
 |  | 
 | 	/* kthread */ | 
 | 	struct task_struct *task; | 
 |  | 
 | 	/* number of things on the pending list */ | 
 | 	atomic_t num_pending; | 
 |  | 
 | 	/* reference counter for this struct */ | 
 | 	atomic_t refs; | 
 |  | 
 | 	unsigned long sequence; | 
 |  | 
 | 	/* protects the pending list. */ | 
 | 	spinlock_t lock; | 
 |  | 
 | 	/* set to non-zero when this thread is already awake and kicking */ | 
 | 	int working; | 
 |  | 
 | 	/* are we currently idle */ | 
 | 	int idle; | 
 | }; | 
 |  | 
 | static int __btrfs_start_workers(struct btrfs_workers *workers); | 
 |  | 
 | /* | 
 |  * btrfs_start_workers uses kthread_run, which can block waiting for memory | 
 |  * for a very long time.  It will actually throttle on page writeback, | 
 |  * and so it may not make progress until after our btrfs worker threads | 
 |  * process all of the pending work structs in their queue | 
 |  * | 
 |  * This means we can't use btrfs_start_workers from inside a btrfs worker | 
 |  * thread that is used as part of cleaning dirty memory, which pretty much | 
 |  * involves all of the worker threads. | 
 |  * | 
 |  * Instead we have a helper queue who never has more than one thread | 
 |  * where we scheduler thread start operations.  This worker_start struct | 
 |  * is used to contain the work and hold a pointer to the queue that needs | 
 |  * another worker. | 
 |  */ | 
 | struct worker_start { | 
 | 	struct btrfs_work work; | 
 | 	struct btrfs_workers *queue; | 
 | }; | 
 |  | 
 | static void start_new_worker_func(struct btrfs_work *work) | 
 | { | 
 | 	struct worker_start *start; | 
 | 	start = container_of(work, struct worker_start, work); | 
 | 	__btrfs_start_workers(start->queue); | 
 | 	kfree(start); | 
 | } | 
 |  | 
 | /* | 
 |  * helper function to move a thread onto the idle list after it | 
 |  * has finished some requests. | 
 |  */ | 
 | static void check_idle_worker(struct btrfs_worker_thread *worker) | 
 | { | 
 | 	if (!worker->idle && atomic_read(&worker->num_pending) < | 
 | 	    worker->workers->idle_thresh / 2) { | 
 | 		unsigned long flags; | 
 | 		spin_lock_irqsave(&worker->workers->lock, flags); | 
 | 		worker->idle = 1; | 
 |  | 
 | 		/* the list may be empty if the worker is just starting */ | 
 | 		if (!list_empty(&worker->worker_list)) { | 
 | 			list_move(&worker->worker_list, | 
 | 				 &worker->workers->idle_list); | 
 | 		} | 
 | 		spin_unlock_irqrestore(&worker->workers->lock, flags); | 
 | 	} | 
 | } | 
 |  | 
 | /* | 
 |  * helper function to move a thread off the idle list after new | 
 |  * pending work is added. | 
 |  */ | 
 | static void check_busy_worker(struct btrfs_worker_thread *worker) | 
 | { | 
 | 	if (worker->idle && atomic_read(&worker->num_pending) >= | 
 | 	    worker->workers->idle_thresh) { | 
 | 		unsigned long flags; | 
 | 		spin_lock_irqsave(&worker->workers->lock, flags); | 
 | 		worker->idle = 0; | 
 |  | 
 | 		if (!list_empty(&worker->worker_list)) { | 
 | 			list_move_tail(&worker->worker_list, | 
 | 				      &worker->workers->worker_list); | 
 | 		} | 
 | 		spin_unlock_irqrestore(&worker->workers->lock, flags); | 
 | 	} | 
 | } | 
 |  | 
 | static void check_pending_worker_creates(struct btrfs_worker_thread *worker) | 
 | { | 
 | 	struct btrfs_workers *workers = worker->workers; | 
 | 	struct worker_start *start; | 
 | 	unsigned long flags; | 
 |  | 
 | 	rmb(); | 
 | 	if (!workers->atomic_start_pending) | 
 | 		return; | 
 |  | 
 | 	start = kzalloc(sizeof(*start), GFP_NOFS); | 
 | 	if (!start) | 
 | 		return; | 
 |  | 
 | 	start->work.func = start_new_worker_func; | 
 | 	start->queue = workers; | 
 |  | 
 | 	spin_lock_irqsave(&workers->lock, flags); | 
 | 	if (!workers->atomic_start_pending) | 
 | 		goto out; | 
 |  | 
 | 	workers->atomic_start_pending = 0; | 
 | 	if (workers->num_workers + workers->num_workers_starting >= | 
 | 	    workers->max_workers) | 
 | 		goto out; | 
 |  | 
 | 	workers->num_workers_starting += 1; | 
 | 	spin_unlock_irqrestore(&workers->lock, flags); | 
 | 	btrfs_queue_worker(workers->atomic_worker_start, &start->work); | 
 | 	return; | 
 |  | 
 | out: | 
 | 	kfree(start); | 
 | 	spin_unlock_irqrestore(&workers->lock, flags); | 
 | } | 
 |  | 
 | static noinline void run_ordered_completions(struct btrfs_workers *workers, | 
 | 					    struct btrfs_work *work) | 
 | { | 
 | 	if (!workers->ordered) | 
 | 		return; | 
 |  | 
 | 	set_bit(WORK_DONE_BIT, &work->flags); | 
 |  | 
 | 	spin_lock(&workers->order_lock); | 
 |  | 
 | 	while (1) { | 
 | 		if (!list_empty(&workers->prio_order_list)) { | 
 | 			work = list_entry(workers->prio_order_list.next, | 
 | 					  struct btrfs_work, order_list); | 
 | 		} else if (!list_empty(&workers->order_list)) { | 
 | 			work = list_entry(workers->order_list.next, | 
 | 					  struct btrfs_work, order_list); | 
 | 		} else { | 
 | 			break; | 
 | 		} | 
 | 		if (!test_bit(WORK_DONE_BIT, &work->flags)) | 
 | 			break; | 
 |  | 
 | 		/* we are going to call the ordered done function, but | 
 | 		 * we leave the work item on the list as a barrier so | 
 | 		 * that later work items that are done don't have their | 
 | 		 * functions called before this one returns | 
 | 		 */ | 
 | 		if (test_and_set_bit(WORK_ORDER_DONE_BIT, &work->flags)) | 
 | 			break; | 
 |  | 
 | 		spin_unlock(&workers->order_lock); | 
 |  | 
 | 		work->ordered_func(work); | 
 |  | 
 | 		/* now take the lock again and call the freeing code */ | 
 | 		spin_lock(&workers->order_lock); | 
 | 		list_del(&work->order_list); | 
 | 		work->ordered_free(work); | 
 | 	} | 
 |  | 
 | 	spin_unlock(&workers->order_lock); | 
 | } | 
 |  | 
 | static void put_worker(struct btrfs_worker_thread *worker) | 
 | { | 
 | 	if (atomic_dec_and_test(&worker->refs)) | 
 | 		kfree(worker); | 
 | } | 
 |  | 
 | static int try_worker_shutdown(struct btrfs_worker_thread *worker) | 
 | { | 
 | 	int freeit = 0; | 
 |  | 
 | 	spin_lock_irq(&worker->lock); | 
 | 	spin_lock(&worker->workers->lock); | 
 | 	if (worker->workers->num_workers > 1 && | 
 | 	    worker->idle && | 
 | 	    !worker->working && | 
 | 	    !list_empty(&worker->worker_list) && | 
 | 	    list_empty(&worker->prio_pending) && | 
 | 	    list_empty(&worker->pending) && | 
 | 	    atomic_read(&worker->num_pending) == 0) { | 
 | 		freeit = 1; | 
 | 		list_del_init(&worker->worker_list); | 
 | 		worker->workers->num_workers--; | 
 | 	} | 
 | 	spin_unlock(&worker->workers->lock); | 
 | 	spin_unlock_irq(&worker->lock); | 
 |  | 
 | 	if (freeit) | 
 | 		put_worker(worker); | 
 | 	return freeit; | 
 | } | 
 |  | 
 | static struct btrfs_work *get_next_work(struct btrfs_worker_thread *worker, | 
 | 					struct list_head *prio_head, | 
 | 					struct list_head *head) | 
 | { | 
 | 	struct btrfs_work *work = NULL; | 
 | 	struct list_head *cur = NULL; | 
 |  | 
 | 	if(!list_empty(prio_head)) | 
 | 		cur = prio_head->next; | 
 |  | 
 | 	smp_mb(); | 
 | 	if (!list_empty(&worker->prio_pending)) | 
 | 		goto refill; | 
 |  | 
 | 	if (!list_empty(head)) | 
 | 		cur = head->next; | 
 |  | 
 | 	if (cur) | 
 | 		goto out; | 
 |  | 
 | refill: | 
 | 	spin_lock_irq(&worker->lock); | 
 | 	list_splice_tail_init(&worker->prio_pending, prio_head); | 
 | 	list_splice_tail_init(&worker->pending, head); | 
 |  | 
 | 	if (!list_empty(prio_head)) | 
 | 		cur = prio_head->next; | 
 | 	else if (!list_empty(head)) | 
 | 		cur = head->next; | 
 | 	spin_unlock_irq(&worker->lock); | 
 |  | 
 | 	if (!cur) | 
 | 		goto out_fail; | 
 |  | 
 | out: | 
 | 	work = list_entry(cur, struct btrfs_work, list); | 
 |  | 
 | out_fail: | 
 | 	return work; | 
 | } | 
 |  | 
 | /* | 
 |  * main loop for servicing work items | 
 |  */ | 
 | static int worker_loop(void *arg) | 
 | { | 
 | 	struct btrfs_worker_thread *worker = arg; | 
 | 	struct list_head head; | 
 | 	struct list_head prio_head; | 
 | 	struct btrfs_work *work; | 
 |  | 
 | 	INIT_LIST_HEAD(&head); | 
 | 	INIT_LIST_HEAD(&prio_head); | 
 |  | 
 | 	do { | 
 | again: | 
 | 		while (1) { | 
 |  | 
 |  | 
 | 			work = get_next_work(worker, &prio_head, &head); | 
 | 			if (!work) | 
 | 				break; | 
 |  | 
 | 			list_del(&work->list); | 
 | 			clear_bit(WORK_QUEUED_BIT, &work->flags); | 
 |  | 
 | 			work->worker = worker; | 
 |  | 
 | 			work->func(work); | 
 |  | 
 | 			atomic_dec(&worker->num_pending); | 
 | 			/* | 
 | 			 * unless this is an ordered work queue, | 
 | 			 * 'work' was probably freed by func above. | 
 | 			 */ | 
 | 			run_ordered_completions(worker->workers, work); | 
 |  | 
 | 			check_pending_worker_creates(worker); | 
 | 			cond_resched(); | 
 | 		} | 
 |  | 
 | 		spin_lock_irq(&worker->lock); | 
 | 		check_idle_worker(worker); | 
 |  | 
 | 		if (freezing(current)) { | 
 | 			worker->working = 0; | 
 | 			spin_unlock_irq(&worker->lock); | 
 | 			try_to_freeze(); | 
 | 		} else { | 
 | 			spin_unlock_irq(&worker->lock); | 
 | 			if (!kthread_should_stop()) { | 
 | 				cpu_relax(); | 
 | 				/* | 
 | 				 * we've dropped the lock, did someone else | 
 | 				 * jump_in? | 
 | 				 */ | 
 | 				smp_mb(); | 
 | 				if (!list_empty(&worker->pending) || | 
 | 				    !list_empty(&worker->prio_pending)) | 
 | 					continue; | 
 |  | 
 | 				/* | 
 | 				 * this short schedule allows more work to | 
 | 				 * come in without the queue functions | 
 | 				 * needing to go through wake_up_process() | 
 | 				 * | 
 | 				 * worker->working is still 1, so nobody | 
 | 				 * is going to try and wake us up | 
 | 				 */ | 
 | 				schedule_timeout(1); | 
 | 				smp_mb(); | 
 | 				if (!list_empty(&worker->pending) || | 
 | 				    !list_empty(&worker->prio_pending)) | 
 | 					continue; | 
 |  | 
 | 				if (kthread_should_stop()) | 
 | 					break; | 
 |  | 
 | 				/* still no more work?, sleep for real */ | 
 | 				spin_lock_irq(&worker->lock); | 
 | 				set_current_state(TASK_INTERRUPTIBLE); | 
 | 				if (!list_empty(&worker->pending) || | 
 | 				    !list_empty(&worker->prio_pending)) { | 
 | 					spin_unlock_irq(&worker->lock); | 
 | 					set_current_state(TASK_RUNNING); | 
 | 					goto again; | 
 | 				} | 
 |  | 
 | 				/* | 
 | 				 * this makes sure we get a wakeup when someone | 
 | 				 * adds something new to the queue | 
 | 				 */ | 
 | 				worker->working = 0; | 
 | 				spin_unlock_irq(&worker->lock); | 
 |  | 
 | 				if (!kthread_should_stop()) { | 
 | 					schedule_timeout(HZ * 120); | 
 | 					if (!worker->working && | 
 | 					    try_worker_shutdown(worker)) { | 
 | 						return 0; | 
 | 					} | 
 | 				} | 
 | 			} | 
 | 			__set_current_state(TASK_RUNNING); | 
 | 		} | 
 | 	} while (!kthread_should_stop()); | 
 | 	return 0; | 
 | } | 
 |  | 
 | /* | 
 |  * this will wait for all the worker threads to shutdown | 
 |  */ | 
 | void btrfs_stop_workers(struct btrfs_workers *workers) | 
 | { | 
 | 	struct list_head *cur; | 
 | 	struct btrfs_worker_thread *worker; | 
 | 	int can_stop; | 
 |  | 
 | 	spin_lock_irq(&workers->lock); | 
 | 	list_splice_init(&workers->idle_list, &workers->worker_list); | 
 | 	while (!list_empty(&workers->worker_list)) { | 
 | 		cur = workers->worker_list.next; | 
 | 		worker = list_entry(cur, struct btrfs_worker_thread, | 
 | 				    worker_list); | 
 |  | 
 | 		atomic_inc(&worker->refs); | 
 | 		workers->num_workers -= 1; | 
 | 		if (!list_empty(&worker->worker_list)) { | 
 | 			list_del_init(&worker->worker_list); | 
 | 			put_worker(worker); | 
 | 			can_stop = 1; | 
 | 		} else | 
 | 			can_stop = 0; | 
 | 		spin_unlock_irq(&workers->lock); | 
 | 		if (can_stop) | 
 | 			kthread_stop(worker->task); | 
 | 		spin_lock_irq(&workers->lock); | 
 | 		put_worker(worker); | 
 | 	} | 
 | 	spin_unlock_irq(&workers->lock); | 
 | } | 
 |  | 
 | /* | 
 |  * simple init on struct btrfs_workers | 
 |  */ | 
 | void btrfs_init_workers(struct btrfs_workers *workers, char *name, int max, | 
 | 			struct btrfs_workers *async_helper) | 
 | { | 
 | 	workers->num_workers = 0; | 
 | 	workers->num_workers_starting = 0; | 
 | 	INIT_LIST_HEAD(&workers->worker_list); | 
 | 	INIT_LIST_HEAD(&workers->idle_list); | 
 | 	INIT_LIST_HEAD(&workers->order_list); | 
 | 	INIT_LIST_HEAD(&workers->prio_order_list); | 
 | 	spin_lock_init(&workers->lock); | 
 | 	spin_lock_init(&workers->order_lock); | 
 | 	workers->max_workers = max; | 
 | 	workers->idle_thresh = 32; | 
 | 	workers->name = name; | 
 | 	workers->ordered = 0; | 
 | 	workers->atomic_start_pending = 0; | 
 | 	workers->atomic_worker_start = async_helper; | 
 | } | 
 |  | 
 | /* | 
 |  * starts new worker threads.  This does not enforce the max worker | 
 |  * count in case you need to temporarily go past it. | 
 |  */ | 
 | static int __btrfs_start_workers(struct btrfs_workers *workers) | 
 | { | 
 | 	struct btrfs_worker_thread *worker; | 
 | 	int ret = 0; | 
 |  | 
 | 	worker = kzalloc(sizeof(*worker), GFP_NOFS); | 
 | 	if (!worker) { | 
 | 		ret = -ENOMEM; | 
 | 		goto fail; | 
 | 	} | 
 |  | 
 | 	INIT_LIST_HEAD(&worker->pending); | 
 | 	INIT_LIST_HEAD(&worker->prio_pending); | 
 | 	INIT_LIST_HEAD(&worker->worker_list); | 
 | 	spin_lock_init(&worker->lock); | 
 |  | 
 | 	atomic_set(&worker->num_pending, 0); | 
 | 	atomic_set(&worker->refs, 1); | 
 | 	worker->workers = workers; | 
 | 	worker->task = kthread_run(worker_loop, worker, | 
 | 				   "btrfs-%s-%d", workers->name, | 
 | 				   workers->num_workers + 1); | 
 | 	if (IS_ERR(worker->task)) { | 
 | 		ret = PTR_ERR(worker->task); | 
 | 		kfree(worker); | 
 | 		goto fail; | 
 | 	} | 
 | 	spin_lock_irq(&workers->lock); | 
 | 	list_add_tail(&worker->worker_list, &workers->idle_list); | 
 | 	worker->idle = 1; | 
 | 	workers->num_workers++; | 
 | 	workers->num_workers_starting--; | 
 | 	WARN_ON(workers->num_workers_starting < 0); | 
 | 	spin_unlock_irq(&workers->lock); | 
 |  | 
 | 	return 0; | 
 | fail: | 
 | 	spin_lock_irq(&workers->lock); | 
 | 	workers->num_workers_starting--; | 
 | 	spin_unlock_irq(&workers->lock); | 
 | 	return ret; | 
 | } | 
 |  | 
 | int btrfs_start_workers(struct btrfs_workers *workers) | 
 | { | 
 | 	spin_lock_irq(&workers->lock); | 
 | 	workers->num_workers_starting++; | 
 | 	spin_unlock_irq(&workers->lock); | 
 | 	return __btrfs_start_workers(workers); | 
 | } | 
 |  | 
 | /* | 
 |  * run through the list and find a worker thread that doesn't have a lot | 
 |  * to do right now.  This can return null if we aren't yet at the thread | 
 |  * count limit and all of the threads are busy. | 
 |  */ | 
 | static struct btrfs_worker_thread *next_worker(struct btrfs_workers *workers) | 
 | { | 
 | 	struct btrfs_worker_thread *worker; | 
 | 	struct list_head *next; | 
 | 	int enforce_min; | 
 |  | 
 | 	enforce_min = (workers->num_workers + workers->num_workers_starting) < | 
 | 		workers->max_workers; | 
 |  | 
 | 	/* | 
 | 	 * if we find an idle thread, don't move it to the end of the | 
 | 	 * idle list.  This improves the chance that the next submission | 
 | 	 * will reuse the same thread, and maybe catch it while it is still | 
 | 	 * working | 
 | 	 */ | 
 | 	if (!list_empty(&workers->idle_list)) { | 
 | 		next = workers->idle_list.next; | 
 | 		worker = list_entry(next, struct btrfs_worker_thread, | 
 | 				    worker_list); | 
 | 		return worker; | 
 | 	} | 
 | 	if (enforce_min || list_empty(&workers->worker_list)) | 
 | 		return NULL; | 
 |  | 
 | 	/* | 
 | 	 * if we pick a busy task, move the task to the end of the list. | 
 | 	 * hopefully this will keep things somewhat evenly balanced. | 
 | 	 * Do the move in batches based on the sequence number.  This groups | 
 | 	 * requests submitted at roughly the same time onto the same worker. | 
 | 	 */ | 
 | 	next = workers->worker_list.next; | 
 | 	worker = list_entry(next, struct btrfs_worker_thread, worker_list); | 
 | 	worker->sequence++; | 
 |  | 
 | 	if (worker->sequence % workers->idle_thresh == 0) | 
 | 		list_move_tail(next, &workers->worker_list); | 
 | 	return worker; | 
 | } | 
 |  | 
 | /* | 
 |  * selects a worker thread to take the next job.  This will either find | 
 |  * an idle worker, start a new worker up to the max count, or just return | 
 |  * one of the existing busy workers. | 
 |  */ | 
 | static struct btrfs_worker_thread *find_worker(struct btrfs_workers *workers) | 
 | { | 
 | 	struct btrfs_worker_thread *worker; | 
 | 	unsigned long flags; | 
 | 	struct list_head *fallback; | 
 | 	int ret; | 
 |  | 
 | 	spin_lock_irqsave(&workers->lock, flags); | 
 | again: | 
 | 	worker = next_worker(workers); | 
 |  | 
 | 	if (!worker) { | 
 | 		if (workers->num_workers + workers->num_workers_starting >= | 
 | 		    workers->max_workers) { | 
 | 			goto fallback; | 
 | 		} else if (workers->atomic_worker_start) { | 
 | 			workers->atomic_start_pending = 1; | 
 | 			goto fallback; | 
 | 		} else { | 
 | 			workers->num_workers_starting++; | 
 | 			spin_unlock_irqrestore(&workers->lock, flags); | 
 | 			/* we're below the limit, start another worker */ | 
 | 			ret = __btrfs_start_workers(workers); | 
 | 			spin_lock_irqsave(&workers->lock, flags); | 
 | 			if (ret) | 
 | 				goto fallback; | 
 | 			goto again; | 
 | 		} | 
 | 	} | 
 | 	goto found; | 
 |  | 
 | fallback: | 
 | 	fallback = NULL; | 
 | 	/* | 
 | 	 * we have failed to find any workers, just | 
 | 	 * return the first one we can find. | 
 | 	 */ | 
 | 	if (!list_empty(&workers->worker_list)) | 
 | 		fallback = workers->worker_list.next; | 
 | 	if (!list_empty(&workers->idle_list)) | 
 | 		fallback = workers->idle_list.next; | 
 | 	BUG_ON(!fallback); | 
 | 	worker = list_entry(fallback, | 
 | 		  struct btrfs_worker_thread, worker_list); | 
 | found: | 
 | 	/* | 
 | 	 * this makes sure the worker doesn't exit before it is placed | 
 | 	 * onto a busy/idle list | 
 | 	 */ | 
 | 	atomic_inc(&worker->num_pending); | 
 | 	spin_unlock_irqrestore(&workers->lock, flags); | 
 | 	return worker; | 
 | } | 
 |  | 
 | /* | 
 |  * btrfs_requeue_work just puts the work item back on the tail of the list | 
 |  * it was taken from.  It is intended for use with long running work functions | 
 |  * that make some progress and want to give the cpu up for others. | 
 |  */ | 
 | void btrfs_requeue_work(struct btrfs_work *work) | 
 | { | 
 | 	struct btrfs_worker_thread *worker = work->worker; | 
 | 	unsigned long flags; | 
 | 	int wake = 0; | 
 |  | 
 | 	if (test_and_set_bit(WORK_QUEUED_BIT, &work->flags)) | 
 | 		return; | 
 |  | 
 | 	spin_lock_irqsave(&worker->lock, flags); | 
 | 	if (test_bit(WORK_HIGH_PRIO_BIT, &work->flags)) | 
 | 		list_add_tail(&work->list, &worker->prio_pending); | 
 | 	else | 
 | 		list_add_tail(&work->list, &worker->pending); | 
 | 	atomic_inc(&worker->num_pending); | 
 |  | 
 | 	/* by definition we're busy, take ourselves off the idle | 
 | 	 * list | 
 | 	 */ | 
 | 	if (worker->idle) { | 
 | 		spin_lock(&worker->workers->lock); | 
 | 		worker->idle = 0; | 
 | 		list_move_tail(&worker->worker_list, | 
 | 			      &worker->workers->worker_list); | 
 | 		spin_unlock(&worker->workers->lock); | 
 | 	} | 
 | 	if (!worker->working) { | 
 | 		wake = 1; | 
 | 		worker->working = 1; | 
 | 	} | 
 |  | 
 | 	if (wake) | 
 | 		wake_up_process(worker->task); | 
 | 	spin_unlock_irqrestore(&worker->lock, flags); | 
 | } | 
 |  | 
 | void btrfs_set_work_high_prio(struct btrfs_work *work) | 
 | { | 
 | 	set_bit(WORK_HIGH_PRIO_BIT, &work->flags); | 
 | } | 
 |  | 
 | /* | 
 |  * places a struct btrfs_work into the pending queue of one of the kthreads | 
 |  */ | 
 | void btrfs_queue_worker(struct btrfs_workers *workers, struct btrfs_work *work) | 
 | { | 
 | 	struct btrfs_worker_thread *worker; | 
 | 	unsigned long flags; | 
 | 	int wake = 0; | 
 |  | 
 | 	/* don't requeue something already on a list */ | 
 | 	if (test_and_set_bit(WORK_QUEUED_BIT, &work->flags)) | 
 | 		return; | 
 |  | 
 | 	worker = find_worker(workers); | 
 | 	if (workers->ordered) { | 
 | 		/* | 
 | 		 * you're not allowed to do ordered queues from an | 
 | 		 * interrupt handler | 
 | 		 */ | 
 | 		spin_lock(&workers->order_lock); | 
 | 		if (test_bit(WORK_HIGH_PRIO_BIT, &work->flags)) { | 
 | 			list_add_tail(&work->order_list, | 
 | 				      &workers->prio_order_list); | 
 | 		} else { | 
 | 			list_add_tail(&work->order_list, &workers->order_list); | 
 | 		} | 
 | 		spin_unlock(&workers->order_lock); | 
 | 	} else { | 
 | 		INIT_LIST_HEAD(&work->order_list); | 
 | 	} | 
 |  | 
 | 	spin_lock_irqsave(&worker->lock, flags); | 
 |  | 
 | 	if (test_bit(WORK_HIGH_PRIO_BIT, &work->flags)) | 
 | 		list_add_tail(&work->list, &worker->prio_pending); | 
 | 	else | 
 | 		list_add_tail(&work->list, &worker->pending); | 
 | 	check_busy_worker(worker); | 
 |  | 
 | 	/* | 
 | 	 * avoid calling into wake_up_process if this thread has already | 
 | 	 * been kicked | 
 | 	 */ | 
 | 	if (!worker->working) | 
 | 		wake = 1; | 
 | 	worker->working = 1; | 
 |  | 
 | 	if (wake) | 
 | 		wake_up_process(worker->task); | 
 | 	spin_unlock_irqrestore(&worker->lock, flags); | 
 | } |