Commit cc9bd53e authored by Johan Hovold's avatar Johan Hovold Committed by Greg Kroah-Hartman

greybus: usb: fix hcd allocation, deregistration and deallocation

Fix allocation, deregistration and deallocation of USB HCD, and update
the hcd_priv helper functions.

The HCD private data was not allocated correctly, something which would
lead to a crash when accessed in hcd_start. The HCD was neither
reregistered or deallocated on connection tear down.
Signed-off-by: default avatarJohan Hovold <johan@hovoldconsulting.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@google.com>
parent 583804f7
...@@ -40,12 +40,19 @@ struct gb_usb_hub_control_response { ...@@ -40,12 +40,19 @@ struct gb_usb_hub_control_response {
struct gb_usb_device { struct gb_usb_device {
struct gb_connection *connection; struct gb_connection *connection;
struct usb_hcd *hcd;
u8 version_major; u8 version_major;
u8 version_minor; u8 version_minor;
}; };
#define to_gb_usb_device(d) ((struct gb_usb_device*) d->hcd_priv) static inline struct gb_usb_device *to_gb_usb_device(struct usb_hcd *hcd)
{
return (struct gb_usb_device *)hcd->hcd_priv;
}
static inline struct usb_hcd *gb_usb_device_to_hcd(struct gb_usb_device *dev)
{
return container_of((void *)dev, struct usb_hcd, hcd_priv);
}
/* Define get_version() routine */ /* Define get_version() routine */
define_get_version(gb_usb_device, USB); define_get_version(gb_usb_device, USB);
...@@ -143,45 +150,44 @@ static int gb_usb_connection_init(struct gb_connection *connection) ...@@ -143,45 +150,44 @@ static int gb_usb_connection_init(struct gb_connection *connection)
{ {
struct device *dev = &connection->dev; struct device *dev = &connection->dev;
struct gb_usb_device *gb_usb_dev; struct gb_usb_device *gb_usb_dev;
struct usb_hcd *hcd;
int retval; int retval;
gb_usb_dev = kzalloc(sizeof(*gb_usb_dev), GFP_KERNEL); hcd = usb_create_hcd(&usb_gb_hc_driver, dev, dev_name(dev));
if (!gb_usb_dev) if (!hcd)
return -ENOMEM; return -ENOMEM;
gb_usb_dev = to_gb_usb_device(hcd);
gb_usb_dev->connection = connection; gb_usb_dev->connection = connection;
connection->private = gb_usb_dev; connection->private = gb_usb_dev;
/* Check for compatible protocol version */ /* Check for compatible protocol version */
retval = get_version(gb_usb_dev); retval = get_version(gb_usb_dev);
if (retval) if (retval)
goto error_create_hcd; goto err_put_hcd;
gb_usb_dev->hcd = usb_create_hcd(&usb_gb_hc_driver, dev, dev_name(dev));
if (!gb_usb_dev->hcd) {
retval = -ENODEV;
goto error_create_hcd;
}
gb_usb_dev->hcd->has_tt = 1; hcd->has_tt = 1;
gb_usb_dev->hcd->hcd_priv[0] = (unsigned long) gb_usb_dev;
retval = usb_add_hcd(gb_usb_dev->hcd, 0, 0); retval = usb_add_hcd(hcd, 0, 0);
if (retval) if (retval)
goto error_add_hcd; goto err_put_hcd;
return 0; return 0;
error_add_hcd:
usb_put_hcd(gb_usb_dev->hcd); err_put_hcd:
error_create_hcd: usb_put_hcd(hcd);
kfree(gb_usb_dev);
return retval; return retval;
} }
static void gb_usb_connection_exit(struct gb_connection *connection) static void gb_usb_connection_exit(struct gb_connection *connection)
{ {
// FIXME - tear everything down! struct gb_usb_device *gb_usb_dev = connection->private;
struct usb_hcd *hcd = gb_usb_device_to_hcd(gb_usb_dev);
usb_remove_hcd(hcd);
usb_put_hcd(hcd);
} }
static struct gb_protocol usb_protocol = { static struct gb_protocol usb_protocol = {
......
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