[libcxx-commits] [libcxx] [libc++][WIP] ADL-proof `std::__wrap_iter` (PR #107766)

Mital Ashok via libcxx-commits libcxx-commits at lists.llvm.org
Sun Sep 8 08:58:03 PDT 2024


https://github.com/MitalAshok created https://github.com/llvm/llvm-project/pull/107766

This is done in an ABI-breaking way.

It's still very messy, just a a proof-of-concept.

Would fix #107747

>From 77c15fc931abd2d4ce1b57faefb79e32a6f6a1fc Mon Sep 17 00:00:00 2001
From: Mital Ashok <mital at mitalashok.co.uk>
Date: Sun, 8 Sep 2024 16:51:58 +0100
Subject: [PATCH] wip

---
 libcxx/include/__configuration/abi.h          |   8 +
 libcxx/include/__format/buffer.h              |   2 +-
 libcxx/include/__iterator/iterator_traits.h   |  10 ++
 libcxx/include/__iterator/wrap_iter.h         | 152 ++++++++++++------
 libcxx/include/__memory/pointer_traits.h      |  38 ++++-
 libcxx/include/array                          |   8 +-
 libcxx/include/regex                          |  21 +++
 libcxx/include/span                           |   4 +-
 libcxx/include/string                         |  32 ++--
 libcxx/include/string_view                    |   2 +-
 libcxx/include/vector                         |  20 +--
 .../iterators/contiguous_iterators.pass.cpp   |  39 +++--
 .../libcxx/iterators/unwrap_iter.pass.cpp     |   8 +-
 .../iterators/wrap_iter_adl.compile.pass.cpp  |  58 +++++++
 ...range_concept_conformance.compile.pass.cpp |  48 ++++--
 ...rator_concept_conformance.compile.pass.cpp | 131 ++++++++-------
 libcxx/test/support/nasty_string.h            |   7 +
 17 files changed, 419 insertions(+), 169 deletions(-)
 create mode 100644 libcxx/test/libcxx/iterators/wrap_iter_adl.compile.pass.cpp

diff --git a/libcxx/include/__configuration/abi.h b/libcxx/include/__configuration/abi.h
index 8efbb42d1d8470..a368586ceb8adf 100644
--- a/libcxx/include/__configuration/abi.h
+++ b/libcxx/include/__configuration/abi.h
@@ -98,6 +98,14 @@
 // and WCHAR_MAX. This ABI setting determines whether we should instead track whether the fill
 // value has been initialized using a separate boolean, which changes the ABI.
 #  define _LIBCPP_ABI_IOS_ALLOW_ARBITRARY_FILL_VALUE
+// Change std::__wrap_iter<T> to not associate the namespaces associated with
+// T when unqualified lookup (e.g., with operators) is done. Also make it so
+// only std::__wrap_iter<pointer> and std::__wrap_iter<const_pointer> can
+// be used together (for comparison, conversion, etc.) instead of with any
+// wrapped iterator. So comprison between std::__wrap_iter<fancy_pointer<T>>
+// and std::__wrap_iter<T*> will no longer compile even if fancy_pointer<T>
+// is comparable with T*.
+#  define _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
 #elif _LIBCPP_ABI_VERSION == 1
 #  if !(defined(_LIBCPP_OBJECT_FORMAT_COFF) || defined(_LIBCPP_OBJECT_FORMAT_XCOFF))
 // Enable compiling copies of now inline methods into the dylib to support
diff --git a/libcxx/include/__format/buffer.h b/libcxx/include/__format/buffer.h
index ce9ac0c81e315a..048af2a00f8ae3 100644
--- a/libcxx/include/__format/buffer.h
+++ b/libcxx/include/__format/buffer.h
@@ -258,7 +258,7 @@ concept __enable_direct_output =
     (same_as<_OutIt, _CharT*>
      // TODO(hardening): the following check might not apply to hardened iterators and might need to be wrapped in an
      // `#ifdef`.
-     || same_as<_OutIt, __wrap_iter<_CharT*>>);
+     || same_as<_OutIt, __wrap_mut_iter<_CharT*, const _CharT*>>);
 
 /// Write policy for directly writing to the underlying output.
 template <class _OutIt, __fmt_char_type _CharT>
diff --git a/libcxx/include/__iterator/iterator_traits.h b/libcxx/include/__iterator/iterator_traits.h
index 4d9ad480cc4a29..a06f9660c8d739 100644
--- a/libcxx/include/__iterator/iterator_traits.h
+++ b/libcxx/include/__iterator/iterator_traits.h
@@ -453,6 +453,14 @@ template <class _Tp>
 struct __libcpp_is_contiguous_iterator
     : _Or< __has_iterator_category_convertible_to<_Tp, contiguous_iterator_tag>,
            __has_iterator_concept_convertible_to<_Tp, contiguous_iterator_tag> > {};
+#elif defined(_LIBCPP_ABI_WRAP_ITER_ADL_PROOF)
+template <class _It, class = void>
+struct __is_contiguous_wrap_iter : false_type {};
+template <class _It>
+struct __is_contiguous_wrap_iter<_It, __void_t<typename _Tp::__is_wrap_iter> > : true_type {};
+
+template <class _Tp>
+struct __libcpp_is_contiguous_iterator : __is_contiguous_wrap_iter<_Tp> {};
 #else
 template <class _Tp>
 struct __libcpp_is_contiguous_iterator : false_type {};
@@ -462,8 +470,10 @@ struct __libcpp_is_contiguous_iterator : false_type {};
 template <class _Up>
 struct __libcpp_is_contiguous_iterator<_Up*> : true_type {};
 
+#ifndef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
 template <class _Iter>
 class __wrap_iter;
+#endif
 
 template <class _Tp>
 using __has_exactly_input_iterator_category =
diff --git a/libcxx/include/__iterator/wrap_iter.h b/libcxx/include/__iterator/wrap_iter.h
index 549d8ff2dbd7db..5f6f1fe5cc59ca 100644
--- a/libcxx/include/__iterator/wrap_iter.h
+++ b/libcxx/include/__iterator/wrap_iter.h
@@ -16,9 +16,11 @@
 #include <__iterator/iterator_traits.h>
 #include <__memory/addressof.h>
 #include <__memory/pointer_traits.h>
+#include <__type_traits/conditional.h>
 #include <__type_traits/enable_if.h>
 #include <__type_traits/integral_constant.h>
 #include <__type_traits/is_convertible.h>
+#include <__type_traits/void_t.h>
 #include <cstddef>
 
 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
