1 // SPDX-License-Identifier: GPL-2.0-only
3 * phy-bcm-kona-usb2.c - Broadcom Kona USB2 Phy Driver
5 * Copyright (C) 2013 Linaro Limited
6 * Matt Porter <mporter@linaro.org>
10 #include <linux/delay.h>
11 #include <linux/err.h>
13 #include <linux/module.h>
15 #include <linux/phy/phy.h>
16 #include <linux/platform_device.h>
19 #define OTGCTL_OTGSTAT2 BIT(31)
20 #define OTGCTL_OTGSTAT1 BIT(30)
21 #define OTGCTL_PRST_N_SW BIT(11)
22 #define OTGCTL_HRESET_N BIT(10)
23 #define OTGCTL_UTMI_LINE_STATE1 BIT(9)
24 #define OTGCTL_UTMI_LINE_STATE0 BIT(8)
27 #define P1CTL_SOFT_RESET BIT(1)
28 #define P1CTL_NON_DRIVING BIT(0)
34 static void bcm_kona_usb_phy_power(struct bcm_kona_usb
*phy
, int on
)
38 val
= readl(phy
->regs
+ OTGCTL
);
40 /* Configure and power PHY */
41 val
&= ~(OTGCTL_OTGSTAT2
| OTGCTL_OTGSTAT1
|
42 OTGCTL_UTMI_LINE_STATE1
| OTGCTL_UTMI_LINE_STATE0
);
43 val
|= OTGCTL_PRST_N_SW
| OTGCTL_HRESET_N
;
45 val
&= ~(OTGCTL_PRST_N_SW
| OTGCTL_HRESET_N
);
47 writel(val
, phy
->regs
+ OTGCTL
);
50 static int bcm_kona_usb_phy_init(struct phy
*gphy
)
52 struct bcm_kona_usb
*phy
= phy_get_drvdata(gphy
);
56 val
= readl(phy
->regs
+ P1CTL
);
57 val
&= ~P1CTL_NON_DRIVING
;
58 val
|= P1CTL_SOFT_RESET
;
59 writel(val
, phy
->regs
+ P1CTL
);
60 writel(val
& ~P1CTL_SOFT_RESET
, phy
->regs
+ P1CTL
);
61 /* Reset needs to be asserted for 2ms */
63 writel(val
| P1CTL_SOFT_RESET
, phy
->regs
+ P1CTL
);
68 static int bcm_kona_usb_phy_power_on(struct phy
*gphy
)
70 struct bcm_kona_usb
*phy
= phy_get_drvdata(gphy
);
72 bcm_kona_usb_phy_power(phy
, 1);
77 static int bcm_kona_usb_phy_power_off(struct phy
*gphy
)
79 struct bcm_kona_usb
*phy
= phy_get_drvdata(gphy
);
81 bcm_kona_usb_phy_power(phy
, 0);
86 static const struct phy_ops ops
= {
87 .init
= bcm_kona_usb_phy_init
,
88 .power_on
= bcm_kona_usb_phy_power_on
,
89 .power_off
= bcm_kona_usb_phy_power_off
,
93 static int bcm_kona_usb2_probe(struct platform_device
*pdev
)
95 struct device
*dev
= &pdev
->dev
;
96 struct bcm_kona_usb
*phy
;
98 struct phy_provider
*phy_provider
;
100 phy
= devm_kzalloc(dev
, sizeof(*phy
), GFP_KERNEL
);
104 phy
->regs
= devm_platform_ioremap_resource(pdev
, 0);
105 if (IS_ERR(phy
->regs
))
106 return PTR_ERR(phy
->regs
);
108 platform_set_drvdata(pdev
, phy
);
110 gphy
= devm_phy_create(dev
, NULL
, &ops
);
112 return PTR_ERR(gphy
);
114 /* The Kona PHY supports an 8-bit wide UTMI interface */
115 phy_set_bus_width(gphy
, 8);
117 phy_set_drvdata(gphy
, phy
);
119 phy_provider
= devm_of_phy_provider_register(dev
,
120 of_phy_simple_xlate
);
122 return PTR_ERR_OR_ZERO(phy_provider
);
125 static const struct of_device_id bcm_kona_usb2_dt_ids
[] = {
126 { .compatible
= "brcm,kona-usb2-phy" },
130 MODULE_DEVICE_TABLE(of
, bcm_kona_usb2_dt_ids
);
132 static struct platform_driver bcm_kona_usb2_driver
= {
133 .probe
= bcm_kona_usb2_probe
,
135 .name
= "bcm-kona-usb2",
136 .of_match_table
= bcm_kona_usb2_dt_ids
,
140 module_platform_driver(bcm_kona_usb2_driver
);
142 MODULE_ALIAS("platform:bcm-kona-usb2");
143 MODULE_AUTHOR("Matt Porter <mporter@linaro.org>");
144 MODULE_DESCRIPTION("BCM Kona USB 2.0 PHY driver");
145 MODULE_LICENSE("GPL v2");