usb: gadget: composite: Fix corruption when changing configuration
authorBenoit Goby <benoit@android.com>
Wed, 16 May 2012 03:44:33 +0000 (20:44 -0700)
committerBenoit Goby <benoit@android.com>
Tue, 22 May 2012 01:55:25 +0000 (18:55 -0700)
Remove the config from the configs list before releasing the spinlock.
Otherwise the other cpu might be processing a SET_CONFIGURATION that
will switch to the configuration that is being released.

Bug: 6521576
Change-Id: Id4da0d0e18ead63e20cb236cd1d3e8e6d116acce
Signed-off-by: Benoit Goby <benoit@android.com>
drivers/usb/gadget/composite.c

index dc06da669739548c109c8002666f163c624c6aed..1d88a80086a36ea7a587ca20d58f4ecde2831f49 100644 (file)
@@ -564,7 +564,7 @@ done:
        return status;
 }
 
-static int remove_config(struct usb_composite_dev *cdev,
+static int unbind_config(struct usb_composite_dev *cdev,
                              struct usb_configuration *config)
 {
        while (!list_empty(&config->functions)) {
@@ -579,7 +579,6 @@ static int remove_config(struct usb_composite_dev *cdev,
                        /* may free memory for "f" */
                }
        }
-       list_del(&config->list);
        if (config->unbind) {
                DBG(cdev, "unbind config '%s'/%p\n", config->label, config);
                config->unbind(config);
@@ -598,9 +597,11 @@ int usb_remove_config(struct usb_composite_dev *cdev,
        if (cdev->config == config)
                reset_config(cdev);
 
+       list_del(&config->list);
+
        spin_unlock_irqrestore(&cdev->lock, flags);
 
-       return remove_config(cdev, config);
+       return unbind_config(cdev, config);
 }
 
 /*-------------------------------------------------------------------------*/
@@ -1084,7 +1085,8 @@ composite_unbind(struct usb_gadget *gadget)
                struct usb_configuration        *c;
                c = list_first_entry(&cdev->configs,
                                struct usb_configuration, list);
-               remove_config(cdev, c);
+               list_del(&c->list);
+               unbind_config(cdev, c);
        }
        if (composite->unbind)
                composite->unbind(cdev);