1 // SPDX-License-Identifier: GPL-2.0+
3 * aspeed-vhub -- Driver for Aspeed SoC "vHub" USB gadget
5 * core.c - Top level support
7 * Copyright 2017 IBM Corporation
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
15 #include <linux/kernel.h>
16 #include <linux/module.h>
17 #include <linux/platform_device.h>
18 #include <linux/delay.h>
19 #include <linux/ioport.h>
20 #include <linux/slab.h>
21 #include <linux/errno.h>
22 #include <linux/list.h>
23 #include <linux/interrupt.h>
24 #include <linux/proc_fs.h>
25 #include <linux/prefetch.h>
26 #include <linux/clk.h>
27 #include <linux/usb/gadget.h>
29 #include <linux/of_gpio.h>
30 #include <linux/regmap.h>
31 #include <linux/dma-mapping.h>
35 void ast_vhub_done(struct ast_vhub_ep
*ep
, struct ast_vhub_req
*req
,
38 bool internal
= req
->internal
;
40 EPVDBG(ep
, "completing request @%p, status %d\n", req
, status
);
42 list_del_init(&req
->queue
);
44 if (req
->req
.status
== -EINPROGRESS
)
45 req
->req
.status
= status
;
48 if (!WARN_ON(!ep
->dev
))
49 usb_gadget_unmap_request(&ep
->dev
->gadget
,
50 &req
->req
, ep
->epn
.is_in
);
55 * If this isn't an internal EP0 request, call the core
56 * to call the gadget completion.
59 spin_unlock(&ep
->vhub
->lock
);
60 usb_gadget_giveback_request(&ep
->ep
, &req
->req
);
61 spin_lock(&ep
->vhub
->lock
);
65 void ast_vhub_nuke(struct ast_vhub_ep
*ep
, int status
)
67 struct ast_vhub_req
*req
;
69 EPDBG(ep
, "Nuking\n");
71 /* Beware, lock will be dropped & req-acquired by done() */
72 while (!list_empty(&ep
->queue
)) {
73 req
= list_first_entry(&ep
->queue
, struct ast_vhub_req
, queue
);
74 ast_vhub_done(ep
, req
, status
);
78 struct usb_request
*ast_vhub_alloc_request(struct usb_ep
*u_ep
,
81 struct ast_vhub_req
*req
;
83 req
= kzalloc(sizeof(*req
), gfp_flags
);
89 void ast_vhub_free_request(struct usb_ep
*u_ep
, struct usb_request
*u_req
)
91 struct ast_vhub_req
*req
= to_ast_req(u_req
);
96 static irqreturn_t
ast_vhub_irq(int irq
, void *data
)
98 struct ast_vhub
*vhub
= data
;
99 irqreturn_t iret
= IRQ_NONE
;
102 /* Stale interrupt while tearing down */
106 spin_lock(&vhub
->lock
);
108 /* Read and ACK interrupts */
109 istat
= readl(vhub
->regs
+ AST_VHUB_ISR
);
112 writel(istat
, vhub
->regs
+ AST_VHUB_ISR
);
115 UDCVDBG(vhub
, "irq status=%08x, ep_acks=%08x ep_nacks=%08x\n",
117 readl(vhub
->regs
+ AST_VHUB_EP_ACK_ISR
),
118 readl(vhub
->regs
+ AST_VHUB_EP_NACK_ISR
));
120 /* Handle generic EPs first */
121 if (istat
& VHUB_IRQ_EP_POOL_ACK_STALL
) {
122 u32 i
, ep_acks
= readl(vhub
->regs
+ AST_VHUB_EP_ACK_ISR
);
123 writel(ep_acks
, vhub
->regs
+ AST_VHUB_EP_ACK_ISR
);
125 for (i
= 0; ep_acks
&& i
< AST_VHUB_NUM_GEN_EPs
; i
++) {
126 u32 mask
= VHUB_EP_IRQ(i
);
127 if (ep_acks
& mask
) {
128 ast_vhub_epn_ack_irq(&vhub
->epns
[i
]);
134 /* Handle device interrupts */
135 if (istat
& (VHUB_IRQ_DEVICE1
|
140 if (istat
& VHUB_IRQ_DEVICE1
)
141 ast_vhub_dev_irq(&vhub
->ports
[0].dev
);
142 if (istat
& VHUB_IRQ_DEVICE2
)
143 ast_vhub_dev_irq(&vhub
->ports
[1].dev
);
144 if (istat
& VHUB_IRQ_DEVICE3
)
145 ast_vhub_dev_irq(&vhub
->ports
[2].dev
);
146 if (istat
& VHUB_IRQ_DEVICE4
)
147 ast_vhub_dev_irq(&vhub
->ports
[3].dev
);
148 if (istat
& VHUB_IRQ_DEVICE5
)
149 ast_vhub_dev_irq(&vhub
->ports
[4].dev
);
152 /* Handle top-level vHub EP0 interrupts */
153 if (istat
& (VHUB_IRQ_HUB_EP0_OUT_ACK_STALL
|
154 VHUB_IRQ_HUB_EP0_IN_ACK_STALL
|
155 VHUB_IRQ_HUB_EP0_SETUP
)) {
156 if (istat
& VHUB_IRQ_HUB_EP0_IN_ACK_STALL
)
157 ast_vhub_ep0_handle_ack(&vhub
->ep0
, true);
158 if (istat
& VHUB_IRQ_HUB_EP0_OUT_ACK_STALL
)
159 ast_vhub_ep0_handle_ack(&vhub
->ep0
, false);
160 if (istat
& VHUB_IRQ_HUB_EP0_SETUP
)
161 ast_vhub_ep0_handle_setup(&vhub
->ep0
);
164 /* Various top level bus events */
165 if (istat
& (VHUB_IRQ_BUS_RESUME
|
166 VHUB_IRQ_BUS_SUSPEND
|
167 VHUB_IRQ_BUS_RESET
)) {
168 if (istat
& VHUB_IRQ_BUS_RESUME
)
169 ast_vhub_hub_resume(vhub
);
170 if (istat
& VHUB_IRQ_BUS_SUSPEND
)
171 ast_vhub_hub_suspend(vhub
);
172 if (istat
& VHUB_IRQ_BUS_RESET
)
173 ast_vhub_hub_reset(vhub
);
177 spin_unlock(&vhub
->lock
);
181 void ast_vhub_init_hw(struct ast_vhub
*vhub
)
185 UDCDBG(vhub
,"(Re)Starting HW ...\n");
188 ctrl
= VHUB_CTRL_PHY_CLK
|
189 VHUB_CTRL_PHY_RESET_DIS
;
192 * We do *NOT* set the VHUB_CTRL_CLK_STOP_SUSPEND bit
193 * to stop the logic clock during suspend because
194 * it causes the registers to become inaccessible and
195 * we haven't yet figured out a good wayt to bring the
196 * controller back into life to issue a wakeup.
200 * Set some ISO & split control bits according to Aspeed
203 * VHUB_CTRL_ISO_RSP_CTRL: When set tells the HW to respond
204 * with 0 bytes data packet to ISO IN endpoints when no data
207 * VHUB_CTRL_SPLIT_IN: This makes a SOF complete a split IN
210 ctrl
|= VHUB_CTRL_ISO_RSP_CTRL
| VHUB_CTRL_SPLIT_IN
;
211 writel(ctrl
, vhub
->regs
+ AST_VHUB_CTRL
);
214 /* Set descriptor ring size */
215 if (AST_VHUB_DESCS_COUNT
== 256) {
216 ctrl
|= VHUB_CTRL_LONG_DESC
;
217 writel(ctrl
, vhub
->regs
+ AST_VHUB_CTRL
);
219 BUILD_BUG_ON(AST_VHUB_DESCS_COUNT
!= 32);
222 /* Reset all devices */
223 writel(VHUB_SW_RESET_ALL
, vhub
->regs
+ AST_VHUB_SW_RESET
);
225 writel(0, vhub
->regs
+ AST_VHUB_SW_RESET
);
227 /* Disable and cleanup EP ACK/NACK interrupts */
228 writel(0, vhub
->regs
+ AST_VHUB_EP_ACK_IER
);
229 writel(0, vhub
->regs
+ AST_VHUB_EP_NACK_IER
);
230 writel(VHUB_EP_IRQ_ALL
, vhub
->regs
+ AST_VHUB_EP_ACK_ISR
);
231 writel(VHUB_EP_IRQ_ALL
, vhub
->regs
+ AST_VHUB_EP_NACK_ISR
);
233 /* Default settings for EP0, enable HW hub EP1 */
234 writel(0, vhub
->regs
+ AST_VHUB_EP0_CTRL
);
235 writel(VHUB_EP1_CTRL_RESET_TOGGLE
|
236 VHUB_EP1_CTRL_ENABLE
,
237 vhub
->regs
+ AST_VHUB_EP1_CTRL
);
238 writel(0, vhub
->regs
+ AST_VHUB_EP1_STS_CHG
);
240 /* Configure EP0 DMA buffer */
241 writel(vhub
->ep0
.buf_dma
, vhub
->regs
+ AST_VHUB_EP0_DATA
);
244 writel(0, vhub
->regs
+ AST_VHUB_CONF
);
246 /* Pullup hub (activate on host) */
247 if (vhub
->force_usb1
)
248 ctrl
|= VHUB_CTRL_FULL_SPEED_ONLY
;
250 ctrl
|= VHUB_CTRL_UPSTREAM_CONNECT
;
251 writel(ctrl
, vhub
->regs
+ AST_VHUB_CTRL
);
253 /* Enable some interrupts */
254 writel(VHUB_IRQ_HUB_EP0_IN_ACK_STALL
|
255 VHUB_IRQ_HUB_EP0_OUT_ACK_STALL
|
256 VHUB_IRQ_HUB_EP0_SETUP
|
257 VHUB_IRQ_EP_POOL_ACK_STALL
|
258 VHUB_IRQ_BUS_RESUME
|
259 VHUB_IRQ_BUS_SUSPEND
|
261 vhub
->regs
+ AST_VHUB_IER
);
264 static int ast_vhub_remove(struct platform_device
*pdev
)
266 struct ast_vhub
*vhub
= platform_get_drvdata(pdev
);
270 if (!vhub
|| !vhub
->regs
)
274 for (i
= 0; i
< AST_VHUB_NUM_PORTS
; i
++)
275 ast_vhub_del_dev(&vhub
->ports
[i
].dev
);
277 spin_lock_irqsave(&vhub
->lock
, flags
);
279 /* Mask & ack all interrupts */
280 writel(0, vhub
->regs
+ AST_VHUB_IER
);
281 writel(VHUB_IRQ_ACK_ALL
, vhub
->regs
+ AST_VHUB_ISR
);
283 /* Pull device, leave PHY enabled */
284 writel(VHUB_CTRL_PHY_CLK
|
285 VHUB_CTRL_PHY_RESET_DIS
,
286 vhub
->regs
+ AST_VHUB_CTRL
);
289 clk_disable_unprepare(vhub
->clk
);
291 spin_unlock_irqrestore(&vhub
->lock
, flags
);
294 dma_free_coherent(&pdev
->dev
,
295 AST_VHUB_EP0_MAX_PACKET
*
296 (AST_VHUB_NUM_PORTS
+ 1),
299 vhub
->ep0_bufs
= NULL
;
304 static int ast_vhub_probe(struct platform_device
*pdev
)
306 enum usb_device_speed max_speed
;
307 struct ast_vhub
*vhub
;
308 struct resource
*res
;
311 vhub
= devm_kzalloc(&pdev
->dev
, sizeof(*vhub
), GFP_KERNEL
);
315 spin_lock_init(&vhub
->lock
);
318 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
319 vhub
->regs
= devm_ioremap_resource(&pdev
->dev
, res
);
320 if (IS_ERR(vhub
->regs
)) {
321 dev_err(&pdev
->dev
, "Failed to map resources\n");
322 return PTR_ERR(vhub
->regs
);
324 UDCDBG(vhub
, "vHub@%pR mapped @%p\n", res
, vhub
->regs
);
326 platform_set_drvdata(pdev
, vhub
);
328 vhub
->clk
= devm_clk_get(&pdev
->dev
, NULL
);
329 if (IS_ERR(vhub
->clk
)) {
330 rc
= PTR_ERR(vhub
->clk
);
333 rc
= clk_prepare_enable(vhub
->clk
);
335 dev_err(&pdev
->dev
, "Error couldn't enable clock (%d)\n", rc
);
339 /* Check if we need to limit the HW to USB1 */
340 max_speed
= usb_get_maximum_speed(&pdev
->dev
);
341 if (max_speed
!= USB_SPEED_UNKNOWN
&& max_speed
< USB_SPEED_HIGH
)
342 vhub
->force_usb1
= true;
344 /* Mask & ack all interrupts before installing the handler */
345 writel(0, vhub
->regs
+ AST_VHUB_IER
);
346 writel(VHUB_IRQ_ACK_ALL
, vhub
->regs
+ AST_VHUB_ISR
);
348 /* Find interrupt and install handler */
349 vhub
->irq
= platform_get_irq(pdev
, 0);
351 dev_err(&pdev
->dev
, "Failed to get interrupt\n");
355 rc
= devm_request_irq(&pdev
->dev
, vhub
->irq
, ast_vhub_irq
, 0,
356 KBUILD_MODNAME
, vhub
);
358 dev_err(&pdev
->dev
, "Failed to request interrupt\n");
363 * Allocate DMA buffers for all EP0s in one chunk,
364 * one per port and one for the vHub itself
366 vhub
->ep0_bufs
= dma_alloc_coherent(&pdev
->dev
,
367 AST_VHUB_EP0_MAX_PACKET
*
368 (AST_VHUB_NUM_PORTS
+ 1),
369 &vhub
->ep0_bufs_dma
, GFP_KERNEL
);
370 if (!vhub
->ep0_bufs
) {
371 dev_err(&pdev
->dev
, "Failed to allocate EP0 DMA buffers\n");
375 UDCVDBG(vhub
, "EP0 DMA buffers @%p (DMA 0x%08x)\n",
376 vhub
->ep0_bufs
, (u32
)vhub
->ep0_bufs_dma
);
379 ast_vhub_init_ep0(vhub
, &vhub
->ep0
, NULL
);
382 for (i
= 0; i
< AST_VHUB_NUM_PORTS
&& rc
== 0; i
++)
383 rc
= ast_vhub_init_dev(vhub
, i
);
387 /* Init hub emulation */
388 ast_vhub_init_hub(vhub
);
391 ast_vhub_init_hw(vhub
);
393 dev_info(&pdev
->dev
, "Initialized virtual hub in USB%d mode\n",
394 vhub
->force_usb1
? 1 : 2);
398 ast_vhub_remove(pdev
);
402 static const struct of_device_id ast_vhub_dt_ids
[] = {
404 .compatible
= "aspeed,ast2400-usb-vhub",
407 .compatible
= "aspeed,ast2500-usb-vhub",
411 MODULE_DEVICE_TABLE(of
, ast_vhub_dt_ids
);
413 static struct platform_driver ast_vhub_driver
= {
414 .probe
= ast_vhub_probe
,
415 .remove
= ast_vhub_remove
,
417 .name
= KBUILD_MODNAME
,
418 .of_match_table
= ast_vhub_dt_ids
,
421 module_platform_driver(ast_vhub_driver
);
423 MODULE_DESCRIPTION("Aspeed vHub udc driver");
424 MODULE_AUTHOR("Benjamin Herrenschmidt <benh@kernel.crashing.org>");
425 MODULE_LICENSE("GPL");