On 28/08/17 21:12 +0200, François Dumont wrote:
Hi
Here is the always equal allocator optimization for associative
containers.
Tested under Linux x86_64.
* include/bits/stl_tree.h
(_Rb_tree_impl(_Rb_tree_impl&&, _Node_allocator&&)): New.
(_Rb_tree(_Rb_tree&&, _Node_allocator&&, std::true_type)): New.
(_Rb_tree(_Rb_tree&&, _Node_allocator&&, std::false_type)): Likewise.
(_Rb_tree(_Rb_tree&&, _Node_allocator&&)): Adapt, use latters.
Ok to apply ?
François
diff --git a/libstdc++-v3/include/bits/stl_tree.h
b/libstdc++-v3/include/bits/stl_tree.h
index c2417f1..f7d34e3 100644
--- a/libstdc++-v3/include/bits/stl_tree.h
+++ b/libstdc++-v3/include/bits/stl_tree.h
@@ -704,6 +704,12 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
#else
_Rb_tree_impl(_Rb_tree_impl&&) = default;
+ _Rb_tree_impl(_Rb_tree_impl&& __x, _Node_allocator&& __a) noexcept
This is an unconditional noexcept, but ...
+ : _Node_allocator(std::move(__a)),
+ _Base_key_compare(std::move(__x)),
This constructor has a conditional noexcept. That's a bug.
+ _Rb_tree_header(std::move(__x))
+ { }
However, I don't think we need this new constructor anyway, see below.
_Rb_tree_impl(const _Key_compare& __comp, _Node_allocator&& __a)
: _Node_allocator(std::move(__a)), _Base_key_compare(__comp)
{ }
@@ -947,7 +953,18 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
: _Rb_tree(std::move(__x), _Node_allocator(__a))
{ }
- _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a);
+ private:
+ _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a, std::true_type) noexcept
This noexcept should be conditional.
+ : _M_impl(std::move(__x._M_impl), std::move(__a))
+ { }
Since we know __a == __x.get_allocator() we could just do:
_Rb_tree(_Rb_tree&& __x, _Node_allocator&&, true_type)
noexcept(is_nothrow_move_constructible<_Rb_tree_impl<_Compare>>::value)
: _M_impl(std::move(__x._M_impl))
{ }
This means we don't need the new constructor.
Or equivalently, just delegate to the move constructor:
_Rb_tree(_Rb_tree&& __x, _Node_allocator&&, true_type)
noexcept(is_nothrow_move_constructible<_Rb_tree>::value)
: _Rb_tree(std::move(__x))
{ }
+ _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a, std::false_type);
+
+ public:
+ _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a)
+ : _Rb_tree(std::move(__x), std::move(__a),
+ typename _Alloc_traits::is_always_equal{})
+ { }
#endif
~_Rb_tree() _GLIBCXX_NOEXCEPT
@@ -1591,12 +1608,11 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION
template<typename _Key, typename _Val, typename _KeyOfValue,
typename _Compare, typename _Alloc>
_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::
- _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a)
+ _Rb_tree(_Rb_tree&& __x, _Node_allocator&& __a, std::false_type __eq)
: _M_impl(__x._M_impl._M_key_compare, std::move(__a))
{
- using __eq = typename _Alloc_traits::is_always_equal;
if (__x._M_root() != nullptr)
- _M_move_data(__x, __eq());
+ _M_move_data(__x, __eq);
}
I think this constructor is simple enough that it should be defined
in-class, so it's inline.
There's no need to qualify true_type and false_type with the std
namespace (I don't know why some of the existing code does that).