Allow specifying StorageType for BitField
This is useful for BitFields that are bools.
This commit is contained in:
parent
6653bd7199
commit
cf95deaf6d
|
@ -110,7 +110,13 @@
|
||||||
* symptoms.
|
* symptoms.
|
||||||
*/
|
*/
|
||||||
#pragma pack(1)
|
#pragma pack(1)
|
||||||
template <std::size_t position, std::size_t bits, typename T>
|
template <std::size_t position, std::size_t bits, typename T,
|
||||||
|
// StorageType is T for non-enum types and the underlying type of T if
|
||||||
|
// T is an enumeration. Note that T is wrapped within an enable_if in the
|
||||||
|
// former case to workaround compile errors which arise when using
|
||||||
|
// std::underlying_type<T>::type directly.
|
||||||
|
typename StorageType = typename std::conditional_t<
|
||||||
|
std::is_enum<T>::value, std::underlying_type<T>, std::enable_if<true, T>>::type>
|
||||||
struct BitField
|
struct BitField
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
@ -149,20 +155,13 @@ public:
|
||||||
constexpr std::size_t NumBits() const { return bits; }
|
constexpr std::size_t NumBits() const { return bits; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// StorageType is T for non-enum types and the underlying type of T if
|
|
||||||
// T is an enumeration. Note that T is wrapped within an enable_if in the
|
|
||||||
// former case to workaround compile errors which arise when using
|
|
||||||
// std::underlying_type<T>::type directly.
|
|
||||||
using StorageType = typename std::conditional_t<std::is_enum<T>::value, std::underlying_type<T>,
|
|
||||||
std::enable_if<true, T>>::type;
|
|
||||||
|
|
||||||
// Unsigned version of StorageType
|
// Unsigned version of StorageType
|
||||||
using StorageTypeU = std::make_unsigned_t<StorageType>;
|
using StorageTypeU = std::make_unsigned_t<StorageType>;
|
||||||
|
|
||||||
constexpr T Value(std::true_type) const
|
constexpr T Value(std::true_type) const
|
||||||
{
|
{
|
||||||
using shift_amount = std::integral_constant<size_t, 8 * sizeof(T) - bits>;
|
const size_t shift_amount = 8 * sizeof(StorageType) - bits;
|
||||||
return static_cast<T>((storage << (shift_amount() - position)) >> shift_amount());
|
return static_cast<T>((storage << (shift_amount - position)) >> shift_amount);
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr T Value(std::false_type) const
|
constexpr T Value(std::false_type) const
|
||||||
|
@ -172,15 +171,17 @@ private:
|
||||||
|
|
||||||
static constexpr StorageType GetMask()
|
static constexpr StorageType GetMask()
|
||||||
{
|
{
|
||||||
return (std::numeric_limits<StorageTypeU>::max() >> (8 * sizeof(T) - bits)) << position;
|
return (std::numeric_limits<StorageTypeU>::max() >> (8 * sizeof(StorageType) - bits))
|
||||||
|
<< position;
|
||||||
}
|
}
|
||||||
|
|
||||||
StorageType storage;
|
StorageType storage;
|
||||||
|
|
||||||
static_assert(bits + position <= 8 * sizeof(T), "Bitfield out of range");
|
static_assert(bits + position <= 8 * sizeof(StorageType), "Bitfield out of range");
|
||||||
|
static_assert(sizeof(T) <= sizeof(StorageType), "T must fit in StorageType");
|
||||||
|
|
||||||
// And, you know, just in case people specify something stupid like bits=position=0x80000000
|
// And, you know, just in case people specify something stupid like bits=position=0x80000000
|
||||||
static_assert(position < 8 * sizeof(T), "Invalid position");
|
static_assert(position < 8 * sizeof(StorageType), "Invalid position");
|
||||||
static_assert(bits <= 8 * sizeof(T), "Invalid number of bits");
|
static_assert(bits <= 8 * sizeof(T), "Invalid number of bits");
|
||||||
static_assert(bits > 0, "Invalid number of bits");
|
static_assert(bits > 0, "Invalid number of bits");
|
||||||
};
|
};
|
||||||
|
|
|
@ -21,6 +21,8 @@ union TestUnion
|
||||||
BitField<30, 4, s64> at_dword_boundary; // goes over the boundary of two u32 values
|
BitField<30, 4, s64> at_dword_boundary; // goes over the boundary of two u32 values
|
||||||
|
|
||||||
BitField<15, 1, s64> signed_1bit; // allowed values: -1 and 0
|
BitField<15, 1, s64> signed_1bit; // allowed values: -1 and 0
|
||||||
|
|
||||||
|
BitField<63, 1, bool, u64> flag;
|
||||||
};
|
};
|
||||||
|
|
||||||
// table of raw numbers to test with
|
// table of raw numbers to test with
|
||||||
|
@ -51,6 +53,7 @@ TEST(BitField, Storage)
|
||||||
EXPECT_EQ(sizeof(TestUnion), sizeof(object.regular_field_signed));
|
EXPECT_EQ(sizeof(TestUnion), sizeof(object.regular_field_signed));
|
||||||
EXPECT_EQ(sizeof(TestUnion), sizeof(object.at_dword_boundary));
|
EXPECT_EQ(sizeof(TestUnion), sizeof(object.at_dword_boundary));
|
||||||
EXPECT_EQ(sizeof(TestUnion), sizeof(object.signed_1bit));
|
EXPECT_EQ(sizeof(TestUnion), sizeof(object.signed_1bit));
|
||||||
|
EXPECT_EQ(sizeof(TestUnion), sizeof(object.flag));
|
||||||
|
|
||||||
// Now write some values to one field and check if this reflects properly
|
// Now write some values to one field and check if this reflects properly
|
||||||
// in the others.
|
// in the others.
|
||||||
|
@ -82,6 +85,7 @@ TEST(BitField, Read)
|
||||||
EXPECT_EQ(object.regular_field_signed, (s64)object.regular_field_signed);
|
EXPECT_EQ(object.regular_field_signed, (s64)object.regular_field_signed);
|
||||||
EXPECT_EQ(object.at_dword_boundary, (s64)object.at_dword_boundary);
|
EXPECT_EQ(object.at_dword_boundary, (s64)object.at_dword_boundary);
|
||||||
EXPECT_EQ(object.signed_1bit, (s64)object.signed_1bit);
|
EXPECT_EQ(object.signed_1bit, (s64)object.signed_1bit);
|
||||||
|
EXPECT_EQ(object.flag, (bool)object.flag);
|
||||||
|
|
||||||
// Now make sure the value is indeed correct
|
// Now make sure the value is indeed correct
|
||||||
EXPECT_EQ(val, object.full_u64);
|
EXPECT_EQ(val, object.full_u64);
|
||||||
|
@ -91,6 +95,7 @@ TEST(BitField, Read)
|
||||||
EXPECT_EQ(((s64)(object.hex << 52)) >> 61, object.regular_field_signed);
|
EXPECT_EQ(((s64)(object.hex << 52)) >> 61, object.regular_field_signed);
|
||||||
EXPECT_EQ(((s64)(object.hex << 30)) >> 60, object.at_dword_boundary);
|
EXPECT_EQ(((s64)(object.hex << 30)) >> 60, object.at_dword_boundary);
|
||||||
EXPECT_EQ(((object.hex >> 15) & 1) ? -1 : 0, object.signed_1bit);
|
EXPECT_EQ(((object.hex >> 15) & 1) ? -1 : 0, object.signed_1bit);
|
||||||
|
EXPECT_EQ((bool)object.flag, ((object.hex >> 63) & 1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -122,6 +127,10 @@ TEST(BitField, Assignment)
|
||||||
// Assignment from other BitField
|
// Assignment from other BitField
|
||||||
object.at_dword_boundary = object.regular_field_signed;
|
object.at_dword_boundary = object.regular_field_signed;
|
||||||
EXPECT_EQ(object.regular_field_signed, object.at_dword_boundary);
|
EXPECT_EQ(object.regular_field_signed, object.at_dword_boundary);
|
||||||
|
|
||||||
|
// Assignment to field of a type with a size smaller than the underlying type
|
||||||
|
object.flag = (val & 2);
|
||||||
|
EXPECT_EQ(object.flag, (val & 2) != 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -165,5 +174,9 @@ TEST(BitField, Alignment)
|
||||||
// Assignment from other BitField
|
// Assignment from other BitField
|
||||||
object.at_dword_boundary = object.regular_field_signed;
|
object.at_dword_boundary = object.regular_field_signed;
|
||||||
EXPECT_EQ(object.regular_field_signed, object.at_dword_boundary);
|
EXPECT_EQ(object.regular_field_signed, object.at_dword_boundary);
|
||||||
|
|
||||||
|
// Assignment to field of a type with a size smaller than the underlying type
|
||||||
|
object.flag = (val & 2);
|
||||||
|
EXPECT_EQ(object.flag, (val & 2) != 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue