blob: 615212b7b2deee10441d3c4f795a265b20d3e115 [file] [log] [blame] [edit]
// Copyright 2017 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include <set>
#include "dawn/common/BitSetIterator.h"
#include "dawn/common/ityp_bitset.h"
#include "gtest/gtest.h"
namespace dawn {
namespace {
// This is ANGLE's BitSetIterator_unittests.cpp file.
class BitSetIteratorTest : public testing::Test {
protected:
std::bitset<40> mStateBits;
};
// Simple iterator test.
TEST_F(BitSetIteratorTest, Iterator) {
std::set<uint32_t> originalValues;
originalValues.insert(2);
originalValues.insert(6);
originalValues.insert(8);
originalValues.insert(35);
for (uint32_t value : originalValues) {
mStateBits.set(value);
}
std::set<uint32_t> readValues;
for (uint32_t bit : IterateBitSet(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.
TEST_F(BitSetIteratorTest, EmptySet) {
// We don't use the FAIL gtest macro here since it returns immediately,
// causing an unreachable code warning in MSVC
bool sawBit = false;
for ([[maybe_unused]] uint32_t bit : IterateBitSet(mStateBits)) {
sawBit = true;
}
EXPECT_FALSE(sawBit);
}
// Test iterating a result of combining two bitsets.
TEST_F(BitSetIteratorTest, NonLValueBitset) {
std::bitset<40> 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<uint32_t> seenBits;
for (uint32_t bit : IterateBitSet(mStateBits & otherBits)) {
EXPECT_EQ(0u, seenBits.count(bit));
seenBits.insert(bit);
EXPECT_TRUE(mStateBits[bit]);
EXPECT_TRUE(otherBits[bit]);
}
EXPECT_EQ((mStateBits & otherBits).count(), seenBits.size());
}
class EnumBitSetIteratorTest : public testing::Test {
protected:
enum class TestEnum { A, B, C, D, E, F, G, H, I, J, EnumCount };
static constexpr size_t kEnumCount = static_cast<size_t>(TestEnum::EnumCount);
ityp::bitset<TestEnum, kEnumCount> mStateBits;
};
// Simple iterator test.
TEST_F(EnumBitSetIteratorTest, Iterator) {
std::set<TestEnum> originalValues;
originalValues.insert(TestEnum::B);
originalValues.insert(TestEnum::F);
originalValues.insert(TestEnum::C);
originalValues.insert(TestEnum::I);
for (TestEnum value : originalValues) {
mStateBits.set(value);
}
std::set<TestEnum> readValues;
for (TestEnum bit : IterateBitSet(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.
TEST_F(EnumBitSetIteratorTest, EmptySet) {
// We don't use the FAIL gtest macro here since it returns immediately,
// causing an unreachable code warning in MSVC
bool sawBit = false;
for ([[maybe_unused]] TestEnum bit : IterateBitSet(mStateBits)) {
sawBit = true;
}
EXPECT_FALSE(sawBit);
}
// Test iterating a result of combining two bitsets.
TEST_F(EnumBitSetIteratorTest, NonLValueBitset) {
ityp::bitset<TestEnum, kEnumCount> otherBits;
mStateBits.set(TestEnum::B);
mStateBits.set(TestEnum::C);
mStateBits.set(TestEnum::D);
mStateBits.set(TestEnum::E);
otherBits.set(TestEnum::A);
otherBits.set(TestEnum::B);
otherBits.set(TestEnum::D);
otherBits.set(TestEnum::F);
std::set<TestEnum> seenBits;
for (TestEnum bit : IterateBitSet(mStateBits & otherBits)) {
EXPECT_EQ(0u, seenBits.count(bit));
seenBits.insert(bit);
EXPECT_TRUE(mStateBits[bit]);
EXPECT_TRUE(otherBits[bit]);
}
EXPECT_EQ((mStateBits & otherBits).count(), seenBits.size());
}
class ITypBitsetIteratorTest : public testing::Test {
protected:
using IntegerT = TypedInteger<struct Foo, uint32_t>;
ityp::bitset<IntegerT, 40> mStateBits;
};
// Simple iterator test.
TEST_F(ITypBitsetIteratorTest, Iterator) {
std::set<IntegerT> originalValues;
originalValues.insert(IntegerT(2));
originalValues.insert(IntegerT(6));
originalValues.insert(IntegerT(8));
originalValues.insert(IntegerT(35));
for (IntegerT value : originalValues) {
mStateBits.set(value);
}
std::set<IntegerT> readValues;
for (IntegerT bit : IterateBitSet(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.
TEST_F(ITypBitsetIteratorTest, EmptySet) {
// We don't use the FAIL gtest macro here since it returns immediately,
// causing an unreachable code warning in MSVC
bool sawBit = false;
for ([[maybe_unused]] IntegerT bit : IterateBitSet(mStateBits)) {
sawBit = true;
}
EXPECT_FALSE(sawBit);
}
// Test iterating a result of combining two bitsets.
TEST_F(ITypBitsetIteratorTest, NonLValueBitset) {
ityp::bitset<IntegerT, 40> otherBits;
mStateBits.set(IntegerT(1));
mStateBits.set(IntegerT(2));
mStateBits.set(IntegerT(3));
mStateBits.set(IntegerT(4));
otherBits.set(IntegerT(0));
otherBits.set(IntegerT(1));
otherBits.set(IntegerT(3));
otherBits.set(IntegerT(5));
std::set<IntegerT> seenBits;
for (IntegerT bit : IterateBitSet(mStateBits & otherBits)) {
EXPECT_EQ(0u, seenBits.count(bit));
seenBits.insert(bit);
EXPECT_TRUE(mStateBits[bit]);
EXPECT_TRUE(otherBits[bit]);
}
EXPECT_EQ((mStateBits & otherBits).count(), seenBits.size());
}
} // anonymous namespace
} // namespace dawn