https://gcc.gnu.org/g:281ac0ef1001aac1b4f4223c5dbba725a5cffb43

commit r15-5719-g281ac0ef1001aac1b4f4223c5dbba725a5cffb43
Author: Jonathan Wakely <jwak...@redhat.com>
Date:   Mon Nov 25 21:57:57 2024 +0000

    libstdc++: Simplify std::list assignment using 'if constexpr'
    
    Use diagnostic pragmas to allow using `if constexpr` in C++11 mode, so
    that we don't need to use tag dispatching.
    
    The _M_move_assign overloads that were previously used for tag
    dispatching are no longer used, but are retained here (at least for the
    default config) so that an explicit instantiation will still define
    those members. This ensures that old code which expects an explicit
    instantiation in some other translation unit will still link. I'm not
    sure if that's really needed, we should probably have a policy about
    whether we support explicit instantiations where the declaration and
    definition use different versions of the headers.
    
    libstdc++-v3/ChangeLog:
    
            * include/bits/stl_list.h (operator=(list&&)): Use if constexpr
            instead of dispatching to _M_move_assign.
            (_M_move_assign): Do not define for versioned namespace.

Diff:
---
 libstdc++-v3/include/bits/stl_list.h | 29 +++++++++++++++++++++++++++--
 1 file changed, 27 insertions(+), 2 deletions(-)

diff --git a/libstdc++-v3/include/bits/stl_list.h 
b/libstdc++-v3/include/bits/stl_list.h
index 7deb04b4bfe4..3f92de42996e 100644
--- a/libstdc++-v3/include/bits/stl_list.h
+++ b/libstdc++-v3/include/bits/stl_list.h
@@ -935,6 +935,8 @@ _GLIBCXX_BEGIN_NAMESPACE_CXX11
       operator=(const list& __x);
 
 #if __cplusplus >= 201103L
+#pragma GCC diagnostic push
+#pragma GCC diagnostic ignored "-Wc++17-extensions" // if constexpr
       /**
        *  @brief  %List move assignment operator.
        *  @param  __x  A %list of identical element and allocator types.
@@ -952,9 +954,29 @@ _GLIBCXX_BEGIN_NAMESPACE_CXX11
        constexpr bool __move_storage =
          _Node_alloc_traits::_S_propagate_on_move_assign()
          || _Node_alloc_traits::_S_always_equal();
-       _M_move_assign(std::move(__x), __bool_constant<__move_storage>());
+       if constexpr (!__move_storage)
+         {
+           if (__x._M_get_Node_allocator() != this->_M_get_Node_allocator())
+             {
+               // The rvalue's allocator cannot be moved, or is not equal,
+               // so we need to individually move each element.
+               _M_assign_dispatch(std::make_move_iterator(__x.begin()),
+                                  std::make_move_iterator(__x.end()),
+                                  __false_type{});
+               return *this;
+             }
+         }
+
+       this->clear();
+       this->_M_move_nodes(std::move(__x));
+
+       if constexpr (_Node_alloc_traits::_S_propagate_on_move_assign())
+         this->_M_get_Node_allocator()
+             = std::move(__x._M_get_Node_allocator());
+
        return *this;
       }
+#pragma GCC diagnostic pop
 
       /**
        *  @brief  %List initializer list assignment operator.
@@ -2156,7 +2178,10 @@ _GLIBCXX_BEGIN_NAMESPACE_CXX11
       const_iterator
       _M_resize_pos(size_type& __new_size) const;
 
-#if __cplusplus >= 201103L
+#if __cplusplus >= 201103L && ! _GLIBCXX_INLINE_VERSION
+      // XXX GLIBCXX_ABI Deprecated
+      // These are unused and only kept so that explicit instantiations will
+      // continue to define the symbols.
       void
       _M_move_assign(list&& __x, true_type) noexcept
       {

Reply via email to