https://gcc.gnu.org/g:ba0245930bc7c99ec9e059a26313bd9aa461d474

commit r13-9124-gba0245930bc7c99ec9e059a26313bd9aa461d474
Author: Jonathan Wakely <[email protected]>
Date:   Sun Oct 13 21:47:14 2024 +0100

    libstdc++: Implement LWG 3564 for ranges::transform_view
    
    The _Iterator<true> type returned by begin() const uses const F& to
    transform the elements, so it should use const F& to determine the
    iterator's value_type and iterator_category as well.
    
    This was accepted into the WP in July 2022.
    
    libstdc++-v3/ChangeLog:
    
            * include/std/ranges (transform_view:_Iterator): Use const F&
            to determine value_type and iterator_category of
            _Iterator<true>, as per LWG 3564.
            * testsuite/std/ranges/adaptors/transform.cc: Check value_type
            and iterator_category.
    
    Reviewed-by: Patrick Palka <[email protected]>
    
    (cherry picked from commit dde19c600c3c8a1d765c9b4961d2556e89edad14)

Diff:
---
 libstdc++-v3/include/std/ranges                       |  9 +++++++--
 .../testsuite/std/ranges/adaptors/transform.cc        | 19 +++++++++++++++++++
 2 files changed, 26 insertions(+), 2 deletions(-)

diff --git a/libstdc++-v3/include/std/ranges b/libstdc++-v3/include/std/ranges
index 421197584873..7bf4746973ff 100644
--- a/libstdc++-v3/include/std/ranges
+++ b/libstdc++-v3/include/std/ranges
@@ -1749,8 +1749,12 @@ namespace views::__adaptor
          static auto
          _S_iter_cat()
          {
+           // _GLIBCXX_RESOLVE_LIB_DEFECTS
+           // 3564. transform_view::iterator<true>::value_type and
+           // iterator_category should use const F&
            using _Base = transform_view::_Base<_Const>;
-           using _Res = invoke_result_t<_Fp&, range_reference_t<_Base>>;
+           using _Res = invoke_result_t<__maybe_const_t<_Const, _Fp>&,
+                                        range_reference_t<_Base>>;
            if constexpr (is_lvalue_reference_v<_Res>)
              {
                using _Cat
@@ -1799,7 +1803,8 @@ namespace views::__adaptor
          using iterator_concept = decltype(_S_iter_concept());
          // iterator_category defined in __transform_view_iter_cat
          using value_type
-           = remove_cvref_t<invoke_result_t<_Fp&, range_reference_t<_Base>>>;
+           = remove_cvref_t<invoke_result_t<__maybe_const_t<_Const, _Fp>&,
+                                            range_reference_t<_Base>>>;
          using difference_type = range_difference_t<_Base>;
 
          _Iterator() requires default_initializable<_Base_iter> = default;
diff --git a/libstdc++-v3/testsuite/std/ranges/adaptors/transform.cc 
b/libstdc++-v3/testsuite/std/ranges/adaptors/transform.cc
index e89ae4f9f3f2..365e90b03ecd 100644
--- a/libstdc++-v3/testsuite/std/ranges/adaptors/transform.cc
+++ b/libstdc++-v3/testsuite/std/ranges/adaptors/transform.cc
@@ -175,6 +175,24 @@ test08()
   static_assert(!requires { views::all | transform; });
 }
 
+void
+test10()
+{
+  struct F {
+    short operator()(int) { return 0; }
+    const int& operator()(const int& i) const { return i; }
+  };
+
+  int x[] {2, 4};
+  const auto xform = x | views::transform(F{});
+  using const_iterator = decltype(xform.begin());
+  // LWG 3564. transform_view::iterator<true>::value_type and iterator_category
+  // should use const F&
+  static_assert(std::same_as<std::iter_value_t<const_iterator>, int>);
+  using cat = std::iterator_traits<const_iterator>::iterator_category;
+  static_assert(std::same_as<cat, std::random_access_iterator_tag>);
+}
+
 int
 main()
 {
@@ -186,4 +204,5 @@ main()
   test06();
   test07();
   test08();
+  test10();
 }

Reply via email to