// // Copyright 2015 The ANGLE Project Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // // bitset_utils_unittest: // Tests bitset helpers and custom classes. // #include #include "common/bitset_utils.h" using namespace angle; namespace { template class BitSetTest : public testing::Test { protected: T mBits; typedef T BitSet; }; using BitSetTypes = ::testing::Types, BitSet32<12>, BitSet64<12>>; TYPED_TEST_SUITE(BitSetTest, BitSetTypes); TYPED_TEST(BitSetTest, Basic) { TypeParam mBits = this->mBits; EXPECT_FALSE(mBits.all()); EXPECT_FALSE(mBits.any()); EXPECT_TRUE(mBits.none()); EXPECT_EQ(mBits.count(), 0u); // Set every bit to 1. for (size_t i = 0; i < mBits.size(); ++i) { mBits.set(i); EXPECT_EQ(mBits.all(), i + 1 == mBits.size()); EXPECT_TRUE(mBits.any()); EXPECT_FALSE(mBits.none()); EXPECT_EQ(mBits.count(), i + 1); } // Reset every other bit to 0. for (size_t i = 0; i < mBits.size(); i += 2) { mBits.reset(i); EXPECT_FALSE(mBits.all()); EXPECT_TRUE(mBits.any()); EXPECT_FALSE(mBits.none()); EXPECT_EQ(mBits.count(), mBits.size() - i / 2 - 1); } // Flip all bits. for (size_t i = 0; i < mBits.size(); ++i) { mBits.flip(i); EXPECT_FALSE(mBits.all()); EXPECT_TRUE(mBits.any()); EXPECT_FALSE(mBits.none()); EXPECT_EQ(mBits.count(), mBits.size() / 2 + (i % 2 == 0)); } // Make sure the bit pattern is what we expect at this point. for (size_t i = 0; i < mBits.size(); ++i) { EXPECT_EQ(mBits.test(i), i % 2 == 0); EXPECT_EQ(static_cast(mBits[i]), i % 2 == 0); } // Test that flip, set and reset all bits at once work. mBits.flip(); EXPECT_FALSE(mBits.all()); EXPECT_TRUE(mBits.any()); EXPECT_FALSE(mBits.none()); EXPECT_EQ(mBits.count(), mBits.size() / 2); mBits.set(); EXPECT_TRUE(mBits.all()); EXPECT_TRUE(mBits.any()); EXPECT_FALSE(mBits.none()); EXPECT_EQ(mBits.count(), mBits.size()); mBits.reset(); EXPECT_FALSE(mBits.all()); EXPECT_FALSE(mBits.any()); EXPECT_TRUE(mBits.none()); EXPECT_EQ(mBits.count(), 0u); // Test that out-of-bound sets don't modify the bitset constexpr uint32_t kMask = (1 << 12) - 1; EXPECT_EQ(mBits.set(12).bits() & ~kMask, 0u); EXPECT_EQ(mBits.set(13).bits() & ~kMask, 0u); EXPECT_EQ(mBits.flip(12).bits() & ~kMask, 0u); EXPECT_EQ(mBits.flip(13).bits() & ~kMask, 0u); } TYPED_TEST(BitSetTest, BitwiseOperators) { TypeParam mBits = this->mBits; // Use a value that has a 1 in the 12th and 13th bits, to make sure masking to exactly 12 bits // does not have an off-by-one error. constexpr uint32_t kSelfValue = 0xF9E4; constexpr uint32_t kOtherValue = 0x5C6A; constexpr uint32_t kMask = (1 << 12) - 1; constexpr uint32_t kSelfMaskedValue = kSelfValue & kMask; constexpr uint32_t kOtherMaskedValue = kOtherValue & kMask; constexpr uint32_t kShift = 3; constexpr uint32_t kSelfShiftedLeft = kSelfMaskedValue << kShift & kMask; constexpr uint32_t kSelfShiftedRight = kSelfMaskedValue >> kShift & kMask; mBits |= kSelfValue; typename TestFixture::BitSet other(kOtherValue); typename TestFixture::BitSet anded(kSelfMaskedValue & kOtherMaskedValue); typename TestFixture::BitSet ored(kSelfMaskedValue | kOtherMaskedValue); typename TestFixture::BitSet xored(kSelfMaskedValue ^ kOtherMaskedValue); EXPECT_EQ(mBits.bits(), kSelfMaskedValue); EXPECT_EQ(other.bits(), kOtherMaskedValue); EXPECT_EQ(mBits & other, anded); EXPECT_EQ(mBits | other, ored); EXPECT_EQ(mBits ^ other, xored); EXPECT_NE(mBits, other); EXPECT_NE(anded, ored); EXPECT_NE(anded, xored); EXPECT_NE(ored, xored); mBits &= other; EXPECT_EQ(mBits, anded); mBits |= ored; EXPECT_EQ(mBits, ored); mBits ^= other; mBits ^= anded; EXPECT_EQ(mBits, typename TestFixture::BitSet(kSelfValue)); EXPECT_EQ(mBits << kShift, typename TestFixture::BitSet(kSelfShiftedLeft)); EXPECT_EQ(mBits >> kShift, typename TestFixture::BitSet(kSelfShiftedRight)); mBits <<= kShift; EXPECT_EQ(mBits, typename TestFixture::BitSet(kSelfShiftedLeft)); EXPECT_EQ(mBits.bits() & ~kMask, 0u); mBits = typename TestFixture::BitSet(kSelfValue); mBits >>= kShift; EXPECT_EQ(mBits, typename TestFixture::BitSet(kSelfShiftedRight)); EXPECT_EQ(mBits.bits() & ~kMask, 0u); mBits |= kSelfMaskedValue; EXPECT_EQ(mBits.bits() & ~kMask, 0u); mBits ^= kOtherMaskedValue; EXPECT_EQ(mBits.bits() & ~kMask, 0u); } template class BitSetIteratorTest : public testing::Test { protected: T mStateBits; typedef T BitSet; }; using BitSetIteratorTypes = ::testing::Types, BitSet64<40>>; TYPED_TEST_SUITE(BitSetIteratorTest, BitSetIteratorTypes); // Simple iterator test. TYPED_TEST(BitSetIteratorTest, Iterator) { TypeParam mStateBits = this->mStateBits; std::set originalValues; originalValues.insert(2); originalValues.insert(6); originalValues.insert(8); originalValues.insert(35); for (size_t value : originalValues) { mStateBits.set(value); } std::set readValues; for (size_t bit : mStateBits) { EXPECT_EQ(1u, originalValues.count(bit)); EXPECT_EQ(0u, readValues.count(bit)); readValues.insert(bit); } EXPECT_EQ(originalValues.size(), readValues.size()); } // Test an empty iterator. TYPED_TEST(BitSetIteratorTest, EmptySet) { TypeParam mStateBits = this->mStateBits; // We don't use the FAIL gtest macro here since it returns immediately, // causing an unreachable code warning in MSVS bool sawBit = false; for (size_t bit : mStateBits) { sawBit = true; ANGLE_UNUSED_VARIABLE(bit); } EXPECT_FALSE(sawBit); } // Test iterating a result of combining two bitsets. TYPED_TEST(BitSetIteratorTest, NonLValueBitset) { TypeParam mStateBits = this->mStateBits; typename TestFixture::BitSet otherBits; mStateBits.set(1); mStateBits.set(2); mStateBits.set(3); mStateBits.set(4); otherBits.set(0); otherBits.set(1); otherBits.set(3); otherBits.set(5); std::set seenBits; typename TestFixture::BitSet maskedBits = (mStateBits & otherBits); for (size_t bit : maskedBits) { EXPECT_EQ(0u, seenBits.count(bit)); seenBits.insert(bit); EXPECT_TRUE(mStateBits[bit]); EXPECT_TRUE(otherBits[bit]); } EXPECT_EQ((mStateBits & otherBits).count(), seenBits.size()); } // Test bit assignments. TYPED_TEST(BitSetIteratorTest, BitAssignment) { TypeParam mStateBits = this->mStateBits; std::set originalValues; originalValues.insert(2); originalValues.insert(6); originalValues.insert(8); originalValues.insert(35); for (size_t value : originalValues) { (mStateBits[value] = false) = true; } for (size_t value : originalValues) { EXPECT_TRUE(mStateBits.test(value)); } } // Tests adding bits to the iterator during iteration. TYPED_TEST(BitSetIteratorTest, SetLaterBit) { TypeParam mStateBits = this->mStateBits; std::set expectedValues = {1, 3, 5, 7, 9}; mStateBits.set(1); std::set actualValues; for (auto iter = mStateBits.begin(), end = mStateBits.end(); iter != end; ++iter) { if (*iter == 1) { iter.setLaterBit(3); iter.setLaterBit(5); iter.setLaterBit(7); iter.setLaterBit(9); } actualValues.insert(*iter); } EXPECT_EQ(expectedValues, actualValues); } // Tests removing bits from the iterator during iteration. TYPED_TEST(BitSetIteratorTest, ResetLaterBit) { TypeParam mStateBits = this->mStateBits; std::set expectedValues = {1, 3, 5, 7, 9}; for (size_t index = 1; index <= 9; ++index) mStateBits.set(index); std::set actualValues; for (auto iter = mStateBits.begin(), end = mStateBits.end(); iter != end; ++iter) { if (*iter == 1) { iter.resetLaterBit(2); iter.resetLaterBit(4); iter.resetLaterBit(6); iter.resetLaterBit(8); } actualValues.insert(*iter); } EXPECT_EQ(expectedValues, actualValues); } } // anonymous namespace