Commit 6243c83b authored by Ricardo Ribalda's avatar Ricardo Ribalda Committed by Laurent Pinchart

media: uvcvideo: Allow hw clock updates with buffers not full

With UVC 1.5 we get as little as one clock sample per frame. Which means
that it takes 32 frames to move from the software timestamp to the
hardware timestamp method.

This results in abrupt changes in the timestamping after 32 frames (~1
second), resulting in noticeable artifacts when used for encoding.

With this patch we modify the update algorithm to work with whatever
amount of values are available.
Tested-by: default avatarHungNien Chen <hn.chen@sunplusit.com>
Reviewed-by: default avatarSergey Senozhatsky <senozhatsky@chromium.org>
Reviewed-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Signed-off-by: default avatarRicardo Ribalda <ribalda@chromium.org>
Reviewed-by: default avatarTomasz Figa <tfiga@chromium.org>
Link: https://lore.kernel.org/r/20240323-resend-hwtimestamp-v10-4-b08e590d97c7@chromium.orgSigned-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
parent 9183c6f1
...@@ -768,10 +768,10 @@ void uvc_video_clock_update(struct uvc_streaming *stream, ...@@ -768,10 +768,10 @@ void uvc_video_clock_update(struct uvc_streaming *stream,
spin_lock_irqsave(&clock->lock, flags); spin_lock_irqsave(&clock->lock, flags);
if (clock->count < clock->size) if (clock->count < 2)
goto done; goto done;
first = &clock->samples[clock->head]; first = &clock->samples[(clock->head - clock->count + clock->size) % clock->size];
last = &clock->samples[(clock->head - 1 + clock->size) % clock->size]; last = &clock->samples[(clock->head - 1 + clock->size) % clock->size];
/* First step, PTS to SOF conversion. */ /* First step, PTS to SOF conversion. */
...@@ -786,6 +786,18 @@ void uvc_video_clock_update(struct uvc_streaming *stream, ...@@ -786,6 +786,18 @@ void uvc_video_clock_update(struct uvc_streaming *stream,
if (y2 < y1) if (y2 < y1)
y2 += 2048 << 16; y2 += 2048 << 16;
/*
* Have at least 1/4 of a second of timestamps before we
* try to do any calculation. Otherwise we do not have enough
* precision. This value was determined by running Android CTS
* on different devices.
*
* dev_sof runs at 1KHz, and we have a fixed point precision of
* 16 bits.
*/
if ((y2 - y1) < ((1000 / 4) << 16))
goto done;
y = (u64)(y2 - y1) * (1ULL << 31) + (u64)y1 * (u64)x2 y = (u64)(y2 - y1) * (1ULL << 31) + (u64)y1 * (u64)x2
- (u64)y2 * (u64)x1; - (u64)y2 * (u64)x1;
y = div_u64(y, x2 - x1); y = div_u64(y, x2 - x1);
......
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