[libc++] Implement ranges::fill{, _n}
Reviewed By: var-const, #libc Spies: libcxx-commits, mgorny Differential Revision: https://reviews.llvm.org/D123462
This commit is contained in:
@@ -42,8 +42,8 @@ Write,copy_n,Nikolas Klauser,`D122982 <https://llvm.org/D122982>`_,✅
|
||||
Write,copy_backward,Nikolas Klauser,`D122982 <https://llvm.org/D122982>`_,✅
|
||||
Write,move,Not assigned,n/a,Not started
|
||||
Write,move_backward,Not assigned,n/a,Not started
|
||||
Write,fill,Not assigned,n/a,Not started
|
||||
Write,fill_n,Not assigned,n/a,Not started
|
||||
Write,fill,Nikolas Klauser,`D123462 <https://reviews.llvm.org/D123462>`_,✅
|
||||
Write,fill_n,Nikolas Klauser,`D123462 <https://reviews.llvm.org/D123462>`_,✅
|
||||
Write,transform,Nikolas Klauser,`D122173 <https://llvm.org/D122173>`_,✅
|
||||
Write,generate,Not assigned,n/a,Not started
|
||||
Write,generate_n,Not assigned,n/a,Not started
|
||||
|
||||
|
@@ -72,6 +72,8 @@ set(files
|
||||
__algorithm/ranges_copy_n.h
|
||||
__algorithm/ranges_count.h
|
||||
__algorithm/ranges_count_if.h
|
||||
__algorithm/ranges_fill.h
|
||||
__algorithm/ranges_fill_n.h
|
||||
__algorithm/ranges_find.h
|
||||
__algorithm/ranges_find_if.h
|
||||
__algorithm/ranges_find_if_not.h
|
||||
|
||||
59
libcxx/include/__algorithm/ranges_fill.h
Normal file
59
libcxx/include/__algorithm/ranges_fill.h
Normal file
@@ -0,0 +1,59 @@
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// 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
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef _LIBCPP___ALGORITHM_RANGES_FILL_H
|
||||
#define _LIBCPP___ALGORITHM_RANGES_FILL_H
|
||||
|
||||
#include <__algorithm/ranges_fill_n.h>
|
||||
#include <__config>
|
||||
#include <__iterator/concepts.h>
|
||||
#include <__ranges/access.h>
|
||||
#include <__ranges/concepts.h>
|
||||
#include <__ranges/dangling.h>
|
||||
|
||||
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
||||
# pragma GCC system_header
|
||||
#endif
|
||||
|
||||
#if _LIBCPP_STD_VER > 17 && !defined(_LIBCPP_HAS_NO_INCOMPLETE_RANGES)
|
||||
|
||||
_LIBCPP_BEGIN_NAMESPACE_STD
|
||||
|
||||
namespace ranges {
|
||||
namespace __fill {
|
||||
struct __fn {
|
||||
template <class _Type, output_iterator<const _Type&> _Iter, sentinel_for<_Iter> _Sent>
|
||||
_LIBCPP_HIDE_FROM_ABI constexpr
|
||||
_Iter operator()(_Iter __first, _Sent __last, const _Type& __value) const {
|
||||
if constexpr(random_access_iterator<_Iter>) {
|
||||
return ranges::fill_n(__first, __last - __first, __value);
|
||||
} else {
|
||||
for (; __first != __last; ++__first)
|
||||
*__first = __value;
|
||||
return __first;
|
||||
}
|
||||
}
|
||||
|
||||
template <class _Type, output_range<const _Type&> _Range>
|
||||
_LIBCPP_HIDE_FROM_ABI constexpr
|
||||
borrowed_iterator_t<_Range> operator()(_Range&& __range, const _Type& __value) const {
|
||||
return (*this)(ranges::begin(__range), ranges::end(__range), __value);
|
||||
}
|
||||
};
|
||||
} // namespace __fill
|
||||
|
||||
inline namespace __cpo {
|
||||
inline constexpr auto fill = __fill::__fn{};
|
||||
} // namespace __cpo
|
||||
} // namespace ranges
|
||||
|
||||
_LIBCPP_END_NAMESPACE_STD
|
||||
|
||||
#endif // _LIBCPP_STD_VER > 17 && !defined(_LIBCPP_HAS_NO_INCOMPLETE_RANGES)
|
||||
|
||||
#endif // _LIBCPP___ALGORITHM_RANGES_FILL_H
|
||||
48
libcxx/include/__algorithm/ranges_fill_n.h
Normal file
48
libcxx/include/__algorithm/ranges_fill_n.h
Normal file
@@ -0,0 +1,48 @@
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// 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
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef _LIBCPP___ALGORITHM_RANGES_FILL_N_H
|
||||
#define _LIBCPP___ALGORITHM_RANGES_FILL_N_H
|
||||
|
||||
#include <__config>
|
||||
#include <__iterator/concepts.h>
|
||||
#include <__iterator/incrementable_traits.h>
|
||||
|
||||
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
||||
# pragma GCC system_header
|
||||
#endif
|
||||
|
||||
#if _LIBCPP_STD_VER > 17 && !defined(_LIBCPP_HAS_NO_INCOMPLETE_RANGES)
|
||||
|
||||
_LIBCPP_BEGIN_NAMESPACE_STD
|
||||
|
||||
namespace ranges {
|
||||
namespace __fill_n {
|
||||
struct __fn {
|
||||
template <class _Type, output_iterator<const _Type&> _Iter>
|
||||
_LIBCPP_HIDE_FROM_ABI constexpr
|
||||
_Iter operator()(_Iter __first, iter_difference_t<_Iter> __n, const _Type& __value) const {
|
||||
for (; __n != 0; --__n) {
|
||||
*__first = __value;
|
||||
++__first;
|
||||
}
|
||||
return __first;
|
||||
}
|
||||
};
|
||||
} // namespace __fill_n
|
||||
|
||||
inline namespace __cpo {
|
||||
inline constexpr auto fill_n = __fill_n::__fn{};
|
||||
} // namespace __cpo
|
||||
} // namespace ranges
|
||||
|
||||
_LIBCPP_END_NAMESPACE_STD
|
||||
|
||||
#endif // _LIBCPP_STD_VER > 17 && !defined(_LIBCPP_HAS_NO_INCOMPLETE_RANGES)
|
||||
|
||||
#endif // _LIBCPP___ALGORITHM_RANGES_FILL_N_H
|
||||
@@ -283,6 +283,14 @@ namespace ranges {
|
||||
requires permutable<iterator_t<R>>
|
||||
constexpr borrowed_iterator_t<R> ranges::reverse(R&& r); // since C++20
|
||||
|
||||
template<class T, output_iterator<const T&> O, sentinel_for<O> S>
|
||||
constexpr O ranges::fill(O first, S last, const T& value); // since C++20
|
||||
|
||||
template<class T, output_range<const T&> R>
|
||||
constexpr borrowed_iterator_t<R> ranges::fill(R&& r, const T& value); // since C++20
|
||||
|
||||
template<class T, output_iterator<const T&> O>
|
||||
constexpr O ranges::fill_n(O first, iter_difference_t<O> n, const T& value); // since C++20
|
||||
}
|
||||
|
||||
constexpr bool // constexpr in C++20
|
||||
@@ -1005,6 +1013,8 @@ template <class BidirectionalIterator, class Compare>
|
||||
#include <__algorithm/ranges_copy_n.h>
|
||||
#include <__algorithm/ranges_count.h>
|
||||
#include <__algorithm/ranges_count_if.h>
|
||||
#include <__algorithm/ranges_fill.h>
|
||||
#include <__algorithm/ranges_fill_n.h>
|
||||
#include <__algorithm/ranges_find.h>
|
||||
#include <__algorithm/ranges_find_if.h>
|
||||
#include <__algorithm/ranges_find_if_not.h>
|
||||
|
||||
@@ -304,6 +304,8 @@ module std [system] {
|
||||
module ranges_copy_n { private header "__algorithm/ranges_copy_n.h" }
|
||||
module ranges_count { private header "__algorithm/ranges_count.h" }
|
||||
module ranges_count_if { private header "__algorithm/ranges_count_if.h" }
|
||||
module ranges_fill { private header "__algorithm/ranges_fill.h" }
|
||||
module ranges_fill_n { private header "__algorithm/ranges_fill_n.h" }
|
||||
module ranges_find { private header "__algorithm/ranges_find.h" }
|
||||
module ranges_find_if { private header "__algorithm/ranges_find_if.h" }
|
||||
module ranges_find_if_not { private header "__algorithm/ranges_find_if_not.h" }
|
||||
|
||||
@@ -109,6 +109,8 @@ END-SCRIPT
|
||||
#include <__algorithm/ranges_copy_n.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_copy_n.h'}}
|
||||
#include <__algorithm/ranges_count.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_count.h'}}
|
||||
#include <__algorithm/ranges_count_if.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_count_if.h'}}
|
||||
#include <__algorithm/ranges_fill.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_fill.h'}}
|
||||
#include <__algorithm/ranges_fill_n.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_fill_n.h'}}
|
||||
#include <__algorithm/ranges_find.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_find.h'}}
|
||||
#include <__algorithm/ranges_find_if.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_find_if.h'}}
|
||||
#include <__algorithm/ranges_find_if_not.h> // expected-error@*:* {{use of private header from outside its module: '__algorithm/ranges_find_if_not.h'}}
|
||||
|
||||
@@ -0,0 +1,143 @@
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// 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
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
// <algorithm>
|
||||
|
||||
// UNSUPPORTED: c++03, c++11, c++14, c++17
|
||||
// UNSUPPORTED: libcpp-has-no-incomplete-ranges
|
||||
|
||||
// template<class T, output_iterator<const T&> O, sentinel_for<O> S>
|
||||
// constexpr O ranges::fill(O first, S last, const T& value);
|
||||
// template<class T, output_range<const T&> R>
|
||||
// constexpr borrowed_iterator_t<R> ranges::fill(R&& r, const T& value);
|
||||
|
||||
#include <algorithm>
|
||||
#include <array>
|
||||
#include <cassert>
|
||||
#include <ranges>
|
||||
#include <string>
|
||||
|
||||
#include "almost_satisfies_types.h"
|
||||
#include "test_iterators.h"
|
||||
|
||||
template <class Iter, class Sent = sentinel_wrapper<Iter>>
|
||||
concept HasFillIt = requires(Iter iter, Sent sent) { std::ranges::fill(iter, sent, int{}); };
|
||||
|
||||
static_assert(HasFillIt<int*>);
|
||||
static_assert(!HasFillIt<OutputIteratorNotIndirectlyWritable>);
|
||||
static_assert(!HasFillIt<OutputIteratorNotInputOrOutputIterator>);
|
||||
static_assert(!HasFillIt<int*, SentinelForNotSemiregular>);
|
||||
static_assert(!HasFillIt<int*, SentinelForNotWeaklyEqualityComparableWith>);
|
||||
|
||||
template <class Range>
|
||||
concept HasFillR = requires(Range range) { std::ranges::fill(range, int{}); };
|
||||
|
||||
static_assert(HasFillR<UncheckedRange<int*>>);
|
||||
static_assert(!HasFillR<OutputRangeNotIndirectlyWritable>);
|
||||
static_assert(!HasFillR<OutputRangeNotInputOrOutputIterator>);
|
||||
static_assert(!HasFillR<OutputRangeNotSentinelSemiregular>);
|
||||
static_assert(!HasFillR<OutputRangeNotSentinelEqualityComparableWith>);
|
||||
|
||||
template <class It, class Sent = It>
|
||||
constexpr void test_iterators() {
|
||||
{ // simple test
|
||||
{
|
||||
int a[3];
|
||||
std::same_as<It> auto ret = std::ranges::fill(It(a), Sent(It(a + 3)), 1);
|
||||
assert(std::all_of(a, a + 3, [](int i) { return i == 1; }));
|
||||
assert(base(ret) == a + 3);
|
||||
}
|
||||
{
|
||||
int a[3];
|
||||
auto range = std::ranges::subrange(It(a), Sent(It(a + 3)));
|
||||
std::same_as<It> auto ret = std::ranges::fill(range, 1);
|
||||
assert(std::all_of(a, a + 3, [](int i) { return i == 1; }));
|
||||
assert(base(ret) == a + 3);
|
||||
}
|
||||
}
|
||||
|
||||
{ // check that an empty range works
|
||||
{
|
||||
std::array<int, 0> a;
|
||||
auto ret = std::ranges::fill(It(a.data()), Sent(It(a.data())), 1);
|
||||
assert(base(ret) == a.data());
|
||||
}
|
||||
{
|
||||
std::array<int, 0> a;
|
||||
auto range = std::ranges::subrange(It(a.data()), Sent(It(a.data())));
|
||||
auto ret = std::ranges::fill(range, 1);
|
||||
assert(base(ret) == a.data());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
constexpr bool test() {
|
||||
test_iterators<cpp17_output_iterator<int*>, sentinel_wrapper<cpp17_output_iterator<int*>>>();
|
||||
test_iterators<cpp20_output_iterator<int*>, sentinel_wrapper<cpp20_output_iterator<int*>>>();
|
||||
test_iterators<forward_iterator<int*>>();
|
||||
test_iterators<bidirectional_iterator<int*>>();
|
||||
test_iterators<random_access_iterator<int*>>();
|
||||
test_iterators<contiguous_iterator<int*>>();
|
||||
test_iterators<int*>();
|
||||
|
||||
{ // check that every element is copied once
|
||||
struct S {
|
||||
bool copied = false;
|
||||
constexpr S& operator=(const S&) {
|
||||
copied = true;
|
||||
return *this;
|
||||
}
|
||||
};
|
||||
{
|
||||
S a[5];
|
||||
std::ranges::fill(a, a + 5, S {true});
|
||||
assert(std::all_of(a, a + 5, [](S& s) { return s.copied; }));
|
||||
}
|
||||
{
|
||||
S a[5];
|
||||
std::ranges::fill(a, S {true});
|
||||
assert(std::all_of(a, a + 5, [](S& s) { return s.copied; }));
|
||||
}
|
||||
}
|
||||
|
||||
{ // check that std::ranges::dangling is returned
|
||||
[[maybe_unused]] std::same_as<std::ranges::dangling> decltype(auto) ret =
|
||||
std::ranges::fill(std::array<int, 10> {}, 1);
|
||||
}
|
||||
|
||||
{ // check that std::ranges::dangling isn't returned with a borrowing range
|
||||
std::array<int, 10> a{};
|
||||
[[maybe_unused]] std::same_as<std::array<int, 10>::iterator> decltype(auto) ret =
|
||||
std::ranges::fill(std::views::all(a), 1);
|
||||
assert(std::all_of(a.begin(), a.end(), [](int i) { return i == 1; }));
|
||||
}
|
||||
|
||||
{ // check that non-trivially copyable items are copied properly
|
||||
{
|
||||
std::array<std::string, 10> a;
|
||||
auto ret = std::ranges::fill(a.begin(), a.end(), "long long string so no SSO");
|
||||
assert(ret == a.data() + a.size());
|
||||
assert(std::all_of(a.begin(), a.end(), [](auto& s) { return s == "long long string so no SSO"; }));
|
||||
}
|
||||
{
|
||||
std::array<std::string, 10> a;
|
||||
auto ret = std::ranges::fill(a, "long long string so no SSO");
|
||||
assert(ret == a.data() + a.size());
|
||||
assert(std::all_of(a.begin(), a.end(), [](auto& s) { return s == "long long string so no SSO"; }));
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
int main(int, char**) {
|
||||
test();
|
||||
static_assert(test());
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -0,0 +1,91 @@
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// 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
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
// <algorithm>
|
||||
|
||||
// UNSUPPORTED: c++03, c++11, c++14, c++17
|
||||
// UNSUPPORTED: libcpp-has-no-incomplete-ranges
|
||||
|
||||
// template<class T, output_iterator<const T&> O>
|
||||
// constexpr O ranges::fill_n(O first, iter_difference_t<O> n, const T& value);
|
||||
|
||||
#include <algorithm>
|
||||
#include <array>
|
||||
#include <cassert>
|
||||
#include <ranges>
|
||||
#include <string>
|
||||
|
||||
#include "almost_satisfies_types.h"
|
||||
#include "test_iterators.h"
|
||||
|
||||
template <class Iter>
|
||||
concept HasFillN = requires(Iter iter) { std::ranges::fill_n(iter, int{}, int{}); };
|
||||
|
||||
struct WrongType {};
|
||||
|
||||
static_assert(HasFillN<int*>);
|
||||
static_assert(!HasFillN<WrongType*>);
|
||||
static_assert(!HasFillN<OutputIteratorNotIndirectlyWritable>);
|
||||
static_assert(!HasFillN<OutputIteratorNotInputOrOutputIterator>);
|
||||
|
||||
template <class It, class Sent = It>
|
||||
constexpr void test_iterators() {
|
||||
{ // simple test
|
||||
int a[3];
|
||||
std::same_as<It> decltype(auto) ret = std::ranges::fill_n(It(a), 3, 1);
|
||||
assert(std::all_of(a, a + 3, [](int i) { return i == 1; }));
|
||||
assert(base(ret) == a + 3);
|
||||
}
|
||||
|
||||
{ // check that an empty range works
|
||||
std::array<int, 0> a;
|
||||
auto ret = std::ranges::fill_n(It(a.data()), 0, 1);
|
||||
assert(base(ret) == a.data());
|
||||
}
|
||||
}
|
||||
|
||||
constexpr bool test() {
|
||||
test_iterators<cpp17_output_iterator<int*>, sentinel_wrapper<cpp17_output_iterator<int*>>>();
|
||||
test_iterators<cpp20_output_iterator<int*>, sentinel_wrapper<cpp20_output_iterator<int*>>>();
|
||||
test_iterators<forward_iterator<int*>>();
|
||||
test_iterators<bidirectional_iterator<int*>>();
|
||||
test_iterators<random_access_iterator<int*>>();
|
||||
test_iterators<contiguous_iterator<int*>>();
|
||||
test_iterators<int*>();
|
||||
|
||||
{ // check that every element is copied once
|
||||
struct S {
|
||||
bool copied = false;
|
||||
constexpr S& operator=(const S&) {
|
||||
assert(!copied);
|
||||
copied = true;
|
||||
return *this;
|
||||
}
|
||||
};
|
||||
|
||||
S a[5];
|
||||
std::ranges::fill_n(a, 5, S {});
|
||||
assert(std::all_of(a, a + 5, [](S& s) { return s.copied; }));
|
||||
}
|
||||
|
||||
{ // check that non-trivially copyable items are copied properly
|
||||
std::array<std::string, 10> a;
|
||||
auto ret = std::ranges::fill_n(a.data(), 10, "long long string so no SSO");
|
||||
assert(ret == a.data() + a.size());
|
||||
assert(std::all_of(a.begin(), a.end(), [](auto& s) { return s == "long long string so no SSO"; }));
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
int main(int, char**) {
|
||||
test();
|
||||
static_assert(test());
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -107,6 +107,7 @@ public:
|
||||
};
|
||||
|
||||
using InputRangeNotSentinelSemiregular = UncheckedRange<cpp20_input_iterator<int*>, SentinelForNotSemiregular>;
|
||||
using OutputRangeNotSentinelSemiregular = UncheckedRange<cpp20_output_iterator<int*>, SentinelForNotSemiregular>;
|
||||
|
||||
static_assert(std::input_or_output_iterator<SentinelForNotSemiregular>);
|
||||
static_assert(!std::semiregular<SentinelForNotSemiregular>);
|
||||
@@ -123,6 +124,8 @@ public:
|
||||
|
||||
using InputRangeNotSentinelEqualityComparableWith =
|
||||
UncheckedRange<cpp20_input_iterator<int*>, SentinelForNotWeaklyEqualityComparableWith>;
|
||||
using OutputRangeNotSentinelEqualityComparableWith =
|
||||
UncheckedRange<cpp20_output_iterator<int*>, SentinelForNotWeaklyEqualityComparableWith>;
|
||||
|
||||
static_assert(std::input_or_output_iterator<SentinelForNotWeaklyEqualityComparableWith>);
|
||||
static_assert(std::semiregular<SentinelForNotWeaklyEqualityComparableWith>);
|
||||
@@ -220,4 +223,39 @@ static_assert(std::forward_iterator<PermutableNotSwappable>);
|
||||
static_assert(!std::permutable<PermutableNotSwappable>);
|
||||
static_assert(!std::indirectly_swappable<PermutableNotSwappable>);
|
||||
|
||||
class OutputIteratorNotInputOrOutputIterator {
|
||||
public:
|
||||
using difference_type = long;
|
||||
using value_type = int;
|
||||
using iterator_category = std::input_iterator_tag;
|
||||
|
||||
int& operator++();
|
||||
void operator++(int);
|
||||
int& operator*();
|
||||
};
|
||||
|
||||
using OutputRangeNotInputOrOutputIterator = UncheckedRange<InputIteratorNotInputOrOutputIterator>;
|
||||
|
||||
static_assert(!std::input_or_output_iterator<OutputIteratorNotInputOrOutputIterator>);
|
||||
static_assert(std::indirectly_writable<OutputIteratorNotInputOrOutputIterator, int>);
|
||||
static_assert(!std::output_iterator<OutputIteratorNotInputOrOutputIterator, int>);
|
||||
static_assert(!std::ranges::input_range<OutputRangeNotInputOrOutputIterator>);
|
||||
|
||||
class OutputIteratorNotIndirectlyWritable {
|
||||
public:
|
||||
using difference_type = long;
|
||||
using iterator_category = std::input_iterator_tag;
|
||||
|
||||
OutputIteratorNotIndirectlyWritable& operator++();
|
||||
void operator++(int);
|
||||
const int& operator*() const;
|
||||
};
|
||||
|
||||
using OutputRangeNotIndirectlyWritable = UncheckedRange<OutputIteratorNotIndirectlyWritable>;
|
||||
|
||||
static_assert(std::input_or_output_iterator<OutputIteratorNotIndirectlyWritable>);
|
||||
static_assert(!std::indirectly_writable<OutputIteratorNotIndirectlyWritable, int>);
|
||||
static_assert(!std::output_iterator<OutputIteratorNotIndirectlyWritable, int>);
|
||||
static_assert(!std::ranges::output_range<OutputIteratorNotIndirectlyWritable, int>);
|
||||
|
||||
#endif // ALMOST_SATISFIES_TYPES_H
|
||||
|
||||
Reference in New Issue
Block a user