//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14, c++17 // constexpr iterator& operator++(); // constexpr void operator++(int); // constexpr iterator operator++(int) // requires ref-is-glvalue && forward_range && // forward_range>; #include #include #include "test_macros.h" #include "../types.h" constexpr bool test() { // This way if we read past end we'll catch the error. int buffer1[2][4] = {{1, 2, 3, 4}, {5, 6, 7, 8}}; int dummy = 42; (void) dummy; int buffer2[2][4] = {{9, 10, 11, 12}, {13, 14, 15, 16}}; // operator++(int); { std::ranges::join_view jv(buffer1); auto iter = jv.begin(); for (int i = 1; i < 9; ++i) { assert(*iter++ == i); } } { using IntView = ValueView; IntView children[4] = {IntView(buffer1[0]), IntView(buffer1[1]), IntView(buffer2[0]), IntView(buffer2[1])}; std::ranges::join_view jv(ValueView{children}); auto iter = jv.begin(); for (int i = 1; i < 17; ++i) { assert(*iter == i); iter++; } ASSERT_SAME_TYPE(decltype(iter++), void); } { std::ranges::join_view jv(buffer1); auto iter = std::next(jv.begin(), 7); assert(*iter++ == 8); assert(iter == jv.end()); } { int small[2][1] = {{1}, {2}}; std::ranges::join_view jv(small); auto iter = jv.begin(); for (int i = 1; i < 3; ++i) { assert(*iter++ == i); } } // Has some empty children. { CopyableChild children[4] = {CopyableChild(buffer1[0], 4), CopyableChild(buffer1[1], 0), CopyableChild(buffer2[0], 1), CopyableChild(buffer2[1], 0)}; auto jv = std::ranges::join_view(ParentView(children)); auto iter = jv.begin(); assert(*iter == 1); iter++; assert(*iter == 2); iter++; assert(*iter == 3); iter++; assert(*iter == 4); iter++; assert(*iter == 9); iter++; assert(iter == jv.end()); } // Parent is empty. { CopyableChild children[4] = {CopyableChild(buffer1[0]), CopyableChild(buffer1[1]), CopyableChild(buffer2[0]), CopyableChild(buffer2[1])}; std::ranges::join_view jv(ParentView(children, 0)); assert(jv.begin() == jv.end()); } // Parent size is one. { CopyableChild children[1] = {CopyableChild(buffer1[0])}; std::ranges::join_view jv(ParentView(children, 1)); auto iter = jv.begin(); assert(*iter == 1); iter++; assert(*iter == 2); iter++; assert(*iter == 3); iter++; assert(*iter == 4); iter++; assert(iter == jv.end()); } // Parent and child size is one. { CopyableChild children[1] = {CopyableChild(buffer1[0], 1)}; std::ranges::join_view jv(ParentView(children, 1)); auto iter = jv.begin(); assert(*iter == 1); iter++; assert(iter == jv.end()); } // Parent size is one child is empty { CopyableChild children[1] = {CopyableChild(buffer1[0], 0)}; std::ranges::join_view jv(ParentView(children, 1)); assert(jv.begin() == jv.end()); } // Has all empty children. { CopyableChild children[4] = {CopyableChild(buffer1[0], 0), CopyableChild(buffer1[1], 0), CopyableChild(buffer2[0], 0), CopyableChild(buffer2[1], 0)}; auto jv = std::ranges::join_view(ParentView(children)); assert(jv.begin() == jv.end()); } // First child is empty, others are not. { CopyableChild children[4] = {CopyableChild(buffer1[0], 4), CopyableChild(buffer1[1], 0), CopyableChild(buffer2[0], 0), CopyableChild(buffer2[1], 0)}; auto jv = std::ranges::join_view(ParentView(children)); auto iter = jv.begin(); assert(*iter == 1); iter++; assert(*iter == 2); iter++; assert(*iter == 3); iter++; assert(*iter == 4); iter++; assert(iter == jv.end()); } // Last child is empty, others are not. { CopyableChild children[4] = {CopyableChild(buffer1[0], 4), CopyableChild(buffer1[1], 4), CopyableChild(buffer2[0], 4), CopyableChild(buffer2[1], 0)}; auto jv = std::ranges::join_view(ParentView(children)); auto iter = jv.begin(); for (int i = 1; i < 13; ++i) { assert(*iter == i); iter++; } } // operator++(); { std::ranges::join_view jv(buffer1); auto iter = jv.begin(); for (int i = 2; i < 9; ++i) { assert(*++iter == i); } } { using IntView = ValueView; IntView children[4] = {IntView(buffer1[0]), IntView(buffer1[1]), IntView(buffer2[0]), IntView(buffer2[1])}; std::ranges::join_view jv(ValueView{children}); auto iter = jv.begin(); for (int i = 2; i < 17; ++i) { assert(*++iter == i); } ASSERT_SAME_TYPE(decltype(++iter), decltype(iter)&); } { // check return value std::ranges::join_view jv(buffer1); auto iter = jv.begin(); using iterator = decltype(iter); decltype(auto) iter2 = ++iter; static_assert(std::is_same_v); assert(&iter2 == &iter); std::same_as decltype(auto) iter3 = iter++; assert(std::next(iter3) == iter); } { // !ref-is-glvalue BidiCommonInner inners[2] = {buffer1[0], buffer1[1]}; InnerRValue> outer{inners}; std::ranges::join_view jv(outer); auto iter = jv.begin(); static_assert(std::is_void_v); } { // !forward_range BufferView inners[2] = {buffer1[0], buffer1[1]}; using Outer = SimpleInputCommonOuter>; std::ranges::join_view jv{Outer(inners)}; auto iter = jv.begin(); static_assert(std::is_void_v); } { // !forward_range> InputCommonInner inners[1] = {buffer1[0]}; std::ranges::join_view jv{inners}; auto iter = jv.begin(); static_assert(std::is_void_v); } { // Check stashing iterators (LWG3698: regex_iterator and join_view don't work together very well) std::ranges::join_view jv; auto it = jv.begin(); assert(*it == 'a'); ++it; assert(*it == 'a'); ++it; assert(*it == 'b'); it++; assert(*it == 'a'); it++; assert(*it == 'b'); ++it; assert(*it == 'c'); } return true; } int main(int, char**) { test(); static_assert(test()); return 0; }