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