| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- // Range v3 library
- //
- // Copyright Eric Niebler 2017-present
- //
- // Use, modification and distribution is subject to the
- // Boost Software License, Version 1.0. (See accompanying
- // file LICENSE_1_0.txt or copy at
- // http://www.boost.org/LICENSE_1_0.txt)
- //
- // Project home: https://github.com/ericniebler/range-v3
- #include <list>
- #include <vector>
- #include <range/v3/core.hpp>
- #include <range/v3/view/all.hpp>
- #include <range/v3/view/ref.hpp>
- #include <range/v3/view/subrange.hpp>
- #include "../simple_test.hpp"
- #include "../test_utils.hpp"
- #include "../test_iterators.hpp"
- CPP_template(class Rng)(
- requires ranges::range<Rng>)
- ranges::borrowed_subrange_t<Rng> algorithm(Rng &&rng);
- struct Base {};
- struct Derived : Base {};
- int main()
- {
- using namespace ranges;
- std::vector<int> vi{1,2,3,4};
- ////////////////////////////////////////////////////////////////////////////
- // borrowed_subrange_t tests:
- // lvalues are ReferenceableRanges and do not dangle:
- CPP_assert(same_as<subrange<int*>,
- decltype(::algorithm(std::declval<int(&)[42]>()))>);
- CPP_assert(same_as<subrange<std::vector<int>::iterator>,
- decltype(::algorithm(vi))>);
- // subrange and ref_view are ReferenceableRanges and do not dangle:
- CPP_assert(same_as<subrange<int*>,
- decltype(::algorithm(std::declval<subrange<int*>>()))>);
- CPP_assert(same_as<subrange<int*>,
- decltype(::algorithm(std::declval<ref_view<int[42]>>()))>);
- // non-ReferenceableRange rvalue ranges dangle:
- CPP_assert(same_as<dangling,
- decltype(::algorithm(std::declval<std::vector<int>>()))>);
- CPP_assert(same_as<dangling,
- decltype(::algorithm(std::move(vi)))>);
- // Test that slicing conversions are not allowed.
- CPP_assert(constructible_from<subrange<Base*, Base*>, Base*, Base*>);
- CPP_assert(!constructible_from<subrange<Base*, Base*>, Derived*, Derived*>);
- CPP_assert(constructible_from<subrange<const Base*, const Base*>, Base*, Base*>);
- CPP_assert(!constructible_from<subrange<const Base*, const Base*>, Derived*, Derived*>);
- CPP_assert(!constructible_from<subrange<Base*, Base*>, subrange<Derived*, Derived*>>);
- CPP_assert(constructible_from<subrange<Base*, unreachable_sentinel_t>, Base*, unreachable_sentinel_t>);
- CPP_assert(!constructible_from<subrange<Base*, unreachable_sentinel_t>, Derived*, unreachable_sentinel_t>);
- CPP_assert(constructible_from<subrange<const Base*, unreachable_sentinel_t>, Base*, unreachable_sentinel_t>);
- CPP_assert(!constructible_from<subrange<const Base*, unreachable_sentinel_t>, Derived*, unreachable_sentinel_t>);
- CPP_assert(!constructible_from<subrange<Base*, unreachable_sentinel_t>, subrange<Derived*, unreachable_sentinel_t>>);
- CPP_assert(constructible_from<subrange<Base*, Base*, subrange_kind::sized>, Base*, Base*, std::size_t>);
- CPP_assert(!constructible_from<subrange<Base*, Base*, subrange_kind::sized>, Derived*, Base*, std::size_t>);
- CPP_assert(constructible_from<subrange<const Base*, const Base*, subrange_kind::sized>, Base*, const Base*, std::size_t>);
- CPP_assert(!constructible_from<subrange<const Base*, const Base*, subrange_kind::sized>, Derived*, const Base*, std::size_t>);
- CPP_assert(!constructible_from<subrange<Base*, Base*, subrange_kind::sized>, subrange<Derived*, Base*>, std::size_t>);
- CPP_assert(convertible_to<subrange<Base*, Base*>, std::pair<const Base*, const Base*>>);
- CPP_assert(!convertible_to<subrange<Derived*, Derived*>, std::pair<Base*, Base*>>);
- subrange<std::vector<int>::iterator> r0 {vi.begin(), vi.end()};
- static_assert(std::tuple_size<decltype(r0)>::value == 2, "");
- CPP_assert(same_as<std::vector<int>::iterator,
- std::tuple_element<0, decltype(r0)>::type>);
- CPP_assert(same_as<std::vector<int>::iterator,
- std::tuple_element<1, decltype(r0)>::type>);
- CPP_assert(sized_range<decltype(r0)>);
- CHECK(r0.size() == 4u);
- CHECK(r0.begin() == vi.begin());
- CHECK(get<0>(r0) == vi.begin());
- CHECK(r0.end() == vi.end());
- CHECK(get<1>(r0) == vi.end());
- r0 = r0.next();
- CHECK(r0.size() == 3u);
- {
- subrange<std::vector<int>::iterator> rng {vi.begin(), vi.end(), ranges::size(vi)};
- CHECK(rng.size() == 4u);
- CHECK(rng.begin() == vi.begin());
- CHECK(rng.end() == vi.end());
- }
- std::pair<std::vector<int>::iterator, std::vector<int>::iterator> p0 = r0;
- CHECK(p0.first == vi.begin()+1);
- CHECK(p0.second == vi.end());
- subrange<std::vector<int>::iterator, unreachable_sentinel_t> r1 { r0.begin(), {} };
- static_assert(std::tuple_size<decltype(r1)>::value == 2, "");
- CPP_assert(same_as<std::vector<int>::iterator,
- std::tuple_element<0, decltype(r1)>::type>);
- CPP_assert(same_as<unreachable_sentinel_t,
- std::tuple_element<1, decltype(r1)>::type>);
- CPP_assert(view_<decltype(r1)>);
- CPP_assert(!sized_range<decltype(r1)>);
- CHECK(r1.begin() == vi.begin()+1);
- r1.end() = unreachable;
- r0 = r0.next();
- ++r0.begin();
- CHECK(r0.begin() == vi.begin()+2);
- CHECK(r0.size() == 2u);
- r0 = {r0.begin(), --r0.end()}; // --r0.end();
- CHECK(r0.end() == vi.end()-1);
- CHECK(r0.size() == 1u);
- CHECK(r0.front() == 3);
- CHECK(r0.back() == 3);
- std::pair<std::vector<int>::iterator, unreachable_sentinel_t> p1 = r1;
- CHECK(p1.first == vi.begin()+1);
- std::list<int> li{1,2,3,4};
- using LI = std::list<int>::iterator;
- subrange<LI, LI, subrange_kind::sized> l0 {li.begin(), li.end(), li.size()};
- CPP_assert(view_<decltype(l0)> && sized_range<decltype(l0)>);
- CHECK(l0.begin() == li.begin());
- CHECK(l0.end() == li.end());
- CHECK(l0.size() == li.size());
- l0 = l0.next();
- CHECK(l0.begin() == next(li.begin()));
- CHECK(l0.end() == li.end());
- CHECK(l0.size() == li.size() - 1u);
- l0 = views::all(li);
- subrange<std::list<int>::iterator> l1 = l0;
- CPP_assert(!sized_range<decltype(l1)>);
- CHECK(l1.begin() == li.begin());
- CHECK(l1.end() == li.end());
- #if RANGES_CXX_DEDUCTION_GUIDES >= RANGES_CXX_DEDUCTION_GUIDES_17
- #if defined(__clang__) && __clang_major__ < 6
- // Workaround https://bugs.llvm.org/show_bug.cgi?id=33314
- RANGES_DIAGNOSTIC_PUSH
- RANGES_DIAGNOSTIC_IGNORE_UNDEFINED_FUNC_TEMPLATE
- #endif
- {
- subrange s0{vi.begin(), vi.end()};
- subrange s1{li.begin(), li.end()};
- CPP_assert(same_as<decltype(r0), decltype(s0)>);
- CPP_assert(same_as<decltype(l1), decltype(s1)>);
- }
- {
- subrange s0{vi.begin(), vi.end(), ranges::size(vi)};
- subrange s1{li.begin(), li.end(), ranges::size(li)};
- CPP_assert(same_as<decltype(r0), decltype(s0)>);
- CPP_assert(same_as<decltype(l0), decltype(s1)>);
- }
- {
- subrange s0{vi};
- subrange s1{li};
- subrange s2{views::all(vi)};
- subrange s3{views::all(li)};
- CPP_assert(same_as<decltype(r0), decltype(s0)>);
- CPP_assert(same_as<decltype(l0), decltype(s1)>);
- CPP_assert(same_as<decltype(r0), decltype(s2)>);
- CPP_assert(same_as<decltype(l0), decltype(s3)>);
- }
- {
- subrange s0{r0};
- subrange s1{l0};
- subrange s2{l1};
- CPP_assert(same_as<decltype(r0), decltype(s0)>);
- CPP_assert(same_as<decltype(l0), decltype(s1)>);
- CPP_assert(same_as<decltype(l1), decltype(s2)>);
- }
- {
- subrange s0{vi, ranges::size(vi)};
- subrange s1{li, ranges::size(li)};
- subrange s2{views::all(vi), ranges::size(vi)};
- subrange s3{views::all(li), ranges::size(li)};
- CPP_assert(same_as<decltype(r0), decltype(s0)>);
- CPP_assert(same_as<decltype(l0), decltype(s1)>);
- CPP_assert(same_as<decltype(r0), decltype(s2)>);
- CPP_assert(same_as<decltype(l0), decltype(s3)>);
- }
- {
- subrange s0{r0, size(r0)};
- subrange s1{l0, size(l0)};
- subrange s2{l1, size(l0)};
- CPP_assert(same_as<decltype(r0), decltype(s0)>);
- CPP_assert(same_as<decltype(l0), decltype(s1)>);
- CPP_assert(same_as<decltype(l0), decltype(s2)>);
- }
- #if defined(__clang__) && __clang_major__ < 6
- RANGES_DIAGNOSTIC_POP
- #endif // clang bug workaround
- #endif // use deduction guides
- return ::test_result();
- }
|