From patchwork Tue Mar 15 14:12:37 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Kieran Bingham X-Patchwork-Id: 15450 Return-Path: X-Original-To: parsemail@patchwork.libcamera.org Delivered-To: parsemail@patchwork.libcamera.org Received: from lancelot.ideasonboard.com (lancelot.ideasonboard.com [92.243.16.209]) by patchwork.libcamera.org (Postfix) with ESMTPS id 87526BF415 for ; Tue, 15 Mar 2022 14:12:43 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id E2E75610F8; Tue, 15 Mar 2022 15:12:42 +0100 (CET) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=libcamera.org; s=mail; t=1647353562; bh=8iNiKW1mFGfu3UKjyK7fUtVGexLlpziC2LQmYiOWYrg=; h=To:Date:In-Reply-To:References:Subject:List-Id:List-Unsubscribe: List-Archive:List-Post:List-Help:List-Subscribe:From:Reply-To: From; b=Z02T9Zhi/6bx6sen1UhDd+++qDY2+SSlU22iykspEGRrwAq6k2cQ0RuvFEhABjCOS PcGWyA71gsRlBYXeQteqVRclMYOjTEj2gnX/hmAuwrbPTZU0JedPsWiA8l1IbIRYm2 Liy6hxc2Pgu37iHl34Lj/MOuk6ztReqjJnlJjwp20Ki2BHnUjB57H7rDI0JKJ6kAjg v8XksQ6ZaKRazBxjxlNHT1i2KuIEVVBPANwARNvQ/makme6FE6DHOfufVob+Mx46FC PoPKXi1nfXj5BDdWgv7JP4c8oGwAU1WAx5ektSw8UOuZocNopLloN14DKmKZpsO/G0 97ZAcd873afUw== Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [213.167.242.64]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id 88A3F604DA for ; Tue, 15 Mar 2022 15:12:41 +0100 (CET) Authentication-Results: lancelot.ideasonboard.com; dkim=pass (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="N9oXk6+j"; dkim-atps=neutral Received: from Monstersaurus.ksquared.org.uk.beta.tailscale.net (cpc89244-aztw30-2-0-cust3082.18-1.cable.virginm.net [86.31.172.11]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id 16346EE; Tue, 15 Mar 2022 15:12:41 +0100 (CET) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1647353561; bh=8iNiKW1mFGfu3UKjyK7fUtVGexLlpziC2LQmYiOWYrg=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=N9oXk6+jeZIgt6xuxjr1KQsilVh58ky1UZ31M6lCjBwT2bBM43f6LdDbAyfoAZzya ul5bptlYSCC3Y3K8Kj2zPupepkBNGytObRrvkRkA3a33bgc8S9XKLweV4mfSa2S4Oy ZQERwQdGZj0P53fYbUdOF5r21D6Crt20D/Q4uL5Q= To: libcamera devel Date: Tue, 15 Mar 2022 14:12:37 +0000 Message-Id: <20220315141237.328859-1-kieran.bingham@ideasonboard.com> X-Mailer: git-send-email 2.32.0 In-Reply-To: <20220214095136.25425-1-hpa@redhat.com> References: <20220214095136.25425-1-hpa@redhat.com> MIME-Version: 1.0 Subject: [libcamera-devel] [PATCH] SQUASH: ipa: ipu3: af: Auto focus for dw9719 Surface Go2 VCM X-BeenThere: libcamera-devel@lists.libcamera.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-Patchwork-Original-From: Kieran Bingham via libcamera-devel From: Kieran Bingham Reply-To: Kieran Bingham Errors-To: libcamera-devel-bounces@lists.libcamera.org Sender: "libcamera-devel" These changes are expected to be squashed into the AF implementation. The changes here include: - Correct alignment of documentation *'s - Normalise return value documentation - Fix spelling of /afEstemateVariance/afEstimateVariance/ - Expand documentation of Af::afEstimateVariance With these, for Kate's patch: Reviewed-by: Kieran Bingham I have a few more potential changes, but they can be handled as patches on top and reviewed independently. Signed-off-by: Kieran Bingham --- src/ipa/ipu3/algorithms/af.cpp | 76 +++++++++++++++++++++++----------- 1 file changed, 52 insertions(+), 24 deletions(-) diff --git a/src/ipa/ipu3/algorithms/af.cpp b/src/ipa/ipu3/algorithms/af.cpp index 17055c043200..541b200bcc61 100644 --- a/src/ipa/ipu3/algorithms/af.cpp +++ b/src/ipa/ipu3/algorithms/af.cpp @@ -33,25 +33,25 @@ * \var kAfMinGridWidth * \brief the minimum width of AF grid. * The minimum grid horizontal dimensions. -*/ + */ /** * \var kAfMinGridHeight * \brief the minimum height of AF grid. * The minimum grid vertical dimensions. -*/ + */ /** * \var kAfMaxGridWidth * \brief the maximum width of AF grid. * The maximum grid horizontal dimensions. -*/ + */ /** * \var kAfMaxGridHeight * \brief The maximum height of AF grid. * The maximum grid vertical dimensions. -*/ + */ /** * \var kAfMinGridBlockWidth @@ -127,6 +127,7 @@ static struct ipu3_uapi_af_filter_config afFilterConfigDefault = { /** * \class Af * \brief An auto-focus algorithm based on IPU3 statistics + * * This algorithm is used to determine the position of the lens to make a * focused image. The IPU3 AF processing block computes the statistics that * are composed by two types of filtered value and stores in a AF buffer. @@ -158,7 +159,7 @@ void Af::prepare(IPAContext &context, ipu3_uapi_params *params) * \brief Configure the Af given a configInfo * \param[in] context The shared IPA context * \param[in] configInfo The IPA configuration data - * \return 0 + * \return 0 on success, a negative error code otherwise */ int Af::configure(IPAContext &context, const IPAConfigInfo &configInfo) { @@ -195,7 +196,9 @@ int Af::configure(IPAContext &context, const IPAConfigInfo &configInfo) /** * \brief AF coarse scan + * * Find a near focused image using a coarse step. The step is determined by coarseSearchStep. + * * \param[in] context The shared IPA context */ void Af::afCoarseScan(IPAContext &context) @@ -220,7 +223,9 @@ void Af::afCoarseScan(IPAContext &context) /** * \brief AF fine scan + * * Find an optimum lens position with moving 1 step for each search. + * * \param[in] context The shared IPA context */ void Af::afFineScan(IPAContext &context) @@ -239,7 +244,9 @@ void Af::afFineScan(IPAContext &context) /** * \brief AF reset + * * Reset all the parameters to start over the AF process. + * * \param[in] context The shared IPA context */ void Af::afReset(IPAContext &context) @@ -260,11 +267,13 @@ void Af::afReset(IPAContext &context) /** * \brief AF variance comparison. - * It always picks the largest variance to replace the previous one. The image - * with a larger variance also indicates it is a clearer image than previous - * one. If it finds the negative sign of derivative, it returns immediately. * \param[in] context The IPA context * \param min_step The VCM movement step. + * + * We always pick the largest variance to replace the previous one. The image + * with a larger variance also indicates it is a clearer image than previous + * one. If we find a negative derivative, we return immediately. + * * \return True, if it finds a AF value. */ bool Af::afScan(IPAContext &context, int min_step) @@ -278,7 +287,7 @@ bool Af::afScan(IPAContext &context, int min_step) * 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 ((currentVariance_ - context.frameContext.af.maxVariance) >= -(context.frameContext.af.maxVariance * 0.1)) { /* @@ -313,8 +322,7 @@ bool Af::afScan(IPAContext &context, int min_step) /** * \brief Determine the frame to be ignored. - * \return Return true the frame is ignored. - * \return Return false the frame should be processed. + * \return Return True if the frame should be ignored, false otherwise */ bool Af::afNeedIgnoreFrame() { @@ -334,9 +342,22 @@ void Af::afIgnoreFrameReset() } /** - * \brief Estemate variance + * \brief Estimate variance + * \param y_item The AF filter data set from the IPU3 statistics buffer + * \param len The quantity of table item entries which are valid to process + * \param isY1 Selects between filter Y1 or Y2 to calculate the variance + * + * Calculate the mean of the data set provided by \a y_item, and then calculate + * the variance of that data set from the mean. + * + * The operation can work on one of two sets of values contained within the + * y_item data set supplied by the IPU3. The two data sets are the results of + * both the Y1 and Y2 filters which are used to support coarse (Y1) and fine + * (Y2) calculations of the contrast. + * + * \return The variance of the values in the data set \a y_item selected by \a isY1 */ -double Af::afEstemateVariance(y_table_item_t *y_item, uint32_t len, +double Af::afEstimateVariance(y_table_item_t *y_item, uint32_t len, bool isY1) { uint32_t z = 0; @@ -363,21 +384,25 @@ double Af::afEstemateVariance(y_table_item_t *y_item, uint32_t len, /** * \brief Determine out-of-focus situation. + * \param context The IPA context. + * * Out-of-focus means that the variance change rate for a focused and a new * variance is greater than a threshold. - * \param context The IPA context. - * \return If it is out-of-focus, return true. - * \return If is is focused, return false. + * + * \return True if the variance threshold is crossed indicating lost focus, + * false otherwise. */ bool Af::afIsOutOfFocus(IPAContext context) { const uint32_t diff_var = std::abs(currentVariance_ - context.frameContext.af.maxVariance); const double var_ratio = diff_var / context.frameContext.af.maxVariance; + LOG(IPU3Af, Debug) << "Variance change rate: " << var_ratio << " Current VCM step: " << context.frameContext.af.focus; + if (var_ratio > kMaxChange) return true; else @@ -386,16 +411,19 @@ bool Af::afIsOutOfFocus(IPAContext context) /** * \brief Determine the max contrast image and lens position. - * Ideally, a clear image also has a raletively higher contrast. So, every - * images for each focus step should be tested to find a optimal focus step. + * \param[in] context The IPA context. + * \param[in] stats The statistic buffer of IPU3. + * + * Ideally, a clear image also has a relatively higher contrast. So, every + * image for each focus step should be tested to find an optimal focus step. + * * The Hill Climbing Algorithm[1] is used to find the maximum variance of the - * AF statistic which is the AF output of IPU3. The focus step is increased + * AF statistics which is the AF output of IPU3. The focus step is increased * then the variance of the AF statistic is estimated. If it finds the negative - * derivative which means we just passed the peak, the best focus is found. + * derivative we have just passed the peak, and we infer that the best focus is + * found. * * [1] Hill Climbing Algorithm, https://en.wikipedia.org/wiki/Hill_climbing - * \param[in] context The IPA context. - * \param[in] stats The statistic buffer of IPU3. */ void Af::process(IPAContext &context, const ipu3_uapi_stats_3a *stats) { @@ -415,9 +443,9 @@ void Af::process(IPAContext &context, const ipu3_uapi_stats_3a *stats) * For fine: y2 results are used. */ if (coarseCompleted_) - currentVariance_ = afEstemateVariance(y_item, afRawBufferLen, false); + currentVariance_ = afEstimateVariance(y_item, afRawBufferLen, false); else - currentVariance_ = afEstemateVariance(y_item, afRawBufferLen, true); + currentVariance_ = afEstimateVariance(y_item, afRawBufferLen, true); if (!context.frameContext.af.stable) { afCoarseScan(context);