1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Broadcom specific Advanced Microcontroller Bus
4*4882a593Smuzhiyun * Broadcom USB-core driver (BCMA bus glue)
5*4882a593Smuzhiyun *
6*4882a593Smuzhiyun * Copyright 2011-2015 Hauke Mehrtens <hauke@hauke-m.de>
7*4882a593Smuzhiyun * Copyright 2015 Felix Fietkau <nbd@openwrt.org>
8*4882a593Smuzhiyun *
9*4882a593Smuzhiyun * Based on ssb-ohci driver
10*4882a593Smuzhiyun * Copyright 2007 Michael Buesch <m@bues.ch>
11*4882a593Smuzhiyun *
12*4882a593Smuzhiyun * Derived from the OHCI-PCI driver
13*4882a593Smuzhiyun * Copyright 1999 Roman Weissgaerber
14*4882a593Smuzhiyun * Copyright 2000-2002 David Brownell
15*4882a593Smuzhiyun * Copyright 1999 Linus Torvalds
16*4882a593Smuzhiyun * Copyright 1999 Gregory P. Smith
17*4882a593Smuzhiyun *
18*4882a593Smuzhiyun * Derived from the USBcore related parts of Broadcom-SB
19*4882a593Smuzhiyun * Copyright 2005-2011 Broadcom Corporation
20*4882a593Smuzhiyun */
21*4882a593Smuzhiyun #include <linux/bcma/bcma.h>
22*4882a593Smuzhiyun #include <linux/delay.h>
23*4882a593Smuzhiyun #include <linux/gpio/consumer.h>
24*4882a593Smuzhiyun #include <linux/platform_device.h>
25*4882a593Smuzhiyun #include <linux/module.h>
26*4882a593Smuzhiyun #include <linux/slab.h>
27*4882a593Smuzhiyun #include <linux/of.h>
28*4882a593Smuzhiyun #include <linux/of_gpio.h>
29*4882a593Smuzhiyun #include <linux/of_platform.h>
30*4882a593Smuzhiyun #include <linux/usb/ehci_pdriver.h>
31*4882a593Smuzhiyun #include <linux/usb/ohci_pdriver.h>
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun MODULE_AUTHOR("Hauke Mehrtens");
34*4882a593Smuzhiyun MODULE_DESCRIPTION("Common USB driver for BCMA Bus");
35*4882a593Smuzhiyun MODULE_LICENSE("GPL");
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun /* See BCMA_CLKCTLST_EXTRESREQ and BCMA_CLKCTLST_EXTRESST */
38*4882a593Smuzhiyun #define USB_BCMA_CLKCTLST_USB_CLK_REQ 0x00000100
39*4882a593Smuzhiyun
40*4882a593Smuzhiyun struct bcma_hcd_device {
41*4882a593Smuzhiyun struct bcma_device *core;
42*4882a593Smuzhiyun struct platform_device *ehci_dev;
43*4882a593Smuzhiyun struct platform_device *ohci_dev;
44*4882a593Smuzhiyun struct gpio_desc *gpio_desc;
45*4882a593Smuzhiyun };
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun /* Wait for bitmask in a register to get set or cleared.
48*4882a593Smuzhiyun * timeout is in units of ten-microseconds.
49*4882a593Smuzhiyun */
bcma_wait_bits(struct bcma_device * dev,u16 reg,u32 bitmask,int timeout)50*4882a593Smuzhiyun static int bcma_wait_bits(struct bcma_device *dev, u16 reg, u32 bitmask,
51*4882a593Smuzhiyun int timeout)
52*4882a593Smuzhiyun {
53*4882a593Smuzhiyun int i;
54*4882a593Smuzhiyun u32 val;
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun for (i = 0; i < timeout; i++) {
57*4882a593Smuzhiyun val = bcma_read32(dev, reg);
58*4882a593Smuzhiyun if ((val & bitmask) == bitmask)
59*4882a593Smuzhiyun return 0;
60*4882a593Smuzhiyun udelay(10);
61*4882a593Smuzhiyun }
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun return -ETIMEDOUT;
64*4882a593Smuzhiyun }
65*4882a593Smuzhiyun
bcma_hcd_4716wa(struct bcma_device * dev)66*4882a593Smuzhiyun static void bcma_hcd_4716wa(struct bcma_device *dev)
67*4882a593Smuzhiyun {
68*4882a593Smuzhiyun #ifdef CONFIG_BCMA_DRIVER_MIPS
69*4882a593Smuzhiyun /* Work around for 4716 failures. */
70*4882a593Smuzhiyun if (dev->bus->chipinfo.id == 0x4716) {
71*4882a593Smuzhiyun u32 tmp;
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun tmp = bcma_cpu_clock(&dev->bus->drv_mips);
74*4882a593Smuzhiyun if (tmp >= 480000000)
75*4882a593Smuzhiyun tmp = 0x1846b; /* set CDR to 0x11(fast) */
76*4882a593Smuzhiyun else if (tmp == 453000000)
77*4882a593Smuzhiyun tmp = 0x1046b; /* set CDR to 0x10(slow) */
78*4882a593Smuzhiyun else
79*4882a593Smuzhiyun tmp = 0;
80*4882a593Smuzhiyun
81*4882a593Smuzhiyun /* Change Shim mdio control reg to fix host not acking at
82*4882a593Smuzhiyun * high frequencies
83*4882a593Smuzhiyun */
84*4882a593Smuzhiyun if (tmp) {
85*4882a593Smuzhiyun bcma_write32(dev, 0x524, 0x1); /* write sel to enable */
86*4882a593Smuzhiyun udelay(500);
87*4882a593Smuzhiyun
88*4882a593Smuzhiyun bcma_write32(dev, 0x524, tmp);
89*4882a593Smuzhiyun udelay(500);
90*4882a593Smuzhiyun bcma_write32(dev, 0x524, 0x4ab);
91*4882a593Smuzhiyun udelay(500);
92*4882a593Smuzhiyun bcma_read32(dev, 0x528);
93*4882a593Smuzhiyun bcma_write32(dev, 0x528, 0x80000000);
94*4882a593Smuzhiyun }
95*4882a593Smuzhiyun }
96*4882a593Smuzhiyun #endif /* CONFIG_BCMA_DRIVER_MIPS */
97*4882a593Smuzhiyun }
98*4882a593Smuzhiyun
99*4882a593Smuzhiyun /* based on arch/mips/brcm-boards/bcm947xx/pcibios.c */
bcma_hcd_init_chip_mips(struct bcma_device * dev)100*4882a593Smuzhiyun static void bcma_hcd_init_chip_mips(struct bcma_device *dev)
101*4882a593Smuzhiyun {
102*4882a593Smuzhiyun u32 tmp;
103*4882a593Smuzhiyun
104*4882a593Smuzhiyun /*
105*4882a593Smuzhiyun * USB 2.0 special considerations:
106*4882a593Smuzhiyun *
107*4882a593Smuzhiyun * 1. Since the core supports both OHCI and EHCI functions, it must
108*4882a593Smuzhiyun * only be reset once.
109*4882a593Smuzhiyun *
110*4882a593Smuzhiyun * 2. In addition to the standard SI reset sequence, the Host Control
111*4882a593Smuzhiyun * Register must be programmed to bring the USB core and various
112*4882a593Smuzhiyun * phy components out of reset.
113*4882a593Smuzhiyun */
114*4882a593Smuzhiyun if (!bcma_core_is_enabled(dev)) {
115*4882a593Smuzhiyun bcma_core_enable(dev, 0);
116*4882a593Smuzhiyun mdelay(10);
117*4882a593Smuzhiyun if (dev->id.rev >= 5) {
118*4882a593Smuzhiyun /* Enable Misc PLL */
119*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x1e0);
120*4882a593Smuzhiyun tmp |= 0x100;
121*4882a593Smuzhiyun bcma_write32(dev, 0x1e0, tmp);
122*4882a593Smuzhiyun if (bcma_wait_bits(dev, 0x1e0, 1 << 24, 100))
123*4882a593Smuzhiyun printk(KERN_EMERG "Failed to enable misc PPL!\n");
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun /* Take out of resets */
126*4882a593Smuzhiyun bcma_write32(dev, 0x200, 0x4ff);
127*4882a593Smuzhiyun udelay(25);
128*4882a593Smuzhiyun bcma_write32(dev, 0x200, 0x6ff);
129*4882a593Smuzhiyun udelay(25);
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun /* Make sure digital and AFE are locked in USB PHY */
132*4882a593Smuzhiyun bcma_write32(dev, 0x524, 0x6b);
133*4882a593Smuzhiyun udelay(50);
134*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x524);
135*4882a593Smuzhiyun udelay(50);
136*4882a593Smuzhiyun bcma_write32(dev, 0x524, 0xab);
137*4882a593Smuzhiyun udelay(50);
138*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x524);
139*4882a593Smuzhiyun udelay(50);
140*4882a593Smuzhiyun bcma_write32(dev, 0x524, 0x2b);
141*4882a593Smuzhiyun udelay(50);
142*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x524);
143*4882a593Smuzhiyun udelay(50);
144*4882a593Smuzhiyun bcma_write32(dev, 0x524, 0x10ab);
145*4882a593Smuzhiyun udelay(50);
146*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x524);
147*4882a593Smuzhiyun
148*4882a593Smuzhiyun if (bcma_wait_bits(dev, 0x528, 0xc000, 10000)) {
149*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x528);
150*4882a593Smuzhiyun printk(KERN_EMERG
151*4882a593Smuzhiyun "USB20H mdio_rddata 0x%08x\n", tmp);
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun bcma_write32(dev, 0x528, 0x80000000);
154*4882a593Smuzhiyun tmp = bcma_read32(dev, 0x314);
155*4882a593Smuzhiyun udelay(265);
156*4882a593Smuzhiyun bcma_write32(dev, 0x200, 0x7ff);
157*4882a593Smuzhiyun udelay(10);
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun /* Take USB and HSIC out of non-driving modes */
160*4882a593Smuzhiyun bcma_write32(dev, 0x510, 0);
161*4882a593Smuzhiyun } else {
162*4882a593Smuzhiyun bcma_write32(dev, 0x200, 0x7ff);
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun udelay(1);
165*4882a593Smuzhiyun }
166*4882a593Smuzhiyun
167*4882a593Smuzhiyun bcma_hcd_4716wa(dev);
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun }
170*4882a593Smuzhiyun
171*4882a593Smuzhiyun /*
172*4882a593Smuzhiyun * bcma_hcd_usb20_old_arm_init - Initialize old USB 2.0 controller on ARM
173*4882a593Smuzhiyun *
174*4882a593Smuzhiyun * Old USB 2.0 core is identified as BCMA_CORE_USB20_HOST and was introduced
175*4882a593Smuzhiyun * long before Northstar devices. It seems some cheaper chipsets like BCM53573
176*4882a593Smuzhiyun * still use it.
177*4882a593Smuzhiyun * Initialization of this old core differs between MIPS and ARM.
178*4882a593Smuzhiyun */
bcma_hcd_usb20_old_arm_init(struct bcma_hcd_device * usb_dev)179*4882a593Smuzhiyun static int bcma_hcd_usb20_old_arm_init(struct bcma_hcd_device *usb_dev)
180*4882a593Smuzhiyun {
181*4882a593Smuzhiyun struct bcma_device *core = usb_dev->core;
182*4882a593Smuzhiyun struct device *dev = &core->dev;
183*4882a593Smuzhiyun struct bcma_device *pmu_core;
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun usleep_range(10000, 20000);
186*4882a593Smuzhiyun if (core->id.rev < 5)
187*4882a593Smuzhiyun return 0;
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun pmu_core = bcma_find_core(core->bus, BCMA_CORE_PMU);
190*4882a593Smuzhiyun if (!pmu_core) {
191*4882a593Smuzhiyun dev_err(dev, "Could not find PMU core\n");
192*4882a593Smuzhiyun return -ENOENT;
193*4882a593Smuzhiyun }
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun /* Take USB core out of reset */
196*4882a593Smuzhiyun bcma_awrite32(core, BCMA_IOCTL, BCMA_IOCTL_CLK | BCMA_IOCTL_FGC);
197*4882a593Smuzhiyun usleep_range(100, 200);
198*4882a593Smuzhiyun bcma_awrite32(core, BCMA_RESET_CTL, BCMA_RESET_CTL_RESET);
199*4882a593Smuzhiyun usleep_range(100, 200);
200*4882a593Smuzhiyun bcma_awrite32(core, BCMA_RESET_CTL, 0);
201*4882a593Smuzhiyun usleep_range(100, 200);
202*4882a593Smuzhiyun bcma_awrite32(core, BCMA_IOCTL, BCMA_IOCTL_CLK);
203*4882a593Smuzhiyun usleep_range(100, 200);
204*4882a593Smuzhiyun
205*4882a593Smuzhiyun /* Enable Misc PLL */
206*4882a593Smuzhiyun bcma_write32(core, BCMA_CLKCTLST, BCMA_CLKCTLST_FORCEHT |
207*4882a593Smuzhiyun BCMA_CLKCTLST_HQCLKREQ |
208*4882a593Smuzhiyun USB_BCMA_CLKCTLST_USB_CLK_REQ);
209*4882a593Smuzhiyun usleep_range(100, 200);
210*4882a593Smuzhiyun
211*4882a593Smuzhiyun bcma_write32(core, 0x510, 0xc7f85000);
212*4882a593Smuzhiyun bcma_write32(core, 0x510, 0xc7f85003);
213*4882a593Smuzhiyun usleep_range(300, 600);
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun /* Program USB PHY PLL parameters */
216*4882a593Smuzhiyun bcma_write32(pmu_core, BCMA_CC_PMU_PLLCTL_ADDR, 0x6);
217*4882a593Smuzhiyun bcma_write32(pmu_core, BCMA_CC_PMU_PLLCTL_DATA, 0x005360c1);
218*4882a593Smuzhiyun usleep_range(100, 200);
219*4882a593Smuzhiyun bcma_write32(pmu_core, BCMA_CC_PMU_PLLCTL_ADDR, 0x7);
220*4882a593Smuzhiyun bcma_write32(pmu_core, BCMA_CC_PMU_PLLCTL_DATA, 0x0);
221*4882a593Smuzhiyun usleep_range(100, 200);
222*4882a593Smuzhiyun bcma_set32(pmu_core, BCMA_CC_PMU_CTL, BCMA_CC_PMU_CTL_PLL_UPD);
223*4882a593Smuzhiyun usleep_range(100, 200);
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun bcma_write32(core, 0x510, 0x7f8d007);
226*4882a593Smuzhiyun udelay(1000);
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun /* Take controller out of reset */
229*4882a593Smuzhiyun bcma_write32(core, 0x200, 0x4ff);
230*4882a593Smuzhiyun usleep_range(25, 50);
231*4882a593Smuzhiyun bcma_write32(core, 0x200, 0x6ff);
232*4882a593Smuzhiyun usleep_range(25, 50);
233*4882a593Smuzhiyun bcma_write32(core, 0x200, 0x7ff);
234*4882a593Smuzhiyun usleep_range(25, 50);
235*4882a593Smuzhiyun
236*4882a593Smuzhiyun of_platform_default_populate(dev->of_node, NULL, dev);
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun return 0;
239*4882a593Smuzhiyun }
240*4882a593Smuzhiyun
bcma_hcd_usb20_ns_init_hc(struct bcma_device * dev)241*4882a593Smuzhiyun static void bcma_hcd_usb20_ns_init_hc(struct bcma_device *dev)
242*4882a593Smuzhiyun {
243*4882a593Smuzhiyun u32 val;
244*4882a593Smuzhiyun
245*4882a593Smuzhiyun /* Set packet buffer OUT threshold */
246*4882a593Smuzhiyun val = bcma_read32(dev, 0x94);
247*4882a593Smuzhiyun val &= 0xffff;
248*4882a593Smuzhiyun val |= 0x80 << 16;
249*4882a593Smuzhiyun bcma_write32(dev, 0x94, val);
250*4882a593Smuzhiyun
251*4882a593Smuzhiyun /* Enable break memory transfer */
252*4882a593Smuzhiyun val = bcma_read32(dev, 0x9c);
253*4882a593Smuzhiyun val |= 1;
254*4882a593Smuzhiyun bcma_write32(dev, 0x9c, val);
255*4882a593Smuzhiyun
256*4882a593Smuzhiyun /*
257*4882a593Smuzhiyun * Broadcom initializes PHY and then waits to ensure HC is ready to be
258*4882a593Smuzhiyun * configured. In our case the order is reversed. We just initialized
259*4882a593Smuzhiyun * controller and we let HCD initialize PHY, so let's wait (sleep) now.
260*4882a593Smuzhiyun */
261*4882a593Smuzhiyun usleep_range(1000, 2000);
262*4882a593Smuzhiyun }
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun /*
265*4882a593Smuzhiyun * bcma_hcd_usb20_ns_init - Initialize Northstar USB 2.0 controller
266*4882a593Smuzhiyun */
bcma_hcd_usb20_ns_init(struct bcma_hcd_device * bcma_hcd)267*4882a593Smuzhiyun static int bcma_hcd_usb20_ns_init(struct bcma_hcd_device *bcma_hcd)
268*4882a593Smuzhiyun {
269*4882a593Smuzhiyun struct bcma_device *core = bcma_hcd->core;
270*4882a593Smuzhiyun struct bcma_chipinfo *ci = &core->bus->chipinfo;
271*4882a593Smuzhiyun struct device *dev = &core->dev;
272*4882a593Smuzhiyun
273*4882a593Smuzhiyun bcma_core_enable(core, 0);
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun if (ci->id == BCMA_CHIP_ID_BCM4707 ||
276*4882a593Smuzhiyun ci->id == BCMA_CHIP_ID_BCM53018)
277*4882a593Smuzhiyun bcma_hcd_usb20_ns_init_hc(core);
278*4882a593Smuzhiyun
279*4882a593Smuzhiyun of_platform_default_populate(dev->of_node, NULL, dev);
280*4882a593Smuzhiyun
281*4882a593Smuzhiyun return 0;
282*4882a593Smuzhiyun }
283*4882a593Smuzhiyun
bcma_hci_platform_power_gpio(struct bcma_device * dev,bool val)284*4882a593Smuzhiyun static void bcma_hci_platform_power_gpio(struct bcma_device *dev, bool val)
285*4882a593Smuzhiyun {
286*4882a593Smuzhiyun struct bcma_hcd_device *usb_dev = bcma_get_drvdata(dev);
287*4882a593Smuzhiyun
288*4882a593Smuzhiyun if (!usb_dev->gpio_desc)
289*4882a593Smuzhiyun return;
290*4882a593Smuzhiyun
291*4882a593Smuzhiyun gpiod_set_value(usb_dev->gpio_desc, val);
292*4882a593Smuzhiyun }
293*4882a593Smuzhiyun
294*4882a593Smuzhiyun static const struct usb_ehci_pdata ehci_pdata = {
295*4882a593Smuzhiyun };
296*4882a593Smuzhiyun
297*4882a593Smuzhiyun static const struct usb_ohci_pdata ohci_pdata = {
298*4882a593Smuzhiyun };
299*4882a593Smuzhiyun
bcma_hcd_create_pdev(struct bcma_device * dev,const char * name,u32 addr,const void * data,size_t size)300*4882a593Smuzhiyun static struct platform_device *bcma_hcd_create_pdev(struct bcma_device *dev,
301*4882a593Smuzhiyun const char *name, u32 addr,
302*4882a593Smuzhiyun const void *data,
303*4882a593Smuzhiyun size_t size)
304*4882a593Smuzhiyun {
305*4882a593Smuzhiyun struct platform_device *hci_dev;
306*4882a593Smuzhiyun struct resource hci_res[2];
307*4882a593Smuzhiyun int ret;
308*4882a593Smuzhiyun
309*4882a593Smuzhiyun memset(hci_res, 0, sizeof(hci_res));
310*4882a593Smuzhiyun
311*4882a593Smuzhiyun hci_res[0].start = addr;
312*4882a593Smuzhiyun hci_res[0].end = hci_res[0].start + 0x1000 - 1;
313*4882a593Smuzhiyun hci_res[0].flags = IORESOURCE_MEM;
314*4882a593Smuzhiyun
315*4882a593Smuzhiyun hci_res[1].start = dev->irq;
316*4882a593Smuzhiyun hci_res[1].flags = IORESOURCE_IRQ;
317*4882a593Smuzhiyun
318*4882a593Smuzhiyun hci_dev = platform_device_alloc(name, 0);
319*4882a593Smuzhiyun if (!hci_dev)
320*4882a593Smuzhiyun return ERR_PTR(-ENOMEM);
321*4882a593Smuzhiyun
322*4882a593Smuzhiyun hci_dev->dev.parent = &dev->dev;
323*4882a593Smuzhiyun hci_dev->dev.dma_mask = &hci_dev->dev.coherent_dma_mask;
324*4882a593Smuzhiyun
325*4882a593Smuzhiyun ret = platform_device_add_resources(hci_dev, hci_res,
326*4882a593Smuzhiyun ARRAY_SIZE(hci_res));
327*4882a593Smuzhiyun if (ret)
328*4882a593Smuzhiyun goto err_alloc;
329*4882a593Smuzhiyun if (data)
330*4882a593Smuzhiyun ret = platform_device_add_data(hci_dev, data, size);
331*4882a593Smuzhiyun if (ret)
332*4882a593Smuzhiyun goto err_alloc;
333*4882a593Smuzhiyun ret = platform_device_add(hci_dev);
334*4882a593Smuzhiyun if (ret)
335*4882a593Smuzhiyun goto err_alloc;
336*4882a593Smuzhiyun
337*4882a593Smuzhiyun return hci_dev;
338*4882a593Smuzhiyun
339*4882a593Smuzhiyun err_alloc:
340*4882a593Smuzhiyun platform_device_put(hci_dev);
341*4882a593Smuzhiyun return ERR_PTR(ret);
342*4882a593Smuzhiyun }
343*4882a593Smuzhiyun
bcma_hcd_usb20_init(struct bcma_hcd_device * usb_dev)344*4882a593Smuzhiyun static int bcma_hcd_usb20_init(struct bcma_hcd_device *usb_dev)
345*4882a593Smuzhiyun {
346*4882a593Smuzhiyun struct bcma_device *dev = usb_dev->core;
347*4882a593Smuzhiyun struct bcma_chipinfo *chipinfo = &dev->bus->chipinfo;
348*4882a593Smuzhiyun u32 ohci_addr;
349*4882a593Smuzhiyun int err;
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun if (dma_set_mask_and_coherent(dev->dma_dev, DMA_BIT_MASK(32)))
352*4882a593Smuzhiyun return -EOPNOTSUPP;
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun bcma_hcd_init_chip_mips(dev);
355*4882a593Smuzhiyun
356*4882a593Smuzhiyun /* In AI chips EHCI is addrspace 0, OHCI is 1 */
357*4882a593Smuzhiyun ohci_addr = dev->addr_s[0];
358*4882a593Smuzhiyun if ((chipinfo->id == BCMA_CHIP_ID_BCM5357 ||
359*4882a593Smuzhiyun chipinfo->id == BCMA_CHIP_ID_BCM4749)
360*4882a593Smuzhiyun && chipinfo->rev == 0)
361*4882a593Smuzhiyun ohci_addr = 0x18009000;
362*4882a593Smuzhiyun
363*4882a593Smuzhiyun usb_dev->ohci_dev = bcma_hcd_create_pdev(dev, "ohci-platform",
364*4882a593Smuzhiyun ohci_addr, &ohci_pdata,
365*4882a593Smuzhiyun sizeof(ohci_pdata));
366*4882a593Smuzhiyun if (IS_ERR(usb_dev->ohci_dev))
367*4882a593Smuzhiyun return PTR_ERR(usb_dev->ohci_dev);
368*4882a593Smuzhiyun
369*4882a593Smuzhiyun usb_dev->ehci_dev = bcma_hcd_create_pdev(dev, "ehci-platform",
370*4882a593Smuzhiyun dev->addr, &ehci_pdata,
371*4882a593Smuzhiyun sizeof(ehci_pdata));
372*4882a593Smuzhiyun if (IS_ERR(usb_dev->ehci_dev)) {
373*4882a593Smuzhiyun err = PTR_ERR(usb_dev->ehci_dev);
374*4882a593Smuzhiyun goto err_unregister_ohci_dev;
375*4882a593Smuzhiyun }
376*4882a593Smuzhiyun
377*4882a593Smuzhiyun return 0;
378*4882a593Smuzhiyun
379*4882a593Smuzhiyun err_unregister_ohci_dev:
380*4882a593Smuzhiyun platform_device_unregister(usb_dev->ohci_dev);
381*4882a593Smuzhiyun return err;
382*4882a593Smuzhiyun }
383*4882a593Smuzhiyun
bcma_hcd_usb30_init(struct bcma_hcd_device * bcma_hcd)384*4882a593Smuzhiyun static int bcma_hcd_usb30_init(struct bcma_hcd_device *bcma_hcd)
385*4882a593Smuzhiyun {
386*4882a593Smuzhiyun struct bcma_device *core = bcma_hcd->core;
387*4882a593Smuzhiyun struct device *dev = &core->dev;
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun bcma_core_enable(core, 0);
390*4882a593Smuzhiyun
391*4882a593Smuzhiyun of_platform_default_populate(dev->of_node, NULL, dev);
392*4882a593Smuzhiyun
393*4882a593Smuzhiyun return 0;
394*4882a593Smuzhiyun }
395*4882a593Smuzhiyun
bcma_hcd_probe(struct bcma_device * core)396*4882a593Smuzhiyun static int bcma_hcd_probe(struct bcma_device *core)
397*4882a593Smuzhiyun {
398*4882a593Smuzhiyun int err;
399*4882a593Smuzhiyun struct bcma_hcd_device *usb_dev;
400*4882a593Smuzhiyun
401*4882a593Smuzhiyun /* TODO: Probably need checks here; is the core connected? */
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun usb_dev = devm_kzalloc(&core->dev, sizeof(struct bcma_hcd_device),
404*4882a593Smuzhiyun GFP_KERNEL);
405*4882a593Smuzhiyun if (!usb_dev)
406*4882a593Smuzhiyun return -ENOMEM;
407*4882a593Smuzhiyun usb_dev->core = core;
408*4882a593Smuzhiyun
409*4882a593Smuzhiyun usb_dev->gpio_desc = devm_gpiod_get_optional(&core->dev, "vcc",
410*4882a593Smuzhiyun GPIOD_OUT_HIGH);
411*4882a593Smuzhiyun if (IS_ERR(usb_dev->gpio_desc))
412*4882a593Smuzhiyun return dev_err_probe(&core->dev, PTR_ERR(usb_dev->gpio_desc),
413*4882a593Smuzhiyun "error obtaining VCC GPIO");
414*4882a593Smuzhiyun
415*4882a593Smuzhiyun switch (core->id.id) {
416*4882a593Smuzhiyun case BCMA_CORE_USB20_HOST:
417*4882a593Smuzhiyun if (IS_ENABLED(CONFIG_ARM))
418*4882a593Smuzhiyun err = bcma_hcd_usb20_old_arm_init(usb_dev);
419*4882a593Smuzhiyun else if (IS_ENABLED(CONFIG_MIPS))
420*4882a593Smuzhiyun err = bcma_hcd_usb20_init(usb_dev);
421*4882a593Smuzhiyun else
422*4882a593Smuzhiyun err = -ENOTSUPP;
423*4882a593Smuzhiyun break;
424*4882a593Smuzhiyun case BCMA_CORE_NS_USB20:
425*4882a593Smuzhiyun err = bcma_hcd_usb20_ns_init(usb_dev);
426*4882a593Smuzhiyun break;
427*4882a593Smuzhiyun case BCMA_CORE_NS_USB30:
428*4882a593Smuzhiyun err = bcma_hcd_usb30_init(usb_dev);
429*4882a593Smuzhiyun break;
430*4882a593Smuzhiyun default:
431*4882a593Smuzhiyun return -ENODEV;
432*4882a593Smuzhiyun }
433*4882a593Smuzhiyun if (err)
434*4882a593Smuzhiyun return err;
435*4882a593Smuzhiyun
436*4882a593Smuzhiyun bcma_set_drvdata(core, usb_dev);
437*4882a593Smuzhiyun return 0;
438*4882a593Smuzhiyun }
439*4882a593Smuzhiyun
bcma_hcd_remove(struct bcma_device * dev)440*4882a593Smuzhiyun static void bcma_hcd_remove(struct bcma_device *dev)
441*4882a593Smuzhiyun {
442*4882a593Smuzhiyun struct bcma_hcd_device *usb_dev = bcma_get_drvdata(dev);
443*4882a593Smuzhiyun struct platform_device *ohci_dev = usb_dev->ohci_dev;
444*4882a593Smuzhiyun struct platform_device *ehci_dev = usb_dev->ehci_dev;
445*4882a593Smuzhiyun
446*4882a593Smuzhiyun if (ohci_dev)
447*4882a593Smuzhiyun platform_device_unregister(ohci_dev);
448*4882a593Smuzhiyun if (ehci_dev)
449*4882a593Smuzhiyun platform_device_unregister(ehci_dev);
450*4882a593Smuzhiyun
451*4882a593Smuzhiyun bcma_core_disable(dev, 0);
452*4882a593Smuzhiyun }
453*4882a593Smuzhiyun
bcma_hcd_shutdown(struct bcma_device * dev)454*4882a593Smuzhiyun static void bcma_hcd_shutdown(struct bcma_device *dev)
455*4882a593Smuzhiyun {
456*4882a593Smuzhiyun bcma_hci_platform_power_gpio(dev, false);
457*4882a593Smuzhiyun bcma_core_disable(dev, 0);
458*4882a593Smuzhiyun }
459*4882a593Smuzhiyun
460*4882a593Smuzhiyun #ifdef CONFIG_PM
461*4882a593Smuzhiyun
bcma_hcd_suspend(struct bcma_device * dev)462*4882a593Smuzhiyun static int bcma_hcd_suspend(struct bcma_device *dev)
463*4882a593Smuzhiyun {
464*4882a593Smuzhiyun bcma_hci_platform_power_gpio(dev, false);
465*4882a593Smuzhiyun bcma_core_disable(dev, 0);
466*4882a593Smuzhiyun
467*4882a593Smuzhiyun return 0;
468*4882a593Smuzhiyun }
469*4882a593Smuzhiyun
bcma_hcd_resume(struct bcma_device * dev)470*4882a593Smuzhiyun static int bcma_hcd_resume(struct bcma_device *dev)
471*4882a593Smuzhiyun {
472*4882a593Smuzhiyun bcma_hci_platform_power_gpio(dev, true);
473*4882a593Smuzhiyun bcma_core_enable(dev, 0);
474*4882a593Smuzhiyun
475*4882a593Smuzhiyun return 0;
476*4882a593Smuzhiyun }
477*4882a593Smuzhiyun
478*4882a593Smuzhiyun #else /* !CONFIG_PM */
479*4882a593Smuzhiyun #define bcma_hcd_suspend NULL
480*4882a593Smuzhiyun #define bcma_hcd_resume NULL
481*4882a593Smuzhiyun #endif /* CONFIG_PM */
482*4882a593Smuzhiyun
483*4882a593Smuzhiyun static const struct bcma_device_id bcma_hcd_table[] = {
484*4882a593Smuzhiyun BCMA_CORE(BCMA_MANUF_BCM, BCMA_CORE_USB20_HOST, BCMA_ANY_REV, BCMA_ANY_CLASS),
485*4882a593Smuzhiyun BCMA_CORE(BCMA_MANUF_BCM, BCMA_CORE_NS_USB20, BCMA_ANY_REV, BCMA_ANY_CLASS),
486*4882a593Smuzhiyun BCMA_CORE(BCMA_MANUF_BCM, BCMA_CORE_NS_USB30, BCMA_ANY_REV, BCMA_ANY_CLASS),
487*4882a593Smuzhiyun {},
488*4882a593Smuzhiyun };
489*4882a593Smuzhiyun MODULE_DEVICE_TABLE(bcma, bcma_hcd_table);
490*4882a593Smuzhiyun
491*4882a593Smuzhiyun static struct bcma_driver bcma_hcd_driver = {
492*4882a593Smuzhiyun .name = KBUILD_MODNAME,
493*4882a593Smuzhiyun .id_table = bcma_hcd_table,
494*4882a593Smuzhiyun .probe = bcma_hcd_probe,
495*4882a593Smuzhiyun .remove = bcma_hcd_remove,
496*4882a593Smuzhiyun .shutdown = bcma_hcd_shutdown,
497*4882a593Smuzhiyun .suspend = bcma_hcd_suspend,
498*4882a593Smuzhiyun .resume = bcma_hcd_resume,
499*4882a593Smuzhiyun };
500*4882a593Smuzhiyun module_bcma_driver(bcma_hcd_driver);
501