From patchwork Thu Mar 31 12:05:56 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Kieran Bingham X-Patchwork-Id: 15586 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 2A564C3256 for ; Thu, 31 Mar 2022 12:06:04 +0000 (UTC) Received: from lancelot.ideasonboard.com (localhost [IPv6:::1]) by lancelot.ideasonboard.com (Postfix) with ESMTP id 384D965633; Thu, 31 Mar 2022 14:06:03 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=libcamera.org; s=mail; t=1648728363; bh=85HrKCwvFGkyTPd55ffsJniWjJclB4LUstQ25z0D1PE=; h=To:Date:Subject:List-Id:List-Unsubscribe:List-Archive:List-Post: List-Help:List-Subscribe:From:Reply-To:From; b=RWrnU8r48SgVJhAwLoOhEEnEX0AAhYFihKS54XHCgkdnVWov5/srTmrDHN9/wXIwb 6nfi9ISSrZDcQ2T6mpxkqDtB/qu2SzqMJJXxEYjmi4glUgBmiGYUaa8z4tZ/HCA42o y1ajdgRPpOHAQFogl0VNx0fjzq4fsnxfCUppWwE1CmyJSF1vbBKTytyZTePNcUOOWv syGQlnmpfGV64A9SIdo9z0j3bLMxI5/vp0wcnbtGbdY6PpTun0ClD/kh64JXUxh2KV UYcMDKLz54qvaQJmiTA5sFDH0tHIBXEIBYViL4BXiuWFxs3nn4MK9A0uyRthlyRX2P WJ8PTfotyw0qw== Received: from perceval.ideasonboard.com (perceval.ideasonboard.com [213.167.242.64]) by lancelot.ideasonboard.com (Postfix) with ESMTPS id 15E666559A for ; Thu, 31 Mar 2022 14:06:01 +0200 (CEST) Authentication-Results: lancelot.ideasonboard.com; dkim=pass (1024-bit key; unprotected) header.d=ideasonboard.com header.i=@ideasonboard.com header.b="quwiCdo8"; dkim-atps=neutral Received: from Monstersaurus.local (cpc89244-aztw30-2-0-cust3082.18-1.cable.virginm.net [86.31.172.11]) by perceval.ideasonboard.com (Postfix) with ESMTPSA id 8956759D; Thu, 31 Mar 2022 14:06:00 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1648728360; bh=85HrKCwvFGkyTPd55ffsJniWjJclB4LUstQ25z0D1PE=; h=From:To:Cc:Subject:Date:From; b=quwiCdo8X22IaKCZEyZ1XQ2M0AeGGloWt0Mhz/f2SmG7tI0r9HWnGma1K6Q8r/Dx5 ZnMm2mQFsWSWT7QOwMfZW1wU3m1m+Sv7Ad6dB79QUvh3CUexoJ53E9S3b2PoLmAB5S KgNkGvDg1hdMW3wXfY1otHuubXG7m3XuXx1GuwlE= To: libcamera devel Date: Thu, 31 Mar 2022 13:05:56 +0100 Message-Id: <20220331120556.2147745-1-kieran.bingham@ideasonboard.com> X-Mailer: git-send-email 2.32.0 MIME-Version: 1.0 Subject: [libcamera-devel] [PATCH] libcamera: geometry: Add Size members to clamp to a min/max 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" Provide two new operations to support clamping a Size to a given minimum or maximum Size, or returning a const variant of the same restriction. Signed-off-by: Kieran Bingham Reviewed-by: Umang Jain --- I was expecting to use this for clamping the block width and height for the AF hardware restrictions on the IPU3 ... but it turns out to be not quite appropriate to use a Size there, as the clamped values are stored in an IPU3 struct directly. However, having made this - I think it is likely to be useful elsewhere so posting so it doesn't get lost. Tests added, so it could be merged already even if there is no current user yet. I expect it's more likely to get used if it exists, than if it doesn't ;-) include/libcamera/geometry.h | 16 ++++++++++++++++ src/libcamera/geometry.cpp | 21 +++++++++++++++++++++ test/geometry.cpp | 24 ++++++++++++++++++++++-- 3 files changed, 59 insertions(+), 2 deletions(-) diff --git a/include/libcamera/geometry.h b/include/libcamera/geometry.h index 7838b6793617..a447beb55965 100644 --- a/include/libcamera/geometry.h +++ b/include/libcamera/geometry.h @@ -93,6 +93,13 @@ public: return *this; } + Size &clamp(const Size &minimum, const Size &maximum) + { + width = std::clamp(width, minimum.width, maximum.width); + height = std::clamp(height, minimum.height, maximum.height); + return *this; + } + Size &growBy(const Size &margins) { width += margins.width; @@ -141,6 +148,15 @@ public: }; } + __nodiscard constexpr Size clampedTo(const Size &minimum, + const Size &maximum) const + { + return { + std::clamp(width, minimum.width, maximum.width), + std::clamp(height, minimum.height, maximum.height) + }; + } + __nodiscard constexpr Size grownBy(const Size &margins) const { return { diff --git a/src/libcamera/geometry.cpp b/src/libcamera/geometry.cpp index cb3c2de18d5e..7ac053a536c1 100644 --- a/src/libcamera/geometry.cpp +++ b/src/libcamera/geometry.cpp @@ -173,6 +173,18 @@ const std::string Size::toString() const * \return A reference to this object */ +/** + * \fn Size::clamp(const Size &minimum, const Size &maximum) + * \brief Restrict the size to be constrained within the \a minimum and \a maximum + * \param[in] minimum The minimum size + * \param[in] maximum The maximum size + * + * This function restricts the width and height to the constraints of the \a + * minimum and the \a maximum sizes given. + * + * \return A reference to this object + */ + /** * \fn Size::growBy(const Size &margins) * \brief Grow the size by \a margins in place @@ -231,6 +243,15 @@ const std::string Size::toString() const * height of this size and the \a expand size */ +/** + * \fn Size::clampedTo(const Size &minimum, const Size &maximum) + * \brief Restrict the size to be constrained within the \a minimum and \a maximum + * \param[in] minimum The minimum size + * \param[in] maximum The maximum size + * \return A size whose width and height match this size within the constraints + * of the \a minimum and \a maximum sizes given. + */ + /** * \fn Size::grownBy(const Size &margins) * \brief Grow the size by \a margins diff --git a/test/geometry.cpp b/test/geometry.cpp index 5125692496b3..1d3d3cad7174 100644 --- a/test/geometry.cpp +++ b/test/geometry.cpp @@ -106,7 +106,7 @@ protected: /* * Test alignDownTo(), alignUpTo(), boundTo(), expandTo(), - * growBy() and shrinkBy() + * clamp() growBy() and shrinkBy() */ Size s(50, 50); @@ -134,6 +134,18 @@ protected: return TestFail; } + s.clamp({ 80, 120 }, { 160, 240 }); + if (s != Size(80, 120)) { + cout << "Size::clamp (minium) test failed" << endl; + return TestFail; + } + + s.clamp({ 20, 30 }, { 50, 50 }); + if (s != Size(50, 50)) { + cout << "Size::clamp (maximum) test failed" << endl; + return TestFail; + } + s.growBy({ 10, 20 }); if (s != Size(60, 70)) { cout << "Size::growBy() test failed" << endl; @@ -162,7 +174,7 @@ protected: /* * Test alignedDownTo(), alignedUpTo(), boundedTo(), - * expandedTo(), grownBy() and shrunkBy() + * expandedTo(), clampedTo(), grownBy() and shrunkBy() */ if (Size(0, 0).alignedDownTo(16, 8) != Size(0, 0) || Size(1, 1).alignedDownTo(16, 8) != Size(0, 0) || @@ -192,6 +204,14 @@ protected: return TestFail; } + if (Size(0, 0).clampedTo({ 60, 40 }, { 80, 90 }) != Size(60, 40) || + Size(100, 100).clampedTo({ 60, 40 }, { 80, 90 }) != Size(80, 90) || + Size(30, 100).clampedTo({ 60, 40 }, { 80, 90 }) != Size(60, 90) || + Size(100, 30).clampedTo({ 60, 40 }, { 80, 90 }) != Size(80, 40)) { + cout << "Size::clampedTo() test failed" << endl; + return TestFail; + } + if (Size(0, 0).grownBy({ 10, 20 }) != Size(10, 20) || Size(200, 50).grownBy({ 10, 20 }) != Size(210, 70)) { cout << "Size::grownBy() test failed" << endl;