@@ -27,15 +29,25 @@
 
 _LIBCPP_BEGIN_NAMESPACE_STD
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+template <class _MutableIter, class _ConstIter>
+struct __wrap_iter_impl {
+  template <bool _IsConst>
+#else
 template <class _Iter>
-class __wrap_iter {
-public:
-  typedef _Iter iterator_type;
-  typedef typename iterator_traits<iterator_type>::value_type value_type;
-  typedef typename iterator_traits<iterator_type>::difference_type difference_type;
-  typedef typename iterator_traits<iterator_type>::pointer pointer;
-  typedef typename iterator_traits<iterator_type>::reference reference;
-  typedef typename iterator_traits<iterator_type>::iterator_category iterator_category;
+#endif
+  class __wrap_iter {
+
+  public:
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+    using _Iter = __conditional_t<_IsConst, _ConstIter, _MutableIter>;
+#endif
+    typedef _Iter iterator_type;
+    typedef typename iterator_traits<iterator_type>::value_type value_type;
+    typedef typename iterator_traits<iterator_type>::difference_type difference_type;
+    typedef typename iterator_traits<iterator_type>::pointer pointer;
+    typedef typename iterator_traits<iterator_type>::reference reference;
+    typedef typename iterator_traits<iterator_type>::iterator_category iterator_category;
 #if _LIBCPP_STD_VER >= 20
   typedef contiguous_iterator_tag iterator_concept;
 #endif
@@ -45,9 +57,15 @@ class __wrap_iter {
 
 public:
   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 __wrap_iter() _NOEXCEPT : __i_() {}
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+  template <bool _B, __enable_if_t<!_B, int> = 0>
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 __wrap_iter(const __wrap_iter<_B>& __u) _NOEXCEPT
+      : __i_(__u.base()) {}
+#else
   template <class _Up, __enable_if_t<is_convertible<_Up, iterator_type>::value, int> = 0>
   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 __wrap_iter(const __wrap_iter<_Up>& __u) _NOEXCEPT
       : __i_(__u.base()) {}
+#endif
   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 reference operator*() const _NOEXCEPT { return *__i_; }
   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 pointer operator->() const _NOEXCEPT {
     return std::__to_address(__i_);
@@ -96,7 +114,13 @@ class __wrap_iter {
 private:
   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 explicit __wrap_iter(iterator_type __x) _NOEXCEPT : __i_(__x) {}
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+  using __is_wrap_iter = void;
+
+  template <bool _OtherConst>
+#else
   template <class _Up>
+#endif
   friend class __wrap_iter;
   template <class _CharT, class _Traits, class _Alloc>
   friend class basic_string;
@@ -108,85 +132,99 @@ class __wrap_iter {
   friend class _LIBCPP_TEMPLATE_VIS span;
   template <class _Tp, size_t _Size>
   friend struct array;
-};
+  };
+
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+#  define _LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD                                                                       \
+    template <bool _Iter1>                                                                                             \
+    friend
+#  define _LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD                                                                         \
+    template <bool _Iter1, bool _Iter2>                                                                                \
+    friend
+#else
+#  define _LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD template <class _Iter1>
+#  define _LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD template <class _Iter1, class _Iter2>
+#endif
 
-template <class _Iter1>
-_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
-operator==(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
-  return __x.base() == __y.base();
-}
+  _LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
+  operator==(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
+    return __x.base() == __y.base();
+  }
 
-template <class _Iter1, class _Iter2>
-_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
-operator==(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
-  return __x.base() == __y.base();
-}
+  _LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
+  operator==(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
+    return __x.base() == __y.base();
+  }
 
-template <class _Iter1>
-_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 bool
-operator<(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
-  return __x.base() < __y.base();
-}
+  _LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 bool
+  operator<(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
+    return __x.base() < __y.base();
+  }
 
-template <class _Iter1, class _Iter2>
-_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 bool
-operator<(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
-  return __x.base() < __y.base();
-}
+  _LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 bool
+  operator<(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
+    return __x.base() < __y.base();
+  }
 
 #if _LIBCPP_STD_VER <= 17
-template <class _Iter1>
-_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
-operator!=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
-  return !(__x == __y);
-}
+  _LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
+  operator!=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
+    return !(__x == __y);
+  }
 
-template <class _Iter1, class _Iter2>
-_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
-operator!=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
-  return !(__x == __y);
+  _LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
+  _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
+  operator!=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
+    return !(__x == __y);
 }
-template <class _Iter1>
+_LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
 operator>(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
   return __y < __x;
 }
 
-template <class _Iter1, class _Iter2>
+_LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
 operator>(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
   return __y < __x;
 }
 
-template <class _Iter1>
+_LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
 operator>=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
   return !(__x < __y);
 }
 
-template <class _Iter1, class _Iter2>
+_LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
 operator>=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
   return !(__x < __y);
 }
 
-template <class _Iter1>
+_LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
 operator<=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter1>& __y) _NOEXCEPT {
   return !(__y < __x);
 }
 
