summaryrefslogtreecommitdiff
path: root/libstdc++-v3/testsuite/std/ranges/adaptors/p2281.cc
blob: 2025ddb1459a7f272089a2656a402940a54bc8ca (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
// Copyright (C) 2021-2023 Free Software Foundation, Inc.
//
// This file is part of the GNU ISO C++ Library.  This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.

// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.

// You should have received a copy of the GNU General Public License along
// with this library; see the file COPYING3.  If not see
// <http://www.gnu.org/licenses/>.

// { dg-options "-std=gnu++2a" }
// { dg-do run { target c++2a } }

#include <algorithm>
#include <ranges>
#include <string>
#include <string_view>
#include <testsuite_hooks.h>

namespace ranges = std::ranges;
namespace views = std::ranges::views;

// Verify P2281 changes to the forwarding semantics of partial application
// and composition of range adaptor objects.

void
test01()
{
  auto split_into_strings = [] (auto p) {
    return views::lazy_split(p) | views::transform([](auto r){
      return std::string(r.begin(), ranges::next(r.begin(), r.end()));
    });
  };
  constexpr std::string_view s = "hello world";
  constexpr std::string_view p = " ";
  constexpr auto v1 = s | split_into_strings(p);
  constexpr auto v2 = split_into_strings(p)(s);
  VERIFY( ranges::equal(v1, (std::string_view[]){"hello", "world"}) );
  VERIFY( ranges::equal(v2, (std::string_view[]){"hello", "world"}) );
}

struct move_only_range
{
  move_only_range() { }
  move_only_range(move_only_range&&);
  move_only_range& operator=(move_only_range&&);
  move_only_range(const move_only_range&) = delete;
  move_only_range& operator=(const move_only_range&) = delete;
  char* begin();
  char* end();
};

template<>
  inline constexpr bool std::ranges::enable_view<move_only_range> = true;

template<auto lazy_split = views::lazy_split>
void
test02()
{
  std::string_view s;
  move_only_range p;
  static_assert(requires { s | lazy_split(std::move(p)); });
  static_assert(requires { lazy_split(std::move(p))(s); });
  static_assert(requires { lazy_split(std::move(p)) | views::all; });
  static_assert(requires { views::all | lazy_split(std::move(p)); });
  static_assert(!requires { lazy_split(p); });
  static_assert(!requires { lazy_split(p) | views::all; });
  static_assert(!requires { views::all | lazy_split(p); });
}

int
main()
{
  test01();
  test02();
}