2 * Copyright (c) 2015, Linaro Limited
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
14 #include <linux/clk.h>
15 #include <linux/delay.h>
16 #include <linux/device.h>
17 #include <linux/err.h>
18 #include <linux/extcon.h>
19 #include <linux/gpio/consumer.h>
21 #include <linux/module.h>
23 #include <linux/platform_device.h>
24 #include <linux/reboot.h>
25 #include <linux/regulator/consumer.h>
26 #include <linux/reset.h>
27 #include <linux/slab.h>
28 #include <linux/usb.h>
29 #include <linux/usb/ulpi.h>
31 #define HSPHY_AHBBURST 0x0090
32 #define HSPHY_AHBMODE 0x0098
33 #define HSPHY_GENCONFIG 0x009c
34 #define HSPHY_GENCONFIG_2 0x00a0
36 #define HSPHY_USBCMD 0x0140
37 #define HSPHY_ULPI_VIEWPORT 0x0170
38 #define HSPHY_CTRL 0x0240
40 #define HSPHY_TXFIFO_IDLE_FORCE_DIS BIT(4)
41 #define HSPHY_SESS_VLD_CTRL_EN BIT(7)
42 #define HSPHY_POR_ASSERT BIT(0)
43 #define HSPHY_RETEN BIT(1)
45 #define HSPHY_SESS_VLD_CTRL BIT(25)
47 #define ULPI_PWR_CLK_MNG_REG 0x88
48 #define ULPI_PWR_OTG_COMP_DISABLE BIT(0)
50 #define ULPI_MISC_A 0x96
51 #define ULPI_MISC_A_VBUSVLDEXTSEL BIT(1)
52 #define ULPI_MISC_A_VBUSVLDEXT BIT(0)
54 #define HSPHY_3P3_MIN 3050000 /* uV */
55 #define HSPHY_3P3_MAX 3300000 /* uV */
57 #define HSPHY_1P8_MIN 1800000 /* uV */
58 #define HSPHY_1P8_MAX 1800000 /* uV */
60 #define HSPHY_VDD_MIN 5
61 #define HSPHY_VDD_MAX 7
67 struct clk
*iface_clk
;
68 struct regulator_bulk_data regulator
[3];
70 struct reset_control
*phy_reset
;
72 struct extcon_dev
*vbus_edev
;
73 struct notifier_block vbus_notify
;
75 struct gpio_desc
*switch_gpio
;
76 struct notifier_block reboot_notify
;
79 static int phy_8x16_notify_connect(struct usb_phy
*phy
,
80 enum usb_device_speed speed
)
82 struct phy_8x16
*qphy
= container_of(phy
, struct phy_8x16
, phy
);
85 val
= ULPI_MISC_A_VBUSVLDEXTSEL
| ULPI_MISC_A_VBUSVLDEXT
;
86 usb_phy_io_write(&qphy
->phy
, val
, ULPI_SET(ULPI_MISC_A
));
88 val
= readl(qphy
->regs
+ HSPHY_USBCMD
);
89 val
|= HSPHY_SESS_VLD_CTRL
;
90 writel(val
, qphy
->regs
+ HSPHY_USBCMD
);
95 static int phy_8x16_notify_disconnect(struct usb_phy
*phy
,
96 enum usb_device_speed speed
)
98 struct phy_8x16
*qphy
= container_of(phy
, struct phy_8x16
, phy
);
101 val
= ULPI_MISC_A_VBUSVLDEXT
| ULPI_MISC_A_VBUSVLDEXTSEL
;
102 usb_phy_io_write(&qphy
->phy
, val
, ULPI_CLR(ULPI_MISC_A
));
104 val
= readl(qphy
->regs
+ HSPHY_USBCMD
);
105 val
&= ~HSPHY_SESS_VLD_CTRL
;
106 writel(val
, qphy
->regs
+ HSPHY_USBCMD
);
111 static int phy_8x16_vbus_on(struct phy_8x16
*qphy
)
113 phy_8x16_notify_connect(&qphy
->phy
, USB_SPEED_UNKNOWN
);
115 /* Switch D+/D- lines to Device connector */
116 gpiod_set_value_cansleep(qphy
->switch_gpio
, 0);
121 static int phy_8x16_vbus_off(struct phy_8x16
*qphy
)
123 phy_8x16_notify_disconnect(&qphy
->phy
, USB_SPEED_UNKNOWN
);
125 /* Switch D+/D- lines to USB HUB */
126 gpiod_set_value_cansleep(qphy
->switch_gpio
, 1);
131 static int phy_8x16_vbus_notify(struct notifier_block
*nb
, unsigned long event
,
134 struct phy_8x16
*qphy
= container_of(nb
, struct phy_8x16
, vbus_notify
);
137 phy_8x16_vbus_on(qphy
);
139 phy_8x16_vbus_off(qphy
);
144 static int phy_8x16_init(struct usb_phy
*phy
)
146 struct phy_8x16
*qphy
= container_of(phy
, struct phy_8x16
, phy
);
147 u32 val
, init
[] = {0x44, 0x6B, 0x24, 0x13};
148 u32 addr
= ULPI_EXT_VENDOR_SPECIFIC
;
151 for (idx
= 0; idx
< ARRAY_SIZE(init
); idx
++)
152 usb_phy_io_write(phy
, init
[idx
], addr
+ idx
);
154 reset_control_reset(qphy
->phy_reset
);
156 /* Assert USB HSPHY_POR */
157 val
= readl(qphy
->regs
+ HSPHY_CTRL
);
158 val
|= HSPHY_POR_ASSERT
;
159 writel(val
, qphy
->regs
+ HSPHY_CTRL
);
162 * wait for minimum 10 microseconds as suggested in HPG.
163 * Use a slightly larger value since the exact value didn't
164 * work 100% of the time.
166 usleep_range(12, 15);
168 /* Deassert USB HSPHY_POR */
169 val
= readl(qphy
->regs
+ HSPHY_CTRL
);
170 val
&= ~HSPHY_POR_ASSERT
;
171 writel(val
, qphy
->regs
+ HSPHY_CTRL
);
173 usleep_range(10, 15);
175 writel(0x00, qphy
->regs
+ HSPHY_AHBBURST
);
176 writel(0x08, qphy
->regs
+ HSPHY_AHBMODE
);
178 /* workaround for rx buffer collision issue */
179 val
= readl(qphy
->regs
+ HSPHY_GENCONFIG
);
180 val
&= ~HSPHY_TXFIFO_IDLE_FORCE_DIS
;
181 writel(val
, qphy
->regs
+ HSPHY_GENCONFIG
);
183 val
= readl(qphy
->regs
+ HSPHY_GENCONFIG_2
);
184 val
|= HSPHY_SESS_VLD_CTRL_EN
;
185 writel(val
, qphy
->regs
+ HSPHY_GENCONFIG_2
);
187 val
= ULPI_PWR_OTG_COMP_DISABLE
;
188 usb_phy_io_write(phy
, val
, ULPI_SET(ULPI_PWR_CLK_MNG_REG
));
190 state
= extcon_get_cable_state_(qphy
->vbus_edev
, EXTCON_USB
);
192 phy_8x16_vbus_on(qphy
);
194 phy_8x16_vbus_off(qphy
);
196 val
= usb_phy_io_read(&qphy
->phy
, ULPI_FUNC_CTRL
);
197 val
&= ~ULPI_FUNC_CTRL_OPMODE_MASK
;
198 val
|= ULPI_FUNC_CTRL_OPMODE_NORMAL
;
199 usb_phy_io_write(&qphy
->phy
, val
, ULPI_FUNC_CTRL
);
204 static void phy_8x16_shutdown(struct usb_phy
*phy
)
208 /* Put the controller in non-driving mode */
209 val
= usb_phy_io_read(phy
, ULPI_FUNC_CTRL
);
210 val
&= ~ULPI_FUNC_CTRL_OPMODE_MASK
;
211 val
|= ULPI_FUNC_CTRL_OPMODE_NONDRIVING
;
212 usb_phy_io_write(phy
, val
, ULPI_FUNC_CTRL
);
215 static int phy_8x16_read_devicetree(struct phy_8x16
*qphy
)
217 struct device
*dev
= qphy
->phy
.dev
;
220 qphy
->core_clk
= devm_clk_get(dev
, "core");
221 if (IS_ERR(qphy
->core_clk
))
222 return PTR_ERR(qphy
->core_clk
);
224 qphy
->iface_clk
= devm_clk_get(dev
, "iface");
225 if (IS_ERR(qphy
->iface_clk
))
226 return PTR_ERR(qphy
->iface_clk
);
228 qphy
->regulator
[0].supply
= "v3p3";
229 qphy
->regulator
[1].supply
= "v1p8";
230 qphy
->regulator
[2].supply
= "vddcx";
232 ret
= devm_regulator_bulk_get(dev
, ARRAY_SIZE(qphy
->regulator
),
237 qphy
->phy_reset
= devm_reset_control_get(dev
, "phy");
238 if (IS_ERR(qphy
->phy_reset
))
239 return PTR_ERR(qphy
->phy_reset
);
241 qphy
->switch_gpio
= devm_gpiod_get_optional(dev
, "switch",
243 return PTR_ERR_OR_ZERO(qphy
->switch_gpio
);
246 static int phy_8x16_reboot_notify(struct notifier_block
*this,
247 unsigned long code
, void *unused
)
249 struct phy_8x16
*qphy
;
251 qphy
= container_of(this, struct phy_8x16
, reboot_notify
);
254 * Ensure that D+/D- lines are routed to uB connector, so
255 * we could load bootloader/kernel at next reboot_notify
257 gpiod_set_value_cansleep(qphy
->switch_gpio
, 0);
261 static int phy_8x16_probe(struct platform_device
*pdev
)
263 struct phy_8x16
*qphy
;
264 struct resource
*res
;
268 qphy
= devm_kzalloc(&pdev
->dev
, sizeof(*qphy
), GFP_KERNEL
);
272 platform_set_drvdata(pdev
, qphy
);
274 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
278 qphy
->regs
= devm_ioremap(&pdev
->dev
, res
->start
, resource_size(res
));
283 phy
->dev
= &pdev
->dev
;
284 phy
->label
= dev_name(&pdev
->dev
);
285 phy
->init
= phy_8x16_init
;
286 phy
->shutdown
= phy_8x16_shutdown
;
287 phy
->notify_connect
= phy_8x16_notify_connect
;
288 phy
->notify_disconnect
= phy_8x16_notify_disconnect
;
289 phy
->io_priv
= qphy
->regs
+ HSPHY_ULPI_VIEWPORT
;
290 phy
->io_ops
= &ulpi_viewport_access_ops
;
291 phy
->type
= USB_PHY_TYPE_USB2
;
293 ret
= phy_8x16_read_devicetree(qphy
);
297 qphy
->vbus_edev
= extcon_get_edev_by_phandle(phy
->dev
, 0);
298 if (IS_ERR(qphy
->vbus_edev
))
299 return PTR_ERR(qphy
->vbus_edev
);
301 ret
= clk_set_rate(qphy
->core_clk
, INT_MAX
);
303 dev_dbg(phy
->dev
, "Can't boost core clock\n");
305 ret
= clk_prepare_enable(qphy
->core_clk
);
309 ret
= clk_prepare_enable(qphy
->iface_clk
);
313 ret
= regulator_bulk_enable(ARRAY_SIZE(qphy
->regulator
),
318 qphy
->vbus_notify
.notifier_call
= phy_8x16_vbus_notify
;
319 ret
= extcon_register_notifier(qphy
->vbus_edev
, EXTCON_USB
,
324 ret
= usb_add_phy_dev(&qphy
->phy
);
328 qphy
->reboot_notify
.notifier_call
= phy_8x16_reboot_notify
;
329 register_reboot_notifier(&qphy
->reboot_notify
);
334 extcon_unregister_notifier(qphy
->vbus_edev
, EXTCON_USB
,
337 regulator_bulk_disable(ARRAY_SIZE(qphy
->regulator
), qphy
->regulator
);
339 clk_disable_unprepare(qphy
->iface_clk
);
341 clk_disable_unprepare(qphy
->core_clk
);
345 static int phy_8x16_remove(struct platform_device
*pdev
)
347 struct phy_8x16
*qphy
= platform_get_drvdata(pdev
);
349 unregister_reboot_notifier(&qphy
->reboot_notify
);
350 extcon_unregister_notifier(qphy
->vbus_edev
, EXTCON_USB
,
354 * Ensure that D+/D- lines are routed to uB connector, so
355 * we could load bootloader/kernel at next reboot_notify
357 gpiod_set_value_cansleep(qphy
->switch_gpio
, 0);
359 usb_remove_phy(&qphy
->phy
);
361 clk_disable_unprepare(qphy
->iface_clk
);
362 clk_disable_unprepare(qphy
->core_clk
);
363 regulator_bulk_disable(ARRAY_SIZE(qphy
->regulator
), qphy
->regulator
);
367 static const struct of_device_id phy_8x16_dt_match
[] = {
368 { .compatible
= "qcom,usb-8x16-phy" },
371 MODULE_DEVICE_TABLE(of
, phy_8x16_dt_match
);
373 static struct platform_driver phy_8x16_driver
= {
374 .probe
= phy_8x16_probe
,
375 .remove
= phy_8x16_remove
,
377 .name
= "phy-qcom-8x16-usb",
378 .of_match_table
= phy_8x16_dt_match
,
381 module_platform_driver(phy_8x16_driver
);
383 MODULE_LICENSE("GPL v2");
384 MODULE_DESCRIPTION("Qualcomm APQ8016/MSM8916 chipsets USB transceiver driver");