risc-v/mpfs: usb: fix ep0 stall/resume and rx reads

Fix EP0 stall and resume properly. EP0 wasn't clearly addressed
on stall / resume operations.

Also fix data reads that provide garbage for the first request.
It has always random data as it's completed prior to any read
operation.

Signed-off-by: Eero Nurkkala <eero.nurkkala@offcode.fi>
This commit is contained in:
Eero Nurkkala 2022-03-15 16:14:02 +02:00 committed by Xiang Xiao
parent e843c441de
commit c38d547900

View File

@ -755,7 +755,14 @@ static int mpfs_ep_stall(struct mpfs_ep_s *privep)
privep->stalled = true;
privep->pending = false;
if (USB_ISEPIN(privep->ep.eplog))
if (epno == EP0)
{
mpfs_modifyreg16(MPFS_USB_INDEXED_CSR_EP0_CSR0,
0,
CSR0L_DEV_SEND_STALL_MASK |
CSR0L_DEV_SERVICED_RX_PKT_RDY_MASK);
}
else if (USB_ISEPIN(privep->ep.eplog))
{
mpfs_modifyreg16(MPFS_USB_ENDPOINT(epno) +
MPFS_USB_ENDPOINT_TX_CSR_OFFSET,
@ -972,6 +979,7 @@ static int mpfs_req_read(struct mpfs_usbdev_s *priv,
struct mpfs_ep_s *privep, uint16_t recvsize)
{
struct mpfs_req_s *privreq;
bool earlyread = false;
int epno;
DEBUGASSERT(priv && privep && privep->epstate == USB_EPSTATE_IDLE);
@ -1016,7 +1024,14 @@ static int mpfs_req_read(struct mpfs_usbdev_s *priv,
{
/* Update the total number of bytes transferred */
privreq->req.xfrd += recvsize;
privep->rxactive = true;
privreq->req.xfrd += mpfs_read_rx_fifo(privreq->req.buf,
privreq->req.len,
epno);
/* Early read indicates a request that was already in place */
earlyread = true;
privreq->inflight = 0;
usbtrace(TRACE_COMPLETE(epno), privreq->req.xfrd);
@ -1036,15 +1051,24 @@ static int mpfs_req_read(struct mpfs_usbdev_s *priv,
DEBUGASSERT(recvsize == 0);
/* activate new read request from queue */
/* Activate new read request from queue */
privreq->inflight = privreq->req.len;
/* It's possible the request has been completed already. If so, don't
* repeat the read as the buffer is empty.
*/
if (!earlyread)
{
privep->rxactive = true;
privreq->req.xfrd = 0;
privreq->inflight = privreq->req.len;
priv->eplist[epno].descb[0]->addr = (uintptr_t)privreq->req.buf;
privreq->req.xfrd += mpfs_read_rx_fifo(privreq->req.buf, privreq->req.len,
privreq->req.xfrd += mpfs_read_rx_fifo(privreq->req.buf,
privreq->req.len,
epno);
}
if (epno == EP0)
{
@ -1736,9 +1760,15 @@ static int mpfs_ep_resume(struct mpfs_ep_s *privep)
privep->pending = false;
privep->epstate = USB_EPSTATE_IDLE;
/* Clear STALLRQx request and reset data toggle */
/* Clear STALLRQx request and reset data toggle if needed */
if (USB_ISEPIN(privep->ep.eplog))
if (epno == EP0)
{
mpfs_modifyreg16(MPFS_USB_INDEXED_CSR_EP0_CSR0,
CSR0L_DEV_STALL_SENT_MASK,
0);
}
else if (USB_ISEPIN(privep->ep.eplog))
{
mpfs_modifyreg16(MPFS_USB_ENDPOINT(epno) +
MPFS_USB_ENDPOINT_TX_CSR_OFFSET,
@ -3573,6 +3603,8 @@ int usbdev_register(struct usbdevclass_driver_s *driver)
struct mpfs_usbdev_s *priv = &g_usbd;
int ret;
DEBUGASSERT(driver != NULL);
/* First hook up the driver */
priv->driver = driver;