-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Btrfs: Add async worker threads for pre and post IO checksumming
Btrfs has been using workqueues to spread the checksumming load across other CPUs in the system. But, workqueues only schedule work on the same CPU that queued the work, giving them a limited benefit for systems with higher CPU counts. This code adds a generic facility to schedule work with pools of kthreads, and changes the bio submission code to queue bios up. The queueing is important to make sure large numbers of procs on the system don't turn streaming workloads into random workloads by sending IO down concurrently. The end result of all of this is much higher performance (and CPU usage) when doing checksumming on large machines. Two worker pools are created, one for writes and one for endio processing. The two could deadlock if we tried to service both from a single pool. Signed-off-by: Chris Mason <chris.mason@oracle.com>
- Loading branch information
Chris Mason
committed
Sep 25, 2008
1 parent
43e570b
commit 8b71284
Showing
8 changed files
with
626 additions
and
132 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,288 @@ | ||
/* | ||
* 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/list.h> | ||
#include <linux/spinlock.h> | ||
#include <linux/freezer.h> | ||
#include "async-thread.h" | ||
|
||
/* | ||
* container for the kthread task pointer and the list of pending work | ||
* One of these is allocated per thread. | ||
*/ | ||
struct btrfs_worker_thread { | ||
/* list of struct btrfs_work that are waiting for service */ | ||
struct list_head 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; | ||
|
||
/* protects the pending list. */ | ||
spinlock_t lock; | ||
|
||
/* set to non-zero when this thread is already awake and kicking */ | ||
int working; | ||
}; | ||
|
||
/* | ||
* main loop for servicing work items | ||
*/ | ||
static int worker_loop(void *arg) | ||
{ | ||
struct btrfs_worker_thread *worker = arg; | ||
struct list_head *cur; | ||
struct btrfs_work *work; | ||
do { | ||
spin_lock_irq(&worker->lock); | ||
while(!list_empty(&worker->pending)) { | ||
cur = worker->pending.next; | ||
work = list_entry(cur, struct btrfs_work, list); | ||
list_del(&work->list); | ||
clear_bit(0, &work->flags); | ||
|
||
work->worker = worker; | ||
spin_unlock_irq(&worker->lock); | ||
|
||
work->func(work); | ||
|
||
atomic_dec(&worker->num_pending); | ||
spin_lock_irq(&worker->lock); | ||
} | ||
worker->working = 0; | ||
if (freezing(current)) { | ||
refrigerator(); | ||
} else { | ||
set_current_state(TASK_INTERRUPTIBLE); | ||
spin_unlock_irq(&worker->lock); | ||
schedule(); | ||
__set_current_state(TASK_RUNNING); | ||
} | ||
} while (!kthread_should_stop()); | ||
return 0; | ||
} | ||
|
||
/* | ||
* this will wait for all the worker threads to shutdown | ||
*/ | ||
int btrfs_stop_workers(struct btrfs_workers *workers) | ||
{ | ||
struct list_head *cur; | ||
struct btrfs_worker_thread *worker; | ||
|
||
while(!list_empty(&workers->worker_list)) { | ||
cur = workers->worker_list.next; | ||
worker = list_entry(cur, struct btrfs_worker_thread, | ||
worker_list); | ||
kthread_stop(worker->task); | ||
list_del(&worker->worker_list); | ||
kfree(worker); | ||
} | ||
return 0; | ||
} | ||
|
||
/* | ||
* simple init on struct btrfs_workers | ||
*/ | ||
void btrfs_init_workers(struct btrfs_workers *workers, int max) | ||
{ | ||
workers->num_workers = 0; | ||
INIT_LIST_HEAD(&workers->worker_list); | ||
workers->last = NULL; | ||
spin_lock_init(&workers->lock); | ||
workers->max_workers = max; | ||
} | ||
|
||
/* | ||
* starts new worker threads. This does not enforce the max worker | ||
* count in case you need to temporarily go past it. | ||
*/ | ||
int btrfs_start_workers(struct btrfs_workers *workers, int num_workers) | ||
{ | ||
struct btrfs_worker_thread *worker; | ||
int ret = 0; | ||
int i; | ||
|
||
for (i = 0; i < num_workers; i++) { | ||
worker = kzalloc(sizeof(*worker), GFP_NOFS); | ||
if (!worker) { | ||
ret = -ENOMEM; | ||
goto fail; | ||
} | ||
|
||
INIT_LIST_HEAD(&worker->pending); | ||
INIT_LIST_HEAD(&worker->worker_list); | ||
spin_lock_init(&worker->lock); | ||
atomic_set(&worker->num_pending, 0); | ||
worker->task = kthread_run(worker_loop, worker, "btrfs"); | ||
if (IS_ERR(worker->task)) { | ||
ret = PTR_ERR(worker->task); | ||
goto fail; | ||
} | ||
|
||
spin_lock_irq(&workers->lock); | ||
list_add_tail(&worker->worker_list, &workers->worker_list); | ||
workers->last = worker; | ||
workers->num_workers++; | ||
spin_unlock_irq(&workers->lock); | ||
} | ||
return 0; | ||
fail: | ||
btrfs_stop_workers(workers); | ||
return ret; | ||
} | ||
|
||
/* | ||
* 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; | ||
struct list_head *start; | ||
int enforce_min = workers->num_workers < workers->max_workers; | ||
|
||
/* start with the last thread if it isn't busy */ | ||
worker = workers->last; | ||
if (atomic_read(&worker->num_pending) < 64) | ||
goto done; | ||
|
||
next = worker->worker_list.next; | ||
start = &worker->worker_list; | ||
|
||
/* | ||
* check all the workers for someone that is bored. FIXME, do | ||
* something smart here | ||
*/ | ||
while(next != start) { | ||
if (next == &workers->worker_list) { | ||
next = workers->worker_list.next; | ||
continue; | ||
} | ||
worker = list_entry(next, struct btrfs_worker_thread, | ||
worker_list); | ||
if (atomic_read(&worker->num_pending) < 64 || !enforce_min) | ||
goto done; | ||
next = next->next; | ||
} | ||
/* | ||
* nobody was bored, if we're already at the max thread count, | ||
* use the last thread | ||
*/ | ||
if (!enforce_min || atomic_read(&workers->last->num_pending) < 64) { | ||
return workers->last; | ||
} | ||
return NULL; | ||
done: | ||
workers->last = worker; | ||
return worker; | ||
} | ||
|
||
static struct btrfs_worker_thread *find_worker(struct btrfs_workers *workers) | ||
{ | ||
struct btrfs_worker_thread *worker; | ||
unsigned long flags; | ||
|
||
again: | ||
spin_lock_irqsave(&workers->lock, flags); | ||
worker = next_worker(workers); | ||
spin_unlock_irqrestore(&workers->lock, flags); | ||
|
||
if (!worker) { | ||
spin_lock_irqsave(&workers->lock, flags); | ||
if (workers->num_workers >= workers->max_workers) { | ||
/* | ||
* we have failed to find any workers, just | ||
* return the force one | ||
*/ | ||
worker = list_entry(workers->worker_list.next, | ||
struct btrfs_worker_thread, worker_list); | ||
spin_unlock_irqrestore(&workers->lock, flags); | ||
} else { | ||
spin_unlock_irqrestore(&workers->lock, flags); | ||
/* we're below the limit, start another worker */ | ||
btrfs_start_workers(workers, 1); | ||
goto again; | ||
} | ||
} | ||
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. | ||
*/ | ||
int btrfs_requeue_work(struct btrfs_work *work) | ||
{ | ||
struct btrfs_worker_thread *worker = work->worker; | ||
unsigned long flags; | ||
|
||
if (test_and_set_bit(0, &work->flags)) | ||
goto out; | ||
|
||
spin_lock_irqsave(&worker->lock, flags); | ||
atomic_inc(&worker->num_pending); | ||
list_add_tail(&work->list, &worker->pending); | ||
spin_unlock_irqrestore(&worker->lock, flags); | ||
out: | ||
return 0; | ||
} | ||
|
||
/* | ||
* places a struct btrfs_work into the pending queue of one of the kthreads | ||
*/ | ||
int 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(0, &work->flags)) | ||
goto out; | ||
|
||
worker = find_worker(workers); | ||
|
||
spin_lock_irqsave(&worker->lock, flags); | ||
atomic_inc(&worker->num_pending); | ||
list_add_tail(&work->list, &worker->pending); | ||
|
||
/* | ||
* avoid calling into wake_up_process if this thread has already | ||
* been kicked | ||
*/ | ||
if (!worker->working) | ||
wake = 1; | ||
worker->working = 1; | ||
|
||
spin_unlock_irqrestore(&worker->lock, flags); | ||
|
||
if (wake) | ||
wake_up_process(worker->task); | ||
out: | ||
return 0; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,78 @@ | ||
/* | ||
* 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. | ||
*/ | ||
|
||
#ifndef __BTRFS_ASYNC_THREAD_ | ||
#define __BTRFS_ASYNC_THREAD_ | ||
|
||
struct btrfs_worker_thread; | ||
|
||
/* | ||
* This is similar to a workqueue, but it is meant to spread the operations | ||
* across all available cpus instead of just the CPU that was used to | ||
* queue the work. There is also some batching introduced to try and | ||
* cut down on context switches. | ||
* | ||
* By default threads are added on demand up to 2 * the number of cpus. | ||
* Changing struct btrfs_workers->max_workers is one way to prevent | ||
* demand creation of kthreads. | ||
* | ||
* the basic model of these worker threads is to embed a btrfs_work | ||
* structure in your own data struct, and use container_of in a | ||
* work function to get back to your data struct. | ||
*/ | ||
struct btrfs_work { | ||
/* | ||
* only func should be set to the function you want called | ||
* your work struct is passed as the only arg | ||
*/ | ||
void (*func)(struct btrfs_work *work); | ||
|
||
/* | ||
* flags should be set to zero. It is used to make sure the | ||
* struct is only inserted once into the list. | ||
*/ | ||
unsigned long flags; | ||
|
||
/* don't touch these */ | ||
struct btrfs_worker_thread *worker; | ||
struct list_head list; | ||
}; | ||
|
||
struct btrfs_workers { | ||
/* current number of running workers */ | ||
int num_workers; | ||
|
||
/* max number of workers allowed. changed by btrfs_start_workers */ | ||
int max_workers; | ||
|
||
/* list with all the work threads */ | ||
struct list_head worker_list; | ||
|
||
/* the last worker thread to have something queued */ | ||
struct btrfs_worker_thread *last; | ||
|
||
/* lock for finding the next worker thread to queue on */ | ||
spinlock_t lock; | ||
}; | ||
|
||
int btrfs_queue_worker(struct btrfs_workers *workers, struct btrfs_work *work); | ||
int btrfs_start_workers(struct btrfs_workers *workers, int num_workers); | ||
int btrfs_stop_workers(struct btrfs_workers *workers); | ||
void btrfs_init_workers(struct btrfs_workers *workers, int max); | ||
int btrfs_requeue_work(struct btrfs_work *work); | ||
#endif |
Oops, something went wrong.