[{"id":29552,"web_url":"https://patchwork.libcamera.org/comment/29552/","msgid":"<20240517102402.errh4vxvbbnjh6vp@jasper>","date":"2024-05-17T10:24:02","subject":"Re: [PATCH v2 3/5] utils: libtuning: modules: Add skeletal CCM\n\tmodule","submitter":{"id":184,"url":"https://patchwork.libcamera.org/api/people/184/","name":"Stefan Klug","email":"stefan.klug@ideasonboard.com"},"content":"Hi Paul,\n\nthanks for the patch.\n\nOn Fri, May 17, 2024 at 04:57:49PM +0900, Paul Elder wrote:\n> Add a skeletal CCM module just so that we can have some CCM tuning\n> values that we can use to test during development of CCM in the IPAs. As\n> rkisp1 is the main target, we only add support for rkisp1 for now.\n> \n> The parameters are mostly copied from the hardcoded values in ctt,\n> except for the metering modes.\n> \n> As CCM is called CTK (for crosstalk) in rkisp1, that is the name that\n> will be used for rkisp1.\n> \n> Signed-off-by: Paul Elder <paul.elder@ideasonboard.com>\n> \n> ---\n> Changes in v2:\n> - move converting floating to fixed point from the tuning script to the\n>   IPA\n> - reorganize the ccm list to mirror the format of lsc\n> - add offset vectors\n> ---\n>  .../tuning/libtuning/modules/ccm/__init__.py  |  6 ++\n>  utils/tuning/libtuning/modules/ccm/ccm.py     | 22 +++++\n>  utils/tuning/libtuning/modules/ccm/rkisp1.py  | 89 +++++++++++++++++++\n>  3 files changed, 117 insertions(+)\n>  create mode 100644 utils/tuning/libtuning/modules/ccm/__init__.py\n>  create mode 100644 utils/tuning/libtuning/modules/ccm/ccm.py\n>  create mode 100644 utils/tuning/libtuning/modules/ccm/rkisp1.py\n> \n> diff --git a/utils/tuning/libtuning/modules/ccm/__init__.py b/utils/tuning/libtuning/modules/ccm/__init__.py\n> new file mode 100644\n> index 000000000..322602afe\n> --- /dev/null\n> +++ b/utils/tuning/libtuning/modules/ccm/__init__.py\n> @@ -0,0 +1,6 @@\n> +# SPDX-License-Identifier: GPL-2.0-or-later\n> +#\n> +# Copyright (C) 2024, Paul Elder <paul.elder@ideasonboard.com>\n> +\n> +from libtuning.modules.ccm.ccm import CCM\n> +from libtuning.modules.ccm.rkisp1 import CCMRkISP1\n> diff --git a/utils/tuning/libtuning/modules/ccm/ccm.py b/utils/tuning/libtuning/modules/ccm/ccm.py\n> new file mode 100644\n> index 000000000..7ddf0a8ca\n> --- /dev/null\n> +++ b/utils/tuning/libtuning/modules/ccm/ccm.py\n> @@ -0,0 +1,22 @@\n> +# SPDX-License-Identifier: BSD-2-Clause\n> +#\n> +# Copyright (C) 2019, Raspberry Pi Ltd\n> +# Copyright (C) 2024, Paul Elder <paul.elder@ideasonboard.com>\n> +\n> +from ..module import Module\n> +\n> +import libtuning as lt\n> +import libtuning.utils as utils\n> +\n> +import numpy as np\n> +\n> +class CCM(Module):\n> +    type = 'ccm'\n> +    hr_name = 'CCM (Base)'\n> +    out_name = 'GenericCCM'\n> +\n> +    def __init__(self, *,\n> +                 debug: list):\n> +        super().__init__()\n> +\n> +        self.debug = debug\n> diff --git a/utils/tuning/libtuning/modules/ccm/rkisp1.py b/utils/tuning/libtuning/modules/ccm/rkisp1.py\n> new file mode 100644\n> index 000000000..acae0a9b0\n> --- /dev/null\n> +++ b/utils/tuning/libtuning/modules/ccm/rkisp1.py\n> @@ -0,0 +1,89 @@\n> +# SPDX-License-Identifier: BSD-2-Clause\n> +#\n> +# Copyright (C) 2019, Raspberry Pi Ltd\n> +# Copyright (C) 2024, Paul Elder <paul.elder@ideasonboard.com>\n> +#\n> +# rkisp1.py - Ccm module for tuning rkisp1\n> +\n> +from .ccm import CCM\n> +\n> +import libtuning as lt\n> +import libtuning.utils as utils\n> +\n> +from numbers import Number\n> +import numpy as np\n> +\n> +\n> +class CCMRkISP1(CCM):\n> +    hr_name = 'Crosstalk Correction (RkISP1)'\n> +    out_name = 'Ccm'\n> +    # \\todo Not sure if this is useful. Probably will remove later.\n> +    compatible = ['rkisp1']\n> +\n> +    def __init__(self, **kwargs):\n> +        super().__init__(**kwargs)\n> +\n> +    # We don't actually need anything from the config file\n> +    def validate_config(self, config: dict) -> bool:\n> +        return True\n> +\n> +    def _generate_ccms(self) -> dict:\n> +        ccms = [\n> +                {\n> +                    'ct': 2860,\n> +                    'ccm': [ 2.12089, -0.52461, -0.59629,\n> +                            -0.85342,  2.80445, -0.95103,\n> +                            -0.26897, -1.14788,  2.41685 ],\n> +                    'offsets': [ 0, 0, 0 ]\n\nI stumbled over that. Should it be named 'offset' (singular)?\n\nWith or without that changed:\n\nReviewed-by: Stefan Klug <stefan.klug@ideasonboard.com> \n\nCheers,\nStefan\n\n> +                },\n> +\n> +                {\n> +                    'ct': 2960,\n> +                    'ccm': [ 2.26962, -0.54174, -0.72789,\n> +                            -0.77008,  2.60271, -0.83262,\n> +                            -0.26036, -1.51254,  2.77289 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 3603,\n> +                    'ccm': [ 2.18644, -0.66148, -0.52496,\n> +                            -0.77828,  2.69474, -0.91645,\n> +                            -0.25239, -0.83059,  2.08298 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 4650,\n> +                    'ccm': [ 2.18174, -0.70887, -0.47287,\n> +                            -0.70196,  2.76426, -1.06231,\n> +                            -0.25157, -0.71978,  1.97135 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 5858,\n> +                    'ccm': [ 2.32392, -0.88421, -0.43971,\n> +                            -0.63821,  2.58348, -0.94527,\n> +                            -0.28541, -0.54112,  1.82653 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 7580,\n> +                    'ccm': [ 2.21175, -0.53242, -0.67933,\n> +                            -0.57875,  3.07922, -1.50047,\n> +                            -0.27709, -0.73338,  2.01048 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +        ]\n> +\n> +        return ccms\n> +\n> +    def process(self, config: dict, images: list, outputs: dict) -> dict:\n> +        output = {}\n> +\n> +        output['ccms'] = self._generate_ccms()\n> +        # \\todo Debug functionality\n> +\n> +        return output\n> -- \n> 2.39.2\n>","headers":{"Return-Path":"<libcamera-devel-bounces@lists.libcamera.org>","X-Original-To":"parsemail@patchwork.libcamera.org","Delivered-To":"parsemail@patchwork.libcamera.org","Received":["from lancelot.ideasonboard.com (lancelot.ideasonboard.com\n\t[92.243.16.209])\n\tby patchwork.libcamera.org (Postfix) with ESMTPS id 87BCCBDE6B\n\tfor <parsemail@patchwork.libcamera.org>;\n\tFri, 17 May 2024 10:24:08 +0000 (UTC)","from lancelot.ideasonboard.com (localhost [IPv6:::1])\n\tby lancelot.ideasonboard.com (Postfix) with ESMTP id 6B14B6347C;\n\tFri, 17 May 2024 12:24:07 +0200 (CEST)","from perceval.ideasonboard.com (perceval.ideasonboard.com\n\t[213.167.242.64])\n\tby lancelot.ideasonboard.com (Postfix) with ESMTPS id 5E22561A5A\n\tfor <libcamera-devel@lists.libcamera.org>;\n\tFri, 17 May 2024 12:24:06 +0200 (CEST)","from ideasonboard.com (unknown\n\t[IPv6:2a00:6020:448c:6c00:d572:8aa2:3e8e:5b99])\n\tby perceval.ideasonboard.com (Postfix) with ESMTPSA id 12EAC82E;\n\tFri, 17 May 2024 12:23:57 +0200 (CEST)"],"Authentication-Results":"lancelot.ideasonboard.com; dkim=pass (1024-bit key;\n\tunprotected) header.d=ideasonboard.com header.i=@ideasonboard.com\n\theader.b=\"S7CQKDRD\"; dkim-atps=neutral","DKIM-Signature":"v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com;\n\ts=mail; t=1715941437;\n\tbh=B31IuGlpqhe8GyWzFa14vYOlRf+EDocOR9ZUM9r6vWc=;\n\th=Date:From:To:Cc:Subject:References:In-Reply-To:From;\n\tb=S7CQKDRDSOkq6jdcH0qnxHMI8Hq4ge9D5xFCOG7b5mX04/NSZ1Jxiel2xVI2hb+Uu\n\tBoCiP0FUF8QJgq3t86EhtHaDN4oZFxM+quVn+07QpVJpzwanKB1YLXYrTfXCtSSBN/\n\tUi4HGRm1u5baOxqnI9ZbJ9Oonr4fnkbmUeiEJU+o=","Date":"Fri, 17 May 2024 12:24:02 +0200","From":"Stefan Klug <stefan.klug@ideasonboard.com>","To":"Paul Elder <paul.elder@ideasonboard.com>","Cc":"libcamera-devel@lists.libcamera.org","Subject":"Re: [PATCH v2 3/5] utils: libtuning: modules: Add skeletal CCM\n\tmodule","Message-ID":"<20240517102402.errh4vxvbbnjh6vp@jasper>","References":"<20240517075751.3866269-1-paul.elder@ideasonboard.com>\n\t<20240517075751.3866269-4-paul.elder@ideasonboard.com>","MIME-Version":"1.0","Content-Type":"text/plain; charset=utf-8","Content-Disposition":"inline","In-Reply-To":"<20240517075751.3866269-4-paul.elder@ideasonboard.com>","X-BeenThere":"libcamera-devel@lists.libcamera.org","X-Mailman-Version":"2.1.29","Precedence":"list","List-Id":"<libcamera-devel.lists.libcamera.org>","List-Unsubscribe":"<https://lists.libcamera.org/options/libcamera-devel>,\n\t<mailto:libcamera-devel-request@lists.libcamera.org?subject=unsubscribe>","List-Archive":"<https://lists.libcamera.org/pipermail/libcamera-devel/>","List-Post":"<mailto:libcamera-devel@lists.libcamera.org>","List-Help":"<mailto:libcamera-devel-request@lists.libcamera.org?subject=help>","List-Subscribe":"<https://lists.libcamera.org/listinfo/libcamera-devel>,\n\t<mailto:libcamera-devel-request@lists.libcamera.org?subject=subscribe>","Errors-To":"libcamera-devel-bounces@lists.libcamera.org","Sender":"\"libcamera-devel\" <libcamera-devel-bounces@lists.libcamera.org>"}},{"id":29580,"web_url":"https://patchwork.libcamera.org/comment/29580/","msgid":"<e5e71cc4-5585-4a80-8d03-777c99c9094c@ideasonboard.com>","date":"2024-05-20T13:30:10","subject":"Re: [PATCH v2 3/5] utils: libtuning: modules: Add skeletal CCM\n\tmodule","submitter":{"id":156,"url":"https://patchwork.libcamera.org/api/people/156/","name":"Dan Scally","email":"dan.scally@ideasonboard.com"},"content":"Hi Paul\n\nOn 17/05/2024 08:57, Paul Elder wrote:\n> Add a skeletal CCM module just so that we can have some CCM tuning\n> values that we can use to test during development of CCM in the IPAs. As\n> rkisp1 is the main target, we only add support for rkisp1 for now.\n>\n> The parameters are mostly copied from the hardcoded values in ctt,\n> except for the metering modes.\n>\n> As CCM is called CTK (for crosstalk) in rkisp1, that is the name that\n> will be used for rkisp1.\n>\n> Signed-off-by: Paul Elder <paul.elder@ideasonboard.com>\n>\n> ---\n> Changes in v2:\n> - move converting floating to fixed point from the tuning script to the\n>    IPA\n> - reorganize the ccm list to mirror the format of lsc\n> - add offset vectors\n> ---\n>   .../tuning/libtuning/modules/ccm/__init__.py  |  6 ++\n>   utils/tuning/libtuning/modules/ccm/ccm.py     | 22 +++++\n>   utils/tuning/libtuning/modules/ccm/rkisp1.py  | 89 +++++++++++++++++++\n>   3 files changed, 117 insertions(+)\n>   create mode 100644 utils/tuning/libtuning/modules/ccm/__init__.py\n>   create mode 100644 utils/tuning/libtuning/modules/ccm/ccm.py\n>   create mode 100644 utils/tuning/libtuning/modules/ccm/rkisp1.py\n>\n> diff --git a/utils/tuning/libtuning/modules/ccm/__init__.py b/utils/tuning/libtuning/modules/ccm/__init__.py\n> new file mode 100644\n> index 000000000..322602afe\n> --- /dev/null\n> +++ b/utils/tuning/libtuning/modules/ccm/__init__.py\n> @@ -0,0 +1,6 @@\n> +# SPDX-License-Identifier: GPL-2.0-or-later\n> +#\n> +# Copyright (C) 2024, Paul Elder <paul.elder@ideasonboard.com>\n> +\n> +from libtuning.modules.ccm.ccm import CCM\n> +from libtuning.modules.ccm.rkisp1 import CCMRkISP1\n> diff --git a/utils/tuning/libtuning/modules/ccm/ccm.py b/utils/tuning/libtuning/modules/ccm/ccm.py\n> new file mode 100644\n> index 000000000..7ddf0a8ca\n> --- /dev/null\n> +++ b/utils/tuning/libtuning/modules/ccm/ccm.py\n> @@ -0,0 +1,22 @@\n> +# SPDX-License-Identifier: BSD-2-Clause\n> +#\n> +# Copyright (C) 2019, Raspberry Pi Ltd\n> +# Copyright (C) 2024, Paul Elder <paul.elder@ideasonboard.com>\n> +\n> +from ..module import Module\n> +\n> +import libtuning as lt\n> +import libtuning.utils as utils\n> +\n> +import numpy as np\n> +\n> +class CCM(Module):\n> +    type = 'ccm'\n> +    hr_name = 'CCM (Base)'\n> +    out_name = 'GenericCCM'\n> +\n> +    def __init__(self, *,\n> +                 debug: list):\n> +        super().__init__()\n> +\n> +        self.debug = debug\n> diff --git a/utils/tuning/libtuning/modules/ccm/rkisp1.py b/utils/tuning/libtuning/modules/ccm/rkisp1.py\n> new file mode 100644\n> index 000000000..acae0a9b0\n> --- /dev/null\n> +++ b/utils/tuning/libtuning/modules/ccm/rkisp1.py\n> @@ -0,0 +1,89 @@\n> +# SPDX-License-Identifier: BSD-2-Clause\n> +#\n> +# Copyright (C) 2019, Raspberry Pi Ltd\n> +# Copyright (C) 2024, Paul Elder <paul.elder@ideasonboard.com>\n> +#\n> +# rkisp1.py - Ccm module for tuning rkisp1\n> +\n> +from .ccm import CCM\n> +\n> +import libtuning as lt\n> +import libtuning.utils as utils\n> +\n> +from numbers import Number\n> +import numpy as np\n> +\n> +\n> +class CCMRkISP1(CCM):\n> +    hr_name = 'Crosstalk Correction (RkISP1)'\n> +    out_name = 'Ccm'\n> +    # \\todo Not sure if this is useful. Probably will remove later.\n> +    compatible = ['rkisp1']\n\nLet's drop it now :)\n\n\nReviewed-by: Daniel Scally <dan.scally@ideasonboard.com>\n\n> +\n> +    def __init__(self, **kwargs):\n> +        super().__init__(**kwargs)\n> +\n> +    # We don't actually need anything from the config file\n> +    def validate_config(self, config: dict) -> bool:\n> +        return True\n> +\n> +    def _generate_ccms(self) -> dict:\n> +        ccms = [\n> +                {\n> +                    'ct': 2860,\n> +                    'ccm': [ 2.12089, -0.52461, -0.59629,\n> +                            -0.85342,  2.80445, -0.95103,\n> +                            -0.26897, -1.14788,  2.41685 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 2960,\n> +                    'ccm': [ 2.26962, -0.54174, -0.72789,\n> +                            -0.77008,  2.60271, -0.83262,\n> +                            -0.26036, -1.51254,  2.77289 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 3603,\n> +                    'ccm': [ 2.18644, -0.66148, -0.52496,\n> +                            -0.77828,  2.69474, -0.91645,\n> +                            -0.25239, -0.83059,  2.08298 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 4650,\n> +                    'ccm': [ 2.18174, -0.70887, -0.47287,\n> +                            -0.70196,  2.76426, -1.06231,\n> +                            -0.25157, -0.71978,  1.97135 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 5858,\n> +                    'ccm': [ 2.32392, -0.88421, -0.43971,\n> +                            -0.63821,  2.58348, -0.94527,\n> +                            -0.28541, -0.54112,  1.82653 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +\n> +                {\n> +                    'ct': 7580,\n> +                    'ccm': [ 2.21175, -0.53242, -0.67933,\n> +                            -0.57875,  3.07922, -1.50047,\n> +                            -0.27709, -0.73338,  2.01048 ],\n> +                    'offsets': [ 0, 0, 0 ]\n> +                },\n> +        ]\n> +\n> +        return ccms\n> +\n> +    def process(self, config: dict, images: list, outputs: dict) -> dict:\n> +        output = {}\n> +\n> +        output['ccms'] = self._generate_ccms()\n> +        # \\todo Debug functionality\n> +\n> +        return output","headers":{"Return-Path":"<libcamera-devel-bounces@lists.libcamera.org>","X-Original-To":"parsemail@patchwork.libcamera.org","Delivered-To":"parsemail@patchwork.libcamera.org","Received":["from lancelot.ideasonboard.com (lancelot.ideasonboard.com\n\t[92.243.16.209])\n\tby patchwork.libcamera.org (Postfix) with ESMTPS id 5B852BD78E\n\tfor <parsemail@patchwork.libcamera.org>;\n\tMon, 20 May 2024 13:30:16 +0000 (UTC)","from lancelot.ideasonboard.com (localhost [IPv6:::1])\n\tby lancelot.ideasonboard.com (Postfix) with ESMTP id 0E47663480;\n\tMon, 20 May 2024 15:30:16 +0200 (CEST)","from perceval.ideasonboard.com (perceval.ideasonboard.com\n\t[IPv6:2001:4b98:dc2:55:216:3eff:fef7:d647])\n\tby lancelot.ideasonboard.com (Postfix) with ESMTPS id 2EE6F61A58\n\tfor <libcamera-devel@lists.libcamera.org>;\n\tMon, 20 May 2024 15:30:14 +0200 (CEST)","from [192.168.0.43]\n\t(cpc141996-chfd3-2-0-cust928.12-3.cable.virginm.net [86.13.91.161])\n\tby perceval.ideasonboard.com (Postfix) with ESMTPSA id 96B06F50\n\tfor <libcamera-devel@lists.libcamera.org>;\n\tMon, 20 May 2024 15:30:02 +0200 (CEST)"],"Authentication-Results":"lancelot.ideasonboard.com; dkim=pass (1024-bit key;\n\tunprotected) header.d=ideasonboard.com header.i=@ideasonboard.com\n\theader.b=\"pzRH0Rvu\"; dkim-atps=neutral","DKIM-Signature":"v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com;\n\ts=mail; t=1716211803;\n\tbh=HBJBzwcG9Lgb7vylJySMJH22F8aCthf/Dlc171avILQ=;\n\th=Date:Subject:To:References:From:In-Reply-To:From;\n\tb=pzRH0RvuaJGvX7LfDxK7Jyc7r0amdH7rTatSJXVQdRboCdlhW2R1Z8qsWDZ17B6S9\n\traIUhVq+zym+Nm1a7cDAJ/4K31VgGrN0XV+xICA/ndE4K1Xs1gDU5tR1axlypMebK3\n\t+jYb78kxGmuYLWdWXCQP9az+paYl22n/61vUiQQ4=","Message-ID":"<e5e71cc4-5585-4a80-8d03-777c99c9094c@ideasonboard.com>","Date":"Mon, 20 May 2024 14:30:10 +0100","MIME-Version":"1.0","User-Agent":"Mozilla Thunderbird","Subject":"Re: [PATCH v2 3/5] utils: libtuning: modules: Add skeletal CCM\n\tmodule","To":"libcamera-devel@lists.libcamera.org","References":"<20240517075751.3866269-1-paul.elder@ideasonboard.com>\n\t<20240517075751.3866269-4-paul.elder@ideasonboard.com>","Content-Language":"en-US","From":"Dan Scally <dan.scally@ideasonboard.com>","Autocrypt":"addr=dan.scally@ideasonboard.com; keydata=\n\t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","In-Reply-To":"<20240517075751.3866269-4-paul.elder@ideasonboard.com>","Content-Type":"text/plain; charset=UTF-8; format=flowed","Content-Transfer-Encoding":"7bit","X-BeenThere":"libcamera-devel@lists.libcamera.org","X-Mailman-Version":"2.1.29","Precedence":"list","List-Id":"<libcamera-devel.lists.libcamera.org>","List-Unsubscribe":"<https://lists.libcamera.org/options/libcamera-devel>,\n\t<mailto:libcamera-devel-request@lists.libcamera.org?subject=unsubscribe>","List-Archive":"<https://lists.libcamera.org/pipermail/libcamera-devel/>","List-Post":"<mailto:libcamera-devel@lists.libcamera.org>","List-Help":"<mailto:libcamera-devel-request@lists.libcamera.org?subject=help>","List-Subscribe":"<https://lists.libcamera.org/listinfo/libcamera-devel>,\n\t<mailto:libcamera-devel-request@lists.libcamera.org?subject=subscribe>","Errors-To":"libcamera-devel-bounces@lists.libcamera.org","Sender":"\"libcamera-devel\" <libcamera-devel-bounces@lists.libcamera.org>"}}]