[libcamera-devel] [PATCH 03/20] libcamera: utils: Add alignUp and alignDown functions
Laurent Pinchart
laurent.pinchart at ideasonboard.com
Tue Jul 14 23:15:53 CEST 2020
Hi Jacopo,
Thank you for the patch.
On Tue, Jul 14, 2020 at 12:41:55PM +0200, Jacopo Mondi wrote:
> Add to libcamera utils library two functions to round up or down a
> value to an alignment and add a test in test/utils.cpp for the two
> new functions.
>
> Signed-off-by: Jacopo Mondi <jacopo at jmondi.org>
> ---
> include/libcamera/internal/utils.h | 3 +++
> src/libcamera/utils.cpp | 22 ++++++++++++++++++++++
> test/utils.cpp | 18 ++++++++++++++++++
> 3 files changed, 43 insertions(+)
>
> diff --git a/include/libcamera/internal/utils.h b/include/libcamera/internal/utils.h
> index 8d026cc6c0fe..25eb24ec2d16 100644
> --- a/include/libcamera/internal/utils.h
> +++ b/include/libcamera/internal/utils.h
> @@ -200,6 +200,9 @@ details::StringSplitter split(const std::string &str, const std::string &delim);
> std::string libcameraBuildPath();
> std::string libcameraSourcePath();
>
> +unsigned int alignUp(unsigned int value, unsigned int alignment);
> +unsigned int alignDown(unsigned int value, unsigned int alignment);
I would inline those two functions, to let the compiler optimize the
code. For instance it's common to have a constexpr alignment value,
which can often translate to more efficient code. I would also mark the
functions as constexpr. As constexpr implies inline,
constexpr unsigned int alignUp(unsigned int value, unsigned int alignment)
{
return (value + alignment - 1) / alignment * alignment;
}
constexpr unsigned int alignDown(unsigned int value, unsigned int alignment)
{
return value / alignment * alignment;
}
> +
> } /* namespace utils */
>
> } /* namespace libcamera */
> diff --git a/src/libcamera/utils.cpp b/src/libcamera/utils.cpp
> index 0567328fe31b..52fb0bb171d8 100644
> --- a/src/libcamera/utils.cpp
> +++ b/src/libcamera/utils.cpp
> @@ -444,6 +444,28 @@ std::string libcameraSourcePath()
> return path + "/";
> }
>
> +/**
> + * \brief Align \a value to \a alignment
Maybe s/value to/value up to/ ? Same for alignDown().
> + * \param[in] value The value to align
> + * \param[in] alignment The alignment
> + * \return The value rounded up to the nearest multiple of \a alignment
> + */
> +unsigned int alignUp(unsigned int value, unsigned int alignment)
> +{
> + return (value + alignment - 1) / alignment * alignment;
> +}
> +
> +/**
> + * \brief Align \a value to \a alignment
> + * \param[in] value The value to align
> + * \param[in] alignment The alignment
> + * \return The value rounded down to the nearest multiple of \a alignment
> + */
> +unsigned int alignDown(unsigned int value, unsigned int alignment)
> +{
> + return value / alignment * alignment;
> +}
> +
> } /* namespace utils */
>
> } /* namespace libcamera */
> diff --git a/test/utils.cpp b/test/utils.cpp
> index f482e6a1d829..02054f46b00e 100644
> --- a/test/utils.cpp
> +++ b/test/utils.cpp
> @@ -7,6 +7,7 @@
>
> #include <iostream>
> #include <map>
> +#include <random>
> #include <sstream>
> #include <string>
> #include <vector>
> @@ -166,6 +167,23 @@ protected:
> return TestFail;
> }
>
> + /* utils::alignUp() and utils::alignDown() tests. */
> + random_device random;
> + unsigned int val = random();
Usage of random number in tests makes it difficult to reproduce
failures. Additionally, the random alignment, could be 0, leading to a
potential division by 0. I think you can just hardcode values :
if (utils::alignUp(6, 3) != 6 || utils::alignUp(7, 3) != 9) {
cerr << "utils::alignUp test failed" << endl;
return TestFail;
}
if (utils::alignDown(6, 3) != 6 || utils::alignDown(7, 3) != 6) {
cerr << "utils::alignDown test failed" << endl;
return TestFail;
}
Reviewed-by: Laurent Pinchart <laurent.pinchart at ideasonboard.com>
> + unsigned int align = random() % (val / 10);
> + unsigned int roundUp = (val + align - 1) / align;
> + unsigned int roundDown = val / align;
> +
> + if (utils::alignUp(val, align) != align * roundUp) {
> + cerr << "utils::alignUp test failed" << endl;
> + return TestFail;
> + }
> +
> + if (utils::alignDown(val, align) != align * roundDown) {
> + cerr << "utils::alignDown test failed" << endl;
> + return TestFail;
> + }
> +
> return TestPass;
> }
> };
--
Regards,
Laurent Pinchart
More information about the libcamera-devel
mailing list