diff options
| author | Takashi Iwai <tiwai@suse.de> | 2011-08-08 08:30:29 -0400 |
|---|---|---|
| committer | Takashi Iwai <tiwai@suse.de> | 2011-08-08 08:30:29 -0400 |
| commit | 0a2d31b62dba9b5b92a38c67c9cc42630513662a (patch) | |
| tree | f755d74ec85248de645e10c45ed1a2ed467530f6 /kernel/workqueue.c | |
| parent | 8039290a91c5dc4414093c086987a5d7738fe2fd (diff) | |
| parent | df944f66784e6d4f2f50739263a4947885d8b6ae (diff) | |
Merge branch 'fix/kconfig' into for-linus
Diffstat (limited to 'kernel/workqueue.c')
| -rw-r--r-- | kernel/workqueue.c | 81 |
1 files changed, 53 insertions, 28 deletions
diff --git a/kernel/workqueue.c b/kernel/workqueue.c index 0400553f0d04..25fb1b0e53fa 100644 --- a/kernel/workqueue.c +++ b/kernel/workqueue.c | |||
| @@ -221,7 +221,7 @@ typedef unsigned long mayday_mask_t; | |||
| 221 | * per-CPU workqueues: | 221 | * per-CPU workqueues: |
| 222 | */ | 222 | */ |
| 223 | struct workqueue_struct { | 223 | struct workqueue_struct { |
| 224 | unsigned int flags; /* I: WQ_* flags */ | 224 | unsigned int flags; /* W: WQ_* flags */ |
| 225 | union { | 225 | union { |
| 226 | struct cpu_workqueue_struct __percpu *pcpu; | 226 | struct cpu_workqueue_struct __percpu *pcpu; |
| 227 | struct cpu_workqueue_struct *single; | 227 | struct cpu_workqueue_struct *single; |
| @@ -240,6 +240,7 @@ struct workqueue_struct { | |||
| 240 | mayday_mask_t mayday_mask; /* cpus requesting rescue */ | 240 | mayday_mask_t mayday_mask; /* cpus requesting rescue */ |
| 241 | struct worker *rescuer; /* I: rescue worker */ | 241 | struct worker *rescuer; /* I: rescue worker */ |
| 242 | 242 | ||
| 243 | int nr_drainers; /* W: drain in progress */ | ||
| 243 | int saved_max_active; /* W: saved cwq max_active */ | 244 | int saved_max_active; /* W: saved cwq max_active */ |
| 244 | const char *name; /* I: workqueue name */ | 245 | const char *name; /* I: workqueue name */ |
| 245 | #ifdef CONFIG_LOCKDEP | 246 | #ifdef CONFIG_LOCKDEP |
| @@ -990,7 +991,7 @@ static void __queue_work(unsigned int cpu, struct workqueue_struct *wq, | |||
| 990 | debug_work_activate(work); | 991 | debug_work_activate(work); |
| 991 | 992 | ||
| 992 | /* if dying, only works from the same workqueue are allowed */ | 993 | /* if dying, only works from the same workqueue are allowed */ |
| 993 | if (unlikely(wq->flags & WQ_DYING) && | 994 | if (unlikely(wq->flags & WQ_DRAINING) && |
| 994 | WARN_ON_ONCE(!is_chained_work(wq))) | 995 | WARN_ON_ONCE(!is_chained_work(wq))) |
| 995 | return; | 996 | return; |
| 996 | 997 | ||
| @@ -2381,6 +2382,54 @@ out_unlock: | |||
| 2381 | } | 2382 | } |
| 2382 | EXPORT_SYMBOL_GPL(flush_workqueue); | 2383 | EXPORT_SYMBOL_GPL(flush_workqueue); |
| 2383 | 2384 | ||
| 2385 | /** | ||
| 2386 | * drain_workqueue - drain a workqueue | ||
| 2387 | * @wq: workqueue to drain | ||
| 2388 | * | ||
| 2389 | * Wait until the workqueue becomes empty. While draining is in progress, | ||
| 2390 | * only chain queueing is allowed. IOW, only currently pending or running | ||
| 2391 | * work items on @wq can queue further work items on it. @wq is flushed | ||
| 2392 | * repeatedly until it becomes empty. The number of flushing is detemined | ||
| 2393 | * by the depth of chaining and should be relatively short. Whine if it | ||
| 2394 | * takes too long. | ||
| 2395 | */ | ||
| 2396 | void drain_workqueue(struct workqueue_struct *wq) | ||
| 2397 | { | ||
| 2398 | unsigned int flush_cnt = 0; | ||
| 2399 | unsigned int cpu; | ||
| 2400 | |||
| 2401 | /* | ||
| 2402 | * __queue_work() needs to test whether there are drainers, is much | ||
| 2403 | * hotter than drain_workqueue() and already looks at @wq->flags. | ||
| 2404 | * Use WQ_DRAINING so that queue doesn't have to check nr_drainers. | ||
| 2405 | */ | ||
| 2406 | spin_lock(&workqueue_lock); | ||
| 2407 | if (!wq->nr_drainers++) | ||
| 2408 | wq->flags |= WQ_DRAINING; | ||
| 2409 | spin_unlock(&workqueue_lock); | ||
| 2410 | reflush: | ||
| 2411 | flush_workqueue(wq); | ||
| 2412 | |||
| 2413 | for_each_cwq_cpu(cpu, wq) { | ||
| 2414 | struct cpu_workqueue_struct *cwq = get_cwq(cpu, wq); | ||
| 2415 | |||
| 2416 | if (!cwq->nr_active && list_empty(&cwq->delayed_works)) | ||
| 2417 | continue; | ||
| 2418 | |||
| 2419 | if (++flush_cnt == 10 || | ||
| 2420 | (flush_cnt % 100 == 0 && flush_cnt <= 1000)) | ||
| 2421 | pr_warning("workqueue %s: flush on destruction isn't complete after %u tries\n", | ||
| 2422 | wq->name, flush_cnt); | ||
| 2423 | goto reflush; | ||
| 2424 | } | ||
| 2425 | |||
| 2426 | spin_lock(&workqueue_lock); | ||
| 2427 | if (!--wq->nr_drainers) | ||
| 2428 | wq->flags &= ~WQ_DRAINING; | ||
| 2429 | spin_unlock(&workqueue_lock); | ||
| 2430 | } | ||
| 2431 | EXPORT_SYMBOL_GPL(drain_workqueue); | ||
| 2432 | |||
| 2384 | static bool start_flush_work(struct work_struct *work, struct wq_barrier *barr, | 2433 | static bool start_flush_work(struct work_struct *work, struct wq_barrier *barr, |
| 2385 | bool wait_executing) | 2434 | bool wait_executing) |
| 2386 | { | 2435 | { |
| @@ -3009,34 +3058,10 @@ EXPORT_SYMBOL_GPL(__alloc_workqueue_key); | |||
| 3009 | */ | 3058 | */ |
| 3010 | void destroy_workqueue(struct workqueue_struct *wq) | 3059 | void destroy_workqueue(struct workqueue_struct *wq) |
| 3011 | { | 3060 | { |
| 3012 | unsigned int flush_cnt = 0; | ||
| 3013 | unsigned int cpu; | 3061 | unsigned int cpu; |
| 3014 | 3062 | ||
| 3015 | /* | 3063 | /* drain it before proceeding with destruction */ |
| 3016 | * Mark @wq dying and drain all pending works. Once WQ_DYING is | 3064 | drain_workqueue(wq); |
| 3017 | * set, only chain queueing is allowed. IOW, only currently | ||
| 3018 | * pending or running work items on @wq can queue further work | ||
| 3019 | * items on it. @wq is flushed repeatedly until it becomes empty. | ||
| 3020 | * The number of flushing is detemined by the depth of chaining and | ||
| 3021 | * should be relatively short. Whine if it takes too long. | ||
| 3022 | */ | ||
| 3023 | wq->flags |= WQ_DYING; | ||
| 3024 | reflush: | ||
| 3025 | flush_workqueue(wq); | ||
| 3026 | |||
| 3027 | for_each_cwq_cpu(cpu, wq) { | ||
| 3028 | struct cpu_workqueue_struct *cwq = get_cwq(cpu, wq); | ||
| 3029 | |||
| 3030 | if (!cwq->nr_active && list_empty(&cwq->delayed_works)) | ||
| 3031 | continue; | ||
| 3032 | |||
| 3033 | if (++flush_cnt == 10 || | ||
| 3034 | (flush_cnt % 100 == 0 && flush_cnt <= 1000)) | ||
| 3035 | printk(KERN_WARNING "workqueue %s: flush on " | ||
| 3036 | "destruction isn't complete after %u tries\n", | ||
| 3037 | wq->name, flush_cnt); | ||
| 3038 | goto reflush; | ||
| 3039 | } | ||
| 3040 | 3065 | ||
| 3041 | /* | 3066 | /* |
| 3042 | * wq list is used to freeze wq, remove from list after | 3067 | * wq list is used to freeze wq, remove from list after |