-template <class _Iter1, class _Iter2>
+_LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR bool
 operator<=(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXCEPT {
   return !(__y < __x);
 }
 
 #else
-template <class _Iter1, class _Iter2>
+_LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI constexpr strong_ordering
 operator<=>(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) noexcept {
-  if constexpr (three_way_comparable_with<_Iter1, _Iter2, strong_ordering>) {
+  if constexpr (three_way_comparable_with<typename __wrap_iter<_Iter1>::iterator_type,
+                                          typename __wrap_iter<_Iter2>::iterator_type,
+                                          strong_ordering>) {
     return __x.base() <=> __y.base();
   } else {
     if (__x.base() < __y.base())
@@ -200,7 +238,7 @@ operator<=>(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) noex
 }
 #endif // _LIBCPP_STD_VER >= 20
 
-template <class _Iter1, class _Iter2>
+_LIBCPP_WRAP_ITER_PAIR_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14
 #ifndef _LIBCPP_CXX03_LANG
     auto
@@ -214,17 +252,25 @@ operator-(const __wrap_iter<_Iter1>& __x, const __wrap_iter<_Iter2>& __y) _NOEXC
   return __x.base() - __y.base();
 }
 
-template <class _Iter1>
+_LIBCPP_WRAP_ITER_SINGLE_TEMPLATE_HEAD
 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 __wrap_iter<_Iter1>
 operator+(typename __wrap_iter<_Iter1>::difference_type __n, __wrap_iter<_Iter1> __x) _NOEXCEPT {
   __x += __n;
   return __x;
 }
 
-#if _LIBCPP_STD_VER <= 17
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+};
+
+template <class _MutableIter, class _ConstIter>
+using __wrap_mut_iter = typename __wrap_iter_impl<_MutableIter, _ConstIter>::template __wrap_iter<false>;
+template <class _MutableIter, class _ConstIter>
+using __wrap_const_iter = typename __wrap_iter_impl<_MutableIter, _ConstIter>::template __wrap_iter<true>;
+#else
+#  if _LIBCPP_STD_VER <= 17
 template <class _It>
 struct __libcpp_is_contiguous_iterator<__wrap_iter<_It> > : true_type {};
-#endif
+#  endif
 
 template <class _It>
 struct _LIBCPP_TEMPLATE_VIS pointer_traits<__wrap_iter<_It> > {
@@ -237,6 +283,12 @@ struct _LIBCPP_TEMPLATE_VIS pointer_traits<__wrap_iter<_It> > {
   }
 };
 
+template <class _MutableIter, class _ConstIter>
+using __wrap_mut_iter = __wrap_iter<_MutableIter>;
+template <class _MutableIter, class _ConstIter>
+using __wrap_const_iter = __wrap_iter<_ConstIter>;
+#endif
+
 _LIBCPP_END_NAMESPACE_STD
 
 #endif // _LIBCPP___ITERATOR_WRAP_ITER_H
diff --git a/libcxx/include/__memory/pointer_traits.h b/libcxx/include/__memory/pointer_traits.h
index 8e08eb74413ee5..0e4196625af4e9 100644
--- a/libcxx/include/__memory/pointer_traits.h
+++ b/libcxx/include/__memory/pointer_traits.h
@@ -43,6 +43,7 @@ _LIBCPP_BEGIN_NAMESPACE_STD
 
 _LIBCPP_CLASS_TRAITS_HAS_XXX(__has_pointer, pointer);
 _LIBCPP_CLASS_TRAITS_HAS_XXX(__has_element_type, element_type);
+_LIBCPP_CLASS_TRAITS_HAS_XXX(__is_wrap_iter, __is_wrap_iter);
 
 template <class _Ptr, bool = __has_element_type<_Ptr>::value>
 struct __pointer_traits_element_type {};
@@ -115,11 +116,21 @@ struct __pointer_traits_rebind<_Sp<_Tp, _Args...>, _Up, false> {
   typedef _Sp<_Up, _Args...> type;
 };
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+template <class _Ptr, bool, class = void>
+struct __pointer_traits_impl {};
+
+template <class _Ptr>
+struct __pointer_traits_impl<_Ptr,
+                             false,
+#else
 template <class _Ptr, class = void>
 struct __pointer_traits_impl {};
 
 template <class _Ptr>
-struct __pointer_traits_impl<_Ptr, __void_t<typename __pointer_traits_element_type<_Ptr>::type> > {
+struct __pointer_traits_impl<_Ptr,
+#endif
+                             __void_t<typename __pointer_traits_element_type<_Ptr>::type> > {
   typedef _Ptr pointer;
   typedef typename __pointer_traits_element_type<pointer>::type element_type;
   typedef typename __pointer_traits_difference_type<pointer>::type difference_type;
@@ -144,8 +155,25 @@ struct __pointer_traits_impl<_Ptr, __void_t<typename __pointer_traits_element_ty
   }
 };
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+template <class _Ptr>
+struct _LIBCPP_TEMPLATE_VIS pointer_traits;
+
+template <class _WrapIt>
+struct __pointer_traits_impl<_WrapIt, true, void> {
+  using pointer         = _WrapIt;
+  using element_type    = typename pointer_traits<typename _WrapIt::iterator_type>::element_type;
+  using difference_type = typename pointer_traits<typename _WrapIt::iterator_type>::difference_type;
+
+  inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR static element_type* to_address(pointer __w) _NOEXCEPT;
+};
+
+template <class _Ptr>
+struct _LIBCPP_TEMPLATE_VIS pointer_traits : __pointer_traits_impl<_Ptr, __is_wrap_iter<_Ptr>::value> {};
+#else
 template <class _Ptr>
 struct _LIBCPP_TEMPLATE_VIS pointer_traits : __pointer_traits_impl<_Ptr> {};
+#endif
 
 template <class _Tp>
 struct _LIBCPP_TEMPLATE_VIS pointer_traits<_Tp*> {
@@ -237,6 +265,14 @@ struct __to_address_helper<_Pointer,
   }
 };
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+template <class _WrapIt>
+inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR typename __pointer_traits_impl<_WrapIt, true, void>::element_type*
+__pointer_traits_impl<_WrapIt, true, void>::to_address(pointer __w) _NOEXCEPT {
+  return std::__to_address(__w);
+}
+#endif
+
 #if _LIBCPP_STD_VER >= 20
 template <class _Tp>
 inline _LIBCPP_HIDE_FROM_ABI constexpr auto to_address(_Tp* __p) noexcept {
diff --git a/libcxx/include/array b/libcxx/include/array
index 588664ace0162a..36fb4963b875c8 100644
--- a/libcxx/include/array
+++ b/libcxx/include/array
@@ -177,8 +177,8 @@ struct _LIBCPP_TEMPLATE_VIS array {
   using pointer         = value_type*;
   using const_pointer   = const value_type*;
 #if defined(_LIBCPP_ABI_USE_WRAP_ITER_IN_STD_ARRAY)
-  using iterator       = __wrap_iter<pointer>;
-  using const_iterator = __wrap_iter<const_pointer>;
+  using iterator       = __wrap_mut_iter<pointer, const_pointer>;
+  using const_iterator = __wrap_const_iter<pointer, const_pointer>;
 #else
   using iterator       = pointer;
   using const_iterator = const_pointer;
@@ -277,8 +277,8 @@ struct _LIBCPP_TEMPLATE_VIS array<_Tp, 0> {
   using pointer         = value_type*;
   using const_pointer   = const value_type*;
 #if defined(_LIBCPP_ABI_USE_WRAP_ITER_IN_STD_ARRAY)
-  using iterator       = __wrap_iter<pointer>;
-  using const_iterator = __wrap_iter<const_pointer>;
+  using iterator       = __wrap_mut_iter<pointer, const_pointer>;
+  using const_iterator = __wrap_const_iter<pointer, const_pointer>;
 #else
   using iterator       = pointer;
   using const_iterator = const_pointer;
diff --git a/libcxx/include/regex b/libcxx/include/regex
index 08aebc2266f5de..0e3e90cc5c634a 100644
--- a/libcxx/include/regex
+++ b/libcxx/include/regex
@@ -2632,6 +2632,16 @@ private:
                            const basic_regex<_Cp, _Tp>& __e,
                            regex_constants::match_flag_type __flags);
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+  template <class _WrapIter, class _Ap, class _Cp, class _Tp, typename _WrapIter::__is_wrap_iter*>
+  friend bool
+  regex_search(_WrapIter __first,
+               _WrapIter __last,
+               match_results<_WrapIter, _Ap>& __m,
+               const basic_regex<_Cp, _Tp>& __e,
+               regex_constants::match_flag_type __flags);
+
+#else
   template <class _Iter, class _Ap, class _Cp, class _Tp>
   friend bool
   regex_search(__wrap_iter<_Iter> __first,
@@ -2639,6 +2649,7 @@ private:
                match_results<__wrap_iter<_Iter>, _Ap>& __m,
                const basic_regex<_Cp, _Tp>& __e,
                regex_constants::match_flag_type __flags);
+#endif
 
   template <class, class>
   friend class __lookahead;
@@ -5149,6 +5160,15 @@ regex_search(_BidirectionalIterator __first,
   return __r;
 }
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+template <class _WrapIter, class _Allocator, class _CharT, class _Traits, typename _WrapIter::__is_wrap_iter* = nullptr>
+inline _LIBCPP_HIDE_FROM_ABI bool
+regex_search(_WrapIter __first,
+             _WrapIter __last,
+             match_results<_WrapIter, _Allocator>& __m,
+             const basic_regex<_CharT, _Traits>& __e,
+             regex_constants::match_flag_type __flags = regex_constants::match_default) {
+#else
 template <class _Iter, class _Allocator, class _CharT, class _Traits>
 inline _LIBCPP_HIDE_FROM_ABI bool
 regex_search(__wrap_iter<_Iter> __first,
@@ -5156,6 +5176,7 @@ regex_search(__wrap_iter<_Iter> __first,
              match_results<__wrap_iter<_Iter>, _Allocator>& __m,
              const basic_regex<_CharT, _Traits>& __e,
              regex_constants::match_flag_type __flags = regex_constants::match_default) {
+#endif
   match_results<const _CharT*> __mc;
   bool __r = __e.__search(__first.base(), __last.base(), __mc, __flags);
   __m.__assign(__first, __last, __mc, __flags & regex_constants::__no_update_pos);
diff --git a/libcxx/include/span b/libcxx/include/span
index a32f7a372e2ae1..a691ba7e8623e4 100644
--- a/libcxx/include/span
+++ b/libcxx/include/span
@@ -239,7 +239,7 @@ public:
 #  ifdef _LIBCPP_ABI_BOUNDED_ITERATORS
   using iterator = __bounded_iter<pointer>;
 #  else
-  using iterator = __wrap_iter<pointer>;
+  using iterator = __wrap_mut_iter<pointer, pointer>;
 #  endif
   using reverse_iterator = std::reverse_iterator<iterator>;
 
@@ -420,7 +420,7 @@ public:
 #  ifdef _LIBCPP_ABI_BOUNDED_ITERATORS
   using iterator = __bounded_iter<pointer>;
 #  else
-  using iterator = __wrap_iter<pointer>;
+  using iterator = __wrap_mut_iter<pointer, pointer>;
 #  endif
   using reverse_iterator = std::reverse_iterator<iterator>;
 
diff --git a/libcxx/include/string b/libcxx/include/string
index 3480b57375c118..a4ccaaec4c9c21 100644
--- a/libcxx/include/string
+++ b/libcxx/include/string
@@ -732,6 +732,17 @@ operator+(type_identity_t<basic_string_view<_CharT, _Traits>> __lhs, basic_strin
 extern template _LIBCPP_EXPORTED_FROM_ABI string operator+
     <char, char_traits<char>, allocator<char> >(char const*, string const&);
 
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+template <class _Iter, class = void>
+struct __string_is_trivial_iterator : public false_type {};
+
+template <class _Tp>
+struct __string_is_trivial_iterator<_Tp*, void> : public is_arithmetic<_Tp> {};
+
+template <class _WrapIter>
+struct __string_is_trivial_iterator<_WrapIter, __void_t<typename _WrapIter::__is_wrap_iter> >
+    : public __string_is_trivial_iterator<typename _WrapIter::iterator_type> {};
+#else
 template <class _Iter>
 struct __string_is_trivial_iterator : public false_type {};
 
@@ -740,6 +751,7 @@ struct __string_is_trivial_iterator<_Tp*> : public is_arithmetic<_Tp> {};
 
 template <class _Iter>
 struct __string_is_trivial_iterator<__wrap_iter<_Iter> > : public __string_is_trivial_iterator<_Iter> {};
+#endif
 
 template <class _CharT, class _Traits, class _Tp>
 struct __can_be_converted_to_string_view
@@ -828,11 +840,11 @@ public:
   // Users might provide custom allocators, and prior to C++20 we have no existing way to detect whether the allocator's
   // pointer type is contiguous (though it has to be by the Standard). Using the wrapper type ensures the iterator is
   // considered contiguous.
-  typedef __bounded_iter<__wrap_iter<pointer> > iterator;
-  typedef __bounded_iter<__wrap_iter<const_pointer> > const_iterator;
+  typedef __bounded_iter<__wrap_mut_iter<pointer, const_pointer> > iterator;
+  typedef __bounded_iter<__wrap_const_iter<pointer, const_pointer> > const_iterator;
 #else
-  typedef __wrap_iter<pointer> iterator;
-  typedef __wrap_iter<const_pointer> const_iterator;
+  typedef __wrap_mut_iter<pointer, const_pointer> iterator;
+  typedef __wrap_const_iter<pointer, const_pointer> const_iterator;
 #endif
   typedef std::reverse_iterator<iterator> reverse_iterator;
   typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
@@ -958,9 +970,9 @@ private:
     // a stricter check, since correct code can never rely on being able to access newly-added elements via an existing
     // iterator.
     return std::__make_bounded_iter(
-        std::__wrap_iter<pointer>(__p),
-        std::__wrap_iter<pointer>(__get_pointer()),
-        std::__wrap_iter<pointer>(__get_pointer() + size()));
+        __wrap_mut_iter<pointer, const_pointer>(__p),
+        __wrap_mut_iter<pointer, const_pointer>(__get_pointer()),
+        __wrap_mut_iter<pointer, const_pointer>(__get_pointer() + size()));
 #else
     return iterator(__p);
 #endif // _LIBCPP_ABI_BOUNDED_ITERATORS_IN_STRING
@@ -970,9 +982,9 @@ private:
 #ifdef _LIBCPP_ABI_BOUNDED_ITERATORS_IN_STRING
     // Bound the iterator according to the size (and not the capacity, unlike vector).
     return std::__make_bounded_iter(
-        std::__wrap_iter<const_pointer>(__p),
-        std::__wrap_iter<const_pointer>(__get_pointer()),
-        std::__wrap_iter<const_pointer>(__get_pointer() + size()));
+        __wrap_const_iter<pointer, const_pointer>(__p),
+        __wrap_const_iter<pointer, const_pointer>(__get_pointer()),
+        __wrap_const_iter<pointer, const_pointer>(__get_pointer() + size()));
 #else
     return const_iterator(__p);
 #endif // _LIBCPP_ABI_BOUNDED_ITERATORS_IN_STRING
diff --git a/libcxx/include/string_view b/libcxx/include/string_view
index cf97e3a9be314d..451b50fb7a7b71 100644
--- a/libcxx/include/string_view
+++ b/libcxx/include/string_view
@@ -284,7 +284,7 @@ public:
 #if defined(_LIBCPP_ABI_BOUNDED_ITERATORS)
   using const_iterator = __bounded_iter<const_pointer>;
 #elif defined(_LIBCPP_ABI_USE_WRAP_ITER_IN_STD_STRING_VIEW)
-  using const_iterator = __wrap_iter<const_pointer>;
+  using const_iterator = __wrap_const_iter<const_pointer, const_pointer>;
 #else
   using const_iterator = const_pointer;
 #endif
diff --git a/libcxx/include/vector b/libcxx/include/vector
index 2442852c764a63..2bf296fef5d063 100644
--- a/libcxx/include/vector
+++ b/libcxx/include/vector
@@ -404,11 +404,11 @@ public:
   // Users might provide custom allocators, and prior to C++20 we have no existing way to detect whether the allocator's
   // pointer type is contiguous (though it has to be by the Standard). Using the wrapper type ensures the iterator is
   // considered contiguous.
-  typedef __bounded_iter<__wrap_iter<pointer> > iterator;
-  typedef __bounded_iter<__wrap_iter<const_pointer> > const_iterator;
+  typedef __bounded_iter<__wrap_mut_iter<pointer, const_pointer> > iterator;
+  typedef __bounded_iter<__wrap_const_iter<pointer, const_pointer> > const_iterator;
 #else
-  typedef __wrap_iter<pointer> iterator;
-  typedef __wrap_iter<const_pointer> const_iterator;
+  typedef __wrap_mut_iter<pointer, const_pointer> iterator;
+  typedef __wrap_const_iter<pointer, const_pointer> const_iterator;
 #endif
   typedef std::reverse_iterator<iterator> reverse_iterator;
   typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
@@ -847,9 +847,9 @@ private:
     // don't have a way to update existing valid iterators when the container is resized and thus have to go with
     // a laxer approach.
     return std::__make_bounded_iter(
-        std::__wrap_iter<pointer>(__p),
-        std::__wrap_iter<pointer>(this->__begin_),
-        std::__wrap_iter<pointer>(this->__end_cap()));
+        __wrap_mut_iter<pointer, const_pointer>(__p),
+        __wrap_mut_iter<pointer, const_pointer>(this->__begin_),
+        __wrap_mut_iter<pointer, const_pointer>(this->__end_cap()));
 #else
     return iterator(__p);
 #endif // _LIBCPP_ABI_BOUNDED_ITERATORS_IN_VECTOR
@@ -859,9 +859,9 @@ private:
 #ifdef _LIBCPP_ABI_BOUNDED_ITERATORS_IN_VECTOR
     // Bound the iterator according to the capacity, rather than the size.
     return std::__make_bounded_iter(
-        std::__wrap_iter<const_pointer>(__p),
-        std::__wrap_iter<const_pointer>(this->__begin_),
-        std::__wrap_iter<const_pointer>(this->__end_cap()));
+        __wrap_const_iter<pointer, const_pointer>(__p),
+        __wrap_const_iter<pointer, const_pointer>(this->__begin_),
+        __wrap_const_iter<pointer, const_pointer>(this->__end_cap()));
 #else
     return const_iterator(__p);
 #endif // _LIBCPP_ABI_BOUNDED_ITERATORS_IN_VECTOR
diff --git a/libcxx/test/libcxx/iterators/contiguous_iterators.pass.cpp b/libcxx/test/libcxx/iterators/contiguous_iterators.pass.cpp
index f00ca4e8794031..a5b1123b269f71 100644
--- a/libcxx/test/libcxx/iterators/contiguous_iterators.pass.cpp
+++ b/libcxx/test/libcxx/iterators/contiguous_iterators.pass.cpp
@@ -181,20 +181,35 @@ int main(int, char**)
     static_assert((!std::__libcpp_is_contiguous_iterator<std::reverse_iterator<my_contiguous_iterator> >::value), "");
 #endif
 
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<char *> >::value), "");
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<const char *> >::value), "");
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<int *> >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<std::__wrap_mut_iter<char*, const char*> >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<std::__wrap_const_iter<char*, const char*> >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<std::__wrap_mut_iter<int*, const int*> >::value), "");
 
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<T *> >::value), "");
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<std::__wrap_iter<T *> > >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<std::__wrap_mut_iter<T*, const T*> >::value), "");
+    static_assert(
+        (std::__libcpp_is_contiguous_iterator<
+            std::__wrap_mut_iter<std::__wrap_mut_iter<T*, const T*>, std::__wrap_mut_iter<T*, const T*> > >::value),
+        "");
 
     // Here my_random_access_iterator is standing in for some user's fancy pointer type, written pre-C++20.
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<my_random_access_iterator> >::value), "");
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<std::__wrap_iter<my_random_access_iterator> > >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<
+                      std::__wrap_mut_iter<my_random_access_iterator, my_random_access_iterator> >::value),
+                  "");
+    static_assert(
+        (std::__libcpp_is_contiguous_iterator<
+            std::__wrap_mut_iter<std::__wrap_mut_iter<my_random_access_iterator, my_random_access_iterator>,
+                                 std::__wrap_mut_iter<my_random_access_iterator, my_random_access_iterator> > >::value),
+        "");
 
 #if TEST_STD_VER >= 20
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<my_contiguous_iterator> >::value), "");
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<std::__wrap_iter<my_contiguous_iterator> > >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<
+                      std::__wrap_mut_iter<my_contiguous_iterator, my_contiguous_iterator> >::value),
+                  "");
+    static_assert(
+        (std::__libcpp_is_contiguous_iterator<
+            std::__wrap_mut_iter<std::__wrap_mut_iter<my_contiguous_iterator, my_contiguous_iterator>,
+                                 std::__wrap_mut_iter<my_contiguous_iterator, my_contiguous_iterator> > >::value),
+        "");
 #endif
 
 //  iterators in the libc++ test suite
