Skip to content

Commit 6c3e921

Browse files
lunndavem330
authored andcommitted
net: fec: Ensure clocks are enabled while using mdio bus
When a switch is attached to the mdio bus, the mdio bus can be used while the interface is not open. If the IPG clock is not enabled, MDIO reads/writes will simply time out. Add support for runtime PM to control this clock. Enable/disable this clock using runtime PM, with open()/close() and mdio read()/write() function triggering runtime PM operations. Since PM is optional, the IPG clock is enabled at probe and is no longer modified by fec_enet_clk_enable(), thus if PM is not enabled in the kernel, it is guaranteed the clock is running when MDIO operations are performed. Signed-off-by: Andrew Lunn <[email protected]> Acked-by: Fugang Duan <[email protected]> Signed-off-by: David S. Miller <[email protected]>
1 parent cfbfd86 commit 6c3e921

File tree

1 file changed

+75
-13
lines changed

1 file changed

+75
-13
lines changed

drivers/net/ethernet/freescale/fec_main.c

Lines changed: 75 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
#include <linux/module.h>
2525
#include <linux/kernel.h>
2626
#include <linux/string.h>
27+
#include <linux/pm_runtime.h>
2728
#include <linux/ptrace.h>
2829
#include <linux/errno.h>
2930
#include <linux/ioport.h>
@@ -77,6 +78,7 @@ static void fec_enet_itr_coal_init(struct net_device *ndev);
7778
#define FEC_ENET_RAEM_V 0x8
7879
#define FEC_ENET_RAFL_V 0x8
7980
#define FEC_ENET_OPD_V 0xFFF0
81+
#define FEC_MDIO_PM_TIMEOUT 100 /* ms */
8082

