greybus: module: fix double free of module
authorGreg Kroah-Hartman <greg@kroah.com>
Wed, 22 Oct 2014 08:38:07 +0000 (16:38 +0800)
committerGreg Kroah-Hartman <greg@kroah.com>
Wed, 22 Oct 2014 08:38:07 +0000 (16:38 +0800)
Also properly clean up all modules when you remove a host driver

Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/staging/greybus/core.c
drivers/staging/greybus/greybus.h

index bc27ad68cc8551db0501519595825b0731c9bf14..480e12bac84a9f1b0bdf267242bfa41db5ab3521 100644 (file)
@@ -190,6 +190,13 @@ err_module:
        greybus_module_release(&gmod->dev);
 }
 
+static void gb_delete_module(struct gb_module *gmod)
+{
+       /* FIXME - tear down interfaces first */
+
+       device_del(&gmod->dev);
+}
+
 void gb_remove_module(struct greybus_host_device *hd, u8 module_id)
 {
        struct gb_module *gmod;
@@ -202,15 +209,18 @@ void gb_remove_module(struct greybus_host_device *hd, u8 module_id)
                }
 
        if (found)
-               greybus_remove_device(gmod);
+               gb_delete_module(gmod);
        else
                dev_err(hd->parent, "module id %d remove error\n", module_id);
 }
 
-void greybus_remove_device(struct gb_module *gmod)
+static void gb_remove_modules(struct greybus_host_device *hd)
 {
-       device_del(&gmod->dev);
-       put_device(&gmod->dev);
+       struct gb_module *gmod, *temp;
+
+       list_for_each_entry_safe(gmod, temp, &hd->modules, links) {
+               gb_delete_module(gmod);
+       }
 }
 
 static DEFINE_MUTEX(hd_mutex);
@@ -248,6 +258,7 @@ EXPORT_SYMBOL_GPL(greybus_create_hd);
 
 void greybus_remove_hd(struct greybus_host_device *hd)
 {
+       gb_remove_modules(hd);
        kref_put_mutex(&hd->kref, free_hd, &hd_mutex);
 }
 EXPORT_SYMBOL_GPL(greybus_remove_hd);
index a4e1f4b2b18e9de57ffd6954c53bd2c72e32a65a..4baa37289334472b267a8085229ed2b095ea2bb8 100644 (file)
@@ -242,8 +242,6 @@ void greybus_deregister(struct greybus_driver *driver);
 
 int greybus_disabled(void);
 
-void greybus_remove_device(struct gb_module *gmod);
-
 /* Internal functions to gb module, move to internal .h file eventually. */
 
 void gb_add_module(struct greybus_host_device *hd, u8 module_id,