@@ -218,9 +233,11 @@ int main(int, char**)
     static_assert(( std::__libcpp_is_contiguous_iterator<std::vector<int>::const_iterator>             ::value), "");
     static_assert((!std::__libcpp_is_contiguous_iterator<std::vector<int>::reverse_iterator>           ::value), "");
     static_assert((!std::__libcpp_is_contiguous_iterator<std::vector<int>::const_reverse_iterator>     ::value), "");
-    static_assert(( std::__libcpp_is_contiguous_iterator<std::__wrap_iter<std::vector<int>::iterator> >::value), "");
+    static_assert((std::__libcpp_is_contiguous_iterator<
+                      std::__wrap_mut_iter<std::vector<int>::iterator, std::vector<int>::const_iterator> >::value),
+                  "");
 
-//  string
+    //  string
     static_assert(( std::__libcpp_is_contiguous_iterator<std::string::iterator>              ::value), "");
     static_assert(( std::__libcpp_is_contiguous_iterator<std::string::const_iterator>        ::value), "");
     static_assert((!std::__libcpp_is_contiguous_iterator<std::string::reverse_iterator>      ::value), "");
diff --git a/libcxx/test/libcxx/iterators/unwrap_iter.pass.cpp b/libcxx/test/libcxx/iterators/unwrap_iter.pass.cpp
index 8ef2be2b010743..c2956cd6120ecc 100644
--- a/libcxx/test/libcxx/iterators/unwrap_iter.pass.cpp
+++ b/libcxx/test/libcxx/iterators/unwrap_iter.pass.cpp
@@ -26,11 +26,13 @@ template <class Iter>
 using rev_rev_iter = rev_iter<rev_iter<Iter> >;
 
 static_assert(std::is_same<UnwrapT<int*>, int*>::value, "");