8183
static struct platform_device_id fec_devtype[] = {
8284
{
@@ -1767,7 +1769,13 @@ static void fec_enet_adjust_link(struct net_device *ndev)
17671769
static int fec_enet_mdio_read(struct mii_bus *bus, int mii_id, int regnum)
17681770
{
17691771
struct fec_enet_private *fep = bus->priv;
1772+
struct device *dev = &fep->pdev->dev;
17701773
unsigned long time_left;
1774+
int ret = 0;
1775+
1776+
ret = pm_runtime_get_sync(dev);
1777+
if (IS_ERR_VALUE(ret))
1778+
return ret;
17711779

17721780
fep->mii_timeout = 0;
17731781
init_completion(&fep->mdio_done);
@@ -1783,18 +1791,30 @@ static int fec_enet_mdio_read(struct mii_bus *bus, int mii_id, int regnum)
17831791
if (time_left == 0) {
17841792
fep->mii_timeout = 1;
17851793
netdev_err(fep->netdev, "MDIO read timeout\n");
1786-
return -ETIMEDOUT;
1794+
ret = -ETIMEDOUT;
1795+
goto out;
17871796
}
17881797

1789-
/* return value */
1790-
return FEC_MMFR_DATA(readl(fep->hwp + FEC_MII_DATA));
1798+
ret = FEC_MMFR_DATA(readl(fep->hwp + FEC_MII_DATA));
1799+
1800+
out:
1801+
pm_runtime_mark_last_busy(dev);
1802+
pm_runtime_put_autosuspend(dev);
1803+
1804+
return ret;
17911805
}
17921806

17931807
static int fec_enet_mdio_write(struct mii_bus *bus, int mii_id, int regnum,
17941808
u16 value)
17951809
{
17961810
struct fec_enet_private *fep = bus->priv;
1811+
struct device *dev = &fep->pdev->dev;
17971812
unsigned long time_left;
1813+
int ret = 0;
1814+
1815+
ret = pm_runtime_get_sync(dev);
1816+
if (IS_ERR_VALUE(ret))
1817+
return ret;
17981818

17991819
fep->mii_timeout = 0;
18001820
init_completion(&fep->mdio_done);
@@ -1811,10 +1831,13 @@ static int fec_enet_mdio_write(struct mii_bus *bus, int mii_id, int regnum,
18111831
if (time_left == 0) {
18121832
fep->mii_timeout = 1;
18131833
netdev_err(fep->netdev, "MDIO write timeout\n");
1814-
return -ETIMEDOUT;
1834+
ret = -ETIMEDOUT;
18151835
}
18161836

1817-
return 0;
1837+
pm_runtime_mark_last_busy(dev);
1838+
pm_runtime_put_autosuspend(dev);
1839+
1840+
return ret;
18181841
}
18191842

18201843
static int fec_enet_clk_enable(struct net_device *ndev, bool enable)
@@ -1826,9 +1849,6 @@ static int fec_enet_clk_enable(struct net_device *ndev, bool enable)
18261849
ret = clk_prepare_enable(fep->clk_ahb);
18271850
if (ret)
18281851
return ret;
1829-
ret = clk_prepare_enable(fep->clk_ipg);
1830-
if (ret)
1831-
goto failed_clk_ipg;
18321852
if (fep->clk_enet_out) {
18331853
ret = clk_prepare_enable(fep->clk_enet_out);
18341854
if (ret)
@@ -1852,7 +1872,6 @@ static int fec_enet_clk_enable(struct net_device *ndev, bool enable)
18521872
}
18531873
} else {
18541874
clk_disable_unprepare(fep->clk_ahb);
1855-
clk_disable_unprepare(fep->clk_ipg);
18561875
if (fep->clk_enet_out)
18571876
clk_disable_unprepare(fep->clk_enet_out);
18581877
if (fep->clk_ptp) {
@@ -1874,8 +1893,6 @@ static int fec_enet_clk_enable(struct net_device *ndev, bool enable)
18741893
if (fep->clk_enet_out)
18751894
clk_disable_unprepare(fep->clk_enet_out);
18761895
failed_clk_enet_out:
1877-
clk_disable_unprepare(fep->clk_ipg);
1878-
failed_clk_ipg:
18791896
clk_disable_unprepare(fep->clk_ahb);
18801897

18811898
return ret;
@@ -2847,10 +2864,14 @@ fec_enet_open(struct net_device *ndev)
28472864
struct fec_enet_private *fep = netdev_priv(ndev);
28482865
int ret;
28492866

2867+
ret = pm_runtime_get_sync(&fep->pdev->dev);
2868+
if (IS_ERR_VALUE(ret))
2869+
return ret;
2870+
28502871
pinctrl_pm_select_default_state(&fep->pdev->dev);
28512872
ret = fec_enet_clk_enable(ndev, true);
28522873
if (ret)
2853-
return ret;
2874+
goto clk_enable;
28542875

28552876
/* I should reset the ring buffers here, but I don't yet know
28562877
* a simple way to do that.
@@ -2881,6 +2902,9 @@ fec_enet_open(struct net_device *ndev)
28812902
fec_enet_free_buffers(ndev);
28822903
err_enet_alloc:
28832904
fec_enet_clk_enable(ndev, false);
2905+
clk_enable:
2906+
pm_runtime_mark_last_busy(&fep->pdev->dev);
2907+
pm_runtime_put_autosuspend(&fep->pdev->dev);
28842908
pinctrl_pm_select_sleep_state(&fep->pdev->dev);
28852909
return ret;
28862910
}
@@ -2903,6 +2927,9 @@ fec_enet_close(struct net_device *ndev)
29032927

29042928
fec_enet_clk_enable(ndev, false);
29052929
pinctrl_pm_select_sleep_state(&fep->pdev->dev);
2930+
pm_runtime_mark_last_busy(&fep->pdev->dev);
2931+
pm_runtime_put_autosuspend(&fep->pdev->dev);
2932+
29062933
fec_enet_free_buffers(ndev);
29072934

29082935
return 0;
@@ -3388,6 +3415,10 @@ fec_probe(struct platform_device *pdev)
33883415
if (ret)
33893416
goto failed_clk;
33903417

3418+
ret = clk_prepare_enable(fep->clk_ipg);
3419+
if (ret)
3420+
goto failed_clk_ipg;
3421+
33913422
fep->reg_phy = devm_regulator_get(&pdev->dev, "phy");
33923423
if (!IS_ERR(fep->reg_phy)) {
33933424
ret = regulator_enable(fep->reg_phy);
@@ -3434,6 +3465,8 @@ fec_probe(struct platform_device *pdev)
34343465
netif_carrier_off(ndev);
34353466
fec_enet_clk_enable(ndev, false);
34363467
pinctrl_pm_select_sleep_state(&pdev->dev);
3468+
pm_runtime_set_active(&pdev->dev);
3469+
pm_runtime_enable(&pdev->dev);
34373470

34383471
ret = register_netdev(ndev);
34393472
if (ret)
@@ -3447,6 +3480,12 @@ fec_probe(struct platform_device *pdev)
34473480

34483481
fep->rx_copybreak = COPYBREAK_DEFAULT;
34493482
INIT_WORK(&fep->tx_timeout_work, fec_enet_timeout_work);
3483+
3484+
pm_runtime_set_autosuspend_delay(&pdev->dev, FEC_MDIO_PM_TIMEOUT);
3485+
pm_runtime_use_autosuspend(&pdev->dev);
3486+
pm_runtime_mark_last_busy(&pdev->dev);
3487+
pm_runtime_put_autosuspend(&pdev->dev);
3488+
34503489
return 0;
34513490

34523491
failed_register:
@@ -3457,6 +3496,8 @@ fec_probe(struct platform_device *pdev)
34573496
if (fep->reg_phy)
34583497
regulator_disable(fep->reg_phy);
34593498
failed_regulator:
3499+
clk_disable_unprepare(fep->clk_ipg);
3500+
failed_clk_ipg:
34603501
fec_enet_clk_enable(ndev, false);
34613502
failed_clk:
34623503
failed_phy:
@@ -3568,7 +3609,28 @@ static int __maybe_unused fec_resume(struct device *dev)
35683609
return ret;
35693610
}
35703611

3571-
static SIMPLE_DEV_PM_OPS(fec_pm_ops, fec_suspend, fec_resume);
3612+
static int __maybe_unused fec_runtime_suspend(struct device *dev)
3613+
{
3614+
struct net_device *ndev = dev_get_drvdata(dev);
3615+
struct fec_enet_private *fep = netdev_priv(ndev);
3616+
3617+
clk_disable_unprepare(fep->clk_ipg);
3618+
3619+
return 0;
3620+
}
3621+
3622+
static int __maybe_unused fec_runtime_resume(struct device *dev)
3623+
{
3624+
struct net_device *ndev = dev_get_drvdata(dev);
3625+
struct fec_enet_private *fep = netdev_priv(ndev);
3626+
3627+
return clk_prepare_enable(fep->clk_ipg);
3628+
}
3629+
3630+
static const struct dev_pm_ops fec_pm_ops = {
3631+
SET_SYSTEM_SLEEP_PM_OPS(fec_suspend, fec_resume)
3632+
SET_RUNTIME_PM_OPS(fec_runtime_suspend, fec_runtime_resume, NULL)
3633+
};
35723634

35733635
static struct platform_driver fec_driver = {
35743636
.driver = {

0 commit comments

Comments
 (0)