зеркало из https://github.com/microsoft/STL.git
2007 строки
81 KiB
C++
2007 строки
81 KiB
C++
// xhash internal header
|
|
|
|
// Copyright (c) Microsoft Corporation.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
#pragma once
|
|
#ifndef _XHASH_
|
|
#define _XHASH_
|
|
#include <yvals_core.h>
|
|
#if _STL_COMPILER_PREPROCESSOR
|
|
#include <cmath>
|
|
#include <cstring>
|
|
#include <cwchar>
|
|
#include <list>
|
|
#include <tuple>
|
|
#include <vector>
|
|
#include <xbit_ops.h>
|
|
#include <xstring>
|
|
|
|
#if _HAS_CXX17
|
|
#include <xnode_handle.h>
|
|
#endif // _HAS_CXX17
|
|
|
|
#pragma pack(push, _CRT_PACKING)
|
|
#pragma warning(push, _STL_WARNING_LEVEL)
|
|
#pragma warning(disable : _STL_DISABLED_WARNINGS)
|
|
_STL_DISABLE_CLANG_WARNINGS
|
|
#pragma push_macro("new")
|
|
#undef new
|
|
|
|
namespace stdext {
|
|
using _STD basic_string;
|
|
using _STD less;
|
|
using _STD size_t;
|
|
|
|
template <class _Kty>
|
|
_NODISCARD size_t hash_value(const _Kty& _Keyval) noexcept {
|
|
if constexpr (_STD is_pointer_v<_Kty> || _STD is_null_pointer_v<_Kty>) {
|
|
return reinterpret_cast<size_t>(_Keyval) ^ 0xdeadbeefu;
|
|
} else {
|
|
return static_cast<size_t>(_Keyval) ^ 0xdeadbeefu;
|
|
}
|
|
}
|
|
|
|
template <class _Elem, class _Traits, class _Alloc>
|
|
_NODISCARD size_t hash_value(const basic_string<_Elem, _Traits, _Alloc>& _Str) noexcept {
|
|
return _STD _Hash_array_representation(_Str.c_str(), _Str.size());
|
|
}
|
|
|
|
_NODISCARD inline size_t hash_value(_In_z_ const char* _Str) noexcept {
|
|
return _STD _Hash_array_representation(_Str, _CSTD strlen(_Str));
|
|
}
|
|
|
|
_NODISCARD inline size_t hash_value(_In_z_ const wchar_t* _Str) noexcept {
|
|
return _STD _Hash_array_representation(_Str, _CSTD wcslen(_Str));
|
|
}
|
|
|
|
template <class _Kty, class _Pr = less<_Kty>>
|
|
class hash_compare { // traits class for hash containers
|
|
public:
|
|
enum { // parameters for hash table
|
|
bucket_size = 1 // 0 < bucket_size
|
|
};
|
|
|
|
hash_compare() = default;
|
|
hash_compare(const _Pr& _Pred) noexcept(_STD is_nothrow_copy_constructible_v<_Pr>) : comp(_Pred) {}
|
|
|
|
_NODISCARD size_t operator()(const _Kty& _Keyval) const noexcept(noexcept(hash_value(_Keyval))) {
|
|
long _Quot = static_cast<long>(hash_value(_Keyval) & LONG_MAX); // TRANSITION, ADL?
|
|
ldiv_t _Qrem = _CSTD ldiv(_Quot, 127773);
|
|
|
|
_Qrem.rem = 16807 * _Qrem.rem - 2836 * _Qrem.quot;
|
|
if (_Qrem.rem < 0) {
|
|
_Qrem.rem += LONG_MAX;
|
|
}
|
|
|
|
return static_cast<size_t>(_Qrem.rem);
|
|
}
|
|
|
|
_NODISCARD bool operator()(const _Kty& _Keyval1, const _Kty& _Keyval2) const
|
|
noexcept(noexcept(comp(_Keyval1, _Keyval2))) {
|
|
// test if _Keyval1 ordered before _Keyval2
|
|
return comp(_Keyval1, _Keyval2);
|
|
}
|
|
|
|
_Pr comp{}; // the comparator object
|
|
};
|
|
} // namespace stdext
|
|
|
|
_STD_BEGIN
|
|
using stdext::hash_compare; // TRANSITION, protobuf used this until v3.7.0, released Feb 28, 2019
|
|
|
|
template <class _Kty, class _Hasher, class _Keyeq, class = void>
|
|
struct _Uhash_choose_transparency {
|
|
// transparency selector for non-transparent hashed containers
|
|
template <class>
|
|
using _Deduce_key = const _Kty&;
|
|
};
|
|
|
|
#if _HAS_CXX20
|
|
template <class _Kty, class _Hasher, class _Keyeq>
|
|
struct _Uhash_choose_transparency<_Kty, _Hasher, _Keyeq,
|
|
void_t<typename _Hasher::is_transparent, typename _Keyeq::is_transparent>> {
|
|
// transparency selector for transparent hashed containers
|
|
template <class _Keyty>
|
|
using _Deduce_key = const _Keyty&;
|
|
|
|
using _Transparent = void;
|
|
};
|
|
#endif // _HAS_CXX20
|
|
|
|
template <class _Hasher, class _Kty>
|
|
_INLINE_VAR constexpr bool _Nothrow_hash = noexcept(
|
|
static_cast<size_t>(_STD declval<const _Hasher&>()(_STD declval<const _Kty&>())));
|
|
|
|
template <class _Kty, class _Hasher, class _Keyeq>
|
|
class _Uhash_compare
|
|
: public _Uhash_choose_transparency<_Kty, _Hasher, _Keyeq> { // traits class for unordered containers
|
|
public:
|
|
enum { // parameters for hash table
|
|
bucket_size = 1 // 0 < bucket_size
|
|
};
|
|
|
|
_Uhash_compare() noexcept(
|
|
conjunction_v<is_nothrow_default_constructible<_Hasher>, is_nothrow_default_constructible<_Keyeq>>)
|
|
: _Mypair(_Zero_then_variadic_args_t{}, _Zero_then_variadic_args_t{}, 0.0f) {}
|
|
|
|
explicit _Uhash_compare(const _Hasher& _Hasharg) noexcept(
|
|
conjunction_v<is_nothrow_copy_constructible<_Hasher>, is_nothrow_default_constructible<_Keyeq>>)
|
|
: _Mypair(_One_then_variadic_args_t{}, _Hasharg, _Zero_then_variadic_args_t{}, 0.0f) {}
|
|
|
|
explicit _Uhash_compare(const _Hasher& _Hasharg, const _Keyeq& _Keyeqarg) noexcept(
|
|
conjunction_v<is_nothrow_copy_constructible<_Hasher>, is_nothrow_copy_constructible<_Keyeq>>)
|
|
: _Mypair(_One_then_variadic_args_t{}, _Hasharg, _One_then_variadic_args_t{}, _Keyeqarg, 0.0f) {}
|
|
|
|
template <class _Keyty>
|
|
_NODISCARD size_t operator()(const _Keyty& _Keyval) const noexcept(_Nothrow_hash<_Hasher, _Keyty>) {
|
|
// hash _Keyval to size_t value
|
|
return static_cast<size_t>(_Mypair._Get_first()(_Keyval));
|
|
}
|
|
|
|
template <class _Keyty1, class _Keyty2>
|
|
_NODISCARD bool operator()(const _Keyty1& _Keyval1, const _Keyty2& _Keyval2) const
|
|
noexcept(_Nothrow_compare<_Keyeq, _Keyty1, _Keyty2>) {
|
|
// test if _Keyval1 NOT equal to _Keyval2
|
|
return !static_cast<bool>(_Mypair._Myval2._Get_first()(_Keyval1, _Keyval2));
|
|
}
|
|
|
|
_NODISCARD float& _Get_max_bucket_size() noexcept {
|
|
return _Mypair._Myval2._Myval2;
|
|
}
|
|
|
|
_NODISCARD const float& _Get_max_bucket_size() const noexcept {
|
|
return _Mypair._Myval2._Myval2;
|
|
}
|
|
|
|
void swap(_Uhash_compare& _Rhs) noexcept(
|
|
conjunction_v<_Is_nothrow_swappable<_Hasher>, _Is_nothrow_swappable<_Keyeq>>) {
|
|
_Swap_adl(_Mypair._Get_first(), _Rhs._Mypair._Get_first());
|
|
auto& _Lsecond = _Mypair._Myval2;
|
|
auto& _Rsecond = _Rhs._Mypair._Myval2;
|
|
_Swap_adl(_Lsecond._Get_first(), _Rsecond._Get_first());
|
|
_STD swap(_Lsecond._Myval2, _Rsecond._Myval2);
|
|
}
|
|
|
|
_Compressed_pair<_Hasher, _Compressed_pair<_Keyeq, float>> _Mypair;
|
|
};
|
|
|
|
template <class _Iter, class _Val>
|
|
struct _Reinterpret_move_iter {
|
|
_Iter _Base;
|
|
|
|
using iterator_category = input_iterator_tag;
|
|
using value_type = typename iterator_traits<_Iter>::value_type;
|
|
using difference_type = typename iterator_traits<_Iter>::difference_type;
|
|
using reference = _Val&&;
|
|
// pointer intentionally omitted
|
|
|
|
reference operator*() const {
|
|
return static_cast<reference>(reinterpret_cast<_Val&>(*_Base));
|
|
}
|
|
|
|
_Reinterpret_move_iter& operator++() {
|
|
++_Base;
|
|
return *this;
|
|
}
|
|
|
|
// post ++ intentionally omitted
|
|
|
|
_NODISCARD_FRIEND bool operator==(const _Reinterpret_move_iter& _Lhs, const _Reinterpret_move_iter& _Rhs) {
|
|
return _Lhs._Base == _Rhs._Base;
|
|
}
|
|
|
|
#if !_HAS_CXX20
|
|
_NODISCARD_FRIEND bool operator!=(const _Reinterpret_move_iter& _Lhs, const _Reinterpret_move_iter& _Rhs) {
|
|
return _Lhs._Base != _Rhs._Base;
|
|
}
|
|
#endif // !_HAS_CXX20
|
|
};
|
|
|
|
template <class _Alnode>
|
|
struct _List_head_construct_ptr {
|
|
using value_type = typename _Alnode::value_type;
|
|
static_assert(_Is_specialization_v<value_type, _List_node>, "_List_head_construct_ptr allocator not rebound");
|
|
using _Alnode_traits = allocator_traits<_Alnode>;
|
|
using pointer = typename _Alnode_traits::pointer;
|
|
_Alnode& _Al;
|
|
pointer _Newhead;
|
|
|
|
explicit _List_head_construct_ptr(_Alnode& _Al_) : _Al(_Al_), _Newhead(value_type::_Buyheadnode(_Al)) {}
|
|
|
|
template <class _Val_types>
|
|
_List_head_construct_ptr(_Alnode& _Al_, _List_val<_Val_types>& _Mycont)
|
|
: _Al(_Al_), _Newhead(value_type::_Buyheadnode(_Al)) {
|
|
_Mycont._Myhead = _Newhead;
|
|
}
|
|
|
|
_List_head_construct_ptr(const _List_head_construct_ptr&) = delete;
|
|
_List_head_construct_ptr& operator=(const _List_head_construct_ptr&) = delete;
|
|
|
|
_NODISCARD pointer _Release() noexcept {
|
|
return _STD exchange(_Newhead, nullptr);
|
|
}
|
|
|
|
~_List_head_construct_ptr() {
|
|
if (_Newhead) {
|
|
value_type::_Freenode0(_Al, _Newhead);
|
|
}
|
|
}
|
|
};
|
|
|
|
template <class _Nodeptr>
|
|
struct _Hash_find_last_result {
|
|
_Nodeptr _Insert_before;
|
|
_Nodeptr _Duplicate;
|
|
};
|
|
|
|
template <class _Aliter>
|
|
struct _Hash_vec {
|
|
// TRANSITION, ABI: "vector" for ABI compatibility that doesn't call allocator::construct
|
|
using _Aliter_traits = allocator_traits<_Aliter>;
|
|
using value_type = typename _Aliter::value_type;
|
|
using size_type = typename _Aliter_traits::size_type;
|
|
using difference_type = typename _Aliter_traits::difference_type;
|
|
using pointer = typename _Aliter_traits::pointer;
|
|
using _Aliter_scary_val = _Vector_val<conditional_t<_Is_simple_alloc_v<_Aliter>, _Simple_types<value_type>,
|
|
_Vec_iter_types<value_type, size_type, difference_type, pointer, typename _Aliter_traits::const_pointer,
|
|
value_type&, const value_type&>>>;
|
|
|
|
_Hash_vec() : _Mypair(_Zero_then_variadic_args_t{}) {
|
|
_Mypair._Myval2._Alloc_proxy(_GET_PROXY_ALLOCATOR(_Aliter, _Mypair._Get_first()));
|
|
}
|
|
|
|
template <class _Any_alloc, enable_if_t<!is_same_v<_Remove_cvref_t<_Any_alloc>, _Hash_vec>, int> = 0>
|
|
explicit _Hash_vec(_Any_alloc&& _Al) noexcept
|
|
: _Mypair(_One_then_variadic_args_t{}, _STD forward<_Any_alloc>(_Al)) { // construct empty vector, allocator
|
|
_Mypair._Myval2._Alloc_proxy(_GET_PROXY_ALLOCATOR(_Aliter, _Mypair._Get_first()));
|
|
}
|
|
|
|
_Hash_vec(const _Hash_vec&) = delete;
|
|
_Hash_vec& operator=(const _Hash_vec&) = delete;
|
|
|
|
_NODISCARD size_type size() const noexcept {
|
|
return static_cast<size_type>(_Mypair._Myval2._Mylast - _Mypair._Myval2._Myfirst);
|
|
}
|
|
|
|
_NODISCARD size_type max_size() const noexcept {
|
|
return (_STD min)(static_cast<size_type>((numeric_limits<difference_type>::max)()),
|
|
_Aliter_traits::max_size(_Mypair._Get_first()));
|
|
}
|
|
|
|
void _Assign_grow(const size_type _Cells, const value_type _Val) {
|
|
// set the elements stored here to _Cells copies of _Val, leaving the value unchanged if an exception is thrown
|
|
const auto _Oldsize = size();
|
|
_STL_INTERNAL_CHECK(_Oldsize <= _Cells);
|
|
auto& _Alvec = _Mypair._Get_first();
|
|
if (_Oldsize < _Cells) {
|
|
const auto _Newvec = _Alvec.allocate(_Cells); // throws
|
|
// nothrow hereafter
|
|
if (_Oldsize != 0) {
|
|
_Destroy_range(_Mypair._Myval2._Myfirst, _Mypair._Myval2._Mylast);
|
|
_Alvec.deallocate(_Mypair._Myval2._Myfirst, _Oldsize);
|
|
}
|
|
|
|
_Mypair._Myval2._Myfirst = _Newvec;
|
|
const auto _Newend = _Newvec + _Cells;
|
|
_Mypair._Myval2._Mylast = _Newend;
|
|
_Mypair._Myval2._Myend = _Newend;
|
|
_STD uninitialized_fill(_Newvec, _Newend, _Val);
|
|
} else {
|
|
_STD fill(_Mypair._Myval2._Myfirst, _Mypair._Myval2._Mylast, _Val);
|
|
}
|
|
}
|
|
|
|
void _Tidy() noexcept {
|
|
_Destroy_range(_Mypair._Myval2._Myfirst, _Mypair._Myval2._Mylast);
|
|
_Mypair._Get_first().deallocate(_Mypair._Myval2._Myfirst, size());
|
|
_Mypair._Myval2._Myfirst = nullptr;
|
|
_Mypair._Myval2._Mylast = nullptr;
|
|
_Mypair._Myval2._Myend = nullptr;
|
|
}
|
|
|
|
~_Hash_vec() {
|
|
_Tidy();
|
|
#if _ITERATOR_DEBUG_LEVEL != 0
|
|
auto&& _Alproxy = _GET_PROXY_ALLOCATOR(_Aliter, _Mypair._Get_first());
|
|
_Delete_plain_internal(_Alproxy, _STD exchange(_Mypair._Myval2._Myproxy, nullptr));
|
|
#endif // _ITERATOR_DEBUG_LEVEL != 0
|
|
}
|
|
|
|
_Compressed_pair<_Aliter, _Aliter_scary_val> _Mypair;
|
|
};
|
|
|
|
template <class _Traits>
|
|
class _Hash { // hash table -- list with vector of iterators for quick access
|
|
protected:
|
|
using _Mylist = list<typename _Traits::value_type, typename _Traits::allocator_type>;
|
|
using _Alnode = typename _Mylist::_Alnode;
|
|
using _Alnode_traits = typename _Mylist::_Alnode_traits;
|
|
using _Node = typename _Mylist::_Node;
|
|
using _Nodeptr = typename _Mylist::_Nodeptr;
|
|
using _Mutable_value_type = typename _Traits::_Mutable_value_type;
|
|
|
|
using _Key_compare = typename _Traits::key_compare;
|
|
using _Value_compare = typename _Traits::value_compare;
|
|
|
|
public:
|
|
using key_type = typename _Traits::key_type;
|
|
|
|
using value_type = typename _Mylist::value_type;
|
|
using allocator_type = typename _Mylist::allocator_type;
|
|
using size_type = typename _Mylist::size_type;
|
|
using difference_type = typename _Mylist::difference_type;
|
|
using pointer = typename _Mylist::pointer;
|
|
using const_pointer = typename _Mylist::const_pointer;
|
|
using reference = value_type&;
|
|
using const_reference = const value_type&;
|
|
|
|
using iterator =
|
|
conditional_t<is_same_v<key_type, value_type>, typename _Mylist::const_iterator, typename _Mylist::iterator>;
|
|
using const_iterator = typename _Mylist::const_iterator;
|
|
|
|
using _Unchecked_iterator = conditional_t<is_same_v<key_type, value_type>,
|
|
typename _Mylist::_Unchecked_const_iterator, typename _Mylist::_Unchecked_iterator>;
|
|
using _Unchecked_const_iterator = typename _Mylist::_Unchecked_const_iterator;
|
|
|
|
using _Aliter = _Rebind_alloc_t<_Alnode, _Unchecked_iterator>;
|
|
|
|
static constexpr size_type _Bucket_size = _Key_compare::bucket_size;
|
|
static constexpr size_type _Min_buckets = 8; // must be a positive power of 2
|
|
static constexpr bool _Multi = _Traits::_Multi;
|
|
|
|
template <class _TraitsT>
|
|
friend bool _Hash_equal(const _Hash<_TraitsT>& _Left, const _Hash<_TraitsT>& _Right);
|
|
|
|
protected:
|
|
_Hash(const _Key_compare& _Parg, const allocator_type& _Al)
|
|
: _Traitsobj(_Parg), _List(_Al), _Vec(_Al), _Mask(_Min_buckets - 1), _Maxidx(_Min_buckets) {
|
|
// construct empty hash table
|
|
_Max_bucket_size() = _Bucket_size;
|
|
_Vec._Assign_grow(_Min_buckets * 2, _List._Unchecked_end());
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Stl_internal_check_container_invariants();
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
}
|
|
|
|
template <class _Any_alloc>
|
|
_Hash(const _Hash& _Right, const _Any_alloc& _Al)
|
|
: _Traitsobj(_Right._Traitsobj), _List(static_cast<allocator_type>(_Al)), _Vec(_Al), _Mask(_Right._Mask),
|
|
_Maxidx(_Right._Maxidx) {
|
|
// construct hash table by copying _Right
|
|
_Vec._Assign_grow(_Right._Vec.size(), _List._Unchecked_end());
|
|
insert(_Right._Unchecked_begin(), _Right._Unchecked_end());
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Stl_internal_check_container_invariants();
|
|
_Right._Stl_internal_check_container_invariants();
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
}
|
|
|
|
_Hash(_Hash&& _Right)
|
|
: _Traitsobj(_Right._Traitsobj), _List(_Move_allocator_tag{}, _Right._List._Getal()),
|
|
_Vec(_STD move(_Right._Vec._Mypair._Get_first())) {
|
|
_Vec._Assign_grow(_Min_buckets * 2, _Unchecked_end());
|
|
_List._Swap_val(_Right._List);
|
|
_Vec._Mypair._Myval2._Swap_val(_Right._Vec._Mypair._Myval2);
|
|
_Mask = _STD exchange(_Right._Mask, _Min_buckets - 1);
|
|
_Maxidx = _STD exchange(_Right._Maxidx, _Min_buckets);
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Stl_internal_check_container_invariants();
|
|
_Right._Stl_internal_check_container_invariants();
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
}
|
|
|
|
private:
|
|
void _Move_construct_equal_alloc(_Hash& _Right) {
|
|
_Vec._Assign_grow(_Min_buckets * 2, _Unchecked_end());
|
|
_List._Swap_val(_Right._List);
|
|
_Vec._Mypair._Myval2._Swap_val(_Right._Vec._Mypair._Myval2);
|
|
_Mask = _STD exchange(_Right._Mask, _Min_buckets - 1);
|
|
_Maxidx = _STD exchange(_Right._Maxidx, _Min_buckets);
|
|
}
|
|
|
|
public:
|
|
_Hash(_Hash&& _Right, const allocator_type& _Al) : _Traitsobj(_Right._Traitsobj), _List(_Al), _Vec(_Al) {
|
|
// construct hash table by moving _Right, allocator
|
|
if constexpr (_Alnode_traits::is_always_equal::value) {
|
|
_Move_construct_equal_alloc(_Right);
|
|
} else if (_List._Getal() == _Right._List._Getal()) {
|
|
_Move_construct_equal_alloc(_Right);
|
|
} else {
|
|
_Maxidx = _Min_buckets;
|
|
const auto _Myhead = _List._Mypair._Myval2._Myhead;
|
|
for (auto& _Val : _Right._List) {
|
|
_List._Emplace(_Myhead, reinterpret_cast<_Mutable_value_type&&>(_Val));
|
|
}
|
|
_Reinsert_with_invalid_vec();
|
|
_Right.clear();
|
|
}
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Stl_internal_check_container_invariants();
|
|
_Right._Stl_internal_check_container_invariants();
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
}
|
|
|
|
private:
|
|
void _Swap_val(_Hash& _Right) noexcept { // swap contents with equal allocator _Hash _Right
|
|
_List._Swap_val(_Right._List);
|
|
_Vec._Mypair._Myval2._Swap_val(_Right._Vec._Mypair._Myval2);
|
|
_STD swap(_Mask, _Right._Mask);
|
|
_STD swap(_Maxidx, _Right._Maxidx);
|
|
}
|
|
|
|
struct _Min_buckets_construct_ptr {
|
|
using pointer = typename allocator_traits<_Aliter>::pointer;
|
|
_Aliter& _Al;
|
|
pointer _Base;
|
|
_Min_buckets_construct_ptr(_Aliter& _Al_) : _Al(_Al_), _Base(_Al.allocate(_Min_buckets * 2)) {}
|
|
_Min_buckets_construct_ptr(const _Min_buckets_construct_ptr&) = delete;
|
|
_NODISCARD pointer _Release(_Unchecked_iterator _Newend) noexcept {
|
|
_STD uninitialized_fill(_Base, _Base + _Min_buckets * 2, _Newend);
|
|
return _STD exchange(_Base, nullptr);
|
|
}
|
|
~_Min_buckets_construct_ptr() {
|
|
if (_Base) {
|
|
_Al.deallocate(_Base, _Min_buckets * 2);
|
|
}
|
|
}
|
|
};
|
|
|
|
void _Pocma_both(_Hash& _Right) {
|
|
_Pocma(_List._Getal(), _Right._List._Getal());
|
|
_Pocma(_Vec._Mypair._Get_first(), _Right._Vec._Mypair._Get_first());
|
|
}
|
|
|
|
struct _NODISCARD _Clear_guard {
|
|
_Hash* _Target;
|
|
|
|
explicit _Clear_guard(_Hash* const _Target_) : _Target(_Target_) {}
|
|
|
|
_Clear_guard(const _Clear_guard&) = delete;
|
|
_Clear_guard& operator=(const _Clear_guard&) = delete;
|
|
|
|
~_Clear_guard() {
|
|
if (_Target) {
|
|
_Target->clear();
|
|
}
|
|
}
|
|
};
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
struct _NODISCARD _Check_container_invariants_guard {
|
|
const _Hash& _Target;
|
|
|
|
explicit _Check_container_invariants_guard(const _Hash& _Target_) : _Target(_Target_) {}
|
|
|
|
_Check_container_invariants_guard(const _Check_container_invariants_guard&) = delete;
|
|
_Check_container_invariants_guard& operator=(const _Check_container_invariants_guard&) = delete;
|
|
|
|
~_Check_container_invariants_guard() {
|
|
_Target._Stl_internal_check_container_invariants();
|
|
}
|
|
};
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
|
|
public:
|
|
_Hash& operator=(_Hash&& _Right) { // assign by moving _Right
|
|
if (this == _STD addressof(_Right)) {
|
|
return *this;
|
|
}
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Check_container_invariants_guard _Check_self{*this};
|
|
_Check_container_invariants_guard _Check_right{_Right};
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
|
|
auto& _Al = _Getal();
|
|
auto& _Right_al = _Right._Getal();
|
|
constexpr auto _Pocma_val = _Choose_pocma_v<_Alnode>;
|
|
if constexpr (_Pocma_val == _Pocma_values::_Propagate_allocators) {
|
|
if (_Al != _Right_al) {
|
|
// allocate all the parts necessary to maintain _Hash invariants using _Right's allocator
|
|
auto&& _Alproxy = _GET_PROXY_ALLOCATOR(_Alnode, _Al);
|
|
auto&& _Right_alproxy = _GET_PROXY_ALLOCATOR(_Alnode, _Right_al);
|
|
_Container_proxy_ptr<_Alnode> _List_proxy(_Right_alproxy, _Leave_proxy_unbound{});
|
|
_Container_proxy_ptr<_Alnode> _Vec_proxy(_Right_alproxy, _Leave_proxy_unbound{});
|
|
_List_head_construct_ptr<_Alnode> _Newhead(_Right_al);
|
|
_Min_buckets_construct_ptr _Buckets(_Right._Vec._Mypair._Get_first());
|
|
|
|
// assign the hash/compare ops; we have no control over whether this throws, and if it does we want
|
|
// to do nothing
|
|
_Traitsobj = _Right._Traitsobj;
|
|
|
|
// nothrow hereafter
|
|
|
|
// release any state we are currently owning, and propagate the allocators
|
|
_List._Tidy();
|
|
_Vec._Tidy();
|
|
_Pocma_both(_Right);
|
|
|
|
// assign the empty list to _Right._List (except the allocators), and take _Right's _List data
|
|
auto& _List_data = _List._Mypair._Myval2;
|
|
auto& _Right_list_data = _Right._List._Mypair._Myval2;
|
|
_List_data._Myhead = _STD exchange(_Right_list_data._Myhead, _Newhead._Release());
|
|
_List_data._Mysize = _STD exchange(_Right_list_data._Mysize, size_type{0});
|
|
_List_proxy._Bind(_Alproxy, _STD addressof(_List_data));
|
|
_List_data._Swap_proxy_and_iterators(_Right_list_data);
|
|
|
|
// assign the _Min_buckets into _Right's _Vec data and take _Right's _Vec data
|
|
auto& _Vec_data = _Vec._Mypair._Myval2;
|
|
auto& _Right_vec_data = _Right._Vec._Mypair._Myval2;
|
|
|
|
const auto _Newfirst = _Buckets._Release(_Right._Unchecked_end());
|
|
const auto _Newlast = _Newfirst + _Min_buckets * 2;
|
|
|
|
_Vec_data._Myfirst = _STD exchange(_Right_vec_data._Myfirst, _Newfirst);
|
|
_Vec_data._Mylast = _STD exchange(_Right_vec_data._Mylast, _Newlast);
|
|
_Vec_data._Myend = _STD exchange(_Right_vec_data._Myend, _Newlast);
|
|
_Vec_proxy._Bind(_Alproxy, _STD addressof(_Vec_data));
|
|
_Vec_data._Swap_proxy_and_iterators(_Right_vec_data);
|
|
|
|
// give _Right the default _Mask and _Maxidx values and take its former values
|
|
_Mask = _STD exchange(_Right._Mask, _Min_buckets - 1);
|
|
_Maxidx = _STD exchange(_Right._Maxidx, _Min_buckets);
|
|
|
|
return *this;
|
|
}
|
|
} else if constexpr (_Pocma_val == _Pocma_values::_No_propagate_allocators) {
|
|
if (_Al != _Right_al) {
|
|
_Clear_guard _Guard{this};
|
|
_Traitsobj = _Right._Traitsobj;
|
|
using _Adapter = _Reinterpret_move_iter<typename _Mylist::_Unchecked_iterator, _Mutable_value_type>;
|
|
_List.template _Assign_cast<_Mutable_value_type&>(
|
|
_Adapter{_Right._List._Unchecked_begin()}, _Adapter{_Right._List._Unchecked_end()});
|
|
_Reinsert_with_invalid_vec();
|
|
_Guard._Target = nullptr;
|
|
|
|
return *this;
|
|
}
|
|
}
|
|
|
|
clear();
|
|
_Traitsobj = _Right._Traitsobj;
|
|
_Pocma_both(_Right);
|
|
_Swap_val(_Right);
|
|
|
|
return *this;
|
|
}
|
|
|
|
template <class... _Valtys>
|
|
conditional_t<_Multi, iterator, pair<iterator, bool>> emplace(_Valtys&&... _Vals) {
|
|
// try to insert value_type(_Vals...)
|
|
using _In_place_key_extractor = typename _Traits::template _In_place_key_extractor<_Remove_cvref_t<_Valtys>...>;
|
|
if constexpr (_Multi) {
|
|
_Check_max_size();
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), _STD forward<_Valtys>(_Vals)...);
|
|
const auto& _Keyval = _Traits::_Kfn(_Newnode._Ptr->_Myval);
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
}
|
|
|
|
const auto _Target = _Find_last(_Keyval, _Hashval);
|
|
return _List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release()));
|
|
} else if constexpr (_In_place_key_extractor::_Extractable) {
|
|
const auto& _Keyval = _In_place_key_extractor::_Extract(_Vals...);
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_last(_Keyval, _Hashval);
|
|
if (_Target._Duplicate) {
|
|
return {_List._Make_iter(_Target._Duplicate), false};
|
|
}
|
|
|
|
_Check_max_size();
|
|
// invalidates _Keyval:
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), _STD forward<_Valtys>(_Vals)...);
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_last(_Traits::_Kfn(_Newnode._Ptr->_Myval), _Hashval);
|
|
}
|
|
|
|
return {
|
|
_List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release())), true};
|
|
} else {
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), _STD forward<_Valtys>(_Vals)...);
|
|
const auto& _Keyval = _Traits::_Kfn(_Newnode._Ptr->_Myval);
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_last(_Keyval, _Hashval);
|
|
if (_Target._Duplicate) {
|
|
return {_List._Make_iter(_Target._Duplicate), false};
|
|
}
|
|
|
|
_Check_max_size();
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_last(_Traits::_Kfn(_Newnode._Ptr->_Myval), _Hashval);
|
|
}
|
|
|
|
return {
|
|
_List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release())), true};
|
|
}
|
|
}
|
|
|
|
template <class... _Valtys>
|
|
iterator emplace_hint(const_iterator _Hint, _Valtys&&... _Vals) { // try to insert value_type(_Vals...)
|
|
using _In_place_key_extractor = typename _Traits::template _In_place_key_extractor<_Remove_cvref_t<_Valtys>...>;
|
|
if constexpr (_Multi) {
|
|
_Check_max_size();
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), _STD forward<_Valtys>(_Vals)...);
|
|
const auto& _Keyval = _Traits::_Kfn(_Newnode._Ptr->_Myval);
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
}
|
|
|
|
const auto _Target = _Find_hint(_Hint._Ptr, _Keyval, _Hashval);
|
|
return _List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release()));
|
|
} else if constexpr (_In_place_key_extractor::_Extractable) {
|
|
const auto& _Keyval = _In_place_key_extractor::_Extract(_Vals...);
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_hint(_Hint._Ptr, _Keyval, _Hashval);
|
|
if (_Target._Duplicate) {
|
|
return _List._Make_iter(_Target._Duplicate);
|
|
}
|
|
|
|
_Check_max_size();
|
|
// invalidates _Keyval:
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), _STD forward<_Valtys>(_Vals)...);
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_hint(_Hint._Ptr, _Traits::_Kfn(_Newnode._Ptr->_Myval), _Hashval);
|
|
}
|
|
|
|
return _List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release()));
|
|
} else {
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), _STD forward<_Valtys>(_Vals)...);
|
|
const auto& _Keyval = _Traits::_Kfn(_Newnode._Ptr->_Myval);
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_hint(_Hint._Ptr, _Keyval, _Hashval);
|
|
if (_Target._Duplicate) {
|
|
return _List._Make_iter(_Target._Duplicate);
|
|
}
|
|
|
|
_Check_max_size();
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_hint(_Hint._Ptr, _Traits::_Kfn(_Newnode._Ptr->_Myval), _Hashval);
|
|
}
|
|
|
|
return _List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release()));
|
|
}
|
|
}
|
|
|
|
protected:
|
|
template <class _Keyty, class... _Mappedty>
|
|
pair<_Nodeptr, bool> _Try_emplace(_Keyty&& _Keyval_arg, _Mappedty&&... _Mapval) {
|
|
const auto& _Keyval = _Keyval_arg;
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_last(_Keyval, _Hashval);
|
|
if (_Target._Duplicate) {
|
|
return {_Target._Duplicate, false};
|
|
}
|
|
|
|
_Check_max_size();
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), piecewise_construct,
|
|
_STD forward_as_tuple(_STD forward<_Keyty>(_Keyval_arg)),
|
|
_STD forward_as_tuple(_STD forward<_Mappedty>(_Mapval)...));
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_last(_Traits::_Kfn(_Newnode._Ptr->_Myval), _Hashval);
|
|
}
|
|
|
|
return {_Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release()), true};
|
|
}
|
|
|
|
template <class _Keyty, class... _Mappedty>
|
|
_Nodeptr _Try_emplace_hint(const _Nodeptr _Hint, _Keyty&& _Keyval_arg, _Mappedty&&... _Mapval) {
|
|
const auto& _Keyval = _Keyval_arg;
|
|
const auto _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_hint(_Hint, _Keyval, _Hashval);
|
|
if (_Target._Duplicate) {
|
|
return _Target._Duplicate;
|
|
}
|
|
|
|
_Check_max_size();
|
|
// might invalidate _Keyval:
|
|
_List_node_emplace_op2<_Alnode> _Newnode(_List._Getal(), piecewise_construct,
|
|
_STD forward_as_tuple(_STD forward<_Keyty>(_Keyval_arg)),
|
|
_STD forward_as_tuple(_STD forward<_Mappedty>(_Mapval)...));
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_hint(_Hint, _Traits::_Kfn(_Newnode._Ptr->_Myval), _Hashval);
|
|
}
|
|
|
|
return _Insert_new_node_before(_Hashval, _Target._Insert_before, _Newnode._Release());
|
|
}
|
|
|
|
private:
|
|
void _Pocca_both(const _Hash& _Right) {
|
|
_Pocca(_List._Getal(), _Right._List._Getal());
|
|
_Pocca(_Vec._Mypair._Get_first(), _Right._Vec._Mypair._Get_first());
|
|
}
|
|
|
|
public:
|
|
_Hash& operator=(const _Hash& _Right) {
|
|
if (this == _STD addressof(_Right)) {
|
|
return *this;
|
|
}
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Check_container_invariants_guard _Check_self{*this};
|
|
_Check_container_invariants_guard _Check_right{_Right};
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
|
|
if constexpr (_Choose_pocca_v<_Alnode>) {
|
|
auto& _Al = _Getal();
|
|
const auto& _Right_al = _Right._Getal();
|
|
if (_Al != _Right_al) {
|
|
auto&& _Alproxy = _GET_PROXY_ALLOCATOR(_Alnode, _Al);
|
|
auto&& _Right_alproxy = _GET_PROXY_ALLOCATOR(_Alnode, _Right_al);
|
|
_Container_proxy_ptr<_Alnode> _Vec_proxy(_Right_alproxy, _Leave_proxy_unbound{});
|
|
_List._Reload_sentinel_and_proxy(_Right._List);
|
|
_Vec._Tidy();
|
|
_Pocca_both(_Right);
|
|
_Vec_proxy._Bind(_Alproxy, _STD addressof(_Vec._Mypair._Myval2));
|
|
|
|
_Clear_guard _Guard{this};
|
|
_Traitsobj = _Right._Traitsobj;
|
|
_List.template _Assign_cast<_Mutable_value_type&>(
|
|
_Right._List._Unchecked_begin(), _Right._List._Unchecked_end());
|
|
_Reinsert_with_invalid_vec();
|
|
_Guard._Target = nullptr;
|
|
|
|
return *this;
|
|
}
|
|
}
|
|
|
|
_Clear_guard _Guard{this};
|
|
_Traitsobj = _Right._Traitsobj;
|
|
_Pocca_both(_Right);
|
|
_List.template _Assign_cast<_Mutable_value_type&>(
|
|
_Right._List._Unchecked_begin(), _Right._List._Unchecked_end());
|
|
_Reinsert_with_invalid_vec();
|
|
_Guard._Target = nullptr;
|
|
|
|
return *this;
|
|
}
|
|
|
|
_NODISCARD iterator begin() noexcept {
|
|
return _List.begin();
|
|
}
|
|
|
|
_NODISCARD const_iterator begin() const noexcept {
|
|
return _List.begin();
|
|
}
|
|
|
|
_NODISCARD iterator end() noexcept {
|
|
return _List.end();
|
|
}
|
|
|
|
_NODISCARD const_iterator end() const noexcept {
|
|
return _List.end();
|
|
}
|
|
|
|
_Unchecked_iterator _Unchecked_begin() noexcept {
|
|
return _List._Unchecked_begin();
|
|
}
|
|
|
|
_Unchecked_const_iterator _Unchecked_begin() const noexcept {
|
|
return _List._Unchecked_begin();
|
|
}
|
|
|
|
_Unchecked_iterator _Unchecked_end() noexcept {
|
|
return _List._Unchecked_end();
|
|
}
|
|
|
|
_Unchecked_const_iterator _Unchecked_end() const noexcept {
|
|
return _List._Unchecked_end();
|
|
}
|
|
|
|
_NODISCARD const_iterator cbegin() const noexcept {
|
|
return begin();
|
|
}
|
|
|
|
_NODISCARD const_iterator cend() const noexcept {
|
|
return end();
|
|
}
|
|
|
|
_NODISCARD size_type size() const noexcept {
|
|
return _List.size();
|
|
}
|
|
|
|
_NODISCARD size_type max_size() const noexcept {
|
|
return _List.max_size();
|
|
}
|
|
|
|
_NODISCARD bool empty() const noexcept {
|
|
return _List.empty();
|
|
}
|
|
|
|
_NODISCARD allocator_type get_allocator() const noexcept {
|
|
return static_cast<allocator_type>(_List.get_allocator());
|
|
}
|
|
|
|
using local_iterator = iterator;
|
|
using const_local_iterator = const_iterator;
|
|
|
|
_NODISCARD size_type bucket_count() const noexcept {
|
|
return _Maxidx;
|
|
}
|
|
|
|
_NODISCARD size_type max_bucket_count() const noexcept {
|
|
return _Vec.max_size() >> 1;
|
|
}
|
|
|
|
_NODISCARD size_type bucket(const key_type& _Keyval) const
|
|
noexcept(_Nothrow_hash<_Traits, key_type>) /* strengthened */ {
|
|
return _Traitsobj(_Keyval) & _Mask;
|
|
}
|
|
|
|
_NODISCARD size_type bucket_size(size_type _Bucket) const noexcept /* strengthened */ {
|
|
_Unchecked_iterator _Bucket_lo = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1];
|
|
if (_Bucket_lo == _Unchecked_end()) {
|
|
return 0;
|
|
}
|
|
|
|
return static_cast<size_type>(_STD distance(_Bucket_lo, _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1])) + 1;
|
|
}
|
|
|
|
_NODISCARD local_iterator begin(size_type _Bucket) noexcept /* strengthened */ {
|
|
return _List._Make_iter(_Vec._Mypair._Myval2._Myfirst[_Bucket << 1]._Ptr);
|
|
}
|
|
|
|
_NODISCARD const_local_iterator begin(size_type _Bucket) const noexcept /* strengthened */ {
|
|
return _List._Make_const_iter(_Vec._Mypair._Myval2._Myfirst[_Bucket << 1]._Ptr);
|
|
}
|
|
|
|
_NODISCARD local_iterator end(size_type _Bucket) noexcept /* strengthened */ {
|
|
_Nodeptr _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1]._Ptr;
|
|
if (_Bucket_hi != _List._Mypair._Myval2._Myhead) {
|
|
_Bucket_hi = _Bucket_hi->_Next;
|
|
}
|
|
|
|
return _List._Make_iter(_Bucket_hi);
|
|
}
|
|
|
|
_NODISCARD const_local_iterator end(size_type _Bucket) const noexcept /* strengthened */ {
|
|
_Nodeptr _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1]._Ptr;
|
|
if (_Bucket_hi != _List._Mypair._Myval2._Myhead) {
|
|
_Bucket_hi = _Bucket_hi->_Next;
|
|
}
|
|
|
|
return _List._Make_const_iter(_Bucket_hi);
|
|
}
|
|
|
|
_NODISCARD const_local_iterator cbegin(size_type _Bucket) const noexcept /* strengthened */ {
|
|
return _List._Make_const_iter(_Vec._Mypair._Myval2._Myfirst[_Bucket << 1]._Ptr);
|
|
}
|
|
|
|
_NODISCARD const_local_iterator cend(size_type _Bucket) const noexcept /* strengthened */ {
|
|
_Nodeptr _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1]._Ptr;
|
|
if (_Bucket_hi != _List._Mypair._Myval2._Myhead) {
|
|
_Bucket_hi = _Bucket_hi->_Next;
|
|
}
|
|
|
|
return _List._Make_const_iter(_Bucket_hi);
|
|
}
|
|
|
|
_NODISCARD float load_factor() const noexcept {
|
|
return static_cast<float>(size()) / static_cast<float>(bucket_count());
|
|
}
|
|
|
|
_NODISCARD float max_load_factor() const noexcept {
|
|
return _Max_bucket_size();
|
|
}
|
|
|
|
void max_load_factor(float _Newmax) noexcept /* strengthened */ {
|
|
_STL_ASSERT(!(_CSTD isnan)(_Newmax) && _Newmax > 0, "invalid hash load factor");
|
|
_Max_bucket_size() = _Newmax;
|
|
}
|
|
|
|
void rehash(size_type _Buckets) { // rebuild table with at least _Buckets buckets
|
|
// don't violate a.bucket_count() >= a.size() / a.max_load_factor() invariant:
|
|
_Buckets = (_STD max)(_Min_load_factor_buckets(_List.size()), _Buckets);
|
|
if (_Buckets <= _Maxidx) { // we already have enough buckets; nothing to do
|
|
return;
|
|
}
|
|
|
|
_Forced_rehash(_Buckets);
|
|
}
|
|
|
|
void reserve(size_type _Maxcount) { // rebuild table with room for _Maxcount elements
|
|
rehash(static_cast<size_type>(static_cast<float>(_Maxcount) / max_load_factor() + 0.5F));
|
|
}
|
|
|
|
conditional_t<_Multi, iterator, pair<iterator, bool>> insert(const value_type& _Val) {
|
|
return emplace(_Val);
|
|
}
|
|
|
|
conditional_t<_Multi, iterator, pair<iterator, bool>> insert(value_type&& _Val) {
|
|
return emplace(_STD move(_Val));
|
|
}
|
|
|
|
iterator insert(const_iterator _Hint, const value_type& _Val) {
|
|
return emplace_hint(_Hint, _Val);
|
|
}
|
|
|
|
iterator insert(const_iterator _Hint, value_type&& _Val) {
|
|
return emplace_hint(_Hint, _STD move(_Val));
|
|
}
|
|
|
|
template <class _Iter>
|
|
void insert(_Iter _First, _Iter _Last) {
|
|
_Adl_verify_range(_First, _Last);
|
|
auto _UFirst = _Get_unwrapped(_First);
|
|
const auto _ULast = _Get_unwrapped(_Last);
|
|
for (; _UFirst != _ULast; ++_UFirst) {
|
|
emplace(*_UFirst);
|
|
}
|
|
}
|
|
|
|
void insert(initializer_list<value_type> _Ilist) {
|
|
insert(_Ilist.begin(), _Ilist.end());
|
|
}
|
|
|
|
private:
|
|
_Nodeptr _Unchecked_erase(_Nodeptr _Plist) noexcept(_Nothrow_hash<_Traits, key_type>) {
|
|
size_type _Bucket = bucket(_Traits::_Kfn(_Plist->_Myval));
|
|
_Erase_bucket(_Plist, _Bucket);
|
|
return _List._Unchecked_erase(_Plist);
|
|
}
|
|
|
|
struct _Range_eraser {
|
|
_Range_eraser(const _Range_eraser&) = delete;
|
|
_Range_eraser& operator=(const _Range_eraser&) = delete;
|
|
|
|
#if _ITERATOR_DEBUG_LEVEL == 2
|
|
// Keep the list nodes around until we have found all those we will erase, for an O(iterators + erasures) update
|
|
// of the iterator chain.
|
|
_Range_eraser(_Mylist& _List_, const _Nodeptr _First_) noexcept
|
|
: _List(_List_), _First(_First_), _Next(_First_) {}
|
|
|
|
void _Bump_erased() noexcept {
|
|
_Next = _Next->_Next;
|
|
}
|
|
|
|
~_Range_eraser() noexcept {
|
|
_List._Unchecked_erase(_First, _Next);
|
|
}
|
|
|
|
_Mylist& _List;
|
|
const _Nodeptr _First;
|
|
_Nodeptr _Next;
|
|
#else // ^^^ _ITERATOR_DEBUG_LEVEL == 2 // _ITERATOR_DEBUG_LEVEL != 2 vvv
|
|
// Destroy the nodes as we encounter them to avoid a second traversal of the linked list.
|
|
_Range_eraser(_Mylist& _List_, const _Nodeptr _First_) noexcept
|
|
: _List(_List_), _Predecessor(_First_->_Prev), _Next(_First_) {}
|
|
|
|
void _Bump_erased() noexcept {
|
|
const auto _Oldnext = _Next;
|
|
_Next = _Oldnext->_Next;
|
|
_Node::_Freenode(_List._Getal(), _Oldnext);
|
|
--_List._Mypair._Myval2._Mysize;
|
|
}
|
|
|
|
~_Range_eraser() noexcept {
|
|
_Predecessor->_Next = _Next;
|
|
_Next->_Prev = _Predecessor;
|
|
}
|
|
|
|
_Mylist& _List;
|
|
const _Nodeptr _Predecessor;
|
|
_Nodeptr _Next;
|
|
#endif
|
|
};
|
|
|
|
_Nodeptr _Unchecked_erase(_Nodeptr _First, const _Nodeptr _Last) noexcept(_Nothrow_hash<_Traits, key_type>) {
|
|
if (_First == _Last) {
|
|
return _Last;
|
|
}
|
|
|
|
const auto _End = _List._Mypair._Myval2._Myhead;
|
|
const auto _Bucket_bounds = _Vec._Mypair._Myval2._Myfirst;
|
|
_Range_eraser _Eraser{_List, _First};
|
|
{
|
|
// process the first bucket, which is special because here _First might not be the beginning of the bucket
|
|
const auto _Predecessor = _First->_Prev;
|
|
const size_type _Bucket = bucket(_Traits::_Kfn(_Eraser._Next->_Myval)); // throws
|
|
// nothrow hereafter this block
|
|
_Nodeptr& _Bucket_lo = _Bucket_bounds[_Bucket << 1]._Ptr;
|
|
_Nodeptr& _Bucket_hi = _Bucket_bounds[(_Bucket << 1) + 1]._Ptr;
|
|
const bool _Update_lo = _Bucket_lo == _Eraser._Next;
|
|
const _Nodeptr _Old_hi = _Bucket_hi;
|
|
for (;;) { // remove elements until we hit the end of the bucket
|
|
const bool _At_bucket_back = _Eraser._Next == _Old_hi;
|
|
_Eraser._Bump_erased();
|
|
if (_At_bucket_back) {
|
|
break;
|
|
}
|
|
|
|
if (_Eraser._Next == _Last) {
|
|
if (_Update_lo) {
|
|
// erased the bucket's prefix
|
|
_Bucket_lo = _Eraser._Next;
|
|
}
|
|
|
|
return _Last;
|
|
}
|
|
}
|
|
|
|
if (_Update_lo) {
|
|
// emptied the bucket
|
|
_Bucket_lo = _End;
|
|
_Bucket_hi = _End;
|
|
} else {
|
|
_Bucket_hi = _Predecessor;
|
|
}
|
|
}
|
|
|
|
// hereafter we are always erasing buckets' prefixes
|
|
while (_Eraser._Next != _Last) {
|
|
const size_type _Bucket = bucket(_Traits::_Kfn(_Eraser._Next->_Myval)); // throws
|
|
// nothrow hereafter this block
|
|
_Nodeptr& _Bucket_lo = _Bucket_bounds[_Bucket << 1]._Ptr;
|
|
_Nodeptr& _Bucket_hi = _Bucket_bounds[(_Bucket << 1) + 1]._Ptr;
|
|
const _Nodeptr _Old_hi = _Bucket_hi;
|
|
for (;;) { // remove elements until we hit the end of the bucket
|
|
const bool _At_bucket_back = _Eraser._Next == _Old_hi;
|
|
_Eraser._Bump_erased();
|
|
if (_At_bucket_back) {
|
|
break;
|
|
}
|
|
|
|
if (_Eraser._Next == _Last) {
|
|
// erased the bucket's prefix
|
|
_Bucket_lo = _Eraser._Next;
|
|
return _Last;
|
|
}
|
|
}
|
|
|
|
// emptied the bucket
|
|
_Bucket_lo = _End;
|
|
_Bucket_hi = _End;
|
|
}
|
|
|
|
return _Last;
|
|
}
|
|
|
|
template <class _Kx>
|
|
static constexpr bool _Noexcept_heterogeneous_erasure() {
|
|
return _Nothrow_hash<_Traits, _Kx> //
|
|
&& (!_Multi || (_Nothrow_compare<_Traits, key_type, _Kx> && _Nothrow_compare<_Traits, _Kx, key_type>) );
|
|
}
|
|
|
|
template <class _Keytype>
|
|
size_type _Erase(const _Keytype& _Keyval) noexcept(_Noexcept_heterogeneous_erasure<_Keytype>()) /* strengthened */ {
|
|
const size_t _Hashval = _Traitsobj(_Keyval);
|
|
if constexpr (_Multi) {
|
|
const auto _Where = _Equal_range(_Keyval, _Hashval);
|
|
_Unchecked_erase(_Where._First._Ptr, _Where._Last._Ptr);
|
|
return _Where._Distance;
|
|
} else {
|
|
const auto _Target = _Find_last(_Keyval, _Hashval)._Duplicate;
|
|
if (_Target) {
|
|
_Erase_bucket(_Target, _Hashval & _Mask);
|
|
_List._Unchecked_erase(_Target);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public:
|
|
template <class _Iter = iterator, enable_if_t<!is_same_v<_Iter, const_iterator>, int> = 0>
|
|
iterator erase(iterator _Plist) noexcept(_Nothrow_hash<_Traits, key_type>) /* strengthened */ {
|
|
return _List._Make_iter(_Unchecked_erase(_Plist._Ptr));
|
|
}
|
|
|
|
iterator erase(const_iterator _Plist) noexcept(_Nothrow_hash<_Traits, key_type>) /* strengthened */ {
|
|
return _List._Make_iter(_Unchecked_erase(_Plist._Ptr));
|
|
}
|
|
|
|
iterator erase(const_iterator _First, const_iterator _Last) noexcept(
|
|
_Nothrow_hash<_Traits, key_type>) /* strengthened */ {
|
|
return _List._Make_iter(_Unchecked_erase(_First._Ptr, _Last._Ptr));
|
|
}
|
|
|
|
size_type erase(const key_type& _Keyval) noexcept(noexcept(_Erase(_Keyval))) /* strengthened */ {
|
|
return _Erase(_Keyval);
|
|
}
|
|
|
|
#if _HAS_CXX23
|
|
template <class _Kx, class _Mytraits = _Traits, class = typename _Mytraits::_Transparent,
|
|
enable_if_t<!disjunction_v<is_convertible<_Kx, const_iterator>, is_convertible<_Kx, iterator>>, int> = 0>
|
|
size_type erase(_Kx&& _Keyval) noexcept(noexcept(_Erase(_Keyval))) /* strengthened */ {
|
|
return _Erase(_Keyval);
|
|
}
|
|
#endif // _HAS_CXX23
|
|
|
|
void clear() noexcept {
|
|
// TRANSITION, ABI:
|
|
// LWG-2550 requires implementations to make clear() O(size()), independent of bucket_count().
|
|
// Unfortunately our current data structure / ABI does not allow achieving this in the general case because:
|
|
// (1) Finding the bucket that goes with an element requires running the hash function
|
|
// (2) The hash function operator() may throw exceptions, and
|
|
// (3) clear() is a noexcept function.
|
|
// We do comply with LWG-2550 if the hash function is noexcept, or if the container was empty.
|
|
const auto _Oldsize = _List._Mypair._Myval2._Mysize;
|
|
if (_Oldsize == 0) {
|
|
return;
|
|
}
|
|
|
|
if constexpr (_Nothrow_hash<_Traits, key_type>) {
|
|
// In testing, hash<size_t>{}(size_t{}) takes about 14 times as much time as assigning a pointer, or
|
|
// ~7-8 times as much as clearing a bucket. Therefore, if we would need to assign over more than 8 times
|
|
// as many buckets as elements, remove element-by-element.
|
|
if (bucket_count() / 8 > _Oldsize) {
|
|
const auto _Head = _List._Mypair._Myval2._Myhead;
|
|
_Unchecked_erase(_Head->_Next, _Head);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Bulk destroy items and reset buckets
|
|
_List.clear();
|
|
_STD fill(_Vec._Mypair._Myval2._Myfirst, _Vec._Mypair._Myval2._Mylast, _Unchecked_end());
|
|
}
|
|
|
|
private:
|
|
template <class _Keyty>
|
|
_NODISCARD _Nodeptr _Find_first(const _Keyty& _Keyval, const size_t _Hashval) const {
|
|
// find node pointer to first node matching _Keyval (with hash _Hashval) if it exists; otherwise, end
|
|
const size_type _Bucket = _Hashval & _Mask;
|
|
_Nodeptr _Where = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1]._Ptr;
|
|
const _Nodeptr _End = _List._Mypair._Myval2._Myhead;
|
|
if (_Where == _End) {
|
|
return _End;
|
|
}
|
|
|
|
const _Nodeptr _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1]._Ptr;
|
|
for (;;) {
|
|
if (!_Traitsobj(_Traits::_Kfn(_Where->_Myval), _Keyval)) {
|
|
if constexpr (!_Traits::_Standard) {
|
|
if (_Traitsobj(_Keyval, _Traits::_Kfn(_Where->_Myval))) {
|
|
return _End;
|
|
}
|
|
}
|
|
|
|
return _Where;
|
|
}
|
|
|
|
if (_Where == _Bucket_hi) {
|
|
return _End;
|
|
}
|
|
|
|
_Where = _Where->_Next;
|
|
}
|
|
}
|
|
|
|
template <class _Keyty>
|
|
_Nodeptr _Find(const _Keyty& _Keyval, const size_t _Hashval) const {
|
|
if constexpr (_Traits::_Multi) {
|
|
return _Find_first(_Keyval, _Hashval);
|
|
} else {
|
|
// use _Find_last for unique containers to avoid increase in code size of instantiating _Find_first
|
|
auto _Target = _Find_last(_Keyval, _Hashval)._Duplicate;
|
|
if (_Target) {
|
|
return _Target;
|
|
}
|
|
|
|
return _List._Mypair._Myval2._Myhead;
|
|
}
|
|
}
|
|
|
|
public:
|
|
template <class _Keyty = void>
|
|
_NODISCARD iterator find(typename _Traits::template _Deduce_key<_Keyty> _Keyval) {
|
|
return _List._Make_iter(_Find(_Keyval, _Traitsobj(_Keyval)));
|
|
}
|
|
|
|
template <class _Keyty = void>
|
|
_NODISCARD const_iterator find(typename _Traits::template _Deduce_key<_Keyty> _Keyval) const {
|
|
return _List._Make_const_iter(_Find(_Keyval, _Traitsobj(_Keyval)));
|
|
}
|
|
|
|
#if _HAS_CXX20
|
|
template <class _Keyty = void>
|
|
_NODISCARD bool contains(typename _Traits::template _Deduce_key<_Keyty> _Keyval) const {
|
|
return static_cast<bool>(_Find_last(_Keyval, _Traitsobj(_Keyval))._Duplicate);
|
|
}
|
|
#endif // _HAS_CXX20
|
|
|
|
template <class _Keyty = void>
|
|
_NODISCARD size_type count(typename _Traits::template _Deduce_key<_Keyty> _Keyval) const {
|
|
const size_t _Hashval = _Traitsobj(_Keyval);
|
|
if constexpr (_Multi) {
|
|
return _Equal_range(_Keyval, _Hashval)._Distance;
|
|
} else {
|
|
return static_cast<bool>(_Find_last(_Keyval, _Hashval)._Duplicate);
|
|
}
|
|
}
|
|
|
|
_DEPRECATE_STDEXT_HASH_LOWER_BOUND _NODISCARD iterator lower_bound(const key_type& _Keyval) {
|
|
return _List._Make_iter(_Find(_Keyval, _Traitsobj(_Keyval)));
|
|
}
|
|
|
|
_DEPRECATE_STDEXT_HASH_LOWER_BOUND _NODISCARD const_iterator lower_bound(const key_type& _Keyval) const {
|
|
return _List._Make_const_iter(_Find(_Keyval, _Traitsobj(_Keyval)));
|
|
}
|
|
|
|
_DEPRECATE_STDEXT_HASH_UPPER_BOUND _NODISCARD iterator upper_bound(const key_type& _Keyval) {
|
|
auto _Target = _Find_last(_Keyval, _Traitsobj(_Keyval))._Duplicate;
|
|
if (_Target) {
|
|
_Target = _Target->_Next;
|
|
} else {
|
|
_Target = _List._Mypair._Myval2._Myhead;
|
|
}
|
|
|
|
return _List._Make_iter(_Target);
|
|
}
|
|
|
|
_DEPRECATE_STDEXT_HASH_UPPER_BOUND _NODISCARD const_iterator upper_bound(const key_type& _Keyval) const {
|
|
auto _Target = _Find_last(_Keyval, _Traitsobj(_Keyval))._Duplicate;
|
|
if (_Target) {
|
|
_Target = _Target->_Next;
|
|
} else {
|
|
_Target = _List._Mypair._Myval2._Myhead;
|
|
}
|
|
|
|
return _List._Make_const_iter(_Target);
|
|
}
|
|
|
|
private:
|
|
struct _Equal_range_result {
|
|
_Unchecked_const_iterator _First;
|
|
_Unchecked_const_iterator _Last;
|
|
size_type _Distance;
|
|
};
|
|
|
|
template <class _Keyty>
|
|
_NODISCARD _Equal_range_result _Equal_range(const _Keyty& _Keyval, const size_t _Hashval) const
|
|
noexcept(_Nothrow_compare<_Traits, key_type, _Keyty>&& _Nothrow_compare<_Traits, _Keyty, key_type>) {
|
|
const size_type _Bucket = _Hashval & _Mask;
|
|
_Unchecked_const_iterator _Where = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1];
|
|
const _Unchecked_const_iterator _End = _Unchecked_end();
|
|
if (_Where == _End) {
|
|
return {_End, _End, 0};
|
|
}
|
|
|
|
const _Unchecked_const_iterator _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1];
|
|
for (; _Traitsobj(_Traits::_Kfn(*_Where), _Keyval); ++_Where) {
|
|
if (_Where == _Bucket_hi) {
|
|
return {_End, _End, 0};
|
|
}
|
|
}
|
|
|
|
if constexpr (!_Traits::_Standard) {
|
|
if (_Traitsobj(_Keyval, _Traits::_Kfn(*_Where))) {
|
|
return {_End, _End, 0};
|
|
}
|
|
}
|
|
|
|
const _Unchecked_const_iterator _First = _Where;
|
|
if constexpr (_Multi) {
|
|
size_type _Distance = 0;
|
|
for (;;) {
|
|
++_Distance;
|
|
|
|
const bool _At_bucket_end = _Where == _Bucket_hi;
|
|
++_Where;
|
|
if (_At_bucket_end) {
|
|
break;
|
|
}
|
|
|
|
if (_Traitsobj(_Keyval, _Traits::_Kfn(*_Where))) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return {_First, _Where, _Distance};
|
|
} else {
|
|
++_Where; // found the unique element
|
|
return {_First, _Where, 1};
|
|
}
|
|
}
|
|
|
|
public:
|
|
template <class _Keyty = void>
|
|
_NODISCARD pair<iterator, iterator> equal_range(typename _Traits::template _Deduce_key<_Keyty> _Keyval) {
|
|
const auto _Result = _Equal_range(_Keyval, _Traitsobj(_Keyval));
|
|
return {_List._Make_iter(_Result._First._Ptr), _List._Make_iter(_Result._Last._Ptr)};
|
|
}
|
|
|
|
template <class _Keyty = void>
|
|
_NODISCARD pair<const_iterator, const_iterator> equal_range(
|
|
typename _Traits::template _Deduce_key<_Keyty> _Keyval) const {
|
|
const auto _Result = _Equal_range(_Keyval, _Traitsobj(_Keyval));
|
|
return {_List._Make_const_iter(_Result._First._Ptr), _List._Make_const_iter(_Result._Last._Ptr)};
|
|
}
|
|
|
|
void swap(_Hash& _Right) noexcept(noexcept(_Traitsobj.swap(_Right._Traitsobj))) /* strengthened */ {
|
|
if (this != _STD addressof(_Right)) {
|
|
_Traitsobj.swap(_Right._Traitsobj);
|
|
_Pocs(_List._Getal(), _Right._List._Getal());
|
|
_Pocs(_Vec._Mypair._Get_first(), _Right._Vec._Mypair._Get_first());
|
|
_Swap_val(_Right);
|
|
}
|
|
}
|
|
|
|
#if _HAS_CXX17
|
|
using node_type = typename _Traits::node_type;
|
|
|
|
node_type extract(const const_iterator _Where) {
|
|
#if _ITERATOR_DEBUG_LEVEL == 2
|
|
const auto _List_data = _STD addressof(_List._Mypair._Myval2);
|
|
_STL_VERIFY(_Where._Getcont() == _List_data, "extract mismatched container");
|
|
_STL_VERIFY(_Where._Ptr != _List_data->_Myhead, "cannot extract end()");
|
|
#endif // _ITERATOR_DEBUG_LEVEL == 2
|
|
|
|
return node_type::_Make(_Extract(_Where._Unwrapped()), _List._Getal());
|
|
}
|
|
|
|
node_type extract(const key_type& _Keyval) {
|
|
const auto _Ptr = _Extract(_Keyval);
|
|
if (!_Ptr) {
|
|
return node_type{};
|
|
}
|
|
|
|
return node_type::_Make(_Ptr, _List._Getal());
|
|
}
|
|
|
|
#if _HAS_CXX23
|
|
template <class _Kx, class _Mytraits = _Traits, class = typename _Mytraits::_Transparent,
|
|
enable_if_t<!disjunction_v<is_convertible<_Kx, const_iterator>, is_convertible<_Kx, iterator>>, int> = 0>
|
|
node_type extract(_Kx&& _Keyval) {
|
|
const auto _Ptr = _Extract(_Keyval);
|
|
if (!_Ptr) {
|
|
return node_type{};
|
|
}
|
|
|
|
return node_type::_Make(_Ptr, _List._Getal());
|
|
}
|
|
#endif // _HAS_CXX23
|
|
|
|
iterator insert(const_iterator _Hint, node_type&& _Handle) {
|
|
if (_Handle.empty()) {
|
|
return end();
|
|
}
|
|
|
|
#if _ITERATOR_DEBUG_LEVEL == 2
|
|
_STL_VERIFY(_List.get_allocator() == _Handle._Getal(), "node handle allocator incompatible for insert");
|
|
#endif // _ITERATOR_DEBUG_LEVEL == 2
|
|
|
|
const auto& _Keyval = _Traits::_Kfn(_Handle._Getptr()->_Myval);
|
|
const size_t _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_hint(_Hint._Ptr, _Keyval, _Hashval);
|
|
if constexpr (!_Traits::_Multi) {
|
|
if (_Target._Duplicate) {
|
|
return _List._Make_iter(_Target._Duplicate);
|
|
}
|
|
}
|
|
|
|
_Check_max_size();
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_hint(_Hint._Ptr, _Keyval, _Hashval);
|
|
}
|
|
|
|
const auto _Released = _Handle._Release();
|
|
_Destroy_in_place(_Released->_Next); // TRANSITION, ABI
|
|
_Destroy_in_place(_Released->_Prev);
|
|
return _List._Make_iter(_Insert_new_node_before(_Hashval, _Target._Insert_before, _Released));
|
|
}
|
|
|
|
template <class>
|
|
friend class _Hash;
|
|
|
|
template <class _Other_traits>
|
|
void merge(_Hash<_Other_traits>& _That) { // transfer all nodes from _That into *this
|
|
static_assert(is_same_v<_Nodeptr, typename _Hash<_Other_traits>::_Nodeptr>,
|
|
"merge() requires an argument with a compatible node type.");
|
|
|
|
static_assert(is_same_v<allocator_type, typename _Hash<_Other_traits>::allocator_type>,
|
|
"merge() requires an argument with the same allocator type.");
|
|
|
|
if constexpr (is_same_v<_Hash, _Hash<_Other_traits>>) {
|
|
if (this == _STD addressof(_That)) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
#if _ITERATOR_DEBUG_LEVEL == 2
|
|
if constexpr (!_Alnode_traits::is_always_equal::value) {
|
|
_STL_VERIFY(_List._Getal() == _That._List._Getal(), "allocator incompatible for merge");
|
|
}
|
|
#endif // _ITERATOR_DEBUG_LEVEL == 2
|
|
|
|
auto _First = _That._Unchecked_begin();
|
|
const auto _Last = _That._Unchecked_end();
|
|
while (_First != _Last) {
|
|
const auto _Candidate = _First._Ptr;
|
|
++_First;
|
|
const auto& _Keyval = _Traits::_Kfn(_Candidate->_Myval);
|
|
const size_t _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_last(_Keyval, _Hashval);
|
|
if constexpr (!_Traits::_Multi) {
|
|
if (_Target._Duplicate) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
_Check_max_size();
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_last(_Keyval, _Hashval);
|
|
}
|
|
|
|
// nothrow hereafter this iteration
|
|
const auto _Source_bucket = _Hashval & _That._Mask;
|
|
_That._Erase_bucket(_Candidate, _Source_bucket);
|
|
_Candidate->_Prev->_Next = _Candidate->_Next;
|
|
_Candidate->_Next->_Prev = _Candidate->_Prev;
|
|
--_That._List._Mypair._Myval2._Mysize;
|
|
_Destroy_in_place(_Candidate->_Next); // TRANSITION, ABI
|
|
_Destroy_in_place(_Candidate->_Prev);
|
|
#if _ITERATOR_DEBUG_LEVEL == 2
|
|
_List._Mypair._Myval2._Adopt_unique(_That._List._Mypair._Myval2, _Candidate);
|
|
#endif // _ITERATOR_DEBUG_LEVEL == 2
|
|
(void) _Insert_new_node_before(_Hashval, _Target._Insert_before, _Candidate);
|
|
}
|
|
}
|
|
|
|
template <class _Other_traits>
|
|
void merge(_Hash<_Other_traits>&& _That) { // transfer all nodes from _That into *this
|
|
static_assert(is_same_v<_Nodeptr, typename _Hash<_Other_traits>::_Nodeptr>,
|
|
"merge() requires an argument with a compatible node type.");
|
|
|
|
static_assert(is_same_v<allocator_type, typename _Hash<_Other_traits>::allocator_type>,
|
|
"merge() requires an argument with the same allocator type.");
|
|
|
|
merge(_That);
|
|
}
|
|
|
|
protected:
|
|
_Nodeptr _Extract(const _Unchecked_const_iterator _Where) {
|
|
const size_type _Bucket = bucket(_Traits::_Kfn(*_Where));
|
|
_Erase_bucket(_Where._Ptr, _Bucket);
|
|
return _List._Mypair._Myval2._Unlinknode(_Where._Ptr);
|
|
}
|
|
|
|
template <class _Kx>
|
|
_Nodeptr _Extract(const _Kx& _Keyval) {
|
|
const size_t _Hashval = _Traitsobj(_Keyval);
|
|
_Nodeptr _Target;
|
|
if constexpr (_Traits::_Multi) {
|
|
_Target = _Find_first(_Keyval, _Hashval);
|
|
if (_Target == _List._Mypair._Myval2._Myhead) {
|
|
return _Nodeptr{};
|
|
}
|
|
} else {
|
|
_Target = _Find_last(_Keyval, _Hashval)._Duplicate;
|
|
if (_Target == nullptr) {
|
|
return _Nodeptr{};
|
|
}
|
|
}
|
|
|
|
_Erase_bucket(_Target, _Hashval & _Mask);
|
|
return _List._Mypair._Myval2._Unlinknode(_Target);
|
|
}
|
|
|
|
public:
|
|
conditional_t<_Traits::_Multi, iterator, _Insert_return_type<iterator, node_type>> insert(node_type&& _Handle) {
|
|
// insert the node (if any) held in _Handle
|
|
if (_Handle.empty()) {
|
|
if constexpr (_Traits::_Multi) {
|
|
return end();
|
|
} else {
|
|
return {end(), false, _STD move(_Handle)};
|
|
}
|
|
}
|
|
|
|
#if _ITERATOR_DEBUG_LEVEL == 2
|
|
_STL_VERIFY(_List.get_allocator() == _Handle._Getal(), "node handle allocator incompatible for insert");
|
|
#endif // _ITERATOR_DEBUG_LEVEL == 2
|
|
|
|
const auto& _Keyval = _Traits::_Kfn(_Handle._Getptr()->_Myval);
|
|
const size_t _Hashval = _Traitsobj(_Keyval);
|
|
auto _Target = _Find_last(_Keyval, _Hashval);
|
|
if constexpr (!_Traits::_Multi) {
|
|
if (_Target._Duplicate) {
|
|
return {_List._Make_iter(_Target._Duplicate), false, _STD move(_Handle)};
|
|
}
|
|
}
|
|
|
|
_Check_max_size();
|
|
if (_Check_rehash_required_1()) {
|
|
_Rehash_for_1();
|
|
_Target = _Find_last(_Keyval, _Hashval);
|
|
}
|
|
|
|
const auto _Released = _Handle._Release();
|
|
_Destroy_in_place(_Released->_Next); // TRANSITION, ABI
|
|
_Destroy_in_place(_Released->_Prev);
|
|
const auto _Newnode = _Insert_new_node_before(_Hashval, _Target._Insert_before, _Released);
|
|
if constexpr (_Traits::_Multi) {
|
|
return _List._Make_iter(_Newnode);
|
|
} else {
|
|
return {_List._Make_iter(_Newnode), true, node_type{}};
|
|
}
|
|
}
|
|
#endif // _HAS_CXX17
|
|
|
|
protected:
|
|
template <class _Keyty>
|
|
_NODISCARD _Hash_find_last_result<_Nodeptr> _Find_last(const _Keyty& _Keyval, const size_t _Hashval) const {
|
|
// find the insertion point for _Keyval and whether an element identical to _Keyval is already in the container
|
|
const size_type _Bucket = _Hashval & _Mask;
|
|
_Nodeptr _Where = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1]._Ptr;
|
|
const _Nodeptr _End = _List._Mypair._Myval2._Myhead;
|
|
if (_Where == _End) {
|
|
return {_End, _Nodeptr{}};
|
|
}
|
|
|
|
const _Nodeptr _Bucket_lo = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1]._Ptr;
|
|
for (;;) {
|
|
// Search backwards to maintain sorted [_Bucket_lo, _Bucket_hi] when !_Standard
|
|
if (!_Traitsobj(_Keyval, _Traits::_Kfn(_Where->_Myval))) {
|
|
if constexpr (!_Traits::_Standard) {
|
|
if (_Traitsobj(_Traits::_Kfn(_Where->_Myval), _Keyval)) {
|
|
return {_Where->_Next, _Nodeptr{}};
|
|
}
|
|
}
|
|
|
|
return {_Where->_Next, _Where};
|
|
}
|
|
|
|
if (_Where == _Bucket_lo) {
|
|
return {_Where, _Nodeptr{}};
|
|
}
|
|
|
|
_Where = _Where->_Prev;
|
|
}
|
|
}
|
|
|
|
template <class _Keyty>
|
|
_NODISCARD _Hash_find_last_result<_Nodeptr> _Find_hint(
|
|
const _Nodeptr _Hint, const _Keyty& _Keyval, const size_t _Hashval) const {
|
|
// if _Hint points to an element equivalent to _Keyval, returns _Hint; otherwise,
|
|
// returns _Find_last(_Keyval, _Hashval)
|
|
if (_Hint != _List._Mypair._Myval2._Myhead && !_Traitsobj(_Traits::_Kfn(_Hint->_Myval), _Keyval)) {
|
|
if constexpr (!_Traits::_Standard) {
|
|
if (_Traitsobj(_Keyval, _Traits::_Kfn(_Hint->_Myval))) {
|
|
return _Find_last(_Keyval, _Hashval);
|
|
}
|
|
}
|
|
|
|
return {_Hint->_Next, _Hint};
|
|
}
|
|
|
|
return _Find_last(_Keyval, _Hashval);
|
|
}
|
|
|
|
_Nodeptr _Insert_new_node_before(
|
|
const size_t _Hashval, const _Nodeptr _Insert_before, const _Nodeptr _Newnode) noexcept {
|
|
const _Nodeptr _Insert_after = _Insert_before->_Prev;
|
|
++_List._Mypair._Myval2._Mysize;
|
|
_Construct_in_place(_Newnode->_Next, _Insert_before);
|
|
_Construct_in_place(_Newnode->_Prev, _Insert_after);
|
|
_Insert_after->_Next = _Newnode;
|
|
_Insert_before->_Prev = _Newnode;
|
|
|
|
const auto _Head = _List._Mypair._Myval2._Myhead;
|
|
const auto _Bucket_array = _Vec._Mypair._Myval2._Myfirst;
|
|
const size_type _Bucket = _Hashval & _Mask;
|
|
_Unchecked_iterator& _Bucket_lo = _Bucket_array[_Bucket << 1];
|
|
_Unchecked_iterator& _Bucket_hi = _Bucket_array[(_Bucket << 1) + 1];
|
|
if (_Bucket_lo._Ptr == _Head) {
|
|
// bucket is empty, set both
|
|
_Bucket_lo._Ptr = _Newnode;
|
|
_Bucket_hi._Ptr = _Newnode;
|
|
} else if (_Bucket_lo._Ptr == _Insert_before) {
|
|
// new node is the lowest element in the bucket
|
|
_Bucket_lo._Ptr = _Newnode;
|
|
} else if (_Bucket_hi._Ptr == _Insert_after) {
|
|
// new node is the highest element in the bucket
|
|
_Bucket_hi._Ptr = _Newnode;
|
|
}
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Stl_internal_check_container_invariants();
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
return _Newnode;
|
|
}
|
|
|
|
void _Check_max_size() const {
|
|
const size_type _Oldsize = _List._Mypair._Myval2._Mysize;
|
|
if (_Oldsize == _List.max_size()) {
|
|
_Xlength_error("unordered_map/set too long");
|
|
}
|
|
}
|
|
|
|
bool _Check_rehash_required_1() const noexcept {
|
|
const size_type _Oldsize = _List._Mypair._Myval2._Mysize;
|
|
const auto _Newsize = _Oldsize + 1;
|
|
return max_load_factor() < static_cast<float>(_Newsize) / static_cast<float>(bucket_count());
|
|
}
|
|
|
|
void _Rehash_for_1() {
|
|
const auto _Oldsize = _List._Mypair._Myval2._Mysize;
|
|
const auto _Newsize = _Oldsize + 1;
|
|
_Forced_rehash(_Desired_grow_bucket_count(_Newsize));
|
|
}
|
|
|
|
void _Erase_bucket(_Nodeptr _Plist, size_type _Bucket) noexcept {
|
|
// remove the node _Plist from its bucket
|
|
_Nodeptr& _Bucket_lo = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1]._Ptr;
|
|
_Nodeptr& _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1]._Ptr;
|
|
if (_Bucket_hi == _Plist) {
|
|
if (_Bucket_lo == _Plist) { // make bucket empty
|
|
const auto _End = _List._Mypair._Myval2._Myhead;
|
|
|
|
_Bucket_lo = _End;
|
|
_Bucket_hi = _End;
|
|
} else {
|
|
_Bucket_hi = _Plist->_Prev; // move end back one element
|
|
}
|
|
} else if (_Bucket_lo == _Plist) {
|
|
_Bucket_lo = _Plist->_Next; // move beginning up one element
|
|
}
|
|
}
|
|
|
|
_NODISCARD size_type _Min_load_factor_buckets(const size_type _For_size) const noexcept {
|
|
// returns the minimum number of buckets necessary for the elements in _List
|
|
return static_cast<size_type>(_CSTD ceilf(static_cast<float>(_For_size) / max_load_factor()));
|
|
}
|
|
|
|
_NODISCARD size_type _Desired_grow_bucket_count(const size_type _For_size) const noexcept {
|
|
const size_type _Old_buckets = bucket_count();
|
|
const size_type _Req_buckets = (_STD max)(_Min_buckets, _Min_load_factor_buckets(_For_size));
|
|
if (_Old_buckets >= _Req_buckets) {
|
|
// we already have enough buckets so there's no need to change the count
|
|
return _Old_buckets;
|
|
}
|
|
|
|
if (_Old_buckets < 512 && _Old_buckets * 8 >= _Req_buckets) {
|
|
// if we are changing the bucket count and have less than 512 buckets, use 8x more buckets
|
|
return _Old_buckets * 8;
|
|
}
|
|
|
|
// power of 2 invariant means this will result in at least 2*_Old_buckets after round up in _Forced_rehash
|
|
return _Req_buckets;
|
|
}
|
|
|
|
void _Reinsert_with_invalid_vec() { // insert elements in [begin(), end()), distrusting existing _Vec elements
|
|
_Forced_rehash(_Desired_grow_bucket_count(_List.size()));
|
|
}
|
|
|
|
void _Forced_rehash(size_type _Buckets) {
|
|
// Force rehash of elements in _List, distrusting existing bucket assignments in _Vec.
|
|
// Assumes _Buckets is greater than _Min_buckets, and that changing to that many buckets doesn't violate
|
|
// load_factor() <= max_load_factor().
|
|
|
|
// Don't violate power of 2, fits in half the bucket vector invariant:
|
|
// (we assume because vector must use single allocations; as a result, its max_size fits in a size_t)
|
|
const unsigned long _Max_storage_buckets_log2 = _Floor_of_log_2(static_cast<size_t>(_Vec.max_size() >> 1));
|
|
const auto _Max_storage_buckets = static_cast<size_type>(1) << _Max_storage_buckets_log2;
|
|
if (_Buckets > _Max_storage_buckets) {
|
|
_Xlength_error("invalid hash bucket count");
|
|
}
|
|
|
|
// The above test also means that we won't perform a forbidden full shift when restoring the power of
|
|
// 2 invariant
|
|
// this round up to power of 2 in addition to the _Buckets > _Maxidx above means
|
|
// we'll at least double in size (the next power of 2 above _Maxidx)
|
|
_Buckets = static_cast<size_type>(1) << _Ceiling_of_log_2(static_cast<size_t>(_Buckets));
|
|
const _Unchecked_iterator _End = _Unchecked_end();
|
|
|
|
_Vec._Assign_grow(_Buckets << 1, _End);
|
|
_Mask = _Buckets - 1;
|
|
_Maxidx = _Buckets;
|
|
|
|
_Clear_guard _Guard{this};
|
|
|
|
_Unchecked_iterator _Inserted = _Unchecked_begin();
|
|
|
|
// Remember the next _Inserted value as splices will change _Inserted's position arbitrarily.
|
|
for (_Unchecked_iterator _Next_inserted = _Inserted; _Inserted != _End; _Inserted = _Next_inserted) {
|
|
++_Next_inserted;
|
|
|
|
auto& _Inserted_key = _Traits::_Kfn(*_Inserted);
|
|
const size_type _Bucket = bucket(_Inserted_key);
|
|
|
|
// _Bucket_lo and _Bucket_hi are the *inclusive* range of elements in the bucket, or _Unchecked_end() if
|
|
// the bucket is empty; if !_Standard then [_Bucket_lo, _Bucket_hi] is a sorted range.
|
|
_Unchecked_iterator& _Bucket_lo = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1];
|
|
_Unchecked_iterator& _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1];
|
|
|
|
if (_Bucket_lo == _End) {
|
|
// The bucket was empty, set it to the inserted element.
|
|
_Bucket_lo = _Inserted;
|
|
_Bucket_hi = _Inserted;
|
|
continue;
|
|
}
|
|
|
|
// Search the bucket for the insertion location and move element if necessary.
|
|
_Unchecked_const_iterator _Insert_before = _Bucket_hi;
|
|
if (!_Traitsobj(_Inserted_key, _Traits::_Kfn(*_Insert_before))) {
|
|
// The inserted element belongs at the end of the bucket; splice it there and set _Bucket_hi to the
|
|
// new bucket inclusive end.
|
|
++_Insert_before;
|
|
if (_Insert_before != _Inserted) { // avoid splice on element already in position
|
|
_Mylist::_Scary_val::_Unchecked_splice(_Insert_before._Ptr, _Inserted._Ptr, _Next_inserted._Ptr);
|
|
}
|
|
|
|
_Bucket_hi = _Inserted;
|
|
continue;
|
|
}
|
|
|
|
// The insertion point isn't *_Bucket_hi, so search [_Bucket_lo, _Bucket_hi) for insertion point; we
|
|
// go backwards to maintain sortedness when !_Standard.
|
|
for (;;) {
|
|
if (_Bucket_lo == _Insert_before) {
|
|
// There are no equivalent keys in the bucket, so insert it at the beginning.
|
|
// Element can't be already in position here because:
|
|
// * (for !_Standard) _Inserted_key < *_Insert_before or
|
|
// * (for _Standard) _Inserted_key != *_Insert_before
|
|
_Mylist::_Scary_val::_Unchecked_splice(_Insert_before._Ptr, _Inserted._Ptr, _Next_inserted._Ptr);
|
|
_Bucket_lo = _Inserted;
|
|
break;
|
|
}
|
|
|
|
if (!_Traitsobj(_Inserted_key, _Traits::_Kfn(*--_Insert_before))) {
|
|
// Found insertion point, move the element here, bucket bounds are already okay.
|
|
++_Insert_before;
|
|
// Element can't be already in position here because all elements we're inserting are after all
|
|
// the elements already in buckets, and *_Insert_before isn't the highest element in the bucket.
|
|
_Mylist::_Scary_val::_Unchecked_splice(_Insert_before._Ptr, _Inserted._Ptr, _Next_inserted._Ptr);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
_Guard._Target = nullptr;
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
_Stl_internal_check_container_invariants();
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
}
|
|
|
|
float& _Max_bucket_size() noexcept {
|
|
return _Traitsobj._Get_max_bucket_size();
|
|
}
|
|
|
|
const float& _Max_bucket_size() const noexcept {
|
|
return _Traitsobj._Get_max_bucket_size();
|
|
}
|
|
|
|
_Alnode& _Getal() noexcept {
|
|
return _List._Getal();
|
|
}
|
|
|
|
const _Alnode& _Getal() const noexcept {
|
|
return _List._Getal();
|
|
}
|
|
|
|
struct _Multi_equal_check_result {
|
|
bool _Equal_possible = false;
|
|
_Unchecked_const_iterator _Subsequent_first{}; // only useful if _Equal_possible
|
|
};
|
|
|
|
_NODISCARD _Multi_equal_check_result _Multi_equal_check_equal_range(
|
|
const _Hash& _Right, _Unchecked_const_iterator _First1) const {
|
|
// check that an equal_range of elements starting with *_First1 are a permutation of the corresponding
|
|
// equal_range of elements in _Right
|
|
auto& _Keyval = _Traits::_Kfn(*_First1);
|
|
// find the start of the matching run in the other container
|
|
const size_t _Hashval = _Right._Traitsobj(_Keyval);
|
|
const size_type _Bucket = _Hashval & _Right._Mask;
|
|
auto _First2 = _Right._Vec._Mypair._Myval2._Myfirst[_Bucket << 1];
|
|
if (_First2 == _Right._Unchecked_end()) {
|
|
// no matching bucket, therefore no matching run
|
|
return {};
|
|
}
|
|
|
|
const auto _Bucket_hi = _Right._Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1];
|
|
for (; _Right._Traitsobj(_Traits::_Kfn(*_First2), _Keyval); ++_First2) {
|
|
// find first matching element in _Right
|
|
if (_First2 == _Bucket_hi) {
|
|
return {};
|
|
}
|
|
}
|
|
|
|
_Unchecked_const_iterator _Left_stop_at;
|
|
if constexpr (_Traits::_Standard) {
|
|
_Left_stop_at = _Unchecked_end();
|
|
} else {
|
|
// check the first elements for equivalence when !_Standard
|
|
if (_Right._Traitsobj(_Keyval, _Traits::_Kfn(*_First2))) {
|
|
return {};
|
|
}
|
|
|
|
const size_t _LHashval = _Traitsobj(_Keyval);
|
|
const size_type _LBucket = _LHashval & _Mask;
|
|
const auto _LBucket_hi = _Vec._Mypair._Myval2._Myfirst[(_LBucket << 1) + 1];
|
|
_Left_stop_at = _LBucket_hi;
|
|
++_Left_stop_at;
|
|
}
|
|
|
|
// trim matching prefixes
|
|
while (*_First1 == *_First2) {
|
|
// the right equal_range ends at the end of the bucket or on the first nonequal element
|
|
bool _Right_range_end = _First2 == _Bucket_hi;
|
|
++_First2;
|
|
if (!_Right_range_end) {
|
|
_Right_range_end = _Right._Traitsobj(_Keyval, _Traits::_Kfn(*_First2));
|
|
}
|
|
|
|
// the left equal_range ends at the end of the container or on the first nonequal element
|
|
++_First1;
|
|
const bool _Left_range_end = _First1 == _Left_stop_at || _Traitsobj(_Keyval, _Traits::_Kfn(*_First1));
|
|
|
|
if (_Left_range_end && _Right_range_end) {
|
|
// the equal_ranges were completely equal
|
|
return {true, _First1};
|
|
}
|
|
|
|
if (_Left_range_end || _Right_range_end) {
|
|
// one equal_range is a prefix of the other; not equal
|
|
return {};
|
|
}
|
|
}
|
|
|
|
// found a mismatched element, find the end of the equal_ranges and dispatch to _Check_match_counts
|
|
auto _Last1 = _First1;
|
|
auto _Last2 = _First2;
|
|
for (;;) {
|
|
bool _Right_range_end = _Last2 == _Bucket_hi;
|
|
++_Last2;
|
|
if (!_Right_range_end) {
|
|
_Right_range_end = _Right._Traitsobj(_Keyval, _Traits::_Kfn(*_Last2));
|
|
}
|
|
|
|
++_Last1;
|
|
const bool _Left_range_end = _Last1 == _Left_stop_at || _Traitsobj(_Keyval, _Traits::_Kfn(*_Last1));
|
|
|
|
if (_Left_range_end && _Right_range_end) {
|
|
// equal_ranges had the same length, check for permutation
|
|
return {_Check_match_counts(_First1, _Last1, _First2, _Last2, equal_to<>{}), _Last1};
|
|
}
|
|
|
|
if (_Left_range_end || _Right_range_end) {
|
|
// different number of elements in the range, not a permutation
|
|
return {};
|
|
}
|
|
}
|
|
}
|
|
|
|
template <bool _Multi2 = _Traits::_Multi, enable_if_t<_Multi2, int> = 0>
|
|
_NODISCARD bool _Multi_equal(const _Hash& _Right) const {
|
|
static_assert(_Traits::_Multi, "This function only works with multi containers");
|
|
_STL_INTERNAL_CHECK(this->size() == _Right.size());
|
|
const auto _Last1 = _Unchecked_end();
|
|
auto _First1 = _Unchecked_begin();
|
|
while (_First1 != _Last1) {
|
|
const auto _Result = _Multi_equal_check_equal_range(_Right, _First1);
|
|
if (!_Result._Equal_possible) {
|
|
return false;
|
|
}
|
|
|
|
_First1 = _Result._Subsequent_first;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
#ifdef _ENABLE_STL_INTERNAL_CHECK
|
|
public:
|
|
void _Stl_internal_check_container_invariants() const noexcept {
|
|
const size_type _Vecsize = _Vec.size();
|
|
_STL_INTERNAL_CHECK(_Vec._Mypair._Myval2._Mylast == _Vec._Mypair._Myval2._Myend);
|
|
_STL_INTERNAL_CHECK(_Vecsize >= _Min_buckets * 2);
|
|
_STL_INTERNAL_CHECK(_Maxidx == (_Vecsize >> 1));
|
|
_STL_INTERNAL_CHECK(_Maxidx - 1 == _Mask);
|
|
_STL_INTERNAL_CHECK(_Maxidx >= _Min_load_factor_buckets(_List.size()));
|
|
// asserts that bucket count is a power of 2:
|
|
_STL_INTERNAL_CHECK((static_cast<size_type>(1) << _Floor_of_log_2(_Vecsize)) == _Vecsize);
|
|
_STL_INTERNAL_CHECK(load_factor() <= max_load_factor());
|
|
// In the test that counts number of allocator copies, avoid an extra rebind that would incorrectly count as
|
|
// a copy; otherwise, allow allocators that support only homogeneous compare.
|
|
#ifdef _USE_HETEROGENEOUS_ALLOCATOR_COMPARE_IN_INTERNAL_CHECK
|
|
_STL_INTERNAL_CHECK(_List._Getal() == _Vec._Mypair._Get_first());
|
|
#else // _USE_HETEROGENEOUS_ALLOCATOR_COMPARE_IN_INTERNAL_CHECK
|
|
_STL_INTERNAL_CHECK(static_cast<_Aliter>(_List._Getal()) == _Vec._Mypair._Get_first());
|
|
#endif // _USE_HETEROGENEOUS_ALLOCATOR_COMPARE_IN_INTERNAL_CHECK
|
|
#ifdef _STL_INTERNAL_CHECK_EXHAUSTIVE
|
|
size_type _Elements = 0;
|
|
const auto _End = _Unchecked_end();
|
|
for (size_type _Bucket = 0; _Bucket < _Maxidx; ++_Bucket) {
|
|
_Unchecked_const_iterator _Where = _Vec._Mypair._Myval2._Myfirst[_Bucket << 1];
|
|
const _Unchecked_const_iterator _Bucket_hi = _Vec._Mypair._Myval2._Myfirst[(_Bucket << 1) + 1];
|
|
if (_Where != _End) {
|
|
// check that the bucket is sorted for legacy hash_meow:
|
|
if constexpr (!_Traits::_Standard) {
|
|
if (_Where != _Bucket_hi) {
|
|
auto _SFirst = _Where;
|
|
auto _SNext = _Where;
|
|
for (;;) {
|
|
++_SNext;
|
|
if constexpr (_Traits::_Multi) {
|
|
_STL_INTERNAL_CHECK(!_Traitsobj(_Traits::_Kfn(*_SNext), _Traits::_Kfn(*_SFirst)));
|
|
} else {
|
|
_STL_INTERNAL_CHECK(_Traitsobj(_Traits::_Kfn(*_SFirst), _Traits::_Kfn(*_SNext)));
|
|
}
|
|
|
|
if (_SNext == _Bucket_hi) {
|
|
break;
|
|
}
|
|
|
|
_SFirst = _SNext;
|
|
}
|
|
}
|
|
}
|
|
// check that all the elements in the bucket belong in the bucket:
|
|
for (;;) {
|
|
++_Elements;
|
|
_STL_INTERNAL_CHECK(bucket(_Traits::_Kfn(*_Where)) == _Bucket);
|
|
if (_Where == _Bucket_hi) {
|
|
break;
|
|
}
|
|
|
|
++_Where;
|
|
}
|
|
}
|
|
}
|
|
|
|
_STL_INTERNAL_CHECK(_List.size() == _Elements);
|
|
#endif // _STL_INTERNAL_CHECK_EXHAUSTIVE
|
|
}
|
|
|
|
protected:
|
|
#endif // _ENABLE_STL_INTERNAL_CHECK
|
|
|
|
_Traits _Traitsobj; // traits to customize behavior
|
|
_Mylist _List; // list of elements, must initialize before _Vec
|
|
_Hash_vec<_Aliter> _Vec; // "vector" of list iterators for buckets:
|
|
// each bucket is 2 iterators denoting the closed range of elements in the bucket,
|
|
// or both iterators set to _Unchecked_end() if the bucket is empty.
|
|
size_type _Mask; // the key mask
|
|
size_type _Maxidx; // current maximum key value, must be a power of 2
|
|
};
|
|
|
|
#if _HAS_CXX17
|
|
// For constraining deduction guides (N4892 [unord.req]/18.3)
|
|
template <class _Hasher>
|
|
using _Is_hasher = negation<disjunction<is_integral<_Hasher>, _Is_allocator<_Hasher>>>;
|
|
#endif // _HAS_CXX17
|
|
|
|
template <class _Traits>
|
|
_NODISCARD bool _Hash_equal(const _Hash<_Traits>& _Left, const _Hash<_Traits>& _Right) {
|
|
if (_Left.size() != _Right.size()) {
|
|
return false;
|
|
}
|
|
|
|
if constexpr (_Traits::_Multi) {
|
|
return _Left._Multi_equal(_Right);
|
|
} else {
|
|
for (const auto& _LVal : _Left) {
|
|
// look for element with equivalent key
|
|
const auto& _Keyval = _Traits::_Kfn(_LVal);
|
|
const auto _Next2 = _Right._Find_last(_Keyval, _Right._Traitsobj(_Keyval))._Duplicate;
|
|
if (!(static_cast<bool>(_Next2) && _Traits::_Nonkfn(_LVal) == _Traits::_Nonkfn(_Next2->_Myval))) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
_STD_END
|
|
|
|
#pragma pop_macro("new")
|
|
_STL_RESTORE_CLANG_WARNINGS
|
|
#pragma warning(pop)
|
|
#pragma pack(pop)
|
|
#endif // _STL_COMPILER_PREPROCESSOR
|
|
#endif // _XHASH_
|