-static_assert(std::is_same<UnwrapT<std::__wrap_iter<int*> >, int*>::value, "");
+static_assert(std::is_same<UnwrapT<std::__wrap_mut_iter<int*, const int*> >, int*>::value, "");
 static_assert(std::is_same<UnwrapT<rev_iter<int*> >, std::reverse_iterator<int*> >::value, "");
 static_assert(std::is_same<UnwrapT<rev_rev_iter<int*> >, int*>::value, "");
-static_assert(std::is_same<UnwrapT<rev_rev_iter<std::__wrap_iter<int*> > >, int*>::value, "");
-static_assert(std::is_same<UnwrapT<rev_rev_iter<rev_iter<std::__wrap_iter<int*> > > >, rev_iter<std::__wrap_iter<int*> > >::value, "");
+static_assert(std::is_same<UnwrapT<rev_rev_iter<std::__wrap_mut_iter<int*, const int*> > >, int*>::value, "");
+static_assert(std::is_same<UnwrapT<rev_rev_iter<rev_iter<std::__wrap_mut_iter<int*, const int*> > > >,
+                           rev_iter<std::__wrap_mut_iter<int*, const int*> > >::value,
+              "");
 
 static_assert(std::is_same<UnwrapT<random_access_iterator<int*> >, random_access_iterator<int*> >::value, "");
 static_assert(std::is_same<UnwrapT<rev_iter<random_access_iterator<int*> > >, rev_iter<random_access_iterator<int*> > >::value, "");
