Add FlagsValueType

This commit is contained in:
Sebastian Messmer 2018-09-14 01:31:38 -07:00
parent dbfd63e8d2
commit ac63b5af85
3 changed files with 126 additions and 1 deletions

View File

@ -47,6 +47,15 @@ namespace value_type {
* - val % scalar (returns val) * - val % scalar (returns val)
* - val / val (returns scalar) * - val / val (returns scalar)
* - val % val (returns scalar) * - val % val (returns scalar)
*
* FlagsValueType: Use this if you want a value type for bitfields (i.e. flags).
* Additionally to what IdValueType offers, this also defines:
* - val |= val (returns val)
* - val &= val (returns val)
* - val ^= val (returns val)
* - val | val (returns val)
* - val & val (returns val)
* - val ^ val (returns val)
*/ */
template <class ConcreteType, class UnderlyingType> template <class ConcreteType, class UnderlyingType>
class IdValueType { class IdValueType {
@ -209,6 +218,46 @@ private:
} }
}; };
template <class ConcreteType, class UnderlyingType>
class FlagsValueType : public IdValueType<ConcreteType, UnderlyingType> {
protected:
using IdValueType<ConcreteType, UnderlyingType>::IdValueType;
public:
constexpr ConcreteType& operator&=(ConcreteType rhs) noexcept(noexcept(*std::declval<UnderlyingType*>() &= std::declval<UnderlyingType>())) {
this->value_ &= rhs.value_;
return *static_cast<ConcreteType*>(this);
}
constexpr ConcreteType& operator|=(ConcreteType rhs) noexcept(noexcept(*std::declval<UnderlyingType*>() |= std::declval<UnderlyingType>())) {
this->value_ |= rhs.value_;
return *static_cast<ConcreteType*>(this);
}
constexpr ConcreteType& operator^=(ConcreteType rhs) noexcept(noexcept(*std::declval<UnderlyingType*>() ^= std::declval<UnderlyingType>())) {
this->value_ ^= rhs.value_;
return *static_cast<ConcreteType*>(this);
}
constexpr ConcreteType operator~() noexcept(noexcept(~*std::declval<UnderlyingType*>())) {
return ConcreteType(~this->value_);
}
friend constexpr ConcreteType operator&(ConcreteType lhs, ConcreteType rhs) noexcept(noexcept(std::declval<ConcreteType>() &= std::declval<ConcreteType>())) {
return lhs &= rhs;
}
friend constexpr ConcreteType operator|(ConcreteType lhs, ConcreteType rhs) noexcept(noexcept(std::declval<ConcreteType>() |= std::declval<ConcreteType>())) {
return lhs |= rhs;
}
friend constexpr ConcreteType operator^(ConcreteType lhs, ConcreteType rhs) noexcept(noexcept(std::declval<ConcreteType>() ^= std::declval<ConcreteType>())) {
return lhs ^= rhs;
}
};
} }
} }

View File

@ -3,6 +3,7 @@
#define MESSMER_FSPP_FSINTERFACE_DIR_H_ #define MESSMER_FSPP_FSINTERFACE_DIR_H_
#include <cpp-utils/pointer/unique_ref.h> #include <cpp-utils/pointer/unique_ref.h>
#include <cpp-utils/value_type/ValueType.h>
#include <string> #include <string>
#include <boost/filesystem/path.hpp> #include <boost/filesystem/path.hpp>

View File

