// Safe container implementation -*- C++ -*- // Copyright (C) 2014-2024 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // . /** @file debug/safe_container.h * This file is a GNU debug extension to the Standard C++ Library. */ #ifndef _GLIBCXX_DEBUG_SAFE_CONTAINER_H #define _GLIBCXX_DEBUG_SAFE_CONTAINER_H 1 #include namespace __gnu_debug { /// Safe class dealing with some allocator dependent operations. template class _SafeBase, bool _IsCxx11AllocatorAware = true> class _Safe_container : public _SafeBase<_SafeContainer> { typedef _SafeBase<_SafeContainer> _Base; _GLIBCXX20_CONSTEXPR _SafeContainer& _M_cont() _GLIBCXX_NOEXCEPT { return *static_cast<_SafeContainer*>(this); } protected: #if __cplusplus >= 201103L _Safe_container() = default; _Safe_container(const _Safe_container&) = default; _Safe_container(_Safe_container&&) = default; private: _GLIBCXX20_CONSTEXPR _Safe_container(_Safe_container&& __x, const _Alloc&, std::true_type) : _Safe_container(std::move(__x)) { } _GLIBCXX20_CONSTEXPR _Safe_container(_Safe_container&& __x, const _Alloc& __a, std::false_type) : _Safe_container() { if (__x._M_cont().get_allocator() == __a) _Base::_M_swap(__x); else if (!std::__is_constant_evaluated()) __x._M_invalidate_all(); } protected: _GLIBCXX20_CONSTEXPR _Safe_container(_Safe_container&& __x, const _Alloc& __a) : _Safe_container(std::move(__x), __a, typename std::allocator_traits<_Alloc>::is_always_equal{}) { } #endif // Copy assignment invalidate all iterators. _GLIBCXX20_CONSTEXPR _Safe_container& operator=(const _Safe_container&) _GLIBCXX_NOEXCEPT { if (!std::__is_constant_evaluated()) this->_M_invalidate_all(); return *this; } #if __cplusplus >= 201103L _GLIBCXX20_CONSTEXPR _Safe_container& operator=(_Safe_container&& __x) noexcept { if (std::__is_constant_evaluated()) return *this; if (std::__addressof(__x) == this) { // Standard containers have a valid but unspecified value after // self-move, so we invalidate all debug iterators even if the // underlying container happens to preserve its contents. this->_M_invalidate_all(); return *this; } if (_IsCxx11AllocatorAware) { typedef __gnu_cxx::__alloc_traits<_Alloc> _Alloc_traits; bool __xfer_memory = _Alloc_traits::_S_propagate_on_move_assign() || _M_cont().get_allocator() == __x._M_cont().get_allocator(); if (__xfer_memory) _Base::_M_swap(__x); else this->_M_invalidate_all(); } else _Base::_M_swap(__x); __x._M_invalidate_all(); return *this; } _GLIBCXX20_CONSTEXPR void _M_swap(_Safe_container& __x) noexcept { if (_IsCxx11AllocatorAware) { typedef __gnu_cxx::__alloc_traits<_Alloc> _Alloc_traits; if (!_Alloc_traits::_S_propagate_on_swap()) __glibcxx_check_equal_allocs(this->_M_cont()._M_base(), __x._M_cont()._M_base()); } _Base::_M_swap(__x); } #endif }; } // namespace __gnu_debug #endif