diff --git a/libcxx/test/libcxx/iterators/wrap_iter_adl.compile.pass.cpp b/libcxx/test/libcxx/iterators/wrap_iter_adl.compile.pass.cpp
new file mode 100644
index 00000000000000..d2cfbf16870ff6
--- /dev/null
+++ b/libcxx/test/libcxx/iterators/wrap_iter_adl.compile.pass.cpp
@@ -0,0 +1,58 @@
+//===----------------------------------------------------------------------===//
+//
+// 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
+//
+//===----------------------------------------------------------------------===//
+
+// Check that whats used for __wrap_iter<T> does not try to look into T
+// for ADL
+
+#include <__iterator/wrap_iter.h>
+#include <vector>
+#include <array>
+#include <iterator>
+
+#include "test_macros.h"
+
+#ifdef _LIBCPP_ABI_WRAP_ITER_ADL_PROOF
+
+struct incomplete;
+template <class T>
+struct holder {
+  T t;
+};
+
+struct nefarious {
+  friend void operator<=>(std::vector<nefarious>::iterator, std::vector<nefarious>::iterator)            = delete;
+  friend void operator==(std::vector<nefarious>::const_iterator, std::vector<nefarious>::const_iterator) = delete;
+};
+
+template <class Container>
+void test() {
+  using iterator       = typename Container::iterator;
+  using const_iterator = typename Container::const_iterator;
+
+  (void)(iterator() < iterator());
+  (void)(iterator() < const_iterator());
+  (void)(const_iterator() < const_iterator());
+  (void)(iterator() == iterator());
+  (void)(iterator() == const_iterator());
+  (void)(const_iterator() == const_iterator());
+
+#  if TEST_STD_VER >= 20
+  static_assert(std::contiguous_iterator<iterator>);
+  static_assert(std::contiguous_iterator<const_iterator>);
+  static_assert(std::ranges::contiguous_range<Container>);
+  static_assert(std::ranges::contiguous_range<const Container>);
+#  endif
+}
+
+void tests() {
+  test<std::vector<holder<incomplete>*>>();
+  test<std::array<holder<incomplete>*, 10>>();
+  test<std::vector<nefarious>>();
+  test<std::array<nefarious, 10>>();
+}
+#endif
diff --git a/libcxx/test/std/strings/basic.string/range_concept_conformance.compile.pass.cpp b/libcxx/test/std/strings/basic.string/range_concept_conformance.compile.pass.cpp
index b6afc20e3cccb0..4712f85078a306 100644
--- a/libcxx/test/std/strings/basic.string/range_concept_conformance.compile.pass.cpp
+++ b/libcxx/test/std/strings/basic.string/range_concept_conformance.compile.pass.cpp
@@ -15,20 +15,34 @@
 #include <concepts>
 #include <ranges>
 
