[3/5] libcamera: v4l2: Add wallclock metadata to video devices
diff mbox series

Message ID 20241206142742.7931-4-david.plowman@raspberrypi.com
State Superseded
Headers show
Series
  • Implement wallclock timestamps for frames
Related show

Commit Message

David Plowman Dec. 6, 2024, 2:27 p.m. UTC
FrameMetadata is extended to include a wallclock timestamp.

When a frame is dequeued, we use the ClockRecovery class to generate a
good estimate of a wallclock timestamp to correspond to the frame's
SensorTimestamp.

Pipeline handlers must enable wallclocks for chosen devices by passing
an appropriate ClockRecovery object.

Signed-off-by: David Plowman <david.plowman@raspberrypi.com>
---
 include/libcamera/framebuffer.h               |  1 +
 include/libcamera/internal/v4l2_videodevice.h |  5 +++
 src/libcamera/v4l2_videodevice.cpp            | 36 ++++++++++++++++++-
 3 files changed, 41 insertions(+), 1 deletion(-)

Comments

Kieran Bingham Dec. 6, 2024, 3:37 p.m. UTC | #1
Quoting David Plowman (2024-12-06 14:27:40)
> FrameMetadata is extended to include a wallclock timestamp.
> 
> When a frame is dequeued, we use the ClockRecovery class to generate a
> good estimate of a wallclock timestamp to correspond to the frame's
> SensorTimestamp.
> 
> Pipeline handlers must enable wallclocks for chosen devices by passing
> an appropriate ClockRecovery object.
> 
> Signed-off-by: David Plowman <david.plowman@raspberrypi.com>
> ---
>  include/libcamera/framebuffer.h               |  1 +
>  include/libcamera/internal/v4l2_videodevice.h |  5 +++
>  src/libcamera/v4l2_videodevice.cpp            | 36 ++++++++++++++++++-
>  3 files changed, 41 insertions(+), 1 deletion(-)
> 
> diff --git a/include/libcamera/framebuffer.h b/include/libcamera/framebuffer.h
> index ff839243..4579d0c6 100644
> --- a/include/libcamera/framebuffer.h
> +++ b/include/libcamera/framebuffer.h
> @@ -35,6 +35,7 @@ struct FrameMetadata {
>         Status status;
>         unsigned int sequence;
>         uint64_t timestamp;
> +       uint64_t wallClock;

This breaks in the CI build I'm afraid:

https://gitlab.freedesktop.org/camera/libcamera/-/jobs/67848569


[481/650] Generating Documentation/doxygen-internal with a custom command
FAILED: Documentation/internal-api-html
/usr/bin/doxygen Documentation/Doxyfile-internal
/builds/camera/libcamera/include/libcamera/framebuffer.h:38: error: Member wallClock (variable) of struct libcamera::FrameMetadata is not documented. (warning treated as error, aborting now)

Either we need to add the doc (which is what I'd do) - or potentially
depending on discussions below, remove the addition...

>  
>         Span<Plane> planes() { return planes_; }
>         Span<const Plane> planes() const { return planes_; }
> diff --git a/include/libcamera/internal/v4l2_videodevice.h b/include/libcamera/internal/v4l2_videodevice.h
> index f021c2a0..5327c112 100644
> --- a/include/libcamera/internal/v4l2_videodevice.h
> +++ b/include/libcamera/internal/v4l2_videodevice.h
> @@ -37,6 +37,7 @@
>  
>  namespace libcamera {
>  
> +class ClockRecovery;
>  class EventNotifier;
>  class MediaDevice;
>  class MediaEntity;
> @@ -232,6 +233,8 @@ public:
>  
>         V4L2PixelFormat toV4L2PixelFormat(const PixelFormat &pixelFormat) const;
>  
> +       void enableWallClock(ClockRecovery *wallClockRecovery);
> +
>  protected:
>         std::string logPrefix() const override;
>  
> @@ -290,6 +293,8 @@ private:
>  
>         Timer watchdog_;
>         utils::Duration watchdogDuration_;
> +
> +       ClockRecovery *wallClockRecovery_;
>  };
>  
>  class V4L2M2MDevice
> diff --git a/src/libcamera/v4l2_videodevice.cpp b/src/libcamera/v4l2_videodevice.cpp
> index a5cf6784..2007dffc 100644
> --- a/src/libcamera/v4l2_videodevice.cpp
> +++ b/src/libcamera/v4l2_videodevice.cpp
> @@ -26,6 +26,7 @@
>  #include <libcamera/base/unique_fd.h>
>  #include <libcamera/base/utils.h>
>  
> +#include "libcamera/internal/clock_recovery.h"
>  #include "libcamera/internal/formats.h"
>  #include "libcamera/internal/framebuffer.h"
>  #include "libcamera/internal/media_device.h"
> @@ -534,7 +535,7 @@ std::ostream &operator<<(std::ostream &out, const V4L2DeviceFormat &f)
>  V4L2VideoDevice::V4L2VideoDevice(const std::string &deviceNode)
>         : V4L2Device(deviceNode), formatInfo_(nullptr), cache_(nullptr),
>           fdBufferNotifier_(nullptr), state_(State::Stopped),
> -         watchdogDuration_(0.0)
> +         watchdogDuration_(0.0), wallClockRecovery_(nullptr)
>  {
>         /*
>          * We default to an MMAP based CAPTURE video device, however this will
> @@ -1865,6 +1866,17 @@ FrameBuffer *V4L2VideoDevice::dequeueBuffer()
>         metadata.timestamp = buf.timestamp.tv_sec * 1000000000ULL
>                            + buf.timestamp.tv_usec * 1000ULL;
>  
> +       metadata.wallClock = 0;
> +       if (wallClockRecovery_) {

I know Laurent discussed passing in the 'ClockRecovery' class as a way
of giving the responsibility to the Pipeline handler - but I'm not
particularly fond of it. (I won't object if this is preferred all
round).

To me - especially if there is an addition of uint64_t wallClock in the
FrameMetadata, then this should just always be enabled.

But if it's /really/ optional and passed in, - then I don't think we
should store a metadata.wallClock below.

> +               /*
> +                * Sample the internal (CLOCK_BOOTTIME) and realtime (CLOCK_REALTIME) clocks and
> +                * update the clock recovery model. Then derive the wallclock estimate for the
> +                * frame timestamp.
> +                */
> +               wallClockRecovery_->addSample();
> +               metadata.wallClock = wallClockRecovery_->getOutput(metadata.timestamp / 1000);

As that could be called in the pipeline handler. And to me - then I
would ask why are we getting the V4L2VideoDevice to manage the
clockRecovery object..

(And I think that it should 'just' be in the V4L2VideoDevice in this
instance if we're going to provide wall clock timestamps of buffers).


> +       }
> +
>         if (V4L2_TYPE_IS_OUTPUT(buf.type))
>                 return buffer;
>  
> @@ -1965,6 +1977,11 @@ int V4L2VideoDevice::streamOn()
>                 return ret;
>         }
>  
> +       if (wallClockRecovery_) {
> +               /* A good moment to sample the clocks to improve the clock recovery model. */
> +               wallClockRecovery_->addSample();
> +       }
> +
>         state_ = State::Streaming;
>         if (watchdogDuration_ && !queuedBuffers_.empty())
>                 watchdog_.start(std::chrono::duration_cast<std::chrono::milliseconds>(watchdogDuration_));
> @@ -2120,6 +2137,23 @@ V4L2PixelFormat V4L2VideoDevice::toV4L2PixelFormat(const PixelFormat &pixelForma
>         return {};
>  }
>  
> +/**
> + * \brief Enable wall clock timestamps for this device
> + * \param[in] wallClockRecovery an appropriately configured ClockRecovery, or
> + * nullptr to disable wall clocks

Is there a reason we can think of for /why/ we would disable the wall
clocks?

It doesn't seem like a 'desired' feature to me - unless the costs of
doing the clock recovery are prohibitively expensive? Which I wouldn't
expect it to be ?



> + *
> + * When buffers are dequeued, wall clock timestamps will be generated that
> + * correspond to the frame's timestamp, as returned by V4l2.
> + */
> +void V4L2VideoDevice::enableWallClock(ClockRecovery *wallClockRecovery)
> +{
> +       wallClockRecovery_ = wallClockRecovery;
> +
> +       /* Also a reasonable moment to sample the two clocks. */
> +       if (wallClockRecovery_)
> +               wallClockRecovery_->addSample();
> +}
> +
>  /**
>   * \class V4L2M2MDevice
>   * \brief Memory-to-Memory video device
> -- 
> 2.39.5
>
David Plowman Dec. 6, 2024, 4:53 p.m. UTC | #2
Hi Kieran

Thanks for the review!

On Fri, 6 Dec 2024 at 15:37, Kieran Bingham
<kieran.bingham@ideasonboard.com> wrote:
>
> Quoting David Plowman (2024-12-06 14:27:40)
> > FrameMetadata is extended to include a wallclock timestamp.
> >
> > When a frame is dequeued, we use the ClockRecovery class to generate a
> > good estimate of a wallclock timestamp to correspond to the frame's
> > SensorTimestamp.
> >
> > Pipeline handlers must enable wallclocks for chosen devices by passing
> > an appropriate ClockRecovery object.
> >
> > Signed-off-by: David Plowman <david.plowman@raspberrypi.com>
> > ---
> >  include/libcamera/framebuffer.h               |  1 +
> >  include/libcamera/internal/v4l2_videodevice.h |  5 +++
> >  src/libcamera/v4l2_videodevice.cpp            | 36 ++++++++++++++++++-
> >  3 files changed, 41 insertions(+), 1 deletion(-)
> >
> > diff --git a/include/libcamera/framebuffer.h b/include/libcamera/framebuffer.h
> > index ff839243..4579d0c6 100644
> > --- a/include/libcamera/framebuffer.h
> > +++ b/include/libcamera/framebuffer.h
> > @@ -35,6 +35,7 @@ struct FrameMetadata {
> >         Status status;
> >         unsigned int sequence;
> >         uint64_t timestamp;
> > +       uint64_t wallClock;
>
> This breaks in the CI build I'm afraid:
>
> https://gitlab.freedesktop.org/camera/libcamera/-/jobs/67848569
>
>
> [481/650] Generating Documentation/doxygen-internal with a custom command
> FAILED: Documentation/internal-api-html
> /usr/bin/doxygen Documentation/Doxyfile-internal
> /builds/camera/libcamera/include/libcamera/framebuffer.h:38: error: Member wallClock (variable) of struct libcamera::FrameMetadata is not documented. (warning treated as error, aborting now)
>
> Either we need to add the doc (which is what I'd do) - or potentially
> depending on discussions below, remove the addition...

Yep, well I think this is a bit above my pay grade. The "wall clock"
sure feels like it belongs with the regular timestamp. If that's a
problem, I'm happy to put it somewhere else, though I don't think I
would know where currently. Perhaps some others can offer an opinion
here?

>
> >
> >         Span<Plane> planes() { return planes_; }
> >         Span<const Plane> planes() const { return planes_; }
> > diff --git a/include/libcamera/internal/v4l2_videodevice.h b/include/libcamera/internal/v4l2_videodevice.h
> > index f021c2a0..5327c112 100644
> > --- a/include/libcamera/internal/v4l2_videodevice.h
> > +++ b/include/libcamera/internal/v4l2_videodevice.h
> > @@ -37,6 +37,7 @@
> >
> >  namespace libcamera {
> >
> > +class ClockRecovery;
> >  class EventNotifier;
> >  class MediaDevice;
> >  class MediaEntity;
> > @@ -232,6 +233,8 @@ public:
> >
> >         V4L2PixelFormat toV4L2PixelFormat(const PixelFormat &pixelFormat) const;
> >
> > +       void enableWallClock(ClockRecovery *wallClockRecovery);
> > +
> >  protected:
> >         std::string logPrefix() const override;
> >
> > @@ -290,6 +293,8 @@ private:
> >
> >         Timer watchdog_;
> >         utils::Duration watchdogDuration_;
> > +
> > +       ClockRecovery *wallClockRecovery_;
> >  };
> >
> >  class V4L2M2MDevice
> > diff --git a/src/libcamera/v4l2_videodevice.cpp b/src/libcamera/v4l2_videodevice.cpp
> > index a5cf6784..2007dffc 100644
> > --- a/src/libcamera/v4l2_videodevice.cpp
> > +++ b/src/libcamera/v4l2_videodevice.cpp
> > @@ -26,6 +26,7 @@
> >  #include <libcamera/base/unique_fd.h>
> >  #include <libcamera/base/utils.h>
> >
> > +#include "libcamera/internal/clock_recovery.h"
> >  #include "libcamera/internal/formats.h"
> >  #include "libcamera/internal/framebuffer.h"
> >  #include "libcamera/internal/media_device.h"
> > @@ -534,7 +535,7 @@ std::ostream &operator<<(std::ostream &out, const V4L2DeviceFormat &f)
> >  V4L2VideoDevice::V4L2VideoDevice(const std::string &deviceNode)
> >         : V4L2Device(deviceNode), formatInfo_(nullptr), cache_(nullptr),
> >           fdBufferNotifier_(nullptr), state_(State::Stopped),
> > -         watchdogDuration_(0.0)
> > +         watchdogDuration_(0.0), wallClockRecovery_(nullptr)
> >  {
> >         /*
> >          * We default to an MMAP based CAPTURE video device, however this will
> > @@ -1865,6 +1866,17 @@ FrameBuffer *V4L2VideoDevice::dequeueBuffer()
> >         metadata.timestamp = buf.timestamp.tv_sec * 1000000000ULL
> >                            + buf.timestamp.tv_usec * 1000ULL;
> >
> > +       metadata.wallClock = 0;
> > +       if (wallClockRecovery_) {
>
> I know Laurent discussed passing in the 'ClockRecovery' class as a way
> of giving the responsibility to the Pipeline handler - but I'm not
> particularly fond of it. (I won't object if this is preferred all
> round).
>
> To me - especially if there is an addition of uint64_t wallClock in the
> FrameMetadata, then this should just always be enabled.
>
> But if it's /really/ optional and passed in, - then I don't think we
> should store a metadata.wallClock below.

Indeed, I don't actually feel terribly strongly either, I'd obviously
prefer to reach a consensus  ahead of time rather than produce
different implementations that we then don't like!

I'd certainly be happy always to enable wall clocks, it depends
whether we think the burden of a bit of floating point arithmetic
every time we add a sample is a problem. Probably not.

To be fair, I do wonder a little bit whether a single ClockRecovery
for the entire system might be the right thing to do. Though I'm not
sure where it would go, or how you would add sporadic samples to it.
But you could easily rate-limit the model updates, if you wanted to.

On the downside, that would certainly make it more difficult for
anyone to configure it differently, but maybe no one would really need
to? Questions, questions...

>
> > +               /*
> > +                * Sample the internal (CLOCK_BOOTTIME) and realtime (CLOCK_REALTIME) clocks and
> > +                * update the clock recovery model. Then derive the wallclock estimate for the
> > +                * frame timestamp.
> > +                */
> > +               wallClockRecovery_->addSample();
> > +               metadata.wallClock = wallClockRecovery_->getOutput(metadata.timestamp / 1000);
>
> As that could be called in the pipeline handler. And to me - then I
> would ask why are we getting the V4L2VideoDevice to manage the
> clockRecovery object..
>
> (And I think that it should 'just' be in the V4L2VideoDevice in this
> instance if we're going to provide wall clock timestamps of buffers).

I would agree that I think I'm against moving more of it into the PH.
After all, the whole point was to put it here so that everyone can get
the new feature easily. But I don't super-love the
pass-the-ClockRecovery-in either. I can certainly see that argument
for a single "global" ClockRecovery. I dunno...

>
>
> > +       }
> > +
> >         if (V4L2_TYPE_IS_OUTPUT(buf.type))
> >                 return buffer;
> >
> > @@ -1965,6 +1977,11 @@ int V4L2VideoDevice::streamOn()
> >                 return ret;
> >         }
> >
> > +       if (wallClockRecovery_) {
> > +               /* A good moment to sample the clocks to improve the clock recovery model. */
> > +               wallClockRecovery_->addSample();
> > +       }
> > +
> >         state_ = State::Streaming;
> >         if (watchdogDuration_ && !queuedBuffers_.empty())
> >                 watchdog_.start(std::chrono::duration_cast<std::chrono::milliseconds>(watchdogDuration_));
> > @@ -2120,6 +2137,23 @@ V4L2PixelFormat V4L2VideoDevice::toV4L2PixelFormat(const PixelFormat &pixelForma
> >         return {};
> >  }
> >
> > +/**
> > + * \brief Enable wall clock timestamps for this device
> > + * \param[in] wallClockRecovery an appropriately configured ClockRecovery, or
> > + * nullptr to disable wall clocks
>
> Is there a reason we can think of for /why/ we would disable the wall
> clocks?
>
> It doesn't seem like a 'desired' feature to me - unless the costs of
> doing the clock recovery are prohibitively expensive? Which I wouldn't
> expect it to be ?

Yeah, no reason... just no compelling reason not to allow it either.
Having them "always enabled" feels nice in many ways...

Thanks!
David

>
>
>
> > + *
> > + * When buffers are dequeued, wall clock timestamps will be generated that
> > + * correspond to the frame's timestamp, as returned by V4l2.
> > + */
> > +void V4L2VideoDevice::enableWallClock(ClockRecovery *wallClockRecovery)
> > +{
> > +       wallClockRecovery_ = wallClockRecovery;
> > +
> > +       /* Also a reasonable moment to sample the two clocks. */
> > +       if (wallClockRecovery_)
> > +               wallClockRecovery_->addSample();
> > +}
> > +
> >  /**
> >   * \class V4L2M2MDevice
> >   * \brief Memory-to-Memory video device
> > --
> > 2.39.5
> >
Naushir Patuck Dec. 12, 2024, 1:04 p.m. UTC | #3
Hi,


On Fri, 6 Dec 2024 at 16:53, David Plowman
<david.plowman@raspberrypi.com> wrote:
>
> Hi Kieran
>
> Thanks for the review!
>
> On Fri, 6 Dec 2024 at 15:37, Kieran Bingham
> <kieran.bingham@ideasonboard.com> wrote:
> >
> > Quoting David Plowman (2024-12-06 14:27:40)
> > > FrameMetadata is extended to include a wallclock timestamp.
> > >
> > > When a frame is dequeued, we use the ClockRecovery class to generate a
> > > good estimate of a wallclock timestamp to correspond to the frame's
> > > SensorTimestamp.
> > >
> > > Pipeline handlers must enable wallclocks for chosen devices by passing
> > > an appropriate ClockRecovery object.
> > >
> > > Signed-off-by: David Plowman <david.plowman@raspberrypi.com>
> > > ---
> > >  include/libcamera/framebuffer.h               |  1 +
> > >  include/libcamera/internal/v4l2_videodevice.h |  5 +++
> > >  src/libcamera/v4l2_videodevice.cpp            | 36 ++++++++++++++++++-
> > >  3 files changed, 41 insertions(+), 1 deletion(-)
> > >
> > > diff --git a/include/libcamera/framebuffer.h b/include/libcamera/framebuffer.h
> > > index ff839243..4579d0c6 100644
> > > --- a/include/libcamera/framebuffer.h
> > > +++ b/include/libcamera/framebuffer.h
> > > @@ -35,6 +35,7 @@ struct FrameMetadata {
> > >         Status status;
> > >         unsigned int sequence;
> > >         uint64_t timestamp;
> > > +       uint64_t wallClock;
> >
> > This breaks in the CI build I'm afraid:
> >
> > https://gitlab.freedesktop.org/camera/libcamera/-/jobs/67848569
> >
> >
> > [481/650] Generating Documentation/doxygen-internal with a custom command
> > FAILED: Documentation/internal-api-html
> > /usr/bin/doxygen Documentation/Doxyfile-internal
> > /builds/camera/libcamera/include/libcamera/framebuffer.h:38: error: Member wallClock (variable) of struct libcamera::FrameMetadata is not documented. (warning treated as error, aborting now)
> >
> > Either we need to add the doc (which is what I'd do) - or potentially
> > depending on discussions below, remove the addition...
>
> Yep, well I think this is a bit above my pay grade. The "wall clock"
> sure feels like it belongs with the regular timestamp. If that's a
> problem, I'm happy to put it somewhere else, though I don't think I
> would know where currently. Perhaps some others can offer an opinion
> here?
>
> >
> > >
> > >         Span<Plane> planes() { return planes_; }
> > >         Span<const Plane> planes() const { return planes_; }
> > > diff --git a/include/libcamera/internal/v4l2_videodevice.h b/include/libcamera/internal/v4l2_videodevice.h
> > > index f021c2a0..5327c112 100644
> > > --- a/include/libcamera/internal/v4l2_videodevice.h
> > > +++ b/include/libcamera/internal/v4l2_videodevice.h
> > > @@ -37,6 +37,7 @@
> > >
> > >  namespace libcamera {
> > >
> > > +class ClockRecovery;
> > >  class EventNotifier;
> > >  class MediaDevice;
> > >  class MediaEntity;
> > > @@ -232,6 +233,8 @@ public:
> > >
> > >         V4L2PixelFormat toV4L2PixelFormat(const PixelFormat &pixelFormat) const;
> > >
> > > +       void enableWallClock(ClockRecovery *wallClockRecovery);
> > > +
> > >  protected:
> > >         std::string logPrefix() const override;
> > >
> > > @@ -290,6 +293,8 @@ private:
> > >
> > >         Timer watchdog_;
> > >         utils::Duration watchdogDuration_;
> > > +
> > > +       ClockRecovery *wallClockRecovery_;
> > >  };
> > >
> > >  class V4L2M2MDevice
> > > diff --git a/src/libcamera/v4l2_videodevice.cpp b/src/libcamera/v4l2_videodevice.cpp
> > > index a5cf6784..2007dffc 100644
> > > --- a/src/libcamera/v4l2_videodevice.cpp
> > > +++ b/src/libcamera/v4l2_videodevice.cpp
> > > @@ -26,6 +26,7 @@
> > >  #include <libcamera/base/unique_fd.h>
> > >  #include <libcamera/base/utils.h>
> > >
> > > +#include "libcamera/internal/clock_recovery.h"
> > >  #include "libcamera/internal/formats.h"
> > >  #include "libcamera/internal/framebuffer.h"
> > >  #include "libcamera/internal/media_device.h"
> > > @@ -534,7 +535,7 @@ std::ostream &operator<<(std::ostream &out, const V4L2DeviceFormat &f)
> > >  V4L2VideoDevice::V4L2VideoDevice(const std::string &deviceNode)
> > >         : V4L2Device(deviceNode), formatInfo_(nullptr), cache_(nullptr),
> > >           fdBufferNotifier_(nullptr), state_(State::Stopped),
> > > -         watchdogDuration_(0.0)
> > > +         watchdogDuration_(0.0), wallClockRecovery_(nullptr)
> > >  {
> > >         /*
> > >          * We default to an MMAP based CAPTURE video device, however this will
> > > @@ -1865,6 +1866,17 @@ FrameBuffer *V4L2VideoDevice::dequeueBuffer()
> > >         metadata.timestamp = buf.timestamp.tv_sec * 1000000000ULL
> > >                            + buf.timestamp.tv_usec * 1000ULL;
> > >
> > > +       metadata.wallClock = 0;
> > > +       if (wallClockRecovery_) {
> >
> > I know Laurent discussed passing in the 'ClockRecovery' class as a way
> > of giving the responsibility to the Pipeline handler - but I'm not
> > particularly fond of it. (I won't object if this is preferred all
> > round).
> >
> > To me - especially if there is an addition of uint64_t wallClock in the
> > FrameMetadata, then this should just always be enabled.
> >
> > But if it's /really/ optional and passed in, - then I don't think we
> > should store a metadata.wallClock below.
>
> Indeed, I don't actually feel terribly strongly either, I'd obviously
> prefer to reach a consensus  ahead of time rather than produce
> different implementations that we then don't like!
>
> I'd certainly be happy always to enable wall clocks, it depends
> whether we think the burden of a bit of floating point arithmetic
> every time we add a sample is a problem. Probably not.
>
> To be fair, I do wonder a little bit whether a single ClockRecovery
> for the entire system might be the right thing to do. Though I'm not
> sure where it would go, or how you would add sporadic samples to it.
> But you could easily rate-limit the model updates, if you wanted to.
>
> On the downside, that would certainly make it more difficult for
> anyone to configure it differently, but maybe no one would really need
> to? Questions, questions...
>
> >
> > > +               /*
> > > +                * Sample the internal (CLOCK_BOOTTIME) and realtime (CLOCK_REALTIME) clocks and
> > > +                * update the clock recovery model. Then derive the wallclock estimate for the
> > > +                * frame timestamp.
> > > +                */
> > > +               wallClockRecovery_->addSample();
> > > +               metadata.wallClock = wallClockRecovery_->getOutput(metadata.timestamp / 1000);
> >
> > As that could be called in the pipeline handler. And to me - then I
> > would ask why are we getting the V4L2VideoDevice to manage the
> > clockRecovery object..
> >
> > (And I think that it should 'just' be in the V4L2VideoDevice in this
> > instance if we're going to provide wall clock timestamps of buffers).
>
> I would agree that I think I'm against moving more of it into the PH.
> After all, the whole point was to put it here so that everyone can get
> the new feature easily. But I don't super-love the
> pass-the-ClockRecovery-in either. I can certainly see that argument
> for a single "global" ClockRecovery. I dunno...

I tend to agree with Kieran that perhaps the wall clock recovery
becomes a standard feature of the V4L2VideoDevice and the framebuffer
metadata.wallClock field will be unconditionally valid.  The amount of
additional computation per frame (per device) is quite tiny, and I
doubt it would make any measurable impact.  As such I would suggest we
move the instantiation of the ClockRecovery object into
V4L2VideoDevice.

As for specifying custom clock recovery config, perhaps we just ignore
this problem (if it is indeed a problem) until we have a definitive
need to change it.  Since this is all in libcamera internals, we have
some leeway with changing this in the future.

Any objections to this?

Naush

>
> >
> >
> > > +       }
> > > +
> > >         if (V4L2_TYPE_IS_OUTPUT(buf.type))
> > >                 return buffer;
> > >
> > > @@ -1965,6 +1977,11 @@ int V4L2VideoDevice::streamOn()
> > >                 return ret;
> > >         }
> > >
> > > +       if (wallClockRecovery_) {
> > > +               /* A good moment to sample the clocks to improve the clock recovery model. */
> > > +               wallClockRecovery_->addSample();
> > > +       }
> > > +
> > >         state_ = State::Streaming;
> > >         if (watchdogDuration_ && !queuedBuffers_.empty())
> > >                 watchdog_.start(std::chrono::duration_cast<std::chrono::milliseconds>(watchdogDuration_));
> > > @@ -2120,6 +2137,23 @@ V4L2PixelFormat V4L2VideoDevice::toV4L2PixelFormat(const PixelFormat &pixelForma
> > >         return {};
> > >  }
> > >
> > > +/**
> > > + * \brief Enable wall clock timestamps for this device
> > > + * \param[in] wallClockRecovery an appropriately configured ClockRecovery, or
> > > + * nullptr to disable wall clocks
> >
> > Is there a reason we can think of for /why/ we would disable the wall
> > clocks?
> >
> > It doesn't seem like a 'desired' feature to me - unless the costs of
> > doing the clock recovery are prohibitively expensive? Which I wouldn't
> > expect it to be ?
>
> Yeah, no reason... just no compelling reason not to allow it either.
> Having them "always enabled" feels nice in many ways...
>
> Thanks!
> David
>
> >
> >
> >
> > > + *
> > > + * When buffers are dequeued, wall clock timestamps will be generated that
> > > + * correspond to the frame's timestamp, as returned by V4l2.
> > > + */
> > > +void V4L2VideoDevice::enableWallClock(ClockRecovery *wallClockRecovery)
> > > +{
> > > +       wallClockRecovery_ = wallClockRecovery;
> > > +
> > > +       /* Also a reasonable moment to sample the two clocks. */
> > > +       if (wallClockRecovery_)
> > > +               wallClockRecovery_->addSample();
> > > +}
> > > +
> > >  /**
> > >   * \class V4L2M2MDevice
> > >   * \brief Memory-to-Memory video device
> > > --
> > > 2.39.5
> > >
Laurent Pinchart Dec. 18, 2024, 2:35 a.m. UTC | #4
On Thu, Dec 12, 2024 at 01:04:11PM +0000, Naushir Patuck wrote:
> On Fri, 6 Dec 2024 at 16:53, David Plowman wrote:
> > On Fri, 6 Dec 2024 at 15:37, Kieran Bingham wrote:
> > > Quoting David Plowman (2024-12-06 14:27:40)
> > > > FrameMetadata is extended to include a wallclock timestamp.
> > > >
> > > > When a frame is dequeued, we use the ClockRecovery class to generate a
> > > > good estimate of a wallclock timestamp to correspond to the frame's
> > > > SensorTimestamp.
> > > >
> > > > Pipeline handlers must enable wallclocks for chosen devices by passing
> > > > an appropriate ClockRecovery object.
> > > >
> > > > Signed-off-by: David Plowman <david.plowman@raspberrypi.com>
> > > > ---
> > > >  include/libcamera/framebuffer.h               |  1 +
> > > >  include/libcamera/internal/v4l2_videodevice.h |  5 +++
> > > >  src/libcamera/v4l2_videodevice.cpp            | 36 ++++++++++++++++++-
> > > >  3 files changed, 41 insertions(+), 1 deletion(-)
> > > >
> > > > diff --git a/include/libcamera/framebuffer.h b/include/libcamera/framebuffer.h
> > > > index ff839243..4579d0c6 100644
> > > > --- a/include/libcamera/framebuffer.h
> > > > +++ b/include/libcamera/framebuffer.h
> > > > @@ -35,6 +35,7 @@ struct FrameMetadata {
> > > >         Status status;
> > > >         unsigned int sequence;
> > > >         uint64_t timestamp;
> > > > +       uint64_t wallClock;
> > >
> > > This breaks in the CI build I'm afraid:
> > >
> > > https://gitlab.freedesktop.org/camera/libcamera/-/jobs/67848569
> > >
> > >
> > > [481/650] Generating Documentation/doxygen-internal with a custom command
> > > FAILED: Documentation/internal-api-html
> > > /usr/bin/doxygen Documentation/Doxyfile-internal
> > > /builds/camera/libcamera/include/libcamera/framebuffer.h:38: error: Member wallClock (variable) of struct libcamera::FrameMetadata is not documented. (warning treated as error, aborting now)
> > >
> > > Either we need to add the doc (which is what I'd do) - or potentially
> > > depending on discussions below, remove the addition...
> >
> > Yep, well I think this is a bit above my pay grade. The "wall clock"
> > sure feels like it belongs with the regular timestamp. If that's a
> > problem, I'm happy to put it somewhere else, though I don't think I
> > would know where currently. Perhaps some others can offer an opinion
> > here?

Even if I'm still considering whether or not we should remove timestamp
from FrameMetadata, I think it makes sense to have either none or both,
so it's fine to keep it here. What's missing is documentation.

> > > >
> > > >         Span<Plane> planes() { return planes_; }
> > > >         Span<const Plane> planes() const { return planes_; }
> > > > diff --git a/include/libcamera/internal/v4l2_videodevice.h b/include/libcamera/internal/v4l2_videodevice.h
> > > > index f021c2a0..5327c112 100644
> > > > --- a/include/libcamera/internal/v4l2_videodevice.h
> > > > +++ b/include/libcamera/internal/v4l2_videodevice.h
> > > > @@ -37,6 +37,7 @@
> > > >
> > > >  namespace libcamera {
> > > >
> > > > +class ClockRecovery;
> > > >  class EventNotifier;
> > > >  class MediaDevice;
> > > >  class MediaEntity;
> > > > @@ -232,6 +233,8 @@ public:
> > > >
> > > >         V4L2PixelFormat toV4L2PixelFormat(const PixelFormat &pixelFormat) const;
> > > >
> > > > +       void enableWallClock(ClockRecovery *wallClockRecovery);
> > > > +
> > > >  protected:
> > > >         std::string logPrefix() const override;
> > > >
> > > > @@ -290,6 +293,8 @@ private:
> > > >
> > > >         Timer watchdog_;
> > > >         utils::Duration watchdogDuration_;
> > > > +
> > > > +       ClockRecovery *wallClockRecovery_;
> > > >  };
> > > >
> > > >  class V4L2M2MDevice
> > > > diff --git a/src/libcamera/v4l2_videodevice.cpp b/src/libcamera/v4l2_videodevice.cpp
> > > > index a5cf6784..2007dffc 100644
> > > > --- a/src/libcamera/v4l2_videodevice.cpp
> > > > +++ b/src/libcamera/v4l2_videodevice.cpp
> > > > @@ -26,6 +26,7 @@
> > > >  #include <libcamera/base/unique_fd.h>
> > > >  #include <libcamera/base/utils.h>
> > > >
> > > > +#include "libcamera/internal/clock_recovery.h"
> > > >  #include "libcamera/internal/formats.h"
> > > >  #include "libcamera/internal/framebuffer.h"
> > > >  #include "libcamera/internal/media_device.h"
> > > > @@ -534,7 +535,7 @@ std::ostream &operator<<(std::ostream &out, const V4L2DeviceFormat &f)
> > > >  V4L2VideoDevice::V4L2VideoDevice(const std::string &deviceNode)
> > > >         : V4L2Device(deviceNode), formatInfo_(nullptr), cache_(nullptr),
> > > >           fdBufferNotifier_(nullptr), state_(State::Stopped),
> > > > -         watchdogDuration_(0.0)
> > > > +         watchdogDuration_(0.0), wallClockRecovery_(nullptr)
> > > >  {
> > > >         /*
> > > >          * We default to an MMAP based CAPTURE video device, however this will
> > > > @@ -1865,6 +1866,17 @@ FrameBuffer *V4L2VideoDevice::dequeueBuffer()
> > > >         metadata.timestamp = buf.timestamp.tv_sec * 1000000000ULL
> > > >                            + buf.timestamp.tv_usec * 1000ULL;
> > > >
> > > > +       metadata.wallClock = 0;
> > > > +       if (wallClockRecovery_) {
> > >
> > > I know Laurent discussed passing in the 'ClockRecovery' class as a way
> > > of giving the responsibility to the Pipeline handler - but I'm not
> > > particularly fond of it. (I won't object if this is preferred all
> > > round).
> > >
> > > To me - especially if there is an addition of uint64_t wallClock in the
> > > FrameMetadata, then this should just always be enabled.
> > >
> > > But if it's /really/ optional and passed in, - then I don't think we
> > > should store a metadata.wallClock below.
> >
> > Indeed, I don't actually feel terribly strongly either, I'd obviously
> > prefer to reach a consensus  ahead of time rather than produce
> > different implementations that we then don't like!
> >
> > I'd certainly be happy always to enable wall clocks, it depends
> > whether we think the burden of a bit of floating point arithmetic
> > every time we add a sample is a problem. Probably not.
> >
> > To be fair, I do wonder a little bit whether a single ClockRecovery
> > for the entire system might be the right thing to do. Though I'm not
> > sure where it would go, or how you would add sporadic samples to it.
> > But you could easily rate-limit the model updates, if you wanted to.
> >
> > On the downside, that would certainly make it more difficult for
> > anyone to configure it differently, but maybe no one would really need
> > to? Questions, questions...
> >
> > > > +               /*
> > > > +                * Sample the internal (CLOCK_BOOTTIME) and realtime (CLOCK_REALTIME) clocks and
> > > > +                * update the clock recovery model. Then derive the wallclock estimate for the
> > > > +                * frame timestamp.
> > > > +                */
> > > > +               wallClockRecovery_->addSample();
> > > > +               metadata.wallClock = wallClockRecovery_->getOutput(metadata.timestamp / 1000);
> > >
> > > As that could be called in the pipeline handler. And to me - then I
> > > would ask why are we getting the V4L2VideoDevice to manage the
> > > clockRecovery object..
> > >
> > > (And I think that it should 'just' be in the V4L2VideoDevice in this
> > > instance if we're going to provide wall clock timestamps of buffers).
> >
> > I would agree that I think I'm against moving more of it into the PH.
> > After all, the whole point was to put it here so that everyone can get
> > the new feature easily. But I don't super-love the
> > pass-the-ClockRecovery-in either. I can certainly see that argument
> > for a single "global" ClockRecovery. I dunno...
> 
> I tend to agree with Kieran that perhaps the wall clock recovery
> becomes a standard feature of the V4L2VideoDevice and the framebuffer
> metadata.wallClock field will be unconditionally valid.  The amount of
> additional computation per frame (per device) is quite tiny, and I
> doubt it would make any measurable impact.  As such I would suggest we
> move the instantiation of the ClockRecovery object into
> V4L2VideoDevice.
> 
> As for specifying custom clock recovery config, perhaps we just ignore
> this problem (if it is indeed a problem) until we have a definitive
> need to change it.  Since this is all in libcamera internals, we have
> some leeway with changing this in the future.
> 
> Any objections to this?
> 
> > > > +       }
> > > > +
> > > >         if (V4L2_TYPE_IS_OUTPUT(buf.type))
> > > >                 return buffer;
> > > >
> > > > @@ -1965,6 +1977,11 @@ int V4L2VideoDevice::streamOn()
> > > >                 return ret;
> > > >         }
> > > >
> > > > +       if (wallClockRecovery_) {
> > > > +               /* A good moment to sample the clocks to improve the clock recovery model. */
> > > > +               wallClockRecovery_->addSample();
> > > > +       }
> > > > +
> > > >         state_ = State::Streaming;
> > > >         if (watchdogDuration_ && !queuedBuffers_.empty())
> > > >                 watchdog_.start(std::chrono::duration_cast<std::chrono::milliseconds>(watchdogDuration_));
> > > > @@ -2120,6 +2137,23 @@ V4L2PixelFormat V4L2VideoDevice::toV4L2PixelFormat(const PixelFormat &pixelForma
> > > >         return {};
> > > >  }
> > > >
> > > > +/**
> > > > + * \brief Enable wall clock timestamps for this device
> > > > + * \param[in] wallClockRecovery an appropriately configured ClockRecovery, or
> > > > + * nullptr to disable wall clocks
> > >
> > > Is there a reason we can think of for /why/ we would disable the wall
> > > clocks?
> > >
> > > It doesn't seem like a 'desired' feature to me - unless the costs of
> > > doing the clock recovery are prohibitively expensive? Which I wouldn't
> > > expect it to be ?
> >
> > Yeah, no reason... just no compelling reason not to allow it either.
> > Having them "always enabled" feels nice in many ways...

Isn't it just a waste of CPU cycles on the video devices that correspond
to the input and output of the ISP, or to the embedded data capture ?
That's why I don't think this belongs to the V4L2VideoDevice class. The
need is to timestamp requests, not frame buffers individually. I'd
rather try to make this simple to wire up in pipeline handlers.

> > > > + *
> > > > + * When buffers are dequeued, wall clock timestamps will be generated that
> > > > + * correspond to the frame's timestamp, as returned by V4l2.
> > > > + */
> > > > +void V4L2VideoDevice::enableWallClock(ClockRecovery *wallClockRecovery)
> > > > +{
> > > > +       wallClockRecovery_ = wallClockRecovery;
> > > > +
> > > > +       /* Also a reasonable moment to sample the two clocks. */
> > > > +       if (wallClockRecovery_)
> > > > +               wallClockRecovery_->addSample();
> > > > +}
> > > > +
> > > >  /**
> > > >   * \class V4L2M2MDevice
> > > >   * \brief Memory-to-Memory video device
David Plowman Dec. 18, 2024, 10:19 a.m. UTC | #5
Hi Laurent

Thanks for the comments!

On Wed, 18 Dec 2024 at 02:35, Laurent Pinchart
<laurent.pinchart@ideasonboard.com> wrote:
>
> On Thu, Dec 12, 2024 at 01:04:11PM +0000, Naushir Patuck wrote:
> > On Fri, 6 Dec 2024 at 16:53, David Plowman wrote:
> > > On Fri, 6 Dec 2024 at 15:37, Kieran Bingham wrote:
> > > > Quoting David Plowman (2024-12-06 14:27:40)
> > > > > FrameMetadata is extended to include a wallclock timestamp.
> > > > >
> > > > > When a frame is dequeued, we use the ClockRecovery class to generate a
> > > > > good estimate of a wallclock timestamp to correspond to the frame's
> > > > > SensorTimestamp.
> > > > >
> > > > > Pipeline handlers must enable wallclocks for chosen devices by passing
> > > > > an appropriate ClockRecovery object.
> > > > >
> > > > > Signed-off-by: David Plowman <david.plowman@raspberrypi.com>
> > > > > ---
> > > > >  include/libcamera/framebuffer.h               |  1 +
> > > > >  include/libcamera/internal/v4l2_videodevice.h |  5 +++
> > > > >  src/libcamera/v4l2_videodevice.cpp            | 36 ++++++++++++++++++-
> > > > >  3 files changed, 41 insertions(+), 1 deletion(-)
> > > > >
> > > > > diff --git a/include/libcamera/framebuffer.h b/include/libcamera/framebuffer.h
> > > > > index ff839243..4579d0c6 100644
> > > > > --- a/include/libcamera/framebuffer.h
> > > > > +++ b/include/libcamera/framebuffer.h
> > > > > @@ -35,6 +35,7 @@ struct FrameMetadata {
> > > > >         Status status;
> > > > >         unsigned int sequence;
> > > > >         uint64_t timestamp;
> > > > > +       uint64_t wallClock;
> > > >
> > > > This breaks in the CI build I'm afraid:
> > > >
> > > > https://gitlab.freedesktop.org/camera/libcamera/-/jobs/67848569
> > > >
> > > >
> > > > [481/650] Generating Documentation/doxygen-internal with a custom command
> > > > FAILED: Documentation/internal-api-html
> > > > /usr/bin/doxygen Documentation/Doxyfile-internal
> > > > /builds/camera/libcamera/include/libcamera/framebuffer.h:38: error: Member wallClock (variable) of struct libcamera::FrameMetadata is not documented. (warning treated as error, aborting now)
> > > >
> > > > Either we need to add the doc (which is what I'd do) - or potentially
> > > > depending on discussions below, remove the addition...
> > >
> > > Yep, well I think this is a bit above my pay grade. The "wall clock"
> > > sure feels like it belongs with the regular timestamp. If that's a
> > > problem, I'm happy to put it somewhere else, though I don't think I
> > > would know where currently. Perhaps some others can offer an opinion
> > > here?
>
> Even if I'm still considering whether or not we should remove timestamp
> from FrameMetadata, I think it makes sense to have either none or both,
> so it's fine to keep it here. What's missing is documentation.
>
> > > > >
> > > > >         Span<Plane> planes() { return planes_; }
> > > > >         Span<const Plane> planes() const { return planes_; }
> > > > > diff --git a/include/libcamera/internal/v4l2_videodevice.h b/include/libcamera/internal/v4l2_videodevice.h
> > > > > index f021c2a0..5327c112 100644
> > > > > --- a/include/libcamera/internal/v4l2_videodevice.h
> > > > > +++ b/include/libcamera/internal/v4l2_videodevice.h
> > > > > @@ -37,6 +37,7 @@
> > > > >
> > > > >  namespace libcamera {
> > > > >
> > > > > +class ClockRecovery;
> > > > >  class EventNotifier;
> > > > >  class MediaDevice;
> > > > >  class MediaEntity;
> > > > > @@ -232,6 +233,8 @@ public:
> > > > >
> > > > >         V4L2PixelFormat toV4L2PixelFormat(const PixelFormat &pixelFormat) const;
> > > > >
> > > > > +       void enableWallClock(ClockRecovery *wallClockRecovery);
> > > > > +
> > > > >  protected:
> > > > >         std::string logPrefix() const override;
> > > > >
> > > > > @@ -290,6 +293,8 @@ private:
> > > > >
> > > > >         Timer watchdog_;
> > > > >         utils::Duration watchdogDuration_;
> > > > > +
> > > > > +       ClockRecovery *wallClockRecovery_;
> > > > >  };
> > > > >
> > > > >  class V4L2M2MDevice
> > > > > diff --git a/src/libcamera/v4l2_videodevice.cpp b/src/libcamera/v4l2_videodevice.cpp
> > > > > index a5cf6784..2007dffc 100644
> > > > > --- a/src/libcamera/v4l2_videodevice.cpp
> > > > > +++ b/src/libcamera/v4l2_videodevice.cpp
> > > > > @@ -26,6 +26,7 @@
> > > > >  #include <libcamera/base/unique_fd.h>
> > > > >  #include <libcamera/base/utils.h>
> > > > >
> > > > > +#include "libcamera/internal/clock_recovery.h"
> > > > >  #include "libcamera/internal/formats.h"
> > > > >  #include "libcamera/internal/framebuffer.h"
> > > > >  #include "libcamera/internal/media_device.h"
> > > > > @@ -534,7 +535,7 @@ std::ostream &operator<<(std::ostream &out, const V4L2DeviceFormat &f)
> > > > >  V4L2VideoDevice::V4L2VideoDevice(const std::string &deviceNode)
> > > > >         : V4L2Device(deviceNode), formatInfo_(nullptr), cache_(nullptr),
> > > > >           fdBufferNotifier_(nullptr), state_(State::Stopped),
> > > > > -         watchdogDuration_(0.0)
> > > > > +         watchdogDuration_(0.0), wallClockRecovery_(nullptr)
> > > > >  {
> > > > >         /*
> > > > >          * We default to an MMAP based CAPTURE video device, however this will
> > > > > @@ -1865,6 +1866,17 @@ FrameBuffer *V4L2VideoDevice::dequeueBuffer()
> > > > >         metadata.timestamp = buf.timestamp.tv_sec * 1000000000ULL
> > > > >                            + buf.timestamp.tv_usec * 1000ULL;
> > > > >
> > > > > +       metadata.wallClock = 0;
> > > > > +       if (wallClockRecovery_) {
> > > >
> > > > I know Laurent discussed passing in the 'ClockRecovery' class as a way
> > > > of giving the responsibility to the Pipeline handler - but I'm not
> > > > particularly fond of it. (I won't object if this is preferred all
> > > > round).
> > > >
> > > > To me - especially if there is an addition of uint64_t wallClock in the
> > > > FrameMetadata, then this should just always be enabled.
> > > >
> > > > But if it's /really/ optional and passed in, - then I don't think we
> > > > should store a metadata.wallClock below.
> > >
> > > Indeed, I don't actually feel terribly strongly either, I'd obviously
> > > prefer to reach a consensus  ahead of time rather than produce
> > > different implementations that we then don't like!
> > >
> > > I'd certainly be happy always to enable wall clocks, it depends
> > > whether we think the burden of a bit of floating point arithmetic
> > > every time we add a sample is a problem. Probably not.
> > >
> > > To be fair, I do wonder a little bit whether a single ClockRecovery
> > > for the entire system might be the right thing to do. Though I'm not
> > > sure where it would go, or how you would add sporadic samples to it.
> > > But you could easily rate-limit the model updates, if you wanted to.
> > >
> > > On the downside, that would certainly make it more difficult for
> > > anyone to configure it differently, but maybe no one would really need
> > > to? Questions, questions...
> > >
> > > > > +               /*
> > > > > +                * Sample the internal (CLOCK_BOOTTIME) and realtime (CLOCK_REALTIME) clocks and
> > > > > +                * update the clock recovery model. Then derive the wallclock estimate for the
> > > > > +                * frame timestamp.
> > > > > +                */
> > > > > +               wallClockRecovery_->addSample();
> > > > > +               metadata.wallClock = wallClockRecovery_->getOutput(metadata.timestamp / 1000);
> > > >
> > > > As that could be called in the pipeline handler. And to me - then I
> > > > would ask why are we getting the V4L2VideoDevice to manage the
> > > > clockRecovery object..
> > > >
> > > > (And I think that it should 'just' be in the V4L2VideoDevice in this
> > > > instance if we're going to provide wall clock timestamps of buffers).
> > >
> > > I would agree that I think I'm against moving more of it into the PH.
> > > After all, the whole point was to put it here so that everyone can get
> > > the new feature easily. But I don't super-love the
> > > pass-the-ClockRecovery-in either. I can certainly see that argument
> > > for a single "global" ClockRecovery. I dunno...
> >
> > I tend to agree with Kieran that perhaps the wall clock recovery
> > becomes a standard feature of the V4L2VideoDevice and the framebuffer
> > metadata.wallClock field will be unconditionally valid.  The amount of
> > additional computation per frame (per device) is quite tiny, and I
> > doubt it would make any measurable impact.  As such I would suggest we
> > move the instantiation of the ClockRecovery object into
> > V4L2VideoDevice.
> >
> > As for specifying custom clock recovery config, perhaps we just ignore
> > this problem (if it is indeed a problem) until we have a definitive
> > need to change it.  Since this is all in libcamera internals, we have
> > some leeway with changing this in the future.
> >
> > Any objections to this?
> >
> > > > > +       }
> > > > > +
> > > > >         if (V4L2_TYPE_IS_OUTPUT(buf.type))
> > > > >                 return buffer;
> > > > >
> > > > > @@ -1965,6 +1977,11 @@ int V4L2VideoDevice::streamOn()
> > > > >                 return ret;
> > > > >         }
> > > > >
> > > > > +       if (wallClockRecovery_) {
> > > > > +               /* A good moment to sample the clocks to improve the clock recovery model. */
> > > > > +               wallClockRecovery_->addSample();
> > > > > +       }
> > > > > +
> > > > >         state_ = State::Streaming;
> > > > >         if (watchdogDuration_ && !queuedBuffers_.empty())
> > > > >                 watchdog_.start(std::chrono::duration_cast<std::chrono::milliseconds>(watchdogDuration_));
> > > > > @@ -2120,6 +2137,23 @@ V4L2PixelFormat V4L2VideoDevice::toV4L2PixelFormat(const PixelFormat &pixelForma
> > > > >         return {};
> > > > >  }
> > > > >
> > > > > +/**
> > > > > + * \brief Enable wall clock timestamps for this device
> > > > > + * \param[in] wallClockRecovery an appropriately configured ClockRecovery, or
> > > > > + * nullptr to disable wall clocks
> > > >
> > > > Is there a reason we can think of for /why/ we would disable the wall
> > > > clocks?
> > > >
> > > > It doesn't seem like a 'desired' feature to me - unless the costs of
> > > > doing the clock recovery are prohibitively expensive? Which I wouldn't
> > > > expect it to be ?
> > >
> > > Yeah, no reason... just no compelling reason not to allow it either.
> > > Having them "always enabled" feels nice in many ways...
>
> Isn't it just a waste of CPU cycles on the video devices that correspond
> to the input and output of the ISP, or to the embedded data capture ?
> That's why I don't think this belongs to the V4L2VideoDevice class. The
> need is to timestamp requests, not frame buffers individually. I'd
> rather try to make this simple to wire up in pipeline handlers.

Just to be sure I've understood here, what you're saying is that you'd
rather have this handled completely in the pipeline handlers, and have
nothing at all in the V4L2VideoDevice? I'm fine with that, just wanted
to clear. At that point, it doesn't feel like there's any need to have
it in the metadata either, and it can just be a control. Does that
sound fair?

Thanks!
David

>
> > > > > + *
> > > > > + * When buffers are dequeued, wall clock timestamps will be generated that
> > > > > + * correspond to the frame's timestamp, as returned by V4l2.
> > > > > + */
> > > > > +void V4L2VideoDevice::enableWallClock(ClockRecovery *wallClockRecovery)
> > > > > +{
> > > > > +       wallClockRecovery_ = wallClockRecovery;
> > > > > +
> > > > > +       /* Also a reasonable moment to sample the two clocks. */
> > > > > +       if (wallClockRecovery_)
> > > > > +               wallClockRecovery_->addSample();
> > > > > +}
> > > > > +
> > > > >  /**
> > > > >   * \class V4L2M2MDevice
> > > > >   * \brief Memory-to-Memory video device
>
> --
> Regards,
>
> Laurent Pinchart

Patch
diff mbox series

diff --git a/include/libcamera/framebuffer.h b/include/libcamera/framebuffer.h
index ff839243..4579d0c6 100644
--- a/include/libcamera/framebuffer.h
+++ b/include/libcamera/framebuffer.h
@@ -35,6 +35,7 @@  struct FrameMetadata {
 	Status status;
 	unsigned int sequence;
 	uint64_t timestamp;
+	uint64_t wallClock;
 
 	Span<Plane> planes() { return planes_; }
 	Span<const Plane> planes() const { return planes_; }
diff --git a/include/libcamera/internal/v4l2_videodevice.h b/include/libcamera/internal/v4l2_videodevice.h
index f021c2a0..5327c112 100644
--- a/include/libcamera/internal/v4l2_videodevice.h
+++ b/include/libcamera/internal/v4l2_videodevice.h
@@ -37,6 +37,7 @@ 
 
 namespace libcamera {
 
+class ClockRecovery;
 class EventNotifier;
 class MediaDevice;
 class MediaEntity;
@@ -232,6 +233,8 @@  public:
 
 	V4L2PixelFormat toV4L2PixelFormat(const PixelFormat &pixelFormat) const;
 
+	void enableWallClock(ClockRecovery *wallClockRecovery);
+
 protected:
 	std::string logPrefix() const override;
 
@@ -290,6 +293,8 @@  private:
 
 	Timer watchdog_;
 	utils::Duration watchdogDuration_;
+
+	ClockRecovery *wallClockRecovery_;
 };
 
 class V4L2M2MDevice
diff --git a/src/libcamera/v4l2_videodevice.cpp b/src/libcamera/v4l2_videodevice.cpp
index a5cf6784..2007dffc 100644
--- a/src/libcamera/v4l2_videodevice.cpp
+++ b/src/libcamera/v4l2_videodevice.cpp
@@ -26,6 +26,7 @@ 
 #include <libcamera/base/unique_fd.h>
 #include <libcamera/base/utils.h>
 
+#include "libcamera/internal/clock_recovery.h"
 #include "libcamera/internal/formats.h"
 #include "libcamera/internal/framebuffer.h"
 #include "libcamera/internal/media_device.h"
@@ -534,7 +535,7 @@  std::ostream &operator<<(std::ostream &out, const V4L2DeviceFormat &f)
 V4L2VideoDevice::V4L2VideoDevice(const std::string &deviceNode)
 	: V4L2Device(deviceNode), formatInfo_(nullptr), cache_(nullptr),
 	  fdBufferNotifier_(nullptr), state_(State::Stopped),
-	  watchdogDuration_(0.0)
+	  watchdogDuration_(0.0), wallClockRecovery_(nullptr)
 {
 	/*
 	 * We default to an MMAP based CAPTURE video device, however this will
@@ -1865,6 +1866,17 @@  FrameBuffer *V4L2VideoDevice::dequeueBuffer()
 	metadata.timestamp = buf.timestamp.tv_sec * 1000000000ULL
 			   + buf.timestamp.tv_usec * 1000ULL;
 
+	metadata.wallClock = 0;
+	if (wallClockRecovery_) {
+		/*
+		 * Sample the internal (CLOCK_BOOTTIME) and realtime (CLOCK_REALTIME) clocks and
+		 * update the clock recovery model. Then derive the wallclock estimate for the
+		 * frame timestamp.
+		 */
+		wallClockRecovery_->addSample();
+		metadata.wallClock = wallClockRecovery_->getOutput(metadata.timestamp / 1000);
+	}
+
 	if (V4L2_TYPE_IS_OUTPUT(buf.type))
 		return buffer;
 
