llvm/libcxx/test/std/ranges/range.adaptors/range.all/range.owning.view/begin_end.pass.cpp

//===----------------------------------------------------------------------===//
//
// 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_t<R> begin();
// constexpr sentinel_t<R> end();
// constexpr auto begin() const requires range<const R>;
// constexpr auto end() const requires range<const R>;

#include <ranges>

#include <array>
#include <cassert>
#include <concepts>
#include <memory>

#include "test_iterators.h"
#include "test_macros.h"

struct Base {
  constexpr int *begin() { return nullptr; }
  constexpr auto end() { return sentinel_wrapper<int*>(nullptr); }
  constexpr char *begin() const { return nullptr; }
  constexpr auto end() const { return sentinel_wrapper<char*>(nullptr); }
};
static_assert(std::same_as<std::ranges::iterator_t<Base>, int*>);
static_assert(std::same_as<std::ranges::sentinel_t<Base>, sentinel_wrapper<int*>>);
static_assert(std::same_as<std::ranges::iterator_t<const Base>, char*>);
static_assert(std::same_as<std::ranges::sentinel_t<const Base>, sentinel_wrapper<char*>>);

struct NoConst {
  int* begin();
  sentinel_wrapper<int*> end();
};

struct DecayChecker {
  int*& begin() const;
  int*& end() const;
};

template <class T>
concept HasBegin = requires (T t) {
  t.begin();
};

template <class T>
concept HasEnd = requires (T t) {
  t.end();
};

constexpr bool test()
{
  {
    using OwningView = std::ranges::owning_view<Base>;
    OwningView ov;
    std::same_as<int*> decltype(auto) b1 = static_cast<OwningView&>(ov).begin();
    std::same_as<int*> decltype(auto) b2 = static_cast<OwningView&&>(ov).begin();
    std::same_as<char*> decltype(auto) b3 = static_cast<const OwningView&>(ov).begin();
    std::same_as<char*> decltype(auto) b4 = static_cast<const OwningView&&>(ov).begin();

    std::same_as<sentinel_wrapper<int*>> decltype(auto) e1 = static_cast<OwningView&>(ov).end();
    std::same_as<sentinel_wrapper<int*>> decltype(auto) e2 = static_cast<OwningView&&>(ov).end();
    std::same_as<sentinel_wrapper<char*>> decltype(auto) e3 = static_cast<const OwningView&>(ov).end();
    std::same_as<sentinel_wrapper<char*>> decltype(auto) e4 = static_cast<const OwningView&&>(ov).end();

    assert(b1 == e1);
    assert(b2 == e2);
    assert(b3 == e3);
    assert(b4 == e4);
  }
  {
    // NoConst has non-const begin() and end(); so does the owning_view.
    using OwningView = std::ranges::owning_view<NoConst>;
    static_assert(HasBegin<OwningView&>);
    static_assert(HasBegin<OwningView&&>);
    static_assert(!HasBegin<const OwningView&>);
    static_assert(!HasBegin<const OwningView&&>);
    static_assert(HasEnd<OwningView&>);
    static_assert(HasEnd<OwningView&&>);
    static_assert(!HasEnd<const OwningView&>);
    static_assert(!HasEnd<const OwningView&&>);
  }
  {
    // DecayChecker's begin() and end() return references; make sure the owning_view decays them.
    using OwningView = std::ranges::owning_view<DecayChecker>;
    OwningView ov;
    ASSERT_SAME_TYPE(decltype(ov.begin()), int*);
    ASSERT_SAME_TYPE(decltype(ov.end()), int*);
  }
  {
    // Test an empty view.
    int a[] = {1};
    auto ov = std::ranges::owning_view(std::ranges::subrange(a, a));
    assert(ov.begin() == a);
    assert(std::as_const(ov).begin() == a);
    assert(ov.end() == a);
    assert(std::as_const(ov).end() == a);
  }
  {
    // Test a non-empty view.
    int a[] = {1};
    auto ov = std::ranges::owning_view(std::ranges::subrange(a, a+1));
    assert(ov.begin() == a);
    assert(std::as_const(ov).begin() == a);
    assert(ov.end() == a+1);
    assert(std::as_const(ov).end() == a+1);
  }
  {
    // Test a non-view.
    std::array<int, 2> a = {1, 2};
    auto ov = std::ranges::owning_view(std::move(a));
    assert(std::to_address(ov.begin()) != std::to_address(a.begin())); // because it points into the copy
    assert(std::to_address(std::as_const(ov).begin()) != std::to_address(a.begin()));
    assert(std::to_address(ov.end()) != std::to_address(a.end()));
    assert(std::to_address(std::as_const(ov).end()) != std::to_address(a.end()));
  }
  return true;
}

int main(int, char**) {
  test();
  static_assert(test());

  return 0;
}