Down the call stack from the ioctl handler for VIDIOC_STREAMON,
uvc_video_alloc_requests contains a BUG_ON, which in the high level,
triggers when VIDIOC_STREAMON ioctl is issued without VIDIOC_STREAMOFF
being issued previously.

This could be triggered by uvc_function_set_alt 0 racing and
winning against uvc_v4l2_streamon, or by userspace neglecting to issue
the VIDIOC_STREAMOFF ioctl.

To fix this, add two more uvc states: starting and stopping. Use these
to prevent the racing, and to detect when VIDIOC_STREAMON is issued
without previously issuing VIDIOC_STREAMOFF.

Signed-off-by: Paul Elder <paul.el...@pitt.edu>
---
 drivers/usb/gadget/function/f_uvc.c    |  8 ++++++--
 drivers/usb/gadget/function/uvc.h      |  2 ++
 drivers/usb/gadget/function/uvc_v4l2.c | 19 +++++++++++++++++--
 3 files changed, 25 insertions(+), 4 deletions(-)

diff --git a/drivers/usb/gadget/function/f_uvc.c 
b/drivers/usb/gadget/function/f_uvc.c
index 439eba660e95..9b63b28a1ee3 100644
--- a/drivers/usb/gadget/function/f_uvc.c
+++ b/drivers/usb/gadget/function/f_uvc.c
@@ -325,17 +325,19 @@ uvc_function_set_alt(struct usb_function *f, unsigned 
interface, unsigned alt)
 
        switch (alt) {
        case 0:
-               if (uvc->state != UVC_STATE_STREAMING)
+               if (uvc->state != UVC_STATE_STREAMING &&
+                               uvc->state != UVC_STATE_STARTING)
                        return 0;
 
                if (uvc->video.ep)
                        usb_ep_disable(uvc->video.ep);
 
+               uvc->state = UVC_STATE_STOPPING;
+
                memset(&v4l2_event, 0, sizeof(v4l2_event));
                v4l2_event.type = UVC_EVENT_STREAMOFF;
                v4l2_event_queue(&uvc->vdev, &v4l2_event);
 
-               uvc->state = UVC_STATE_CONNECTED;
                return 0;
 
        case 1:
@@ -354,6 +356,8 @@ uvc_function_set_alt(struct usb_function *f, unsigned 
interface, unsigned alt)
                        return ret;
                usb_ep_enable(uvc->video.ep);
 
+               uvc->state = UVC_STATE_STARTING;
+
                memset(&v4l2_event, 0, sizeof(v4l2_event));
                v4l2_event.type = UVC_EVENT_STREAMON;
                v4l2_event_queue(&uvc->vdev, &v4l2_event);
diff --git a/drivers/usb/gadget/function/uvc.h 
b/drivers/usb/gadget/function/uvc.h
index a64e07e61f8c..afb2eac1f337 100644
--- a/drivers/usb/gadget/function/uvc.h
+++ b/drivers/usb/gadget/function/uvc.h
@@ -131,6 +131,8 @@ enum uvc_state {
        UVC_STATE_DISCONNECTED,
        UVC_STATE_CONNECTED,
        UVC_STATE_STREAMING,
+       UVC_STATE_STARTING,
+       UVC_STATE_STOPPING,
 };
 
 struct uvc_device {
diff --git a/drivers/usb/gadget/function/uvc_v4l2.c 
b/drivers/usb/gadget/function/uvc_v4l2.c
index 9a9019625496..fdf02b6987c0 100644
--- a/drivers/usb/gadget/function/uvc_v4l2.c
+++ b/drivers/usb/gadget/function/uvc_v4l2.c
@@ -193,6 +193,9 @@ uvc_v4l2_streamon(struct file *file, void *fh, enum 
v4l2_buf_type type)
        struct uvc_video *video = &uvc->video;
        int ret;
 
+       if (uvc->state != UVC_STATE_STARTING)
+               return 0;
+
        if (type != video->queue.queue.type)
                return -EINVAL;
 
@@ -201,12 +204,13 @@ uvc_v4l2_streamon(struct file *file, void *fh, enum 
v4l2_buf_type type)
        if (ret < 0)
                return ret;
 
+       uvc->state = UVC_STATE_STREAMING;
+
        /*
         * Complete the alternate setting selection setup phase now that
         * userspace is ready to provide video frames.
         */
        uvc_function_setup_continue(uvc);
-       uvc->state = UVC_STATE_STREAMING;
 
        return 0;
 }
@@ -217,11 +221,22 @@ uvc_v4l2_streamoff(struct file *file, void *fh, enum 
v4l2_buf_type type)
        struct video_device *vdev = video_devdata(file);
        struct uvc_device *uvc = video_get_drvdata(vdev);
        struct uvc_video *video = &uvc->video;
+       int ret;
+
+       if (uvc->state != UVC_STATE_STOPPING)
+               return 0;
 
        if (type != video->queue.queue.type)
                return -EINVAL;
 
-       return uvcg_video_enable(video, 0);
+       ret = uvcg_video_enable(video, 0);
+       if (ret < 0)
+               return ret;
+
+       uvc->state = UVC_STATE_CONNECTED;
+
+       return 0;
+
 }
 
 static int
-- 
2.17.0

--
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to