@@ -1965,6 +1977,11 @@  int V4L2VideoDevice::streamOn()
 		return ret;
 	}
 
+	if (wallClockRecovery_) {
+		/* A good moment to sample the clocks to improve the clock recovery model. */
+		wallClockRecovery_->addSample();
+	}
+
 	state_ = State::Streaming;
 	if (watchdogDuration_ && !queuedBuffers_.empty())
 		watchdog_.start(std::chrono::duration_cast<std::chrono::milliseconds>(watchdogDuration_));
@@ -2120,6 +2137,23 @@  V4L2PixelFormat V4L2VideoDevice::toV4L2PixelFormat(const PixelFormat &pixelForma
 	return {};
 }
 
+/**
+ * \brief Enable wall clock timestamps for this device
+ * \param[in] wallClockRecovery an appropriately configured ClockRecovery, or
+ * nullptr to disable wall clocks
+ *
+ * When buffers are dequeued, wall clock timestamps will be generated that
+ * correspond to the frame's timestamp, as returned by V4l2.
+ */
+void V4L2VideoDevice::enableWallClock(ClockRecovery *wallClockRecovery)
+{
+	wallClockRecovery_ = wallClockRecovery;
+
+	/* Also a reasonable moment to sample the two clocks. */
+	if (wallClockRecovery_)
+		wallClockRecovery_->addSample();
+}
+
 /**
  * \class V4L2M2MDevice
  * \brief Memory-to-Memory video device