&& invocable<_Fn, iter_reference_t<_Is>...>
using indirect_result_t = invoke_result_t<_Fn, iter_reference_t<_Is>...>;
+ namespace __detail
+ {
+ template<typename _Iter, typename _Proj>
+ struct __projected
+ {
+ struct __type
+ {
+ using value_type = remove_cvref_t<indirect_result_t<_Proj&, _Iter>>;
+ indirect_result_t<_Proj&, _Iter> operator*() const; // not defined
+ };
+ };
+
+ template<weakly_incrementable _Iter, typename _Proj>
+ struct __projected<_Iter, _Proj>
+ {
+ struct __type
+ {
+ using value_type = remove_cvref_t<indirect_result_t<_Proj&, _Iter>>;
+ using difference_type = iter_difference_t<_Iter>;
+ indirect_result_t<_Proj&, _Iter> operator*() const; // not defined
+ };
+ };
+ } // namespace __detail
+
/// [projected], projected
template<indirectly_readable _Iter,
indirectly_regular_unary_invocable<_Iter> _Proj>
- struct projected
- {
- using value_type = remove_cvref_t<indirect_result_t<_Proj&, _Iter>>;
-
- indirect_result_t<_Proj&, _Iter> operator*() const; // not defined
- };
-
- template<weakly_incrementable _Iter, typename _Proj>
- struct incrementable_traits<projected<_Iter, _Proj>>
- { using difference_type = iter_difference_t<_Iter>; };
+ using projected = __detail::__projected<_Iter, _Proj>::__type;
// [alg.req], common algorithm requirements
--- /dev/null
+// { dg-options "-std=gnu++20" }
+// { dg-do compile { target c++20 } }
+
+// P2538R1 ADL-proof std::projected
+// https://www.open-std.org/jtc1/sc22/wg21/docs/papers/2022/p2538r1.html
+
+#include <iterator>
+
+template<typename T>
+ concept has_diff_type = requires { typename T::difference_type; };
+
+static_assert( has_diff_type<std::projected<int*, void(*)(int)>> );
+
+struct Indy {
+ using value_type = int;
+ int operator*() const { return 0; }
+};
+static_assert( ! std::weakly_incrementable<Indy> );
+static_assert( ! has_diff_type<std::projected<Indy, void(*)(int)>> );
+
+
+// Examples from the paper:
+
+template<class T> struct Holder { T t; };
+struct Incomplete;
+
+void test_concepts()
+{
+ using T = Holder<Incomplete>*;
+ static_assert(std::equality_comparable<T>);
+ (void) std::indirectly_comparable<T*, T*, std::equal_to<>>;
+ (void) std::sortable<T*>;
+}
+
+#include <algorithm>
+
+void test_count()
+{
+ Holder<Incomplete>* a = nullptr;
+ (void) std::count(&a, &a, nullptr);
+ (void) std::ranges::count(&a, &a, nullptr); // { dg-bogus "." }
+}