blob: 2463bf733196d015145ed46074403c0b36061df6 [file] [log] [blame]
//===----------------------------------------------------------------------===//
//
// 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
// <algorithm>
// template<random_access_iterator I, sentinel_for<I> S, class Comp = ranges::less,
// class Proj = identity>
// requires sortable<I, Comp, Proj>
// constexpr I
// sort(I first, S last, Comp comp = {}, Proj proj = {}); // since C++20
//
// template<random_access_range R, class Comp = ranges::less, class Proj = identity>
// requires sortable<iterator_t<R>, Comp, Proj>
// constexpr borrowed_iterator_t<R>
// sort(R&& r, Comp comp = {}, Proj proj = {}); // since C++20
#include <algorithm>
#include <array>
#include <concepts>
#include <functional>
#include <ranges>
#include "almost_satisfies_types.h"
#include "boolean_testable.h"
#include "test_iterators.h"
// SFINAE tests.
using BadComparator = ComparatorNotCopyable<int*>;
static_assert(!std::sortable<int*, BadComparator>);
template <class Iter, class Sent = sentinel_wrapper<Iter>, class Comp = std::ranges::less>
concept HasSortIt = requires(Iter first, Sent last, Comp comp) { std::ranges::sort(first, last, comp); };
static_assert(HasSortIt<int*>);
static_assert(!HasSortIt<RandomAccessIteratorNotDerivedFrom>);
static_assert(!HasSortIt<RandomAccessIteratorBadIndex>);
static_assert(!HasSortIt<int*, SentinelForNotSemiregular>);
static_assert(!HasSortIt<int*, SentinelForNotWeaklyEqualityComparableWith>);
static_assert(!HasSortIt<int*, int*, BadComparator>);
static_assert(!HasSortIt<const int*>); // Doesn't satisfy `sortable`.
template <class Range, class Comp = std::ranges::less>
concept HasSortR = requires(Range range, Comp comp) { std::ranges::sort(range, comp); };
static_assert(HasSortR<UncheckedRange<int*>>);
static_assert(!HasSortR<RandomAccessRangeNotDerivedFrom>);
static_assert(!HasSortR<RandomAccessRangeBadIndex>);
static_assert(!HasSortR<UncheckedRange<int*, SentinelForNotSemiregular>>);
static_assert(!HasSortR<UncheckedRange<int*, SentinelForNotWeaklyEqualityComparableWith>>);
static_assert(!HasSortR<UncheckedRange<int*>, BadComparator>);
static_assert(!HasSortR<UncheckedRange<const int*>>); // Doesn't satisfy `sortable`.
template <class Iter, class Sent, std::size_t N>
constexpr void test_one(std::array<int, N> input, std::array<int, N> expected) {
{ // (iterator, sentinel) overload.
auto sorted = input;
auto b = Iter(sorted.data());
auto e = Sent(Iter(sorted.data() + sorted.size()));
std::same_as<Iter> decltype(auto) last = std::ranges::sort(b, e);
assert(sorted == expected);
assert(base(last) == sorted.data() + sorted.size());
}
{ // (range) overload.
auto sorted = input;
auto b = Iter(sorted.data());
auto e = Sent(Iter(sorted.data() + sorted.size()));
auto range = std::ranges::subrange(b, e);
std::same_as<Iter> decltype(auto) last = std::ranges::sort(range);
assert(sorted == expected);
assert(base(last) == sorted.data() + sorted.size());
}
}
template <class Iter, class Sent>
constexpr void test_iterators_2() {
// Empty sequence.
test_one<Iter, Sent, 0>({}, {});
// 1-element sequence.
test_one<Iter, Sent, 1>({1}, {1});
// 2-element sequence.
test_one<Iter, Sent, 2>({2, 1}, {1, 2});
// 3-element sequence.
test_one<Iter, Sent, 3>({2, 1, 3}, {1, 2, 3});
// Longer sequence.
test_one<Iter, Sent, 8>({2, 1, 3, 6, 8, 4, 11, 5}, {1, 2, 3, 4, 5, 6, 8, 11});
// Longer sequence with duplicates.
test_one<Iter, Sent, 7>({2, 1, 3, 6, 2, 8, 6}, {1, 2, 2, 3, 6, 6, 8});
// All elements are the same.
test_one<Iter, Sent, 3>({1, 1, 1}, {1, 1, 1});
// Already sorted.
test_one<Iter, Sent, 5>({1, 2, 3, 4, 5}, {1, 2, 3, 4, 5});
// Reverse-sorted.
test_one<Iter, Sent, 5>({5, 4, 3, 2, 1}, {1, 2, 3, 4, 5});
// Repeating pattern.
test_one<Iter, Sent, 6>({1, 2, 1, 2, 1, 2}, {1, 1, 1, 2, 2, 2});
}
template <class Iter>
constexpr void test_iterators_1() {
test_iterators_2<Iter, Iter>();
test_iterators_2<Iter, sentinel_wrapper<Iter>>();
}
constexpr void test_iterators() {
test_iterators_1<random_access_iterator<int*>>();
test_iterators_1<contiguous_iterator<int*>>();
test_iterators_1<int*>();
}
constexpr bool test() {
test_iterators();
{ // A custom comparator works.
{
std::array in = {1, 2, 3, 4, 5};
auto last = std::ranges::sort(in.begin(), in.end(), std::ranges::greater{});
assert((in == std::array{5, 4, 3, 2, 1}));
assert(last == in.end());
}
{
std::array in = {1, 2, 3, 4, 5};
auto last = std::ranges::sort(in, std::ranges::greater{});
assert((in == std::array{5, 4, 3, 2, 1}));
assert(last == in.end());
}
}
{ // A custom projection works.
struct A {
int a;
constexpr bool operator==(const A&) const = default;
};
{
std::array in = {A{2}, A{3}, A{1}};
auto last = std::ranges::sort(in.begin(), in.end(), {}, &A::a);
assert((in == std::array{A{1}, A{2}, A{3}}));
assert(last == in.end());
}
{
std::array in = {A{2}, A{3}, A{1}};
auto last = std::ranges::sort(in, {}, &A::a);
assert((in == std::array{A{1}, A{2}, A{3}}));
assert(last == in.end());
}
}
{ // `std::invoke` is used in the implementation.
struct S {
int i;
constexpr S(int i_) : i(i_) {}
constexpr bool comparator(const S& rhs) const { return i < rhs.i; }
constexpr const S& projection() const { return *this; }
constexpr bool operator==(const S&) const = default;
};
{
std::array in = {S{2}, S{3}, S{1}};
auto last = std::ranges::sort(in.begin(), in.end(), &S::comparator, &S::projection);
assert((in == std::array{S{1}, S{2}, S{3}}));
assert(last == in.end());
}
{
std::array in = {S{2}, S{3}, S{1}};
auto last = std::ranges::sort(in, &S::comparator, &S::projection);
assert((in == std::array{S{1}, S{2}, S{3}}));
assert(last == in.end());
}
}
{ // The comparator can return any type that's convertible to `bool`.
{
std::array in = {2, 1, 3};
auto last = std::ranges::sort(in.begin(), in.end(), [](int i, int j) { return BooleanTestable{i < j}; });
assert((in == std::array{1, 2, 3}));
assert(last == in.end());
}
{
std::array in = {2, 1, 3};
auto last = std::ranges::sort(in, [](int i, int j) { return BooleanTestable{i < j}; });
assert((in == std::array{1, 2, 3}));
assert(last == in.end());
}
}
{ // `std::ranges::dangling` is returned.
[[maybe_unused]] std::same_as<std::ranges::dangling> decltype(auto) result = std::ranges::sort(std::array{1, 2, 3});
}
// TODO: Enable the tests once the implementation switched to use iter_move/iter_swap
/*
{ // ProxyIterator
{
std::array in = {2, 1, 3};
ProxyRange proxy{in};
std::ranges::sort(proxy.begin(), proxy.end(), [](auto i, auto j) { return i.data < j.data; });
assert((in == std::array{1, 2, 3}));
}
{
std::array in = {2, 1, 3};
ProxyRange proxy{in};
std::ranges::sort(proxy, [](auto i, auto j) { return i.data < j.data; });
assert((in == std::array{1, 2, 3}));
}
}
*/
return true;
}
int main(int, char**) {
test();
static_assert(test());
return 0;
}