1 // Debug support for the circular buffer library.
2
3 // Copyright (c) 2003-2008 Jan Gaspar
4
5 // Use, modification, and distribution is subject to the Boost Software
6 // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
7 // http://www.boost.org/LICENSE_1_0.txt)
8
9 #if !defined(BOOST_CIRCULAR_BUFFER_DEBUG_HPP)
10 #define BOOST_CIRCULAR_BUFFER_DEBUG_HPP
11
12 #if defined(_MSC_VER)
13 #pragma once
14 #endif
15
16 #if BOOST_CB_ENABLE_DEBUG
17 #include <cstring>
18
19 #if defined(BOOST_NO_STDC_NAMESPACE)
20 namespace std {
21 using ::memset;
22 }
23 #endif
24
25 #endif // BOOST_CB_ENABLE_DEBUG
26 namespace boost {
27
28 namespace cb_details {
29
30 #if BOOST_CB_ENABLE_DEBUG
31
32 // The value the uninitialized memory is filled with.
33 const int UNINITIALIZED = 0xcc;
34
35 template <class T>
do_fill_uninitialized_memory(T * data,std::size_t size_in_bytes)36 inline void do_fill_uninitialized_memory(T* data, std::size_t size_in_bytes) BOOST_NOEXCEPT {
37 std::memset(static_cast<void*>(data), UNINITIALIZED, size_in_bytes);
38 }
39
40 template <class T>
do_fill_uninitialized_memory(T &,std::size_t)41 inline void do_fill_uninitialized_memory(T& /*data*/, std::size_t /*size_in_bytes*/) BOOST_NOEXCEPT {
42 // Do nothing
43 }
44
45
46 class debug_iterator_registry;
47
48 /*!
49 \class debug_iterator_base
50 \brief Registers/unregisters iterators into the registry of valid iterators.
51
52 This class is intended to be a base class of an iterator.
53 */
54 class debug_iterator_base {
55
56 private:
57 // Members
58
59 //! Iterator registry.
60 mutable const debug_iterator_registry* m_registry;
61
62 //! Next iterator in the iterator chain.
63 mutable const debug_iterator_base* m_next;
64
65 public:
66 // Construction/destruction
67
68 //! Default constructor.
69 debug_iterator_base();
70
71 //! Constructor taking the iterator registry as a parameter.
72 debug_iterator_base(const debug_iterator_registry* registry);
73
74 //! Copy constructor.
75 debug_iterator_base(const debug_iterator_base& rhs);
76
77 //! Destructor.
78 ~debug_iterator_base();
79
80 // Methods
81
82 //! Assign operator.
83 debug_iterator_base& operator = (const debug_iterator_base& rhs);
84
85 //! Is the iterator valid?
86 bool is_valid(const debug_iterator_registry* registry) const;
87
88 //! Invalidate the iterator.
89 /*!
90 \note The method is const in order to invalidate const iterators, too.
91 */
92 void invalidate() const;
93
94 //! Return the next iterator in the iterator chain.
95 const debug_iterator_base* next() const;
96
97 //! Set the next iterator in the iterator chain.
98 /*!
99 \note The method is const in order to set a next iterator to a const iterator, too.
100 */
101 void set_next(const debug_iterator_base* it) const;
102
103 private:
104 // Helpers
105
106 //! Register self as a valid iterator.
107 void register_self();
108
109 //! Unregister self from valid iterators.
110 void unregister_self();
111 };
112
113 /*!
114 \class debug_iterator_registry
115 \brief Registry of valid iterators.
116
117 This class is intended to be a base class of a container.
118 */
119 class debug_iterator_registry {
120
121 //! Pointer to the chain of valid iterators.
122 mutable const debug_iterator_base* m_iterators;
123
124 public:
125 // Methods
126
127 //! Default constructor.
debug_iterator_registry()128 debug_iterator_registry() : m_iterators(0) {}
129
130 //! Register an iterator into the list of valid iterators.
131 /*!
132 \note The method is const in order to register iterators into const containers, too.
133 */
register_iterator(const debug_iterator_base * it) const134 void register_iterator(const debug_iterator_base* it) const {
135 it->set_next(m_iterators);
136 m_iterators = it;
137 }
138
139 //! Unregister an iterator from the list of valid iterators.
140 /*!
141 \note The method is const in order to unregister iterators from const containers, too.
142 */
unregister_iterator(const debug_iterator_base * it) const143 void unregister_iterator(const debug_iterator_base* it) const {
144 const debug_iterator_base* previous = 0;
145 for (const debug_iterator_base* p = m_iterators; p != it; previous = p, p = p->next()) {}
146 remove(it, previous);
147 }
148
149 //! Invalidate every iterator pointing to the same element as the iterator passed as a parameter.
150 template <class Iterator>
invalidate_iterators(const Iterator & it)151 void invalidate_iterators(const Iterator& it) {
152 const debug_iterator_base* previous = 0;
153 for (const debug_iterator_base* p = m_iterators; p != 0; p = p->next()) {
154 if (((Iterator*)p)->m_it == it.m_it) {
155 p->invalidate();
156 remove(p, previous);
157 continue;
158 }
159 previous = p;
160 }
161 }
162
163 //! Invalidate all iterators except an iterator poining to the same element as the iterator passed as a parameter.
164 template <class Iterator>
invalidate_iterators_except(const Iterator & it)165 void invalidate_iterators_except(const Iterator& it) {
166 const debug_iterator_base* previous = 0;
167 for (const debug_iterator_base* p = m_iterators; p != 0; p = p->next()) {
168 if (((Iterator*)p)->m_it != it.m_it) {
169 p->invalidate();
170 remove(p, previous);
171 continue;
172 }
173 previous = p;
174 }
175 }
176
177 //! Invalidate all iterators.
invalidate_all_iterators()178 void invalidate_all_iterators() {
179 for (const debug_iterator_base* p = m_iterators; p != 0; p = p->next())
180 p->invalidate();
181 m_iterators = 0;
182 }
183
184 private:
185 // Helpers
186
187 //! Remove the current iterator from the iterator chain.
remove(const debug_iterator_base * current,const debug_iterator_base * previous) const188 void remove(const debug_iterator_base* current,
189 const debug_iterator_base* previous) const {
190 if (previous == 0)
191 m_iterators = m_iterators->next();
192 else
193 previous->set_next(current->next());
194 }
195 };
196
197 // Implementation of the debug_iterator_base methods.
198
debug_iterator_base()199 inline debug_iterator_base::debug_iterator_base() : m_registry(0), m_next(0) {}
200
debug_iterator_base(const debug_iterator_registry * registry)201 inline debug_iterator_base::debug_iterator_base(const debug_iterator_registry* registry)
202 : m_registry(registry), m_next(0) {
203 register_self();
204 }
205
debug_iterator_base(const debug_iterator_base & rhs)206 inline debug_iterator_base::debug_iterator_base(const debug_iterator_base& rhs)
207 : m_registry(rhs.m_registry), m_next(0) {
208 register_self();
209 }
210
~debug_iterator_base()211 inline debug_iterator_base::~debug_iterator_base() { unregister_self(); }
212
operator =(const debug_iterator_base & rhs)213 inline debug_iterator_base& debug_iterator_base::operator = (const debug_iterator_base& rhs) {
214 if (m_registry == rhs.m_registry)
215 return *this;
216 unregister_self();
217 m_registry = rhs.m_registry;
218 register_self();
219 return *this;
220 }
221
is_valid(const debug_iterator_registry * registry) const222 inline bool debug_iterator_base::is_valid(const debug_iterator_registry* registry) const {
223 return m_registry == registry;
224 }
225
invalidate() const226 inline void debug_iterator_base::invalidate() const { m_registry = 0; }
227
next() const228 inline const debug_iterator_base* debug_iterator_base::next() const { return m_next; }
229
set_next(const debug_iterator_base * it) const230 inline void debug_iterator_base::set_next(const debug_iterator_base* it) const { m_next = it; }
231
register_self()232 inline void debug_iterator_base::register_self() {
233 if (m_registry != 0)
234 m_registry->register_iterator(this);
235 }
236
unregister_self()237 inline void debug_iterator_base::unregister_self() {
238 if (m_registry != 0)
239 m_registry->unregister_iterator(this);
240 }
241
242 #endif // #if BOOST_CB_ENABLE_DEBUG
243
244 } // namespace cb_details
245
246 } // namespace boost
247
248 #endif // #if !defined(BOOST_CIRCULAR_BUFFER_DEBUG_HPP)
249