Commit b24d90f4 authored by Richard Weinberger's avatar Richard Weinberger Committed by Greg Kroah-Hartman

ubi: fastmap: Cancel work upon detach

commit 6e7d8016 upstream.

Ben Hutchings pointed out that 29b7a6fa ("ubi: fastmap: Don't flush
fastmap work on detach") does not really fix the problem, it just
reduces the risk to hit the race window where fastmap work races against
free()'ing ubi->volumes[].

The correct approach is making sure that no more fastmap work is in
progress before we free ubi data structures.
So we cancel fastmap work right after the ubi background thread is
stopped.
By setting ubi->thread_enabled to zero we make sure that no further work
tries to wake the thread.

Fixes: 29b7a6fa ("ubi: fastmap: Don't flush fastmap work on detach")
Fixes: 74cdaf24 ("UBI: Fastmap: Fix memory leaks while closing the WL sub-system")
Cc: stable@vger.kernel.org
Cc: Ben Hutchings <ben.hutchings@codethink.co.uk>
Cc: Martin Townsend <mtownsend1973@gmail.com>
Signed-off-by: default avatarRichard Weinberger <richard@nod.at>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent db04f92b
...@@ -1082,6 +1082,9 @@ int ubi_detach_mtd_dev(int ubi_num, int anyway) ...@@ -1082,6 +1082,9 @@ int ubi_detach_mtd_dev(int ubi_num, int anyway)
if (ubi->bgt_thread) if (ubi->bgt_thread)
kthread_stop(ubi->bgt_thread); kthread_stop(ubi->bgt_thread);
#ifdef CONFIG_MTD_UBI_FASTMAP
cancel_work_sync(&ubi->fm_work);
#endif
ubi_debugfs_exit_dev(ubi); ubi_debugfs_exit_dev(ubi);
uif_close(ubi); uif_close(ubi);
......
...@@ -1505,6 +1505,7 @@ int ubi_thread(void *u) ...@@ -1505,6 +1505,7 @@ int ubi_thread(void *u)
} }
dbg_wl("background thread \"%s\" is killed", ubi->bgt_name); dbg_wl("background thread \"%s\" is killed", ubi->bgt_name);
ubi->thread_enabled = 0;
return 0; return 0;
} }
...@@ -1514,9 +1515,6 @@ int ubi_thread(void *u) ...@@ -1514,9 +1515,6 @@ int ubi_thread(void *u)
*/ */
static void shutdown_work(struct ubi_device *ubi) static void shutdown_work(struct ubi_device *ubi)
{ {
#ifdef CONFIG_MTD_UBI_FASTMAP
flush_work(&ubi->fm_work);
#endif
while (!list_empty(&ubi->works)) { while (!list_empty(&ubi->works)) {
struct ubi_work *wrk; struct ubi_work *wrk;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment