aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb
diff options
context:
space:
mode:
authorJanusz Dziedzic <januszx.dziedzic@intel.com>2017-03-13 08:11:32 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2017-04-12 06:41:15 -0400
commitd9a97dcdf5c23f5a074e204e7a8fd355412146b8 (patch)
treeaf236a165f9ef416cfacbdbc122773924de58d5d /drivers/usb
parentee670af5feede68f0ad22b329626594b8aebefc3 (diff)
usb: dwc3: gadget: delay unmap of bounced requests
commit de288e36fe33f7e06fa272bc8e2f85aa386d99aa upstream. In the case of bounced ep0 requests, we must delay DMA operation until after ->complete() otherwise we might overwrite contents of req->buf. This caused problems with RNDIS gadget. Signed-off-by: Janusz Dziedzic <januszx.dziedzic@intel.com> Signed-off-by: Felipe Balbi <felipe.balbi@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb')
-rw-r--r--drivers/usb/dwc3/gadget.c21
1 files changed, 17 insertions, 4 deletions
diff --git a/drivers/usb/dwc3/gadget.c b/drivers/usb/dwc3/gadget.c
index 5dc6bfc91f4b..ce603dcbd493 100644
--- a/drivers/usb/dwc3/gadget.c
+++ b/drivers/usb/dwc3/gadget.c
@@ -174,6 +174,7 @@ void dwc3_gadget_giveback(struct dwc3_ep *dep, struct dwc3_request *req,
174 int status) 174 int status)
175{ 175{
176 struct dwc3 *dwc = dep->dwc; 176 struct dwc3 *dwc = dep->dwc;
177 unsigned int unmap_after_complete = false;
177 178
178 req->started = false; 179 req->started = false;
179 list_del(&req->list); 180 list_del(&req->list);
@@ -182,11 +183,19 @@ void dwc3_gadget_giveback(struct dwc3_ep *dep, struct dwc3_request *req,
182 if (req->request.status == -EINPROGRESS) 183 if (req->request.status == -EINPROGRESS)
183 req->request.status = status; 184 req->request.status = status;
184 185
185 if (dwc->ep0_bounced && dep->number <= 1) 186 /*
187 * NOTICE we don't want to unmap before calling ->complete() if we're
188 * dealing with a bounced ep0 request. If we unmap it here, we would end
189 * up overwritting the contents of req->buf and this could confuse the
190 * gadget driver.
191 */
192 if (dwc->ep0_bounced && dep->number <= 1) {
186 dwc->ep0_bounced = false; 193 dwc->ep0_bounced = false;
187 194 unmap_after_complete = true;
188 usb_gadget_unmap_request(&dwc->gadget, &req->request, 195 } else {
189 req->direction); 196 usb_gadget_unmap_request(&dwc->gadget,
197 &req->request, req->direction);
198 }
190 199
191 trace_dwc3_gadget_giveback(req); 200 trace_dwc3_gadget_giveback(req);
192 201
@@ -194,6 +203,10 @@ void dwc3_gadget_giveback(struct dwc3_ep *dep, struct dwc3_request *req,
194 usb_gadget_giveback_request(&dep->endpoint, &req->request); 203 usb_gadget_giveback_request(&dep->endpoint, &req->request);
195 spin_lock(&dwc->lock); 204 spin_lock(&dwc->lock);
196 205
206 if (unmap_after_complete)
207 usb_gadget_unmap_request(&dwc->gadget,
208 &req->request, req->direction);
209
197 if (dep->number > 1) 210 if (dep->number > 1)
198 pm_runtime_put(dwc->dev); 211 pm_runtime_put(dwc->dev);
199} 212}