-static_assert(std::same_as<std::ranges::iterator_t<std::string>, std::string::iterator>);
-static_assert(std::ranges::common_range<std::string>);
-static_assert(std::ranges::random_access_range<std::string>);
-static_assert(std::ranges::contiguous_range<std::string>);
-static_assert(!std::ranges::view<std::string>);
-static_assert(std::ranges::sized_range<std::string>);
-static_assert(!std::ranges::borrowed_range<std::string>);
-static_assert(std::ranges::viewable_range<std::string>);
-
-static_assert(std::same_as<std::ranges::iterator_t<std::string const>, std::string::const_iterator>);
-static_assert(std::ranges::common_range<std::string const>);
-static_assert(std::ranges::random_access_range<std::string const>);
-static_assert(std::ranges::contiguous_range<std::string const>);
-static_assert(!std::ranges::view<std::string const>);
-static_assert(std::ranges::sized_range<std::string const>);
-static_assert(!std::ranges::borrowed_range<std::string const>);
-static_assert(!std::ranges::viewable_range<std::string const>);
+#include "nasty_string.h"
+#include "test_macros.h"
+
+template <class String>
+void test() {
+  static_assert(std::same_as<std::ranges::iterator_t<String>, typename String::iterator>);
+  static_assert(std::ranges::common_range<String>);
+  static_assert(std::ranges::random_access_range<String>);
+  static_assert(std::ranges::contiguous_range<String>);
+  static_assert(!std::ranges::view<String>);
+  static_assert(std::ranges::sized_range<String>);
+  static_assert(!std::ranges::borrowed_range<String>);
+  static_assert(std::ranges::viewable_range<String>);
+
+  static_assert(std::same_as<std::ranges::iterator_t<String const>, typename String::const_iterator>);
+  static_assert(std::ranges::common_range<String const>);
+  static_assert(std::ranges::random_access_range<String const>);
+  static_assert(std::ranges::contiguous_range<String const>);
+  static_assert(!std::ranges::view<String const>);
+  static_assert(std::ranges::sized_range<String const>);
+  static_assert(!std::ranges::borrowed_range<String const>);
+  static_assert(!std::ranges::viewable_range<String const>);
+}
+
+void tests() {
+  test<std::string>();
+  test<std::wstring>();
+#ifndef TEST_HAS_NO_NASTY_STRING
+  test<nasty_string>();
+#endif
+}
diff --git a/libcxx/test/std/strings/basic.string/string.iterators/iterator_concept_conformance.compile.pass.cpp b/libcxx/test/std/strings/basic.string/string.iterators/iterator_concept_conformance.compile.pass.cpp
index bde2f9e9cde327..77212e24d7a644 100644
--- a/libcxx/test/std/strings/basic.string/string.iterators/iterator_concept_conformance.compile.pass.cpp
+++ b/libcxx/test/std/strings/basic.string/string.iterators/iterator_concept_conformance.compile.pass.cpp
@@ -14,64 +14,77 @@
 
 #include <iterator>
 
-using iterator               = std::string::iterator;
-using const_iterator         = std::string::const_iterator;
-using reverse_iterator       = std::string::reverse_iterator;
-using const_reverse_iterator = std::string::const_reverse_iterator;
-using value_type             = char;
+#include "nasty_string.h"
 
