1f554add5SHoward Hinnant// -*- C++ -*- 2eb8650a7SLouis Dionne//===----------------------------------------------------------------------===// 3f554add5SHoward Hinnant// 457b08b09SChandler Carruth// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 557b08b09SChandler Carruth// See https://llvm.org/LICENSE.txt for license information. 657b08b09SChandler Carruth// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 7f554add5SHoward Hinnant// 8f554add5SHoward Hinnant//===----------------------------------------------------------------------===// 9f554add5SHoward Hinnant 10cc82a1b0SLouis Dionne#ifndef _LIBCPP___DEBUG 11cc82a1b0SLouis Dionne#define _LIBCPP___DEBUG 12f554add5SHoward Hinnant 13f87aa19bSLouis Dionne#include <__assert> 14c1bd9197SEric Fiselier#include <__config> 15*23d6cde1SLouis Dionne#include <cstddef> 16e3cf7050SNikolas Klauser#include <type_traits> 17c1bd9197SEric Fiselier 18fc88dbd2SHoward Hinnant#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) 19fc88dbd2SHoward Hinnant# pragma GCC system_header 20fc88dbd2SHoward Hinnant#endif 21fc88dbd2SHoward Hinnant 22f3966eafSLouis Dionne// Catch invalid uses of the legacy _LIBCPP_DEBUG toggle. 23f3966eafSLouis Dionne#if defined(_LIBCPP_DEBUG) && _LIBCPP_DEBUG != 0 && !defined(_LIBCPP_ENABLE_DEBUG_MODE) 24f3966eafSLouis Dionne# error "Enabling the debug mode now requires having configured the library with support for the debug mode" 25f3966eafSLouis Dionne#endif 26f3966eafSLouis Dionne 27f3966eafSLouis Dionne#if defined(_LIBCPP_ENABLE_DEBUG_MODE) && !defined(_LIBCPP_CXX03_LANG) && !defined(_LIBCPP_DEBUG_RANDOMIZE_UNSPECIFIED_STABILITY) 28f3966eafSLouis Dionne# define _LIBCPP_DEBUG_RANDOMIZE_UNSPECIFIED_STABILITY 29f3966eafSLouis Dionne#endif 30f3966eafSLouis Dionne 31f3966eafSLouis Dionne#ifdef _LIBCPP_ENABLE_DEBUG_MODE 32f3966eafSLouis Dionne# define _LIBCPP_DEBUG_ASSERT(x, m) _LIBCPP_ASSERT(::std::__libcpp_is_constant_evaluated() || (x), m) 33f3966eafSLouis Dionne#else 34f3966eafSLouis Dionne# define _LIBCPP_DEBUG_ASSERT(x, m) ((void)0) 35f3966eafSLouis Dionne#endif 36f3966eafSLouis Dionne 37f3966eafSLouis Dionne#if defined(_LIBCPP_ENABLE_DEBUG_MODE) || defined(_LIBCPP_BUILDING_LIBRARY) 38f3966eafSLouis Dionne 39f554add5SHoward Hinnant_LIBCPP_BEGIN_NAMESPACE_STD 40f554add5SHoward Hinnant 416e41256fSHoward Hinnantstruct _LIBCPP_TYPE_VIS __c_node; 42f554add5SHoward Hinnant 436e41256fSHoward Hinnantstruct _LIBCPP_TYPE_VIS __i_node 44f554add5SHoward Hinnant{ 45f554add5SHoward Hinnant void* __i_; 46f554add5SHoward Hinnant __i_node* __next_; 47f554add5SHoward Hinnant __c_node* __c_; 48f554add5SHoward Hinnant 49f554add5SHoward Hinnant __i_node(const __i_node&) = delete; 50f554add5SHoward Hinnant __i_node& operator=(const __i_node&) = delete; 51318507edSNikolas Klauser 52f554add5SHoward Hinnant _LIBCPP_INLINE_VISIBILITY 53f554add5SHoward Hinnant __i_node(void* __i, __i_node* __next, __c_node* __c) 54f554add5SHoward Hinnant : __i_(__i), __next_(__next), __c_(__c) {} 55f554add5SHoward Hinnant ~__i_node(); 56f554add5SHoward Hinnant}; 57f554add5SHoward Hinnant 586e41256fSHoward Hinnantstruct _LIBCPP_TYPE_VIS __c_node 59f554add5SHoward Hinnant{ 60f554add5SHoward Hinnant void* __c_; 61f554add5SHoward Hinnant __c_node* __next_; 62f554add5SHoward Hinnant __i_node** beg_; 63f554add5SHoward Hinnant __i_node** end_; 64f554add5SHoward Hinnant __i_node** cap_; 65f554add5SHoward Hinnant 66f554add5SHoward Hinnant __c_node(const __c_node&) = delete; 67f554add5SHoward Hinnant __c_node& operator=(const __c_node&) = delete; 68318507edSNikolas Klauser 69f554add5SHoward Hinnant _LIBCPP_INLINE_VISIBILITY 70f86c2b6fSArthur O'Dwyer explicit __c_node(void* __c, __c_node* __next) 71f554add5SHoward Hinnant : __c_(__c), __next_(__next), beg_(nullptr), end_(nullptr), cap_(nullptr) {} 72f554add5SHoward Hinnant virtual ~__c_node(); 73f554add5SHoward Hinnant 74f554add5SHoward Hinnant virtual bool __dereferenceable(const void*) const = 0; 75f554add5SHoward Hinnant virtual bool __decrementable(const void*) const = 0; 76f554add5SHoward Hinnant virtual bool __addable(const void*, ptrdiff_t) const = 0; 77f554add5SHoward Hinnant virtual bool __subscriptable(const void*, ptrdiff_t) const = 0; 78f554add5SHoward Hinnant 79920b56caSHoward Hinnant void __add(__i_node* __i); 80f554add5SHoward Hinnant _LIBCPP_HIDDEN void __remove(__i_node* __i); 81f554add5SHoward Hinnant}; 82f554add5SHoward Hinnant 83f554add5SHoward Hinnanttemplate <class _Cont> 84f554add5SHoward Hinnantstruct _C_node 85f554add5SHoward Hinnant : public __c_node 86f554add5SHoward Hinnant{ 87f86c2b6fSArthur O'Dwyer explicit _C_node(void* __c, __c_node* __n) 88f554add5SHoward Hinnant : __c_node(__c, __n) {} 89f554add5SHoward Hinnant 90f554add5SHoward Hinnant virtual bool __dereferenceable(const void*) const; 91f554add5SHoward Hinnant virtual bool __decrementable(const void*) const; 92f554add5SHoward Hinnant virtual bool __addable(const void*, ptrdiff_t) const; 93f554add5SHoward Hinnant virtual bool __subscriptable(const void*, ptrdiff_t) const; 94f554add5SHoward Hinnant}; 95f554add5SHoward Hinnant 96f554add5SHoward Hinnanttemplate <class _Cont> 97687d3213SEric Fiselierinline bool 98f554add5SHoward Hinnant_C_node<_Cont>::__dereferenceable(const void* __i) const 99f554add5SHoward Hinnant{ 100f554add5SHoward Hinnant typedef typename _Cont::const_iterator iterator; 101f554add5SHoward Hinnant const iterator* __j = static_cast<const iterator*>(__i); 102c003db1fSHoward Hinnant _Cont* _Cp = static_cast<_Cont*>(__c_); 103c003db1fSHoward Hinnant return _Cp->__dereferenceable(__j); 104f554add5SHoward Hinnant} 105f554add5SHoward Hinnant 106f554add5SHoward Hinnanttemplate <class _Cont> 107687d3213SEric Fiselierinline bool 108f554add5SHoward Hinnant_C_node<_Cont>::__decrementable(const void* __i) const 109f554add5SHoward Hinnant{ 110f554add5SHoward Hinnant typedef typename _Cont::const_iterator iterator; 111f554add5SHoward Hinnant const iterator* __j = static_cast<const iterator*>(__i); 112c003db1fSHoward Hinnant _Cont* _Cp = static_cast<_Cont*>(__c_); 113c003db1fSHoward Hinnant return _Cp->__decrementable(__j); 114f554add5SHoward Hinnant} 115f554add5SHoward Hinnant 116f554add5SHoward Hinnanttemplate <class _Cont> 117687d3213SEric Fiselierinline bool 118f554add5SHoward Hinnant_C_node<_Cont>::__addable(const void* __i, ptrdiff_t __n) const 119f554add5SHoward Hinnant{ 120f554add5SHoward Hinnant typedef typename _Cont::const_iterator iterator; 121f554add5SHoward Hinnant const iterator* __j = static_cast<const iterator*>(__i); 122c003db1fSHoward Hinnant _Cont* _Cp = static_cast<_Cont*>(__c_); 123c003db1fSHoward Hinnant return _Cp->__addable(__j, __n); 124f554add5SHoward Hinnant} 125f554add5SHoward Hinnant 126f554add5SHoward Hinnanttemplate <class _Cont> 127687d3213SEric Fiselierinline bool 128f554add5SHoward Hinnant_C_node<_Cont>::__subscriptable(const void* __i, ptrdiff_t __n) const 129f554add5SHoward Hinnant{ 130f554add5SHoward Hinnant typedef typename _Cont::const_iterator iterator; 131f554add5SHoward Hinnant const iterator* __j = static_cast<const iterator*>(__i); 132c003db1fSHoward Hinnant _Cont* _Cp = static_cast<_Cont*>(__c_); 133c003db1fSHoward Hinnant return _Cp->__subscriptable(__j, __n); 134f554add5SHoward Hinnant} 135f554add5SHoward Hinnant 1366e41256fSHoward Hinnantclass _LIBCPP_TYPE_VIS __libcpp_db 137f554add5SHoward Hinnant{ 138f554add5SHoward Hinnant __c_node** __cbeg_; 139f554add5SHoward Hinnant __c_node** __cend_; 140f554add5SHoward Hinnant size_t __csz_; 141f554add5SHoward Hinnant __i_node** __ibeg_; 142f554add5SHoward Hinnant __i_node** __iend_; 143f554add5SHoward Hinnant size_t __isz_; 144f554add5SHoward Hinnant 145f86c2b6fSArthur O'Dwyer explicit __libcpp_db(); 146f554add5SHoward Hinnantpublic: 147f554add5SHoward Hinnant __libcpp_db(const __libcpp_db&) = delete; 148f554add5SHoward Hinnant __libcpp_db& operator=(const __libcpp_db&) = delete; 149318507edSNikolas Klauser 150f554add5SHoward Hinnant ~__libcpp_db(); 151f554add5SHoward Hinnant 152f554add5SHoward Hinnant class __db_c_iterator; 153f554add5SHoward Hinnant class __db_c_const_iterator; 154f554add5SHoward Hinnant class __db_i_iterator; 155f554add5SHoward Hinnant class __db_i_const_iterator; 156f554add5SHoward Hinnant 157f554add5SHoward Hinnant __db_c_const_iterator __c_end() const; 158f554add5SHoward Hinnant __db_i_const_iterator __i_end() const; 159f554add5SHoward Hinnant 1601c014d75SEric Fiselier typedef __c_node*(_InsertConstruct)(void*, void*, __c_node*); 1611c014d75SEric Fiselier 1621c014d75SEric Fiselier template <class _Cont> 1631c014d75SEric Fiselier _LIBCPP_INLINE_VISIBILITY static __c_node* __create_C_node(void *__mem, void *__c, __c_node *__next) { 1641c014d75SEric Fiselier return ::new (__mem) _C_node<_Cont>(__c, __next); 1651c014d75SEric Fiselier } 1661c014d75SEric Fiselier 167f554add5SHoward Hinnant template <class _Cont> 168f554add5SHoward Hinnant _LIBCPP_INLINE_VISIBILITY 169f554add5SHoward Hinnant void __insert_c(_Cont* __c) 170f554add5SHoward Hinnant { 1711c014d75SEric Fiselier __insert_c(static_cast<void*>(__c), &__create_C_node<_Cont>); 172f554add5SHoward Hinnant } 173f554add5SHoward Hinnant 174c36bfc49SHoward Hinnant void __insert_i(void* __i); 1751c014d75SEric Fiselier void __insert_c(void* __c, _InsertConstruct* __fn); 176f554add5SHoward Hinnant void __erase_c(void* __c); 177f554add5SHoward Hinnant 178f554add5SHoward Hinnant void __insert_ic(void* __i, const void* __c); 179f554add5SHoward Hinnant void __iterator_copy(void* __i, const void* __i0); 180f554add5SHoward Hinnant void __erase_i(void* __i); 181f554add5SHoward Hinnant 182f554add5SHoward Hinnant void* __find_c_from_i(void* __i) const; 183f554add5SHoward Hinnant void __invalidate_all(void* __c); 184f554add5SHoward Hinnant __c_node* __find_c_and_lock(void* __c) const; 185920b56caSHoward Hinnant __c_node* __find_c(void* __c) const; 186f554add5SHoward Hinnant void unlock() const; 187f554add5SHoward Hinnant 188f554add5SHoward Hinnant void swap(void* __c1, void* __c2); 189f554add5SHoward Hinnant 190f554add5SHoward Hinnant 191f554add5SHoward Hinnant bool __dereferenceable(const void* __i) const; 192f554add5SHoward Hinnant bool __decrementable(const void* __i) const; 193f554add5SHoward Hinnant bool __addable(const void* __i, ptrdiff_t __n) const; 194f554add5SHoward Hinnant bool __subscriptable(const void* __i, ptrdiff_t __n) const; 19542a3046eSHoward Hinnant bool __less_than_comparable(const void* __i, const void* __j) const; 196f554add5SHoward Hinnantprivate: 197f554add5SHoward Hinnant _LIBCPP_HIDDEN 198f554add5SHoward Hinnant __i_node* __insert_iterator(void* __i); 199f554add5SHoward Hinnant _LIBCPP_HIDDEN 200f554add5SHoward Hinnant __i_node* __find_iterator(const void* __i) const; 201f554add5SHoward Hinnant 2026e41256fSHoward Hinnant friend _LIBCPP_FUNC_VIS __libcpp_db* __get_db(); 203f554add5SHoward Hinnant}; 204f554add5SHoward Hinnant 2056e41256fSHoward Hinnant_LIBCPP_FUNC_VIS __libcpp_db* __get_db(); 2066e41256fSHoward Hinnant_LIBCPP_FUNC_VIS const __libcpp_db* __get_const_db(); 207f554add5SHoward Hinnant 208f3966eafSLouis Dionne_LIBCPP_END_NAMESPACE_STD 209f554add5SHoward Hinnant 210f3966eafSLouis Dionne#endif // defined(_LIBCPP_ENABLE_DEBUG_MODE) || defined(_LIBCPP_BUILDING_LIBRARY) 211f3966eafSLouis Dionne 212f3966eafSLouis Dionne_LIBCPP_BEGIN_NAMESPACE_STD 213f554add5SHoward Hinnant 214e3cf7050SNikolas Klausertemplate <class _Tp> 215e3cf7050SNikolas Klauser_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_AFTER_CXX11 inline void __debug_db_insert_c(_Tp* __c) { 216f3966eafSLouis Dionne#ifdef _LIBCPP_ENABLE_DEBUG_MODE 217e3cf7050SNikolas Klauser if (!__libcpp_is_constant_evaluated()) 218e3cf7050SNikolas Klauser __get_db()->__insert_c(__c); 219e3cf7050SNikolas Klauser#else 220e3cf7050SNikolas Klauser (void)(__c); 221e3cf7050SNikolas Klauser#endif 222e3cf7050SNikolas Klauser} 223e3cf7050SNikolas Klauser 224caf5548cSNikolas Klausertemplate <class _Tp> 225caf5548cSNikolas Klauser_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_AFTER_CXX11 inline void __debug_db_insert_i(_Tp* __i) { 226f3966eafSLouis Dionne#ifdef _LIBCPP_ENABLE_DEBUG_MODE 227caf5548cSNikolas Klauser if (!__libcpp_is_constant_evaluated()) 228caf5548cSNikolas Klauser __get_db()->__insert_i(__i); 229caf5548cSNikolas Klauser#else 230caf5548cSNikolas Klauser (void)(__i); 231caf5548cSNikolas Klauser#endif 232caf5548cSNikolas Klauser} 233caf5548cSNikolas Klauser 23408f68dfeSNikolas Klausertemplate <class _Tp> 23508f68dfeSNikolas Klauser_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_AFTER_CXX11 inline void __debug_db_erase_c(_Tp* __c) { 236f3966eafSLouis Dionne#ifdef _LIBCPP_ENABLE_DEBUG_MODE 23708f68dfeSNikolas Klauser if (!__libcpp_is_constant_evaluated()) 23808f68dfeSNikolas Klauser __get_db()->__erase_c(__c); 23908f68dfeSNikolas Klauser#else 24008f68dfeSNikolas Klauser (void)(__c); 24108f68dfeSNikolas Klauser#endif 24208f68dfeSNikolas Klauser} 24308f68dfeSNikolas Klauser 24408f68dfeSNikolas Klausertemplate <class _Tp> 24508f68dfeSNikolas Klauser_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_AFTER_CXX11 inline void __debug_db_swap(_Tp* __lhs, _Tp* __rhs) { 246f3966eafSLouis Dionne#ifdef _LIBCPP_ENABLE_DEBUG_MODE 24708f68dfeSNikolas Klauser if (!__libcpp_is_constant_evaluated()) 24808f68dfeSNikolas Klauser __get_db()->swap(__lhs, __rhs); 24908f68dfeSNikolas Klauser#else 25008f68dfeSNikolas Klauser (void)(__lhs); 25108f68dfeSNikolas Klauser (void)(__rhs); 25208f68dfeSNikolas Klauser#endif 25308f68dfeSNikolas Klauser} 25408f68dfeSNikolas Klauser 25508f68dfeSNikolas Klausertemplate <class _Tp> 25608f68dfeSNikolas Klauser_LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_AFTER_CXX11 inline void __debug_db_invalidate_all(_Tp* __c) { 257f3966eafSLouis Dionne#ifdef _LIBCPP_ENABLE_DEBUG_MODE 25808f68dfeSNikolas Klauser if (!__libcpp_is_constant_evaluated()) 25908f68dfeSNikolas Klauser __get_db()->__invalidate_all(__c); 26008f68dfeSNikolas Klauser#else 26108f68dfeSNikolas Klauser (void)(__c); 26208f68dfeSNikolas Klauser#endif 26308f68dfeSNikolas Klauser} 26408f68dfeSNikolas Klauser 265687d3213SEric Fiselier_LIBCPP_END_NAMESPACE_STD 266cec9af9eSHoward Hinnant 267cc82a1b0SLouis Dionne#endif // _LIBCPP___DEBUG 268