]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
usb: chipidea: core: convert ci_role_switch to local variable
authorXu Yang <xu.yang_2@nxp.com>
Mon, 27 Apr 2026 07:57:55 +0000 (15:57 +0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 22 May 2026 09:20:56 +0000 (11:20 +0200)
When a system contains multiple USB controllers, the global ci_role_switch
variable may be overwritten by subsequent driver initialization code.

This can cause issues in the following cases:
 - The 2nd ci_hdrc_probe() sees ci_role_switch.fwnode as non-NULL even
   though the "usb-role-switch" property is not present for the controller.
 - When the ci_hdrc device is unbound and bound again, ci_role_switch
   fwnode will not be reassigned, and the old value will be used instead.

Convert ci_role_switch to a local variable to fix these issues.

Fixes: 05559f10ed79 ("usb: chipidea: add role switch class support")
Cc: stable <stable@kernel.org>
Acked-by: Peter Chen <peter.chen@kernel.org>
Reviewed-by: Frank Li <Frank.Li@nxp.com>
Signed-off-by: Xu Yang <xu.yang_2@nxp.com>
Link: https://patch.msgid.link/20260427075755.3611217-1-xu.yang_2@nxp.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/usb/chipidea/core.c

index 7cfabb04a4fb80c6db56ccb430d290dbac61b716..2ab3db3c1015103fdf51677cfcd7c6a73cfaebfd 100644 (file)
@@ -655,12 +655,6 @@ static enum ci_role ci_get_role(struct ci_hdrc *ci)
        return role;
 }
 
-static struct usb_role_switch_desc ci_role_switch = {
-       .set = ci_usb_role_switch_set,
-       .get = ci_usb_role_switch_get,
-       .allow_userspace_control = true,
-};
-
 static int ci_get_platdata(struct device *dev,
                struct ci_hdrc_platform_data *platdata)
 {
@@ -787,9 +781,6 @@ static int ci_get_platdata(struct device *dev,
                        cable->connected = false;
        }
 
-       if (device_property_read_bool(dev, "usb-role-switch"))
-               ci_role_switch.fwnode = dev->fwnode;
-
        platdata->pctl = devm_pinctrl_get(dev);
        if (!IS_ERR(platdata->pctl)) {
                struct pinctrl_state *p;
@@ -1033,6 +1024,7 @@ ATTRIBUTE_GROUPS(ci);
 
 static int ci_hdrc_probe(struct platform_device *pdev)
 {
+       struct usb_role_switch_desc ci_role_switch = {};
        struct device   *dev = &pdev->dev;
        struct ci_hdrc  *ci;
        struct resource *res;
@@ -1179,7 +1171,11 @@ static int ci_hdrc_probe(struct platform_device *pdev)
                }
        }
 
-       if (ci_role_switch.fwnode) {
+       if (device_property_read_bool(dev, "usb-role-switch")) {
+               ci_role_switch.set = ci_usb_role_switch_set;
+               ci_role_switch.get = ci_usb_role_switch_get;
+               ci_role_switch.allow_userspace_control = true;
+               ci_role_switch.fwnode = dev_fwnode(dev);
                ci_role_switch.driver_data = ci;
                ci->role_switch = usb_role_switch_register(dev,
                                        &ci_role_switch);