diff options
Diffstat (limited to 'sys/arch/arm/xscale/pxa27x_udc.c')
-rw-r--r-- | sys/arch/arm/xscale/pxa27x_udc.c | 87 |
1 files changed, 57 insertions, 30 deletions
diff --git a/sys/arch/arm/xscale/pxa27x_udc.c b/sys/arch/arm/xscale/pxa27x_udc.c index 2dc6e7cffeb..dcb3f266c99 100644 --- a/sys/arch/arm/xscale/pxa27x_udc.c +++ b/sys/arch/arm/xscale/pxa27x_udc.c @@ -1,8 +1,9 @@ -/* $OpenBSD: pxa27x_udc.c,v 1.16 2007/02/23 06:12:43 drahn Exp $ */ +/* $OpenBSD: pxa27x_udc.c,v 1.17 2007/02/24 22:08:20 drahn Exp $ */ /* - * Copyright (c) 2005 David Gwynne <dlg@openbsd.org> + * Copyright (c) 2007 Dale Rahn <drahn@openbsd.org> * Copyright (c) 2006 Uwe Stuehler <uwe@openbsd.org> + * Copyright (c) 2005 David Gwynne <dlg@openbsd.org> * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above @@ -104,7 +105,7 @@ int pxaudc_connect_intr(void *); int pxaudc_intr(void *); void pxaudc_intr1(struct pxaudc_softc *); void pxaudc_ep0_intr(struct pxaudc_softc *); -void pxaudc_epN_intr(struct pxaudc_softc *sc, int ep); +void pxaudc_epN_intr(struct pxaudc_softc *sc, int ep, int isr); usbf_status pxaudc_open(struct usbf_pipe *); void pxaudc_softintr(void *); @@ -542,8 +543,10 @@ pxaudc_read_epN(struct pxaudc_softc *sc, int ep) again: xfer = SIMPLEQ_FIRST(&pipe->queue); - if (xfer == NULL) + if (xfer == NULL) { + printf("pxaudc_read_epN: ep %d, no xfer\n", ep); return; + } count = CSR_READ_4(sc, USBDC_UDCBCR(ep)); tlen = len = MIN(count, xfer->length - xfer->actlen); @@ -674,12 +677,19 @@ pxaudc_write(struct pxaudc_softc *sc, usbf_xfer_handle xfer) int ep = sc->sc_ep_map[usbf_endpoint_index(xfer->pipe->endpoint)]; int tlen = 0; int maxp = UGETW(xfer->pipe->endpoint->edesc->wMaxPacketSize); + u_int32_t csr, csr_o; + +#ifdef DEBUG_TX_PKT + if (xfer->actlen == 0) + printf("new packet len %x\n", xfer->length); +#endif #ifdef DEBUG_TX printf("writing data to endpoint %x, xlen %x xact %x\n", ep, xfer->length, xfer->actlen); #endif + if (xfer->actlen == xfer->length) { /* * If the packet size is wMaxPacketSize byte multiple @@ -687,16 +697,16 @@ pxaudc_write(struct pxaudc_softc *sc, usbf_xfer_handle xfer) */ if ((xfer->actlen % maxp) == 0 && xfer->status != USBF_NORMAL_COMPLETION) { - CSR_SET_4(sc, USBDC_UDCCSR(ep), USBDC_UDCCSR_SP); - /* - * if we send a zero packet, we are 'done', but dont - * to usbf_transfer_complete() just yet because the - * short packet will cause another interrupt. - */ - xfer->status = USBF_NORMAL_COMPLETION; - return; + if (CSR_READ_4(sc, USBDC_UDCCSR(ep)) & USBDC_UDCCSR_BNF) { + CSR_SET_4(sc, USBDC_UDCCSR(ep), USBDC_UDCCSR_SP); + } else { + printf("fifo full when trying to set short packet\n"); + } } xfer->status = USBF_NORMAL_COMPLETION; +#ifdef DEBUG_TX_PKT + printf("packet complete %x\n", xfer->actlen); +#endif usbf_transfer_complete(xfer); return; } @@ -704,8 +714,18 @@ pxaudc_write(struct pxaudc_softc *sc, usbf_xfer_handle xfer) p = xfer->buffer + xfer->actlen; - if (CSR_READ_4(sc, USBDC_UDCCSR(ep)) & USBDC_UDCCSR_PC) - CSR_SET_4(sc, USBDC_UDCCSR(ep), USBDC_UDCCSR_PC); + + csr_o = 0; + csr = CSR_READ_4(sc, USBDC_UDCCSR(ep)); + if (csr & USBDC_UDCCSR_PC) + csr_o |= USBDC_UDCCSR_PC; + if (csr & USBDC_UDCCSR_TRN) + csr_o |= USBDC_UDCCSR_TRN; + if (csr & USBDC_UDCCSR_SST) + csr_o |= USBDC_UDCCSR_SST; + if (csr_o != 0) + CSR_WRITE_4(sc, USBDC_UDCCSR(ep), csr_o); + while (CSR_READ_4(sc, USBDC_UDCCSR(ep)) & USBDC_UDCCSR_BNF) { u_int32_t v; @@ -737,6 +757,9 @@ pxaudc_write(struct pxaudc_softc *sc, usbf_xfer_handle xfer) #ifdef DEBUG_TX printf(" wrote tlen %x %x\n", tlen, xfer->actlen); #endif + if (xfer->actlen == 0) { + printf("whoa, write_ep called, but no free space\n"); + } if (xfer->actlen >= xfer->length) { if ((xfer->actlen % maxp) != 0) { CSR_SET_4(sc, USBDC_UDCCSR(ep), USBDC_UDCCSR_SP); @@ -849,16 +872,6 @@ pxaudc_intr1(struct pxaudc_softc *sc) otgisr, USBDC_UDCOTGISR_BITS); #endif - for (i = 1; i < 24; i++) { - if (i < 16) { - if (isr0 & USBDC_UDCISR0_IR(i)) - pxaudc_epN_intr(sc, i); - } else { - if (isr1 & USBDC_UDCISR1_IR(i-16)) - pxaudc_epN_intr(sc, i); - } - } - /* Handle USB RESET condition. */ if (isr1 & USBDC_UDCISR1_IRRS) { sc->sc_ep0state = EP0_SETUP; @@ -871,6 +884,18 @@ pxaudc_intr1(struct pxaudc_softc *sc) if (isr0 & USBDC_UDCISR0_IR(0)) pxaudc_ep0_intr(sc); + for (i = 1; i < 24; i++) { + if (i < 16) { + if (USBDC_UDCISR0_IRs(isr0,i)) + pxaudc_epN_intr(sc, i, + USBDC_UDCISR0_IRs(isr0,i)); + } else { + if (USBDC_UDCISR1_IRs(isr1,i)) + pxaudc_epN_intr(sc, i, + USBDC_UDCISR1_IRs(isr1,i)); + } + } + if (isr1 & USBDC_UDCISR1_IRSU) { /* suspend ?? */ } @@ -883,16 +908,19 @@ ret: } void -pxaudc_epN_intr(struct pxaudc_softc *sc, int ep) +pxaudc_epN_intr(struct pxaudc_softc *sc, int ep, int isr) { + struct pxaudc_pipe *ppipe; + usbf_pipe_handle pipe; + int dir; + /* should not occur before device is configured */ if (sc->sc_cn == 0) return; + if (isr & 2) + printf("ep%d: fifo error\n", ep); /* XXX */ /* faster method of determining direction? */ - struct pxaudc_pipe *ppipe; - usbf_pipe_handle pipe = NULL; - int dir; ppipe = sc->sc_pipe[ep]; if (ppipe == NULL) @@ -935,8 +963,7 @@ pxaudc_ep0_intr(struct pxaudc_softc *sc) csr0 = CSR_READ_4(sc, USBDC_UDCCSR0); DPRINTF(10,("pxaudc_ep0_intr: csr0=%b\n", csr0, USBDC_UDCCSR0_BITS)); - - delay(100); + delay (25); ppipe = sc->sc_pipe[0]; if (ppipe != NULL) { |