#ifndef Magnum_Math_BoolVector_h #define Magnum_Math_BoolVector_h /* This file is part of Magnum. Copyright © 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019 Vladimír Vondruš Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /** @file * @brief Class @ref Magnum::Math::BoolVector */ #include #ifndef CORRADE_NO_DEBUG #include #endif #include "Magnum/Types.h" #include "Magnum/Math/Math.h" #include "Magnum/Math/Tags.h" namespace Magnum { namespace Math { namespace Implementation { template struct BoolVectorConverter; /** @todo C++14: use std::make_index_sequence and std::integer_sequence */ template struct Sequence {}; #ifndef DOXYGEN_GENERATING_OUTPUT /* E.g. GenerateSequence<3>::Type is Sequence<0, 1, 2> */ template struct GenerateSequence: GenerateSequence {}; template struct GenerateSequence<0, sequence...> { typedef Sequence Type; }; #endif template constexpr T repeat(T value, std::size_t) { return value; } } /** @brief Vector storing boolean values @tparam size Bit count Result of component-wise comparison from @ref Vector. The boolean values are stored as bits in array of unsigned bytes, unused bits have undefined value which doesn't affect comparison or @ref all() / @ref none() / @ref any() functions. See also @ref matrix-vector for brief introduction. @section Math-BoolVector-indexing Bit indexing Value at position 0 is the lowest bit of the first byte passed in constructor. Value at position 8 is the lowest bit of the second byte passed in constructor. For example: @snippet MagnumMath-cpp14.cpp BoolVector-indexing @section Math-BoolVector-boolean Boolean operations The class implements @cpp && @ce, @cpp || @ce and @cpp ! @ce operators component-wise, in other words equivalently to @cpp & @ce, @cpp | @ce and @cpp ~ @ce. This is done in order to have consistent behavior with boolean operations on scalar types --- in the following example, it causes the final conversion to @cpp bool @ce done at the end (instead of it happening already in the boolean subexpressions). Combined with @ref operator bool() returning @cpp true @ce only if all bits are set, this means the condition will be passed only if @cpp b @ce is around @cpp a @ce in *all dimensions*, and work the same way as if the variables were just scalars: @snippet MagnumMath.cpp BoolVector-boolean @see @ref Magnum::BoolVector2, @ref Magnum::BoolVector3, @ref Magnum::BoolVector4 */ template class BoolVector { static_assert(size != 0, "BoolVector cannot have zero elements"); public: enum: std::size_t { Size = size, /**< Vector size */ DataSize = (size-1)/8+1 /**< Vector storage size */ }; /** * @brief Default constructor * * Equivalent to @ref BoolVector(ZeroInitT). */ constexpr /*implicit*/ BoolVector() noexcept: _data{} {} /** @brief Construct a zero-filled boolean vector */ constexpr explicit BoolVector(ZeroInitT) noexcept: _data{} {} /** @brief Construct without initializing the contents */ explicit BoolVector(NoInitT) noexcept {} /** * @brief Construct a boolean vector from segment values * @param first Value for first 8bit segment * @param next Values for next Bbit segments */ #ifdef DOXYGEN_GENERATING_OUTPUT template constexpr /*implicit*/ BoolVector(UnsignedByte first, T... next) noexcept; #else template::type> constexpr /*implicit*/ BoolVector(UnsignedByte first, T... next) noexcept: _data{first, UnsignedByte(next)...} {} #endif /** @brief Construct a boolean vector with one value for all fields */ #ifdef DOXYGEN_GENERATING_OUTPUT explicit BoolVector(T value) noexcept; #else template::value && size != 1, bool>::type> constexpr explicit BoolVector(T value) noexcept: BoolVector(typename Implementation::GenerateSequence::Type(), value ? FullSegmentMask : 0) {} #endif /** @brief Construct a boolean vector from external representation */ template::from(std::declval()))> constexpr explicit BoolVector(const U& other) noexcept: BoolVector{Implementation::BoolVectorConverter::from(other)} {} /** @brief Copy constructor */ constexpr /*implicit*/ BoolVector(const BoolVector&) noexcept = default; /** @brief Convert a boolean vector to external representation */ template::to(std::declval>()))> constexpr explicit operator U() const { return Implementation::BoolVectorConverter::to(*this); } /** * @brief Raw data * @return Array of DataSize length * * @see @ref operator[](), @ref set() */ UnsignedByte* data() { return _data; } constexpr const UnsignedByte* data() const { return _data; } /**< @overload */ /** @brief Bit at given position */ constexpr bool operator[](std::size_t i) const { return (_data[i/8] >> i%8) & 0x01; } /** @brief Set a bit at given position */ BoolVector& set(std::size_t i, bool value) { value ? _data[i/8] |= (1 << i%8) : _data[i/8] &= ~(1 << i%8); return *this; } /** @brief Equality comparison */ bool operator==(const BoolVector& other) const; /** @brief Non-equality comparison */ bool operator!=(const BoolVector& other) const { return !operator==(other); } /** * @brief Boolean conversion * * Equivalent to @ref all(). * @see @ref any(), @ref none() */ explicit operator bool() const { return all(); } /** * @brief Whether all bits are set * * @see @ref none(), @ref any(), @ref operator bool() */ bool all() const; /** * @brief Whether no bits are set * * @see @ref all(), @ref any(), @ref operator bool() */ bool none() const; /** * @brief Whether any bit is set * * @see @ref all(), @ref none(), @ref operator bool() */ bool any() const { return !none(); } /** @brief Bitwise inversion */ BoolVector operator~() const; /** * @brief Component-wise boolean negation * @m_since{2019,10} * * Equivalent to @ref operator~(). See @ref Math-BoolVector-boolean for * more information. */ BoolVector operator!() const { return operator~(); } /** * @brief Bitwise AND and assign * * The computation is done in-place. */ BoolVector& operator&=(const BoolVector& other) { for(std::size_t i = 0; i != DataSize; ++i) _data[i] &= other._data[i]; return *this; } /** * @brief Bitwise AND * * @see @ref operator&=() */ BoolVector operator&(const BoolVector& other) const { return BoolVector(*this) &= other; } /** * @brief Component-wise boolean AND * @m_since{2019,10} * * Equivalent to @ref operator&(). See @ref Math-BoolVector-boolean for * more information. */ BoolVector operator&&(const BoolVector& other) const { return BoolVector(*this) &= other; } /** * @brief Bitwise OR and assign * * The computation is done in-place. */ BoolVector& operator|=(const BoolVector& other) { for(std::size_t i = 0; i != DataSize; ++i) _data[i] |= other._data[i]; return *this; } /** * @brief Bitwise OR * * @see @ref operator|=() */ BoolVector operator|(const BoolVector& other) const { return BoolVector(*this) |= other; } /** * @brief Component-wise boolean OR * @m_since{2019,10} * * Equivalent to @ref operator|(). See @ref Math-BoolVector-boolean for * more information. */ BoolVector operator||(const BoolVector& other) const { return BoolVector(*this) |= other; } /** * @brief Bitwise XOR and assign * * The computation is done in-place. */ BoolVector& operator^=(const BoolVector& other) { for(std::size_t i = 0; i != DataSize; ++i) _data[i] ^= other._data[i]; return *this; } /** * @brief Bitwise XOR * * @see @ref operator^=() */ BoolVector operator^(const BoolVector& other) const { return BoolVector(*this) ^= other; } private: enum: UnsignedByte { FullSegmentMask = 0xFF, LastSegmentMask = (1 << size%8) - 1 }; /* Implementation for Vector::Vector(U) */ template constexpr explicit BoolVector(Implementation::Sequence, UnsignedByte value): _data{Implementation::repeat(value, sequence)...} {} UnsignedByte _data[(size-1)/8+1]; }; #ifndef CORRADE_NO_DEBUG /** @debugoperator{BoolVector} In order to avoid potential confusion, prints the value as a comma-separated sequence of binary literals, so the output corresponds to how the value would be constructed. For example, @snippet MagnumMath-cpp14.cpp BoolVector-debug @m_class{m-noindent} prints as @code{.shell-session} BoolVector(0b1010) BoolVector(0b00001000, 0b00000011, 0b100) @endcode Note that this, on the other hand, makes mapping to bit indices less obvious --- see @ref Math-BoolVector-indexing for more information. */ template Corrade::Utility::Debug& operator<<(Corrade::Utility::Debug& debug, const BoolVector& value) { debug << "BoolVector(0b" << Corrade::Utility::Debug::nospace; /* Print the full bytes comma-separated */ for(std::size_t byte = 0; byte != BoolVector::DataSize - 1; ++byte) { for(std::size_t i = 0; i != 8; ++i) debug << (((value.data()[byte] >> (8 - i - 1)) & 1) ? "1" : "0") << Corrade::Utility::Debug::nospace; debug << ", 0b" << Corrade::Utility::Debug::nospace; } /* Print the last (potentially) partial byte */ constexpr std::size_t suffixSize = size%8 ? size%8 : 8; for(std::size_t i = 0; i != suffixSize; ++i) debug << (((value.data()[size/8] >> (suffixSize - i - 1)) & 1) ? "1" : "0") << Corrade::Utility::Debug::nospace; return debug << ")"; } #endif template inline bool BoolVector::operator==(const BoolVector& other) const { for(std::size_t i = 0; i != size/8; ++i) if(_data[i] != other._data[i]) return false; /* Check last segment */ if(size%8 && (_data[DataSize-1] & LastSegmentMask) != (other._data[DataSize-1] & LastSegmentMask)) return false; return true; } template inline bool BoolVector::all() const { /* Check all full segments */ for(std::size_t i = 0; i != size/8; ++i) if(_data[i] != FullSegmentMask) return false; /* Check last segment */ if(size%8 && (_data[DataSize-1] & LastSegmentMask) != LastSegmentMask) return false; return true; } template inline bool BoolVector::none() const { /* Check all full segments */ for(std::size_t i = 0; i != size/8; ++i) if(_data[i]) return false; /* Check last segment */ if(size%8 && (_data[DataSize-1] & LastSegmentMask)) return false; return true; } template inline BoolVector BoolVector::operator~() const { BoolVector out{NoInit}; for(std::size_t i = 0; i != DataSize; ++i) out._data[i] = ~_data[i]; return out; } namespace Implementation { template struct StrictWeakOrdering> { bool operator()(const BoolVector& a, const BoolVector& b) const { auto ad = a.data(); auto bd = b.data(); for(std::size_t i = 0; i < BoolVector::DataSize - 1; ++i) { if(ad[i] < bd[i]) return true; if(ad[i] > bd[i]) return false; } /* Mask last element with to hide padding bits */ constexpr UnsignedByte mask = UnsignedByte(0xFF) >> (BoolVector::DataSize * 8 - size); constexpr std::size_t i = BoolVector::DataSize - 1; return (ad[i] & mask) < (bd[i] & mask); } }; } }} #endif