Hi Guennadi,
On Monday, 11 December 2017 23:44:09 EET Guennadi Liakhovetski wrote:
> On Mon, 11 Dec 2017, Laurent Pinchart wrote:
> > On Monday, 11 December 2017 22:16:23 EET Laurent Pinchart wrote:
> >> On Wednesday, 6 December 2017 17:15:40 EET Guennadi Liakhovetski wrote:
> >>> From: Guennadi Liakhovetski <[email protected]>
> >>>
> >>> Some UVC video cameras contain metadata in their payload headers. This
> >>> patch extracts that data, adding more clock synchronisation
> >>> information, on both bulk and isochronous endpoints and makes it
> >>> available to the user space on a separate video node, using the
> >>> V4L2_CAP_META_CAPTURE capability and the V4L2_BUF_TYPE_META_CAPTURE
> >>> buffer queue type. By default, only the V4L2_META_FMT_UVC pixel format
> >>> is available from those nodes. However, cameras can be added to the
> >>> device ID table to additionally specify their own metadata format, in
> >>> which case that format will also become available from the metadata
> >>> node.
> >>>
> >>> Signed-off-by: Guennadi Liakhovetski <[email protected]>
> >>> ---
> >>>
> >>> v8: addressed comments and integrated changes from Laurent, thanks
> >>> again, e.g.:
> >>>
> >>> - multiple stylistic changes
> >>> - remove the UVC_DEV_FLAG_METADATA_NODE flag / quirk: nodes are now
> >>> created unconditionally
> >>> - reuse uvc_ioctl_querycap()
> >>> - reuse code in uvc_register_video()
> >>> - set an error flag when the metadata buffer overflows
> >>>
> >>> drivers/media/usb/uvc/Makefile | 2 +-
> >>> drivers/media/usb/uvc/uvc_driver.c | 15 ++-
> >>> drivers/media/usb/uvc/uvc_isight.c | 2 +-
> >>> drivers/media/usb/uvc/uvc_metadata.c | 179 ++++++++++++++++++++++++++++
> >>> drivers/media/usb/uvc/uvc_queue.c | 44 +++++++--
> >>> drivers/media/usb/uvc/uvc_video.c | 132 ++++++++++++++++++++++++--
> >>> drivers/media/usb/uvc/uvcvideo.h | 16 +++-
> >>> include/uapi/linux/uvcvideo.h | 26 +++++
> >>> 8 files changed, 394 insertions(+), 22 deletions(-)
> >>> create mode 100644 drivers/media/usb/uvc/uvc_metadata.c
> >>
> >> [snip]
> >>
> >> > diff --git a/drivers/media/usb/uvc/uvc_video.c
> >> > b/drivers/media/usb/uvc/uvc_video.c index 13f459e..2fc0bf2 100644
> >> > --- a/drivers/media/usb/uvc/uvc_video.c
> >> > +++ b/drivers/media/usb/uvc/uvc_video.c
> >>
> >> [snip]
> >>
> >>> +static void uvc_video_decode_meta(struct uvc_streaming *stream,
> >>> + struct uvc_buffer *meta_buf,
> >>> + const u8 *mem, unsigned int length)
> >>> +{
> >>> + struct uvc_meta_buf *meta;
> >>> + size_t len_std = 2;
> >>> + bool has_pts, has_scr;
> >>> + unsigned long flags;
> >>> + ktime_t time;
> >>> + const u8 *scr;
> >>> +
> >>> + if (!meta_buf || length == 2)
> >>> + return;
> >>> +
> >>> + if (meta_buf->length - meta_buf->bytesused <
> >>> + length + sizeof(meta->ns) + sizeof(meta->sof)) {
> >>> + meta_buf->error = 1;
> >>> + return;
> >>> + }
> >>> +
> >>> + has_pts = mem[1] & UVC_STREAM_PTS;
> >>> + has_scr = mem[1] & UVC_STREAM_SCR;
> >>> +
> >>> + if (has_pts) {
> >>> + len_std += 4;
> >>> + scr = mem + 6;
> >>> + } else {
> >>> + scr = mem + 2;
> >>> + }
> >>> +
> >>> + if (has_scr)
> >>> + len_std += 6;
> >>> +
> >>> + if (stream->meta.format == V4L2_META_FMT_UVC)
> >>> + length = len_std;
> >>> +
> >>> + if (length == len_std && (!has_scr ||
> >>> + !memcmp(scr, stream->clock.last_scr, 6)))
> >>> + return;
> >>> +
> >>> + meta = (struct uvc_meta_buf *)((u8 *)meta_buf->mem +
> >>> meta_buf->bytesused); + local_irq_save(flags);
> >>> + time = uvc_video_get_time();
> >>> + meta->sof = usb_get_current_frame_number(stream->dev->udev);
> >>
> >> You need a put_unaligned here too. If you're fine with the patch below
> >> there's no need to resubmit, and
> >
> > One more thing, __put_unaligned_cpu16() and __put_unaligned_cpu64() don't
> > compile on x86_64 with v4.12 (using media_build.git). I propose replacing
> > them with put_unaligned() which compiles and should do the right thing.
> Sure, thanks for catching! Shall I fix and resubmit?
If you're fine with
git://linuxtv.org/pinchartl/media.git uvc/next
there's no need to resubmit.
By the way, could you please review "uvcvideo: Factor out video device
registration to a function" and "uvcvideo: Report V4L2 device caps through the
video_device structure" ?
I will send a pull request after testing the code and getting those two
patches reviewed.
> >> Reviewed-by: Laurent Pinchart <[email protected]>
> >>
> >> Would you mind sending me your test tool patch ?
>
> Will send it offline.
>
> Thanks
> Guennadi
>
> >> diff --git a/drivers/media/usb/uvc/uvc_video.c b/drivers/media/usb/uvc/
> >> uvc_video.c
> >> index 2fc0bf2221db..02e4997a32bb 100644
> >> --- a/drivers/media/usb/uvc/uvc_video.c
> >> +++ b/drivers/media/usb/uvc/uvc_video.c
> >> @@ -1142,6 +1142,7 @@ static void uvc_video_decode_meta(struct
> >> uvc_streaming *stream,
> >> size_t len_std = 2;
> >> bool has_pts, has_scr;
> >> unsigned long flags;
> >> + unsigned int sof;
> >> ktime_t time;
> >> const u8 *scr;
> >>
> >> @@ -1177,9 +1178,10 @@ static void uvc_video_decode_meta(struct
> >> uvc_streaming *stream,
> >> meta = (struct uvc_meta_buf *)((u8 *)meta_buf->mem +
> >> meta_buf->bytesused);
> >>
> >> local_irq_save(flags);
> >> time = uvc_video_get_time();
> >> - meta->sof = usb_get_current_frame_number(stream->dev->udev);
> >> + sof = usb_get_current_frame_number(stream->dev->udev);
> >> local_irq_restore(flags);
> >> __put_unaligned_cpu64(ktime_to_ns(time), &meta->ns);
> >> + __put_unaligned_cpu16(sof, &meta->sof);
> >>
> >> if (has_scr)
> >> memcpy(stream->clock.last_scr, scr, 6);
> >>
> >>> + local_irq_restore(flags);
> >>> + __put_unaligned_cpu64(ktime_to_ns(time), &meta->ns);
> >>> +
> >>> + if (has_scr)
> >>> + memcpy(stream->clock.last_scr, scr, 6);
> >>> +
> >>> + memcpy(&meta->length, mem, length);
> >>> + meta_buf->bytesused += length + sizeof(meta->ns) +
> >>> sizeof(meta->sof);
> >>> +
> >>> + uvc_trace(UVC_TRACE_FRAME,
> >>> + "%s(): t-sys %lluns, SOF %u, len %u, flags 0x%x, PTS %u, STC
> >>> %u
> >>> frame
> >>> SOF %u\n", + __func__, time, meta->sof, meta->length, meta-
>flags,
> >>> + has_pts ? *(u32 *)meta->buf : 0,
> >>> + has_scr ? *(u32 *)scr : 0,
> >>> + has_scr ? *(u32 *)(scr + 4) & 0x7ff : 0);
> >>> +}
> >>
> >> [snip]
--
Regards,
Laurent Pinchart