allocator.h
1 /***************************************************************************
2  Copyright (C) 2002-2015 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13 ***************************************************************************/
14 
15 #ifndef ALLOCATOR_H_
16 #define ALLOCATOR_H_
17 
18 #include <array>
19 #include <vector>
20 #include "atomic.h"
21 
22 namespace Transactional {
23 
24 enum {mempool_max_size = 4};
25 struct MemoryPool : public std::array<atomic<void*>, mempool_max_size> {
26  MemoryPool() {
27  std::fill(this->begin(), this->end(), nullptr);
28  }
29  ~MemoryPool() {
30  memoryBarrier();
31  for(auto &&x: *this) {
32  operator delete((void*)x);
33 // x = (void*)((uintptr_t)(void*)x | 1u); //invalid address.
34  }
35  }
36 };
37 
38 template<typename T>
39 class allocator {
40 public:
41  typedef size_t size_type;
42  typedef ptrdiff_t difference_type;
43  typedef T* pointer;
44  typedef const T* const_pointer;
45  typedef T& reference;
46  typedef const T& const_reference;
47  typedef T value_type;
48 
49  template<class Y>
50  struct rebind {
51  typedef allocator<Y> other;
52  };
53 
54  allocator(MemoryPool *pool) noexcept : m_pool(pool) {}
55  allocator(const allocator& x) noexcept : m_pool(x.m_pool) {}
56 
57  template<typename Y> allocator(const allocator<Y> &x) noexcept : m_pool(x.m_pool) {}
58 
59  ~allocator() {
60  }
61 
62  unsigned int pool_size() const {
63  return std::min((int)mempool_max_size, std::max(4, 256 / (int)sizeof(T)));
64  }
65  pointer allocate(size_type num, const void * /*hint*/ = 0) {
66  for(unsigned int i = 0; i < pool_size(); ++i) {
67  auto &x = (*m_pool)[i];
68  void *ptr = x.exchange(nullptr);
69  if(ptr) {
70 // if((uintptr_t)ptr & 1u)
71 // throw ptr; //invalid address.
72  return static_cast<pointer>(ptr);
73  }
74  }
75  return (pointer) (operator new(num * sizeof(T)));
76  }
77  template <class U, class... Args>
78  void construct(U* p, Args&&... args) {
79  new((void*) p) U(std::forward<Args>(args)...);
80  }
81 
82  void deallocate(pointer ptr, size_type /*num*/) {
83  void *p = ptr;
84  for(unsigned int i = 0; i < pool_size(); ++i) {
85  auto &x = (*m_pool)[i];
86  p = x.exchange(p);
87 // if((uintptr_t)p & 1u)
88 // throw ptr; //invalid address.
89  if( !p) {
90  return; //left in the pool.
91  }
92  }
93  operator delete(p);
94  }
95  template <class U>
96  void destroy(U* p) {
97  p->~U();
98  }
99 
100  pointer address(reference value) const noexcept {
101  return &value;
102  }
103  const_pointer address(const_reference value) const noexcept {
104  return &value;
105  }
106 
107  size_type max_size() const noexcept {
108  return std::numeric_limits<size_t>::max() / sizeof(T);
109  }
110 private:
111  template <typename Y> friend class allocator;
112  MemoryPool *m_pool;
113 };
114 
115 template <class T1, class T2>
116 bool operator==(const allocator<T1>&, const allocator<T2>&) noexcept {
117  return true;
118 }
119 
120 template <class T1, class T2>
121 bool operator!=(const allocator<T1>&, const allocator<T2>&) noexcept {
122  return false;
123 }
124 
125 template <typename T, int max_fixed_size = 4>
126 class fast_vector {
127 public:
128  using reference = T&;
129  using const_reference = const T&;
130  using size_type = size_t;
131  using pointer = T*;
132  using const_pointer = const T*;
133  using iterator = pointer;
134  using const_iterator = const_pointer;
135  fast_vector() : m_size(0) {}
136  ~fast_vector() { destroy(); }
137  fast_vector(size_type size) {
138  if(size > max_fixed_size) {
139  new (&m_vector) std::vector<T>(size);
140  m_size = HAS_STD_VECTOR;
141  }
142  else {
143  for(size_type i = 0; i < size; ++i)
144  new (m_array + i) T();
145  m_size = size;
146  }
147  }
148  fast_vector(const fast_vector &r) : m_size(0) {
149  this->operator=(r);
150  }
151  fast_vector(fast_vector &&r) : m_size(0) {
152  this->operator=(std::move(r));
153  }
154  fast_vector& operator=(const fast_vector &r) {
155  destroy();
156  if(r.is_fixed()) {
157  m_size = r.m_size;
158  for(size_type i = 0; i < m_size; ++i) {
159  new (m_array + i) T(r.m_array[i]);
160  }
161  }
162  else if(r.m_vector.size() <= max_fixed_size) {
163  m_size = r.m_vector.size();
164  for(size_type i = 0; i < m_size; ++i) {
165  new (m_array + i) T(r.m_vector[i]);
166  }
167  }
168  else {
169  m_size = HAS_STD_VECTOR;
170  new (&m_vector) std::vector<T>(r.m_vector);
171  }
172  return *this;
173  }
174  fast_vector& operator=(fast_vector &&r) {
175  destroy();
176  if(r.is_fixed()) {
177  m_size = r.m_size;
178  for(size_type i = 0; i < m_size; ++i) {
179  new (m_array + i) T(std::move(r.m_array[i]));
180  }
181  r.clear_fixed();
182  }
183  else {
184  m_size = HAS_STD_VECTOR;
185  new (&m_vector) std::vector<T>(std::move(r.m_vector));
186  }
187  return *this;
188  }
189  iterator begin() noexcept {return is_fixed() ? &m_array[0] : &m_vector[0];}
190  const_iterator begin() const noexcept {return is_fixed() ? &m_array[0] : &m_vector[0];}
191  iterator end() noexcept {return is_fixed() ? &m_array[m_size] : &m_vector[m_vector.size()];}
192  const_iterator end() const noexcept {return is_fixed() ? &m_array[m_size] : &m_vector[m_vector.size()];}
193  size_type size() const noexcept {return is_fixed() ? m_size : m_vector.size();}
194  bool empty() const noexcept {return !size();}
195  reference operator[](size_type n) {return is_fixed() ? m_array[n] : m_vector[n];}
196  const_reference operator[](size_type n) const {return is_fixed() ? m_array[n] : m_vector[n];}
197  const_reference at(size_type n) const {if(n >= size()) throw std::out_of_range(""); return (*this)[n];}
198  reference at(size_type n) {if(n >= size()) throw std::out_of_range(""); return (*this)[n];}
199  reference front() {return is_fixed() ? m_array.front() : m_vector.front();}
200  const_reference front() const {return is_fixed() ? m_array.front() : m_vector.front();}
201  reference back() {return (*this)[this->size() - 1];}
202  const_reference back() const {return (*this)[this->size() - 1];}
203  void push_back(const T& x) {
204  emplace_back(x);
205  }
206  void push_back(T&& x) {
207  emplace_back(std::move(x));
208  }
209  template <class... Args>
210  void emplace_back(Args&&... args) {
211  if(m_size < max_fixed_size) {
212  new (m_array + m_size) T(std::forward<Args>(args)...);
213  ++m_size;
214  }
215  else {
216  if(m_size == max_fixed_size) {
217  move_fixed_to_var(m_size);
218  }
219  m_vector.emplace_back(std::forward<Args>(args)...);
220  }
221  }
222  iterator erase(const_iterator position) {
223  if(is_fixed()) {
224  for(auto it = const_cast<iterator>(position);;) {
225  auto nit = it + 1;
226  if(nit == end()) {
227  it->~T();
228  break;
229  }
230  else
231  *it = std::move(*nit);
232  it = nit;
233  }
234  --m_size;
235  return const_cast<iterator>(position);
236  }
237  else {
238  auto it = m_vector.erase(m_vector.begin() + (position - begin()));
239  return &*it;
240  }
241  }
242 // iterator erase(const_iterator first, const_iterator last);
243  void clear() {
244  if(is_fixed()) {
245  clear_fixed();
246  }
247  else {
248  m_vector.clear();
249  }
250  }
251  void resize(size_type sz) {
252  if(is_fixed()) {
253  if(sz > max_fixed_size) {
254  move_fixed_to_var(sz);
255  }
256  else {
257  for(size_type i = m_size; i < sz; ++i)
258  new (m_array + i) T();
259  for(size_type i = sz; i < m_size; ++i)
260  m_array[i].~T();
261  m_size = sz;
262  }
263  }
264  else {
265  m_vector.resize(sz);
266 // shrink_to_fit();
267  }
268  }
269  void shrink_to_fit() {
270  if( !is_fixed()) return;
271  if(m_vector.capacity() - m_vector.size() > max_fixed_size) {
272  m_vector.shrink_to_fit();
273  }
274  }
275 private:
276  void destroy() {
277  clear();
278  if(!is_fixed())
279  m_vector.~vector();
280  }
281  bool is_fixed() const noexcept {return m_size != (size_type)HAS_STD_VECTOR;}
282  void clear_fixed() noexcept {
283  assert(is_fixed());
284  for(size_type i = 0; i < m_size; ++i)
285  m_array[i].~T();
286  m_size = 0;
287  }
288  void move_fixed_to_var(size_type new_size) {
289  std::vector<T> tmp;
290  tmp.reserve(m_size);
291  for(size_type i = 0; i < m_size; ++i) {
292  tmp.emplace_back(std::move(m_array[i]));
293  m_array[i].~T();
294  }
295  new (&m_vector) std::vector<T>();
296  m_vector.reserve(std::max(new_size, (size_type)(max_fixed_size * 2)));
297  assert(new_size >= m_size);
298  for(auto &&x: tmp)
299  m_vector.emplace_back(std::move(x));
300  m_vector.resize(new_size);
301  m_size = HAS_STD_VECTOR;
302  }
303  size_type m_size;
304  enum : size_type {HAS_STD_VECTOR = (size_type)-1};
305  union {
306  T m_array[max_fixed_size];
307  std::vector<T> m_vector;
308  };
309 };
310 
311 }
312 #endif /* ALLOCATOR_H_ */

Generated for KAME4 by  doxygen 1.8.3