2010-01-05 23:04:49 +08:00
|
|
|
//===- llvm/unittest/ADT/BitVectorTest.cpp - BitVector tests --------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-05-01 04:50:28 +08:00
|
|
|
// Some of these tests fail on PowerPC for unknown reasons.
|
2013-07-27 06:13:57 +08:00
|
|
|
#ifndef __ppc__
|
2010-05-01 04:50:28 +08:00
|
|
|
|
2010-01-05 23:04:49 +08:00
|
|
|
#include "llvm/ADT/BitVector.h"
|
2012-06-16 18:51:07 +08:00
|
|
|
#include "llvm/ADT/SmallBitVector.h"
|
2010-01-05 23:04:49 +08:00
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
// Test fixture
|
|
|
|
template <typename T>
|
|
|
|
class BitVectorTest : public ::testing::Test { };
|
|
|
|
|
|
|
|
// Test both BitVector and SmallBitVector with the same suite of tests.
|
|
|
|
typedef ::testing::Types<BitVector, SmallBitVector> BitVectorTestTypes;
|
|
|
|
TYPED_TEST_CASE(BitVectorTest, BitVectorTestTypes);
|
|
|
|
|
|
|
|
TYPED_TEST(BitVectorTest, TrivialOperation) {
|
|
|
|
TypeParam Vec;
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_EQ(0U, Vec.count());
|
|
|
|
EXPECT_EQ(0U, Vec.size());
|
|
|
|
EXPECT_FALSE(Vec.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_TRUE(Vec.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_TRUE(Vec.none());
|
|
|
|
EXPECT_TRUE(Vec.empty());
|
|
|
|
|
|
|
|
Vec.resize(5, true);
|
|
|
|
EXPECT_EQ(5U, Vec.count());
|
|
|
|
EXPECT_EQ(5U, Vec.size());
|
|
|
|
EXPECT_TRUE(Vec.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_TRUE(Vec.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_FALSE(Vec.none());
|
|
|
|
EXPECT_FALSE(Vec.empty());
|
|
|
|
|
|
|
|
Vec.resize(11);
|
|
|
|
EXPECT_EQ(5U, Vec.count());
|
|
|
|
EXPECT_EQ(11U, Vec.size());
|
|
|
|
EXPECT_TRUE(Vec.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_FALSE(Vec.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_FALSE(Vec.none());
|
|
|
|
EXPECT_FALSE(Vec.empty());
|
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TypeParam Inv = Vec;
|
2012-05-14 23:46:27 +08:00
|
|
|
Inv.flip();
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_EQ(6U, Inv.count());
|
|
|
|
EXPECT_EQ(11U, Inv.size());
|
|
|
|
EXPECT_TRUE(Inv.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_FALSE(Inv.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_FALSE(Inv.none());
|
|
|
|
EXPECT_FALSE(Inv.empty());
|
|
|
|
|
|
|
|
EXPECT_FALSE(Inv == Vec);
|
|
|
|
EXPECT_TRUE(Inv != Vec);
|
2012-05-14 23:46:27 +08:00
|
|
|
Vec.flip();
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_TRUE(Inv == Vec);
|
|
|
|
EXPECT_FALSE(Inv != Vec);
|
|
|
|
|
|
|
|
// Add some "interesting" data to Vec.
|
|
|
|
Vec.resize(23, true);
|
|
|
|
Vec.resize(25, false);
|
|
|
|
Vec.resize(26, true);
|
|
|
|
Vec.resize(29, false);
|
|
|
|
Vec.resize(33, true);
|
2010-05-01 04:50:28 +08:00
|
|
|
Vec.resize(57, false);
|
2010-01-05 23:04:49 +08:00
|
|
|
unsigned Count = 0;
|
|
|
|
for (unsigned i = Vec.find_first(); i != -1u; i = Vec.find_next(i)) {
|
|
|
|
++Count;
|
|
|
|
EXPECT_TRUE(Vec[i]);
|
|
|
|
EXPECT_TRUE(Vec.test(i));
|
|
|
|
}
|
|
|
|
EXPECT_EQ(Count, Vec.count());
|
|
|
|
EXPECT_EQ(Count, 23u);
|
|
|
|
EXPECT_FALSE(Vec[0]);
|
|
|
|
EXPECT_TRUE(Vec[32]);
|
2010-05-01 04:50:28 +08:00
|
|
|
EXPECT_FALSE(Vec[56]);
|
|
|
|
Vec.resize(61, false);
|
2010-01-05 23:04:49 +08:00
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TypeParam Copy = Vec;
|
|
|
|
TypeParam Alt(3, false);
|
2010-01-05 23:04:49 +08:00
|
|
|
Alt.resize(6, true);
|
|
|
|
std::swap(Alt, Vec);
|
|
|
|
EXPECT_TRUE(Copy == Alt);
|
|
|
|
EXPECT_TRUE(Vec.size() == 6);
|
|
|
|
EXPECT_TRUE(Vec.count() == 3);
|
|
|
|
EXPECT_TRUE(Vec.find_first() == 3);
|
|
|
|
std::swap(Copy, Vec);
|
|
|
|
|
|
|
|
// Add some more "interesting" data.
|
|
|
|
Vec.resize(68, true);
|
|
|
|
Vec.resize(78, false);
|
|
|
|
Vec.resize(89, true);
|
|
|
|
Vec.resize(90, false);
|
|
|
|
Vec.resize(91, true);
|
|
|
|
Vec.resize(130, false);
|
|
|
|
Count = 0;
|
|
|
|
for (unsigned i = Vec.find_first(); i != -1u; i = Vec.find_next(i)) {
|
|
|
|
++Count;
|
|
|
|
EXPECT_TRUE(Vec[i]);
|
|
|
|
EXPECT_TRUE(Vec.test(i));
|
|
|
|
}
|
|
|
|
EXPECT_EQ(Count, Vec.count());
|
|
|
|
EXPECT_EQ(Count, 42u);
|
|
|
|
EXPECT_FALSE(Vec[0]);
|
|
|
|
EXPECT_TRUE(Vec[32]);
|
|
|
|
EXPECT_FALSE(Vec[60]);
|
|
|
|
EXPECT_FALSE(Vec[129]);
|
|
|
|
|
|
|
|
Vec.flip(60);
|
|
|
|
EXPECT_TRUE(Vec[60]);
|
|
|
|
EXPECT_EQ(Count + 1, Vec.count());
|
|
|
|
Vec.flip(60);
|
|
|
|
EXPECT_FALSE(Vec[60]);
|
|
|
|
EXPECT_EQ(Count, Vec.count());
|
|
|
|
|
|
|
|
Vec.reset(32);
|
|
|
|
EXPECT_FALSE(Vec[32]);
|
|
|
|
EXPECT_EQ(Count - 1, Vec.count());
|
|
|
|
Vec.set(32);
|
|
|
|
EXPECT_TRUE(Vec[32]);
|
|
|
|
EXPECT_EQ(Count, Vec.count());
|
|
|
|
|
|
|
|
Vec.flip();
|
|
|
|
EXPECT_EQ(Vec.size() - Count, Vec.count());
|
|
|
|
|
|
|
|
Vec.reset();
|
|
|
|
EXPECT_EQ(0U, Vec.count());
|
|
|
|
EXPECT_EQ(130U, Vec.size());
|
|
|
|
EXPECT_FALSE(Vec.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_FALSE(Vec.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_TRUE(Vec.none());
|
|
|
|
EXPECT_FALSE(Vec.empty());
|
|
|
|
|
2013-06-07 22:14:38 +08:00
|
|
|
Vec.flip();
|
|
|
|
EXPECT_EQ(130U, Vec.count());
|
|
|
|
EXPECT_EQ(130U, Vec.size());
|
|
|
|
EXPECT_TRUE(Vec.any());
|
|
|
|
EXPECT_TRUE(Vec.all());
|
|
|
|
EXPECT_FALSE(Vec.none());
|
|
|
|
EXPECT_FALSE(Vec.empty());
|
|
|
|
|
2013-06-07 23:14:31 +08:00
|
|
|
Vec.resize(64);
|
|
|
|
EXPECT_EQ(64U, Vec.count());
|
|
|
|
EXPECT_EQ(64U, Vec.size());
|
|
|
|
EXPECT_TRUE(Vec.any());
|
|
|
|
EXPECT_TRUE(Vec.all());
|
|
|
|
EXPECT_FALSE(Vec.none());
|
|
|
|
EXPECT_FALSE(Vec.empty());
|
|
|
|
|
|
|
|
Vec.flip();
|
|
|
|
EXPECT_EQ(0U, Vec.count());
|
|
|
|
EXPECT_EQ(64U, Vec.size());
|
|
|
|
EXPECT_FALSE(Vec.any());
|
|
|
|
EXPECT_FALSE(Vec.all());
|
|
|
|
EXPECT_TRUE(Vec.none());
|
|
|
|
EXPECT_FALSE(Vec.empty());
|
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
Inv = TypeParam().flip();
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_EQ(0U, Inv.count());
|
|
|
|
EXPECT_EQ(0U, Inv.size());
|
|
|
|
EXPECT_FALSE(Inv.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_TRUE(Inv.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_TRUE(Inv.none());
|
|
|
|
EXPECT_TRUE(Inv.empty());
|
|
|
|
|
|
|
|
Vec.clear();
|
|
|
|
EXPECT_EQ(0U, Vec.count());
|
|
|
|
EXPECT_EQ(0U, Vec.size());
|
|
|
|
EXPECT_FALSE(Vec.any());
|
2010-09-27 23:48:37 +08:00
|
|
|
EXPECT_TRUE(Vec.all());
|
2010-01-05 23:04:49 +08:00
|
|
|
EXPECT_TRUE(Vec.none());
|
|
|
|
EXPECT_TRUE(Vec.empty());
|
|
|
|
}
|
|
|
|
|
2017-04-11 01:18:54 +08:00
|
|
|
TYPED_TEST(BitVectorTest, FindOperations) {
|
|
|
|
// Test finding in an empty BitVector.
|
|
|
|
TypeParam A;
|
|
|
|
EXPECT_EQ(-1, A.find_first());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(-1, A.find_last());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(-1, A.find_first_unset());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(-1, A.find_last_unset());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(-1, A.find_next(0));
|
|
|
|
EXPECT_EQ(-1, A.find_next_unset(0));
|
|
|
|
|
|
|
|
// Test finding next set and unset bits in a BitVector with multiple words
|
|
|
|
A.resize(100);
|
|
|
|
A.set(12);
|
|
|
|
A.set(13);
|
|
|
|
A.set(75);
|
|
|
|
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(75, A.find_last());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(12, A.find_first());
|
|
|
|
EXPECT_EQ(13, A.find_next(12));
|
|
|
|
EXPECT_EQ(75, A.find_next(13));
|
|
|
|
EXPECT_EQ(-1, A.find_next(75));
|
|
|
|
|
|
|
|
EXPECT_EQ(0, A.find_first_unset());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(99, A.find_last_unset());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(14, A.find_next_unset(11));
|
|
|
|
EXPECT_EQ(14, A.find_next_unset(12));
|
|
|
|
EXPECT_EQ(14, A.find_next_unset(13));
|
|
|
|
EXPECT_EQ(16, A.find_next_unset(15));
|
|
|
|
EXPECT_EQ(76, A.find_next_unset(74));
|
|
|
|
EXPECT_EQ(76, A.find_next_unset(75));
|
|
|
|
EXPECT_EQ(-1, A.find_next_unset(99));
|
|
|
|
|
|
|
|
A.set(0, 100);
|
2017-04-11 04:01:50 +08:00
|
|
|
EXPECT_EQ(100U, A.count());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(0, A.find_first());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(99, A.find_last());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(-1, A.find_first_unset());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(-1, A.find_last_unset());
|
2017-04-11 01:18:54 +08:00
|
|
|
|
|
|
|
A.reset(0, 100);
|
2017-04-11 04:01:50 +08:00
|
|
|
EXPECT_EQ(0U, A.count());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(-1, A.find_first());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(-1, A.find_last());
|
2017-04-11 01:18:54 +08:00
|
|
|
EXPECT_EQ(0, A.find_first_unset());
|
2017-04-22 02:07:46 +08:00
|
|
|
EXPECT_EQ(99, A.find_last_unset());
|
2017-04-11 01:18:54 +08:00
|
|
|
}
|
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TYPED_TEST(BitVectorTest, CompoundAssignment) {
|
|
|
|
TypeParam A;
|
2010-02-10 13:54:04 +08:00
|
|
|
A.resize(10);
|
|
|
|
A.set(4);
|
|
|
|
A.set(7);
|
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TypeParam B;
|
2010-02-10 13:54:04 +08:00
|
|
|
B.resize(50);
|
|
|
|
B.set(5);
|
|
|
|
B.set(18);
|
|
|
|
|
|
|
|
A |= B;
|
|
|
|
EXPECT_TRUE(A.test(4));
|
|
|
|
EXPECT_TRUE(A.test(5));
|
|
|
|
EXPECT_TRUE(A.test(7));
|
|
|
|
EXPECT_TRUE(A.test(18));
|
2010-02-10 21:34:02 +08:00
|
|
|
EXPECT_EQ(4U, A.count());
|
|
|
|
EXPECT_EQ(50U, A.size());
|
2010-02-10 13:54:04 +08:00
|
|
|
|
|
|
|
B.resize(10);
|
|
|
|
B.set();
|
|
|
|
B.reset(2);
|
|
|
|
B.reset(7);
|
|
|
|
A &= B;
|
|
|
|
EXPECT_FALSE(A.test(2));
|
|
|
|
EXPECT_FALSE(A.test(7));
|
2010-02-10 21:34:02 +08:00
|
|
|
EXPECT_EQ(2U, A.count());
|
|
|
|
EXPECT_EQ(50U, A.size());
|
2010-02-10 13:54:04 +08:00
|
|
|
|
|
|
|
B.resize(100);
|
|
|
|
B.set();
|
|
|
|
|
|
|
|
A ^= B;
|
|
|
|
EXPECT_TRUE(A.test(2));
|
|
|
|
EXPECT_TRUE(A.test(7));
|
2010-02-10 21:34:02 +08:00
|
|
|
EXPECT_EQ(98U, A.count());
|
|
|
|
EXPECT_EQ(100U, A.size());
|
2010-01-05 23:04:49 +08:00
|
|
|
}
|
2010-02-10 13:54:04 +08:00
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TYPED_TEST(BitVectorTest, ProxyIndex) {
|
|
|
|
TypeParam Vec(3);
|
2010-05-01 04:50:28 +08:00
|
|
|
EXPECT_TRUE(Vec.none());
|
|
|
|
Vec[0] = Vec[1] = Vec[2] = true;
|
|
|
|
EXPECT_EQ(Vec.size(), Vec.count());
|
|
|
|
Vec[2] = Vec[1] = Vec[0] = false;
|
|
|
|
EXPECT_TRUE(Vec.none());
|
|
|
|
}
|
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TYPED_TEST(BitVectorTest, PortableBitMask) {
|
|
|
|
TypeParam A;
|
2012-01-17 09:24:32 +08:00
|
|
|
const uint32_t Mask1[] = { 0x80000000, 6, 5 };
|
|
|
|
|
|
|
|
A.resize(10);
|
2015-09-18 23:08:24 +08:00
|
|
|
A.setBitsInMask(Mask1, 1);
|
2012-01-17 09:24:32 +08:00
|
|
|
EXPECT_EQ(10u, A.size());
|
|
|
|
EXPECT_FALSE(A.test(0));
|
|
|
|
|
|
|
|
A.resize(32);
|
2015-09-18 23:08:24 +08:00
|
|
|
A.setBitsInMask(Mask1, 1);
|
2012-01-17 09:24:32 +08:00
|
|
|
EXPECT_FALSE(A.test(0));
|
|
|
|
EXPECT_TRUE(A.test(31));
|
|
|
|
EXPECT_EQ(1u, A.count());
|
|
|
|
|
|
|
|
A.resize(33);
|
|
|
|
A.setBitsInMask(Mask1, 1);
|
|
|
|
EXPECT_EQ(1u, A.count());
|
|
|
|
A.setBitsInMask(Mask1, 2);
|
|
|
|
EXPECT_EQ(1u, A.count());
|
|
|
|
|
|
|
|
A.resize(34);
|
|
|
|
A.setBitsInMask(Mask1, 2);
|
|
|
|
EXPECT_EQ(2u, A.count());
|
|
|
|
|
|
|
|
A.resize(65);
|
|
|
|
A.setBitsInMask(Mask1, 3);
|
|
|
|
EXPECT_EQ(4u, A.count());
|
|
|
|
|
|
|
|
A.setBitsNotInMask(Mask1, 1);
|
|
|
|
EXPECT_EQ(32u+3u, A.count());
|
|
|
|
|
|
|
|
A.setBitsNotInMask(Mask1, 3);
|
|
|
|
EXPECT_EQ(65u, A.count());
|
|
|
|
|
|
|
|
A.resize(96);
|
|
|
|
EXPECT_EQ(65u, A.count());
|
|
|
|
|
|
|
|
A.clear();
|
|
|
|
A.resize(128);
|
|
|
|
A.setBitsNotInMask(Mask1, 3);
|
|
|
|
EXPECT_EQ(96u-5u, A.count());
|
|
|
|
|
|
|
|
A.clearBitsNotInMask(Mask1, 1);
|
|
|
|
EXPECT_EQ(64-4u, A.count());
|
|
|
|
}
|
2010-02-10 13:54:04 +08:00
|
|
|
|
2012-06-16 18:51:07 +08:00
|
|
|
TYPED_TEST(BitVectorTest, BinOps) {
|
|
|
|
TypeParam A;
|
|
|
|
TypeParam B;
|
2012-05-14 23:01:19 +08:00
|
|
|
|
|
|
|
A.resize(65);
|
|
|
|
EXPECT_FALSE(A.anyCommon(B));
|
|
|
|
EXPECT_FALSE(B.anyCommon(B));
|
|
|
|
|
|
|
|
B.resize(64);
|
|
|
|
A.set(64);
|
|
|
|
EXPECT_FALSE(A.anyCommon(B));
|
|
|
|
EXPECT_FALSE(B.anyCommon(A));
|
|
|
|
|
|
|
|
B.set(63);
|
|
|
|
EXPECT_FALSE(A.anyCommon(B));
|
|
|
|
EXPECT_FALSE(B.anyCommon(A));
|
|
|
|
|
|
|
|
A.set(63);
|
|
|
|
EXPECT_TRUE(A.anyCommon(B));
|
|
|
|
EXPECT_TRUE(B.anyCommon(A));
|
|
|
|
|
|
|
|
B.resize(70);
|
|
|
|
B.set(64);
|
|
|
|
B.reset(63);
|
|
|
|
A.resize(64);
|
|
|
|
EXPECT_FALSE(A.anyCommon(B));
|
|
|
|
EXPECT_FALSE(B.anyCommon(A));
|
|
|
|
}
|
2012-10-16 06:05:27 +08:00
|
|
|
|
2017-04-21 00:56:54 +08:00
|
|
|
typedef std::vector<std::pair<int, int>> RangeList;
|
|
|
|
|
|
|
|
template <typename VecType>
|
|
|
|
static inline VecType createBitVector(uint32_t Size,
|
|
|
|
const RangeList &setRanges) {
|
|
|
|
VecType V;
|
|
|
|
V.resize(Size);
|
|
|
|
for (auto &R : setRanges)
|
|
|
|
V.set(R.first, R.second);
|
|
|
|
return V;
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(BitVectorTest, ShiftOpsSingleWord) {
|
|
|
|
// Test that shift ops work when the desired shift amount is less
|
|
|
|
// than one word.
|
|
|
|
|
|
|
|
// 1. Case where the number of bits in the BitVector also fit into a single
|
|
|
|
// word.
|
|
|
|
TypeParam A = createBitVector<TypeParam>(12, {{2, 4}, {8, 10}});
|
|
|
|
TypeParam B = A;
|
|
|
|
|
|
|
|
EXPECT_EQ(4U, A.count());
|
|
|
|
EXPECT_TRUE(A.test(2));
|
|
|
|
EXPECT_TRUE(A.test(3));
|
|
|
|
EXPECT_TRUE(A.test(8));
|
|
|
|
EXPECT_TRUE(A.test(9));
|
|
|
|
|
|
|
|
A >>= 1;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(12, {{1, 3}, {7, 9}}), A);
|
|
|
|
|
|
|
|
A <<= 1;
|
|
|
|
EXPECT_EQ(B, A);
|
|
|
|
|
|
|
|
A >>= 10;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(12, {}), A);
|
|
|
|
|
|
|
|
A = B;
|
|
|
|
A <<= 10;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(12, {}), A);
|
|
|
|
|
|
|
|
// 2. Case where the number of bits in the BitVector do not fit into a single
|
|
|
|
// word.
|
|
|
|
|
|
|
|
// 31----------------------------------------------------------------------0
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 00000111 | 11111110 00000000 00001111 11111111
|
|
|
|
A = createBitVector<TypeParam>(40, {{0, 12}, {25, 35}});
|
|
|
|
EXPECT_EQ(40U, A.size());
|
|
|
|
EXPECT_EQ(22U, A.count());
|
|
|
|
|
|
|
|
// 2a. Make sure that left shifting some 1 bits out of the vector works.
|
|
|
|
// 31----------------------------------------------------------------------0
|
|
|
|
// Before:
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 00000111 | 11111110 00000000 00001111 11111111
|
|
|
|
// After:
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 11111100 | 00000000 00011111 11111110 00000000
|
|
|
|
A <<= 9;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(40, {{9, 21}, {34, 40}}), A);
|
|
|
|
|
|
|
|
// 2b. Make sure that keeping the number of one bits unchanged works.
|
|
|
|
// 31----------------------------------------------------------------------0
|
|
|
|
// Before:
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 11111100 | 00000000 00011111 11111110 00000000
|
|
|
|
// After:
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 00000011 | 11110000 00000000 01111111 11111000
|
|
|
|
A >>= 6;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(40, {{3, 15}, {28, 34}}), A);
|
|
|
|
|
|
|
|
// 2c. Make sure that right shifting some 1 bits out of the vector works.
|
|
|
|
// 31----------------------------------------------------------------------0
|
|
|
|
// Before:
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 00000011 | 11110000 00000000 01111111 11111000
|
|
|
|
// After:
|
|
|
|
// XXXXXXXX XXXXXXXX XXXXXXXX 00000000 | 00000000 11111100 00000000 00011111
|
|
|
|
A >>= 10;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(40, {{0, 5}, {18, 24}}), A);
|
|
|
|
|
|
|
|
// 3. Big test.
|
|
|
|
A = createBitVector<TypeParam>(300, {{1, 30}, {60, 95}, {200, 275}});
|
|
|
|
A <<= 29;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(
|
|
|
|
300, {{1 + 29, 30 + 29}, {60 + 29, 95 + 29}, {200 + 29, 300}}),
|
|
|
|
A);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(BitVectorTest, ShiftOpsMultiWord) {
|
|
|
|
// Test that shift ops work when the desired shift amount is greater than or
|
|
|
|
// equal to the size of a single word.
|
|
|
|
auto A = createBitVector<TypeParam>(300, {{1, 30}, {60, 95}, {200, 275}});
|
|
|
|
|
|
|
|
// Make a copy so we can re-use it later.
|
|
|
|
auto B = A;
|
|
|
|
|
|
|
|
// 1. Shift left by an exact multiple of the word size. This should invoke
|
|
|
|
// only a memmove and no per-word bit operations.
|
|
|
|
A <<= 64;
|
|
|
|
auto Expected = createBitVector<TypeParam>(
|
|
|
|
300, {{1 + 64, 30 + 64}, {60 + 64, 95 + 64}, {200 + 64, 300}});
|
|
|
|
EXPECT_EQ(Expected, A);
|
|
|
|
|
|
|
|
// 2. Shift left by a non multiple of the word size. This should invoke both
|
|
|
|
// a memmove and per-word bit operations.
|
|
|
|
A = B;
|
|
|
|
A <<= 93;
|
|
|
|
EXPECT_EQ(createBitVector<TypeParam>(
|
|
|
|
300, {{1 + 93, 30 + 93}, {60 + 93, 95 + 93}, {200 + 93, 300}}),
|
|
|
|
A);
|
|
|
|
|
|
|
|
// 1. Shift right by an exact multiple of the word size. This should invoke
|
|
|
|
// only a memmove and no per-word bit operations.
|
|
|
|
A = B;
|
|
|
|
A >>= 64;
|
|
|
|
EXPECT_EQ(
|
|
|
|
createBitVector<TypeParam>(300, {{0, 95 - 64}, {200 - 64, 275 - 64}}), A);
|
|
|
|
|
|
|
|
// 2. Shift left by a non multiple of the word size. This should invoke both
|
|
|
|
// a memmove and per-word bit operations.
|
|
|
|
A = B;
|
|
|
|
A >>= 93;
|
|
|
|
EXPECT_EQ(
|
|
|
|
createBitVector<TypeParam>(300, {{0, 95 - 93}, {200 - 93, 275 - 93}}), A);
|
|
|
|
}
|
|
|
|
|
2012-10-16 06:05:27 +08:00
|
|
|
TYPED_TEST(BitVectorTest, RangeOps) {
|
|
|
|
TypeParam A;
|
|
|
|
A.resize(256);
|
|
|
|
A.reset();
|
|
|
|
A.set(1, 255);
|
|
|
|
|
|
|
|
EXPECT_FALSE(A.test(0));
|
|
|
|
EXPECT_TRUE( A.test(1));
|
|
|
|
EXPECT_TRUE( A.test(23));
|
|
|
|
EXPECT_TRUE( A.test(254));
|
|
|
|
EXPECT_FALSE(A.test(255));
|
|
|
|
|
|
|
|
TypeParam B;
|
|
|
|
B.resize(256);
|
|
|
|
B.set();
|
|
|
|
B.reset(1, 255);
|
|
|
|
|
|
|
|
EXPECT_TRUE( B.test(0));
|
|
|
|
EXPECT_FALSE(B.test(1));
|
|
|
|
EXPECT_FALSE(B.test(23));
|
|
|
|
EXPECT_FALSE(B.test(254));
|
|
|
|
EXPECT_TRUE( B.test(255));
|
|
|
|
|
|
|
|
TypeParam C;
|
|
|
|
C.resize(3);
|
|
|
|
C.reset();
|
|
|
|
C.set(0, 1);
|
|
|
|
|
|
|
|
EXPECT_TRUE(C.test(0));
|
|
|
|
EXPECT_FALSE( C.test(1));
|
|
|
|
EXPECT_FALSE( C.test(2));
|
|
|
|
|
|
|
|
TypeParam D;
|
|
|
|
D.resize(3);
|
|
|
|
D.set();
|
|
|
|
D.reset(0, 1);
|
|
|
|
|
|
|
|
EXPECT_FALSE(D.test(0));
|
|
|
|
EXPECT_TRUE( D.test(1));
|
|
|
|
EXPECT_TRUE( D.test(2));
|
2012-10-16 14:04:27 +08:00
|
|
|
|
|
|
|
TypeParam E;
|
|
|
|
E.resize(128);
|
|
|
|
E.reset();
|
|
|
|
E.set(1, 33);
|
|
|
|
|
|
|
|
EXPECT_FALSE(E.test(0));
|
|
|
|
EXPECT_TRUE( E.test(1));
|
|
|
|
EXPECT_TRUE( E.test(32));
|
|
|
|
EXPECT_FALSE(E.test(33));
|
2013-12-19 10:35:26 +08:00
|
|
|
|
|
|
|
TypeParam BufferOverrun;
|
|
|
|
unsigned size = sizeof(unsigned long) * 8;
|
|
|
|
BufferOverrun.resize(size);
|
|
|
|
BufferOverrun.reset(0, size);
|
|
|
|
BufferOverrun.set(0, size);
|
2012-10-16 06:05:27 +08:00
|
|
|
}
|
2013-07-12 05:59:16 +08:00
|
|
|
|
|
|
|
TYPED_TEST(BitVectorTest, CompoundTestReset) {
|
|
|
|
TypeParam A(50, true);
|
|
|
|
TypeParam B(50, false);
|
|
|
|
|
|
|
|
TypeParam C(100, true);
|
|
|
|
TypeParam D(100, false);
|
|
|
|
|
|
|
|
EXPECT_FALSE(A.test(A));
|
|
|
|
EXPECT_TRUE(A.test(B));
|
|
|
|
EXPECT_FALSE(A.test(C));
|
|
|
|
EXPECT_TRUE(A.test(D));
|
|
|
|
EXPECT_FALSE(B.test(A));
|
|
|
|
EXPECT_FALSE(B.test(B));
|
|
|
|
EXPECT_FALSE(B.test(C));
|
|
|
|
EXPECT_FALSE(B.test(D));
|
|
|
|
EXPECT_TRUE(C.test(A));
|
|
|
|
EXPECT_TRUE(C.test(B));
|
|
|
|
EXPECT_FALSE(C.test(C));
|
|
|
|
EXPECT_TRUE(C.test(D));
|
|
|
|
|
|
|
|
A.reset(B);
|
|
|
|
A.reset(D);
|
|
|
|
EXPECT_TRUE(A.all());
|
|
|
|
A.reset(A);
|
|
|
|
EXPECT_TRUE(A.none());
|
|
|
|
A.set();
|
|
|
|
A.reset(C);
|
|
|
|
EXPECT_TRUE(A.none());
|
|
|
|
A.set();
|
|
|
|
|
|
|
|
C.reset(A);
|
|
|
|
EXPECT_EQ(50, C.find_first());
|
|
|
|
C.reset(C);
|
|
|
|
EXPECT_TRUE(C.none());
|
|
|
|
}
|
2016-06-17 05:45:13 +08:00
|
|
|
|
|
|
|
TYPED_TEST(BitVectorTest, MoveConstructor) {
|
|
|
|
TypeParam A(10, true);
|
|
|
|
TypeParam B(std::move(A));
|
|
|
|
// Check that the move ctor leaves the moved-from object in a valid state.
|
|
|
|
// The following line used to crash.
|
|
|
|
A = B;
|
|
|
|
|
|
|
|
TypeParam C(10, true);
|
|
|
|
EXPECT_EQ(C, A);
|
|
|
|
EXPECT_EQ(C, B);
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_TEST(BitVectorTest, MoveAssignment) {
|
|
|
|
TypeParam A(10, true);
|
|
|
|
TypeParam B;
|
|
|
|
B = std::move(A);
|
|
|
|
// Check that move assignment leaves the moved-from object in a valid state.
|
|
|
|
// The following line used to crash.
|
|
|
|
A = B;
|
|
|
|
|
|
|
|
TypeParam C(10, true);
|
|
|
|
EXPECT_EQ(C, A);
|
|
|
|
EXPECT_EQ(C, B);
|
|
|
|
}
|
|
|
|
|
2017-01-24 03:06:54 +08:00
|
|
|
template<class TypeParam>
|
|
|
|
static void testEmpty(const TypeParam &A) {
|
|
|
|
EXPECT_TRUE(A.empty());
|
|
|
|
EXPECT_EQ((size_t)0, A.size());
|
|
|
|
EXPECT_EQ((size_t)0, A.count());
|
|
|
|
EXPECT_FALSE(A.any());
|
|
|
|
EXPECT_TRUE(A.all());
|
|
|
|
EXPECT_TRUE(A.none());
|
|
|
|
EXPECT_EQ(-1, A.find_first());
|
|
|
|
EXPECT_EQ(A, TypeParam());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Tests whether BitVector behaves well with Bits==nullptr, Capacity==0
|
|
|
|
TYPED_TEST(BitVectorTest, EmptyVector) {
|
|
|
|
TypeParam A;
|
|
|
|
testEmpty(A);
|
|
|
|
|
|
|
|
TypeParam B;
|
|
|
|
B.reset();
|
|
|
|
testEmpty(B);
|
|
|
|
|
|
|
|
TypeParam C;
|
|
|
|
C.clear();
|
|
|
|
testEmpty(C);
|
|
|
|
|
|
|
|
TypeParam D(A);
|
|
|
|
testEmpty(D);
|
|
|
|
|
|
|
|
TypeParam E;
|
|
|
|
E = A;
|
|
|
|
testEmpty(E);
|
|
|
|
|
|
|
|
TypeParam F;
|
|
|
|
E.reset(A);
|
|
|
|
testEmpty(E);
|
|
|
|
}
|
|
|
|
|
2012-05-14 23:01:19 +08:00
|
|
|
}
|
2010-02-10 06:15:27 +08:00
|
|
|
#endif
|