1 
2 //----------------------------------------------------------------------------
3 // Anti-Grain Geometry - Version 2.3
4 // Copyright (C) 2002-2005 Maxim Shemanarev (http://www.antigrain.com)
5 //
6 // Permission to copy, use, modify, sell and distribute this software
7 // is granted provided this copyright notice appears in all copies.
8 // This software is provided "as is" without express or implied
9 // warranty, and with no claim as to its suitability for any purpose.
10 //
11 //----------------------------------------------------------------------------
12 // Contact: mcseem@antigrain.com
13 //          mcseemagg@yahoo.com
14 //          http://www.antigrain.com
15 //----------------------------------------------------------------------------
16 #ifndef AGG_ARRAY_INCLUDED
17 #define AGG_ARRAY_INCLUDED
18 
19 #include "agg_basics.h"
20 #include "core/fxcrt/fx_memory.h"  // For FXSYS_* macros.
21 
22 namespace agg
23 {
24 template <class T>
25 class pod_array {
26 public:
27     typedef T value_type;
~pod_array()28     ~pod_array()
29     {
30         FX_Free(m_array);
31     }
pod_array()32     pod_array() : m_size(0), m_capacity(0), m_array(0) {}
33     pod_array(unsigned cap, unsigned extra_tail = 0);
34     pod_array(const pod_array<T>&);
35     const pod_array<T>& operator = (const pod_array<T>&);
36     void capacity(unsigned cap, unsigned extra_tail = 0);
capacity()37     unsigned capacity() const
38     {
39         return m_capacity;
40     }
41     void allocate(unsigned size, unsigned extra_tail = 0);
42     void resize(unsigned new_size);
zero()43     void zero()
44     {
45         FXSYS_memset(m_array, 0, sizeof(T) * m_size);
46     }
add(const T & v)47     void add(const T& v)
48     {
49         m_array[m_size++] = v;
50     }
inc_size(unsigned size)51     void inc_size(unsigned size)
52     {
53         m_size += size;
54     }
size()55     unsigned size()      const
56     {
57         return m_size;
58     }
byte_size()59     unsigned byte_size() const
60     {
61         return m_size * sizeof(T);
62     }
63     const T& operator [] (unsigned i) const
64     {
65         return m_array[i];
66     }
67     T& operator [] (unsigned i)
68     {
69         return m_array[i];
70     }
at(unsigned i)71     const T& at(unsigned i) const
72     {
73         return m_array[i];
74     }
at(unsigned i)75     T& at(unsigned i)
76     {
77         return m_array[i];
78     }
value_at(unsigned i)79     T  value_at(unsigned i) const
80     {
81         return m_array[i];
82     }
data()83     const T* data() const
84     {
85         return m_array;
86     }
data()87     T* data()
88     {
89         return m_array;
90     }
remove_all()91     void remove_all()
92     {
93         m_size = 0;
94     }
cut_at(unsigned num)95     void cut_at(unsigned num)
96     {
97         if(num < m_size) {
98             m_size = num;
99         }
100     }
101 private:
102     unsigned m_size;
103     unsigned m_capacity;
104     T*       m_array;
105 };
106 template<class T>
capacity(unsigned cap,unsigned extra_tail)107 void pod_array<T>::capacity(unsigned cap, unsigned extra_tail)
108 {
109     m_size = 0;
110     unsigned full_cap = cap + extra_tail;
111     if(full_cap < cap) {
112         FX_Free(m_array);
113         m_array = 0;
114         m_capacity = 0;
115     } else if(full_cap > m_capacity) {
116         FX_Free(m_array);
117         m_array = FX_Alloc(T, full_cap);
118         m_capacity = full_cap;
119     }
120 }
121 template<class T>
allocate(unsigned size,unsigned extra_tail)122 void pod_array<T>::allocate(unsigned size, unsigned extra_tail)
123 {
124     capacity(size, extra_tail);
125     m_size = size;
126 }
127 template<class T>
resize(unsigned new_size)128 void pod_array<T>::resize(unsigned new_size)
129 {
130     if(new_size > m_size) {
131         if(new_size > m_capacity) {
132             T* data = FX_Alloc(T, new_size);
133             FXSYS_memcpy(data, m_array, m_size * sizeof(T));
134             FX_Free(m_array);
135             m_array = data;
136         }
137     } else {
138         m_size = new_size;
139     }
140 }
pod_array(unsigned cap,unsigned extra_tail)141 template<class T> pod_array<T>::pod_array(unsigned cap, unsigned extra_tail) :
142     m_size(0), m_capacity(cap + extra_tail), m_array(FX_Alloc(T, m_capacity)) {}
pod_array(const pod_array<T> & v)143 template<class T> pod_array<T>::pod_array(const pod_array<T>& v) :
144     m_size(v.m_size),
145     m_capacity(v.m_capacity),
146     m_array(v.m_capacity ? FX_Alloc(T, v.m_capacity) : 0)
147 {
148     FXSYS_memcpy(m_array, v.m_array, sizeof(T) * v.m_size);
149 }
150 template<class T> const pod_array<T>&
151 pod_array<T>::operator = (const pod_array<T>&v)
152 {
153     allocate(v.m_size);
154     if(v.m_size) {
155         FXSYS_memcpy(m_array, v.m_array, sizeof(T) * v.m_size);
156     }
157     return *this;
158 }
159 template<class T, unsigned S = 6> class pod_deque
160 {
161 public:
162     enum block_scale_e {
163         block_shift = S,
164         block_size  = 1 << block_shift,
165         block_mask  = block_size - 1
166     };
167     typedef T value_type;
168     ~pod_deque();
169     pod_deque();
170     pod_deque(unsigned block_ptr_inc);
171     pod_deque(const pod_deque<T, S>& v);
172     const pod_deque<T, S>& operator = (const pod_deque<T, S>& v);
remove_all()173     void remove_all()
174     {
175         m_size = 0;
176     }
free_all()177     void free_all()
178     {
179         free_tail(0);
180     }
181     void free_tail(unsigned size);
182     void add(const T& val);
183     void modify_last(const T& val);
184     void remove_last();
185     int allocate_continuous_block(unsigned num_elements);
add_array(const T * ptr,unsigned num_elem)186     void add_array(const T* ptr, unsigned num_elem)
187     {
188         while(num_elem--) {
189             add(*ptr++);
190         }
191     }
add_data(DataAccessor & data)192     template<class DataAccessor> void add_data(DataAccessor& data)
193     {
194         while(data.size()) {
195             add(*data);
196             ++data;
197         }
198     }
cut_at(unsigned size)199     void cut_at(unsigned size)
200     {
201         if(size < m_size) {
202             m_size = size;
203         }
204     }
size()205     unsigned size() const
206     {
207         return m_size;
208     }
209     const T& operator [] (unsigned i) const
210     {
211         return m_blocks[i >> block_shift][i & block_mask];
212     }
213     T& operator [] (unsigned i)
214     {
215         return m_blocks[i >> block_shift][i & block_mask];
216     }
at(unsigned i)217     const T& at(unsigned i) const
218     {
219         return m_blocks[i >> block_shift][i & block_mask];
220     }
at(unsigned i)221     T& at(unsigned i)
222     {
223         return m_blocks[i >> block_shift][i & block_mask];
224     }
value_at(unsigned i)225     T value_at(unsigned i) const
226     {
227         return m_blocks[i >> block_shift][i & block_mask];
228     }
curr(unsigned idx)229     const T& curr(unsigned idx) const
230     {
231         return (*this)[idx];
232     }
curr(unsigned idx)233     T& curr(unsigned idx)
234     {
235         return (*this)[idx];
236     }
prev(unsigned idx)237     const T& prev(unsigned idx) const
238     {
239         return (*this)[(idx + m_size - 1) % m_size];
240     }
prev(unsigned idx)241     T& prev(unsigned idx)
242     {
243         return (*this)[(idx + m_size - 1) % m_size];
244     }
next(unsigned idx)245     const T& next(unsigned idx) const
246     {
247         return (*this)[(idx + 1) % m_size];
248     }
next(unsigned idx)249     T& next(unsigned idx)
250     {
251         return (*this)[(idx + 1) % m_size];
252     }
last()253     const T& last() const
254     {
255         return (*this)[m_size - 1];
256     }
last()257     T& last()
258     {
259         return (*this)[m_size - 1];
260     }
261     unsigned byte_size() const;
block(unsigned nb)262     const T* block(unsigned nb) const
263     {
264         return m_blocks[nb];
265     }
266 public:
267     void allocate_block(unsigned nb);
268     T*   data_ptr();
269     unsigned        m_size;
270     unsigned        m_num_blocks;
271     unsigned        m_max_blocks;
272     T**             m_blocks;
273     unsigned        m_block_ptr_inc;
274 };
~pod_deque()275 template<class T, unsigned S> pod_deque<T, S>::~pod_deque()
276 {
277     if(m_num_blocks) {
278         T** blk = m_blocks + m_num_blocks - 1;
279         while(m_num_blocks--) {
280             FX_Free(*blk);
281             --blk;
282         }
283         FX_Free(m_blocks);
284     }
285 }
286 template<class T, unsigned S>
free_tail(unsigned size)287 void pod_deque<T, S>::free_tail(unsigned size)
288 {
289     if(size < m_size) {
290         unsigned nb = (size + block_mask) >> block_shift;
291         while(m_num_blocks > nb) {
292             FX_Free(m_blocks[--m_num_blocks]);
293         }
294         m_size = size;
295     }
296 }
pod_deque()297 template<class T, unsigned S> pod_deque<T, S>::pod_deque() :
298     m_size(0),
299     m_num_blocks(0),
300     m_max_blocks(0),
301     m_blocks(0),
302     m_block_ptr_inc(block_size)
303 {
304 }
305 template<class T, unsigned S>
pod_deque(unsigned block_ptr_inc)306 pod_deque<T, S>::pod_deque(unsigned block_ptr_inc) :
307     m_size(0),
308     m_num_blocks(0),
309     m_max_blocks(0),
310     m_blocks(0),
311     m_block_ptr_inc(block_ptr_inc)
312 {
313 }
314 template<class T, unsigned S>
pod_deque(const pod_deque<T,S> & v)315 pod_deque<T, S>::pod_deque(const pod_deque<T, S>& v) :
316     m_size(v.m_size),
317     m_num_blocks(v.m_num_blocks),
318     m_max_blocks(v.m_max_blocks),
319     m_blocks(v.m_max_blocks ? FX_Alloc(T*, v.m_max_blocks) : 0),
320     m_block_ptr_inc(v.m_block_ptr_inc)
321 {
322     unsigned i;
323     for(i = 0; i < v.m_num_blocks; ++i) {
324         m_blocks[i] = FX_Alloc(T, block_size);
325         FXSYS_memcpy(m_blocks[i], v.m_blocks[i], block_size * sizeof(T));
326     }
327 }
328 template<class T, unsigned S>
329 const pod_deque<T, S>& pod_deque<T, S>::operator = (const pod_deque<T, S>& v)
330 {
331     unsigned i;
332     for(i = m_num_blocks; i < v.m_num_blocks; ++i) {
333         allocate_block(i);
334     }
335     for(i = 0; i < v.m_num_blocks; ++i) {
336         FXSYS_memcpy(m_blocks[i], v.m_blocks[i], block_size * sizeof(T));
337     }
338     m_size = v.m_size;
339     return *this;
340 }
341 template<class T, unsigned S>
allocate_block(unsigned nb)342 void pod_deque<T, S>::allocate_block(unsigned nb)
343 {
344     if(nb >= m_max_blocks) {
345         T** new_blocks = FX_Alloc(T*, m_max_blocks + m_block_ptr_inc);
346         if(m_blocks) {
347             FXSYS_memcpy(new_blocks,
348                          m_blocks,
349                          m_num_blocks * sizeof(T*));
350             FX_Free(m_blocks);
351         }
352         m_blocks = new_blocks;
353         m_max_blocks += m_block_ptr_inc;
354     }
355     m_blocks[nb] = FX_Alloc(T, block_size);
356     m_num_blocks++;
357 }
358 template<class T, unsigned S>
data_ptr()359 inline T* pod_deque<T, S>::data_ptr()
360 {
361     unsigned nb = m_size >> block_shift;
362     if(nb >= m_num_blocks) {
363         allocate_block(nb);
364     }
365     return m_blocks[nb] + (m_size & block_mask);
366 }
367 template<class T, unsigned S>
add(const T & val)368 inline void pod_deque<T, S>::add(const T& val)
369 {
370     *data_ptr() = val;
371     ++m_size;
372 }
373 template<class T, unsigned S>
remove_last()374 inline void pod_deque<T, S>::remove_last()
375 {
376     if(m_size) {
377         --m_size;
378     }
379 }
380 template<class T, unsigned S>
modify_last(const T & val)381 void pod_deque<T, S>::modify_last(const T& val)
382 {
383     remove_last();
384     add(val);
385 }
386 template<class T, unsigned S>
allocate_continuous_block(unsigned num_elements)387 int pod_deque<T, S>::allocate_continuous_block(unsigned num_elements)
388 {
389     if(num_elements < block_size) {
390         data_ptr();
391         unsigned rest = block_size - (m_size & block_mask);
392         unsigned index;
393         if(num_elements <= rest) {
394             index = m_size;
395             m_size += num_elements;
396             return index;
397         }
398         m_size += rest;
399         data_ptr();
400         index = m_size;
401         m_size += num_elements;
402         return index;
403     }
404     return -1;
405 }
406 template<class T, unsigned S>
byte_size()407 unsigned pod_deque<T, S>::byte_size() const
408 {
409     return m_size * sizeof(T);
410 }
411 class pod_allocator
412 {
413 public:
remove_all()414     void remove_all()
415     {
416         if(m_num_blocks) {
417             int8u** blk = m_blocks + m_num_blocks - 1;
418             while(m_num_blocks--) {
419                 FX_Free(*blk);
420                 --blk;
421             }
422             FX_Free(m_blocks);
423         }
424         m_num_blocks = 0;
425         m_max_blocks = 0;
426         m_blocks = 0;
427         m_buf_ptr = 0;
428         m_rest = 0;
429     }
~pod_allocator()430     ~pod_allocator()
431     {
432         remove_all();
433     }
434     pod_allocator(unsigned block_size, unsigned block_ptr_inc = 256 - 8) :
m_block_size(block_size)435         m_block_size(block_size),
436         m_block_ptr_inc(block_ptr_inc),
437         m_num_blocks(0),
438         m_max_blocks(0),
439         m_blocks(0),
440         m_buf_ptr(0),
441         m_rest(0)
442     {
443     }
444     int8u* allocate(unsigned size, unsigned alignment = 1)
445     {
446         if(size == 0) {
447             return 0;
448         }
449         if(size <= m_rest) {
450             int8u* ptr = m_buf_ptr;
451             if(alignment > 1) {
452                 unsigned align = (alignment - unsigned((size_t)ptr) % alignment) % alignment;
453                 size += align;
454                 ptr += align;
455                 if(size <= m_rest) {
456                     m_rest -= size;
457                     m_buf_ptr += size;
458                     return ptr;
459                 }
460                 allocate_block(size);
461                 return allocate(size - align, alignment);
462             }
463             m_rest -= size;
464             m_buf_ptr += size;
465             return ptr;
466         }
467         allocate_block(size + alignment - 1);
468         return allocate(size, alignment);
469     }
470 private:
allocate_block(unsigned size)471     void allocate_block(unsigned size)
472     {
473         if(size < m_block_size) {
474             size = m_block_size;
475         }
476         if(m_num_blocks >= m_max_blocks) {
477             int8u** new_blocks = FX_Alloc(int8u*, m_max_blocks + m_block_ptr_inc);
478             if(m_blocks) {
479                 FXSYS_memcpy(new_blocks,
480                              m_blocks,
481                              m_num_blocks * sizeof(int8u*));
482                 FX_Free(m_blocks);
483             }
484             m_blocks = new_blocks;
485             m_max_blocks += m_block_ptr_inc;
486         }
487         m_blocks[m_num_blocks] = m_buf_ptr = FX_Alloc(int8u, size);
488         m_num_blocks++;
489         m_rest = size;
490     }
491     unsigned m_block_size;
492     unsigned m_block_ptr_inc;
493     unsigned m_num_blocks;
494     unsigned m_max_blocks;
495     int8u**  m_blocks;
496     int8u*   m_buf_ptr;
497     unsigned m_rest;
498 };
499 enum quick_sort_threshold_e {
500     quick_sort_threshold = 9
501 };
swap_elements(T & a,T & b)502 template<class T> inline void swap_elements(T& a, T& b)
503 {
504     T temp = a;
505     a = b;
506     b = temp;
507 }
508 }
509 #endif
510