//===----------------------------------------------------------------------===// // // 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 // UNSUPPORTED: libcpp-has-no-incomplete-ranges // constexpr iterator& operator--(); // requires ref-is-glvalue && bidirectional_­range && // bidirectional_­range> && // common_­range>; // constexpr iterator operator--(int); // requires ref-is-glvalue && bidirectional_­range && // bidirectional_­range> && // common_­range>; #include #include #include #include "../types.h" template concept CanPreDecrement = requires(T& t) { --t; }; template concept CanPostDecrement = requires(T& t) { t--; }; constexpr void noDecrementTest(auto&& jv) { auto iter = jv.begin(); static_assert(!CanPreDecrement); static_assert(!CanPostDecrement); } constexpr bool test() { int buffer[4][4] = {{1, 2, 3, 4}, {5, 6, 7, 8}, {9, 10, 11, 12}, {13, 14, 15, 16}}; { // outer == ranges::end std::ranges::join_view jv(buffer); auto iter = std::next(jv.begin(), 16); for (int i = 16; i != 0; --i) { assert(*--iter == i); } } { // outer == ranges::end std::ranges::join_view jv(buffer); auto iter = std::next(jv.begin(), 13); for (int i = 13; i != 0; --i) { assert(*--iter == i); } } { // outer != ranges::end std::ranges::join_view jv(buffer); auto iter = std::next(jv.begin(), 12); for (int i = 12; i != 0; --i) { assert(*--iter == i); } } { // outer != ranges::end std::ranges::join_view jv(buffer); auto iter = std::next(jv.begin()); for (int i = 1; i != 0; --i) { assert(*--iter == i); } } { int small[2][1] = {{1}, {2}}; std::ranges::join_view jv(small); auto iter = std::next(jv.begin(), 2); for (int i = 2; i != 0; --i) { assert(*--iter == i); } } { #if defined(__GNUG__) && !defined(__clang__) // This seems to be a gcc bug where evaluating the following code // at compile time results in wrong array index if (!std::is_constant_evaluated()) { #endif // skip empty inner BidiCommonInner inners[4] = {buffer[0], {nullptr, 0}, {nullptr, 0}, buffer[1]}; std::ranges::join_view jv(inners); auto iter = jv.end(); for (int i = 8; i != 0; --i) { assert(*--iter == i); } #if defined(__GNUG__) && !defined(__clang__) } #endif } { // basic type checking std::ranges::join_view jv(buffer); auto iter1 = std::ranges::next(jv.begin(), 4); using iterator = decltype(iter1); decltype(auto) iter2 = --iter1; static_assert(std::same_as); assert(&iter1 == &iter2); std::same_as decltype(auto) iter3 = iter1--; assert(iter3 == std::next(iter1)); } { // !ref-is-glvalue BidiCommonInner inners[2] = {buffer[0], buffer[1]}; InnerRValue> outer{inners}; std::ranges::join_view jv(outer); noDecrementTest(jv); } { // !bidirectional_­range BidiCommonInner inners[2] = {buffer[0], buffer[1]}; SimpleForwardCommonOuter outer{inners}; std::ranges::join_view jv(outer); noDecrementTest(jv); } { // !bidirectional_­range> ForwardCommonInner inners[2] = {buffer[0], buffer[1]}; std::ranges::join_view jv(inners); noDecrementTest(jv); } { // LWG3313 `join_view::iterator::operator--` is incorrectly constrained // `join_view::iterator` should not have `operator--` if // !common_­range> BidiNonCommonInner inners[2] = {buffer[0], buffer[1]}; std::ranges::join_view jv(inners); auto iter = jv.begin(); static_assert(!CanPreDecrement); static_assert(!CanPostDecrement); } return true; } int main(int, char**) { test(); static_assert(test()); return 0; }