blob: cb44374e81a615c2c4cccdc7cb526ee5b5444721 [file] [log] [blame]
Lukasz Majewskiac3c8312011-10-27 10:36:46 +02001/*
Marek Vasutf8995db2015-12-04 02:34:46 +01002 * drivers/usb/gadget/dwc2_udc_otg.c
Marek Vasut4a7629a2015-12-04 02:55:37 +01003 * Designware DWC2 on-chip full/high speed USB OTG 2.0 device controllers
Lukasz Majewskiac3c8312011-10-27 10:36:46 +02004 *
5 * Copyright (C) 2008 for Samsung Electronics
6 *
7 * BSP Support for Samsung's UDC driver
8 * available at:
9 * git://git.kernel.org/pub/scm/linux/kernel/git/kki_ap/linux-2.6-samsung.git
10 *
11 * State machine bugfixes:
12 * Marek Szyprowski <m.szyprowski@samsung.com>
13 *
14 * Ported to u-boot:
15 * Marek Szyprowski <m.szyprowski@samsung.com>
16 * Lukasz Majewski <l.majewski@samsumg.com>
17 *
Wolfgang Denkd79de1d2013-07-08 09:37:19 +020018 * SPDX-License-Identifier: GPL-2.0+
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020019 */
Lukasz Majewski8fb78fd2012-05-02 13:11:38 +020020#undef DEBUG
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020021#include <common.h>
Masahiro Yamada56a931c2016-09-21 11:28:55 +090022#include <linux/errno.h>
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020023#include <linux/list.h>
24#include <malloc.h>
25
26#include <linux/usb/ch9.h>
27#include <linux/usb/gadget.h>
28
29#include <asm/byteorder.h>
Tom Rini73424522011-12-15 08:40:51 -070030#include <asm/unaligned.h>
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020031#include <asm/io.h>
32
33#include <asm/mach-types.h>
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020034
Marek Vasut4811c662015-12-04 02:32:22 +010035#include "dwc2_udc_otg_regs.h"
36#include "dwc2_udc_otg_priv.h"
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020037#include <usb/lin_gadget_compat.h>
38
39/***********************************************************/
40
41#define OTG_DMA_MODE 1
42
Lukasz Majewski0dfeb002012-05-02 13:11:35 +020043#define DEBUG_SETUP 0
44#define DEBUG_EP0 0
45#define DEBUG_ISR 0
46#define DEBUG_OUT_EP 0
47#define DEBUG_IN_EP 0
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020048
Marek Vasutf1be9cb2015-12-04 02:51:20 +010049#include <usb/dwc2_udc.h>
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020050
51#define EP0_CON 0
52#define EP_MASK 0xF
53
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020054static char *state_names[] = {
55 "WAIT_FOR_SETUP",
56 "DATA_STATE_XMIT",
57 "DATA_STATE_NEED_ZLP",
58 "WAIT_FOR_OUT_STATUS",
59 "DATA_STATE_RECV",
60 "WAIT_FOR_COMPLETE",
61 "WAIT_FOR_OUT_COMPLETE",
62 "WAIT_FOR_IN_COMPLETE",
63 "WAIT_FOR_NULL_COMPLETE",
64};
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020065
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020066#define DRIVER_VERSION "15 March 2009"
67
Marek Vasut1a639ff2015-12-04 00:57:58 +010068struct dwc2_udc *the_controller;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020069
Marek Vasutdbdd58f2015-12-04 02:28:40 +010070static const char driver_name[] = "dwc2-udc";
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020071static const char ep0name[] = "ep0-control";
72
73/* Max packet size*/
74static unsigned int ep0_fifo_size = 64;
75static unsigned int ep_fifo_size = 512;
76static unsigned int ep_fifo_size2 = 1024;
77static int reset_available = 1;
78
79static struct usb_ctrlrequest *usb_ctrl;
80static dma_addr_t usb_ctrl_dma_addr;
81
82/*
83 Local declarations.
84*/
Marek Vasut4e7c29e2015-12-04 02:13:42 +010085static int dwc2_ep_enable(struct usb_ep *ep,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020086 const struct usb_endpoint_descriptor *);
Marek Vasut4e7c29e2015-12-04 02:13:42 +010087static int dwc2_ep_disable(struct usb_ep *ep);
Marek Vasut153150f2015-12-04 02:17:40 +010088static struct usb_request *dwc2_alloc_request(struct usb_ep *ep,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020089 gfp_t gfp_flags);
Marek Vasut153150f2015-12-04 02:17:40 +010090static void dwc2_free_request(struct usb_ep *ep, struct usb_request *);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +020091
Marek Vasut153150f2015-12-04 02:17:40 +010092static int dwc2_queue(struct usb_ep *ep, struct usb_request *, gfp_t gfp_flags);
93static int dwc2_dequeue(struct usb_ep *ep, struct usb_request *);
94static int dwc2_fifo_status(struct usb_ep *ep);
95static void dwc2_fifo_flush(struct usb_ep *ep);
Marek Vasutfe92a512015-12-04 01:59:12 +010096static void dwc2_ep0_read(struct dwc2_udc *dev);
97static void dwc2_ep0_kick(struct dwc2_udc *dev, struct dwc2_ep *ep);
Marek Vasut153150f2015-12-04 02:17:40 +010098static void dwc2_handle_ep0(struct dwc2_udc *dev);
Marek Vasutfe92a512015-12-04 01:59:12 +010099static int dwc2_ep0_write(struct dwc2_udc *dev);
Marek Vasut32f931c2015-12-04 01:51:07 +0100100static int write_fifo_ep0(struct dwc2_ep *ep, struct dwc2_request *req);
101static void done(struct dwc2_ep *ep, struct dwc2_request *req, int status);
Marek Vasut1a639ff2015-12-04 00:57:58 +0100102static void stop_activity(struct dwc2_udc *dev,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200103 struct usb_gadget_driver *driver);
Marek Vasut1a639ff2015-12-04 00:57:58 +0100104static int udc_enable(struct dwc2_udc *dev);
105static void udc_set_address(struct dwc2_udc *dev, unsigned char address);
106static void reconfig_usbd(struct dwc2_udc *dev);
107static void set_max_pktsize(struct dwc2_udc *dev, enum usb_device_speed speed);
Marek Vasut5b309f72015-12-04 01:48:57 +0100108static void nuke(struct dwc2_ep *ep, int status);
Marek Vasutfa03c812015-12-04 02:03:45 +0100109static int dwc2_udc_set_halt(struct usb_ep *_ep, int value);
110static void dwc2_udc_set_nak(struct dwc2_ep *ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200111
Lukasz Majewski8fb78fd2012-05-02 13:11:38 +0200112void set_udc_gadget_private_data(void *p)
113{
114 debug_cond(DEBUG_SETUP != 0,
115 "%s: the_controller: 0x%p, p: 0x%p\n", __func__,
116 the_controller, p);
117 the_controller->gadget.dev.device_data = p;
118}
119
120void *get_udc_gadget_private_data(struct usb_gadget *gadget)
121{
122 return gadget->dev.device_data;
123}
124
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100125static struct usb_ep_ops dwc2_ep_ops = {
126 .enable = dwc2_ep_enable,
127 .disable = dwc2_ep_disable,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200128
Marek Vasut153150f2015-12-04 02:17:40 +0100129 .alloc_request = dwc2_alloc_request,
130 .free_request = dwc2_free_request,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200131
Marek Vasut153150f2015-12-04 02:17:40 +0100132 .queue = dwc2_queue,
133 .dequeue = dwc2_dequeue,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200134
Marek Vasutfa03c812015-12-04 02:03:45 +0100135 .set_halt = dwc2_udc_set_halt,
Marek Vasut153150f2015-12-04 02:17:40 +0100136 .fifo_status = dwc2_fifo_status,
137 .fifo_flush = dwc2_fifo_flush,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200138};
139
140#define create_proc_files() do {} while (0)
141#define remove_proc_files() do {} while (0)
142
143/***********************************************************/
144
145void __iomem *regs_otg;
Marek Vasut67ec9dc2015-12-04 01:11:45 +0100146struct dwc2_usbotg_reg *reg;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200147
Lukasz Majewskief731022014-08-25 11:07:29 +0200148bool dfu_usb_get_reset(void)
149{
150 return !!(readl(&reg->gintsts) & INT_RESET);
151}
152
Marek Vasut1a639ff2015-12-04 00:57:58 +0100153__weak void otg_phy_init(struct dwc2_udc *dev) {}
154__weak void otg_phy_off(struct dwc2_udc *dev) {}
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200155
156/***********************************************************/
157
Marek Vasutf8995db2015-12-04 02:34:46 +0100158#include "dwc2_udc_otg_xfer_dma.c"
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200159
160/*
161 * udc_disable - disable USB device controller
162 */
Marek Vasut1a639ff2015-12-04 00:57:58 +0100163static void udc_disable(struct dwc2_udc *dev)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200164{
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200165 debug_cond(DEBUG_SETUP != 0, "%s: %p\n", __func__, dev);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200166
167 udc_set_address(dev, 0);
168
169 dev->ep0state = WAIT_FOR_SETUP;
170 dev->gadget.speed = USB_SPEED_UNKNOWN;
171 dev->usb_address = 0;
172
173 otg_phy_off(dev);
174}
175
176/*
177 * udc_reinit - initialize software state
178 */
Marek Vasut1a639ff2015-12-04 00:57:58 +0100179static void udc_reinit(struct dwc2_udc *dev)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200180{
181 unsigned int i;
182
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200183 debug_cond(DEBUG_SETUP != 0, "%s: %p\n", __func__, dev);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200184
185 /* device/ep0 records init */
186 INIT_LIST_HEAD(&dev->gadget.ep_list);
187 INIT_LIST_HEAD(&dev->gadget.ep0->ep_list);
188 dev->ep0state = WAIT_FOR_SETUP;
189
190 /* basic endpoint records init */
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100191 for (i = 0; i < DWC2_MAX_ENDPOINTS; i++) {
Marek Vasut5b309f72015-12-04 01:48:57 +0100192 struct dwc2_ep *ep = &dev->ep[i];
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200193
194 if (i != 0)
195 list_add_tail(&ep->ep.ep_list, &dev->gadget.ep_list);
196
197 ep->desc = 0;
198 ep->stopped = 0;
199 INIT_LIST_HEAD(&ep->queue);
200 ep->pio_irqs = 0;
201 }
202
203 /* the rest was statically initialized, and is read-only */
204}
205
206#define BYTES2MAXP(x) (x / 8)
207#define MAXP2BYTES(x) (x * 8)
208
209/* until it's enabled, this UDC should be completely invisible
210 * to any USB host.
211 */
Marek Vasut1a639ff2015-12-04 00:57:58 +0100212static int udc_enable(struct dwc2_udc *dev)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200213{
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200214 debug_cond(DEBUG_SETUP != 0, "%s: %p\n", __func__, dev);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200215
216 otg_phy_init(dev);
Marek Vasut80300882014-11-04 04:23:25 +0100217 reconfig_usbd(dev);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200218
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200219 debug_cond(DEBUG_SETUP != 0,
Marek Vasutdbdd58f2015-12-04 02:28:40 +0100220 "DWC2 USB 2.0 OTG Controller Core Initialized : 0x%x\n",
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200221 readl(&reg->gintmsk));
222
223 dev->gadget.speed = USB_SPEED_UNKNOWN;
224
225 return 0;
226}
227
228/*
229 Register entry point for the peripheral controller driver.
230*/
231int usb_gadget_register_driver(struct usb_gadget_driver *driver)
232{
Marek Vasut1a639ff2015-12-04 00:57:58 +0100233 struct dwc2_udc *dev = the_controller;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200234 int retval = 0;
Bo Shen60dc40c2014-08-27 17:28:18 +0800235 unsigned long flags = 0;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200236
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200237 debug_cond(DEBUG_SETUP != 0, "%s: %s\n", __func__, "no name");
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200238
239 if (!driver
240 || (driver->speed != USB_SPEED_FULL
241 && driver->speed != USB_SPEED_HIGH)
242 || !driver->bind || !driver->disconnect || !driver->setup)
243 return -EINVAL;
244 if (!dev)
245 return -ENODEV;
246 if (dev->driver)
247 return -EBUSY;
248
249 spin_lock_irqsave(&dev->lock, flags);
250 /* first hook up the driver ... */
251 dev->driver = driver;
252 spin_unlock_irqrestore(&dev->lock, flags);
253
254 if (retval) { /* TODO */
255 printf("target device_add failed, error %d\n", retval);
256 return retval;
257 }
258
259 retval = driver->bind(&dev->gadget);
260 if (retval) {
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200261 debug_cond(DEBUG_SETUP != 0,
262 "%s: bind to driver --> error %d\n",
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200263 dev->gadget.name, retval);
264 dev->driver = 0;
265 return retval;
266 }
267
268 enable_irq(IRQ_OTG);
269
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200270 debug_cond(DEBUG_SETUP != 0,
271 "Registered gadget driver %s\n", dev->gadget.name);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200272 udc_enable(dev);
273
274 return 0;
275}
276
277/*
278 * Unregister entry point for the peripheral controller driver.
279 */
280int usb_gadget_unregister_driver(struct usb_gadget_driver *driver)
281{
Marek Vasut1a639ff2015-12-04 00:57:58 +0100282 struct dwc2_udc *dev = the_controller;
Bo Shen60dc40c2014-08-27 17:28:18 +0800283 unsigned long flags = 0;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200284
285 if (!dev)
286 return -ENODEV;
287 if (!driver || driver != dev->driver)
288 return -EINVAL;
289
290 spin_lock_irqsave(&dev->lock, flags);
291 dev->driver = 0;
292 stop_activity(dev, driver);
293 spin_unlock_irqrestore(&dev->lock, flags);
294
295 driver->unbind(&dev->gadget);
296
297 disable_irq(IRQ_OTG);
298
299 udc_disable(dev);
300 return 0;
301}
302
303/*
304 * done - retire a request; caller blocked irqs
305 */
Marek Vasut32f931c2015-12-04 01:51:07 +0100306static void done(struct dwc2_ep *ep, struct dwc2_request *req, int status)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200307{
308 unsigned int stopped = ep->stopped;
309
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000310 debug("%s: %s %p, req = %p, stopped = %d\n",
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200311 __func__, ep->ep.name, ep, &req->req, stopped);
312
313 list_del_init(&req->queue);
314
315 if (likely(req->req.status == -EINPROGRESS))
316 req->req.status = status;
317 else
318 status = req->req.status;
319
320 if (status && status != -ESHUTDOWN) {
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000321 debug("complete %s req %p stat %d len %u/%u\n",
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200322 ep->ep.name, &req->req, status,
323 req->req.actual, req->req.length);
324 }
325
326 /* don't modify queue heads during completion callback */
327 ep->stopped = 1;
328
Lukasz Majewski0dfeb002012-05-02 13:11:35 +0200329#ifdef DEBUG
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200330 printf("calling complete callback\n");
331 {
332 int i, len = req->req.length;
333
334 printf("pkt[%d] = ", req->req.length);
335 if (len > 64)
336 len = 64;
337 for (i = 0; i < len; i++) {
338 printf("%02x", ((u8 *)req->req.buf)[i]);
339 if ((i & 7) == 7)
340 printf(" ");
341 }
342 printf("\n");
343 }
344#endif
345 spin_unlock(&ep->dev->lock);
346 req->req.complete(&ep->ep, &req->req);
347 spin_lock(&ep->dev->lock);
348
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000349 debug("callback completed\n");
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200350
351 ep->stopped = stopped;
352}
353
354/*
355 * nuke - dequeue ALL requests
356 */
Marek Vasut5b309f72015-12-04 01:48:57 +0100357static void nuke(struct dwc2_ep *ep, int status)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200358{
Marek Vasut32f931c2015-12-04 01:51:07 +0100359 struct dwc2_request *req;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200360
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000361 debug("%s: %s %p\n", __func__, ep->ep.name, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200362
363 /* called with irqs blocked */
364 while (!list_empty(&ep->queue)) {
Marek Vasut32f931c2015-12-04 01:51:07 +0100365 req = list_entry(ep->queue.next, struct dwc2_request, queue);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200366 done(ep, req, status);
367 }
368}
369
Marek Vasut1a639ff2015-12-04 00:57:58 +0100370static void stop_activity(struct dwc2_udc *dev,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200371 struct usb_gadget_driver *driver)
372{
373 int i;
374
375 /* don't disconnect drivers more than once */
376 if (dev->gadget.speed == USB_SPEED_UNKNOWN)
377 driver = 0;
378 dev->gadget.speed = USB_SPEED_UNKNOWN;
379
380 /* prevent new request submissions, kill any outstanding requests */
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100381 for (i = 0; i < DWC2_MAX_ENDPOINTS; i++) {
Marek Vasut5b309f72015-12-04 01:48:57 +0100382 struct dwc2_ep *ep = &dev->ep[i];
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200383 ep->stopped = 1;
384 nuke(ep, -ESHUTDOWN);
385 }
386
387 /* report disconnect; the driver is already quiesced */
388 if (driver) {
389 spin_unlock(&dev->lock);
390 driver->disconnect(&dev->gadget);
391 spin_lock(&dev->lock);
392 }
393
394 /* re-init driver-visible data structures */
395 udc_reinit(dev);
396}
397
Marek Vasut1a639ff2015-12-04 00:57:58 +0100398static void reconfig_usbd(struct dwc2_udc *dev)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200399{
400 /* 2. Soft-reset OTG Core and then unreset again. */
401 int i;
402 unsigned int uTemp = writel(CORE_SOFT_RESET, &reg->grstctl);
Marek Vasut80300882014-11-04 04:23:25 +0100403 uint32_t dflt_gusbcfg;
Xu Ziyuanb729dc02016-07-14 14:52:33 +0800404 uint32_t rx_fifo_sz, tx_fifo_sz, np_tx_fifo_sz;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200405
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000406 debug("Reseting OTG controller\n");
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200407
Marek Vasut80300882014-11-04 04:23:25 +0100408 dflt_gusbcfg =
409 0<<15 /* PHY Low Power Clock sel*/
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200410 |1<<14 /* Non-Periodic TxFIFO Rewind Enable*/
411 |0x5<<10 /* Turnaround time*/
412 |0<<9 | 0<<8 /* [0:HNP disable,1:HNP enable][ 0:SRP disable*/
413 /* 1:SRP enable] H1= 1,1*/
414 |0<<7 /* Ulpi DDR sel*/
415 |0<<6 /* 0: high speed utmi+, 1: full speed serial*/
416 |0<<4 /* 0: utmi+, 1:ulpi*/
Steve Raed7198f32016-06-07 15:35:21 -0700417#ifdef CONFIG_USB_GADGET_DWC2_OTG_PHY_BUS_WIDTH_8
418 |0<<3 /* phy i/f 0:8bit, 1:16bit*/
419#else
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200420 |1<<3 /* phy i/f 0:8bit, 1:16bit*/
Steve Raed7198f32016-06-07 15:35:21 -0700421#endif
Marek Vasut80300882014-11-04 04:23:25 +0100422 |0x7<<0; /* HS/FS Timeout**/
423
424 if (dev->pdata->usb_gusbcfg)
425 dflt_gusbcfg = dev->pdata->usb_gusbcfg;
426
427 writel(dflt_gusbcfg, &reg->gusbcfg);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200428
429 /* 3. Put the OTG device core in the disconnected state.*/
430 uTemp = readl(&reg->dctl);
431 uTemp |= SOFT_DISCONNECT;
432 writel(uTemp, &reg->dctl);
433
434 udelay(20);
435
436 /* 4. Make the OTG device core exit from the disconnected state.*/
437 uTemp = readl(&reg->dctl);
438 uTemp = uTemp & ~SOFT_DISCONNECT;
439 writel(uTemp, &reg->dctl);
440
441 /* 5. Configure OTG Core to initial settings of device mode.*/
442 /* [][1: full speed(30Mhz) 0:high speed]*/
443 writel(EP_MISS_CNT(1) | DEV_SPEED_HIGH_SPEED_20, &reg->dcfg);
444
445 mdelay(1);
446
447 /* 6. Unmask the core interrupts*/
448 writel(GINTMSK_INIT, &reg->gintmsk);
449
450 /* 7. Set NAK bit of EP0, EP1, EP2*/
451 writel(DEPCTL_EPDIS|DEPCTL_SNAK, &reg->out_endp[EP0_CON].doepctl);
452 writel(DEPCTL_EPDIS|DEPCTL_SNAK, &reg->in_endp[EP0_CON].diepctl);
453
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100454 for (i = 1; i < DWC2_MAX_ENDPOINTS; i++) {
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200455 writel(DEPCTL_EPDIS|DEPCTL_SNAK, &reg->out_endp[i].doepctl);
456 writel(DEPCTL_EPDIS|DEPCTL_SNAK, &reg->in_endp[i].diepctl);
457 }
458
459 /* 8. Unmask EPO interrupts*/
460 writel(((1 << EP0_CON) << DAINT_OUT_BIT)
461 | (1 << EP0_CON), &reg->daintmsk);
462
463 /* 9. Unmask device OUT EP common interrupts*/
464 writel(DOEPMSK_INIT, &reg->doepmsk);
465
466 /* 10. Unmask device IN EP common interrupts*/
467 writel(DIEPMSK_INIT, &reg->diepmsk);
468
Xu Ziyuanb729dc02016-07-14 14:52:33 +0800469 rx_fifo_sz = RX_FIFO_SIZE;
470 np_tx_fifo_sz = NPTX_FIFO_SIZE;
471 tx_fifo_sz = PTX_FIFO_SIZE;
472
473 if (dev->pdata->rx_fifo_sz)
474 rx_fifo_sz = dev->pdata->rx_fifo_sz;
475 if (dev->pdata->np_tx_fifo_sz)
476 np_tx_fifo_sz = dev->pdata->np_tx_fifo_sz;
477 if (dev->pdata->tx_fifo_sz)
478 tx_fifo_sz = dev->pdata->tx_fifo_sz;
479
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200480 /* 11. Set Rx FIFO Size (in 32-bit words) */
Xu Ziyuanb729dc02016-07-14 14:52:33 +0800481 writel(rx_fifo_sz, &reg->grxfsiz);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200482
483 /* 12. Set Non Periodic Tx FIFO Size */
Xu Ziyuanb729dc02016-07-14 14:52:33 +0800484 writel((np_tx_fifo_sz << 16) | rx_fifo_sz,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200485 &reg->gnptxfsiz);
486
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100487 for (i = 1; i < DWC2_MAX_HW_ENDPOINTS; i++)
Xu Ziyuanb729dc02016-07-14 14:52:33 +0800488 writel((rx_fifo_sz + np_tx_fifo_sz + tx_fifo_sz*(i-1)) |
489 tx_fifo_sz << 16, &reg->dieptxf[i-1]);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200490
491 /* Flush the RX FIFO */
492 writel(RX_FIFO_FLUSH, &reg->grstctl);
493 while (readl(&reg->grstctl) & RX_FIFO_FLUSH)
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100494 debug("%s: waiting for DWC2_UDC_OTG_GRSTCTL\n", __func__);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200495
496 /* Flush all the Tx FIFO's */
497 writel(TX_FIFO_FLUSH_ALL, &reg->grstctl);
498 writel(TX_FIFO_FLUSH_ALL | TX_FIFO_FLUSH, &reg->grstctl);
499 while (readl(&reg->grstctl) & TX_FIFO_FLUSH)
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100500 debug("%s: waiting for DWC2_UDC_OTG_GRSTCTL\n", __func__);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200501
502 /* 13. Clear NAK bit of EP0, EP1, EP2*/
503 /* For Slave mode*/
504 /* EP0: Control OUT */
505 writel(DEPCTL_EPDIS | DEPCTL_CNAK,
506 &reg->out_endp[EP0_CON].doepctl);
507
508 /* 14. Initialize OTG Link Core.*/
509 writel(GAHBCFG_INIT, &reg->gahbcfg);
510}
511
Marek Vasut1a639ff2015-12-04 00:57:58 +0100512static void set_max_pktsize(struct dwc2_udc *dev, enum usb_device_speed speed)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200513{
514 unsigned int ep_ctrl;
515 int i;
516
517 if (speed == USB_SPEED_HIGH) {
518 ep0_fifo_size = 64;
519 ep_fifo_size = 512;
520 ep_fifo_size2 = 1024;
521 dev->gadget.speed = USB_SPEED_HIGH;
522 } else {
523 ep0_fifo_size = 64;
524 ep_fifo_size = 64;
525 ep_fifo_size2 = 64;
526 dev->gadget.speed = USB_SPEED_FULL;
527 }
528
529 dev->ep[0].ep.maxpacket = ep0_fifo_size;
Marek Vasutcb9c5d02015-12-04 02:44:33 +0100530 for (i = 1; i < DWC2_MAX_ENDPOINTS; i++)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200531 dev->ep[i].ep.maxpacket = ep_fifo_size;
532
533 /* EP0 - Control IN (64 bytes)*/
534 ep_ctrl = readl(&reg->in_endp[EP0_CON].diepctl);
535 writel(ep_ctrl|(0<<0), &reg->in_endp[EP0_CON].diepctl);
536
537 /* EP0 - Control OUT (64 bytes)*/
538 ep_ctrl = readl(&reg->out_endp[EP0_CON].doepctl);
539 writel(ep_ctrl|(0<<0), &reg->out_endp[EP0_CON].doepctl);
540}
541
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100542static int dwc2_ep_enable(struct usb_ep *_ep,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200543 const struct usb_endpoint_descriptor *desc)
544{
Marek Vasut5b309f72015-12-04 01:48:57 +0100545 struct dwc2_ep *ep;
Marek Vasut1a639ff2015-12-04 00:57:58 +0100546 struct dwc2_udc *dev;
Bo Shen60dc40c2014-08-27 17:28:18 +0800547 unsigned long flags = 0;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200548
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000549 debug("%s: %p\n", __func__, _ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200550
Marek Vasut5b309f72015-12-04 01:48:57 +0100551 ep = container_of(_ep, struct dwc2_ep, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200552 if (!_ep || !desc || ep->desc || _ep->name == ep0name
553 || desc->bDescriptorType != USB_DT_ENDPOINT
554 || ep->bEndpointAddress != desc->bEndpointAddress
Tom Rini73424522011-12-15 08:40:51 -0700555 || ep_maxpacket(ep) <
556 le16_to_cpu(get_unaligned(&desc->wMaxPacketSize))) {
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200557
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000558 debug("%s: bad ep or descriptor\n", __func__);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200559 return -EINVAL;
560 }
561
562 /* xfer types must match, except that interrupt ~= bulk */
563 if (ep->bmAttributes != desc->bmAttributes
564 && ep->bmAttributes != USB_ENDPOINT_XFER_BULK
565 && desc->bmAttributes != USB_ENDPOINT_XFER_INT) {
566
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000567 debug("%s: %s type mismatch\n", __func__, _ep->name);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200568 return -EINVAL;
569 }
570
571 /* hardware _could_ do smaller, but driver doesn't */
Frank Wang0d64b5b2016-01-27 12:39:40 -0800572 if ((desc->bmAttributes == USB_ENDPOINT_XFER_BULK &&
573 le16_to_cpu(get_unaligned(&desc->wMaxPacketSize)) >
Tom Rini73424522011-12-15 08:40:51 -0700574 ep_maxpacket(ep)) || !get_unaligned(&desc->wMaxPacketSize)) {
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200575
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000576 debug("%s: bad %s maxpacket\n", __func__, _ep->name);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200577 return -ERANGE;
578 }
579
580 dev = ep->dev;
581 if (!dev->driver || dev->gadget.speed == USB_SPEED_UNKNOWN) {
582
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000583 debug("%s: bogus device state\n", __func__);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200584 return -ESHUTDOWN;
585 }
586
587 ep->stopped = 0;
588 ep->desc = desc;
589 ep->pio_irqs = 0;
Tom Rini73424522011-12-15 08:40:51 -0700590 ep->ep.maxpacket = le16_to_cpu(get_unaligned(&desc->wMaxPacketSize));
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200591
592 /* Reset halt state */
Marek Vasutfa03c812015-12-04 02:03:45 +0100593 dwc2_udc_set_nak(ep);
594 dwc2_udc_set_halt(_ep, 0);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200595
596 spin_lock_irqsave(&ep->dev->lock, flags);
Marek Vasutfa03c812015-12-04 02:03:45 +0100597 dwc2_udc_ep_activate(ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200598 spin_unlock_irqrestore(&ep->dev->lock, flags);
599
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000600 debug("%s: enabled %s, stopped = %d, maxpacket = %d\n",
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200601 __func__, _ep->name, ep->stopped, ep->ep.maxpacket);
602 return 0;
603}
604
605/*
606 * Disable EP
607 */
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100608static int dwc2_ep_disable(struct usb_ep *_ep)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200609{
Marek Vasut5b309f72015-12-04 01:48:57 +0100610 struct dwc2_ep *ep;
Bo Shen60dc40c2014-08-27 17:28:18 +0800611 unsigned long flags = 0;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200612
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000613 debug("%s: %p\n", __func__, _ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200614
Marek Vasut5b309f72015-12-04 01:48:57 +0100615 ep = container_of(_ep, struct dwc2_ep, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200616 if (!_ep || !ep->desc) {
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000617 debug("%s: %s not enabled\n", __func__,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200618 _ep ? ep->ep.name : NULL);
619 return -EINVAL;
620 }
621
622 spin_lock_irqsave(&ep->dev->lock, flags);
623
624 /* Nuke all pending requests */
625 nuke(ep, -ESHUTDOWN);
626
627 ep->desc = 0;
628 ep->stopped = 1;
629
630 spin_unlock_irqrestore(&ep->dev->lock, flags);
631
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000632 debug("%s: disabled %s\n", __func__, _ep->name);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200633 return 0;
634}
635
Marek Vasut153150f2015-12-04 02:17:40 +0100636static struct usb_request *dwc2_alloc_request(struct usb_ep *ep,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200637 gfp_t gfp_flags)
638{
Marek Vasut32f931c2015-12-04 01:51:07 +0100639 struct dwc2_request *req;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200640
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000641 debug("%s: %s %p\n", __func__, ep->name, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200642
Mike Frysinger1f19e3f2012-04-26 02:34:44 +0000643 req = memalign(CONFIG_SYS_CACHELINE_SIZE, sizeof(*req));
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200644 if (!req)
645 return 0;
646
647 memset(req, 0, sizeof *req);
648 INIT_LIST_HEAD(&req->queue);
649
650 return &req->req;
651}
652
Marek Vasut153150f2015-12-04 02:17:40 +0100653static void dwc2_free_request(struct usb_ep *ep, struct usb_request *_req)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200654{
Marek Vasut32f931c2015-12-04 01:51:07 +0100655 struct dwc2_request *req;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200656
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000657 debug("%s: %p\n", __func__, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200658
Marek Vasut32f931c2015-12-04 01:51:07 +0100659 req = container_of(_req, struct dwc2_request, req);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200660 WARN_ON(!list_empty(&req->queue));
661 kfree(req);
662}
663
664/* dequeue JUST ONE request */
Marek Vasut153150f2015-12-04 02:17:40 +0100665static int dwc2_dequeue(struct usb_ep *_ep, struct usb_request *_req)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200666{
Marek Vasut5b309f72015-12-04 01:48:57 +0100667 struct dwc2_ep *ep;
Marek Vasut32f931c2015-12-04 01:51:07 +0100668 struct dwc2_request *req;
Bo Shen60dc40c2014-08-27 17:28:18 +0800669 unsigned long flags = 0;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200670
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000671 debug("%s: %p\n", __func__, _ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200672
Marek Vasut5b309f72015-12-04 01:48:57 +0100673 ep = container_of(_ep, struct dwc2_ep, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200674 if (!_ep || ep->ep.name == ep0name)
675 return -EINVAL;
676
677 spin_lock_irqsave(&ep->dev->lock, flags);
678
679 /* make sure it's actually queued on this endpoint */
680 list_for_each_entry(req, &ep->queue, queue) {
681 if (&req->req == _req)
682 break;
683 }
684 if (&req->req != _req) {
685 spin_unlock_irqrestore(&ep->dev->lock, flags);
686 return -EINVAL;
687 }
688
689 done(ep, req, -ECONNRESET);
690
691 spin_unlock_irqrestore(&ep->dev->lock, flags);
692 return 0;
693}
694
695/*
696 * Return bytes in EP FIFO
697 */
Marek Vasut153150f2015-12-04 02:17:40 +0100698static int dwc2_fifo_status(struct usb_ep *_ep)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200699{
700 int count = 0;
Marek Vasut5b309f72015-12-04 01:48:57 +0100701 struct dwc2_ep *ep;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200702
Marek Vasut5b309f72015-12-04 01:48:57 +0100703 ep = container_of(_ep, struct dwc2_ep, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200704 if (!_ep) {
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000705 debug("%s: bad ep\n", __func__);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200706 return -ENODEV;
707 }
708
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000709 debug("%s: %d\n", __func__, ep_index(ep));
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200710
711 /* LPD can't report unclaimed bytes from IN fifos */
712 if (ep_is_in(ep))
713 return -EOPNOTSUPP;
714
715 return count;
716}
717
718/*
719 * Flush EP FIFO
720 */
Marek Vasut153150f2015-12-04 02:17:40 +0100721static void dwc2_fifo_flush(struct usb_ep *_ep)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200722{
Marek Vasut5b309f72015-12-04 01:48:57 +0100723 struct dwc2_ep *ep;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200724
Marek Vasut5b309f72015-12-04 01:48:57 +0100725 ep = container_of(_ep, struct dwc2_ep, ep);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200726 if (unlikely(!_ep || (!ep->desc && ep->ep.name != ep0name))) {
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000727 debug("%s: bad ep\n", __func__);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200728 return;
729 }
730
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000731 debug("%s: %d\n", __func__, ep_index(ep));
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200732}
733
Marek Vasutfa03c812015-12-04 02:03:45 +0100734static const struct usb_gadget_ops dwc2_udc_ops = {
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200735 /* current versions must always be self-powered */
736};
737
Marek Vasut1a639ff2015-12-04 00:57:58 +0100738static struct dwc2_udc memory = {
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200739 .usb_address = 0,
740 .gadget = {
Marek Vasutfa03c812015-12-04 02:03:45 +0100741 .ops = &dwc2_udc_ops,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200742 .ep0 = &memory.ep[0].ep,
743 .name = driver_name,
744 },
745
746 /* control endpoint */
747 .ep[0] = {
748 .ep = {
749 .name = ep0name,
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100750 .ops = &dwc2_ep_ops,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200751 .maxpacket = EP0_FIFO_SIZE,
752 },
753 .dev = &memory,
754
755 .bEndpointAddress = 0,
756 .bmAttributes = 0,
757
758 .ep_type = ep_control,
759 },
760
761 /* first group of endpoints */
762 .ep[1] = {
763 .ep = {
764 .name = "ep1in-bulk",
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100765 .ops = &dwc2_ep_ops,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200766 .maxpacket = EP_FIFO_SIZE,
767 },
768 .dev = &memory,
769
770 .bEndpointAddress = USB_DIR_IN | 1,
771 .bmAttributes = USB_ENDPOINT_XFER_BULK,
772
773 .ep_type = ep_bulk_out,
774 .fifo_num = 1,
775 },
776
777 .ep[2] = {
778 .ep = {
779 .name = "ep2out-bulk",
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100780 .ops = &dwc2_ep_ops,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200781 .maxpacket = EP_FIFO_SIZE,
782 },
783 .dev = &memory,
784
785 .bEndpointAddress = USB_DIR_OUT | 2,
786 .bmAttributes = USB_ENDPOINT_XFER_BULK,
787
788 .ep_type = ep_bulk_in,
789 .fifo_num = 2,
790 },
791
792 .ep[3] = {
793 .ep = {
794 .name = "ep3in-int",
Marek Vasut4e7c29e2015-12-04 02:13:42 +0100795 .ops = &dwc2_ep_ops,
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200796 .maxpacket = EP_FIFO_SIZE,
797 },
798 .dev = &memory,
799
800 .bEndpointAddress = USB_DIR_IN | 3,
801 .bmAttributes = USB_ENDPOINT_XFER_INT,
802
803 .ep_type = ep_interrupt,
804 .fifo_num = 3,
805 },
806};
807
808/*
809 * probe - binds to the platform device
810 */
811
Marek Vasut01b61fa2015-12-04 02:26:33 +0100812int dwc2_udc_probe(struct dwc2_plat_otg_data *pdata)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200813{
Marek Vasut1a639ff2015-12-04 00:57:58 +0100814 struct dwc2_udc *dev = &memory;
Lukasz Majewski904f8f72014-02-05 10:10:44 +0100815 int retval = 0;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200816
Anatolij Gustschine63ce4f2011-12-19 04:20:35 +0000817 debug("%s: %p\n", __func__, pdata);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200818
819 dev->pdata = pdata;
820
Marek Vasut67ec9dc2015-12-04 01:11:45 +0100821 reg = (struct dwc2_usbotg_reg *)pdata->regs_otg;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200822
823 /* regs_otg = (void *)pdata->regs_otg; */
824
825 dev->gadget.is_dualspeed = 1; /* Hack only*/
826 dev->gadget.is_otg = 0;
827 dev->gadget.is_a_peripheral = 0;
828 dev->gadget.b_hnp_enable = 0;
829 dev->gadget.a_hnp_support = 0;
830 dev->gadget.a_alt_hnp_support = 0;
831
832 the_controller = dev;
833
Lukasz Majewski904f8f72014-02-05 10:10:44 +0100834 usb_ctrl = memalign(CONFIG_SYS_CACHELINE_SIZE,
835 ROUND(sizeof(struct usb_ctrlrequest),
836 CONFIG_SYS_CACHELINE_SIZE));
837 if (!usb_ctrl) {
838 error("No memory available for UDC!\n");
839 return -ENOMEM;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200840 }
Lukasz Majewski904f8f72014-02-05 10:10:44 +0100841
842 usb_ctrl_dma_addr = (dma_addr_t) usb_ctrl;
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200843
844 udc_reinit(dev);
845
846 return retval;
847}
848
Kishon Vijay Abraham I4763e162015-02-23 18:40:23 +0530849int usb_gadget_handle_interrupts(int index)
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200850{
851 u32 intr_status = readl(&reg->gintsts);
852 u32 gintmsk = readl(&reg->gintmsk);
853
854 if (intr_status & gintmsk)
Marek Vasutfa03c812015-12-04 02:03:45 +0100855 return dwc2_udc_irq(1, (void *)the_controller);
Lukasz Majewskiac3c8312011-10-27 10:36:46 +0200856 return 0;
857}