@ -7,6 +7,7 @@
using cpputils::value_type::IdValueType; using cpputils::value_type::IdValueType;
using cpputils::value_type::OrderedIdValueType; using cpputils::value_type::OrderedIdValueType;
using cpputils::value_type::QuantityValueType; using cpputils::value_type::QuantityValueType;
using cpputils::value_type::FlagsValueType;
namespace { namespace {
@ -22,10 +23,15 @@ struct MyQuantityValueType : QuantityValueType<MyQuantityValueType, int64_t> {
constexpr explicit MyQuantityValueType(int64_t val): QuantityValueType(val) {} constexpr explicit MyQuantityValueType(int64_t val): QuantityValueType(val) {}
}; };
struct MyFlagsValueType : FlagsValueType<MyFlagsValueType, int64_t> {
constexpr explicit MyFlagsValueType(int64_t val): FlagsValueType(val) {}
};
} }
DEFINE_HASH_FOR_VALUE_TYPE(MyIdValueType); DEFINE_HASH_FOR_VALUE_TYPE(MyIdValueType);
DEFINE_HASH_FOR_VALUE_TYPE(MyOrderedIdValueType); DEFINE_HASH_FOR_VALUE_TYPE(MyOrderedIdValueType);
DEFINE_HASH_FOR_VALUE_TYPE(MyQuantityValueType); DEFINE_HASH_FOR_VALUE_TYPE(MyQuantityValueType);
DEFINE_HASH_FOR_VALUE_TYPE(MyFlagsValueType);
namespace { namespace {
/** /**
@ -57,6 +63,7 @@ struct IdValueTypeTest_constexpr_test {
static_assert(IdValueTypeTest_constexpr_test<MyIdValueType>::success, ""); static_assert(IdValueTypeTest_constexpr_test<MyIdValueType>::success, "");
static_assert(IdValueTypeTest_constexpr_test<MyOrderedIdValueType>::success, ""); static_assert(IdValueTypeTest_constexpr_test<MyOrderedIdValueType>::success, "");
static_assert(IdValueTypeTest_constexpr_test<MyQuantityValueType>::success, ""); static_assert(IdValueTypeTest_constexpr_test<MyQuantityValueType>::success, "");
static_assert(IdValueTypeTest_constexpr_test<MyFlagsValueType>::success, "");
namespace IdValueTypeTest_constexpr_test_extras { namespace IdValueTypeTest_constexpr_test_extras {
// For some reason, MSVC crashes when these are part of IdValueTypeTest_constexpr_test. // For some reason, MSVC crashes when these are part of IdValueTypeTest_constexpr_test.
@ -67,12 +74,14 @@ namespace IdValueTypeTest_constexpr_test_extras {
static_assert(MyOrderedIdValueType(5) != MyOrderedIdValueType(6), ""); static_assert(MyOrderedIdValueType(5) != MyOrderedIdValueType(6), "");
static_assert(!(MyQuantityValueType(5) == MyQuantityValueType(6)), ""); static_assert(!(MyQuantityValueType(5) == MyQuantityValueType(6)), "");
static_assert(MyQuantityValueType(5) != MyQuantityValueType(6), ""); static_assert(MyQuantityValueType(5) != MyQuantityValueType(6), "");
static_assert(!(MyFlagsValueType(5) == MyFlagsValueType(6)), "");
static_assert(MyFlagsValueType(5) != MyFlagsValueType(6), "");
} }
template<class Type> class IdValueTypeTest : public testing::Test { template<class Type> class IdValueTypeTest : public testing::Test {
}; };
using IdValueTypeTest_types = testing::Types<MyIdValueType, MyOrderedIdValueType, MyQuantityValueType>; using IdValueTypeTest_types = testing::Types<MyIdValueType, MyOrderedIdValueType, MyQuantityValueType, MyFlagsValueType>;
TYPED_TEST_CASE(IdValueTypeTest, IdValueTypeTest_types); TYPED_TEST_CASE(IdValueTypeTest, IdValueTypeTest_types);
@ -397,4 +406,70 @@ TYPED_TEST(QuantityValueTypeTest, ModValue) {
EXPECT_EQ(1, TypeParam(7) % TypeParam(3)); EXPECT_EQ(1, TypeParam(7) % TypeParam(3));
} }
/**
* Tests for FlagsValueType
*/
namespace FlagsValueTypeTest_constexpr_test {
static_assert(~MyFlagsValueType(3) != MyFlagsValueType(3), "");
static_assert(~~MyFlagsValueType(3) == MyFlagsValueType(3), "");
static_assert(~MyFlagsValueType(3) == MyFlagsValueType(~3), "");
static_assert((MyFlagsValueType(3) & MyFlagsValueType(5)) == MyFlagsValueType(3 & 5), "");
static_assert((MyFlagsValueType(3) | MyFlagsValueType(5)) == MyFlagsValueType(3 | 5), "");
static_assert((MyFlagsValueType(3) ^ MyFlagsValueType(5)) == MyFlagsValueType(3 ^ 5), "");
static_assert((MyFlagsValueType(3) &= MyFlagsValueType(5)) == MyFlagsValueType(3 & 5), "");
static_assert((MyFlagsValueType(3) |= MyFlagsValueType(5)) == MyFlagsValueType(3 | 5), "");
static_assert((MyFlagsValueType(3) ^= MyFlagsValueType(5)) == MyFlagsValueType(3 ^ 5), "");
}
template<class Type> class FlagsValueTypeTest : public testing::Test {};
using FlagsValueType_types = testing::Types<MyFlagsValueType>;
TYPED_TEST_CASE(FlagsValueTypeTest, FlagsValueType_types);
TYPED_TEST(FlagsValueTypeTest, Invert) {
TypeParam a(3);
TypeParam b(~3);
EXPECT_EQ(b, ~a);
a = ~a;
EXPECT_EQ(b, a);
}
TYPED_TEST(FlagsValueTypeTest, And) {
TypeParam a(3);
TypeParam b(5);
TypeParam c(3 & 5);
EXPECT_EQ(c, a & b);
EXPECT_EQ(c, b &= a);
EXPECT_EQ(c, b);
}
TYPED_TEST(FlagsValueTypeTest, Or) {
TypeParam a(3);
TypeParam b(5);
TypeParam c(3 | 5);
EXPECT_EQ(c, a | b);
EXPECT_EQ(c, b |= a);
EXPECT_EQ(c, b);
}
TYPED_TEST(FlagsValueTypeTest, Xor) {
TypeParam a(3);
TypeParam b(5);
TypeParam c(3 ^ 5);
EXPECT_EQ(c, a ^ b);
EXPECT_EQ(c, b ^= a);
EXPECT_EQ(c, b);
}
} }