1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3 * aspeed-vhub -- Driver for Aspeed SoC "vHub" USB gadget
4 *
5 * core.c - Top level support
6 *
7 * Copyright 2017 IBM Corporation
8 */
9
10 #include <linux/kernel.h>
11 #include <linux/module.h>
12 #include <linux/platform_device.h>
13 #include <linux/delay.h>
14 #include <linux/ioport.h>
15 #include <linux/slab.h>
16 #include <linux/errno.h>
17 #include <linux/list.h>
18 #include <linux/interrupt.h>
19 #include <linux/proc_fs.h>
20 #include <linux/prefetch.h>
21 #include <linux/clk.h>
22 #include <linux/usb/gadget.h>
23 #include <linux/of.h>
24 #include <linux/regmap.h>
25 #include <linux/dma-mapping.h>
26 #include <linux/reset.h>
27
28 #include "vhub.h"
29
ast_vhub_done(struct ast_vhub_ep * ep,struct ast_vhub_req * req,int status)30 void ast_vhub_done(struct ast_vhub_ep *ep, struct ast_vhub_req *req,
31 int status)
32 {
33 bool internal = req->internal;
34 struct ast_vhub *vhub = ep->vhub;
35
36 EPVDBG(ep, "completing request @%p, status %d\n", req, status);
37
38 list_del_init(&req->queue);
39
40 if ((req->req.status == -EINPROGRESS) || (status == -EOVERFLOW))
41 req->req.status = status;
42
43 if (req->req.dma) {
44 if (!WARN_ON(!ep->dev))
45 usb_gadget_unmap_request_by_dev(&vhub->pdev->dev,
46 &req->req, ep->epn.is_in);
47 req->req.dma = 0;
48 }
49
50 /*
51 * If this isn't an internal EP0 request, call the core
52 * to call the gadget completion.
53 */
54 if (!internal) {
55 spin_unlock(&ep->vhub->lock);
56 usb_gadget_giveback_request(&ep->ep, &req->req);
57 spin_lock(&ep->vhub->lock);
58 }
59 }
60
ast_vhub_nuke(struct ast_vhub_ep * ep,int status)61 void ast_vhub_nuke(struct ast_vhub_ep *ep, int status)
62 {
63 struct ast_vhub_req *req;
64 int count = 0;
65
66 /* Beware, lock will be dropped & req-acquired by done() */
67 while (!list_empty(&ep->queue)) {
68 req = list_first_entry(&ep->queue, struct ast_vhub_req, queue);
69 ast_vhub_done(ep, req, status);
70 count++;
71 }
72 if (count)
73 EPDBG(ep, "Nuked %d request(s)\n", count);
74 }
75
ast_vhub_alloc_request(struct usb_ep * u_ep,gfp_t gfp_flags)76 struct usb_request *ast_vhub_alloc_request(struct usb_ep *u_ep,
77 gfp_t gfp_flags)
78 {
79 struct ast_vhub_req *req;
80
81 req = kzalloc_obj(*req, gfp_flags);
82 if (!req)
83 return NULL;
84 return &req->req;
85 }
86
ast_vhub_free_request(struct usb_ep * u_ep,struct usb_request * u_req)87 void ast_vhub_free_request(struct usb_ep *u_ep, struct usb_request *u_req)
88 {
89 struct ast_vhub_req *req = to_ast_req(u_req);
90
91 kfree(req);
92 }
93
ast_vhub_irq(int irq,void * data)94 static irqreturn_t ast_vhub_irq(int irq, void *data)
95 {
96 struct ast_vhub *vhub = data;
97 irqreturn_t iret = IRQ_NONE;
98 u32 i, istat;
99
100 /* Stale interrupt while tearing down */
101 if (!vhub->ep0_bufs)
102 return IRQ_NONE;
103
104 spin_lock(&vhub->lock);
105
106 /* Read and ACK interrupts */
107 istat = readl(vhub->regs + AST_VHUB_ISR);
108 if (!istat)
109 goto bail;
110 writel(istat, vhub->regs + AST_VHUB_ISR);
111 iret = IRQ_HANDLED;
112
113 UDCVDBG(vhub, "irq status=%08x, ep_acks=%08x ep_nacks=%08x\n",
114 istat,
115 readl(vhub->regs + AST_VHUB_EP_ACK_ISR),
116 readl(vhub->regs + AST_VHUB_EP_NACK_ISR));
117
118 /* Handle generic EPs first */
119 if (istat & VHUB_IRQ_EP_POOL_ACK_STALL) {
120 u32 ep_acks = readl(vhub->regs + AST_VHUB_EP_ACK_ISR);
121 writel(ep_acks, vhub->regs + AST_VHUB_EP_ACK_ISR);
122
123 for (i = 0; ep_acks && i < vhub->max_epns; i++) {
124 u32 mask = VHUB_EP_IRQ(i);
125 if (ep_acks & mask) {
126 ast_vhub_epn_ack_irq(&vhub->epns[i]);
127 ep_acks &= ~mask;
128 }
129 }
130 }
131
132 /* Handle device interrupts */
133 if (istat & vhub->port_irq_mask) {
134 for (i = 0; i < vhub->max_ports; i++) {
135 if (istat & VHUB_DEV_IRQ(i))
136 ast_vhub_dev_irq(&vhub->ports[i].dev);
137 }
138 }
139
140 /* Handle top-level vHub EP0 interrupts */
141 if (istat & (VHUB_IRQ_HUB_EP0_OUT_ACK_STALL |
142 VHUB_IRQ_HUB_EP0_IN_ACK_STALL |
143 VHUB_IRQ_HUB_EP0_SETUP)) {
144 if (istat & VHUB_IRQ_HUB_EP0_IN_ACK_STALL)
145 ast_vhub_ep0_handle_ack(&vhub->ep0, true);
146 if (istat & VHUB_IRQ_HUB_EP0_OUT_ACK_STALL)
147 ast_vhub_ep0_handle_ack(&vhub->ep0, false);
148 if (istat & VHUB_IRQ_HUB_EP0_SETUP)
149 ast_vhub_ep0_handle_setup(&vhub->ep0);
150 }
151
152 /* Various top level bus events */
153 if (istat & (VHUB_IRQ_BUS_RESUME |
154 VHUB_IRQ_BUS_SUSPEND |
155 VHUB_IRQ_BUS_RESET)) {
156 if (istat & VHUB_IRQ_BUS_RESUME)
157 ast_vhub_hub_resume(vhub);
158 if (istat & VHUB_IRQ_BUS_SUSPEND)
159 ast_vhub_hub_suspend(vhub);
160 if (istat & VHUB_IRQ_BUS_RESET)
161 ast_vhub_hub_reset(vhub);
162 }
163
164 bail:
165 spin_unlock(&vhub->lock);
166 return iret;
167 }
168
ast_vhub_init_hw(struct ast_vhub * vhub)169 void ast_vhub_init_hw(struct ast_vhub *vhub)
170 {
171 u32 ctrl, port_mask, epn_mask;
172
173 UDCDBG(vhub,"(Re)Starting HW ...\n");
174
175 /* Enable PHY */
176 ctrl = VHUB_CTRL_PHY_CLK |
177 VHUB_CTRL_PHY_RESET_DIS;
178
179 /*
180 * We do *NOT* set the VHUB_CTRL_CLK_STOP_SUSPEND bit
181 * to stop the logic clock during suspend because
182 * it causes the registers to become inaccessible and
183 * we haven't yet figured out a good wayt to bring the
184 * controller back into life to issue a wakeup.
185 */
186
187 /*
188 * Set some ISO & split control bits according to Aspeed
189 * recommendation
190 *
191 * VHUB_CTRL_ISO_RSP_CTRL: When set tells the HW to respond
192 * with 0 bytes data packet to ISO IN endpoints when no data
193 * is available.
194 *
195 * VHUB_CTRL_SPLIT_IN: This makes a SOF complete a split IN
196 * transaction.
197 */
198 ctrl |= VHUB_CTRL_ISO_RSP_CTRL | VHUB_CTRL_SPLIT_IN;
199 writel(ctrl, vhub->regs + AST_VHUB_CTRL);
200 udelay(1);
201
202 /* Set descriptor ring size */
203 if (AST_VHUB_DESCS_COUNT == 256) {
204 ctrl |= VHUB_CTRL_LONG_DESC;
205 writel(ctrl, vhub->regs + AST_VHUB_CTRL);
206 } else {
207 BUILD_BUG_ON(AST_VHUB_DESCS_COUNT != 32);
208 }
209
210 /* Reset all devices */
211 port_mask = GENMASK(vhub->max_ports, 1);
212 writel(VHUB_SW_RESET_ROOT_HUB |
213 VHUB_SW_RESET_DMA_CONTROLLER |
214 VHUB_SW_RESET_EP_POOL |
215 port_mask, vhub->regs + AST_VHUB_SW_RESET);
216 udelay(1);
217 writel(0, vhub->regs + AST_VHUB_SW_RESET);
218
219 /* Disable and cleanup EP ACK/NACK interrupts */
220 epn_mask = GENMASK(vhub->max_epns - 1, 0);
221 writel(0, vhub->regs + AST_VHUB_EP_ACK_IER);
222 writel(0, vhub->regs + AST_VHUB_EP_NACK_IER);
223 writel(epn_mask, vhub->regs + AST_VHUB_EP_ACK_ISR);
224 writel(epn_mask, vhub->regs + AST_VHUB_EP_NACK_ISR);
225
226 /* Default settings for EP0, enable HW hub EP1 */
227 writel(0, vhub->regs + AST_VHUB_EP0_CTRL);
228 writel(VHUB_EP1_CTRL_RESET_TOGGLE |
229 VHUB_EP1_CTRL_ENABLE,
230 vhub->regs + AST_VHUB_EP1_CTRL);
231 writel(0, vhub->regs + AST_VHUB_EP1_STS_CHG);
232
233 /* Configure EP0 DMA buffer */
234 writel(vhub->ep0.buf_dma, vhub->regs + AST_VHUB_EP0_DATA);
235
236 /* Clear address */
237 writel(0, vhub->regs + AST_VHUB_CONF);
238
239 /* Pullup hub (activate on host) */
240 if (vhub->force_usb1)
241 ctrl |= VHUB_CTRL_FULL_SPEED_ONLY;
242
243 ctrl |= VHUB_CTRL_UPSTREAM_CONNECT;
244 writel(ctrl, vhub->regs + AST_VHUB_CTRL);
245
246 /* Enable some interrupts */
247 writel(VHUB_IRQ_HUB_EP0_IN_ACK_STALL |
248 VHUB_IRQ_HUB_EP0_OUT_ACK_STALL |
249 VHUB_IRQ_HUB_EP0_SETUP |
250 VHUB_IRQ_EP_POOL_ACK_STALL |
251 VHUB_IRQ_BUS_RESUME |
252 VHUB_IRQ_BUS_SUSPEND |
253 VHUB_IRQ_BUS_RESET,
254 vhub->regs + AST_VHUB_IER);
255 }
256
ast_vhub_remove(struct platform_device * pdev)257 static void ast_vhub_remove(struct platform_device *pdev)
258 {
259 struct ast_vhub *vhub = platform_get_drvdata(pdev);
260 unsigned long flags;
261 int i;
262
263 if (!vhub || !vhub->regs)
264 return;
265
266 /* Remove devices */
267 for (i = 0; i < vhub->max_ports; i++)
268 ast_vhub_del_dev(&vhub->ports[i].dev);
269
270 spin_lock_irqsave(&vhub->lock, flags);
271
272 /* Mask & ack all interrupts */
273 writel(0, vhub->regs + AST_VHUB_IER);
274 writel(VHUB_IRQ_ACK_ALL, vhub->regs + AST_VHUB_ISR);
275
276 /* Pull device, leave PHY enabled */
277 writel(VHUB_CTRL_PHY_CLK |
278 VHUB_CTRL_PHY_RESET_DIS,
279 vhub->regs + AST_VHUB_CTRL);
280
281 if (vhub->clk)
282 clk_disable_unprepare(vhub->clk);
283
284 reset_control_assert(vhub->rst);
285
286 spin_unlock_irqrestore(&vhub->lock, flags);
287
288 if (vhub->ep0_bufs)
289 dma_free_coherent(&pdev->dev,
290 AST_VHUB_EP0_MAX_PACKET *
291 (vhub->max_ports + 1),
292 vhub->ep0_bufs,
293 vhub->ep0_bufs_dma);
294 vhub->ep0_bufs = NULL;
295 }
296
ast_vhub_probe(struct platform_device * pdev)297 static int ast_vhub_probe(struct platform_device *pdev)
298 {
299 enum usb_device_speed max_speed;
300 const u64 *dma_mask_ptr;
301 struct ast_vhub *vhub;
302 struct resource *res;
303 int i, rc = 0;
304 const struct device_node *np = pdev->dev.of_node;
305
306 vhub = devm_kzalloc(&pdev->dev, sizeof(*vhub), GFP_KERNEL);
307 if (!vhub)
308 return -ENOMEM;
309
310 rc = of_property_read_u32(np, "aspeed,vhub-downstream-ports",
311 &vhub->max_ports);
312 if (rc < 0)
313 vhub->max_ports = AST_VHUB_NUM_PORTS;
314
315 vhub->ports = devm_kcalloc(&pdev->dev, vhub->max_ports,
316 sizeof(*vhub->ports), GFP_KERNEL);
317 if (!vhub->ports)
318 return -ENOMEM;
319
320 rc = of_property_read_u32(np, "aspeed,vhub-generic-endpoints",
321 &vhub->max_epns);
322 if (rc < 0)
323 vhub->max_epns = AST_VHUB_NUM_GEN_EPs;
324
325 vhub->epns = devm_kcalloc(&pdev->dev, vhub->max_epns,
326 sizeof(*vhub->epns), GFP_KERNEL);
327 if (!vhub->epns)
328 return -ENOMEM;
329
330 spin_lock_init(&vhub->lock);
331 vhub->pdev = pdev;
332 vhub->port_irq_mask = GENMASK(VHUB_IRQ_DEV1_BIT + vhub->max_ports - 1,
333 VHUB_IRQ_DEV1_BIT);
334
335 vhub->regs = devm_platform_get_and_ioremap_resource(pdev, 0, &res);
336 if (IS_ERR(vhub->regs)) {
337 dev_err(&pdev->dev, "Failed to map resources\n");
338 return PTR_ERR(vhub->regs);
339 }
340 UDCDBG(vhub, "vHub@%pR mapped @%p\n", res, vhub->regs);
341
342 platform_set_drvdata(pdev, vhub);
343
344 vhub->clk = devm_clk_get(&pdev->dev, NULL);
345 if (IS_ERR(vhub->clk)) {
346 rc = PTR_ERR(vhub->clk);
347 goto err;
348 }
349 rc = clk_prepare_enable(vhub->clk);
350 if (rc) {
351 dev_err(&pdev->dev, "Error couldn't enable clock (%d)\n", rc);
352 goto err;
353 }
354
355 vhub->rst = devm_reset_control_get_optional_shared(&pdev->dev, NULL);
356 if (IS_ERR(vhub->rst)) {
357 rc = PTR_ERR(vhub->rst);
358 goto err;
359 }
360
361 rc = reset_control_deassert(vhub->rst);
362 if (rc)
363 goto err;
364
365 /* Check if we need to limit the HW to USB1 */
366 max_speed = usb_get_maximum_speed(&pdev->dev);
367 if (max_speed != USB_SPEED_UNKNOWN && max_speed < USB_SPEED_HIGH)
368 vhub->force_usb1 = true;
369
370 /* Mask & ack all interrupts before installing the handler */
371 writel(0, vhub->regs + AST_VHUB_IER);
372 writel(VHUB_IRQ_ACK_ALL, vhub->regs + AST_VHUB_ISR);
373
374 /* Find interrupt and install handler */
375 vhub->irq = platform_get_irq(pdev, 0);
376 if (vhub->irq < 0) {
377 rc = vhub->irq;
378 goto err;
379 }
380 rc = devm_request_irq(&pdev->dev, vhub->irq, ast_vhub_irq, 0,
381 KBUILD_MODNAME, vhub);
382 if (rc) {
383 dev_err(&pdev->dev, "Failed to request interrupt\n");
384 goto err;
385 }
386
387 dma_mask_ptr = (u64 *)of_device_get_match_data(&pdev->dev);
388 if (dma_mask_ptr) {
389 rc = dma_coerce_mask_and_coherent(&pdev->dev, *dma_mask_ptr);
390 if (rc)
391 goto err;
392 }
393 /*
394 * Allocate DMA buffers for all EP0s in one chunk,
395 * one per port and one for the vHub itself
396 */
397 vhub->ep0_bufs = dma_alloc_coherent(&pdev->dev,
398 AST_VHUB_EP0_MAX_PACKET *
399 (vhub->max_ports + 1),
400 &vhub->ep0_bufs_dma, GFP_KERNEL);
401 if (!vhub->ep0_bufs) {
402 dev_err(&pdev->dev, "Failed to allocate EP0 DMA buffers\n");
403 rc = -ENOMEM;
404 goto err;
405 }
406 UDCVDBG(vhub, "EP0 DMA buffers @%p (DMA 0x%08x)\n",
407 vhub->ep0_bufs, (u32)vhub->ep0_bufs_dma);
408
409 /* Init vHub EP0 */
410 ast_vhub_init_ep0(vhub, &vhub->ep0, NULL);
411
412 /* Init devices */
413 for (i = 0; i < vhub->max_ports && rc == 0; i++)
414 rc = ast_vhub_init_dev(vhub, i);
415 if (rc)
416 goto err;
417
418 /* Init hub emulation */
419 rc = ast_vhub_init_hub(vhub);
420 if (rc)
421 goto err;
422
423 /* Initialize HW */
424 ast_vhub_init_hw(vhub);
425
426 dev_info(&pdev->dev, "Initialized virtual hub in USB%d mode\n",
427 vhub->force_usb1 ? 1 : 2);
428
429 return 0;
430 err:
431 ast_vhub_remove(pdev);
432 return rc;
433 }
434
435 static const u64 dma_mask_32 = DMA_BIT_MASK(32);
436 static const u64 dma_mask_64 = DMA_BIT_MASK(64);
437
438 static const struct of_device_id ast_vhub_dt_ids[] = {
439 {
440 .compatible = "aspeed,ast2400-usb-vhub",
441 .data = &dma_mask_32,
442 },
443 {
444 .compatible = "aspeed,ast2500-usb-vhub",
445 .data = &dma_mask_32,
446 },
447 {
448 .compatible = "aspeed,ast2600-usb-vhub",
449 .data = &dma_mask_32,
450 },
451 {
452 .compatible = "aspeed,ast2700-usb-vhub",
453 .data = &dma_mask_64,
454 },
455 { }
456 };
457 MODULE_DEVICE_TABLE(of, ast_vhub_dt_ids);
458
459 static struct platform_driver ast_vhub_driver = {
460 .probe = ast_vhub_probe,
461 .remove = ast_vhub_remove,
462 .driver = {
463 .name = KBUILD_MODNAME,
464 .of_match_table = ast_vhub_dt_ids,
465 },
466 };
467 module_platform_driver(ast_vhub_driver);
468
469 MODULE_DESCRIPTION("Aspeed vHub udc driver");
470 MODULE_AUTHOR("Benjamin Herrenschmidt <benh@kernel.crashing.org>");
471 MODULE_LICENSE("GPL");
472