[libcamera-devel] [PATCH v5 04/10] ipa: Add common contrast based AF implementation

Jacopo Mondi jacopo.mondi at ideasonboard.com
Tue Mar 28 09:07:22 CEST 2023


Hi Daniel
G
On Tue, Mar 28, 2023 at 07:42:25AM +0200, Daniel Semkowicz wrote:
> Hi Jacopo,
>
> On Mon, Mar 27, 2023 at 3:03 PM Jacopo Mondi
> <jacopo.mondi at ideasonboard.com> wrote:
> >
> > Hi Daniel
> >
> > On Mon, Mar 27, 2023 at 02:29:45PM +0200, Daniel Semkowicz wrote:
> > > Hi Jacopo,
> > >
> > > On Fri, Mar 24, 2023 at 4:30 PM Jacopo Mondi
> > > <jacopo.mondi at ideasonboard.com> wrote:
> > > >
> > > > Hi Daniel
> > > >
> > > > On Fri, Mar 24, 2023 at 03:29:02PM +0100, Daniel Semkowicz via libcamera-devel wrote:
> > > > > Create a new class with contrast based Auto Focus implementation
> > > > > using hill climbing algorithm. This common implementation is independent
> > > > > of platform specific code. This way, each platform can just implement
> > > > > contrast calculation and run the AF control loop basing on this class.
> > > > > This implementation is based on the code that was common for IPU3
> > > > > and RPi AF algorithms.
> > > > >
> > > > > Signed-off-by: Daniel Semkowicz <dse at thaumatec.com>
> > > > > ---
> > > > >  .../libipa/algorithms/af_hill_climbing.cpp    | 379 ++++++++++++++++++
> > > > >  src/ipa/libipa/algorithms/af_hill_climbing.h  |  91 +++++
> > > > >  src/ipa/libipa/algorithms/meson.build         |   2 +
> > > > >  3 files changed, 472 insertions(+)
> > > > >  create mode 100644 src/ipa/libipa/algorithms/af_hill_climbing.cpp
> > > > >  create mode 100644 src/ipa/libipa/algorithms/af_hill_climbing.h
> > > > >
> > > > > diff --git a/src/ipa/libipa/algorithms/af_hill_climbing.cpp b/src/ipa/libipa/algorithms/af_hill_climbing.cpp
> > > > > new file mode 100644
> > > > > index 00000000..6ee090eb
> > > > > --- /dev/null
> > > > > +++ b/src/ipa/libipa/algorithms/af_hill_climbing.cpp
> > > > > @@ -0,0 +1,379 @@
> > > > > +/* SPDX-License-Identifier: LGPL-2.1-or-later */
> > > > > +/*
> > > > > + * Copyright (C) 2021, Red Hat
> > > > > + * Copyright (C) 2022, Ideas On Board
> > > > > + * Copyright (C) 2023, Theobroma Systems
> > > > > + *
> > > > > + * af_hill_climbing.cpp - AF contrast based hill climbing common algorithm
> > > > > + */
> > > > > +
> > > > > +#include "af_hill_climbing.h"
> > > > > +
> > > > > +#include "libcamera/internal/yaml_parser.h"
> > > > > +
> > > > > +/**
> > > > > + * \file af_hill_climbing.h
> > > > > + * \brief AF contrast based hill climbing common algorithm
> > > > > + */
> > > > > +
> > > > > +namespace libcamera {
> > > > > +
> > > > > +namespace ipa::algorithms {
> > > > > +
> > > > > +LOG_DEFINE_CATEGORY(Af)
> > > > > +
> > > > > +/**
> > > > > + * \class AfHillClimbing
> > > > > + * \brief Contrast based hill climbing auto focus control algorithm
> > > > > + * implementation
> > > > > + *
> > > > > + * Control part of the auto focus algorithm. It calculates the lens position
> > > > > + * based on the contrast measure supplied by the platform-specific
> > > > > + * implementation. This way it is independent from the platform.
> > > > > + *
> > > > > + * Platform layer that use this algorithm should call process() function
> > > > > + * for each each frame and set the lens to the calculated position.
> > > > > + *
> > > > > + * Focus search is divided into two phases:
> > > > > + * 1. coarse search,
> > > > > + * 2. fine search.
> > > > > + *
> > > > > + * In the first phase, the lens is moved with bigger steps to quickly find
> > > > > + * a rough position for the best focus. Then, based on the outcome of coarse
> > > > > + * search, the second phase is executed. Lens is moved with smaller steps
> > > > > + * in a limited range within the rough position to find the exact position
> > > > > + * for best focus.
> > > > > + *
> > > > > + * Tuning file parameters:
> > > > > + * - **coarse-search-step:** The value by which the lens position will change
> > > > > + *   in one step in the *coarse search* phase. Unit is lens specific.
> > > > > + * - **fine-search-step:** The value by which the lens position will change
> > > > > + *   in one step in the *fine search* phase. Unit is lens specific.
> > > > > + * - **fine-search-range:** Search range in the *fine search* phase, expressed
> > > > > + *   as a percentage of the coarse search result. Valid values are
> > > > > + *   in the [0, 100] interval. Value 5 means 5%. If coarse search stopped
> > > > > + *   at value 200, the fine search range will be [190, 210].
> > > > > + * - **max-variance-change:** ratio of contrast variance change in the
> > > > > + *   *continuous mode* needed for triggering the focus change. When the variance
> > > > > + *   change exceeds this value, focus search will be triggered. Valid values are
> > > > > + *   in the [0.0, 1.0] interval.
> > > > > + * .
> > > > > + *
> > > > > + * \todo Search range in the *fine search* phase should depend on the lens
> > > > > + *   movement range rather than coarse search result.
> > > > > + * \todo Implement setRange.
> > > > > + * \todo Implement setSpeed.
> > > > > + * \todo Implement setMeteringMode.
> > > > > + * \todo Implement setWindows.
> > > > > + * \todo Implement the AfPauseDeferred mode.
> > > > > + */
> > > > > +
> > > > > +/**
> > > > > + * \brief Initialize the AfHillClimbing with tuning data
> > > > > + * \param[in] tuningData The tuning data for the algorithm
> > > > > + *
> > > > > + * This method should be called in the libcamera::ipa::Algorithm::init()
> > > > > + * method of the platform layer.
> > > > > + *
> > > > > + * \return 0 if successful, an error code otherwise
> > > > > + */
> > > > > +int AfHillClimbing::init(const YamlObject &tuningData)
> > > > > +{
> > > > > +     coarseSearchStep_ = tuningData["coarse-search-step"].get<uint32_t>(30);
> > > > > +     fineSearchStep_ = tuningData["fine-search-step"].get<uint32_t>(1);
> > > > > +     fineRange_ = tuningData["fine-search-range"].get<uint32_t>(5);
> > > > > +     fineRange_ /= 100;
> > > > > +     maxChange_ = tuningData["max-variance-change"].get<double>(0.5);
> > > > > +
> > > > > +     LOG(Af, Debug) << "coarseSearchStep_: " << coarseSearchStep_
> > > > > +                    << ", fineSearchStep_: " << fineSearchStep_
> > > > > +                    << ", fineRange_: " << fineRange_
> > > > > +                    << ", maxChange_: " << maxChange_;
> > > > > +
> > > > > +     return 0;
> > > > > +}
> > > > > +
> > > > > +/**
> > > > > + * \brief Configure the AfHillClimbing with sensor and lens information
> > > > > + * \param[in] minFocusPosition Minimum position supported by camera lens
> > > > > + * \param[in] maxFocusPosition Maximum position supported by camera lens
> > > > > + *
> > > > > + * This method should be called in the libcamera::ipa::Algorithm::configure()
> > > > > + * method of the platform layer.
> > > > > + */
> > > > > +void AfHillClimbing::configure(int32_t minFocusPosition,
> > > > > +                            int32_t maxFocusPosition)
> > > > > +{
> > > > > +     minLensPosition_ = minFocusPosition;
> > > > > +     maxLensPosition_ = maxFocusPosition;
> > > > > +}
> > > > > +
> > > > > +/**
> > > > > + * \brief Run the auto focus algorithm loop
> > > > > + * \param[in] currentContrast New value of contrast measured for current frame
> > > > > + *
> > > > > + * This method should be called in the libcamera::ipa::Algorithm::process()
> > > > > + * method of the platform layer for each frame.
> > > > > + *
> > > > > + * Contrast value supplied in the \p currentContrast parameter can be platform
> > > > > + * specific. The only requirement is the contrast value must increase with
> > > > > + * the increasing image focus. Contrast value must be highest when image is in
> > > > > + * focus.
> > > > > + *
> > > > > + * \return New lens position calculated by the AF algorithm
> > > > > + */
> > > > > +int32_t AfHillClimbing::process(double currentContrast)
> > > > > +{
> > > > > +     currentContrast_ = currentContrast;
> > > > > +
> > > > > +     if (shouldSkipFrame())
> > > > > +             return lensPosition_;
> > > > > +
> > > > > +     switch (mode_) {
> > > > > +     case controls::AfModeManual:
> > > > > +             /* Nothing to process. */
> > > > > +             break;
> > > > > +     case controls::AfModeAuto:
> > > > > +             processAutoMode();
> > > > > +             break;
> > > > > +     case controls::AfModeContinuous:
> > > > > +             processContinuousMode();
> > > > > +             break;
> > > > > +     }
> > > > > +
> > > > > +     return lensPosition_;
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::processAutoMode()
> > > > > +{
> > > > > +     if (state_ == controls::AfStateScanning) {
> > > > > +             afCoarseScan();
> > > > > +             afFineScan();
> > > > > +     }
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::processContinuousMode()
> > > > > +{
> > > > > +     /* If we are in a paused state, we won't process the stats. */
> > > > > +     if (pauseState_ == controls::AfPauseStatePaused)
> > > > > +             return;
> > > > > +
> > > > > +     if (state_ == controls::AfStateScanning) {
> > > > > +             afCoarseScan();
> > > > > +             afFineScan();
> > > > > +             return;
> > > > > +     }
> > > > > +
> > > > > +     /*
> > > > > +      * AF scan can be started at any moment in AfModeContinuous,
> > > > > +      * except when the state is already AfStateScanning.
> > > > > +      */
> > > > > +     if (afIsOutOfFocus())
> > > > > +             afReset();
> > > > > +}
> > > > > +
> > > > > +/**
> > > > > + * \brief Request AF to skip n frames
> > > > > + * \param[in] n Number of frames to be skipped
> > > > > + *
> > > > > + * For the requested number of frames, the AF calculation will be skipped
> > > > > + * and lens position will not change. The platform layer still needs to
> > > > > + * call process() function for each frame during this time.
> > > > > + * This function can be used by the platform layer if the hardware needs more
> > > > > + * time for some operations.
> > > > > + *
> > > > > + * The number of the requested frames (\p n) will be applied only if \p n has
> > > > > + * higher value than the number of frames already requested to be skipped.
> > > > > + * For example, if *setFramesToSkip(5)* was already called for the current
> > > > > + * frame, then calling *setFramesToSkip(3)* will not override the previous
> > > > > + * request and 5 frames will be skipped.
> > > >
> > > > This still puzzles me a bit as it is not clear to me when the platform
> > > > layer is supposed to set the number of frames to skip.
> > > >
> > > > Looking at the implementation it happens everytime the lens is moved.
> > > > I guess this is fine for now, but this will need to be handled better
> > > > by the CameraLens class modeling the lens movement delay.
> > > >
> > >
> > > In RkISP it is also needed when setting AF Windows. ISP needs a one
> > > additional frame to calculate the new statistics after setting window.
> > > This is why I exposed this method as public.
> > >
> > > > > + */
> > > > > +void AfHillClimbing::setFramesToSkip(uint32_t n)
> > > > > +{
> > > > > +     if (n > framesToSkip_)
> > > > > +             framesToSkip_ = n;
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setMode(controls::AfModeEnum mode)
> > > > > +{
> > > > > +     if (mode == mode_)
> > > > > +             return;
> > > > > +
> > > > > +     LOG(Af, Debug) << "Switched AF mode from " << mode_ << " to " << mode;
> > > > > +     mode_ = mode;
> > > > > +
> > > > > +     state_ = controls::AfStateIdle;
> > > > > +     pauseState_ = controls::AfPauseStateRunning;
> > > > > +
> > > > > +     if (mode_ == controls::AfModeContinuous)
> > > > > +             afReset();
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setRange([[maybe_unused]] controls::AfRangeEnum range)
> > > > > +{
> > > > > +     LOG(Af, Error) << __FUNCTION__ << " not implemented!";
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setSpeed([[maybe_unused]] controls::AfSpeedEnum speed)
> > > > > +{
> > > > > +     LOG(Af, Error) << __FUNCTION__ << " not implemented!";
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setMeteringMode([[maybe_unused]] controls::AfMeteringEnum metering)
> > > > > +{
> > > > > +     LOG(Af, Error) << __FUNCTION__ << " not implemented!";
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setWindows([[maybe_unused]] Span<const Rectangle> windows)
> > > > > +{
> > > > > +     LOG(Af, Error) << __FUNCTION__ << " not implemented!";
> > > > > +}
> > > > > +
> > > >
> > > > I'm still of the idea you should better not implement these functions
> > > > at all. If anything tries to call into them will fail at compile time
> > > > instead of getting a run-time error in the logs
> > >
> > > These functions are pure virtual, so you are forced by compiler
> > > to implement them.
> > >
> >
> > Ah! I removed the definition in the cpp file but I left the
> > declaration in the header, so I guess it worked here because of the
> > compiler generated one for me. Sorry for the overlook
> >
> > > >
> > > > > +void AfHillClimbing::setTrigger(controls::AfTriggerEnum trigger)
> > > > > +{
> > > > > +     if (mode_ != controls::AfModeAuto) {
> > > > > +             LOG(Af, Warning)
> > > > > +                     << __FUNCTION__ << " not valid in mode " << mode_;
> > > >
> > > > We don't use __FUNCTION__ for the reason that as far as I know it's
> > > > not a standard language keyword (while __func__ is iirc). I'm also not
> > > > sure how it works with name mangling as I see gcc has a
> > > > __PRETTY_FUNCTION__ specific extensions to beautify names.
> > > >
> > > > I would just
> > > >                         << "setTrigger() not valid in mode " << mode_;
> > > >
> > > > > +             return;
> > > > > +     }
> > > > > +
> > > > > +     LOG(Af, Debug) << "Trigger called with " << trigger;
> > > > > +
> > > > > +     if (trigger == controls::AfTriggerStart)
> > > > > +             afReset();
> > > > > +     else
> > > > > +             state_ = controls::AfStateIdle;
> > > >
> > > > I would switch to be sure to handle all cases. A minor though
> > > >
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setPause(controls::AfPauseEnum pause)
> > > > > +{
> > > > > +     if (mode_ != controls::AfModeContinuous) {
> > > > > +             LOG(Af, Warning)
> > > > > +                     << __FUNCTION__ << " not valid in mode " << mode_;
> > > >
> > > > ditto
> > > >
> > > > > +             return;
> > > > > +     }
> > > > > +
> > > > > +     switch (pause) {
> > > > > +     case controls::AfPauseImmediate:
> > > > > +             pauseState_ = controls::AfPauseStatePaused;
> > > > > +             break;
> > > > > +     case controls::AfPauseDeferred:
> > > > > +             LOG(Af, Warning) << "AfPauseDeferred is not supported!";
> > > > > +             break;
> > > > > +     case controls::AfPauseResume:
> > > > > +             pauseState_ = controls::AfPauseStateRunning;
> > > > > +             break;
> > > > > +     }
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::setLensPosition(float lensPosition)
> > > > > +{
> > > > > +     if (mode_ != controls::AfModeManual) {
> > > > > +             LOG(Af, Warning)
> > > > > +                     << __FUNCTION__ << " not valid in mode " << mode_;
> > > > > +             return;
> > > > > +     }
> > > > > +
> > > > > +     lensPosition_ = static_cast<int32_t>(lensPosition);
> > > > > +
> > > > > +     LOG(Af, Debug) << "Requesting lens position " << lensPosition_;
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::afCoarseScan()
> > > > > +{
> > > > > +     if (coarseCompleted_)
> > > > > +             return;
> > > > > +
> > > > > +     if (afScan(coarseSearchStep_)) {
> > > > > +             coarseCompleted_ = true;
> > > > > +             maxContrast_ = 0;
> > > > > +             const auto diff = static_cast<int32_t>(
> > > > > +                     std::abs(lensPosition_) * fineRange_);
> > > > > +             lensPosition_ = std::max(lensPosition_ - diff, minLensPosition_);
> > > > > +             maxStep_ = std::min(lensPosition_ + diff, maxLensPosition_);
> > > > > +     }
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::afFineScan()
> > > > > +{
> > > > > +     if (!coarseCompleted_)
> > > > > +             return;
> > > > > +
> > > > > +     if (afScan(fineSearchStep_)) {
> > > > > +             LOG(Af, Debug) << "AF found the best focus position!";
> > > > > +             state_ = controls::AfStateFocused;
> > > > > +     }
> > > > > +}
> > > > > +
> > > > > +bool AfHillClimbing::afScan(uint32_t steps)
> > > > > +{
> > > > > +     if (lensPosition_ + static_cast<int32_t>(steps) > maxStep_) {
> > > > > +             /* If the max step is reached, move lens to the position. */
> > > > > +             lensPosition_ = bestPosition_;
> > > > > +             return true;
> > > > > +     } else {
> > > >
> > > > If you return in the if() clause , you can remove else { } and save
> > > > one indendation level
> > >
> > > Good points! I will rework the code :)
> > >
> > > >
> > > > > +             /*
> > > > > +              * Find the maximum of the variance by estimating its
> > > > > +              * derivative. If the direction changes, it means we have passed
> > > > > +              * a maximum one step before.
> > > > > +              */
> > > > > +             if ((currentContrast_ - maxContrast_) >= -(maxContrast_ * 0.1)) {
> > > > > +                     /*
> > > > > +                      * Positive and zero derivative:
> > > > > +                      * The variance is still increasing. The focus could be
> > > > > +                      * increased for the next comparison. Also, the max
> > > > > +                      * variance and previous focus value are updated.
> > > > > +                      */
> > > > > +                     bestPosition_ = lensPosition_;
> > > > > +                     lensPosition_ += static_cast<int32_t>(steps);
> > > > > +                     maxContrast_ = currentContrast_;
> > > > > +             } else {
> > > > > +                     /*
> > > > > +                      * Negative derivative:
> > > > > +                      * The variance starts to decrease which means the maximum
> > > > > +                      * variance is found. Set focus step to previous good one
> > > > > +                      * then return immediately.
> > > > > +                      */
> > > > > +                     lensPosition_ = bestPosition_;
> > > > > +                     return true;
> > > > > +             }
> > > > > +     }
> > > > > +
> > > > > +     LOG(Af, Debug) << "Previous step is " << bestPosition_
> > > > > +                    << ", Current step is " << lensPosition_;
> > > > > +     return false;
> > > > > +}
> > > > > +
> > > > > +void AfHillClimbing::afReset()
> > > > > +{
> > > > > +     LOG(Af, Debug) << "Reset AF parameters";
> > > > > +     lensPosition_ = minLensPosition_;
> > > > > +     maxStep_ = maxLensPosition_;
> > > > > +     state_ = controls::AfStateScanning;
> > > > > +     coarseCompleted_ = false;
> > > > > +     maxContrast_ = 0.0;
> > > > > +     setFramesToSkip(1);
> > > > > +}
> > > > > +
> > > > > +bool AfHillClimbing::afIsOutOfFocus() const
> > > > > +{
> > > > > +     const double diff_var = std::abs(currentContrast_ - maxContrast_);
> > > > > +     const double var_ratio = diff_var / maxContrast_;
> > > > > +     LOG(Af, Debug) << "Variance change rate: " << var_ratio
> > > > > +                    << ", Current lens step: " << lensPosition_;
> > > > > +     return var_ratio > maxChange_;
> > > > > +}
> > > > > +
> > > > > +bool AfHillClimbing::shouldSkipFrame()
> > > > > +{
> > > > > +     if (framesToSkip_ > 0) {
> > > > > +             framesToSkip_--;
> > > > > +             return true;
> > > > > +     }
> > > > > +
> > > > > +     return false;
> > > > > +}
> > > > > +
> > > > > +} /* namespace ipa::algorithms */
> > > > > +
> > > > > +} /* namespace libcamera */
> > > > > diff --git a/src/ipa/libipa/algorithms/af_hill_climbing.h b/src/ipa/libipa/algorithms/af_hill_climbing.h
> > > > > new file mode 100644
> > > > > index 00000000..47d2bbec
> > > > > --- /dev/null
> > > > > +++ b/src/ipa/libipa/algorithms/af_hill_climbing.h
> > > > > @@ -0,0 +1,91 @@
> > > > > +/* SPDX-License-Identifier: LGPL-2.1-or-later */
> > > > > +/*
> > > > > + * Copyright (C) 2021, Red Hat
> > > > > + * Copyright (C) 2022, Ideas On Board
> > > > > + * Copyright (C) 2023, Theobroma Systems
> > > > > + *
> > > > > + * af_hill_climbing.h - AF contrast based hill climbing common algorithm
> > > > > + */
> > > > > +
> > > > > +#pragma once
> > > > > +
> > > > > +#include <libcamera/base/log.h>
> > > > > +
> > > > > +#include "af.h"
> > > > > +
> > > > > +namespace libcamera {
> > > > > +
> > > > > +class YamlObject;
> > > > > +
> > > > > +namespace ipa::algorithms {
> > > > > +
> > > > > +LOG_DECLARE_CATEGORY(Af)
> > > > > +
> > > > > +class AfHillClimbing : public Af
> > > >
> > > > Isn't it better to declare the class final instead of the single
> > > > functions ?
> > >
> > > It is not the same.
> > > When used in a class definition, final specifies
> > > that the class cannot be derived from.
> >
> > Yep
> >
> > > When used in a virtual function declaration or definition, final
> > > specifier ensures that the function is virtual and specifies that it
> > > may not be overridden by derived classes.
> >
> > I almost agree. final when applied to a virtual function ensures that
> > it cannot be overriden in derived classes.
> >
> > The part I don't get in you comment is "final specifier ensures that
> > the function is -virtual-" the rest I agree on.
>
> When "final" is used in a function definition, it implies function
> to be virtual. This is why I said it is a different for class and
> function. But now I see, I probably misunderstood your intention
> and your comment was more about architecture - what should be final.
> Not that class final should replace function finals as I understood it.
>
> >
> > > It is technically possible to derive from AfHillClimbing to build
> > > on it, even if it was meant to be used in composition style.
> > > Do We want to force a composition approach or leave possibility to
> > > derive from it?
> >
> > The thing that doesn't click for me is that you have declared as final
> > the methods that implement the AF state machine, which is agnostic
> > from the HillClimbing algorithm. Is this correct ?
> >
> >         void setMode(controls::AfModeEnum mode) final;
> >         void setRange(controls::AfRangeEnum range) final;
> >         void setSpeed(controls::AfSpeedEnum speed) final;
> >         void setMeteringMode(controls::AfMeteringEnum metering) final;
> >         void setWindows(Span<const Rectangle> windows) final;
> >         void setTrigger(controls::AfTriggerEnum trigger) final;
> >         void setPause(controls::AfPauseEnum pause) final;
> >         void setLensPosition(float lensPosition) final;
> >
> >         void processAutoMode();
> >         void processContinuousMode();
> >         void afCoarseScan();
> >         void afFineScan();
> >         bool afScan(uint32_t steps);
> >         void afReset();
> >         [[nodiscard]] bool afIsOutOfFocus() const;
> >         bool shouldSkipFrame();
> >
> > As I see it, the algorithm state machine method might later be broken
> > out to a common AfBase class (or even making the Af class non-purely
> > virtual) as they do not depend on the contrast-computation method.
> >
> > If one would like to derive from AfHillClimbing I guess it would be to
> > tweak the algorithm's computation, not the state machine which adheres
> > to the libcamera's controls definition. That's why I was suggesting
> > that this fine-grained control on what can be overridden is probably a
> > bit premature.
>
> This is a valid point. Algorithms class hierarchy (and especially this
> one) is in early stage. Then maybe I should drop the final specifiers
> completely in this class? As it is hard to decide the future direction
> at this point?
>

I would say so. I guess it's a tad early to divine how this hierarchy
will be expanded in future.

Thanks
  j

> >
> > >
> > > Best regards
> > > Daniel
> > >
> > > >
> > > > class AfHillClimbing final : public Af
> > > >
> > > > > +{
> > > > > +public:
> > > > > +     int init(const YamlObject &tuningData);
> > > > > +     void configure(int32_t minFocusPosition, int32_t maxFocusPosition);
> > > > > +     int32_t process(double currentContrast);
> > > > > +     void setFramesToSkip(uint32_t n);
> > > > > +
> > > > > +     controls::AfStateEnum state() final { return state_; }
> > > > > +     controls::AfPauseStateEnum pauseState() final { return pauseState_; }
> > > > > +
> > > > > +private:
> > > > > +     void setMode(controls::AfModeEnum mode) final;
> > > > > +     void setRange(controls::AfRangeEnum range) final;
> > > > > +     void setSpeed(controls::AfSpeedEnum speed) final;
> > > > > +     void setMeteringMode(controls::AfMeteringEnum metering) final;
> > > > > +     void setWindows(Span<const Rectangle> windows) final;
> > > > > +     void setTrigger(controls::AfTriggerEnum trigger) final;
> > > > > +     void setPause(controls::AfPauseEnum pause) final;
> > > > > +     void setLensPosition(float lensPosition) final;
> > > >
> > > > These functions should have the 'override' specifier
> > > >
> > > > > +
> > > > > +     void processAutoMode();
> > > > > +     void processContinuousMode();
> > > > > +     void afCoarseScan();
> > > > > +     void afFineScan();
> > > > > +     bool afScan(uint32_t steps);
> > > > > +     void afReset();
> > > > > +     [[nodiscard]] bool afIsOutOfFocus() const;
> > > > > +     bool shouldSkipFrame();
> > > > > +
> > > > > +     controls::AfModeEnum mode_ = controls::AfModeManual;
> > > > > +     controls::AfStateEnum state_ = controls::AfStateIdle;
> > > > > +     controls::AfPauseStateEnum pauseState_ = controls::AfPauseStateRunning;
> > > > > +
> > > > > +     /* Current focus lens position. */
> > > > > +     int32_t lensPosition_ = 0;
> > > > > +     /* Local optimum focus lens position during scanning. */
> > > > > +     int32_t bestPosition_ = 0;
> > > > > +
> > > > > +     /* Current AF statistic contrast. */
> > > > > +     double currentContrast_ = 0;
> > > > > +     /* It is used to determine the derivative during scanning */
> > > > > +     double maxContrast_ = 0;
> > > > > +     /* The designated maximum range of focus scanning. */
> > > > > +     int32_t maxStep_ = 0;
> > > > > +     /* If the coarse scan completes, it is set to true. */
> > > > > +     bool coarseCompleted_ = false;
> > > > > +
> > > > > +     uint32_t framesToSkip_ = 0;
> > > > > +
> > > > > +     /* Position limits of the focus lens. */
> > > > > +     int32_t minLensPosition_;
> > > > > +     int32_t maxLensPosition_;
> > > > > +
> > > > > +     /* Minimum position step for searching appropriate focus. */
> > > > > +     uint32_t coarseSearchStep_;
> > > > > +     uint32_t fineSearchStep_;
> > > > > +
> > > > > +     /* Fine scan range 0 < fineRange_ < 1. */
> > > > > +     double fineRange_;
> > > > > +
> > > > > +     /* Max ratio of variance change, 0.0 < maxChange_ < 1.0. */
> > > > > +     double maxChange_;
> > > > > +};
> > > > > +
> > > > > +} /* namespace ipa::algorithms */
> > > > > +
> > > > > +} /* namespace libcamera */
> > > > > diff --git a/src/ipa/libipa/algorithms/meson.build b/src/ipa/libipa/algorithms/meson.build
> > > > > index 3df4798f..20e437fc 100644
> > > > > --- a/src/ipa/libipa/algorithms/meson.build
> > > > > +++ b/src/ipa/libipa/algorithms/meson.build
> > > > > @@ -2,8 +2,10 @@
> > > > >
> > > > >  libipa_algorithms_headers = files([
> > > > >      'af.h',
> > > > > +    'af_hill_climbing.h',
> > > > >  ])
> > > > >
> > > > >  libipa_algorithms_sources = files([
> > > > >      'af.cpp',
> > > > > +    'af_hill_climbing.cpp',
> > > > >  ])
> > > > > --
> > > > > 2.39.2
> > > > >
>
> Best regards
> Daniel


More information about the libcamera-devel mailing list