Commit f11d893d authored by Ming Lei's avatar Ming Lei Committed by Greg Kroah-Hartman

usb: musb: support ISO high bandwidth for gadget mode

This patch has been tested OK on beagle B5 board and
use usbtest #15 and #16 as testcase.
Signed-off-by: default avatarMing Lei <tom.leiming@gmail.com>
Reviewed-by: default avatarSergei Shtylyov <sshtylyov@mvista.com>
Cc: David Brownell <dbrownell@users.sourceforge.net>
Cc: Anand Gadiyar <gadiyar@ti.com>
Cc: Mike Frysinger <vapier@gentoo.org>
Cc: Sergei Shtylyov <sshtylyov@ru.mvista.com>
Signed-off-by: default avatarFelipe Balbi <balbi@ti.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 3ee076de
...@@ -337,13 +337,15 @@ static void txstate(struct musb *musb, struct musb_request *req) ...@@ -337,13 +337,15 @@ static void txstate(struct musb *musb, struct musb_request *req)
csr |= (MUSB_TXCSR_DMAENAB | csr |= (MUSB_TXCSR_DMAENAB |
MUSB_TXCSR_MODE); MUSB_TXCSR_MODE);
/* against programming guide */ /* against programming guide */
} else } else {
csr |= (MUSB_TXCSR_AUTOSET csr |= (MUSB_TXCSR_DMAENAB
| MUSB_TXCSR_DMAENAB
| MUSB_TXCSR_DMAMODE | MUSB_TXCSR_DMAMODE
| MUSB_TXCSR_MODE); | MUSB_TXCSR_MODE);
if (!musb_ep->hb_mult)
csr |= MUSB_TXCSR_AUTOSET;
}
csr &= ~MUSB_TXCSR_P_UNDERRUN; csr &= ~MUSB_TXCSR_P_UNDERRUN;
musb_writew(epio, MUSB_TXCSR, csr); musb_writew(epio, MUSB_TXCSR, csr);
} }
} }
...@@ -643,7 +645,8 @@ static void rxstate(struct musb *musb, struct musb_request *req) ...@@ -643,7 +645,8 @@ static void rxstate(struct musb *musb, struct musb_request *req)
*/ */
csr |= MUSB_RXCSR_DMAENAB; csr |= MUSB_RXCSR_DMAENAB;
csr |= MUSB_RXCSR_AUTOCLEAR; if (!musb_ep->hb_mult)
csr |= MUSB_RXCSR_AUTOCLEAR;
#ifdef USE_MODE1 #ifdef USE_MODE1
/* csr |= MUSB_RXCSR_DMAMODE; */ /* csr |= MUSB_RXCSR_DMAMODE; */
...@@ -875,9 +878,25 @@ static int musb_gadget_enable(struct usb_ep *ep, ...@@ -875,9 +878,25 @@ static int musb_gadget_enable(struct usb_ep *ep,
/* REVISIT this rules out high bandwidth periodic transfers */ /* REVISIT this rules out high bandwidth periodic transfers */
tmp = le16_to_cpu(desc->wMaxPacketSize); tmp = le16_to_cpu(desc->wMaxPacketSize);
if (tmp & ~0x07ff) if (tmp & ~0x07ff) {
goto fail; int ok;
musb_ep->packet_sz = tmp;
if (usb_endpoint_dir_in(desc))
ok = musb->hb_iso_tx;
else
ok = musb->hb_iso_rx;
if (!ok) {
DBG(4, "%s: not support ISO high bandwidth\n", __func__);
goto fail;
}
musb_ep->hb_mult = (tmp >> 11) & 3;
} else {
musb_ep->hb_mult = 0;
}
musb_ep->packet_sz = tmp & 0x7ff;
tmp = musb_ep->packet_sz * (musb_ep->hb_mult + 1);
/* enable the interrupts for the endpoint, set the endpoint /* enable the interrupts for the endpoint, set the endpoint
* packet size (or fail), set the mode, clear the fifo * packet size (or fail), set the mode, clear the fifo
...@@ -890,8 +909,11 @@ static int musb_gadget_enable(struct usb_ep *ep, ...@@ -890,8 +909,11 @@ static int musb_gadget_enable(struct usb_ep *ep,
musb_ep->is_in = 1; musb_ep->is_in = 1;
if (!musb_ep->is_in) if (!musb_ep->is_in)
goto fail; goto fail;
if (tmp > hw_ep->max_packet_sz_tx)
if (tmp > hw_ep->max_packet_sz_tx) {
DBG(4, "%s: packet size beyond hw fifo size\n", __func__);
goto fail; goto fail;
}
int_txe |= (1 << epnum); int_txe |= (1 << epnum);
musb_writew(mbase, MUSB_INTRTXE, int_txe); musb_writew(mbase, MUSB_INTRTXE, int_txe);
...@@ -906,7 +928,7 @@ static int musb_gadget_enable(struct usb_ep *ep, ...@@ -906,7 +928,7 @@ static int musb_gadget_enable(struct usb_ep *ep,
if (musb->hwvers < MUSB_HWVERS_2000) if (musb->hwvers < MUSB_HWVERS_2000)
musb_writew(regs, MUSB_TXMAXP, hw_ep->max_packet_sz_tx); musb_writew(regs, MUSB_TXMAXP, hw_ep->max_packet_sz_tx);
else else
musb_writew(regs, MUSB_TXMAXP, tmp); musb_writew(regs, MUSB_TXMAXP, musb_ep->packet_sz | (musb_ep->hb_mult << 11));
csr = MUSB_TXCSR_MODE | MUSB_TXCSR_CLRDATATOG; csr = MUSB_TXCSR_MODE | MUSB_TXCSR_CLRDATATOG;
if (musb_readw(regs, MUSB_TXCSR) if (musb_readw(regs, MUSB_TXCSR)
...@@ -927,8 +949,11 @@ static int musb_gadget_enable(struct usb_ep *ep, ...@@ -927,8 +949,11 @@ static int musb_gadget_enable(struct usb_ep *ep,
musb_ep->is_in = 0; musb_ep->is_in = 0;
if (musb_ep->is_in) if (musb_ep->is_in)
goto fail; goto fail;
if (tmp > hw_ep->max_packet_sz_rx)
if (tmp > hw_ep->max_packet_sz_rx) {
DBG(4, "%s: packet size beyond hw fifo size\n", __func__);
goto fail; goto fail;
}
int_rxe |= (1 << epnum); int_rxe |= (1 << epnum);
musb_writew(mbase, MUSB_INTRRXE, int_rxe); musb_writew(mbase, MUSB_INTRRXE, int_rxe);
...@@ -942,7 +967,7 @@ static int musb_gadget_enable(struct usb_ep *ep, ...@@ -942,7 +967,7 @@ static int musb_gadget_enable(struct usb_ep *ep,
if (musb->hwvers < MUSB_HWVERS_2000) if (musb->hwvers < MUSB_HWVERS_2000)
musb_writew(regs, MUSB_RXMAXP, hw_ep->max_packet_sz_rx); musb_writew(regs, MUSB_RXMAXP, hw_ep->max_packet_sz_rx);
else else
musb_writew(regs, MUSB_RXMAXP, tmp); musb_writew(regs, MUSB_RXMAXP, musb_ep->packet_sz | (musb_ep->hb_mult << 11));
/* force shared fifo to OUT-only mode */ /* force shared fifo to OUT-only mode */
if (hw_ep->is_shared_fifo) { if (hw_ep->is_shared_fifo) {
......
...@@ -79,6 +79,8 @@ struct musb_ep { ...@@ -79,6 +79,8 @@ struct musb_ep {
/* true if lock must be dropped but req_list may not be advanced */ /* true if lock must be dropped but req_list may not be advanced */
u8 busy; u8 busy;
u8 hb_mult;
}; };
static inline struct musb_ep *to_musb_ep(struct usb_ep *ep) static inline struct musb_ep *to_musb_ep(struct usb_ep *ep)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment