| // Copyright 2015 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/base/interval_set.h" |
| |
| #include <stdarg.h> |
| |
| #include <iterator> |
| #include <limits> |
| |
| #include "net/test/gtest_util.h" |
| #include "starboard/types.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| using std::string; |
| |
| namespace net { |
| namespace test { |
| namespace { |
| |
| using ::testing::ElementsAreArray; |
| |
| class IntervalSetTest : public ::testing::Test { |
| protected: |
| void SetUp() override { |
| // Initialize two IntervalSets for union, intersection, and difference |
| // tests |
| is.Add(100, 200); |
| is.Add(300, 400); |
| is.Add(500, 600); |
| is.Add(700, 800); |
| is.Add(900, 1000); |
| is.Add(1100, 1200); |
| is.Add(1300, 1400); |
| is.Add(1500, 1600); |
| is.Add(1700, 1800); |
| is.Add(1900, 2000); |
| is.Add(2100, 2200); |
| |
| // Lots of different cases: |
| other.Add(50, 70); // disjoint, at the beginning |
| other.Add(2250, 2270); // disjoint, at the end |
| other.Add(650, 670); // disjoint, in the middle |
| other.Add(350, 360); // included |
| other.Add(370, 380); // also included (two at once) |
| other.Add(470, 530); // overlaps low end |
| other.Add(770, 830); // overlaps high end |
| other.Add(870, 900); // meets at low end |
| other.Add(1200, 1230); // meets at high end |
| other.Add(1270, 1830); // overlaps multiple ranges |
| } |
| |
| void TearDown() override { |
| is.Clear(); |
| EXPECT_TRUE(is.Empty()); |
| other.Clear(); |
| EXPECT_TRUE(other.Empty()); |
| } |
| IntervalSet<int> is; |
| IntervalSet<int> other; |
| }; |
| |
| TEST_F(IntervalSetTest, IsDisjoint) { |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(0, 99))); |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(0, 100))); |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(200, 200))); |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(200, 299))); |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(400, 407))); |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(405, 499))); |
| EXPECT_TRUE(is.IsDisjoint(Interval<int>(2300, 2300))); |
| EXPECT_TRUE( |
| is.IsDisjoint(Interval<int>(2300, std::numeric_limits<int>::max()))); |
| EXPECT_FALSE(is.IsDisjoint(Interval<int>(100, 100))); |
| EXPECT_FALSE(is.IsDisjoint(Interval<int>(100, 105))); |
| EXPECT_FALSE(is.IsDisjoint(Interval<int>(199, 300))); |
| EXPECT_FALSE(is.IsDisjoint(Interval<int>(250, 450))); |
| EXPECT_FALSE(is.IsDisjoint(Interval<int>(299, 400))); |
| EXPECT_FALSE(is.IsDisjoint(Interval<int>(250, 2000))); |
| EXPECT_FALSE( |
| is.IsDisjoint(Interval<int>(2199, std::numeric_limits<int>::max()))); |
| } |
| |
| // Base helper method for verifying the contents of an interval set. |
| // Returns true iff <is> contains <count> intervals whose successive |
| // endpoints match the sequence of args in <ap>: |
| static bool VA_Check(const IntervalSet<int>& is, size_t count, va_list ap) { |
| std::vector<Interval<int>> intervals; |
| is.Get(&intervals); |
| if (count != intervals.size()) { |
| LOG(ERROR) << "Expected " << count << " intervals, got " << intervals.size() |
| << ": " << is.ToString(); |
| return false; |
| } |
| if (count != is.Size()) { |
| LOG(ERROR) << "Expected " << count << " intervals, got Size " << is.Size() |
| << ": " << is.ToString(); |
| return false; |
| } |
| bool result = true; |
| for (size_t i = 0; i < count; i++) { |
| int min = va_arg(ap, int); |
| int max = va_arg(ap, int); |
| if (min != intervals[i].min() || max != intervals[i].max()) { |
| LOG(ERROR) << "Expected: [" << min << ", " << max << ") got " |
| << intervals[i] << " in " << is.ToString(); |
| result = false; |
| } |
| } |
| return result; |
| } |
| |
| static bool Check(const IntervalSet<int>& is, int count, ...) { |
| va_list ap; |
| va_start(ap, count); |
| const bool result = VA_Check(is, count, ap); |
| va_end(ap); |
| return result; |
| } |
| |
| // Some helper functions for testing Contains and Find, which are logically the |
| // same. |
| static void TestContainsAndFind(const IntervalSet<int>& is, int value) { |
| EXPECT_TRUE(is.Contains(value)) << "Set does not contain " << value; |
| auto it = is.Find(value); |
| EXPECT_NE(it, is.end()) << "No iterator to interval containing " << value; |
| EXPECT_TRUE(it->Contains(value)) << "Iterator does not contain " << value; |
| } |
| |
| static void TestContainsAndFind(const IntervalSet<int>& is, int min, int max) { |
| EXPECT_TRUE(is.Contains(min, max)) |
| << "Set does not contain interval with min " << min << "and max " << max; |
| auto it = is.Find(min, max); |
| EXPECT_NE(it, is.end()) << "No iterator to interval with min " << min |
| << "and max " << max; |
| EXPECT_TRUE(it->Contains(Interval<int>(min, max))) |
| << "Iterator does not contain interval with min " << min << "and max " |
| << max; |
| } |
| |
| static void TestNotContainsAndFind(const IntervalSet<int>& is, int value) { |
| EXPECT_FALSE(is.Contains(value)) << "Set contains " << value; |
| auto it = is.Find(value); |
| EXPECT_EQ(it, is.end()) << "There is iterator to interval containing " |
| << value; |
| } |
| |
| static void TestNotContainsAndFind(const IntervalSet<int>& is, |
| int min, |
| int max) { |
| EXPECT_FALSE(is.Contains(min, max)) << "Set contains interval with min " |
| << min << "and max " << max; |
| auto it = is.Find(min, max); |
| EXPECT_EQ(it, is.end()) << "There is iterator to interval with min " << min |
| << "and max " << max; |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetBasic) { |
| // Test Add, Get, Contains and Find |
| IntervalSet<int> iset; |
| EXPECT_TRUE(iset.Empty()); |
| EXPECT_EQ(0u, iset.Size()); |
| iset.Add(100, 200); |
| EXPECT_FALSE(iset.Empty()); |
| EXPECT_EQ(1u, iset.Size()); |
| iset.Add(100, 150); |
| iset.Add(150, 200); |
| iset.Add(130, 170); |
| iset.Add(90, 150); |
| iset.Add(170, 220); |
| iset.Add(300, 400); |
| iset.Add(250, 450); |
| EXPECT_FALSE(iset.Empty()); |
| EXPECT_EQ(2u, iset.Size()); |
| EXPECT_TRUE(Check(iset, 2, 90, 220, 250, 450)); |
| |
| // Test two intervals with a.max == b.min, that will just join up. |
| iset.Clear(); |
| iset.Add(100, 200); |
| iset.Add(200, 300); |
| EXPECT_FALSE(iset.Empty()); |
| EXPECT_EQ(1u, iset.Size()); |
| EXPECT_TRUE(Check(iset, 1, 100, 300)); |
| |
| // Test adding two sets together. |
| iset.Clear(); |
| IntervalSet<int> iset_add; |
| iset.Add(100, 200); |
| iset.Add(100, 150); |
| iset.Add(150, 200); |
| iset.Add(130, 170); |
| iset_add.Add(90, 150); |
| iset_add.Add(170, 220); |
| iset_add.Add(300, 400); |
| iset_add.Add(250, 450); |
| |
| iset.Add(iset_add); |
| EXPECT_FALSE(iset.Empty()); |
| EXPECT_EQ(2u, iset.Size()); |
| EXPECT_TRUE(Check(iset, 2, 90, 220, 250, 450)); |
| |
| // Test Get() (using an output iterator), begin()/end(), and rbegin()/rend() |
| // to iterate over intervals. |
| { |
| std::vector<Interval<int>> expected; |
| iset.Get(&expected); |
| |
| std::vector<Interval<int>> actual1; |
| iset.Get(back_inserter(actual1)); |
| ASSERT_EQ(expected.size(), actual1.size()); |
| |
| std::vector<Interval<int>> actual2; |
| std::copy(iset.begin(), iset.end(), back_inserter(actual2)); |
| ASSERT_EQ(expected.size(), actual2.size()); |
| |
| for (size_t i = 0; i < expected.size(); i++) { |
| EXPECT_EQ(expected[i].min(), actual1[i].min()); |
| EXPECT_EQ(expected[i].max(), actual1[i].max()); |
| |
| EXPECT_EQ(expected[i].min(), actual2[i].min()); |
| EXPECT_EQ(expected[i].max(), actual2[i].max()); |
| } |
| |
| // Ensure that the rbegin()/rend() iterators correctly yield the intervals |
| // in reverse order. |
| EXPECT_THAT(std::vector<Interval<int>>(iset.rbegin(), iset.rend()), |
| ElementsAreArray(expected.rbegin(), expected.rend())); |
| } |
| |
| TestNotContainsAndFind(iset, 89); |
| TestContainsAndFind(iset, 90); |
| TestContainsAndFind(iset, 120); |
| TestContainsAndFind(iset, 219); |
| TestNotContainsAndFind(iset, 220); |
| TestNotContainsAndFind(iset, 235); |
| TestNotContainsAndFind(iset, 249); |
| TestContainsAndFind(iset, 250); |
| TestContainsAndFind(iset, 300); |
| TestContainsAndFind(iset, 449); |
| TestNotContainsAndFind(iset, 450); |
| TestNotContainsAndFind(iset, 451); |
| |
| TestNotContainsAndFind(iset, 50, 60); |
| TestNotContainsAndFind(iset, 50, 90); |
| TestNotContainsAndFind(iset, 50, 200); |
| TestNotContainsAndFind(iset, 90, 90); |
| TestContainsAndFind(iset, 90, 200); |
| TestContainsAndFind(iset, 100, 200); |
| TestContainsAndFind(iset, 100, 220); |
| TestNotContainsAndFind(iset, 100, 221); |
| TestNotContainsAndFind(iset, 220, 220); |
| TestNotContainsAndFind(iset, 240, 300); |
| TestContainsAndFind(iset, 250, 300); |
| TestContainsAndFind(iset, 260, 300); |
| TestContainsAndFind(iset, 300, 450); |
| TestNotContainsAndFind(iset, 300, 451); |
| |
| IntervalSet<int> iset_contains; |
| iset_contains.Add(50, 90); |
| EXPECT_FALSE(iset.Contains(iset_contains)); |
| iset_contains.Clear(); |
| |
| iset_contains.Add(90, 200); |
| EXPECT_TRUE(iset.Contains(iset_contains)); |
| iset_contains.Add(100, 200); |
| EXPECT_TRUE(iset.Contains(iset_contains)); |
| iset_contains.Add(100, 220); |
| EXPECT_TRUE(iset.Contains(iset_contains)); |
| iset_contains.Add(250, 300); |
| EXPECT_TRUE(iset.Contains(iset_contains)); |
| iset_contains.Add(300, 450); |
| EXPECT_TRUE(iset.Contains(iset_contains)); |
| iset_contains.Add(300, 451); |
| EXPECT_FALSE(iset.Contains(iset_contains)); |
| EXPECT_FALSE(iset.Contains(Interval<int>())); |
| EXPECT_FALSE(iset.Contains(IntervalSet<int>())); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetContainsEmpty) { |
| const IntervalSet<int> empty; |
| const IntervalSet<int> other_empty; |
| EXPECT_FALSE(empty.Contains(empty)); |
| EXPECT_FALSE(empty.Contains(other_empty)); |
| // TODO(rtenneti): Implement after suupport for std::initializer_list. |
| #if 0 |
| const IntervalSet<int> non_empty({{10, 20}, {40, 50}}); |
| EXPECT_FALSE(empty.Contains(non_empty)); |
| EXPECT_FALSE(non_empty.Contains(empty)); |
| #endif |
| } |
| |
| TEST_F(IntervalSetTest, Equality) { |
| IntervalSet<int> is_copy = is; |
| EXPECT_TRUE(is.Equals(is)); |
| EXPECT_EQ(is, is); |
| EXPECT_TRUE(is.Equals(is_copy)); |
| EXPECT_EQ(is, is_copy); |
| EXPECT_FALSE(is.Equals(other)); |
| EXPECT_NE(is, other); |
| EXPECT_FALSE(is.Equals(IntervalSet<int>())); |
| EXPECT_NE(is, IntervalSet<int>()); |
| EXPECT_TRUE(IntervalSet<int>().Equals(IntervalSet<int>())); |
| EXPECT_EQ(IntervalSet<int>(), IntervalSet<int>()); |
| } |
| |
| TEST_F(IntervalSetTest, SpanningInterval) { |
| // Spanning interval of an empty set is empty: |
| { |
| IntervalSet<int> iset; |
| const Interval<int>& ival = iset.SpanningInterval(); |
| EXPECT_TRUE(ival.Empty()); |
| } |
| |
| // Spanning interval of a set with one interval is that interval: |
| { |
| IntervalSet<int> iset; |
| iset.Add(100, 200); |
| const Interval<int>& ival = iset.SpanningInterval(); |
| EXPECT_EQ(100, ival.min()); |
| EXPECT_EQ(200, ival.max()); |
| } |
| |
| // Spanning interval of a set with multiple elements is determined |
| // by the endpoints of the first and last element: |
| { |
| const Interval<int>& ival = is.SpanningInterval(); |
| EXPECT_EQ(100, ival.min()); |
| EXPECT_EQ(2200, ival.max()); |
| } |
| { |
| const Interval<int>& ival = other.SpanningInterval(); |
| EXPECT_EQ(50, ival.min()); |
| EXPECT_EQ(2270, ival.max()); |
| } |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetUnion) { |
| is.Union(other); |
| EXPECT_TRUE(Check(is, 12, 50, 70, 100, 200, 300, 400, 470, 600, 650, 670, 700, |
| 830, 870, 1000, 1100, 1230, 1270, 1830, 1900, 2000, 2100, |
| 2200, 2250, 2270)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersection) { |
| EXPECT_TRUE(is.Intersects(other)); |
| EXPECT_TRUE(other.Intersects(is)); |
| is.Intersection(other); |
| EXPECT_TRUE(Check(is, 7, 350, 360, 370, 380, 500, 530, 770, 800, 1300, 1400, |
| 1500, 1600, 1700, 1800)); |
| EXPECT_TRUE(is.Intersects(other)); |
| EXPECT_TRUE(other.Intersects(is)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionBothEmpty) { |
| IntervalSet<string> mine, theirs; |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionEmptyMine) { |
| IntervalSet<string> mine; |
| IntervalSet<string> theirs("a", "b"); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionEmptyTheirs) { |
| IntervalSet<string> mine("a", "b"); |
| IntervalSet<string> theirs; |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionTheirsBeforeMine) { |
| IntervalSet<string> mine("y", "z"); |
| IntervalSet<string> theirs; |
| theirs.Add("a", "b"); |
| theirs.Add("c", "d"); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionMineBeforeTheirs) { |
| IntervalSet<string> mine; |
| mine.Add("a", "b"); |
| mine.Add("c", "d"); |
| IntervalSet<string> theirs("y", "z"); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| // TODO(rtenneti): Implement after suupport for std::initializer_list. |
| #if 0 |
| TEST_F(IntervalSetTest, |
| IntervalSetIntersectionTheirsBeforeMineInt64Singletons) { |
| IntervalSet<int64_t> mine({{10, 15}}); |
| IntervalSet<int64_t> theirs({{-20, -5}}); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionMineBeforeTheirsIntSingletons) { |
| IntervalSet<int> mine({{10, 15}}); |
| IntervalSet<int> theirs({{90, 95}}); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionTheirsBetweenMine) { |
| IntervalSet<int64_t> mine({{0, 5}, {40, 50}}); |
| IntervalSet<int64_t> theirs({{10, 15}}); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionMineBetweenTheirs) { |
| IntervalSet<int> mine({{20, 25}}); |
| IntervalSet<int> theirs({{10, 15}, {30, 32}}); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| #endif // 0 |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionAlternatingIntervals) { |
| IntervalSet<int> mine, theirs; |
| mine.Add(10, 20); |
| mine.Add(40, 50); |
| mine.Add(60, 70); |
| theirs.Add(25, 39); |
| theirs.Add(55, 59); |
| theirs.Add(75, 79); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| EXPECT_FALSE(mine.Intersects(theirs)); |
| EXPECT_FALSE(theirs.Intersects(mine)); |
| } |
| |
| // TODO(rtenneti): Implement after suupport for std::initializer_list. |
| #if 0 |
| TEST_F(IntervalSetTest, |
| IntervalSetIntersectionAdjacentAlternatingNonIntersectingIntervals) { |
| // Make sure that intersection with adjacent interval set is empty. |
| const IntervalSet<int> x1({{0, 10}}); |
| const IntervalSet<int> y1({{-50, 0}, {10, 95}}); |
| |
| IntervalSet<int> result1 = x1; |
| result1.Intersection(y1); |
| EXPECT_TRUE(result1.Empty()) << result1; |
| |
| const IntervalSet<int16_t> x2({{0, 10}, {20, 30}, {40, 90}}); |
| const IntervalSet<int16_t> y2( |
| {{-50, -40}, {-2, 0}, {10, 20}, {32, 40}, {90, 95}}); |
| |
| IntervalSet<int16_t> result2 = x2; |
| result2.Intersection(y2); |
| EXPECT_TRUE(result2.Empty()) << result2; |
| |
| const IntervalSet<int64_t> x3({{-1, 5}, {5, 10}}); |
| const IntervalSet<int64_t> y3({{-10, -1}, {10, 95}}); |
| |
| IntervalSet<int64_t> result3 = x3; |
| result3.Intersection(y3); |
| EXPECT_TRUE(result3.Empty()) << result3; |
| } |
| |
| TEST_F(IntervalSetTest, |
| IntervalSetIntersectionAlternatingIntersectingIntervals) { |
| const IntervalSet<int> x1({{0, 10}}); |
| const IntervalSet<int> y1({{-50, 1}, {9, 95}}); |
| const IntervalSet<int> expected_result1({{0, 1}, {9, 10}}); |
| |
| IntervalSet<int> result1 = x1; |
| result1.Intersection(y1); |
| EXPECT_EQ(result1, expected_result1); |
| |
| const IntervalSet<int16_t> x2({{0, 10}, {20, 30}, {40, 90}}); |
| const IntervalSet<int16_t> y2( |
| {{-50, -40}, {-2, 2}, {9, 21}, {32, 41}, {85, 95}}); |
| const IntervalSet<int16_t> expected_result2( |
| {{0, 2}, {9, 10}, {20, 21}, {40, 41}, {85, 90}}); |
| |
| IntervalSet<int16_t> result2 = x2; |
| result2.Intersection(y2); |
| EXPECT_EQ(result2, expected_result2); |
| |
| const IntervalSet<int64_t> x3({{-1, 5}, {5, 10}}); |
| const IntervalSet<int64_t> y3({{-10, 3}, {4, 95}}); |
| const IntervalSet<int64_t> expected_result3({{-1, 3}, {4, 10}}); |
| |
| IntervalSet<int64_t> result3 = x3; |
| result3.Intersection(y3); |
| EXPECT_EQ(result3, expected_result3); |
| } |
| |
| #endif // 0 |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionIdentical) { |
| IntervalSet<int> copy(is); |
| EXPECT_TRUE(copy.Intersects(is)); |
| EXPECT_TRUE(is.Intersects(copy)); |
| is.Intersection(copy); |
| EXPECT_EQ(copy, is); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionSuperset) { |
| IntervalSet<int> mine(-1, 10000); |
| EXPECT_TRUE(mine.Intersects(is)); |
| EXPECT_TRUE(is.Intersects(mine)); |
| mine.Intersection(is); |
| EXPECT_EQ(is, mine); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionSubset) { |
| IntervalSet<int> copy(is); |
| IntervalSet<int> theirs(-1, 10000); |
| EXPECT_TRUE(copy.Intersects(theirs)); |
| EXPECT_TRUE(theirs.Intersects(copy)); |
| is.Intersection(theirs); |
| EXPECT_EQ(copy, is); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetIntersectionLargeSet) { |
| IntervalSet<int> mine, theirs; |
| // mine: [0, 9), [10, 19), ..., [990, 999) |
| for (int i = 0; i < 1000; i += 10) { |
| mine.Add(i, i + 9); |
| } |
| |
| theirs.Add(500, 520); |
| theirs.Add(535, 545); |
| theirs.Add(801, 809); |
| EXPECT_TRUE(mine.Intersects(theirs)); |
| EXPECT_TRUE(theirs.Intersects(mine)); |
| mine.Intersection(theirs); |
| EXPECT_TRUE(Check(mine, 5, 500, 509, 510, 519, 535, 539, 540, 545, 801, 809)); |
| EXPECT_TRUE(mine.Intersects(theirs)); |
| EXPECT_TRUE(theirs.Intersects(mine)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifference) { |
| is.Difference(other); |
| EXPECT_TRUE(Check(is, 10, 100, 200, 300, 350, 360, 370, 380, 400, 530, 600, |
| 700, 770, 900, 1000, 1100, 1200, 1900, 2000, 2100, 2200)); |
| IntervalSet<int> copy = is; |
| is.Difference(copy); |
| EXPECT_TRUE(is.Empty()); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceSingleBounds) { |
| std::vector<Interval<int>> ivals; |
| other.Get(&ivals); |
| for (size_t i = 0; i < ivals.size(); ++i) { |
| is.Difference(ivals[i].min(), ivals[i].max()); |
| } |
| EXPECT_TRUE(Check(is, 10, 100, 200, 300, 350, 360, 370, 380, 400, 530, 600, |
| 700, 770, 900, 1000, 1100, 1200, 1900, 2000, 2100, 2200)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceSingleInterval) { |
| std::vector<Interval<int>> ivals; |
| other.Get(&ivals); |
| for (size_t i = 0; i < ivals.size(); ++i) { |
| is.Difference(ivals[i]); |
| } |
| EXPECT_TRUE(Check(is, 10, 100, 200, 300, 350, 360, 370, 380, 400, 530, 600, |
| 700, 770, 900, 1000, 1100, 1200, 1900, 2000, 2100, 2200)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceAlternatingIntervals) { |
| IntervalSet<int> mine, theirs; |
| mine.Add(10, 20); |
| mine.Add(40, 50); |
| mine.Add(60, 70); |
| theirs.Add(25, 39); |
| theirs.Add(55, 59); |
| theirs.Add(75, 79); |
| |
| mine.Difference(theirs); |
| EXPECT_TRUE(Check(mine, 3, 10, 20, 40, 50, 60, 70)); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceEmptyMine) { |
| IntervalSet<string> mine, theirs; |
| theirs.Add("a", "b"); |
| |
| mine.Difference(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceEmptyTheirs) { |
| IntervalSet<string> mine, theirs; |
| mine.Add("a", "b"); |
| |
| mine.Difference(theirs); |
| EXPECT_EQ(1u, mine.Size()); |
| EXPECT_EQ("a", mine.begin()->min()); |
| EXPECT_EQ("b", mine.begin()->max()); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceTheirsBeforeMine) { |
| IntervalSet<string> mine, theirs; |
| mine.Add("y", "z"); |
| theirs.Add("a", "b"); |
| |
| mine.Difference(theirs); |
| EXPECT_EQ(1u, mine.Size()); |
| EXPECT_EQ("y", mine.begin()->min()); |
| EXPECT_EQ("z", mine.begin()->max()); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceMineBeforeTheirs) { |
| IntervalSet<string> mine, theirs; |
| mine.Add("a", "b"); |
| theirs.Add("y", "z"); |
| |
| mine.Difference(theirs); |
| EXPECT_EQ(1u, mine.Size()); |
| EXPECT_EQ("a", mine.begin()->min()); |
| EXPECT_EQ("b", mine.begin()->max()); |
| } |
| |
| TEST_F(IntervalSetTest, IntervalSetDifferenceIdentical) { |
| IntervalSet<string> mine; |
| mine.Add("a", "b"); |
| mine.Add("c", "d"); |
| IntervalSet<string> theirs(mine); |
| |
| mine.Difference(theirs); |
| EXPECT_TRUE(mine.Empty()); |
| } |
| |
| TEST_F(IntervalSetTest, EmptyComplement) { |
| // The complement of an empty set is the input interval: |
| IntervalSet<int> iset; |
| iset.Complement(100, 200); |
| EXPECT_TRUE(Check(iset, 1, 100, 200)); |
| } |
| |
| TEST(IntervalSetMultipleCompactionTest, OuterCovering) { |
| IntervalSet<int> iset; |
| // First add a bunch of disjoint ranges |
| iset.Add(100, 150); |
| iset.Add(200, 250); |
| iset.Add(300, 350); |
| iset.Add(400, 450); |
| EXPECT_TRUE(Check(iset, 4, 100, 150, 200, 250, 300, 350, 400, 450)); |
| // Now add a big range that covers all of these ranges |
| iset.Add(0, 500); |
| EXPECT_TRUE(Check(iset, 1, 0, 500)); |
| } |
| |
| TEST(IntervalSetMultipleCompactionTest, InnerCovering) { |
| IntervalSet<int> iset; |
| // First add a bunch of disjoint ranges |
| iset.Add(100, 150); |
| iset.Add(200, 250); |
| iset.Add(300, 350); |
| iset.Add(400, 450); |
| EXPECT_TRUE(Check(iset, 4, 100, 150, 200, 250, 300, 350, 400, 450)); |
| // Now add a big range that partially covers the left and right most ranges. |
| iset.Add(125, 425); |
| EXPECT_TRUE(Check(iset, 1, 100, 450)); |
| } |
| |
| TEST(IntervalSetMultipleCompactionTest, LeftCovering) { |
| IntervalSet<int> iset; |
| // First add a bunch of disjoint ranges |
| iset.Add(100, 150); |
| iset.Add(200, 250); |
| iset.Add(300, 350); |
| iset.Add(400, 450); |
| EXPECT_TRUE(Check(iset, 4, 100, 150, 200, 250, 300, 350, 400, 450)); |
| // Now add a big range that partially covers the left most range. |
| iset.Add(125, 500); |
| EXPECT_TRUE(Check(iset, 1, 100, 500)); |
| } |
| |
| TEST(IntervalSetMultipleCompactionTest, RightCovering) { |
| IntervalSet<int> iset; |
| // First add a bunch of disjoint ranges |
| iset.Add(100, 150); |
| iset.Add(200, 250); |
| iset.Add(300, 350); |
| iset.Add(400, 450); |
| EXPECT_TRUE(Check(iset, 4, 100, 150, 200, 250, 300, 350, 400, 450)); |
| // Now add a big range that partially covers the right most range. |
| iset.Add(0, 425); |
| EXPECT_TRUE(Check(iset, 1, 0, 450)); |
| } |
| |
| // Helper method for testing and verifying the results of a one-interval |
| // completement case. |
| static bool CheckOneComplement(int add_min, |
| int add_max, |
| int comp_min, |
| int comp_max, |
| int count, |
| ...) { |
| IntervalSet<int> iset; |
| iset.Add(add_min, add_max); |
| iset.Complement(comp_min, comp_max); |
| bool result = true; |
| va_list ap; |
| va_start(ap, count); |
| if (!VA_Check(iset, count, ap)) { |
| result = false; |
| } |
| va_end(ap); |
| return result; |
| } |
| |
| TEST_F(IntervalSetTest, SingleIntervalComplement) { |
| // Verify the complement of a set with one interval (i): |
| // |----- i -----| |
| // |----- args -----| |
| EXPECT_TRUE(CheckOneComplement(0, 10, 50, 150, 1, 50, 150)); |
| |
| // |----- i -----| |
| // |----- args -----| |
| EXPECT_TRUE(CheckOneComplement(50, 150, 0, 100, 1, 0, 50)); |
| |
| // |----- i -----| |
| // |----- args -----| |
| EXPECT_TRUE(CheckOneComplement(50, 150, 50, 150, 0)); |
| |
| // |---------- i ----------| |
| // |----- args -----| |
| EXPECT_TRUE(CheckOneComplement(50, 500, 100, 300, 0)); |
| |
| // |----- i -----| |
| // |---------- args ----------| |
| EXPECT_TRUE(CheckOneComplement(50, 500, 0, 800, 2, 0, 50, 500, 800)); |
| |
| // |----- i -----| |
| // |----- args -----| |
| EXPECT_TRUE(CheckOneComplement(50, 150, 100, 300, 1, 150, 300)); |
| |
| // |----- i -----| |
| // |----- args -----| |
| EXPECT_TRUE(CheckOneComplement(50, 150, 200, 300, 1, 200, 300)); |
| } |
| |
| // Helper method that copies <iset> and takes its complement, |
| // returning false if Check succeeds. |
| static bool CheckComplement(const IntervalSet<int>& iset, |
| int comp_min, |
| int comp_max, |
| int count, |
| ...) { |
| IntervalSet<int> iset_copy = iset; |
| iset_copy.Complement(comp_min, comp_max); |
| bool result = true; |
| va_list ap; |
| va_start(ap, count); |
| if (!VA_Check(iset_copy, count, ap)) { |
| result = false; |
| } |
| va_end(ap); |
| return result; |
| } |
| |
| TEST_F(IntervalSetTest, MultiIntervalComplement) { |
| // Initialize a small test set: |
| IntervalSet<int> iset; |
| iset.Add(100, 200); |
| iset.Add(300, 400); |
| iset.Add(500, 600); |
| |
| // |----- i -----| |
| // |----- comp -----| |
| EXPECT_TRUE(CheckComplement(iset, 0, 50, 1, 0, 50)); |
| |
| // |----- i -----| |
| // |----- comp -----| |
| EXPECT_TRUE(CheckComplement(iset, 0, 200, 1, 0, 100)); |
| EXPECT_TRUE(CheckComplement(iset, 0, 220, 2, 0, 100, 200, 220)); |
| |
| // |----- i -----| |
| // |----- comp -----| |
| EXPECT_TRUE(CheckComplement(iset, 100, 600, 2, 200, 300, 400, 500)); |
| |
| // |---------- i ----------| |
| // |----- comp -----| |
| EXPECT_TRUE(CheckComplement(iset, 300, 400, 0)); |
| EXPECT_TRUE(CheckComplement(iset, 250, 400, 1, 250, 300)); |
| EXPECT_TRUE(CheckComplement(iset, 300, 450, 1, 400, 450)); |
| EXPECT_TRUE(CheckComplement(iset, 250, 450, 2, 250, 300, 400, 450)); |
| |
| // |----- i -----| |
| // |---------- comp ----------| |
| EXPECT_TRUE( |
| CheckComplement(iset, 0, 700, 4, 0, 100, 200, 300, 400, 500, 600, 700)); |
| |
| // |----- i -----| |
| // |----- comp -----| |
| EXPECT_TRUE(CheckComplement(iset, 400, 700, 2, 400, 500, 600, 700)); |
| EXPECT_TRUE(CheckComplement(iset, 350, 700, 2, 400, 500, 600, 700)); |
| |
| // |----- i -----| |
| // |----- comp -----| |
| EXPECT_TRUE(CheckComplement(iset, 700, 800, 1, 700, 800)); |
| } |
| |
| // Verifies ToString, operator<< don't assert. |
| // TODO(rtenneti): Implement ToString() method of IntervalSet. |
| TEST_F(IntervalSetTest, DISABLED_ToString) { |
| IntervalSet<int> iset; |
| iset.Add(300, 400); |
| iset.Add(100, 200); |
| iset.Add(500, 600); |
| EXPECT_TRUE(!iset.ToString().empty()); |
| VLOG(2) << iset.ToString(); |
| // Order and format of ToString() output is guaranteed. |
| EXPECT_EQ("[100, 200) [300, 400) [500, 600)", iset.ToString()); |
| EXPECT_EQ("[1, 2)", IntervalSet<int>(1, 2).ToString()); |
| EXPECT_EQ("", IntervalSet<int>().ToString()); |
| } |
| |
| TEST_F(IntervalSetTest, ConstructionDiscardsEmptyInterval) { |
| EXPECT_TRUE(IntervalSet<int>(Interval<int>(2, 2)).Empty()); |
| EXPECT_TRUE(IntervalSet<int>(2, 2).Empty()); |
| EXPECT_FALSE(IntervalSet<int>(Interval<int>(2, 3)).Empty()); |
| EXPECT_FALSE(IntervalSet<int>(2, 3).Empty()); |
| } |
| |
| TEST_F(IntervalSetTest, Swap) { |
| IntervalSet<int> a, b; |
| a.Add(300, 400); |
| b.Add(100, 200); |
| b.Add(500, 600); |
| a.Swap(&b); |
| EXPECT_TRUE(Check(a, 2, 100, 200, 500, 600)); |
| EXPECT_TRUE(Check(b, 1, 300, 400)); |
| swap(a, b); |
| EXPECT_TRUE(Check(a, 1, 300, 400)); |
| EXPECT_TRUE(Check(b, 2, 100, 200, 500, 600)); |
| } |
| |
| // TODO(rtenneti): Enabled these tests. |
| #if 0 |
| static void BM_Difference(int iters) { |
| // StopBenchmarkTiming(); |
| IntervalSet<int> difference_set; |
| int start = 10; |
| for (int i = 0; i < 1000000; ++i) { |
| difference_set.Add(start, start+5); |
| start += 7; |
| } |
| |
| // Create an interval somewhere in the middle of the difference set. |
| // StartBenchmarkTiming(); |
| for (int i = 0; i < iters; ++i) { |
| IntervalSet<int> initial(1000000, 1000020); |
| initial.Difference(difference_set); |
| } |
| } |
| |
| BENCHMARK(BM_Difference); |
| |
| static void BM_IntersectionSmallAndLarge(int iters, int size) { |
| // Intersects constant size 'mine' with large 'theirs'. |
| StopBenchmarkTiming(); |
| IntervalSet<int> theirs; |
| for (int i = 0; i < size; ++i) { |
| theirs.Add(2 * i, 2 * i + 1); |
| } |
| |
| StartBenchmarkTiming(); |
| for (int i = 0; i < iters; ++i) { |
| // 'mine' starts in the middle of 'theirs'. |
| IntervalSet<int> mine(size, size + 10); |
| mine.Intersection(theirs); |
| } |
| } |
| |
| BENCHMARK_RANGE(BM_IntersectionSmallAndLarge, 0, 1 << 23); |
| |
| static void BM_IntersectionIdentical(int iters, int size) { |
| // Intersects identical 'mine' and 'theirs'. |
| StopBenchmarkTiming(); |
| IntervalSet<int> mine; |
| for (int i = 0; i < size; ++i) { |
| mine.Add(2 * i, 2 * i + 1); |
| } |
| IntervalSet<int> theirs(mine); |
| |
| StartBenchmarkTiming(); |
| for (int i = 0; i < iters; ++i) { |
| mine.Intersection(theirs); |
| } |
| } |
| |
| BENCHMARK_RANGE(BM_IntersectionIdentical, 0, 1 << 23); |
| |
| class IntervalSetInitTest : public testing::Test { |
| protected: |
| const std::vector<Interval<int>> intervals_{{0, 1}, {2, 4}}; |
| }; |
| |
| TEST_F(IntervalSetInitTest, DirectInit) { |
| std::initializer_list<Interval<int>> il = {{0, 1}, {2, 3}, {3, 4}}; |
| IntervalSet<int> s(il); |
| EXPECT_THAT(s, ElementsAreArray(intervals_)); |
| } |
| |
| TEST_F(IntervalSetInitTest, CopyInit) { |
| std::initializer_list<Interval<int>> il = {{0, 1}, {2, 3}, {3, 4}}; |
| IntervalSet<int> s = il; |
| EXPECT_THAT(s, ElementsAreArray(intervals_)); |
| } |
| |
| TEST_F(IntervalSetInitTest, AssignIterPair) { |
| IntervalSet<int> s(0, 1000); // Make sure assign clears. |
| s.assign(intervals_.begin(), intervals_.end()); |
| EXPECT_THAT(s, ElementsAreArray(intervals_)); |
| } |
| |
| TEST_F(IntervalSetInitTest, AssignInitList) { |
| IntervalSet<int> s(0, 1000); // Make sure assign clears. |
| s.assign({{0, 1}, {2, 3}, {3, 4}}); |
| EXPECT_THAT(s, ElementsAreArray(intervals_)); |
| } |
| |
| TEST_F(IntervalSetInitTest, AssignmentInitList) { |
| std::initializer_list<Interval<int>> il = {{0, 1}, {2, 3}, {3, 4}}; |
| IntervalSet<int> s; |
| s = il; |
| EXPECT_THAT(s, ElementsAreArray(intervals_)); |
| } |
| |
| TEST_F(IntervalSetInitTest, BracedInitThenBracedAssign) { |
| IntervalSet<int> s{{0, 1}, {2, 3}, {3, 4}}; |
| s = {{0, 1}, {2, 4}}; |
| EXPECT_THAT(s, ElementsAreArray(intervals_)); |
| } |
| |
| #endif // 0 |
| |
| } // namespace |
| } // namespace test |
| } // namespace net |