usb: dwc3: gadget: Revise setting IOC when no TRB left
authorThinh Nguyen <Thinh.Nguyen@synopsys.com>
Thu, 1 Oct 2020 00:44:25 +0000 (17:44 -0700)
committerFelipe Balbi <balbi@kernel.org>
Fri, 2 Oct 2020 06:57:46 +0000 (09:57 +0300)
To keep the setting of interrupt-on-completion (IOC) when out of TRBs
consistent and easier to read, the caller of dwc3_prepare_one_trb()
will determine if the TRB must have IOC bit set. This also reduces the
number of times we need to call dwc3_calc_trbs_left(). Note that we only
care about setting IOC from insufficient number of TRBs for SG and not
linear requests (because we don't need to split linear requests).

Signed-off-by: Thinh Nguyen <Thinh.Nguyen@synopsys.com>
Signed-off-by: Felipe Balbi <balbi@kernel.org>
drivers/usb/dwc3/gadget.c

index 0387b94b8f942d9c1a989473c343b85afc98eb68..327cd556e8dbfc6fb02766ce0418f1845a86d3a1 100644 (file)
@@ -1034,8 +1034,7 @@ static void __dwc3_prepare_one_trb(struct dwc3_ep *dep, struct dwc3_trb *trb,
                        trb->ctrl |= DWC3_TRB_CTRL_ISP_IMI;
        }
 
-       if ((!no_interrupt && !chain) || must_interrupt ||
-                       (dwc3_calc_trbs_left(dep) == 1))
+       if ((!no_interrupt && !chain) || must_interrupt)
                trb->ctrl |= DWC3_TRB_CTRL_IOC;
 
        if (chain)
@@ -1169,6 +1168,7 @@ static int dwc3_prepare_trbs_sg(struct dwc3_ep *dep,
                length -= sg_dma_len(s);
 
        for_each_sg(sg, s, remaining, i) {
+               unsigned int num_trbs_left = dwc3_calc_trbs_left(dep);
                unsigned int trb_length;
                bool must_interrupt = false;
                bool last_sg = false;
@@ -1187,7 +1187,7 @@ static int dwc3_prepare_trbs_sg(struct dwc3_ep *dep,
                if ((i == remaining - 1) || !length)
                        last_sg = true;
 
-               if (!dwc3_calc_trbs_left(dep))
+               if (!num_trbs_left)
                        break;
 
                if (last_sg) {
@@ -1196,12 +1196,13 @@ static int dwc3_prepare_trbs_sg(struct dwc3_ep *dep,
                } else {
                        /*
                         * Look ahead to check if we have enough TRBs for the
-                        * last SG entry. If not, set interrupt on this TRB to
-                        * resume preparing the last SG entry when more TRBs are
+                        * next SG entry. If not, set interrupt on this TRB to
+                        * resume preparing the next SG entry when more TRBs are
                         * free.
                         */
-                       if (needs_extra_trb && dwc3_calc_trbs_left(dep) <= 2 &&
-                                       sg_dma_len(sg_next(s)) >= length)
+                       if (num_trbs_left == 1 || (needs_extra_trb &&
+                                       num_trbs_left <= 2 &&
+                                       sg_dma_len(sg_next(s)) >= length))
                                must_interrupt = true;
 
                        dwc3_prepare_one_trb(dep, req, trb_length, 1, i, false,
@@ -1230,7 +1231,7 @@ static int dwc3_prepare_trbs_sg(struct dwc3_ep *dep,
                        break;
                }
 
-               if (!dwc3_calc_trbs_left(dep) || must_interrupt)
+               if (must_interrupt)
                        break;
        }