diff options
Diffstat (limited to 'drivers/md')
-rw-r--r-- | drivers/md/dm-kcopyd.c | 17 |
1 files changed, 11 insertions, 6 deletions
diff --git a/drivers/md/dm-kcopyd.c b/drivers/md/dm-kcopyd.c index 9d379070918b..3e3fc06cb861 100644 --- a/drivers/md/dm-kcopyd.c +++ b/drivers/md/dm-kcopyd.c | |||
@@ -511,13 +511,16 @@ static void segment_complete(int read_err, unsigned long write_err, | |||
511 | } else if (atomic_dec_and_test(&job->sub_jobs)) { | 511 | } else if (atomic_dec_and_test(&job->sub_jobs)) { |
512 | 512 | ||
513 | /* | 513 | /* |
514 | * To avoid a race we must keep the job around | 514 | * Queue the completion callback to the kcopyd thread. |
515 | * until after the notify function has completed. | 515 | * |
516 | * Otherwise the client may try and stop the job | 516 | * Some callers assume that all the completions are called |
517 | * after we've completed. | 517 | * from a single thread and don't race with each other. |
518 | * | ||
519 | * We must not call the callback directly here because this | ||
520 | * code may not be executing in the thread. | ||
518 | */ | 521 | */ |
519 | job->fn(read_err, write_err, job->context); | 522 | push(&kc->complete_jobs, job); |
520 | mempool_free(job, job->kc->job_pool); | 523 | wake(kc); |
521 | } | 524 | } |
522 | } | 525 | } |
523 | 526 | ||
@@ -530,6 +533,8 @@ static void split_job(struct kcopyd_job *job) | |||
530 | { | 533 | { |
531 | int i; | 534 | int i; |
532 | 535 | ||
536 | atomic_inc(&job->kc->nr_jobs); | ||
537 | |||
533 | atomic_set(&job->sub_jobs, SPLIT_COUNT); | 538 | atomic_set(&job->sub_jobs, SPLIT_COUNT); |
534 | for (i = 0; i < SPLIT_COUNT; i++) | 539 | for (i = 0; i < SPLIT_COUNT; i++) |
535 | segment_complete(0, 0u, job); | 540 | segment_complete(0, 0u, job); |