aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
Diffstat (limited to 'drivers')
-rw-r--r--drivers/usb/gadget/inode.c18
1 files changed, 11 insertions, 7 deletions
diff --git a/drivers/usb/gadget/inode.c b/drivers/usb/gadget/inode.c
index ea8e3160d05e..e5ce4f0bb7cc 100644
--- a/drivers/usb/gadget/inode.c
+++ b/drivers/usb/gadget/inode.c
@@ -933,28 +933,24 @@ static void clean_req (struct usb_ep *ep, struct usb_request *req)
933static void ep0_complete (struct usb_ep *ep, struct usb_request *req) 933static void ep0_complete (struct usb_ep *ep, struct usb_request *req)
934{ 934{
935 struct dev_data *dev = ep->driver_data; 935 struct dev_data *dev = ep->driver_data;
936 unsigned long flags;
936 int free = 1; 937 int free = 1;
937 938
938 /* for control OUT, data must still get to userspace */ 939 /* for control OUT, data must still get to userspace */
940 spin_lock_irqsave(&dev->lock, flags);
939 if (!dev->setup_in) { 941 if (!dev->setup_in) {
940 dev->setup_out_error = (req->status != 0); 942 dev->setup_out_error = (req->status != 0);
941 if (!dev->setup_out_error) 943 if (!dev->setup_out_error)
942 free = 0; 944 free = 0;
943 dev->setup_out_ready = 1; 945 dev->setup_out_ready = 1;
944 ep0_readable (dev); 946 ep0_readable (dev);
945 } else {
946 unsigned long flags;
947
948 spin_lock_irqsave(&dev->lock, flags);
949 if (dev->state == STATE_DEV_SETUP)
950 dev->state = STATE_DEV_CONNECTED;
951 spin_unlock_irqrestore(&dev->lock, flags);
952 } 947 }
953 948
954 /* clean up as appropriate */ 949 /* clean up as appropriate */
955 if (free && req->buf != &dev->rbuf) 950 if (free && req->buf != &dev->rbuf)
956 clean_req (ep, req); 951 clean_req (ep, req);
957 req->complete = epio_complete; 952 req->complete = epio_complete;
953 spin_unlock_irqrestore(&dev->lock, flags);
958} 954}
959 955
960static int setup_req (struct usb_ep *ep, struct usb_request *req, u16 len) 956static int setup_req (struct usb_ep *ep, struct usb_request *req, u16 len)
@@ -1036,6 +1032,13 @@ ep0_read (struct file *fd, char __user *buf, size_t len, loff_t *ptr)
1036 spin_lock_irq (&dev->lock); 1032 spin_lock_irq (&dev->lock);
1037 if (retval) 1033 if (retval)
1038 goto done; 1034 goto done;
1035
1036 if (dev->state != STATE_DEV_SETUP) {
1037 retval = -ECANCELED;
1038 goto done;
1039 }
1040 dev->state = STATE_DEV_CONNECTED;
1041
1039 if (dev->setup_out_error) 1042 if (dev->setup_out_error)
1040 retval = -EIO; 1043 retval = -EIO;
1041 else { 1044 else {
@@ -1187,6 +1190,7 @@ ep0_write (struct file *fd, const char __user *buf, size_t len, loff_t *ptr)
1187 if (dev->setup_in) { 1190 if (dev->setup_in) {
1188 retval = setup_req (dev->gadget->ep0, dev->req, len); 1191 retval = setup_req (dev->gadget->ep0, dev->req, len);
1189 if (retval == 0) { 1192 if (retval == 0) {
1193 dev->state = STATE_DEV_CONNECTED;
1190 spin_unlock_irq (&dev->lock); 1194 spin_unlock_irq (&dev->lock);
1191 if (copy_from_user (dev->req->buf, buf, len)) 1195 if (copy_from_user (dev->req->buf, buf, len))
1192 retval = -EFAULT; 1196 retval = -EFAULT;