SAMA5 EHCI: Did not work with DEBUG off. Appears to be because of some D-Cache flushing that was performed only with DEBUG ON. Now is unconditional

This commit is contained in:
Gregory Nutt 2013-09-10 10:12:51 -06:00
parent c6bf25bca0
commit 89c829d1ae
2 changed files with 41 additions and 21 deletions

View File

@ -1158,14 +1158,16 @@ static int sam_qh_invalidate(struct sam_qh_s *qh)
static int sam_qtd_flush(struct sam_qtd_s *qtd, uint32_t **bp, void *arg) static int sam_qtd_flush(struct sam_qtd_s *qtd, uint32_t **bp, void *arg)
{ {
/* Flush the D-Cache, i.e., make the contents of the memory match the contents /* Flush the D-Cache, i.e., make the contents of the memory match the contents
* of the D-Cache in the specified address range. If debug is enabled, then * of the D-Cache in the specified address range and invalidate the D-Cache
* we will also invalidate the contents of the D-cache so that we can examine * to force re-loading of the data from memory when next accessed.
* the modified memory contents in the event of a failure.
*/ */
#if 0 /* Didn't behave as expected */
cp15_flush_dcache((uintptr_t)&qtd->hw,
(uintptr_t)&qtd->hw + sizeof(struct ehci_qtd_s));
#else
cp15_clean_dcache((uintptr_t)&qtd->hw, cp15_clean_dcache((uintptr_t)&qtd->hw,
(uintptr_t)&qtd->hw + sizeof(struct ehci_qtd_s)); (uintptr_t)&qtd->hw + sizeof(struct ehci_qtd_s));
#ifdef CONFIG_DEBUG_USB
cp15_invalidate_dcache((uintptr_t)&qtd->hw, cp15_invalidate_dcache((uintptr_t)&qtd->hw,
(uintptr_t)&qtd->hw + sizeof(struct ehci_qtd_s)); (uintptr_t)&qtd->hw + sizeof(struct ehci_qtd_s));
#endif #endif
@ -1183,15 +1185,17 @@ static int sam_qtd_flush(struct sam_qtd_s *qtd, uint32_t **bp, void *arg)
static int sam_qh_flush(struct sam_qh_s *qh) static int sam_qh_flush(struct sam_qh_s *qh)
{ {
/* Flush the QH first. Normally this just means writing the contents of the /* Flush the QH first. This will write the contents of the D-cache to RAM and
* D-cache to RAM. If debug is enabled, then we will also invalidate the * invalidate the contents of the D-cache so that the next access will be
* contents of the D-cache so that we can examine the modified memory contents * reloaded from D-Cache.
* in the event of a failure.
*/ */
#if 0 /* Didn't behave as expected */
cp15_flush_dcache((uintptr_t)&qh->hw,
(uintptr_t)&qh->hw + sizeof(struct ehci_qh_s));
#else
cp15_clean_dcache((uintptr_t)&qh->hw, cp15_clean_dcache((uintptr_t)&qh->hw,
(uintptr_t)&qh->hw + sizeof(struct ehci_qh_s)); (uintptr_t)&qh->hw + sizeof(struct ehci_qh_s));
#ifdef CONFIG_DEBUG_USB
cp15_invalidate_dcache((uintptr_t)&qh->hw, cp15_invalidate_dcache((uintptr_t)&qh->hw,
(uintptr_t)&qh->hw + sizeof(struct ehci_qh_s)); (uintptr_t)&qh->hw + sizeof(struct ehci_qh_s));
#endif #endif
@ -1540,14 +1544,10 @@ static int sam_qtd_addbpl(struct sam_qtd_s *qtd, const void *buffer, size_t bufl
* will be accessed for an OUT DMA. * will be accessed for an OUT DMA.
*/ */
#if 0 /* Didn't behave as expected */
cp15_flush_dcache((uintptr_t)buffer, (uintptr_t)buffer + buflen);
#else
cp15_clean_dcache((uintptr_t)buffer, (uintptr_t)buffer + buflen); cp15_clean_dcache((uintptr_t)buffer, (uintptr_t)buffer + buflen);
/* If debug is enabled, then we will also invalidate the contents of the
* D-cache so that we can examine the modified memory contents in the event
* of a failure.
*/
#ifdef CONFIG_DEBUG_USB
cp15_invalidate_dcache((uintptr_t)buffer, (uintptr_t)buffer + buflen); cp15_invalidate_dcache((uintptr_t)buffer, (uintptr_t)buffer + buflen);
#endif #endif
@ -1826,6 +1826,7 @@ static ssize_t sam_async_transfer(struct sam_rhport_s *rhport,
uint32_t *alt; uint32_t *alt;
uint32_t toggle; uint32_t toggle;
uint32_t regval; uint32_t regval;
bool dirin = false;
int ret; int ret;
/* Terse output only if we are tracing */ /* Terse output only if we are tracing */
@ -1912,10 +1913,9 @@ static ssize_t sam_async_transfer(struct sam_rhport_s *rhport,
*/ */
alt = NULL; alt = NULL;
if (buffer && buflen > 0) if (buffer != NULL && buflen > 0)
{ {
uint32_t tokenbits; uint32_t tokenbits;
bool dirin;
/* Extra TOKEN bits include the data toggle, the data PID, and if /* Extra TOKEN bits include the data toggle, the data PID, and if
* there is no request, an indication to interrupt at the end of this * there is no request, an indication to interrupt at the end of this
@ -2082,6 +2082,24 @@ static ssize_t sam_async_transfer(struct sam_rhport_s *rhport,
sam_takesem(&g_ehci.exclsem); sam_takesem(&g_ehci.exclsem);
#if 0 /* Does not seem to be needed */
/* Was there a data buffer? Was this an OUT transfer? */
if (buffer != NULL && buflen > 0 && !dirin)
{
/* We have received data from the host -- unless there was an error.
* in any event, we will invalidate the data buffer so that we will
* reload any new data freshly DMAed into the user buffer.
*
* NOTE: This might be un-necessary. We cleaned and invalidated the
* D-Cache prior to starting the DMA so the D-Cache should still be
* invalid in this memory region.
*/
cp15_invalidate_dcache((uintptr_t)buffer, (uintptr_t)buffer + buflen);
}
#endif
/* Did sam_ioc_wait() report an error? */ /* Did sam_ioc_wait() report an error? */
if (ret < 0) if (ret < 0)
@ -2090,7 +2108,9 @@ static ssize_t sam_async_transfer(struct sam_rhport_s *rhport,
goto errout_with_iocwait; goto errout_with_iocwait;
} }
/* Transfer completed successfully. Return the number of bytes transferred */ /* Transfer completed successfully. Return the number of bytes
* transferred.
*/
return epinfo->xfrd; return epinfo->xfrd;

View File

@ -379,7 +379,7 @@ int usbhost_enumerate(FAR struct usbhost_driver_s *drvr, uint8_t funcaddr,
DRVR_EP0CONFIGURE(drvr, 0, maxpacketsize); DRVR_EP0CONFIGURE(drvr, 0, maxpacketsize);
/* Read first 8 bytes of the device descriptor */ /* Read first bytes of the device descriptor */
ctrlreq->type = USB_REQ_DIR_IN|USB_REQ_RECIPIENT_DEVICE; ctrlreq->type = USB_REQ_DIR_IN|USB_REQ_RECIPIENT_DEVICE;
ctrlreq->req = USB_REQ_GETDESCRIPTOR; ctrlreq->req = USB_REQ_GETDESCRIPTOR;