//===- IteratorTest.cpp - Unit tests for iterator utilities ---------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/ADT/iterator.h" #include "llvm/ADT/STLExtras.h" #include "llvm/ADT/SmallVector.h" #include "gtest/gtest.h" using namespace llvm; namespace { template struct Shadow; struct WeirdIter : std::iterator, Shadow<1>, Shadow<2>, Shadow<3>> {}; struct AdaptedIter : iterator_adaptor_base {}; // Test that iterator_adaptor_base forwards typedefs, if value_type is // unchanged. static_assert(std::is_same>::value, ""); static_assert( std::is_same>::value, ""); static_assert(std::is_same>::value, ""); static_assert(std::is_same>::value, ""); TEST(PointeeIteratorTest, Basic) { int arr[4] = {1, 2, 3, 4}; SmallVector V; V.push_back(&arr[0]); V.push_back(&arr[1]); V.push_back(&arr[2]); V.push_back(&arr[3]); typedef pointee_iterator::const_iterator> test_iterator; test_iterator Begin, End; Begin = V.begin(); End = test_iterator(V.end()); test_iterator I = Begin; for (int i = 0; i < 4; ++i) { EXPECT_EQ(*V[i], *I); EXPECT_EQ(I, Begin + i); EXPECT_EQ(I, std::next(Begin, i)); test_iterator J = Begin; J += i; EXPECT_EQ(I, J); EXPECT_EQ(*V[i], Begin[i]); EXPECT_NE(I, End); EXPECT_GT(End, I); EXPECT_LT(I, End); EXPECT_GE(I, Begin); EXPECT_LE(Begin, I); EXPECT_EQ(i, I - Begin); EXPECT_EQ(i, std::distance(Begin, I)); EXPECT_EQ(Begin, I - i); test_iterator K = I++; EXPECT_EQ(K, std::prev(I)); } EXPECT_EQ(End, I); } TEST(PointeeIteratorTest, SmartPointer) { SmallVector, 4> V; V.push_back(make_unique(1)); V.push_back(make_unique(2)); V.push_back(make_unique(3)); V.push_back(make_unique(4)); typedef pointee_iterator< SmallVectorImpl>::const_iterator> test_iterator; test_iterator Begin, End; Begin = V.begin(); End = test_iterator(V.end()); test_iterator I = Begin; for (int i = 0; i < 4; ++i) { EXPECT_EQ(*V[i], *I); EXPECT_EQ(I, Begin + i); EXPECT_EQ(I, std::next(Begin, i)); test_iterator J = Begin; J += i; EXPECT_EQ(I, J); EXPECT_EQ(*V[i], Begin[i]); EXPECT_NE(I, End); EXPECT_GT(End, I); EXPECT_LT(I, End); EXPECT_GE(I, Begin); EXPECT_LE(Begin, I); EXPECT_EQ(i, I - Begin); EXPECT_EQ(i, std::distance(Begin, I)); EXPECT_EQ(Begin, I - i); test_iterator K = I++; EXPECT_EQ(K, std::prev(I)); } EXPECT_EQ(End, I); } TEST(PointeeIteratorTest, Range) { int A[] = {1, 2, 3, 4}; SmallVector V{&A[0], &A[1], &A[2], &A[3]}; int I = 0; for (int II : make_pointee_range(V)) EXPECT_EQ(A[I++], II); } TEST(FilterIteratorTest, Lambda) { auto IsOdd = [](int N) { return N % 2 == 1; }; int A[] = {0, 1, 2, 3, 4, 5, 6}; auto Range = make_filter_range(A, IsOdd); SmallVector Actual(Range.begin(), Range.end()); EXPECT_EQ((SmallVector{1, 3, 5}), Actual); } TEST(FilterIteratorTest, CallableObject) { int Counter = 0; struct Callable { int &Counter; Callable(int &Counter) : Counter(Counter) {} bool operator()(int N) { Counter++; return N % 2 == 1; } }; Callable IsOdd(Counter); int A[] = {0, 1, 2, 3, 4, 5, 6}; auto Range = make_filter_range(A, IsOdd); EXPECT_EQ(2, Counter); SmallVector Actual(Range.begin(), Range.end()); EXPECT_GE(Counter, 7); EXPECT_EQ((SmallVector{1, 3, 5}), Actual); } TEST(FilterIteratorTest, FunctionPointer) { bool (*IsOdd)(int) = [](int N) { return N % 2 == 1; }; int A[] = {0, 1, 2, 3, 4, 5, 6}; auto Range = make_filter_range(A, IsOdd); SmallVector Actual(Range.begin(), Range.end()); EXPECT_EQ((SmallVector{1, 3, 5}), Actual); } TEST(FilterIteratorTest, Composition) { auto IsOdd = [](int N) { return N % 2 == 1; }; std::unique_ptr A[] = {make_unique(0), make_unique(1), make_unique(2), make_unique(3), make_unique(4), make_unique(5), make_unique(6)}; using PointeeIterator = pointee_iterator *>; auto Range = make_filter_range( make_range(PointeeIterator(std::begin(A)), PointeeIterator(std::end(A))), IsOdd); SmallVector Actual(Range.begin(), Range.end()); EXPECT_EQ((SmallVector{1, 3, 5}), Actual); } TEST(FilterIteratorTest, InputIterator) { struct InputIterator : iterator_adaptor_base { using BaseT = iterator_adaptor_base; InputIterator(int *It) : BaseT(It) {} }; auto IsOdd = [](int N) { return N % 2 == 1; }; int A[] = {0, 1, 2, 3, 4, 5, 6}; auto Range = make_filter_range( make_range(InputIterator(std::begin(A)), InputIterator(std::end(A))), IsOdd); SmallVector Actual(Range.begin(), Range.end()); EXPECT_EQ((SmallVector{1, 3, 5}), Actual); } TEST(PointerIterator, Basic) { int A[] = {1, 2, 3, 4}; pointer_iterator Begin(std::begin(A)), End(std::end(A)); EXPECT_EQ(A, *Begin); ++Begin; EXPECT_EQ(A + 1, *Begin); ++Begin; EXPECT_EQ(A + 2, *Begin); ++Begin; EXPECT_EQ(A + 3, *Begin); ++Begin; EXPECT_EQ(Begin, End); } TEST(PointerIterator, Const) { int A[] = {1, 2, 3, 4}; const pointer_iterator Begin(std::begin(A)); EXPECT_EQ(A, *Begin); EXPECT_EQ(A + 1, std::next(*Begin, 1)); EXPECT_EQ(A + 2, std::next(*Begin, 2)); EXPECT_EQ(A + 3, std::next(*Begin, 3)); EXPECT_EQ(A + 4, std::next(*Begin, 4)); } TEST(PointerIterator, Range) { int A[] = {1, 2, 3, 4}; int I = 0; for (int *P : make_pointer_range(A)) EXPECT_EQ(A + I++, P); } TEST(ZipIteratorTest, Basic) { using namespace std; const SmallVector pi{3, 1, 4, 1, 5, 9}; SmallVector odd{1, 1, 0, 1, 1, 1}; const char message[] = "yynyyy\0"; for (auto tup : zip(pi, odd, message)) { EXPECT_EQ(get<0>(tup) & 0x01, get<1>(tup)); EXPECT_EQ(get<0>(tup) & 0x01 ? 'y' : 'n', get<2>(tup)); } // note the rvalue for (auto tup : zip(pi, SmallVector{1, 1, 0, 1, 1})) { EXPECT_EQ(get<0>(tup) & 0x01, get<1>(tup)); } } TEST(ZipIteratorTest, ZipFirstBasic) { using namespace std; const SmallVector pi{3, 1, 4, 1, 5, 9}; unsigned iters = 0; for (auto tup : zip_first(SmallVector{1, 1, 0, 1}, pi)) { EXPECT_EQ(get<0>(tup), get<1>(tup) & 0x01); iters += 1; } EXPECT_EQ(iters, 4u); } TEST(ZipIteratorTest, Mutability) { using namespace std; const SmallVector pi{3, 1, 4, 1, 5, 9}; char message[] = "hello zip\0"; for (auto tup : zip(pi, message, message)) { EXPECT_EQ(get<1>(tup), get<2>(tup)); get<2>(tup) = get<0>(tup) & 0x01 ? 'y' : 'n'; } // note the rvalue for (auto tup : zip(message, "yynyyyzip\0")) { EXPECT_EQ(get<0>(tup), get<1>(tup)); } } TEST(ZipIteratorTest, ZipFirstMutability) { using namespace std; vector pi{3, 1, 4, 1, 5, 9}; unsigned iters = 0; for (auto tup : zip_first(SmallVector{1, 1, 0, 1}, pi)) { get<1>(tup) = get<0>(tup); iters += 1; } EXPECT_EQ(iters, 4u); for (auto tup : zip_first(SmallVector{1, 1, 0, 1}, pi)) { EXPECT_EQ(get<0>(tup), get<1>(tup)); } } TEST(ZipIteratorTest, Filter) { using namespace std; vector pi{3, 1, 4, 1, 5, 9}; unsigned iters = 0; // pi is length 6, but the zip RHS is length 7. auto zipped = zip_first(pi, vector{1, 1, 0, 1, 1, 1, 0}); for (auto tup : make_filter_range( zipped, [](decltype(zipped)::value_type t) { return get<1>(t); })) { EXPECT_EQ(get<0>(tup) & 0x01, get<1>(tup)); get<0>(tup) += 1; iters += 1; } // Should have skipped pi[2]. EXPECT_EQ(iters, 5u); // Ensure that in-place mutation works. EXPECT_TRUE(all_of(pi, [](unsigned n) { return (n & 0x01) == 0; })); } TEST(ZipIteratorTest, Reverse) { using namespace std; vector ascending{0, 1, 2, 3, 4, 5}; auto zipped = zip_first(ascending, vector{0, 1, 0, 1, 0, 1}); unsigned last = 6; for (auto tup : reverse(zipped)) { // Check that this is in reverse. EXPECT_LT(get<0>(tup), last); last = get<0>(tup); EXPECT_EQ(get<0>(tup) & 0x01, get<1>(tup)); } auto odds = [](decltype(zipped)::value_type tup) { return get<1>(tup); }; last = 6; for (auto tup : make_filter_range(reverse(zipped), odds)) { EXPECT_LT(get<0>(tup), last); last = get<0>(tup); EXPECT_TRUE(get<0>(tup) & 0x01); get<0>(tup) += 1; } // Ensure that in-place mutation works. EXPECT_TRUE(all_of(ascending, [](unsigned n) { return (n & 0x01) == 0; })); } } // anonymous namespace