forked from andrewprock/ustl
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathuvector.h
328 lines (298 loc) · 12.2 KB
/
uvector.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
// This file is part of the uSTL library, an STL implementation.
//
// Copyright (c) 2005 by Mike Sharov <[email protected]>
// This file is free software, distributed under the MIT License.
#pragma once
#include "memblock.h"
#include "umemory.h"
#include "upredalgo.h"
namespace ustl {
/// \class vector uvector.h ustl.h
/// \ingroup Sequences
///
/// \brief STL vector equivalent.
///
/// Provides a typed array-like interface to a managed memory block, including
/// element access, iteration, modification, resizing, and serialization. In
/// this design elements frequently undergo bitwise move, so don't put it in
/// here if it doesn't support it. This mostly means having no self-pointers.
///
template <typename T>
class vector {
public:
typedef T value_type;
typedef value_type* pointer;
typedef const value_type* const_pointer;
typedef value_type& reference;
typedef const value_type& const_reference;
typedef pointer iterator;
typedef const_pointer const_iterator;
typedef memblock::size_type size_type;
typedef memblock::written_size_type written_size_type;
typedef memblock::difference_type difference_type;
typedef ::ustl::reverse_iterator<iterator> reverse_iterator;
typedef ::ustl::reverse_iterator<const_iterator> const_reverse_iterator;
public:
inline vector (void);
inline explicit vector (size_type n);
vector (size_type n, const T& v);
vector (const vector& v);
vector (const_iterator i1, const_iterator i2);
inline ~vector (void) noexcept;
inline const vector& operator= (const vector& v);
inline bool operator== (const vector& v) const { return _data == v._data; }
inline operator cmemlink (void) const { return cmemlink (_data); }
inline operator cmemlink (void) { return cmemlink (_data); }
inline operator memlink (void) { return memlink (_data); }
inline void reserve (size_type n, bool bExact = false);
inline void resize (size_type n, bool bExact = true);
inline size_type capacity (void) const { return _data.capacity() / sizeof(T); }
inline size_type size (void) const { return _data.size() / sizeof(T); }
inline size_type max_size (void) const { return _data.max_size() / sizeof(T); }
inline bool empty (void) const { return _data.empty(); }
inline iterator begin (void) { return iterator (_data.begin()); }
inline const_iterator begin (void) const { return const_iterator (_data.begin()); }
inline iterator end (void) { return iterator (_data.end()); }
inline const_iterator end (void) const { return const_iterator (_data.end()); }
inline const_iterator cbegin (void) const { return begin(); }
inline const_iterator cend (void) const { return end(); }
inline reverse_iterator rbegin (void) { return reverse_iterator (end()); }
inline const_reverse_iterator rbegin (void) const { return const_reverse_iterator (end()); }
inline reverse_iterator rend (void) { return reverse_iterator (begin()); }
inline const_reverse_iterator rend (void) const { return const_reverse_iterator (begin()); }
inline const_reverse_iterator crbegin (void) const { return rbegin(); }
inline const_reverse_iterator crend (void) const { return rend(); }
inline pointer data (void) { return pointer (_data.data()); }
inline const_pointer data (void) const { return const_pointer (_data.data()); }
inline const_pointer cdata (void) const { return const_pointer (_data.cdata()); }
inline iterator iat (size_type i) { assert (i <= size()); return begin() + i; }
inline const_iterator iat (size_type i) const { assert (i <= size()); return begin() + i; }
inline reference at (size_type i) { assert (i < size()); return begin()[i]; }
inline const_reference at (size_type i) const { assert (i < size()); return begin()[i]; }
inline reference operator[] (size_type i) { return at (i); }
inline const_reference operator[] (size_type i) const { return at (i); }
inline reference front (void) { return at(0); }
inline const_reference front (void) const { return at(0); }
inline reference back (void) { assert (!empty()); return end()[-1]; }
inline const_reference back (void) const { assert (!empty()); return end()[-1]; }
inline void push_back (const T& v = T());
inline void pop_back (void) { destroy (end()-1); _data.memlink::resize (_data.size() - sizeof(T)); }
inline void clear (void) { destroy (begin(), end()); _data.clear(); }
inline void shrink_to_fit (void) { _data.shrink_to_fit(); }
inline void deallocate (void) noexcept;
inline void assign (const_iterator i1, const_iterator i2);
inline void assign (size_type n, const T& v);
inline void swap (vector& v) { _data.swap (v._data); }
inline iterator insert (const_iterator ip, const T& v);
inline iterator insert (const_iterator ip, size_type n, const T& v);
inline iterator insert (const_iterator ip, const_iterator i1, const_iterator i2);
inline iterator erase (const_iterator ep, size_type n = 1);
inline iterator erase (const_iterator ep1, const_iterator ep2);
inline void manage (pointer p, size_type n) { _data.manage (p, n * sizeof(T)); }
inline bool is_linked (void) const { return _data.is_linked(); }
inline void unlink (void) { _data.unlink(); }
inline void copy_link (void) { _data.copy_link(); }
inline void link (const_pointer p, size_type n) { _data.link (p, n * sizeof(T)); }
inline void link (pointer p, size_type n) { _data.link (p, n * sizeof(T)); }
inline void link (const vector& v) { _data.link (v); }
inline void link (vector& v) { _data.link (v); }
inline void link (const_pointer first, const_pointer last) { _data.link (first, last); }
inline void link (pointer first, pointer last) { _data.link (first, last); }
inline void read (istream& is) { container_read (is, *this); }
inline void write (ostream& os) const { container_write (os, *this); }
inline void text_write (ostringstream& os) const { container_text_write (os, *this); }
inline size_t stream_size (void) const { return container_stream_size (*this); }
#if HAVE_CPP11
inline vector (vector&& v) : _data(move(v._data)) {}
inline vector (std::initializer_list<T> v) : _data() { uninitialized_copy_n (v.begin(), v.size(), append_hole(v.size())); }
inline vector& operator= (vector&& v) { swap (v); return *this; }
template <typename... Args>
inline iterator emplace (const_iterator ip, Args&&... args);
template <typename... Args>
inline void emplace_back (Args&&... args);
inline void push_back (T&& v) { emplace_back (move(v)); }
inline iterator insert (const_iterator ip, T&& v) { return emplace (ip, move(v)); }
inline iterator insert (const_iterator ip, std::initializer_list<T> v) { return insert (ip, v.begin(), v.end()); }
#endif
protected:
inline iterator insert_space (const_iterator ip, size_type n);
private:
inline iterator insert_hole (const_iterator ip, size_type n);
inline iterator append_hole (size_type n);
private:
memblock _data; ///< Raw element data, consecutively stored.
};
/// Allocates space for at least \p n elements.
template <typename T>
inline void vector<T>::reserve (size_type n, bool bExact)
{
_data.reserve (n * sizeof(T), bExact);
}
template <typename T>
inline typename vector<T>::iterator vector<T>::append_hole (size_type n)
{
_data.reserve (_data.size() + n*sizeof(T));
_data.memlink::resize (_data.size()+n*sizeof(T));
return end()-n;
}
/// Resizes the vector to contain \p n elements.
template <typename T>
inline void vector<T>::resize (size_type n, bool bExact)
{
destroy (begin()+n, end());
const size_type nb = n * sizeof(T);
if (_data.capacity() < nb)
reserve (n, bExact);
uninitialized_default_construct_n (end(), (nb - _data.size())/sizeof(T));
_data.memlink::resize (nb);
}
/// Calls element destructors and frees storage.
template <typename T>
inline void vector<T>::deallocate (void) noexcept
{
destroy (begin(), end());
_data.deallocate();
}
/// Initializes empty vector.
template <typename T>
inline vector<T>::vector (void)
:_data()
{
}
/// Initializes a vector of size \p n.
template <typename T>
inline vector<T>::vector (size_type n)
:_data()
{
resize (n);
}
/// Copies \p n elements from \p v.
template <typename T>
vector<T>::vector (size_type n, const T& v)
:_data()
{
uninitialized_fill_n (append_hole (n), n, v);
}
/// Copies \p v.
template <typename T>
vector<T>::vector (const vector<T>& v)
:_data()
{
uninitialized_copy_n (v.begin(), v.size(), append_hole(v.size()));
}
/// Copies range [\p i1, \p i2]
template <typename T>
vector<T>::vector (const_iterator i1, const_iterator i2)
:_data()
{
uninitialized_copy (i1, i2, append_hole(distance(i1,i2)));
}
/// Destructor
template <typename T>
inline vector<T>::~vector (void) noexcept
{
destroy (begin(), end());
}
/// Copies the range [\p i1, \p i2]
template <typename T>
inline void vector<T>::assign (const_iterator i1, const_iterator i2)
{
assert (i1 <= i2);
resize (distance (i1, i2));
::ustl::copy (i1, i2, begin());
}
/// Copies \p n elements with value \p v.
template <typename T>
inline void vector<T>::assign (size_type n, const T& v)
{
resize (n);
::ustl::fill (begin(), end(), v);
}
/// Copies contents of \p v.
template <typename T>
inline const vector<T>& vector<T>::operator= (const vector<T>& v)
{
assign (v.begin(), v.end());
return *this;
}
/// Inserts \p n uninitialized elements at \p ip.
template <typename T>
inline typename vector<T>::iterator vector<T>::insert_hole (const_iterator ip, size_type n)
{
const uoff_t ipmi = distance (_data.begin(), memblock::const_iterator(ip));
reserve (size() + n);
return iterator (_data.insert (_data.iat(ipmi), n * sizeof(T)));
}
/// Inserts \p n uninitialized elements at \p ip.
template <typename T>
inline typename vector<T>::iterator vector<T>::insert_space (const_iterator ip, size_type n)
{
iterator ih = insert_hole (ip, n);
uninitialized_default_construct_n (ih, n);
return ih;
}
/// Inserts \p n elements with value \p v at offsets \p ip.
template <typename T>
inline typename vector<T>::iterator vector<T>::insert (const_iterator ip, size_type n, const T& v)
{
iterator d = insert_hole (ip, n);
uninitialized_fill_n (d, n, v);
return d;
}
/// Inserts value \p v at offset \p ip.
template <typename T>
inline typename vector<T>::iterator vector<T>::insert (const_iterator ip, const T& v)
{
iterator d = insert_hole (ip, 1);
construct_at (d, v);
return d;
}
/// Inserts range [\p i1, \p i2] at offset \p ip.
template <typename T>
inline typename vector<T>::iterator vector<T>::insert (const_iterator ip, const_iterator i1, const_iterator i2)
{
assert (i1 <= i2);
iterator d = insert_hole (ip, distance (i1, i2));
uninitialized_copy (i1, i2, d);
return d;
}
/// Removes \p count elements at offset \p ep.
template <typename T>
inline typename vector<T>::iterator vector<T>::erase (const_iterator ep, size_type n)
{
iterator d = const_cast<iterator>(ep);
destroy_n (d, n);
return iterator (_data.erase (memblock::iterator(d), n * sizeof(T)));
}
/// Removes elements from \p ep1 to \p ep2.
template <typename T>
inline typename vector<T>::iterator vector<T>::erase (const_iterator ep1, const_iterator ep2)
{
assert (ep1 <= ep2);
return erase (ep1, distance(ep1, ep2));
}
/// Inserts value \p v at the end of the vector.
template <typename T>
inline void vector<T>::push_back (const T& v)
{
construct_at (append_hole(1), v);
}
#if HAVE_CPP11
/// Constructs value at \p ip
template <typename T>
template <typename... Args>
inline typename vector<T>::iterator vector<T>::emplace (const_iterator ip, Args&&... args)
{
return new (insert_hole(ip,1)) T (forward<Args>(args)...);
}
/// Constructs value at the end of the vector.
template <typename T>
template <typename... Args>
inline void vector<T>::emplace_back (Args&&... args)
{
new (append_hole(1)) T (forward<Args>(args)...);
}
#endif
/// Use with vector classes to allocate and link to stack space. \p n is in elements.
#define typed_alloca_link(m,T,n) (m).link ((T*) alloca ((n) * sizeof(T)), (n))
} // namespace ustl