Hi Sergey,

Thank you for the patch.

On Wed, Oct 02, 2019 at 01:01:02PM +0000, Sergey Zakharchenko wrote:
> This device does not function correctly in raw mode in kernel
> versions validating buffer sizes in bulk mode. It erroneously
> announces 16 bits per pixel instead of 12 for NV12 format, so it
> needs this quirk to fix computed frame size and avoid legitimate
> frames getting discarded.
> 
> Signed-off-by: Sergey Zakharchenko <szakharche...@digital-loggers.com>
> ---
>  drivers/media/usb/uvc/uvc_driver.c | 27 +++++++++++++++++++++++++++
>  drivers/media/usb/uvc/uvcvideo.h   |  1 +
>  2 files changed, 28 insertions(+)
> 
> diff --git a/drivers/media/usb/uvc/uvc_driver.c 
> b/drivers/media/usb/uvc/uvc_driver.c
> index 66ee168ddc7e..23f62456946d 100644
> --- a/drivers/media/usb/uvc/uvc_driver.c
> +++ b/drivers/media/usb/uvc/uvc_driver.c
> @@ -446,10 +446,12 @@ static int uvc_parse_format(struct uvc_device *dev,
>       struct usb_host_interface *alts = intf->cur_altsetting;
>       struct uvc_format_desc *fmtdesc;
>       struct uvc_frame *frame;
> +     const struct v4l2_format_info *info;
>       const unsigned char *start = buffer;
>       unsigned int width_multiplier = 1;
>       unsigned int interval;
>       unsigned int i, n;
> +     unsigned int div;
>       u8 ftype;
>  
>       format->type = buffer[2];
> @@ -497,6 +499,18 @@ static int uvc_parse_format(struct uvc_device *dev,
>                       }
>               }
>  
> +             /* Some devices report bpp that doesn't match the format. */
> +             if (dev->quirks & UVC_QUIRK_FORCE_BPP) {
> +                     info = v4l2_format_info(format->fcc);
> +                     if (info) {
> +                             div = info->hdiv * info->vdiv;
> +                             n = info->bpp[0] * div;
> +                             for (i = 1; i < info->comp_planes; i++)
> +                                     n += info->bpp[i];
> +                             format->bpp = DIV_ROUND_UP(8 * n, div);
> +                     }
> +             }

Do you think it would make sense to do this by default, without
requiring a quirk ? Or are there cases where this calculation would lead
to incorrect results while the bpp reported by the camera would be right
?

> +
>               if (buffer[2] == UVC_VS_FORMAT_UNCOMPRESSED) {
>                       ftype = UVC_VS_FRAME_UNCOMPRESSED;
>               } else {
> @@ -2384,6 +2398,10 @@ static const struct uvc_device_info uvc_quirk_force_y8 
> = {
>       .quirks = UVC_QUIRK_FORCE_Y8,
>  };
>  
> +static const struct uvc_device_info uvc_quirk_force_bpp = {
> +     .quirks = UVC_QUIRK_FORCE_BPP,
> +};
> +
>  #define UVC_INFO_QUIRK(q) (kernel_ulong_t)&(struct uvc_device_info){.quirks 
> = q}
>  #define UVC_INFO_META(m) (kernel_ulong_t)&(struct uvc_device_info) \
>       {.meta_format = m}
> @@ -2869,6 +2887,15 @@ static const struct usb_device_id uvc_ids[] = {
>         .bInterfaceSubClass   = 1,
>         .bInterfaceProtocol   = 0,
>         .driver_info          = UVC_INFO_META(V4L2_META_FMT_D4XX) },
> +     /* GEO Semiconductor GC6500 */
> +     { .match_flags          = USB_DEVICE_ID_MATCH_DEVICE
> +                             | USB_DEVICE_ID_MATCH_INT_INFO,
> +       .idVendor             = 0x29fe,
> +       .idProduct            = 0x4d53,

Could you please keep the entries sorted by idVendor/idProduct ?

> +       .bInterfaceClass      = USB_CLASS_VIDEO,
> +       .bInterfaceSubClass   = 1,
> +       .bInterfaceProtocol   = 0,
> +       .driver_info          = (kernel_ulong_t)&uvc_quirk_force_bpp },

As this is the only device using this quirk, you can drop
uvc_quirk_force_bpp and use

        .driver_info            = UVC_INFO_QUIRK(UVC_QUIRK_FORCE_BPP) },

>       /* Generic USB Video Class */
>       { USB_INTERFACE_INFO(USB_CLASS_VIDEO, 1, UVC_PC_PROTOCOL_UNDEFINED) },
>       { USB_INTERFACE_INFO(USB_CLASS_VIDEO, 1, UVC_PC_PROTOCOL_15) },
> diff --git a/drivers/media/usb/uvc/uvcvideo.h 
> b/drivers/media/usb/uvc/uvcvideo.h
> index c7c1baa90dea..24e3d8c647e7 100644
> --- a/drivers/media/usb/uvc/uvcvideo.h
> +++ b/drivers/media/usb/uvc/uvcvideo.h
> @@ -198,6 +198,7 @@
>  #define UVC_QUIRK_RESTRICT_FRAME_RATE        0x00000200
>  #define UVC_QUIRK_RESTORE_CTRLS_ON_INIT      0x00000400
>  #define UVC_QUIRK_FORCE_Y8           0x00000800
> +#define UVC_QUIRK_FORCE_BPP          0x00001000
>  
>  /* Format flags */
>  #define UVC_FMT_FLAG_COMPRESSED              0x00000001
> 
> base-commit: 20a438d53fd9d12a894161bc56cbeab7a9993c39
> prerequisite-patch-id: 521eb9602d395ea667eecc75cd2273b59cd3ed76

-- 
Regards,

Laurent Pinchart

Reply via email to