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 gpio_desc
*switch_gpio
;
73 struct notifier_block reboot_notify
;
76 static int phy_8x16_notify_connect(struct usb_phy
*phy
,
77 enum usb_device_speed speed
)
79 struct phy_8x16
*qphy
= container_of(phy
, struct phy_8x16
, phy
);
82 val
= ULPI_MISC_A_VBUSVLDEXTSEL
| ULPI_MISC_A_VBUSVLDEXT
;
83 usb_phy_io_write(&qphy
->phy
, val
, ULPI_SET(ULPI_MISC_A
));
85 val
= readl(qphy
->regs
+ HSPHY_USBCMD
);
86 val
|= HSPHY_SESS_VLD_CTRL
;
87 writel(val
, qphy
->regs
+ HSPHY_USBCMD
);
92 static int phy_8x16_notify_disconnect(struct usb_phy
*phy
,
93 enum usb_device_speed speed
)
95 struct phy_8x16
*qphy
= container_of(phy
, struct phy_8x16
, phy
);
98 val
= ULPI_MISC_A_VBUSVLDEXT
| ULPI_MISC_A_VBUSVLDEXTSEL
;
99 usb_phy_io_write(&qphy
->phy
, val
, ULPI_CLR(ULPI_MISC_A
));
101 val
= readl(qphy
->regs
+ HSPHY_USBCMD
);
102 val
&= ~HSPHY_SESS_VLD_CTRL
;
103 writel(val
, qphy
->regs
+ HSPHY_USBCMD
);
108 static int phy_8x16_vbus_on(struct phy_8x16
*qphy
)
110 phy_8x16_notify_connect(&qphy
->phy
, USB_SPEED_UNKNOWN
);
112 /* Switch D+/D- lines to Device connector */
113 gpiod_set_value_cansleep(qphy
->switch_gpio
, 0);
118 static int phy_8x16_vbus_off(struct phy_8x16
*qphy
)
120 phy_8x16_notify_disconnect(&qphy
->phy
, USB_SPEED_UNKNOWN
);
122 /* Switch D+/D- lines to USB HUB */
123 gpiod_set_value_cansleep(qphy
->switch_gpio
, 1);
128 static int phy_8x16_vbus_notify(struct notifier_block
*nb
, unsigned long event
,
131 struct usb_phy
*usb_phy
= container_of(nb
, struct usb_phy
, vbus_nb
);
132 struct phy_8x16
*qphy
= container_of(usb_phy
, struct phy_8x16
, phy
);
135 phy_8x16_vbus_on(qphy
);
137 phy_8x16_vbus_off(qphy
);
142 static int phy_8x16_init(struct usb_phy
*phy
)
144 struct phy_8x16
*qphy
= container_of(phy
, struct phy_8x16
, phy
);
145 u32 val
, init
[] = {0x44, 0x6B, 0x24, 0x13};
146 u32 addr
= ULPI_EXT_VENDOR_SPECIFIC
;
149 for (idx
= 0; idx
< ARRAY_SIZE(init
); idx
++)
150 usb_phy_io_write(phy
, init
[idx
], addr
+ idx
);
152 reset_control_reset(qphy
->phy_reset
);
154 /* Assert USB HSPHY_POR */
155 val
= readl(qphy
->regs
+ HSPHY_CTRL
);
156 val
|= HSPHY_POR_ASSERT
;
157 writel(val
, qphy
->regs
+ HSPHY_CTRL
);
160 * wait for minimum 10 microseconds as suggested in HPG.
161 * Use a slightly larger value since the exact value didn't
162 * work 100% of the time.
164 usleep_range(12, 15);
166 /* Deassert USB HSPHY_POR */
167 val
= readl(qphy
->regs
+ HSPHY_CTRL
);
168 val
&= ~HSPHY_POR_ASSERT
;
169 writel(val
, qphy
->regs
+ HSPHY_CTRL
);
171 usleep_range(10, 15);
173 writel(0x00, qphy
->regs
+ HSPHY_AHBBURST
);
174 writel(0x08, qphy
->regs
+ HSPHY_AHBMODE
);
176 /* workaround for rx buffer collision issue */
177 val
= readl(qphy
->regs
+ HSPHY_GENCONFIG
);
178 val
&= ~HSPHY_TXFIFO_IDLE_FORCE_DIS
;
179 writel(val
, qphy
->regs
+ HSPHY_GENCONFIG
);
181 val
= readl(qphy
->regs
+ HSPHY_GENCONFIG_2
);
182 val
|= HSPHY_SESS_VLD_CTRL_EN
;
183 writel(val
, qphy
->regs
+ HSPHY_GENCONFIG_2
);
185 val
= ULPI_PWR_OTG_COMP_DISABLE
;
186 usb_phy_io_write(phy
, val
, ULPI_SET(ULPI_PWR_CLK_MNG_REG
));
188 state
= extcon_get_state(qphy
->phy
.edev
, EXTCON_USB
);
190 phy_8x16_vbus_on(qphy
);
192 phy_8x16_vbus_off(qphy
);
194 val
= usb_phy_io_read(&qphy
->phy
, ULPI_FUNC_CTRL
);
195 val
&= ~ULPI_FUNC_CTRL_OPMODE_MASK
;
196 val
|= ULPI_FUNC_CTRL_OPMODE_NORMAL
;
197 usb_phy_io_write(&qphy
->phy
, val
, ULPI_FUNC_CTRL
);
202 static void phy_8x16_shutdown(struct usb_phy
*phy
)
206 /* Put the controller in non-driving mode */
207 val
= usb_phy_io_read(phy
, ULPI_FUNC_CTRL
);
208 val
&= ~ULPI_FUNC_CTRL_OPMODE_MASK
;
209 val
|= ULPI_FUNC_CTRL_OPMODE_NONDRIVING
;
210 usb_phy_io_write(phy
, val
, ULPI_FUNC_CTRL
);
213 static int phy_8x16_read_devicetree(struct phy_8x16
*qphy
)
215 struct device
*dev
= qphy
->phy
.dev
;
218 qphy
->core_clk
= devm_clk_get(dev
, "core");
219 if (IS_ERR(qphy
->core_clk
))
220 return PTR_ERR(qphy
->core_clk
);
222 qphy
->iface_clk
= devm_clk_get(dev
, "iface");
223 if (IS_ERR(qphy
->iface_clk
))
224 return PTR_ERR(qphy
->iface_clk
);
226 qphy
->regulator
[0].supply
= "v3p3";
227 qphy
->regulator
[1].supply
= "v1p8";
228 qphy
->regulator
[2].supply
= "vddcx";
230 ret
= devm_regulator_bulk_get(dev
, ARRAY_SIZE(qphy
->regulator
),
235 qphy
->phy_reset
= devm_reset_control_get(dev
, "phy");
236 if (IS_ERR(qphy
->phy_reset
))
237 return PTR_ERR(qphy
->phy_reset
);
239 qphy
->switch_gpio
= devm_gpiod_get_optional(dev
, "switch",
241 return PTR_ERR_OR_ZERO(qphy
->switch_gpio
);
244 static int phy_8x16_reboot_notify(struct notifier_block
*this,
245 unsigned long code
, void *unused
)
247 struct phy_8x16
*qphy
;
249 qphy
= container_of(this, struct phy_8x16
, reboot_notify
);
252 * Ensure that D+/D- lines are routed to uB connector, so
253 * we could load bootloader/kernel at next reboot_notify
255 gpiod_set_value_cansleep(qphy
->switch_gpio
, 0);
259 static int phy_8x16_probe(struct platform_device
*pdev
)
261 struct phy_8x16
*qphy
;
262 struct resource
*res
;
266 qphy
= devm_kzalloc(&pdev
->dev
, sizeof(*qphy
), GFP_KERNEL
);
270 platform_set_drvdata(pdev
, qphy
);
272 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
276 qphy
->regs
= devm_ioremap(&pdev
->dev
, res
->start
, resource_size(res
));
281 phy
->dev
= &pdev
->dev
;
282 phy
->label
= dev_name(&pdev
->dev
);
283 phy
->init
= phy_8x16_init
;
284 phy
->shutdown
= phy_8x16_shutdown
;
285 phy
->notify_connect
= phy_8x16_notify_connect
;
286 phy
->notify_disconnect
= phy_8x16_notify_disconnect
;
287 phy
->io_priv
= qphy
->regs
+ HSPHY_ULPI_VIEWPORT
;
288 phy
->io_ops
= &ulpi_viewport_access_ops
;
289 phy
->type
= USB_PHY_TYPE_USB2
;
290 phy
->vbus_nb
.notifier_call
= phy_8x16_vbus_notify
;
291 phy
->id_nb
.notifier_call
= NULL
;
293 ret
= phy_8x16_read_devicetree(qphy
);
297 ret
= clk_set_rate(qphy
->core_clk
, INT_MAX
);
299 dev_dbg(phy
->dev
, "Can't boost core clock\n");
301 ret
= clk_prepare_enable(qphy
->core_clk
);
305 ret
= clk_prepare_enable(qphy
->iface_clk
);
309 ret
= regulator_bulk_enable(ARRAY_SIZE(qphy
->regulator
),
314 ret
= usb_add_phy_dev(&qphy
->phy
);
318 qphy
->reboot_notify
.notifier_call
= phy_8x16_reboot_notify
;
319 register_reboot_notifier(&qphy
->reboot_notify
);
324 regulator_bulk_disable(ARRAY_SIZE(qphy
->regulator
), qphy
->regulator
);
326 clk_disable_unprepare(qphy
->iface_clk
);
328 clk_disable_unprepare(qphy
->core_clk
);
332 static int phy_8x16_remove(struct platform_device
*pdev
)
334 struct phy_8x16
*qphy
= platform_get_drvdata(pdev
);
336 unregister_reboot_notifier(&qphy
->reboot_notify
);
339 * Ensure that D+/D- lines are routed to uB connector, so
340 * we could load bootloader/kernel at next reboot_notify
342 gpiod_set_value_cansleep(qphy
->switch_gpio
, 0);
344 usb_remove_phy(&qphy
->phy
);
346 clk_disable_unprepare(qphy
->iface_clk
);
347 clk_disable_unprepare(qphy
->core_clk
);
348 regulator_bulk_disable(ARRAY_SIZE(qphy
->regulator
), qphy
->regulator
);
352 static const struct of_device_id phy_8x16_dt_match
[] = {
353 { .compatible
= "qcom,usb-8x16-phy" },
356 MODULE_DEVICE_TABLE(of
, phy_8x16_dt_match
);
358 static struct platform_driver phy_8x16_driver
= {
359 .probe
= phy_8x16_probe
,
360 .remove
= phy_8x16_remove
,
362 .name
= "phy-qcom-8x16-usb",
363 .of_match_table
= phy_8x16_dt_match
,
366 module_platform_driver(phy_8x16_driver
);
368 MODULE_LICENSE("GPL v2");
369 MODULE_DESCRIPTION("Qualcomm APQ8016/MSM8916 chipsets USB transceiver driver");