-static_assert(std::contiguous_iterator<iterator>);
-static_assert(std::indirectly_writable<iterator, value_type>);
-static_assert(std::sentinel_for<iterator, iterator>);
-static_assert(std::sentinel_for<iterator, const_iterator>);
-static_assert(!std::sentinel_for<iterator, reverse_iterator>);
-static_assert(!std::sentinel_for<iterator, const_reverse_iterator>);
-static_assert(std::sized_sentinel_for<iterator, iterator>);
-static_assert(std::sized_sentinel_for<iterator, const_iterator>);
-static_assert(!std::sized_sentinel_for<iterator, reverse_iterator>);
-static_assert(!std::sized_sentinel_for<iterator, const_reverse_iterator>);
-static_assert(std::indirectly_movable<iterator, iterator>);
-static_assert(std::indirectly_movable_storable<iterator, iterator>);
-static_assert(!std::indirectly_movable<iterator, const_iterator>);
-static_assert(!std::indirectly_movable_storable<iterator, const_iterator>);
-static_assert(std::indirectly_movable<iterator, reverse_iterator>);
-static_assert(std::indirectly_movable_storable<iterator, reverse_iterator>);
-static_assert(!std::indirectly_movable<iterator, const_reverse_iterator>);
-static_assert(!std::indirectly_movable_storable<iterator, const_reverse_iterator>);
-static_assert(std::indirectly_copyable<iterator, iterator>);
-static_assert(std::indirectly_copyable_storable<iterator, iterator>);
-static_assert(!std::indirectly_copyable<iterator, const_iterator>);
-static_assert(!std::indirectly_copyable_storable<iterator, const_iterator>);
-static_assert(std::indirectly_copyable<iterator, reverse_iterator>);
-static_assert(std::indirectly_copyable_storable<iterator, reverse_iterator>);
-static_assert(!std::indirectly_copyable<iterator, const_reverse_iterator>);
-static_assert(!std::indirectly_copyable_storable<iterator, const_reverse_iterator>);
-static_assert(std::indirectly_swappable<iterator, iterator>);
+template <class String>
+void test() {
+  using iterator               = typename String::iterator;
+  using const_iterator         = typename String::const_iterator;
+  using reverse_iterator       = typename String::reverse_iterator;
+  using const_reverse_iterator = typename String::const_reverse_iterator;
+  using value_type             = typename String::value_type;
 
-static_assert(std::contiguous_iterator<const_iterator>);
-static_assert(!std::indirectly_writable<const_iterator, value_type>);
-static_assert(std::sentinel_for<const_iterator, iterator>);
-static_assert(std::sentinel_for<const_iterator, const_iterator>);
-static_assert(!std::sentinel_for<const_iterator, reverse_iterator>);
-static_assert(!std::sentinel_for<const_iterator, const_reverse_iterator>);
-static_assert(std::sized_sentinel_for<const_iterator, iterator>);
-static_assert(std::sized_sentinel_for<const_iterator, const_iterator>);
-static_assert(!std::sized_sentinel_for<const_iterator, reverse_iterator>);
-static_assert(!std::sized_sentinel_for<const_iterator, const_reverse_iterator>);
-static_assert(std::indirectly_movable<const_iterator, iterator>);
-static_assert(std::indirectly_movable_storable<const_iterator, iterator>);
-static_assert(!std::indirectly_movable<const_iterator, const_iterator>);
-static_assert(!std::indirectly_movable_storable<const_iterator, const_iterator>);
-static_assert(std::indirectly_movable<const_iterator, reverse_iterator>);
-static_assert(std::indirectly_movable_storable<const_iterator, reverse_iterator>);
-static_assert(!std::indirectly_movable<const_iterator, const_reverse_iterator>);
-static_assert(!std::indirectly_movable_storable<const_iterator, const_reverse_iterator>);
-static_assert(std::indirectly_copyable<const_iterator, iterator>);
-static_assert(std::indirectly_copyable_storable<const_iterator, iterator>);
-static_assert(!std::indirectly_copyable<const_iterator, const_iterator>);
-static_assert(!std::indirectly_copyable_storable<const_iterator, const_iterator>);
-static_assert(std::indirectly_copyable<const_iterator, reverse_iterator>);
-static_assert(std::indirectly_copyable_storable<const_iterator, reverse_iterator>);
-static_assert(!std::indirectly_copyable<const_iterator, const_reverse_iterator>);
-static_assert(!std::indirectly_copyable_storable<const_iterator, const_reverse_iterator>);
-static_assert(!std::indirectly_swappable<const_iterator, const_iterator>);
+  static_assert(std::contiguous_iterator<iterator>);
+  static_assert(std::indirectly_writable<iterator, value_type>);
+  static_assert(std::sentinel_for<iterator, iterator>);
+  static_assert(std::sentinel_for<iterator, const_iterator>);
+  static_assert(!std::sentinel_for<iterator, reverse_iterator>);
+  static_assert(!std::sentinel_for<iterator, const_reverse_iterator>);
+  static_assert(std::sized_sentinel_for<iterator, iterator>);
+  static_assert(std::sized_sentinel_for<iterator, const_iterator>);
+  static_assert(!std::sized_sentinel_for<iterator, reverse_iterator>);
+  static_assert(!std::sized_sentinel_for<iterator, const_reverse_iterator>);
+  static_assert(std::indirectly_movable<iterator, iterator>);
+  static_assert(std::indirectly_movable_storable<iterator, iterator>);
+  static_assert(!std::indirectly_movable<iterator, const_iterator>);
+  static_assert(!std::indirectly_movable_storable<iterator, const_iterator>);
+  static_assert(std::indirectly_movable<iterator, reverse_iterator>);
+  static_assert(std::indirectly_movable_storable<iterator, reverse_iterator>);
+  static_assert(!std::indirectly_movable<iterator, const_reverse_iterator>);
+  static_assert(!std::indirectly_movable_storable<iterator, const_reverse_iterator>);
+  static_assert(std::indirectly_copyable<iterator, iterator>);
+  static_assert(std::indirectly_copyable_storable<iterator, iterator>);
+  static_assert(!std::indirectly_copyable<iterator, const_iterator>);
+  static_assert(!std::indirectly_copyable_storable<iterator, const_iterator>);
+  static_assert(std::indirectly_copyable<iterator, reverse_iterator>);
+  static_assert(std::indirectly_copyable_storable<iterator, reverse_iterator>);
+  static_assert(!std::indirectly_copyable<iterator, const_reverse_iterator>);
+  static_assert(!std::indirectly_copyable_storable<iterator, const_reverse_iterator>);
+  static_assert(std::indirectly_swappable<iterator, iterator>);
+
+  static_assert(std::contiguous_iterator<const_iterator>);
+  static_assert(!std::indirectly_writable<const_iterator, value_type>);
+  static_assert(std::sentinel_for<const_iterator, iterator>);
+  static_assert(std::sentinel_for<const_iterator, const_iterator>);
+  static_assert(!std::sentinel_for<const_iterator, reverse_iterator>);
+  static_assert(!std::sentinel_for<const_iterator, const_reverse_iterator>);
+  static_assert(std::sized_sentinel_for<const_iterator, iterator>);
+  static_assert(std::sized_sentinel_for<const_iterator, const_iterator>);
+  static_assert(!std::sized_sentinel_for<const_iterator, reverse_iterator>);
+  static_assert(!std::sized_sentinel_for<const_iterator, const_reverse_iterator>);
+  static_assert(std::indirectly_movable<const_iterator, iterator>);
+  static_assert(std::indirectly_movable_storable<const_iterator, iterator>);
+  static_assert(!std::indirectly_movable<const_iterator, const_iterator>);
+  static_assert(!std::indirectly_movable_storable<const_iterator, const_iterator>);
+  static_assert(std::indirectly_movable<const_iterator, reverse_iterator>);
+  static_assert(std::indirectly_movable_storable<const_iterator, reverse_iterator>);
+  static_assert(!std::indirectly_movable<const_iterator, const_reverse_iterator>);
+  static_assert(!std::indirectly_movable_storable<const_iterator, const_reverse_iterator>);
+  static_assert(std::indirectly_copyable<const_iterator, iterator>);
+  static_assert(std::indirectly_copyable_storable<const_iterator, iterator>);
+  static_assert(!std::indirectly_copyable<const_iterator, const_iterator>);
+  static_assert(!std::indirectly_copyable_storable<const_iterator, const_iterator>);
+  static_assert(std::indirectly_copyable<const_iterator, reverse_iterator>);
+  static_assert(std::indirectly_copyable_storable<const_iterator, reverse_iterator>);
+  static_assert(!std::indirectly_copyable<const_iterator, const_reverse_iterator>);
+  static_assert(!std::indirectly_copyable_storable<const_iterator, const_reverse_iterator>);
+  static_assert(!std::indirectly_swappable<const_iterator, const_iterator>);
+}
+
+void tests() {
+  test<std::string>();
+  test<std::wstring>();
+#if !defined(TEST_HAS_NO_NASTY_STRING)
+  test<nasty_string>();
+#endif
+}
diff --git a/libcxx/test/support/nasty_string.h b/libcxx/test/support/nasty_string.h
index ea9d83ccf282aa..e526ccd3365b6f 100644
--- a/libcxx/test/support/nasty_string.h
+++ b/libcxx/test/support/nasty_string.h
@@ -40,8 +40,15 @@
 #ifndef TEST_HAS_NO_NASTY_STRING
 // Make sure the char-like operations in strings do not depend on the char-like type.
 struct nasty_char {
+#  if defined(_LIBCPP_VERSION) && !defined(_LIBCPP_ABI_WRAP_ITER_ADL_PROOF)
+  // T, T would win overload resolution when comparing two wrap_iters, so
+  // make sure this is less specific
+  template <typename T, typename U>
+  friend auto operator<=>(T, U) = delete;
+#  else
   template <typename T>
   friend auto operator<=>(T, T) = delete;
+#  endif
 
   template <typename T>
   friend void operator+(T&&) = delete;



More information about the libcxx-commits mailing list