зеркало из https://github.com/microsoft/STL.git
216 строки
6.9 KiB
C++
216 строки
6.9 KiB
C++
// xnode_handle.h internal header
|
|
|
|
// Copyright (c) Microsoft Corporation.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
#pragma once
|
|
#ifndef _XNODE_HANDLE_H
|
|
#define _XNODE_HANDLE_H
|
|
#include <yvals_core.h>
|
|
#if _STL_COMPILER_PREPROCESSOR
|
|
#include <xmemory>
|
|
|
|
#if !_HAS_CXX17
|
|
#error Node handles are only available with C++17. (Also, you should not include this internal header.)
|
|
#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
|
|
_STD_BEGIN
|
|
|
|
// STRUCT TEMPLATE _Insert_return_type
|
|
template <class _Iterator, class _NodeType>
|
|
struct _Insert_return_type {
|
|
_Iterator position;
|
|
bool inserted;
|
|
_NodeType node;
|
|
};
|
|
|
|
// CLASS TEMPLATE _Node_handle
|
|
template <class _Derived_type, class _KeyTy, class _ValueTy>
|
|
struct _Node_handle_map_base { // map-specific node handle behavior
|
|
using key_type = _KeyTy;
|
|
using mapped_type = _ValueTy;
|
|
|
|
key_type& key() const noexcept /* strengthened */ {
|
|
return _Datum().first;
|
|
}
|
|
mapped_type& mapped() const noexcept /* strengthened */ {
|
|
return _Datum().second;
|
|
}
|
|
|
|
private:
|
|
using _Pair_type = pair<key_type, mapped_type>;
|
|
_Pair_type& _Datum() const {
|
|
const auto& _Self = static_cast<const _Derived_type&>(*this);
|
|
auto& _Data = _Self._Getptr()->_Myval;
|
|
// NB: Node handle requires formally-undefined behavior *somewhere*, this
|
|
// implementation chooses to type-pun pair<const K, T> as pair<K, T>.
|
|
return reinterpret_cast<_Pair_type&>(_Data);
|
|
}
|
|
};
|
|
|
|
template <class _Derived_type, class _ValueTy>
|
|
struct _Node_handle_set_base { // set-specific node handle behavior
|
|
using value_type = _ValueTy;
|
|
|
|
value_type& value() const noexcept /* strengthened */ {
|
|
const auto& _Self = static_cast<const _Derived_type&>(*this);
|
|
return _Self._Getptr()->_Myval;
|
|
}
|
|
};
|
|
|
|
template <class _Node, class _Alloc, template <class...> class _Base, class... _Types>
|
|
class _Node_handle : public _Base<_Node_handle<_Node, _Alloc, _Base, _Types...>, _Types...> {
|
|
// storage for a node from one of the node-based standard containers
|
|
public:
|
|
using allocator_type = _Alloc;
|
|
|
|
private:
|
|
using _Alty_traits = allocator_traits<_Alloc>;
|
|
using _Alnode = _Rebind_alloc_t<_Alloc, _Node>;
|
|
using _Alnode_traits = allocator_traits<_Alnode>;
|
|
using _Nodeptr = typename _Alnode_traits::pointer;
|
|
|
|
_Nodeptr _Ptr{};
|
|
aligned_union_t<0, _Alloc> _Alloc_storage; // Invariant: contains a live _Alloc iff _Ptr != nullptr
|
|
|
|
void _Clear() noexcept { // destroy any contained node and return to the empty state
|
|
if (_Ptr) {
|
|
_Alloc& _Al = _Getal();
|
|
_Alnode _Node_alloc{_Al};
|
|
_Alnode_traits::destroy(_Node_alloc, _STD addressof(_Ptr->_Myval));
|
|
_Node::_Freenode0(_Node_alloc, _Ptr);
|
|
_Destroy_in_place(_Al);
|
|
_Ptr = nullptr;
|
|
}
|
|
}
|
|
|
|
_Node_handle(const _Nodeptr _My_ptr, const _Alloc& _My_alloc) noexcept
|
|
: _Ptr{_My_ptr} { // Initialize a _Node_handle that holds the specified node
|
|
// pre: _My_ptr != nullptr
|
|
// pre: _Alloc can release _Ptr
|
|
_Construct_in_place(_Getal(), _My_alloc);
|
|
}
|
|
|
|
public:
|
|
constexpr _Node_handle() noexcept : _Alloc_storage{} {}
|
|
|
|
~_Node_handle() noexcept {
|
|
_Clear();
|
|
}
|
|
|
|
_Node_handle(_Node_handle&& _That) noexcept : _Ptr{_That._Ptr} { // steal node and allocator (if any) from _That
|
|
if (_Ptr) {
|
|
_That._Ptr = nullptr;
|
|
_Alloc& _That_al = _That._Getal();
|
|
_Construct_in_place(_Getal(), _STD move(_That_al));
|
|
_Destroy_in_place(_That_al);
|
|
}
|
|
}
|
|
|
|
_Node_handle& operator=(_Node_handle&& _That) noexcept /* strengthened */ {
|
|
// steal state from _That
|
|
if (!_Ptr) {
|
|
if (_That._Ptr) {
|
|
_Alloc& _That_al = _That._Getal();
|
|
_Construct_in_place(_Getal(), _STD move(_That_al));
|
|
_Destroy_in_place(_That_al);
|
|
_Ptr = _STD exchange(_That._Ptr, nullptr);
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
if (!_That._Ptr || this == _STD addressof(_That)) {
|
|
_Clear();
|
|
return *this;
|
|
}
|
|
|
|
_Alloc& _Al = _Getal();
|
|
_Alnode _Node_alloc{_Al};
|
|
_Alnode_traits::destroy(_Node_alloc, _STD addressof(_Ptr->_Myval));
|
|
_Alnode_traits::deallocate(_Node_alloc, _Ptr, 1);
|
|
|
|
_Alloc& _That_al = _That._Getal();
|
|
_Pocma(_Al, _That_al);
|
|
_Destroy_in_place(_That_al);
|
|
|
|
_Ptr = _STD exchange(_That._Ptr, nullptr);
|
|
return *this;
|
|
}
|
|
|
|
_Nodeptr _Getptr() const noexcept {
|
|
return _Ptr;
|
|
}
|
|
|
|
_Alloc& _Getal() noexcept { // pre: !empty()
|
|
return reinterpret_cast<_Alloc&>(_Alloc_storage);
|
|
}
|
|
const _Alloc& _Getal() const noexcept { // pre: !empty()
|
|
return reinterpret_cast<const _Alloc&>(_Alloc_storage);
|
|
}
|
|
|
|
_NODISCARD allocator_type get_allocator() const noexcept /* strengthened */ {
|
|
// pre: !empty()
|
|
return _Getal();
|
|
}
|
|
|
|
explicit operator bool() const noexcept {
|
|
return _Ptr != nullptr;
|
|
}
|
|
|
|
_NODISCARD bool empty() const noexcept {
|
|
return _Ptr == nullptr;
|
|
}
|
|
|
|
_Nodeptr _Release() noexcept { // extract the node from *this
|
|
// pre: !empty()
|
|
_Destroy_in_place(_Getal());
|
|
return _STD exchange(_Ptr, nullptr);
|
|
}
|
|
|
|
void swap(_Node_handle& _That) noexcept /* strengthened */ {
|
|
if (_Ptr) {
|
|
if (_That._Ptr) {
|
|
_Pocs(_Getal(), _That._Getal());
|
|
} else {
|
|
_Alloc& _Al = _Getal();
|
|
_Construct_in_place(_That._Getal(), _STD move(_Al));
|
|
_Destroy_in_place(_Al);
|
|
}
|
|
} else {
|
|
if (!_That._Ptr) {
|
|
return;
|
|
}
|
|
|
|
_Alloc& _That_al = _That._Getal();
|
|
_Construct_in_place(_Getal(), _STD move(_That_al));
|
|
_Destroy_in_place(_That_al);
|
|
}
|
|
_Swap_adl(_Ptr, _That._Ptr);
|
|
}
|
|
friend void swap(_Node_handle& _Left, _Node_handle& _Right) noexcept /* strengthened */ {
|
|
_Left.swap(_Right);
|
|
}
|
|
|
|
static _Node_handle _Make(const _Nodeptr _Ptr, const allocator_type& _Al) {
|
|
// initialize a _Node_handle that holds _Ptr and _Al
|
|
// pre: _Ptr != nullptr
|
|
// pre: _Al can release _Ptr
|
|
return _Node_handle{_Ptr, _Al};
|
|
}
|
|
};
|
|
|
|
_STD_END
|
|
#pragma pop_macro("new")
|
|
_STL_RESTORE_CLANG_WARNINGS
|
|
#pragma warning(pop)
|
|
#pragma pack(pop)
|
|
#endif // _STL_COMPILER_PREPROCESSOR
|
|
#endif // _XNODE_HANDLE_H
|