// Copyright (c) 2016 Google Inc. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include <memory> #include <vector> #include "gmock/gmock.h" #include "source/opt/iterator.h" #include "source/util/make_unique.h" namespace spvtools { namespace opt { namespace { using ::testing::ContainerEq; TEST(Iterator, IncrementDeref) { const int count = 100; std::vector<std::unique_ptr<int>> data; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); } UptrVectorIterator<int> it(&data, data.begin()); UptrVectorIterator<int> end(&data, data.end()); EXPECT_EQ(*data[0], *it); for (int i = 1; i < count; ++i) { EXPECT_NE(end, it); EXPECT_EQ(*data[i], *(++it)); } EXPECT_EQ(end, ++it); } TEST(Iterator, DecrementDeref) { const int count = 100; std::vector<std::unique_ptr<int>> data; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); } UptrVectorIterator<int> begin(&data, data.begin()); UptrVectorIterator<int> it(&data, data.end()); for (int i = count - 1; i >= 0; --i) { EXPECT_NE(begin, it); EXPECT_EQ(*data[i], *(--it)); } EXPECT_EQ(begin, it); } TEST(Iterator, PostIncrementDeref) { const int count = 100; std::vector<std::unique_ptr<int>> data; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); } UptrVectorIterator<int> it(&data, data.begin()); UptrVectorIterator<int> end(&data, data.end()); for (int i = 0; i < count; ++i) { EXPECT_NE(end, it); EXPECT_EQ(*data[i], *(it++)); } EXPECT_EQ(end, it); } TEST(Iterator, PostDecrementDeref) { const int count = 100; std::vector<std::unique_ptr<int>> data; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); } UptrVectorIterator<int> begin(&data, data.begin()); UptrVectorIterator<int> end(&data, data.end()); UptrVectorIterator<int> it(&data, data.end()); EXPECT_EQ(end, it--); for (int i = count - 1; i >= 1; --i) { EXPECT_EQ(*data[i], *(it--)); } // Decrementing .begin() is undefined behavior. EXPECT_EQ(*data[0], *it); } TEST(Iterator, Access) { const int count = 100; std::vector<std::unique_ptr<int>> data; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); } UptrVectorIterator<int> it(&data, data.begin()); for (int i = 0; i < count; ++i) EXPECT_EQ(*data[i], it[i]); } TEST(Iterator, Comparison) { const int count = 100; std::vector<std::unique_ptr<int>> data; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); } UptrVectorIterator<int> it(&data, data.begin()); UptrVectorIterator<int> end(&data, data.end()); for (int i = 0; i < count; ++i, ++it) EXPECT_TRUE(it < end); EXPECT_EQ(end, it); } TEST(Iterator, InsertBeginEnd) { const int count = 100; std::vector<std::unique_ptr<int>> data; std::vector<int> expected; std::vector<int> actual; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); expected.push_back(i); } // Insert at the beginning expected.insert(expected.begin(), -100); UptrVectorIterator<int> begin(&data, data.begin()); auto insert_point = begin.InsertBefore(MakeUnique<int>(-100)); for (int i = 0; i < count + 1; ++i) { actual.push_back(*(insert_point++)); } EXPECT_THAT(actual, ContainerEq(expected)); // Insert at the end expected.push_back(-42); expected.push_back(-36); expected.push_back(-77); UptrVectorIterator<int> end(&data, data.end()); end = end.InsertBefore(MakeUnique<int>(-77)); end = end.InsertBefore(MakeUnique<int>(-36)); end = end.InsertBefore(MakeUnique<int>(-42)); actual.clear(); begin = UptrVectorIterator<int>(&data, data.begin()); for (int i = 0; i < count + 4; ++i) { actual.push_back(*(begin++)); } EXPECT_THAT(actual, ContainerEq(expected)); } TEST(Iterator, InsertMiddle) { const int count = 100; std::vector<std::unique_ptr<int>> data; std::vector<int> expected; std::vector<int> actual; for (int i = 0; i < count; ++i) { data.emplace_back(new int(i)); expected.push_back(i); } const int insert_pos = 42; expected.insert(expected.begin() + insert_pos, -100); expected.insert(expected.begin() + insert_pos, -42); UptrVectorIterator<int> it(&data, data.begin()); for (int i = 0; i < insert_pos; ++i) ++it; it = it.InsertBefore(MakeUnique<int>(-100)); it = it.InsertBefore(MakeUnique<int>(-42)); auto begin = UptrVectorIterator<int>(&data, data.begin()); for (int i = 0; i < count + 2; ++i) { actual.push_back(*(begin++)); } EXPECT_THAT(actual, ContainerEq(expected)); } TEST(IteratorRange, Interface) { const uint32_t count = 100; std::vector<std::unique_ptr<uint32_t>> data; for (uint32_t i = 0; i < count; ++i) { data.emplace_back(new uint32_t(i)); } auto b = UptrVectorIterator<uint32_t>(&data, data.begin()); auto e = UptrVectorIterator<uint32_t>(&data, data.end()); auto range = IteratorRange<decltype(b)>(b, e); EXPECT_EQ(b, range.begin()); EXPECT_EQ(e, range.end()); EXPECT_FALSE(range.empty()); EXPECT_EQ(count, range.size()); EXPECT_EQ(0u, *range.begin()); EXPECT_EQ(99u, *(--range.end())); // IteratorRange itself is immutable. ++b, --e; EXPECT_EQ(count, range.size()); ++range.begin(), --range.end(); EXPECT_EQ(count, range.size()); } TEST(Iterator, FilterIterator) { struct Placeholder { int val; }; std::vector<Placeholder> data = {{1}, {2}, {3}, {4}, {5}, {6}, {7}, {8}, {9}, {10}}; // Predicate to only consider odd values. struct Predicate { bool operator()(const Placeholder& data) { return data.val % 2; } }; Predicate pred; auto filter_range = MakeFilterIteratorRange(data.begin(), data.end(), pred); EXPECT_EQ(filter_range.begin().Get(), data.begin()); EXPECT_EQ(filter_range.end(), filter_range.begin().GetEnd()); for (Placeholder& data : filter_range) { EXPECT_EQ(data.val % 2, 1); } for (auto it = filter_range.begin(); it != filter_range.end(); it++) { EXPECT_EQ(it->val % 2, 1); EXPECT_EQ((*it).val % 2, 1); } for (auto it = filter_range.begin(); it != filter_range.end(); ++it) { EXPECT_EQ(it->val % 2, 1); EXPECT_EQ((*it).val % 2, 1); } EXPECT_EQ(MakeFilterIterator(data.begin(), data.end(), pred).Get(), data.begin()); EXPECT_EQ(MakeFilterIterator(data.end(), data.end(), pred).Get(), data.end()); EXPECT_EQ(MakeFilterIterator(data.begin(), data.end(), pred).GetEnd(), MakeFilterIterator(data.end(), data.end(), pred)); EXPECT_NE(MakeFilterIterator(data.begin(), data.end(), pred), MakeFilterIterator(data.end(), data.end(), pred)); // Empty range: no values satisfies the predicate. auto empty_range = MakeFilterIteratorRange( data.begin(), data.end(), [](const Placeholder& data) { return data.val > 10; }); EXPECT_EQ(empty_range.begin(), empty_range.end()); } } // namespace } // namespace opt } // namespace spvtools