blob: aee44aa06479bfe0cf0bfe9b5d3dc4ee46ac457e [file] [log] [blame]
Vipin KUMAR62db1c02010-01-15 19:15:47 +05301/*
2 * Based on drivers/usb/gadget/omap1510_udc.c
3 * TI OMAP1510 USB bus interface driver
4 *
5 * (C) Copyright 2009
6 * Vipin Kumar, ST Micoelectronics, vipin.kumar@st.com.
7 *
8 * See file CREDITS for list of people who contributed to this
9 * project.
10 *
11 * This program is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU General Public License as
13 * published by the Free Software Foundation; either version 2 of
14 * the License, or (at your option) any later version.
15 *
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
24 * MA 02111-1307 USA
25 */
26
27#include <common.h>
28#include <asm/io.h>
29
30#include <usbdevice.h>
31#include "ep0.h"
Vipin KUMAR27215512012-03-06 23:39:37 +000032#include <usb/designware_udc.h>
Vipin KUMAR62db1c02010-01-15 19:15:47 +053033#include <asm/arch/hardware.h>
Vipin KUMAR62db1c02010-01-15 19:15:47 +053034
35#define UDC_INIT_MDELAY 80 /* Device settle delay */
36
37/* Some kind of debugging output... */
Vipin KUMAR27215512012-03-06 23:39:37 +000038#ifndef DEBUG_DWUSBTTY
Vipin KUMAR62db1c02010-01-15 19:15:47 +053039#define UDCDBG(str)
40#define UDCDBGA(fmt, args...)
41#else
42#define UDCDBG(str) serial_printf(str "\n")
43#define UDCDBGA(fmt, args...) serial_printf(fmt "\n", ##args)
44#endif
45
46static struct urb *ep0_urb;
47static struct usb_device_instance *udc_device;
48
49static struct plug_regs *const plug_regs_p =
50 (struct plug_regs * const)CONFIG_SYS_PLUG_BASE;
51static struct udc_regs *const udc_regs_p =
52 (struct udc_regs * const)CONFIG_SYS_USBD_BASE;
53static struct udc_endp_regs *const outep_regs_p =
54 &((struct udc_regs * const)CONFIG_SYS_USBD_BASE)->out_regs[0];
55static struct udc_endp_regs *const inep_regs_p =
56 &((struct udc_regs * const)CONFIG_SYS_USBD_BASE)->in_regs[0];
57
58/*
59 * udc_state_transition - Write the next packet to TxFIFO.
60 * @initial: Initial state.
61 * @final: Final state.
62 *
63 * Helper function to implement device state changes. The device states and
64 * the events that transition between them are:
65 *
66 * STATE_ATTACHED
67 * || /\
68 * \/ ||
69 * DEVICE_HUB_CONFIGURED DEVICE_HUB_RESET
70 * || /\
71 * \/ ||
72 * STATE_POWERED
73 * || /\
74 * \/ ||
75 * DEVICE_RESET DEVICE_POWER_INTERRUPTION
76 * || /\
77 * \/ ||
78 * STATE_DEFAULT
79 * || /\
80 * \/ ||
81 * DEVICE_ADDRESS_ASSIGNED DEVICE_RESET
82 * || /\
83 * \/ ||
84 * STATE_ADDRESSED
85 * || /\
86 * \/ ||
87 * DEVICE_CONFIGURED DEVICE_DE_CONFIGURED
88 * || /\
89 * \/ ||
90 * STATE_CONFIGURED
91 *
92 * udc_state_transition transitions up (in the direction from STATE_ATTACHED
93 * to STATE_CONFIGURED) from the specified initial state to the specified final
94 * state, passing through each intermediate state on the way. If the initial
95 * state is at or above (i.e. nearer to STATE_CONFIGURED) the final state, then
96 * no state transitions will take place.
97 *
98 * udc_state_transition also transitions down (in the direction from
99 * STATE_CONFIGURED to STATE_ATTACHED) from the specified initial state to the
100 * specified final state, passing through each intermediate state on the way.
101 * If the initial state is at or below (i.e. nearer to STATE_ATTACHED) the final
102 * state, then no state transitions will take place.
103 *
104 * This function must only be called with interrupts disabled.
105 */
106static void udc_state_transition(usb_device_state_t initial,
107 usb_device_state_t final)
108{
109 if (initial < final) {
110 switch (initial) {
111 case STATE_ATTACHED:
112 usbd_device_event_irq(udc_device,
113 DEVICE_HUB_CONFIGURED, 0);
114 if (final == STATE_POWERED)
115 break;
116 case STATE_POWERED:
117 usbd_device_event_irq(udc_device, DEVICE_RESET, 0);
118 if (final == STATE_DEFAULT)
119 break;
120 case STATE_DEFAULT:
121 usbd_device_event_irq(udc_device,
122 DEVICE_ADDRESS_ASSIGNED, 0);
123 if (final == STATE_ADDRESSED)
124 break;
125 case STATE_ADDRESSED:
126 usbd_device_event_irq(udc_device, DEVICE_CONFIGURED, 0);
127 case STATE_CONFIGURED:
128 break;
129 default:
130 break;
131 }
132 } else if (initial > final) {
133 switch (initial) {
134 case STATE_CONFIGURED:
135 usbd_device_event_irq(udc_device,
136 DEVICE_DE_CONFIGURED, 0);
137 if (final == STATE_ADDRESSED)
138 break;
139 case STATE_ADDRESSED:
140 usbd_device_event_irq(udc_device, DEVICE_RESET, 0);
141 if (final == STATE_DEFAULT)
142 break;
143 case STATE_DEFAULT:
144 usbd_device_event_irq(udc_device,
145 DEVICE_POWER_INTERRUPTION, 0);
146 if (final == STATE_POWERED)
147 break;
148 case STATE_POWERED:
149 usbd_device_event_irq(udc_device, DEVICE_HUB_RESET, 0);
150 case STATE_ATTACHED:
151 break;
152 default:
153 break;
154 }
155 }
156}
157
158/* Stall endpoint */
159static void udc_stall_ep(u32 ep_num)
160{
161 writel(readl(&inep_regs_p[ep_num].endp_cntl) | ENDP_CNTL_STALL,
162 &inep_regs_p[ep_num].endp_cntl);
163
164 writel(readl(&outep_regs_p[ep_num].endp_cntl) | ENDP_CNTL_STALL,
165 &outep_regs_p[ep_num].endp_cntl);
166}
167
168static void *get_fifo(int ep_num, int in)
169{
170 u32 *fifo_ptr = (u32 *)CONFIG_SYS_FIFO_BASE;
171
172 switch (ep_num) {
173 case UDC_EP3:
174 fifo_ptr += readl(&inep_regs_p[1].endp_bsorfn);
175 /* break intentionally left out */
176
177 case UDC_EP1:
178 fifo_ptr += readl(&inep_regs_p[0].endp_bsorfn);
179 /* break intentionally left out */
180
181 case UDC_EP0:
182 default:
183 if (in) {
184 fifo_ptr +=
185 readl(&outep_regs_p[2].endp_maxpacksize) >> 16;
186 /* break intentionally left out */
187 } else {
188 break;
189 }
190
191 case UDC_EP2:
192 fifo_ptr += readl(&outep_regs_p[0].endp_maxpacksize) >> 16;
193 /* break intentionally left out */
194 }
195
196 return (void *)fifo_ptr;
197}
198
199static int usbgetpckfromfifo(int epNum, u8 *bufp, u32 len)
200{
201 u8 *fifo_ptr = (u8 *)get_fifo(epNum, 0);
202 u32 i, nw, nb;
203 u32 *wrdp;
204 u8 *bytp;
205
206 if (readl(&udc_regs_p->dev_stat) & DEV_STAT_RXFIFO_EMPTY)
207 return -1;
208
209 nw = len / sizeof(u32);
210 nb = len % sizeof(u32);
211
212 wrdp = (u32 *)bufp;
213 for (i = 0; i < nw; i++) {
214 writel(readl(fifo_ptr), wrdp);
215 wrdp++;
216 }
217
218 bytp = (u8 *)wrdp;
219 for (i = 0; i < nb; i++) {
220 writeb(readb(fifo_ptr), bytp);
221 fifo_ptr++;
222 bytp++;
223 }
224 readl(&outep_regs_p[epNum].write_done);
225
226 return 0;
227}
228
229static void usbputpcktofifo(int epNum, u8 *bufp, u32 len)
230{
231 u32 i, nw, nb;
232 u32 *wrdp;
233 u8 *bytp;
234 u8 *fifo_ptr = get_fifo(epNum, 1);
235
236 nw = len / sizeof(int);
237 nb = len % sizeof(int);
238 wrdp = (u32 *)bufp;
239 for (i = 0; i < nw; i++) {
240 writel(*wrdp, fifo_ptr);
241 wrdp++;
242 }
243
244 bytp = (u8 *)wrdp;
245 for (i = 0; i < nb; i++) {
246 writeb(*bytp, fifo_ptr);
247 fifo_ptr++;
248 bytp++;
249 }
250}
251
252/*
Vipin KUMAR27215512012-03-06 23:39:37 +0000253 * dw_write_noniso_tx_fifo - Write the next packet to TxFIFO.
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530254 * @endpoint: Endpoint pointer.
255 *
256 * If the endpoint has an active tx_urb, then the next packet of data from the
257 * URB is written to the tx FIFO. The total amount of data in the urb is given
258 * by urb->actual_length. The maximum amount of data that can be sent in any
259 * one packet is given by endpoint->tx_packetSize. The number of data bytes
260 * from this URB that have already been transmitted is given by endpoint->sent.
261 * endpoint->last is updated by this routine with the number of data bytes
262 * transmitted in this packet.
263 *
264 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000265static void dw_write_noniso_tx_fifo(struct usb_endpoint_instance
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530266 *endpoint)
267{
268 struct urb *urb = endpoint->tx_urb;
269 int align;
270
271 if (urb) {
272 u32 last;
273
274 UDCDBGA("urb->buffer %p, buffer_length %d, actual_length %d",
275 urb->buffer, urb->buffer_length, urb->actual_length);
276
277 last = MIN(urb->actual_length - endpoint->sent,
278 endpoint->tx_packetSize);
279
280 if (last) {
281 u8 *cp = urb->buffer + endpoint->sent;
282
283 /*
284 * This ensures that USBD packet fifo is accessed
285 * - through word aligned pointer or
286 * - through non word aligned pointer but only
287 * with a max length to make the next packet
288 * word aligned
289 */
290
291 align = ((ulong)cp % sizeof(int));
292 if (align)
293 last = MIN(last, sizeof(int) - align);
294
295 UDCDBGA("endpoint->sent %d, tx_packetSize %d, last %d",
296 endpoint->sent, endpoint->tx_packetSize, last);
297
298 usbputpcktofifo(endpoint->endpoint_address &
299 USB_ENDPOINT_NUMBER_MASK, cp, last);
300 }
301 endpoint->last = last;
302 }
303}
304
305/*
306 * Handle SETUP USB interrupt.
307 * This function implements TRM Figure 14-14.
308 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000309static void dw_udc_setup(struct usb_endpoint_instance *endpoint)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530310{
311 u8 *datap = (u8 *)&ep0_urb->device_request;
312 int ep_addr = endpoint->endpoint_address;
313
314 UDCDBG("-> Entering device setup");
315 usbgetpckfromfifo(ep_addr, datap, 8);
316
317 /* Try to process setup packet */
318 if (ep0_recv_setup(ep0_urb)) {
319 /* Not a setup packet, stall next EP0 transaction */
320 udc_stall_ep(0);
321 UDCDBG("can't parse setup packet, still waiting for setup");
322 return;
323 }
324
325 /* Check direction */
326 if ((ep0_urb->device_request.bmRequestType & USB_REQ_DIRECTION_MASK)
327 == USB_REQ_HOST2DEVICE) {
328 UDCDBG("control write on EP0");
329 if (le16_to_cpu(ep0_urb->device_request.wLength)) {
330 /* Stall this request */
331 UDCDBG("Stalling unsupported EP0 control write data "
332 "stage.");
333 udc_stall_ep(0);
334 }
335 } else {
336
337 UDCDBG("control read on EP0");
338 /*
339 * The ep0_recv_setup function has already placed our response
340 * packet data in ep0_urb->buffer and the packet length in
341 * ep0_urb->actual_length.
342 */
343 endpoint->tx_urb = ep0_urb;
344 endpoint->sent = 0;
345 /*
Vipin KUMAR27215512012-03-06 23:39:37 +0000346 * Write packet data to the FIFO. dw_write_noniso_tx_fifo
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530347 * will update endpoint->last with the number of bytes written
348 * to the FIFO.
349 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000350 dw_write_noniso_tx_fifo(endpoint);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530351
352 writel(0x0, &inep_regs_p[ep_addr].write_done);
353 }
354
355 udc_unset_nak(endpoint->endpoint_address);
356
357 UDCDBG("<- Leaving device setup");
358}
359
360/*
361 * Handle endpoint 0 RX interrupt
362 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000363static void dw_udc_ep0_rx(struct usb_endpoint_instance *endpoint)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530364{
365 u8 dummy[64];
366
367 UDCDBG("RX on EP0");
368
369 /* Check direction */
370 if ((ep0_urb->device_request.bmRequestType
371 & USB_REQ_DIRECTION_MASK) == USB_REQ_HOST2DEVICE) {
372 /*
373 * This rx interrupt must be for a control write data
374 * stage packet.
375 *
376 * We don't support control write data stages.
377 * We should never end up here.
378 */
379
380 UDCDBG("Stalling unexpected EP0 control write "
381 "data stage packet");
382 udc_stall_ep(0);
383 } else {
384 /*
385 * This rx interrupt must be for a control read status
386 * stage packet.
387 */
388 UDCDBG("ACK on EP0 control read status stage packet");
389 u32 len = (readl(&outep_regs_p[0].endp_status) >> 11) & 0xfff;
390 usbgetpckfromfifo(0, dummy, len);
391 }
392}
393
394/*
395 * Handle endpoint 0 TX interrupt
396 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000397static void dw_udc_ep0_tx(struct usb_endpoint_instance *endpoint)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530398{
399 struct usb_device_request *request = &ep0_urb->device_request;
400 int ep_addr;
401
402 UDCDBG("TX on EP0");
403
404 /* Check direction */
405 if ((request->bmRequestType & USB_REQ_DIRECTION_MASK) ==
406 USB_REQ_HOST2DEVICE) {
407 /*
408 * This tx interrupt must be for a control write status
409 * stage packet.
410 */
411 UDCDBG("ACK on EP0 control write status stage packet");
412 } else {
413 /*
414 * This tx interrupt must be for a control read data
415 * stage packet.
416 */
417 int wLength = le16_to_cpu(request->wLength);
418
419 /*
420 * Update our count of bytes sent so far in this
421 * transfer.
422 */
423 endpoint->sent += endpoint->last;
424
425 /*
426 * We are finished with this transfer if we have sent
427 * all of the bytes in our tx urb (urb->actual_length)
428 * unless we need a zero-length terminating packet. We
429 * need a zero-length terminating packet if we returned
430 * fewer bytes than were requested (wLength) by the host,
431 * and the number of bytes we returned is an exact
432 * multiple of the packet size endpoint->tx_packetSize.
433 */
434 if ((endpoint->sent == ep0_urb->actual_length) &&
435 ((ep0_urb->actual_length == wLength) ||
436 (endpoint->last != endpoint->tx_packetSize))) {
437 /* Done with control read data stage. */
438 UDCDBG("control read data stage complete");
439 } else {
440 /*
441 * We still have another packet of data to send
442 * in this control read data stage or else we
443 * need a zero-length terminating packet.
444 */
445 UDCDBG("ACK control read data stage packet");
Vipin KUMAR27215512012-03-06 23:39:37 +0000446 dw_write_noniso_tx_fifo(endpoint);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530447
448 ep_addr = endpoint->endpoint_address;
449 writel(0x0, &inep_regs_p[ep_addr].write_done);
450 }
451 }
452}
453
Vipin KUMAR27215512012-03-06 23:39:37 +0000454static struct usb_endpoint_instance *dw_find_ep(int ep)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530455{
456 int i;
457
458 for (i = 0; i < udc_device->bus->max_endpoints; i++) {
459 if ((udc_device->bus->endpoint_array[i].endpoint_address &
460 USB_ENDPOINT_NUMBER_MASK) == ep)
461 return &udc_device->bus->endpoint_array[i];
462 }
463 return NULL;
464}
465
466/*
467 * Handle RX transaction on non-ISO endpoint.
468 * The ep argument is a physical endpoint number for a non-ISO IN endpoint
469 * in the range 1 to 15.
470 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000471static void dw_udc_epn_rx(int ep)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530472{
473 int nbytes = 0;
474 struct urb *urb;
Vipin KUMAR27215512012-03-06 23:39:37 +0000475 struct usb_endpoint_instance *endpoint = dw_find_ep(ep);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530476
477 if (endpoint) {
478 urb = endpoint->rcv_urb;
479
480 if (urb) {
481 u8 *cp = urb->buffer + urb->actual_length;
482
483 nbytes = (readl(&outep_regs_p[ep].endp_status) >> 11) &
484 0xfff;
485 usbgetpckfromfifo(ep, cp, nbytes);
486 usbd_rcv_complete(endpoint, nbytes, 0);
487 }
488 }
489}
490
491/*
492 * Handle TX transaction on non-ISO endpoint.
493 * The ep argument is a physical endpoint number for a non-ISO IN endpoint
494 * in the range 16 to 30.
495 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000496static void dw_udc_epn_tx(int ep)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530497{
Vipin KUMAR27215512012-03-06 23:39:37 +0000498 struct usb_endpoint_instance *endpoint = dw_find_ep(ep);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530499
500 /*
501 * We need to transmit a terminating zero-length packet now if
502 * we have sent all of the data in this URB and the transfer
503 * size was an exact multiple of the packet size.
504 */
505 if (endpoint && endpoint->tx_urb && endpoint->tx_urb->actual_length) {
506 if (endpoint->last == endpoint->tx_packetSize) {
507 /* handle zero length packet here */
508 writel(0x0, &inep_regs_p[ep].write_done);
509 }
510 /* retire the data that was just sent */
511 usbd_tx_complete(endpoint);
512 /*
513 * Check to see if we have more data ready to transmit
514 * now.
515 */
516 if (endpoint->tx_urb && endpoint->tx_urb->actual_length) {
517 /* write data to FIFO */
Vipin KUMAR27215512012-03-06 23:39:37 +0000518 dw_write_noniso_tx_fifo(endpoint);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530519 writel(0x0, &inep_regs_p[ep].write_done);
520
521 } else if (endpoint->tx_urb
522 && (endpoint->tx_urb->actual_length == 0)) {
523 /* udc_set_nak(ep); */
524 }
525 }
526}
527
528/*
529 * Start of public functions.
530 */
531
532/* Called to start packet transmission. */
533int udc_endpoint_write(struct usb_endpoint_instance *endpoint)
534{
535 udc_unset_nak(endpoint->endpoint_address & USB_ENDPOINT_NUMBER_MASK);
536 return 0;
537}
538
539/* Start to initialize h/w stuff */
540int udc_init(void)
541{
542 int i;
543 u32 plug_st;
544
545 udc_device = NULL;
546
547 UDCDBG("starting");
548
549 readl(&plug_regs_p->plug_pending);
550
551 udc_disconnect();
552
553 for (i = 0; i < UDC_INIT_MDELAY; i++)
554 udelay(1000);
555
556 plug_st = readl(&plug_regs_p->plug_state);
557 writel(plug_st | PLUG_STATUS_EN, &plug_regs_p->plug_state);
558
559 writel(~0x0, &udc_regs_p->endp_int);
560 writel(~0x0, &udc_regs_p->dev_int_mask);
561 writel(~0x0, &udc_regs_p->endp_int_mask);
562
563 writel(DEV_CONF_FS_SPEED | DEV_CONF_REMWAKEUP | DEV_CONF_SELFPOW |
564 /* Dev_Conf_SYNCFRAME | */
565 DEV_CONF_PHYINT_16, &udc_regs_p->dev_conf);
566
567 writel(0x0, &udc_regs_p->dev_cntl);
568
569 /* Clear all interrupts pending */
570 writel(DEV_INT_MSK, &udc_regs_p->dev_int);
571
572 return 0;
573}
574
575/*
576 * udc_setup_ep - setup endpoint
577 * Associate a physical endpoint with endpoint_instance
578 */
579void udc_setup_ep(struct usb_device_instance *device,
580 u32 ep, struct usb_endpoint_instance *endpoint)
581{
582 UDCDBGA("setting up endpoint addr %x", endpoint->endpoint_address);
583 int ep_addr;
584 int ep_num, ep_type;
585 int packet_size;
586 int buffer_size;
587 int attributes;
588 char *tt;
589 u32 endp_intmask;
590
591 tt = getenv("usbtty");
592 if (!tt)
593 tt = "generic";
594
595 ep_addr = endpoint->endpoint_address;
596 ep_num = ep_addr & USB_ENDPOINT_NUMBER_MASK;
597
598 if ((ep_addr & USB_ENDPOINT_DIR_MASK) == USB_DIR_IN) {
599 /* IN endpoint */
600 packet_size = endpoint->tx_packetSize;
601 buffer_size = packet_size * 2;
602 attributes = endpoint->tx_attributes;
603 } else {
604 /* OUT endpoint */
605 packet_size = endpoint->rcv_packetSize;
606 buffer_size = packet_size * 2;
607 attributes = endpoint->rcv_attributes;
608 }
609
610 switch (attributes & USB_ENDPOINT_XFERTYPE_MASK) {
611 case USB_ENDPOINT_XFER_CONTROL:
612 ep_type = ENDP_EPTYPE_CNTL;
613 break;
614 case USB_ENDPOINT_XFER_BULK:
615 default:
616 ep_type = ENDP_EPTYPE_BULK;
617 break;
618 case USB_ENDPOINT_XFER_INT:
619 ep_type = ENDP_EPTYPE_INT;
620 break;
621 case USB_ENDPOINT_XFER_ISOC:
622 ep_type = ENDP_EPTYPE_ISO;
623 break;
624 }
625
626 struct udc_endp_regs *out_p = &outep_regs_p[ep_num];
627 struct udc_endp_regs *in_p = &inep_regs_p[ep_num];
628
629 if (!ep_addr) {
630 /* Setup endpoint 0 */
631 buffer_size = packet_size;
632
633 writel(readl(&in_p->endp_cntl) | ENDP_CNTL_CNAK,
634 &in_p->endp_cntl);
635
636 writel(readl(&out_p->endp_cntl) | ENDP_CNTL_CNAK,
637 &out_p->endp_cntl);
638
639 writel(ENDP_CNTL_CONTROL | ENDP_CNTL_FLUSH, &in_p->endp_cntl);
640
641 writel(buffer_size / sizeof(int), &in_p->endp_bsorfn);
642
643 writel(packet_size, &in_p->endp_maxpacksize);
644
645 writel(ENDP_CNTL_CONTROL | ENDP_CNTL_RRDY, &out_p->endp_cntl);
646
647 writel(packet_size | ((buffer_size / sizeof(int)) << 16),
648 &out_p->endp_maxpacksize);
649
650 writel((packet_size << 19) | ENDP_EPTYPE_CNTL,
651 &udc_regs_p->udc_endp_reg[ep_num]);
652
653 } else if ((ep_addr & USB_ENDPOINT_DIR_MASK) == USB_DIR_IN) {
654 /* Setup the IN endpoint */
655 writel(0x0, &in_p->endp_status);
656 writel((ep_type << 4) | ENDP_CNTL_RRDY, &in_p->endp_cntl);
657 writel(buffer_size / sizeof(int), &in_p->endp_bsorfn);
658 writel(packet_size, &in_p->endp_maxpacksize);
659
660 if (!strcmp(tt, "cdc_acm")) {
661 if (ep_type == ENDP_EPTYPE_INT) {
662 /* Conf no. 1 Interface no. 0 */
663 writel((packet_size << 19) |
664 ENDP_EPDIR_IN | (1 << 7) |
665 (0 << 11) | (ep_type << 5) | ep_num,
666 &udc_regs_p->udc_endp_reg[ep_num]);
667 } else {
668 /* Conf no. 1 Interface no. 1 */
669 writel((packet_size << 19) |
670 ENDP_EPDIR_IN | (1 << 7) |
671 (1 << 11) | (ep_type << 5) | ep_num,
672 &udc_regs_p->udc_endp_reg[ep_num]);
673 }
674 } else {
675 /* Conf no. 1 Interface no. 0 */
676 writel((packet_size << 19) |
677 ENDP_EPDIR_IN | (1 << 7) |
678 (0 << 11) | (ep_type << 5) | ep_num,
679 &udc_regs_p->udc_endp_reg[ep_num]);
680 }
681
682 } else {
683 /* Setup the OUT endpoint */
684 writel(0x0, &out_p->endp_status);
685 writel((ep_type << 4) | ENDP_CNTL_RRDY, &out_p->endp_cntl);
686 writel(packet_size | ((buffer_size / sizeof(int)) << 16),
687 &out_p->endp_maxpacksize);
688
689 if (!strcmp(tt, "cdc_acm")) {
690 writel((packet_size << 19) |
691 ENDP_EPDIR_OUT | (1 << 7) |
692 (1 << 11) | (ep_type << 5) | ep_num,
693 &udc_regs_p->udc_endp_reg[ep_num]);
694 } else {
695 writel((packet_size << 19) |
696 ENDP_EPDIR_OUT | (1 << 7) |
697 (0 << 11) | (ep_type << 5) | ep_num,
698 &udc_regs_p->udc_endp_reg[ep_num]);
699 }
700
701 }
702
703 endp_intmask = readl(&udc_regs_p->endp_int_mask);
704 endp_intmask &= ~((1 << ep_num) | 0x10000 << ep_num);
705 writel(endp_intmask, &udc_regs_p->endp_int_mask);
706}
707
708/* Turn on the USB connection by enabling the pullup resistor */
709void udc_connect(void)
710{
711 u32 plug_st;
712
713 plug_st = readl(&plug_regs_p->plug_state);
714 plug_st &= ~(PLUG_STATUS_PHY_RESET | PLUG_STATUS_PHY_MODE);
715 writel(plug_st, &plug_regs_p->plug_state);
716}
717
718/* Turn off the USB connection by disabling the pullup resistor */
719void udc_disconnect(void)
720{
721 u32 plug_st;
722
723 plug_st = readl(&plug_regs_p->plug_state);
724 plug_st |= (PLUG_STATUS_PHY_RESET | PLUG_STATUS_PHY_MODE);
725 writel(plug_st, &plug_regs_p->plug_state);
726}
727
728/* Switch on the UDC */
729void udc_enable(struct usb_device_instance *device)
730{
731 UDCDBGA("enable device %p, status %d", device, device->status);
732
733 /* Save the device structure pointer */
734 udc_device = device;
735
736 /* Setup ep0 urb */
737 if (!ep0_urb) {
738 ep0_urb =
739 usbd_alloc_urb(udc_device, udc_device->bus->endpoint_array);
740 } else {
741 serial_printf("udc_enable: ep0_urb already allocated %p\n",
742 ep0_urb);
743 }
744
745 writel(DEV_INT_SOF, &udc_regs_p->dev_int_mask);
746}
747
748/**
749 * udc_startup - allow udc code to do any additional startup
750 */
751void udc_startup_events(struct usb_device_instance *device)
752{
753 /* The DEVICE_INIT event puts the USB device in the state STATE_INIT. */
754 usbd_device_event_irq(device, DEVICE_INIT, 0);
755
756 /*
757 * The DEVICE_CREATE event puts the USB device in the state
758 * STATE_ATTACHED.
759 */
760 usbd_device_event_irq(device, DEVICE_CREATE, 0);
761
762 /*
763 * Some USB controller driver implementations signal
764 * DEVICE_HUB_CONFIGURED and DEVICE_RESET events here.
765 * DEVICE_HUB_CONFIGURED causes a transition to the state STATE_POWERED,
766 * and DEVICE_RESET causes a transition to the state STATE_DEFAULT.
Vipin KUMAR27215512012-03-06 23:39:37 +0000767 * The DW USB client controller has the capability to detect when the
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530768 * USB cable is connected to a powered USB bus, so we will defer the
769 * DEVICE_HUB_CONFIGURED and DEVICE_RESET events until later.
770 */
771
772 udc_enable(device);
773}
774
775/*
776 * Plug detection interrupt handling
777 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000778void dw_udc_plug_irq(void)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530779{
780 if (readl(&plug_regs_p->plug_state) & PLUG_STATUS_ATTACHED) {
781 /*
782 * USB cable attached
783 * Turn off PHY reset bit (PLUG detect).
784 * Switch PHY opmode to normal operation (PLUG detect).
785 */
786 udc_connect();
787 writel(DEV_INT_SOF, &udc_regs_p->dev_int_mask);
788
789 UDCDBG("device attached and powered");
790 udc_state_transition(udc_device->device_state, STATE_POWERED);
791 } else {
792 /*
793 * USB cable detached
794 * Reset the PHY and switch the mode.
795 */
796 udc_disconnect();
797 writel(~0x0, &udc_regs_p->dev_int_mask);
798
799 UDCDBG("device detached or unpowered");
800 udc_state_transition(udc_device->device_state, STATE_ATTACHED);
801 }
802}
803
804/*
805 * Device interrupt handling
806 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000807void dw_udc_dev_irq(void)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530808{
809 if (readl(&udc_regs_p->dev_int) & DEV_INT_USBRESET) {
810 writel(~0x0, &udc_regs_p->endp_int_mask);
811
812 udc_connect();
813
814 writel(readl(&inep_regs_p[0].endp_cntl) | ENDP_CNTL_FLUSH,
815 &inep_regs_p[0].endp_cntl);
816
817 writel(DEV_INT_USBRESET, &udc_regs_p->dev_int);
818
819 UDCDBG("device reset in progess");
820 udc_state_transition(udc_device->device_state, STATE_DEFAULT);
821 }
822
823 /* Device Enumeration completed */
824 if (readl(&udc_regs_p->dev_int) & DEV_INT_ENUM) {
825 writel(DEV_INT_ENUM, &udc_regs_p->dev_int);
826
827 /* Endpoint interrupt enabled for Ctrl IN & Ctrl OUT */
828 writel(readl(&udc_regs_p->endp_int_mask) & ~0x10001,
829 &udc_regs_p->endp_int_mask);
830
831 UDCDBG("default -> addressed");
832 udc_state_transition(udc_device->device_state, STATE_ADDRESSED);
833 }
834
835 /* The USB will be in SUSPEND in 3 ms */
836 if (readl(&udc_regs_p->dev_int) & DEV_INT_INACTIVE) {
837 writel(DEV_INT_INACTIVE, &udc_regs_p->dev_int);
838
839 UDCDBG("entering inactive state");
840 /* usbd_device_event_irq(udc_device, DEVICE_BUS_INACTIVE, 0); */
841 }
842
843 /* SetConfiguration command received */
844 if (readl(&udc_regs_p->dev_int) & DEV_INT_SETCFG) {
845 writel(DEV_INT_SETCFG, &udc_regs_p->dev_int);
846
847 UDCDBG("entering configured state");
848 udc_state_transition(udc_device->device_state,
849 STATE_CONFIGURED);
850 }
851
852 /* SetInterface command received */
853 if (readl(&udc_regs_p->dev_int) & DEV_INT_SETINTF)
854 writel(DEV_INT_SETINTF, &udc_regs_p->dev_int);
855
856 /* USB Suspend detected on cable */
857 if (readl(&udc_regs_p->dev_int) & DEV_INT_SUSPUSB) {
858 writel(DEV_INT_SUSPUSB, &udc_regs_p->dev_int);
859
860 UDCDBG("entering suspended state");
861 usbd_device_event_irq(udc_device, DEVICE_BUS_INACTIVE, 0);
862 }
863
864 /* USB Start-Of-Frame detected on cable */
865 if (readl(&udc_regs_p->dev_int) & DEV_INT_SOF)
866 writel(DEV_INT_SOF, &udc_regs_p->dev_int);
867}
868
869/*
870 * Endpoint interrupt handling
871 */
Vipin KUMAR27215512012-03-06 23:39:37 +0000872void dw_udc_endpoint_irq(void)
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530873{
874 while (readl(&udc_regs_p->endp_int) & ENDP0_INT_CTRLOUT) {
875
876 writel(ENDP0_INT_CTRLOUT, &udc_regs_p->endp_int);
877
878 if ((readl(&outep_regs_p[0].endp_status) & ENDP_STATUS_OUTMSK)
879 == ENDP_STATUS_OUT_SETUP) {
Vipin KUMAR27215512012-03-06 23:39:37 +0000880 dw_udc_setup(udc_device->bus->endpoint_array + 0);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530881 writel(ENDP_STATUS_OUT_SETUP,
882 &outep_regs_p[0].endp_status);
883
884 } else if ((readl(&outep_regs_p[0].endp_status) &
885 ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_DATA) {
Vipin KUMAR27215512012-03-06 23:39:37 +0000886 dw_udc_ep0_rx(udc_device->bus->endpoint_array + 0);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530887 writel(ENDP_STATUS_OUT_DATA,
888 &outep_regs_p[0].endp_status);
889
890 } else if ((readl(&outep_regs_p[0].endp_status) &
891 ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_NONE) {
892 /* NONE received */
893 }
894
895 writel(0x0, &outep_regs_p[0].endp_status);
896 }
897
898 if (readl(&udc_regs_p->endp_int) & ENDP0_INT_CTRLIN) {
Vipin KUMAR27215512012-03-06 23:39:37 +0000899 dw_udc_ep0_tx(udc_device->bus->endpoint_array + 0);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530900
901 writel(ENDP_STATUS_IN, &inep_regs_p[0].endp_status);
902 writel(ENDP0_INT_CTRLIN, &udc_regs_p->endp_int);
903 }
904
905 while (readl(&udc_regs_p->endp_int) & ENDP_INT_NONISOOUT_MSK) {
906 u32 epnum = 0;
907 u32 ep_int = readl(&udc_regs_p->endp_int) &
908 ENDP_INT_NONISOOUT_MSK;
909
910 ep_int >>= 16;
911 while (0x0 == (ep_int & 0x1)) {
912 ep_int >>= 1;
913 epnum++;
914 }
915
916 writel((1 << 16) << epnum, &udc_regs_p->endp_int);
917
918 if ((readl(&outep_regs_p[epnum].endp_status) &
919 ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_DATA) {
920
Vipin KUMAR27215512012-03-06 23:39:37 +0000921 dw_udc_epn_rx(epnum);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530922 writel(ENDP_STATUS_OUT_DATA,
923 &outep_regs_p[epnum].endp_status);
924 } else if ((readl(&outep_regs_p[epnum].endp_status) &
925 ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_NONE) {
926 writel(0x0, &outep_regs_p[epnum].endp_status);
927 }
928 }
929
930 if (readl(&udc_regs_p->endp_int) & ENDP_INT_NONISOIN_MSK) {
931 u32 epnum = 0;
932 u32 ep_int = readl(&udc_regs_p->endp_int) &
933 ENDP_INT_NONISOIN_MSK;
934
935 while (0x0 == (ep_int & 0x1)) {
936 ep_int >>= 1;
937 epnum++;
938 }
939
940 if (readl(&inep_regs_p[epnum].endp_status) & ENDP_STATUS_IN) {
941 writel(ENDP_STATUS_IN,
942 &outep_regs_p[epnum].endp_status);
Vipin KUMAR27215512012-03-06 23:39:37 +0000943 dw_udc_epn_tx(epnum);
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530944
945 writel(ENDP_STATUS_IN,
946 &outep_regs_p[epnum].endp_status);
947 }
948
949 writel((1 << epnum), &udc_regs_p->endp_int);
950 }
951}
952
953/*
954 * UDC interrupts
955 */
956void udc_irq(void)
957{
958 /*
959 * Loop while we have interrupts.
960 * If we don't do this, the input chain
961 * polling delay is likely to miss
962 * host requests.
963 */
964 while (readl(&plug_regs_p->plug_pending))
Vipin KUMAR27215512012-03-06 23:39:37 +0000965 dw_udc_plug_irq();
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530966
967 while (readl(&udc_regs_p->dev_int))
Vipin KUMAR27215512012-03-06 23:39:37 +0000968 dw_udc_dev_irq();
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530969
970 if (readl(&udc_regs_p->endp_int))
Vipin KUMAR27215512012-03-06 23:39:37 +0000971 dw_udc_endpoint_irq();
Vipin KUMAR62db1c02010-01-15 19:15:47 +0530972}
973
974/* Flow control */
975void udc_set_nak(int epid)
976{
977 writel(readl(&inep_regs_p[epid].endp_cntl) | ENDP_CNTL_SNAK,
978 &inep_regs_p[epid].endp_cntl);
979
980 writel(readl(&outep_regs_p[epid].endp_cntl) | ENDP_CNTL_SNAK,
981 &outep_regs_p[epid].endp_cntl);
982}
983
984void udc_unset_nak(int epid)
985{
986 u32 val;
987
988 val = readl(&inep_regs_p[epid].endp_cntl);
989 val &= ~ENDP_CNTL_SNAK;
990 val |= ENDP_CNTL_CNAK;
991 writel(val, &inep_regs_p[epid].endp_cntl);
992
993 val = readl(&outep_regs_p[epid].endp_cntl);
994 val &= ~ENDP_CNTL_SNAK;
995 val |= ENDP_CNTL_CNAK;
996 writel(val, &outep_regs_p[epid].endp_cntl);
997}