usb: roles: Provide the switch drivers handle to the switch in the API
The USB role callback functions had a parameter pointing to the parent device (struct device) of the switch. The assumption was that the switch parent is always the controller. Firstly, that may not be true in every case, and secondly, it prevents us from supporting devices that supply multiple muxes. Changing the first parameter of usb_role_switch_set_t and usb_role_switch_get_t from struct device to struct usb_role_switch. Cc: Peter Chen <Peter.Chen@nxp.com> Cc: Felipe Balbi <balbi@kernel.org> Cc: Chunfeng Yun <chunfeng.yun@mediatek.com> Cc: Bin Liu <b-liu@ti.com> Signed-off-by: Heikki Krogerus <heikki.krogerus@linux.intel.com> Link: https://lore.kernel.org/r/20200302135353.56659-6-heikki.krogerus@linux.intel.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
69af044a77
commit
bce3052f0c
@ -330,9 +330,9 @@ exit:
|
|||||||
*
|
*
|
||||||
* Returns role
|
* Returns role
|
||||||
*/
|
*/
|
||||||
static enum usb_role cdns3_role_get(struct device *dev)
|
static enum usb_role cdns3_role_get(struct usb_role_switch *sw)
|
||||||
{
|
{
|
||||||
struct cdns3 *cdns = dev_get_drvdata(dev);
|
struct cdns3 *cdns = usb_role_switch_get_drvdata(sw);
|
||||||
|
|
||||||
return cdns->role;
|
return cdns->role;
|
||||||
}
|
}
|
||||||
@ -346,9 +346,9 @@ static enum usb_role cdns3_role_get(struct device *dev)
|
|||||||
* - Role switch for dual-role devices
|
* - Role switch for dual-role devices
|
||||||
* - USB_ROLE_GADGET <--> USB_ROLE_NONE for peripheral-only devices
|
* - USB_ROLE_GADGET <--> USB_ROLE_NONE for peripheral-only devices
|
||||||
*/
|
*/
|
||||||
static int cdns3_role_set(struct device *dev, enum usb_role role)
|
static int cdns3_role_set(struct usb_role_switch *sw, enum usb_role role)
|
||||||
{
|
{
|
||||||
struct cdns3 *cdns = dev_get_drvdata(dev);
|
struct cdns3 *cdns = usb_role_switch_get_drvdata(sw);
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
pm_runtime_get_sync(cdns->dev);
|
pm_runtime_get_sync(cdns->dev);
|
||||||
@ -423,12 +423,6 @@ pm_put:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct usb_role_switch_desc cdns3_switch_desc = {
|
|
||||||
.set = cdns3_role_set,
|
|
||||||
.get = cdns3_role_get,
|
|
||||||
.allow_userspace_control = true,
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* cdns3_probe - probe for cdns3 core device
|
* cdns3_probe - probe for cdns3 core device
|
||||||
* @pdev: Pointer to cdns3 core platform device
|
* @pdev: Pointer to cdns3 core platform device
|
||||||
@ -437,6 +431,7 @@ static const struct usb_role_switch_desc cdns3_switch_desc = {
|
|||||||
*/
|
*/
|
||||||
static int cdns3_probe(struct platform_device *pdev)
|
static int cdns3_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
|
struct usb_role_switch_desc sw_desc = { };
|
||||||
struct device *dev = &pdev->dev;
|
struct device *dev = &pdev->dev;
|
||||||
struct resource *res;
|
struct resource *res;
|
||||||
struct cdns3 *cdns;
|
struct cdns3 *cdns;
|
||||||
@ -529,7 +524,12 @@ static int cdns3_probe(struct platform_device *pdev)
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto err3;
|
goto err3;
|
||||||
|
|
||||||
cdns->role_sw = usb_role_switch_register(dev, &cdns3_switch_desc);
|
sw_desc.set = cdns3_role_set;
|
||||||
|
sw_desc.get = cdns3_role_get;
|
||||||
|
sw_desc.allow_userspace_control = true;
|
||||||
|
sw_desc.driver_data = cdns;
|
||||||
|
|
||||||
|
cdns->role_sw = usb_role_switch_register(dev, &sw_desc);
|
||||||
if (IS_ERR(cdns->role_sw)) {
|
if (IS_ERR(cdns->role_sw)) {
|
||||||
ret = PTR_ERR(cdns->role_sw);
|
ret = PTR_ERR(cdns->role_sw);
|
||||||
dev_warn(dev, "Unable to register Role Switch\n");
|
dev_warn(dev, "Unable to register Role Switch\n");
|
||||||
|
@ -600,9 +600,9 @@ static int ci_cable_notifier(struct notifier_block *nb, unsigned long event,
|
|||||||
return NOTIFY_DONE;
|
return NOTIFY_DONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum usb_role ci_usb_role_switch_get(struct device *dev)
|
static enum usb_role ci_usb_role_switch_get(struct usb_role_switch *sw)
|
||||||
{
|
{
|
||||||
struct ci_hdrc *ci = dev_get_drvdata(dev);
|
struct ci_hdrc *ci = usb_role_switch_get_drvdata(sw);
|
||||||
enum usb_role role;
|
enum usb_role role;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
@ -613,9 +613,10 @@ static enum usb_role ci_usb_role_switch_get(struct device *dev)
|
|||||||
return role;
|
return role;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ci_usb_role_switch_set(struct device *dev, enum usb_role role)
|
static int ci_usb_role_switch_set(struct usb_role_switch *sw,
|
||||||
|
enum usb_role role)
|
||||||
{
|
{
|
||||||
struct ci_hdrc *ci = dev_get_drvdata(dev);
|
struct ci_hdrc *ci = usb_role_switch_get_drvdata(sw);
|
||||||
struct ci_hdrc_cable *cable = NULL;
|
struct ci_hdrc_cable *cable = NULL;
|
||||||
enum usb_role current_role = ci_role_to_usb_role(ci);
|
enum usb_role current_role = ci_role_to_usb_role(ci);
|
||||||
enum ci_role ci_role = usb_role_to_ci_role(role);
|
enum ci_role ci_role = usb_role_to_ci_role(role);
|
||||||
@ -1118,6 +1119,7 @@ static int ci_hdrc_probe(struct platform_device *pdev)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (ci_role_switch.fwnode) {
|
if (ci_role_switch.fwnode) {
|
||||||
|
ci_role_switch.driver_data = ci;
|
||||||
ci->role_switch = usb_role_switch_register(dev,
|
ci->role_switch = usb_role_switch_register(dev,
|
||||||
&ci_role_switch);
|
&ci_role_switch);
|
||||||
if (IS_ERR(ci->role_switch)) {
|
if (IS_ERR(ci->role_switch)) {
|
||||||
|
@ -321,9 +321,10 @@ static int dwc3_meson_g12a_otg_mode_set(struct dwc3_meson_g12a *priv,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int dwc3_meson_g12a_role_set(struct device *dev, enum usb_role role)
|
static int dwc3_meson_g12a_role_set(struct usb_role_switch *sw,
|
||||||
|
enum usb_role role)
|
||||||
{
|
{
|
||||||
struct dwc3_meson_g12a *priv = dev_get_drvdata(dev);
|
struct dwc3_meson_g12a *priv = usb_role_switch_get_drvdata(sw);
|
||||||
enum phy_mode mode;
|
enum phy_mode mode;
|
||||||
|
|
||||||
if (role == USB_ROLE_NONE)
|
if (role == USB_ROLE_NONE)
|
||||||
@ -338,9 +339,9 @@ static int dwc3_meson_g12a_role_set(struct device *dev, enum usb_role role)
|
|||||||
return dwc3_meson_g12a_otg_mode_set(priv, mode);
|
return dwc3_meson_g12a_otg_mode_set(priv, mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum usb_role dwc3_meson_g12a_role_get(struct device *dev)
|
static enum usb_role dwc3_meson_g12a_role_get(struct usb_role_switch *sw)
|
||||||
{
|
{
|
||||||
struct dwc3_meson_g12a *priv = dev_get_drvdata(dev);
|
struct dwc3_meson_g12a *priv = usb_role_switch_get_drvdata(sw);
|
||||||
|
|
||||||
return priv->otg_phy_mode == PHY_MODE_USB_HOST ?
|
return priv->otg_phy_mode == PHY_MODE_USB_HOST ?
|
||||||
USB_ROLE_HOST : USB_ROLE_DEVICE;
|
USB_ROLE_HOST : USB_ROLE_DEVICE;
|
||||||
@ -499,6 +500,7 @@ static int dwc3_meson_g12a_probe(struct platform_device *pdev)
|
|||||||
priv->switch_desc.allow_userspace_control = true;
|
priv->switch_desc.allow_userspace_control = true;
|
||||||
priv->switch_desc.set = dwc3_meson_g12a_role_set;
|
priv->switch_desc.set = dwc3_meson_g12a_role_set;
|
||||||
priv->switch_desc.get = dwc3_meson_g12a_role_get;
|
priv->switch_desc.get = dwc3_meson_g12a_role_get;
|
||||||
|
priv->switch_desc.driver_data = priv;
|
||||||
|
|
||||||
priv->role_switch = usb_role_switch_register(dev, &priv->switch_desc);
|
priv->role_switch = usb_role_switch_register(dev, &priv->switch_desc);
|
||||||
if (IS_ERR(priv->role_switch))
|
if (IS_ERR(priv->role_switch))
|
||||||
|
@ -2355,14 +2355,14 @@ static const struct usb_gadget_ops renesas_usb3_gadget_ops = {
|
|||||||
.set_selfpowered = renesas_usb3_set_selfpowered,
|
.set_selfpowered = renesas_usb3_set_selfpowered,
|
||||||
};
|
};
|
||||||
|
|
||||||
static enum usb_role renesas_usb3_role_switch_get(struct device *dev)
|
static enum usb_role renesas_usb3_role_switch_get(struct usb_role_switch *sw)
|
||||||
{
|
{
|
||||||
struct renesas_usb3 *usb3 = dev_get_drvdata(dev);
|
struct renesas_usb3 *usb3 = usb_role_switch_get_drvdata(sw);
|
||||||
enum usb_role cur_role;
|
enum usb_role cur_role;
|
||||||
|
|
||||||
pm_runtime_get_sync(dev);
|
pm_runtime_get_sync(usb3_to_dev(usb3));
|
||||||
cur_role = usb3_is_host(usb3) ? USB_ROLE_HOST : USB_ROLE_DEVICE;
|
cur_role = usb3_is_host(usb3) ? USB_ROLE_HOST : USB_ROLE_DEVICE;
|
||||||
pm_runtime_put(dev);
|
pm_runtime_put(usb3_to_dev(usb3));
|
||||||
|
|
||||||
return cur_role;
|
return cur_role;
|
||||||
}
|
}
|
||||||
@ -2372,7 +2372,7 @@ static void handle_ext_role_switch_states(struct device *dev,
|
|||||||
{
|
{
|
||||||
struct renesas_usb3 *usb3 = dev_get_drvdata(dev);
|
struct renesas_usb3 *usb3 = dev_get_drvdata(dev);
|
||||||
struct device *host = usb3->host_dev;
|
struct device *host = usb3->host_dev;
|
||||||
enum usb_role cur_role = renesas_usb3_role_switch_get(dev);
|
enum usb_role cur_role = renesas_usb3_role_switch_get(usb3->role_sw);
|
||||||
|
|
||||||
switch (role) {
|
switch (role) {
|
||||||
case USB_ROLE_NONE:
|
case USB_ROLE_NONE:
|
||||||
@ -2424,7 +2424,7 @@ static void handle_role_switch_states(struct device *dev,
|
|||||||
{
|
{
|
||||||
struct renesas_usb3 *usb3 = dev_get_drvdata(dev);
|
struct renesas_usb3 *usb3 = dev_get_drvdata(dev);
|
||||||
struct device *host = usb3->host_dev;
|
struct device *host = usb3->host_dev;
|
||||||
enum usb_role cur_role = renesas_usb3_role_switch_get(dev);
|
enum usb_role cur_role = renesas_usb3_role_switch_get(usb3->role_sw);
|
||||||
|
|
||||||
if (cur_role == USB_ROLE_HOST && role == USB_ROLE_DEVICE) {
|
if (cur_role == USB_ROLE_HOST && role == USB_ROLE_DEVICE) {
|
||||||
device_release_driver(host);
|
device_release_driver(host);
|
||||||
@ -2438,19 +2438,19 @@ static void handle_role_switch_states(struct device *dev,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int renesas_usb3_role_switch_set(struct device *dev,
|
static int renesas_usb3_role_switch_set(struct usb_role_switch *sw,
|
||||||
enum usb_role role)
|
enum usb_role role)
|
||||||
{
|
{
|
||||||
struct renesas_usb3 *usb3 = dev_get_drvdata(dev);
|
struct renesas_usb3 *usb3 = usb_role_switch_get_drvdata(sw);
|
||||||
|
|
||||||
pm_runtime_get_sync(dev);
|
pm_runtime_get_sync(usb3_to_dev(usb3));
|
||||||
|
|
||||||
if (usb3->role_sw_by_connector)
|
if (usb3->role_sw_by_connector)
|
||||||
handle_ext_role_switch_states(dev, role);
|
handle_ext_role_switch_states(usb3_to_dev(usb3), role);
|
||||||
else
|
else
|
||||||
handle_role_switch_states(dev, role);
|
handle_role_switch_states(usb3_to_dev(usb3), role);
|
||||||
|
|
||||||
pm_runtime_put(dev);
|
pm_runtime_put(usb3_to_dev(usb3));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -2831,6 +2831,8 @@ static int renesas_usb3_probe(struct platform_device *pdev)
|
|||||||
renesas_usb3_role_switch_desc.fwnode = dev_fwnode(&pdev->dev);
|
renesas_usb3_role_switch_desc.fwnode = dev_fwnode(&pdev->dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
renesas_usb3_role_switch_desc.driver_data = usb3;
|
||||||
|
|
||||||
INIT_WORK(&usb3->role_work, renesas_usb3_role_work);
|
INIT_WORK(&usb3->role_work, renesas_usb3_role_work);
|
||||||
usb3->role_sw = usb_role_switch_register(&pdev->dev,
|
usb3->role_sw = usb_role_switch_register(&pdev->dev,
|
||||||
&renesas_usb3_role_switch_desc);
|
&renesas_usb3_role_switch_desc);
|
||||||
|
@ -676,12 +676,13 @@ static void tegra_xudc_usb_role_sw_work(struct work_struct *work)
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int tegra_xudc_usb_role_sw_set(struct device *dev, enum usb_role role)
|
static int tegra_xudc_usb_role_sw_set(struct usb_role_switch *sw,
|
||||||
|
enum usb_role role)
|
||||||
{
|
{
|
||||||
struct tegra_xudc *xudc = dev_get_drvdata(dev);
|
struct tegra_xudc *xudc = usb_role_switch_get_drvdata(sw);
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
dev_dbg(dev, "%s role is %d\n", __func__, role);
|
dev_dbg(xudc->dev, "%s role is %d\n", __func__, role);
|
||||||
|
|
||||||
spin_lock_irqsave(&xudc->lock, flags);
|
spin_lock_irqsave(&xudc->lock, flags);
|
||||||
|
|
||||||
@ -3590,6 +3591,7 @@ static int tegra_xudc_probe(struct platform_device *pdev)
|
|||||||
if (of_property_read_bool(xudc->dev->of_node, "usb-role-switch")) {
|
if (of_property_read_bool(xudc->dev->of_node, "usb-role-switch")) {
|
||||||
role_sx_desc.set = tegra_xudc_usb_role_sw_set;
|
role_sx_desc.set = tegra_xudc_usb_role_sw_set;
|
||||||
role_sx_desc.fwnode = dev_fwnode(xudc->dev);
|
role_sx_desc.fwnode = dev_fwnode(xudc->dev);
|
||||||
|
role_sx_desc.driver_data = xudc;
|
||||||
|
|
||||||
xudc->usb_role_sw = usb_role_switch_register(xudc->dev,
|
xudc->usb_role_sw = usb_role_switch_register(xudc->dev,
|
||||||
&role_sx_desc);
|
&role_sx_desc);
|
||||||
|
@ -320,9 +320,9 @@ void ssusb_set_force_mode(struct ssusb_mtk *ssusb,
|
|||||||
mtu3_writel(ssusb->ippc_base, SSUSB_U2_CTRL(0), value);
|
mtu3_writel(ssusb->ippc_base, SSUSB_U2_CTRL(0), value);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ssusb_role_sw_set(struct device *dev, enum usb_role role)
|
static int ssusb_role_sw_set(struct usb_role_switch *sw, enum usb_role role)
|
||||||
{
|
{
|
||||||
struct ssusb_mtk *ssusb = dev_get_drvdata(dev);
|
struct ssusb_mtk *ssusb = usb_role_switch_get_drvdata(sw);
|
||||||
bool to_host = false;
|
bool to_host = false;
|
||||||
|
|
||||||
if (role == USB_ROLE_HOST)
|
if (role == USB_ROLE_HOST)
|
||||||
@ -334,9 +334,9 @@ static int ssusb_role_sw_set(struct device *dev, enum usb_role role)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum usb_role ssusb_role_sw_get(struct device *dev)
|
static enum usb_role ssusb_role_sw_get(struct usb_role_switch *sw)
|
||||||
{
|
{
|
||||||
struct ssusb_mtk *ssusb = dev_get_drvdata(dev);
|
struct ssusb_mtk *ssusb = usb_role_switch_get_drvdata(sw);
|
||||||
enum usb_role role;
|
enum usb_role role;
|
||||||
|
|
||||||
role = ssusb->is_host ? USB_ROLE_HOST : USB_ROLE_DEVICE;
|
role = ssusb->is_host ? USB_ROLE_HOST : USB_ROLE_DEVICE;
|
||||||
@ -356,6 +356,7 @@ static int ssusb_role_sw_register(struct otg_switch_mtk *otg_sx)
|
|||||||
role_sx_desc.set = ssusb_role_sw_set;
|
role_sx_desc.set = ssusb_role_sw_set;
|
||||||
role_sx_desc.get = ssusb_role_sw_get;
|
role_sx_desc.get = ssusb_role_sw_get;
|
||||||
role_sx_desc.fwnode = dev_fwnode(ssusb->dev);
|
role_sx_desc.fwnode = dev_fwnode(ssusb->dev);
|
||||||
|
role_sx_desc.driver_data = ssusb;
|
||||||
otg_sx->role_sw = usb_role_switch_register(ssusb->dev, &role_sx_desc);
|
otg_sx->role_sw = usb_role_switch_register(ssusb->dev, &role_sx_desc);
|
||||||
|
|
||||||
return PTR_ERR_OR_ZERO(otg_sx->role_sw);
|
return PTR_ERR_OR_ZERO(otg_sx->role_sw);
|
||||||
|
@ -115,9 +115,8 @@ static void mtk_musb_clks_disable(struct mtk_glue *glue)
|
|||||||
clk_disable_unprepare(glue->main);
|
clk_disable_unprepare(glue->main);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int musb_usb_role_sx_set(struct device *dev, enum usb_role role)
|
static int mtk_otg_switch_set(struct mtk_glue *glue, enum usb_role role)
|
||||||
{
|
{
|
||||||
struct mtk_glue *glue = dev_get_drvdata(dev);
|
|
||||||
struct musb *musb = glue->musb;
|
struct musb *musb = glue->musb;
|
||||||
u8 devctl = readb(musb->mregs + MUSB_DEVCTL);
|
u8 devctl = readb(musb->mregs + MUSB_DEVCTL);
|
||||||
enum usb_role new_role;
|
enum usb_role new_role;
|
||||||
@ -168,9 +167,14 @@ static int musb_usb_role_sx_set(struct device *dev, enum usb_role role)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum usb_role musb_usb_role_sx_get(struct device *dev)
|
static int musb_usb_role_sx_set(struct usb_role_switch *sw, enum usb_role role)
|
||||||
{
|
{
|
||||||
struct mtk_glue *glue = dev_get_drvdata(dev);
|
return mtk_otg_switch_set(usb_role_switch_get_drvdata(sw), role);
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum usb_role musb_usb_role_sx_get(struct usb_role_switch *sw)
|
||||||
|
{
|
||||||
|
struct mtk_glue *glue = usb_role_switch_get_drvdata(sw);
|
||||||
|
|
||||||
return glue->role;
|
return glue->role;
|
||||||
}
|
}
|
||||||
@ -182,6 +186,7 @@ static int mtk_otg_switch_init(struct mtk_glue *glue)
|
|||||||
role_sx_desc.set = musb_usb_role_sx_set;
|
role_sx_desc.set = musb_usb_role_sx_set;
|
||||||
role_sx_desc.get = musb_usb_role_sx_get;
|
role_sx_desc.get = musb_usb_role_sx_get;
|
||||||
role_sx_desc.fwnode = dev_fwnode(glue->dev);
|
role_sx_desc.fwnode = dev_fwnode(glue->dev);
|
||||||
|
role_sx_desc.driver_data = glue;
|
||||||
glue->role_sw = usb_role_switch_register(glue->dev, &role_sx_desc);
|
glue->role_sw = usb_role_switch_register(glue->dev, &role_sx_desc);
|
||||||
|
|
||||||
return PTR_ERR_OR_ZERO(glue->role_sw);
|
return PTR_ERR_OR_ZERO(glue->role_sw);
|
||||||
@ -288,8 +293,7 @@ static int mtk_musb_set_mode(struct musb *musb, u8 mode)
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
glue->role = new_role;
|
mtk_otg_switch_set(glue, new_role);
|
||||||
musb_usb_role_sx_set(dev, glue->role);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,7 +48,7 @@ int usb_role_switch_set_role(struct usb_role_switch *sw, enum usb_role role)
|
|||||||
|
|
||||||
mutex_lock(&sw->lock);
|
mutex_lock(&sw->lock);
|
||||||
|
|
||||||
ret = sw->set(sw->dev.parent, role);
|
ret = sw->set(sw, role);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
sw->role = role;
|
sw->role = role;
|
||||||
|
|
||||||
@ -75,7 +75,7 @@ enum usb_role usb_role_switch_get_role(struct usb_role_switch *sw)
|
|||||||
mutex_lock(&sw->lock);
|
mutex_lock(&sw->lock);
|
||||||
|
|
||||||
if (sw->get)
|
if (sw->get)
|
||||||
role = sw->get(sw->dev.parent);
|
role = sw->get(sw);
|
||||||
else
|
else
|
||||||
role = sw->role;
|
role = sw->role;
|
||||||
|
|
||||||
|
@ -42,6 +42,7 @@
|
|||||||
#define DRV_NAME "intel_xhci_usb_sw"
|
#define DRV_NAME "intel_xhci_usb_sw"
|
||||||
|
|
||||||
struct intel_xhci_usb_data {
|
struct intel_xhci_usb_data {
|
||||||
|
struct device *dev;
|
||||||
struct usb_role_switch *role_sw;
|
struct usb_role_switch *role_sw;
|
||||||
void __iomem *base;
|
void __iomem *base;
|
||||||
bool enable_sw_switch;
|
bool enable_sw_switch;
|
||||||
@ -51,9 +52,10 @@ static const struct software_node intel_xhci_usb_node = {
|
|||||||
"intel-xhci-usb-sw",
|
"intel-xhci-usb-sw",
|
||||||
};
|
};
|
||||||
|
|
||||||
static int intel_xhci_usb_set_role(struct device *dev, enum usb_role role)
|
static int intel_xhci_usb_set_role(struct usb_role_switch *sw,
|
||||||
|
enum usb_role role)
|
||||||
{
|
{
|
||||||
struct intel_xhci_usb_data *data = dev_get_drvdata(dev);
|
struct intel_xhci_usb_data *data = usb_role_switch_get_drvdata(sw);
|
||||||
unsigned long timeout;
|
unsigned long timeout;
|
||||||
acpi_status status;
|
acpi_status status;
|
||||||
u32 glk, val;
|
u32 glk, val;
|
||||||
@ -66,11 +68,11 @@ static int intel_xhci_usb_set_role(struct device *dev, enum usb_role role)
|
|||||||
*/
|
*/
|
||||||
status = acpi_acquire_global_lock(ACPI_WAIT_FOREVER, &glk);
|
status = acpi_acquire_global_lock(ACPI_WAIT_FOREVER, &glk);
|
||||||
if (ACPI_FAILURE(status) && status != AE_NOT_CONFIGURED) {
|
if (ACPI_FAILURE(status) && status != AE_NOT_CONFIGURED) {
|
||||||
dev_err(dev, "Error could not acquire lock\n");
|
dev_err(data->dev, "Error could not acquire lock\n");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
pm_runtime_get_sync(dev);
|
pm_runtime_get_sync(data->dev);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Set idpin value as requested.
|
* Set idpin value as requested.
|
||||||
@ -112,7 +114,7 @@ static int intel_xhci_usb_set_role(struct device *dev, enum usb_role role)
|
|||||||
do {
|
do {
|
||||||
val = readl(data->base + DUAL_ROLE_CFG1);
|
val = readl(data->base + DUAL_ROLE_CFG1);
|
||||||
if (!!(val & HOST_MODE) == (role == USB_ROLE_HOST)) {
|
if (!!(val & HOST_MODE) == (role == USB_ROLE_HOST)) {
|
||||||
pm_runtime_put(dev);
|
pm_runtime_put(data->dev);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -120,21 +122,21 @@ static int intel_xhci_usb_set_role(struct device *dev, enum usb_role role)
|
|||||||
usleep_range(5000, 10000);
|
usleep_range(5000, 10000);
|
||||||
} while (time_before(jiffies, timeout));
|
} while (time_before(jiffies, timeout));
|
||||||
|
|
||||||
pm_runtime_put(dev);
|
pm_runtime_put(data->dev);
|
||||||
|
|
||||||
dev_warn(dev, "Timeout waiting for role-switch\n");
|
dev_warn(data->dev, "Timeout waiting for role-switch\n");
|
||||||
return -ETIMEDOUT;
|
return -ETIMEDOUT;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum usb_role intel_xhci_usb_get_role(struct device *dev)
|
static enum usb_role intel_xhci_usb_get_role(struct usb_role_switch *sw)
|
||||||
{
|
{
|
||||||
struct intel_xhci_usb_data *data = dev_get_drvdata(dev);
|
struct intel_xhci_usb_data *data = usb_role_switch_get_drvdata(sw);
|
||||||
enum usb_role role;
|
enum usb_role role;
|
||||||
u32 val;
|
u32 val;
|
||||||
|
|
||||||
pm_runtime_get_sync(dev);
|
pm_runtime_get_sync(data->dev);
|
||||||
val = readl(data->base + DUAL_ROLE_CFG0);
|
val = readl(data->base + DUAL_ROLE_CFG0);
|
||||||
pm_runtime_put(dev);
|
pm_runtime_put(data->dev);
|
||||||
|
|
||||||
if (!(val & SW_IDPIN))
|
if (!(val & SW_IDPIN))
|
||||||
role = USB_ROLE_HOST;
|
role = USB_ROLE_HOST;
|
||||||
@ -175,7 +177,9 @@ static int intel_xhci_usb_probe(struct platform_device *pdev)
|
|||||||
sw_desc.get = intel_xhci_usb_get_role,
|
sw_desc.get = intel_xhci_usb_get_role,
|
||||||
sw_desc.allow_userspace_control = true,
|
sw_desc.allow_userspace_control = true,
|
||||||
sw_desc.fwnode = software_node_fwnode(&intel_xhci_usb_node);
|
sw_desc.fwnode = software_node_fwnode(&intel_xhci_usb_node);
|
||||||
|
sw_desc.driver_data = data;
|
||||||
|
|
||||||
|
data->dev = dev;
|
||||||
data->enable_sw_switch = !device_property_read_bool(dev,
|
data->enable_sw_switch = !device_property_read_bool(dev,
|
||||||
"sw_switch_disable");
|
"sw_switch_disable");
|
||||||
|
|
||||||
|
@ -13,8 +13,9 @@ enum usb_role {
|
|||||||
USB_ROLE_DEVICE,
|
USB_ROLE_DEVICE,
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef int (*usb_role_switch_set_t)(struct device *dev, enum usb_role role);
|
typedef int (*usb_role_switch_set_t)(struct usb_role_switch *sw,
|
||||||
typedef enum usb_role (*usb_role_switch_get_t)(struct device *dev);
|
enum usb_role role);
|
||||||
|
typedef enum usb_role (*usb_role_switch_get_t)(struct usb_role_switch *sw);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct usb_role_switch_desc - USB Role Switch Descriptor
|
* struct usb_role_switch_desc - USB Role Switch Descriptor
|
||||||
|
Loading…
x
Reference in New Issue
Block a user