[libcamera-devel] [PATCH 1/2] libcamera: flags: Add type-safe enum-based flags
Laurent Pinchart
laurent.pinchart at ideasonboard.com
Sat Jul 25 01:08:40 CEST 2020
Add a Flags template class that provide type-safe bitwise operators on
enum values. This allows using enum types for bit fields, without giving
away type-safety as usually done when storing combined flags in integer
variables.
Signed-off-by: Laurent Pinchart <laurent.pinchart at ideasonboard.com>
---
include/libcamera/flags.h | 195 ++++++++++++++++++++++++++++++++++
include/libcamera/meson.build | 1 +
src/libcamera/flags.cpp | 192 +++++++++++++++++++++++++++++++++
src/libcamera/meson.build | 1 +
4 files changed, 389 insertions(+)
create mode 100644 include/libcamera/flags.h
create mode 100644 src/libcamera/flags.cpp
diff --git a/include/libcamera/flags.h b/include/libcamera/flags.h
new file mode 100644
index 000000000000..74525ccb1050
--- /dev/null
+++ b/include/libcamera/flags.h
@@ -0,0 +1,195 @@
+/* SPDX-License-Identifier: LGPL-2.1-or-later */
+/*
+ * Copyright (C) 2020, Google Inc.
+ *
+ * flags.h - Type-safe enum-based bitfields
+ */
+#ifndef __LIBCAMERA_FLAGS_H__
+#define __LIBCAMERA_FLAGS_H__
+
+#include <type_traits>
+
+namespace libcamera {
+
+template<typename E>
+class Flags
+{
+public:
+ static_assert(std::is_enum<E>::value,
+ "Flags<> template parameter must be an enum");
+
+ using Type = std::underlying_type_t<E>;
+
+ constexpr Flags()
+ : value_(0)
+ {
+ }
+
+ constexpr Flags(E flag)
+ : value_(static_cast<Type>(flag))
+ {
+ }
+
+ constexpr Flags &operator&=(E flag)
+ {
+ value_ &= static_cast<Type>(flag);
+ return *this;
+ }
+
+ constexpr Flags &operator&=(Flags other)
+ {
+ value_ &= other.value_;
+ return *this;
+ }
+
+ constexpr Flags &operator|=(E flag)
+ {
+ value_ |= static_cast<Type>(flag);
+ return *this;
+ }
+
+ constexpr Flags &operator|=(Flags other)
+ {
+ value_ |= other.value_;
+ return *this;
+ }
+
+ constexpr Flags &operator^=(E flag)
+ {
+ value_ ^= static_cast<Type>(flag);
+ return *this;
+ }
+
+ constexpr Flags &operator^=(Flags other)
+ {
+ value_ ^= other.value_;
+ return *this;
+ }
+
+ constexpr bool operator==(E flag)
+ {
+ return value_ == static_cast<Type>(flag);
+ }
+
+ constexpr bool operator==(Flags other)
+ {
+ return value_ == static_cast<Type>(other);
+ }
+
+ constexpr bool operator!=(E flag)
+ {
+ return value_ != static_cast<Type>(flag);
+ }
+
+ constexpr bool operator!=(Flags other)
+ {
+ return value_ != static_cast<Type>(other);
+ }
+
+ constexpr explicit operator Type() const
+ {
+ return value_;
+ }
+
+ constexpr explicit operator bool() const
+ {
+ return !!value_;
+ }
+
+ constexpr Flags operator&(E flag) const
+ {
+ return Flags(static_cast<E>(value_ & static_cast<Type>(flag)));
+ }
+
+ constexpr Flags operator&(Flags other) const
+ {
+ return Flags(static_cast<E>(value_ & other.value_));
+ }
+
+ constexpr Flags operator|(E flag) const
+ {
+ return Flags(static_cast<E>(value_ | static_cast<Type>(flag)));
+ }
+
+ constexpr Flags operator|(Flags other) const
+ {
+ return Flags(static_cast<E>(value_ | other.value_));
+ }
+
+ constexpr Flags operator^(E flag) const
+ {
+ return Flags(static_cast<E>(value_ ^ static_cast<Type>(flag)));
+ }
+
+ constexpr Flags operator^(Flags other) const
+ {
+ return Flags(static_cast<E>(value_ ^ other.value_));
+ }
+
+ constexpr Flags operator~() const
+ {
+ return Flags(static_cast<E>(~value_));
+ }
+
+ constexpr bool operator!() const
+ {
+ return !value_;
+ }
+
+private:
+ Type value_;
+};
+
+#ifndef __DOXYGEN__
+template<typename E>
+struct flags_enable_operators {
+ static const bool enable = false;
+};
+
+template<typename E>
+typename std::enable_if_t<flags_enable_operators<E>::enable, Flags<E>>
+operator|(E lhs, E rhs)
+{
+ using type = std::underlying_type_t<E>;
+ return Flags<E>(static_cast<E>(static_cast<type>(lhs) | static_cast<type>(rhs)));
+}
+
+template<typename E>
+typename std::enable_if_t<flags_enable_operators<E>::enable, Flags<E>>
+operator&(E lhs, E rhs)
+{
+ using type = std::underlying_type_t<E>;
+ return Flags<E>(static_cast<E>(static_cast<type>(lhs) & static_cast<type>(rhs)));
+}
+
+template<typename E>
+typename std::enable_if_t<flags_enable_operators<E>::enable, Flags<E>>
+operator^(E lhs, E rhs)
+{
+ using type = std::underlying_type_t<E>;
+ return Flags<E>(static_cast<E>(static_cast<type>(lhs) ^ static_cast<type>(rhs)));
+}
+
+template<typename E>
+typename std::enable_if_t<flags_enable_operators<E>::enable, Flags<E>>
+operator~(E rhs)
+{
+ using type = std::underlying_type_t<E>;
+ return Flags<E>(static_cast<E>(~static_cast<type>(rhs)));
+}
+
+#define LIBCAMERA_FLAGS_ENABLE_OPERATORS(_enum) \
+template<> \
+struct flags_enable_operators<_enum> { \
+ static const bool enable = true; \
+};
+
+#else /* __DOXYGEN__ */
+
+#define LIBCAMERA_FLAGS_ENABLE_OPERATORS(_enum)
+
+#endif /* __DOXYGEN__ */
+
+} /* namespace libcamera */
+
+#endif /* __LIBCAMERA_FLAGS_H__ */
diff --git a/include/libcamera/meson.build b/include/libcamera/meson.build
index cdb8e0372e77..cd51a1f09ffb 100644
--- a/include/libcamera/meson.build
+++ b/include/libcamera/meson.build
@@ -9,6 +9,7 @@ libcamera_public_headers = files([
'event_dispatcher.h',
'event_notifier.h',
'file_descriptor.h',
+ 'flags.h',
'framebuffer_allocator.h',
'geometry.h',
'logging.h',
diff --git a/src/libcamera/flags.cpp b/src/libcamera/flags.cpp
new file mode 100644
index 000000000000..d0521755a83e
--- /dev/null
+++ b/src/libcamera/flags.cpp
@@ -0,0 +1,192 @@
+/* SPDX-License-Identifier: LGPL-2.1-or-later */
+/*
+ * Copyright (C) 2020, Google Inc.
+ *
+ * flags.cpp - Type-safe enum-based bitfields
+ */
+
+#include <libcamera/flags.h>
+
+/**
+ * \file flags.h
+ * \brief Enum-based bit fields
+ */
+
+namespace libcamera {
+
+/**
+ * \class Flags
+ * \brief Type-safe container for enum-based bitfields
+ *
+ * The Flags template class provides type-safe bitwise operators on enum values.
+ * It allows using enum types for bitfields, while preventing unsafe casts from
+ * integer types and mixing of flags from different enum types.
+ *
+ * To use the Flags class, declare an enum containing the desired bit flags, and
+ * use the Flags<enum> class to store bitfields based on the enum. If bitwise
+ * operators on the underlying enum are also desired, they can be enabled with
+ * the LIBCAMERA_FLAGS_ENABLE_OPERATORS(enum) macro.
+ */
+
+/**
+ * \typedef Flags::Type
+ * \brief The underlying data type of the enum
+ */
+
+/**
+ * \fn Flags::Flags()
+ * \brief Construct a Flags instance with a zero value
+ */
+
+/**
+ * \fn Flags::Flags(E flag)
+ * \brief Construct a Flags instance storing the \a flag
+ * \param[in] flag The initial value
+ */
+
+/**
+ * \fn Flags &Flags::operator&=(E flag)
+ * \brief Store the bitwise AND of this Flags and the \a flag in this Flags
+ * \param[in] flag The second operand
+ * \return A reference to this Flags
+ */
+
+/**
+ * \fn Flags &Flags::operator&=(Flags other)
+ * \brief Store the bitwise AND of this Flags and the \a other Flags in this Flags
+ * \param[in] other The second operand
+ * \return A reference to this Flags
+ */
+
+/**
+ * \fn Flags &Flags::operator|=(E flag)
+ * \brief Store the bitwise OR of this Flags and the \a flag in this Flags
+ * \param[in] flag The second operand
+ * \return A reference to this Flags
+ */
+
+/**
+ * \fn Flags &Flags::operator|=(Flags other)
+ * \brief Store the bitwise OR of this Flags and the \a other Flags in this Flags
+ * \param[in] other The second operand
+ * \return A reference to this Flags
+ */
+
+/**
+ * \fn Flags &Flags::operator^=(E flag)
+ * \brief Store the bitwise XOR of this Flags and the \a flag in this Flags
+ * \param[in] flag The second operand
+ * \return A reference to this Flags
+ */
+
+/**
+ * \fn Flags &Flags::operator^=(Flags other)
+ * \brief Store the bitwise XOR of this Flags and the \a other Flags in this Flags
+ * \param[in] other The second operand
+ * \return A reference to this Flags
+ */
+
+/**
+ * \fn bool Flags::operator==(E flag)
+ * \brief Compare flags for equality
+ * \param[in] flag The second operand
+ * \return True if the Flags and \a flag are equal, false otherwise
+ */
+
+/**
+ * \fn bool Flags::operator==(Flags other)
+ * \brief Compare flags for equality
+ * \param[in] other The second operand
+ * \return True if the Flags and \a other are equal, false otherwise
+ */
+
+/**
+ * \fn bool Flags::operator!=(E flag)
+ * \brief Compare flags for non-equality
+ * \param[in] flag The second operand
+ * \return True if the Flags and \a flag are not equal, false otherwise
+ */
+
+/**
+ * \fn bool Flags::operator!=(Flags other)
+ * \brief Compare flags for non-equality
+ * \param[in] other The second operand
+ * \return True if the Flags and \a other are not equal, false otherwise
+ */
+
+/**
+ * \fn Flags::operator Type() const
+ * \brief Convert the Flags to the underlying integer type
+ * \return The Flags value as an integer
+ */
+
+/**
+ * \fn Flags::operator bool() const
+ * \brief Convert the Flags to a boolean
+ * \return True if at least one flag is set, false otherwise
+ */
+
+/**
+ * \fn Flags Flags::operator&(E flag) const
+ * \brief Compute the bitwise AND of this Flags and the \a flag
+ * \param[in] flag The second operand
+ * \return A Flags containing the result of the AND operation
+ */
+
+/**
+ * \fn Flags Flags::operator&(Flags other) const
+ * \brief Compute the bitwise AND of this Flags and the \a other Flags
+ * \param[in] other The second operand
+ * \return A Flags containing the result of the AND operation
+ */
+
+/**
+ * \fn Flags Flags::operator|(E flag) const
+ * \brief Compute the bitwise OR of this Flags and the \a flag
+ * \param[in] flag The second operand
+ * \return A Flags containing the result of the OR operation
+ */
+
+/**
+ * \fn Flags Flags::operator|(Flags other) const
+ * \brief Compute the bitwise OR of this Flags and the \a other Flags
+ * \param[in] other The second operand
+ * \return A Flags containing the result of the OR operation
+ */
+
+/**
+ * \fn Flags Flags::operator^(E flag) const
+ * \brief Compute the bitwise XOR of this Flags and the \a flag
+ * \param[in] flag The second operand
+ * \return A Flags containing the result of the XOR operation
+ */
+
+/**
+ * \fn Flags Flags::operator^(Flags other) const
+ * \brief Compute the bitwise XOR of this Flags and the \a other Flags
+ * \param[in] other The second operand
+ * \return A Flags containing the result of the XOR operation
+ */
+
+/**
+ * \fn Flags Flags::operator~() const
+ * \brief Compute the bitwise NOT of this Flags
+ * \return A Flags containing the result of the NOT operation
+ */
+
+/**
+ * \fn bool Flags::operator!() const
+ * \brief Check if flags are set
+ * \return True if no flags is set, false otherwise
+ */
+
+/**
+ * \def LIBCAMERA_FLAGS_ENABLE_OPERATORS(enum)
+ * \brief Enable bitwise operations on the \a enum enumeration
+ *
+ * This macro enables the bitwise AND, OR, XOR and NOT operators on the given
+ * \a enum. This allows the enum values to be safely used in bitwise operations
+ * with the Flags<> class.
+ */
+
+} /* namespace libcamera */
diff --git a/src/libcamera/meson.build b/src/libcamera/meson.build
index 3aad4386ffc2..2461200fa772 100644
--- a/src/libcamera/meson.build
+++ b/src/libcamera/meson.build
@@ -18,6 +18,7 @@ libcamera_sources = files([
'event_notifier.cpp',
'file.cpp',
'file_descriptor.cpp',
+ 'flags.cpp',
'formats.cpp',
'framebuffer_allocator.cpp',
'geometry.cpp',
--
Regards,
Laurent Pinchart
More information about the libcamera-devel
mailing list