Commit b950bdbc authored by Alan Stern's avatar Alan Stern Committed by Greg Kroah-Hartman

USB: g_file_storage: ignore bulk-out data after invalid CBW

This patch (as1061) makes g_file_storage more compliant with the
Bulk-Only Transport specification.  After an invalid CBW is received,
the gadget must ignore any further bulk-OUT data until it is reset.
Signed-off-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 9aebfd6b
...@@ -644,7 +644,7 @@ struct fsg_dev { ...@@ -644,7 +644,7 @@ struct fsg_dev {
unsigned long atomic_bitflags; unsigned long atomic_bitflags;
#define REGISTERED 0 #define REGISTERED 0
#define CLEAR_BULK_HALTS 1 #define IGNORE_BULK_OUT 1
#define SUSPENDED 2 #define SUSPENDED 2
struct usb_ep *bulk_in; struct usb_ep *bulk_in;
...@@ -2936,8 +2936,8 @@ static int received_cbw(struct fsg_dev *fsg, struct fsg_buffhd *bh) ...@@ -2936,8 +2936,8 @@ static int received_cbw(struct fsg_dev *fsg, struct fsg_buffhd *bh)
struct usb_request *req = bh->outreq; struct usb_request *req = bh->outreq;
struct bulk_cb_wrap *cbw = req->buf; struct bulk_cb_wrap *cbw = req->buf;
/* Was this a real packet? */ /* Was this a real packet? Should it be ignored? */
if (req->status) if (req->status || test_bit(IGNORE_BULK_OUT, &fsg->atomic_bitflags))
return -EINVAL; return -EINVAL;
/* Is the CBW valid? */ /* Is the CBW valid? */
...@@ -2948,13 +2948,17 @@ static int received_cbw(struct fsg_dev *fsg, struct fsg_buffhd *bh) ...@@ -2948,13 +2948,17 @@ static int received_cbw(struct fsg_dev *fsg, struct fsg_buffhd *bh)
req->actual, req->actual,
le32_to_cpu(cbw->Signature)); le32_to_cpu(cbw->Signature));
/* The Bulk-only spec says we MUST stall the bulk pipes! /* The Bulk-only spec says we MUST stall the IN endpoint
* If we want to avoid stalls, set a flag so that we will * (6.6.1), so it's unavoidable. It also says we must
* clear the endpoint halts at the next reset. */ * retain this state until the next reset, but there's
if (!mod_data.can_stall) * no way to tell the controller driver it should ignore
set_bit(CLEAR_BULK_HALTS, &fsg->atomic_bitflags); * Clear-Feature(HALT) requests.
fsg_set_halt(fsg, fsg->bulk_out); *
* We aren't required to halt the OUT endpoint; instead
* we can simply accept and discard any data received
* until the next reset. */
halt_bulk_in_endpoint(fsg); halt_bulk_in_endpoint(fsg);
set_bit(IGNORE_BULK_OUT, &fsg->atomic_bitflags);
return -EINVAL; return -EINVAL;
} }
...@@ -3140,6 +3144,7 @@ static int do_set_interface(struct fsg_dev *fsg, int altsetting) ...@@ -3140,6 +3144,7 @@ static int do_set_interface(struct fsg_dev *fsg, int altsetting)
goto reset; goto reset;
fsg->bulk_out_enabled = 1; fsg->bulk_out_enabled = 1;
fsg->bulk_out_maxpacket = le16_to_cpu(d->wMaxPacketSize); fsg->bulk_out_maxpacket = le16_to_cpu(d->wMaxPacketSize);
clear_bit(IGNORE_BULK_OUT, &fsg->atomic_bitflags);
if (transport_is_cbi()) { if (transport_is_cbi()) {
d = ep_desc(fsg->gadget, &fs_intr_in_desc, &hs_intr_in_desc); d = ep_desc(fsg->gadget, &fs_intr_in_desc, &hs_intr_in_desc);
...@@ -3321,11 +3326,8 @@ static void handle_exception(struct fsg_dev *fsg) ...@@ -3321,11 +3326,8 @@ static void handle_exception(struct fsg_dev *fsg)
/* In case we were forced against our will to halt a /* In case we were forced against our will to halt a
* bulk endpoint, clear the halt now. (The SuperH UDC * bulk endpoint, clear the halt now. (The SuperH UDC
* requires this.) */ * requires this.) */
if (test_and_clear_bit(CLEAR_BULK_HALTS, if (test_and_clear_bit(IGNORE_BULK_OUT, &fsg->atomic_bitflags))
&fsg->atomic_bitflags)) {
usb_ep_clear_halt(fsg->bulk_in); usb_ep_clear_halt(fsg->bulk_in);
usb_ep_clear_halt(fsg->bulk_out);
}
if (transport_is_bbb()) { if (transport_is_bbb()) {
if (fsg->ep0_req_tag == exception_req_tag) if (fsg->ep0_req_tag == exception_req_tag)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment