123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 |
- // SPDX-License-Identifier: GPL-2.0+
- /*
- * Atheros AR71XX/9XXX USB PHY driver
- *
- * Copyright (C) 2015-2018 Alban Bedel <[email protected]>
- */
- #include <linux/module.h>
- #include <linux/platform_device.h>
- #include <linux/phy/phy.h>
- #include <linux/reset.h>
- struct ath79_usb_phy {
- struct reset_control *reset;
- /* The suspend override logic is inverted, hence the no prefix
- * to make the code a bit easier to understand.
- */
- struct reset_control *no_suspend_override;
- };
- static int ath79_usb_phy_power_on(struct phy *phy)
- {
- struct ath79_usb_phy *priv = phy_get_drvdata(phy);
- int err = 0;
- if (priv->no_suspend_override) {
- err = reset_control_assert(priv->no_suspend_override);
- if (err)
- return err;
- }
- err = reset_control_deassert(priv->reset);
- if (err && priv->no_suspend_override)
- reset_control_deassert(priv->no_suspend_override);
- return err;
- }
- static int ath79_usb_phy_power_off(struct phy *phy)
- {
- struct ath79_usb_phy *priv = phy_get_drvdata(phy);
- int err = 0;
- err = reset_control_assert(priv->reset);
- if (err)
- return err;
- if (priv->no_suspend_override) {
- err = reset_control_deassert(priv->no_suspend_override);
- if (err)
- reset_control_deassert(priv->reset);
- }
- return err;
- }
- static const struct phy_ops ath79_usb_phy_ops = {
- .power_on = ath79_usb_phy_power_on,
- .power_off = ath79_usb_phy_power_off,
- .owner = THIS_MODULE,
- };
- static int ath79_usb_phy_probe(struct platform_device *pdev)
- {
- struct ath79_usb_phy *priv;
- struct phy *phy;
- priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
- if (!priv)
- return -ENOMEM;
- priv->reset = devm_reset_control_get(&pdev->dev, "phy");
- if (IS_ERR(priv->reset))
- return PTR_ERR(priv->reset);
- priv->no_suspend_override = devm_reset_control_get_optional(
- &pdev->dev, "usb-suspend-override");
- if (IS_ERR(priv->no_suspend_override))
- return PTR_ERR(priv->no_suspend_override);
- phy = devm_phy_create(&pdev->dev, NULL, &ath79_usb_phy_ops);
- if (IS_ERR(phy))
- return PTR_ERR(phy);
- phy_set_drvdata(phy, priv);
- return PTR_ERR_OR_ZERO(devm_of_phy_provider_register(
- &pdev->dev, of_phy_simple_xlate));
- }
- static const struct of_device_id ath79_usb_phy_of_match[] = {
- { .compatible = "qca,ar7100-usb-phy" },
- {}
- };
- MODULE_DEVICE_TABLE(of, ath79_usb_phy_of_match);
- static struct platform_driver ath79_usb_phy_driver = {
- .probe = ath79_usb_phy_probe,
- .driver = {
- .of_match_table = ath79_usb_phy_of_match,
- .name = "ath79-usb-phy",
- }
- };
- module_platform_driver(ath79_usb_phy_driver);
- MODULE_DESCRIPTION("ATH79 USB PHY driver");
- MODULE_AUTHOR("Alban Bedel <[email protected]>");
- MODULE_LICENSE("GPL");
|