diff options
author | Davidlohr Bueso <dave@stgolabs.net> | 2014-09-11 23:40:21 -0400 |
---|---|---|
committer | Paul E. McKenney <paulmck@linux.vnet.ibm.com> | 2014-09-16 16:41:06 -0400 |
commit | d36a7a0d5e8b5bff1671723d733eb61621b0cee4 (patch) | |
tree | 17cbd1c2542a4b587890a593446757606a4a7834 /kernel/torture.c | |
parent | 1e6757a92189278c484799ea98fc69bdc528940e (diff) |
torture: Address race in module cleanup
When performing module cleanups by calling torture_cleanup() the
'torture_type' string in nullified However, callers are not necessarily
done, and might still need to reference the variable. This impacts
both rcutorture and locktorture, causing printing things like:
[ 94.226618] (null)-torture: Stopping lock_torture_writer task
[ 94.226624] (null)-torture: Stopping lock_torture_stats task
Thus delay this operation until the very end of the cleanup process.
The consequence (which shouldn't matter for this kid of program) is,
of course, that we delay the window between rmmod and modprobing,
for instance in module_torture_begin().
Signed-off-by: Davidlohr Bueso <dbueso@suse.de>
Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Diffstat (limited to 'kernel/torture.c')
-rw-r--r-- | kernel/torture.c | 16 |
1 files changed, 13 insertions, 3 deletions
diff --git a/kernel/torture.c b/kernel/torture.c index ede8b25ec1ae..dd70993c266c 100644 --- a/kernel/torture.c +++ b/kernel/torture.c | |||
@@ -633,8 +633,13 @@ EXPORT_SYMBOL_GPL(torture_init_end); | |||
633 | * | 633 | * |
634 | * This must be called before the caller starts shutting down its own | 634 | * This must be called before the caller starts shutting down its own |
635 | * kthreads. | 635 | * kthreads. |
636 | * | ||
637 | * Both torture_cleanup_begin() and torture_cleanup_end() must be paired, | ||
638 | * in order to correctly perform the cleanup. They are separated because | ||
639 | * threads can still need to reference the torture_type type, thus nullify | ||
640 | * only after completing all other relevant calls. | ||
636 | */ | 641 | */ |
637 | bool torture_cleanup(void) | 642 | bool torture_cleanup_begin(void) |
638 | { | 643 | { |
639 | mutex_lock(&fullstop_mutex); | 644 | mutex_lock(&fullstop_mutex); |
640 | if (ACCESS_ONCE(fullstop) == FULLSTOP_SHUTDOWN) { | 645 | if (ACCESS_ONCE(fullstop) == FULLSTOP_SHUTDOWN) { |
@@ -649,12 +654,17 @@ bool torture_cleanup(void) | |||
649 | torture_shuffle_cleanup(); | 654 | torture_shuffle_cleanup(); |
650 | torture_stutter_cleanup(); | 655 | torture_stutter_cleanup(); |
651 | torture_onoff_cleanup(); | 656 | torture_onoff_cleanup(); |
657 | return false; | ||
658 | } | ||
659 | EXPORT_SYMBOL_GPL(torture_cleanup_begin); | ||
660 | |||
661 | void torture_cleanup_end(void) | ||
662 | { | ||
652 | mutex_lock(&fullstop_mutex); | 663 | mutex_lock(&fullstop_mutex); |
653 | torture_type = NULL; | 664 | torture_type = NULL; |
654 | mutex_unlock(&fullstop_mutex); | 665 | mutex_unlock(&fullstop_mutex); |
655 | return false; | ||
656 | } | 666 | } |
657 | EXPORT_SYMBOL_GPL(torture_cleanup); | 667 | EXPORT_SYMBOL_GPL(torture_cleanup_end); |
658 | 668 | ||
659 | /* | 669 | /* |
660 | * Is it time for the current torture test to stop? | 670 | * Is it time for the current torture test to stop? |