Commit 2352a732 authored by Viresh Kumar's avatar Viresh Kumar Committed by Greg Kroah-Hartman

greybus: Unregister devices to get them freed

Devices registered with the device-core needs to be freed by calling
device_unregister(). For module we are calling just put_device() and for
bundle, connection and interface we are calling device_del().

All of these are incomplete and so none of them get freed, i.e. the
.release() routine is never called for their devices.

Module being a special case that it needs to maintain a refcount or a
list of interfaces to trace its usage count. I have chosen refcount.

And so once the refcount is zero, we can Unregister the device and
module will get free as well.

Because of this bug in freeing devices, their sysfs directories were not
getting removed properly and after a manifest is parsed with the help of
gbsim, removing modules was creating problems. The sysfs directory
'greybus' wasn't getting removed. And inserting the modules again
resulted in warnings and insmod failure.

WARNING: CPU: 3 PID: 4277 at
/build/buildd/linux-3.13.0/fs/sysfs/dir.c:486
sysfs_warn_dup+0x86/0xa0()
Signed-off-by: default avatarViresh Kumar <viresh.kumar@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@google.com>
parent 045235f1
...@@ -140,7 +140,7 @@ void gb_bundle_destroy(struct gb_interface *intf) ...@@ -140,7 +140,7 @@ void gb_bundle_destroy(struct gb_interface *intf)
list_for_each_entry_safe(bundle, temp, &list, links) { list_for_each_entry_safe(bundle, temp, &list, links) {
list_del(&bundle->links); list_del(&bundle->links);
gb_bundle_connections_exit(bundle); gb_bundle_connections_exit(bundle);
device_del(&bundle->dev); device_unregister(&bundle->dev);
} }
} }
......
...@@ -252,7 +252,7 @@ void gb_connection_destroy(struct gb_connection *connection) ...@@ -252,7 +252,7 @@ void gb_connection_destroy(struct gb_connection *connection)
gb_connection_hd_cport_id_free(connection); gb_connection_hd_cport_id_free(connection);
gb_protocol_put(connection->protocol); gb_protocol_put(connection->protocol);
device_del(&connection->dev); device_unregister(&connection->dev);
} }
int gb_connection_init(struct gb_connection *connection) int gb_connection_init(struct gb_connection *connection)
......
...@@ -173,6 +173,8 @@ static struct gb_interface *gb_interface_create(struct greybus_host_device *hd, ...@@ -173,6 +173,8 @@ static struct gb_interface *gb_interface_create(struct greybus_host_device *hd,
*/ */
static void gb_interface_destroy(struct gb_interface *intf) static void gb_interface_destroy(struct gb_interface *intf)
{ {
struct gb_module *module;
if (WARN_ON(!intf)) if (WARN_ON(!intf))
return; return;
...@@ -184,10 +186,11 @@ static void gb_interface_destroy(struct gb_interface *intf) ...@@ -184,10 +186,11 @@ static void gb_interface_destroy(struct gb_interface *intf)
kfree(intf->product_string); kfree(intf->product_string);
kfree(intf->vendor_string); kfree(intf->vendor_string);
put_device(&intf->module->dev);
/* kref_put(module->hd); */ /* kref_put(module->hd); */
device_del(&intf->dev); module = intf->module;
device_unregister(&intf->dev);
gb_module_remove(module);
} }
/** /**
......
...@@ -101,6 +101,7 @@ static struct gb_module *gb_module_create(struct greybus_host_device *hd, ...@@ -101,6 +101,7 @@ static struct gb_module *gb_module_create(struct greybus_host_device *hd,
return NULL; return NULL;
module->module_id = module_id; module->module_id = module_id;
module->refcount = 1;
module->dev.parent = hd->parent; module->dev.parent = hd->parent;
module->dev.bus = &greybus_bus_type; module->dev.bus = &greybus_bus_type;
module->dev.type = &greybus_module_type; module->dev.type = &greybus_module_type;
...@@ -127,9 +128,20 @@ struct gb_module *gb_module_find_or_create(struct greybus_host_device *hd, ...@@ -127,9 +128,20 @@ struct gb_module *gb_module_find_or_create(struct greybus_host_device *hd,
struct gb_module *module; struct gb_module *module;
module = gb_module_find(module_id); module = gb_module_find(module_id);
if (module) if (module) {
module->refcount++;
return module; return module;
}
return gb_module_create(hd, module_id); return gb_module_create(hd, module_id);
} }
void gb_module_remove(struct gb_module *module)
{
if (!module)
return;
if (!--module->refcount)
device_unregister(&module->dev);
}
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
struct gb_module { struct gb_module {
struct device dev; struct device dev;
u8 module_id; /* Physical location within the Endo */ u8 module_id; /* Physical location within the Endo */
u16 refcount;
}; };
#define to_gb_module(d) container_of(d, struct gb_module, dev) #define to_gb_module(d) container_of(d, struct gb_module, dev)
...@@ -21,6 +22,7 @@ struct greybus_host_device; ...@@ -21,6 +22,7 @@ struct greybus_host_device;
/* Greybus "private" definitions */ /* Greybus "private" definitions */
struct gb_module *gb_module_find_or_create(struct greybus_host_device *hd, struct gb_module *gb_module_find_or_create(struct greybus_host_device *hd,
u8 module_id); u8 module_id);
void gb_module_remove(struct gb_module *module);
#endif /* __MODULE_H */ #endif /* __MODULE_H */
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