usb: dwc2: gadget: fix zero length packet transfers
According to programming guide, zero length packet should be programmed on its own and should not be counted in DIEPTSIZ.PktCnt with other packets. For ep0, this is the zlp for DATA IN stage (if required) and not for the STATUS stage. Tested-by: Robert Baldyga <r.baldyga@samsung.com> Acked-by: Paul Zimmerman <paulz@synopsys.com> Signed-off-by: Gregory Herrero <gregory.herrero@intel.com> Signed-off-by: Mian Yousaf Kaukab <yousaf.kaukab@intel.com> Signed-off-by: Felipe Balbi <balbi@ti.com>
This commit is contained in:
parent
fe0b94abcd
commit
f71b5e2533
@ -598,14 +598,15 @@ static void s3c_hsotg_start_req(struct dwc2_hsotg *hsotg,
|
|||||||
else
|
else
|
||||||
epsize = 0;
|
epsize = 0;
|
||||||
|
|
||||||
if (index != 0 && ureq->zero) {
|
/*
|
||||||
/*
|
* zero length packet should be programmed on its own and should not
|
||||||
* test for the packets being exactly right for the
|
* be counted in DIEPTSIZ.PktCnt with other packets.
|
||||||
* transfer
|
*/
|
||||||
*/
|
if (dir_in && ureq->zero && !continuing) {
|
||||||
|
/* Test if zlp is actually required. */
|
||||||
if (length == (packets * hs_ep->ep.maxpacket))
|
if ((ureq->length >= hs_ep->ep.maxpacket) &&
|
||||||
packets++;
|
!(ureq->length % hs_ep->ep.maxpacket))
|
||||||
|
hs_ep->sent_zlp = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
epsize |= DXEPTSIZ_PKTCNT(packets);
|
epsize |= DXEPTSIZ_PKTCNT(packets);
|
||||||
@ -857,7 +858,11 @@ static int s3c_hsotg_send_reply(struct dwc2_hsotg *hsotg,
|
|||||||
|
|
||||||
req->buf = hsotg->ep0_buff;
|
req->buf = hsotg->ep0_buff;
|
||||||
req->length = length;
|
req->length = length;
|
||||||
req->zero = 1; /* always do zero-length final transfer */
|
/*
|
||||||
|
* zero flag is for sending zlp in DATA IN stage. It has no impact on
|
||||||
|
* STATUS stage.
|
||||||
|
*/
|
||||||
|
req->zero = 0;
|
||||||
req->complete = s3c_hsotg_complete_oursetup;
|
req->complete = s3c_hsotg_complete_oursetup;
|
||||||
|
|
||||||
if (length)
|
if (length)
|
||||||
@ -1744,32 +1749,20 @@ static void s3c_hsotg_complete_in(struct dwc2_hsotg *hsotg,
|
|||||||
dev_dbg(hsotg->dev, "req->length:%d req->actual:%d req->zero:%d\n",
|
dev_dbg(hsotg->dev, "req->length:%d req->actual:%d req->zero:%d\n",
|
||||||
hs_req->req.length, hs_req->req.actual, hs_req->req.zero);
|
hs_req->req.length, hs_req->req.actual, hs_req->req.zero);
|
||||||
|
|
||||||
/*
|
|
||||||
* Check if dealing with Maximum Packet Size(MPS) IN transfer at EP0
|
|
||||||
* When sent data is a multiple MPS size (e.g. 64B ,128B ,192B
|
|
||||||
* ,256B ... ), after last MPS sized packet send IN ZLP packet to
|
|
||||||
* inform the host that no more data is available.
|
|
||||||
* The state of req.zero member is checked to be sure that the value to
|
|
||||||
* send is smaller than wValue expected from host.
|
|
||||||
* Check req.length to NOT send another ZLP when the current one is
|
|
||||||
* under completion (the one for which this completion has been called).
|
|
||||||
*/
|
|
||||||
if (hs_req->req.length && hs_ep->index == 0 && hs_req->req.zero &&
|
|
||||||
hs_req->req.length == hs_req->req.actual &&
|
|
||||||
!(hs_req->req.length % hs_ep->ep.maxpacket)) {
|
|
||||||
|
|
||||||
dev_dbg(hsotg->dev, "ep0 zlp IN packet sent\n");
|
|
||||||
s3c_hsotg_program_zlp(hsotg, hs_ep);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!size_left && hs_req->req.actual < hs_req->req.length) {
|
if (!size_left && hs_req->req.actual < hs_req->req.length) {
|
||||||
dev_dbg(hsotg->dev, "%s trying more for req...\n", __func__);
|
dev_dbg(hsotg->dev, "%s trying more for req...\n", __func__);
|
||||||
s3c_hsotg_start_req(hsotg, hs_ep, hs_req, true);
|
s3c_hsotg_start_req(hsotg, hs_ep, hs_req, true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Zlp for all endpoints, for ep0 only in DATA IN stage */
|
||||||
|
if (hs_ep->sent_zlp) {
|
||||||
|
s3c_hsotg_program_zlp(hsotg, hs_ep);
|
||||||
|
hs_ep->sent_zlp = 0;
|
||||||
|
/* transfer will be completed on next complete interrupt */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (hs_ep->index == 0 && hsotg->ep0_state == DWC2_EP0_DATA_IN) {
|
if (hs_ep->index == 0 && hsotg->ep0_state == DWC2_EP0_DATA_IN) {
|
||||||
/* Move to STATUS OUT */
|
/* Move to STATUS OUT */
|
||||||
s3c_hsotg_ep0_zlp(hsotg, false);
|
s3c_hsotg_ep0_zlp(hsotg, false);
|
||||||
|
Loading…
Reference in New Issue
Block a user