blob: 26fee141f6ee6978953d0137d7d2cce0e5f4a9b4 [file] [log] [blame]
Chunfeng Yune09b88c2020-10-16 11:38:39 +08001// SPDX-License-Identifier: GPL-2.0
2/*
3 * Copyright (C) 2016 MediaTek Inc.
4 *
5 * Author: Chunfeng Yun <chunfeng.yun@mediatek.com>
6 */
7
Chunfeng Yune09b88c2020-10-16 11:38:39 +08008#include <dm/lists.h>
9#include <linux/iopoll.h>
10
11#include "mtu3.h"
12#include "mtu3_dr.h"
13
14void ssusb_set_force_mode(struct ssusb_mtk *ssusb,
15 enum mtu3_dr_force_mode mode)
16{
17 u32 value;
18
19 value = mtu3_readl(ssusb->ippc_base, SSUSB_U2_CTRL(0));
20 switch (mode) {
21 case MTU3_DR_FORCE_DEVICE:
22 value |= SSUSB_U2_PORT_FORCE_IDDIG | SSUSB_U2_PORT_RG_IDDIG;
23 break;
24 case MTU3_DR_FORCE_HOST:
25 value |= SSUSB_U2_PORT_FORCE_IDDIG;
26 value &= ~SSUSB_U2_PORT_RG_IDDIG;
27 break;
28 case MTU3_DR_FORCE_NONE:
29 value &= ~(SSUSB_U2_PORT_FORCE_IDDIG | SSUSB_U2_PORT_RG_IDDIG);
30 break;
31 default:
32 return;
33 }
34 mtu3_writel(ssusb->ippc_base, SSUSB_U2_CTRL(0), value);
35}
36
37/* u2-port0 should be powered on and enabled; */
38int ssusb_check_clocks(struct ssusb_mtk *ssusb, u32 ex_clks)
39{
40 void __iomem *ibase = ssusb->ippc_base;
41 u32 value, check_val;
42 int ret;
43
44 check_val = ex_clks | SSUSB_SYS125_RST_B_STS | SSUSB_SYSPLL_STABLE |
45 SSUSB_REF_RST_B_STS;
46
47 ret = readl_poll_timeout(ibase + U3D_SSUSB_IP_PW_STS1, value,
48 ((value & check_val) == check_val), 10000);
49 if (ret) {
50 dev_err(ssusb->dev, "clks of sts1 are not stable!\n");
51 return ret;
52 }
53
54 ret = readl_poll_timeout(ibase + U3D_SSUSB_IP_PW_STS2, value,
55 (value & SSUSB_U2_MAC_SYS_RST_B_STS), 10000);
56 if (ret) {
57 dev_err(ssusb->dev, "mac2 clock is not stable\n");
58 return ret;
59 }
60
61 return 0;
62}
63
64int ssusb_phy_setup(struct ssusb_mtk *ssusb)
65{
66 struct udevice *dev = ssusb->dev;
67 struct phy_bulk *phys = &ssusb->phys;
68 int ret;
69
70 ret = generic_phy_get_bulk(dev, phys);
71 if (ret)
72 return ret;
73
74 ret = generic_phy_init_bulk(phys);
75 if (ret)
76 return ret;
77
78 ret = generic_phy_power_on_bulk(phys);
79 if (ret)
80 generic_phy_exit_bulk(phys);
81
82 return ret;
83}
84
85void ssusb_phy_shutdown(struct ssusb_mtk *ssusb)
86{
87 generic_phy_power_off_bulk(&ssusb->phys);
88 generic_phy_exit_bulk(&ssusb->phys);
89}
90
91static int ssusb_rscs_init(struct ssusb_mtk *ssusb)
92{
93 int ret = 0;
94
95 ret = regulator_set_enable(ssusb->vusb33_supply, true);
96 if (ret < 0 && ret != -ENOSYS) {
97 dev_err(ssusb->dev, "failed to enable vusb33\n");
98 goto vusb33_err;
99 }
100
101 ret = clk_enable_bulk(&ssusb->clks);
102 if (ret)
103 goto clks_err;
104
105 ret = ssusb_phy_setup(ssusb);
106 if (ret) {
107 dev_err(ssusb->dev, "failed to setup phy\n");
108 goto phy_err;
109 }
110
111 return 0;
112
113phy_err:
114 clk_disable_bulk(&ssusb->clks);
115clks_err:
116 regulator_set_enable(ssusb->vusb33_supply, false);
117vusb33_err:
118 return ret;
119}
120
121static void ssusb_rscs_exit(struct ssusb_mtk *ssusb)
122{
123 clk_disable_bulk(&ssusb->clks);
124 regulator_set_enable(ssusb->vusb33_supply, false);
125 ssusb_phy_shutdown(ssusb);
126}
127
128static void ssusb_ip_sw_reset(struct ssusb_mtk *ssusb)
129{
130 /* reset whole ip (xhci & u3d) */
131 mtu3_setbits(ssusb->ippc_base, U3D_SSUSB_IP_PW_CTRL0, SSUSB_IP_SW_RST);
132 udelay(1);
133 mtu3_clrbits(ssusb->ippc_base, U3D_SSUSB_IP_PW_CTRL0, SSUSB_IP_SW_RST);
134}
135
136static int get_ssusb_rscs(struct udevice *dev, struct ssusb_mtk *ssusb)
137{
138 struct udevice *child;
139 int ret;
140
141 ret = device_get_supply_regulator(dev, "vusb33-supply",
142 &ssusb->vusb33_supply);
143 if (ret) /* optional, ignore error */
144 dev_warn(dev, "can't get optional vusb33 %d\n", ret);
145
146 ret = device_get_supply_regulator(dev, "vbus-supply",
147 &ssusb->vbus_supply);
148 if (ret) /* optional, ignore error */
149 dev_warn(dev, "can't get optional vbus regulator %d!\n", ret);
150
151 ret = clk_get_bulk(dev, &ssusb->clks);
152 if (ret) {
153 dev_err(dev, "failed to get clocks %d!\n", ret);
154 return ret;
155 }
156
157 ssusb->ippc_base = devfdt_remap_addr_name(dev, "ippc");
158 if (!ssusb->ippc_base) {
159 dev_err(dev, "error mapping memory for ippc\n");
160 return -ENODEV;
161 }
162
163 ret = device_find_first_child(dev, &child);
164 if (ret || !child) {
165 dev_err(dev, "failed to get child %d!\n", ret);
166 return ret;
167 }
168
169 ssusb->mac_base = devfdt_remap_addr_name(child, "mac");
170 if (!ssusb->mac_base) {
171 dev_err(dev, "error mapping memory for mac\n");
172 return -ENODEV;
173 }
174
Simon Glassf10643c2020-12-19 10:40:14 -0700175 ssusb->dr_mode = usb_get_dr_mode(dev_ofnode(child));
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800176
177 if (ssusb->dr_mode == USB_DR_MODE_UNKNOWN ||
178 ssusb->dr_mode == USB_DR_MODE_OTG)
179 ssusb->dr_mode = USB_DR_MODE_PERIPHERAL;
180
181 if (IS_ENABLED(CONFIG_USB_MTU3_GADGET))
182 ssusb->dr_mode = USB_DR_MODE_PERIPHERAL;
183 else if (IS_ENABLED(CONFIG_USB_MTU3_HOST))
184 ssusb->dr_mode = USB_DR_MODE_HOST;
185
186 dev_info(dev, "dr_mode: %d, ippc: 0x%p, mac: 0x%p\n",
187 ssusb->dr_mode, ssusb->ippc_base, ssusb->mac_base);
188
189 return 0;
190}
191
192static int mtu3_probe(struct udevice *dev)
193{
194 struct ssusb_mtk *ssusb = dev_get_priv(dev);
195 int ret = -ENOMEM;
196
197 ssusb->dev = dev;
198
199 ret = get_ssusb_rscs(dev, ssusb);
200 if (ret)
201 return ret;
202
203 ret = ssusb_rscs_init(ssusb);
204 if (ret)
205 return ret;
206
207 ssusb_ip_sw_reset(ssusb);
208
209 return 0;
210}
211
212static int mtu3_remove(struct udevice *dev)
213{
214 struct ssusb_mtk *ssusb = dev_to_ssusb(dev);
215
216 ssusb_rscs_exit(ssusb);
217 return 0;
218}
219
220static const struct udevice_id ssusb_of_match[] = {
221 {.compatible = "mediatek,ssusb",},
222 {},
223};
224
225#if CONFIG_IS_ENABLED(DM_USB_GADGET)
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800226static int mtu3_gadget_probe(struct udevice *dev)
227{
228 struct ssusb_mtk *ssusb = dev_to_ssusb(dev->parent);
229 struct mtu3 *mtu = dev_get_priv(dev);
230
231 mtu->dev = dev;
232 ssusb->u3d = mtu;
233 return ssusb_gadget_init(ssusb);
234}
235
236static int mtu3_gadget_remove(struct udevice *dev)
237{
238 struct mtu3 *mtu = dev_get_priv(dev);
239
240 ssusb_gadget_exit(mtu->ssusb);
241 return 0;
242}
243
Marek Vasut2c2d7c22024-06-14 02:51:21 +0200244static int mtu3_gadget_handle_interrupts(struct udevice *dev)
245{
246 struct mtu3 *mtu = dev_get_priv(dev);
247
248 mtu3_irq(0, mtu);
249
250 return 0;
251}
252
253static const struct usb_gadget_generic_ops mtu3_gadget_ops = {
254 .handle_interrupts = mtu3_gadget_handle_interrupts,
255};
256
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800257U_BOOT_DRIVER(mtu3_peripheral) = {
258 .name = "mtu3-peripheral",
259 .id = UCLASS_USB_GADGET_GENERIC,
260 .of_match = ssusb_of_match,
Marek Vasut2c2d7c22024-06-14 02:51:21 +0200261 .ops = &mtu3_gadget_ops,
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800262 .probe = mtu3_gadget_probe,
263 .remove = mtu3_gadget_remove,
Simon Glass41575d82020-12-03 16:55:17 -0700264 .priv_auto = sizeof(struct mtu3),
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800265};
266#endif
267
Simon Glass333e4a62021-07-10 21:14:29 -0600268#if defined(CONFIG_SPL_USB_HOST) || \
Simon Glass371dc062024-09-29 19:49:48 -0600269 (!defined(CONFIG_XPL_BUILD) && defined(CONFIG_USB_HOST))
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800270static int mtu3_host_probe(struct udevice *dev)
271{
272 struct ssusb_mtk *ssusb = dev_to_ssusb(dev->parent);
273 struct mtu3_host *u3h = dev_get_priv(dev);
274 struct xhci_hcor *hcor;
275 int rc;
276
277 u3h->dev = dev;
278 ssusb->u3h = u3h;
279 rc = ssusb_host_init(ssusb);
280 if (rc)
281 return rc;
282
283 u3h->ctrl.quirks = XHCI_MTK_HOST;
284 hcor = (struct xhci_hcor *)((uintptr_t)u3h->hcd +
285 HC_LENGTH(xhci_readl(&u3h->hcd->cr_capbase)));
286
287 return xhci_register(dev, u3h->hcd, hcor);
288}
289
290static int mtu3_host_remove(struct udevice *dev)
291{
292 struct mtu3_host *u3h = dev_get_priv(dev);
293
294 xhci_deregister(dev);
295 ssusb_host_exit(u3h->ssusb);
296 return 0;
297}
298
299U_BOOT_DRIVER(mtu3_host) = {
300 .name = "mtu3-host",
301 .id = UCLASS_USB,
302 .of_match = ssusb_of_match,
303 .probe = mtu3_host_probe,
304 .remove = mtu3_host_remove,
Simon Glass41575d82020-12-03 16:55:17 -0700305 .priv_auto = sizeof(struct mtu3_host),
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800306 .ops = &xhci_usb_ops,
307 .flags = DM_FLAG_ALLOC_PRIV_DMA,
308};
309#endif
310
311static int mtu3_glue_bind(struct udevice *parent)
312{
313 struct udevice *dev;
314 enum usb_dr_mode dr_mode;
315 const char *driver;
316 const char *name;
317 ofnode node;
318 int ret;
319
Simon Glassf10643c2020-12-19 10:40:14 -0700320 node = ofnode_by_compatible(dev_ofnode(parent), "mediatek,ssusb");
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800321 if (!ofnode_valid(node))
322 return -ENODEV;
323
324 name = ofnode_get_name(node);
325 dr_mode = usb_get_dr_mode(node);
326
327 switch (dr_mode) {
328#if CONFIG_IS_ENABLED(DM_USB_GADGET)
329 case USB_DR_MODE_PERIPHERAL:
330 case USB_DR_MODE_OTG:
331 dev_dbg(parent, "%s: dr_mode: peripheral\n", __func__);
332 driver = "mtu3-peripheral";
333 break;
334#endif
335
Simon Glass333e4a62021-07-10 21:14:29 -0600336#if defined(CONFIG_SPL_USB_HOST) || \
Simon Glass371dc062024-09-29 19:49:48 -0600337 (!defined(CONFIG_XPL_BUILD) && defined(CONFIG_USB_HOST))
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800338 case USB_DR_MODE_HOST:
339 dev_dbg(parent, "%s: dr_mode: host\n", __func__);
340 driver = "mtu3-host";
341 break;
342#endif
343 default:
344 dev_err(parent, "%s: unsupported dr_mode %d\n",
345 __func__, dr_mode);
346 return -ENODEV;
347 };
348
349 dev_dbg(parent, "%s: node name: %s, driver %s, dr_mode %d\n",
350 __func__, name, driver, dr_mode);
351
352 ret = device_bind_driver_to_node(parent, driver, name, node, &dev);
353 if (ret)
354 dev_err(parent, "%s: not able to bind usb device mode\n",
355 __func__);
356
357 return ret;
358}
359
360static const struct udevice_id mtu3_of_match[] = {
361 {.compatible = "mediatek,mtu3",},
362 {},
363};
364
365U_BOOT_DRIVER(mtu3) = {
366 .name = "mtu3",
367 .id = UCLASS_NOP,
368 .of_match = mtu3_of_match,
369 .bind = mtu3_glue_bind,
370 .probe = mtu3_probe,
371 .remove = mtu3_remove,
Simon Glass41575d82020-12-03 16:55:17 -0700372 .priv_auto = sizeof(struct ssusb_mtk),
Chunfeng Yune09b88c2020-10-16 11:38:39 +0800373};