BlackCat_Tensors
A GPU-supported autograd and linear algebra library, designed for neural network construction
stack_allocator.h
Go to the documentation of this file.
1 /*
2  * Stack_Allocator.h
3  *
4  * Created on: Mar 16, 2019
5  * Author: joseph
6  */
7 
8 #ifndef BC_CORE_CONTEXT_WORKSPACE_H_
9 #define BC_CORE_CONTEXT_WORKSPACE_H_
10 
11 #include "polymorphic_allocator.h"
12 #include <vector>
13 
14 namespace bc {
15 namespace allocators {
16 namespace detail {
17 
18 template<class SystemTag>
20 
21  using system_tag = SystemTag;
22  using value_type = Byte;
24 
25  std::size_t m_data_sz=0;
26  std::size_t m_curr_index=0;
27 
28  Byte* m_data=nullptr;
29  allocator_type m_allocator;
30 
31 public:
32  Stack_Allocator_Base(std::size_t sz=0) : m_data_sz(sz){
33  if (sz)
34  m_data = m_allocator.allocate(sz);
35  }
36  void reserve(std::size_t sz) {
37  if (sz == 0 || (m_data_sz - m_curr_index) > sz) {
38  return;
39  }
40 
41  if (!(m_curr_index==0)){
42  std::cout << "BC_Memory Allocation failure: \n" <<
43  "\tcurrent_stack_index == " << m_curr_index << " out of " << m_data_sz << \
44  "attempting to reserve " << sz << " bytes " << std::endl;
45  }
46 
47  BC_ASSERT(m_curr_index==0,
48  "Stack_Allocator reserve called while memory is still allocated");
49 
50  if (m_data_sz < sz) {
51  if (m_data_sz > 0) {
52  m_allocator.deallocate(m_data, m_data_sz);
53  }
54  m_data = m_allocator.allocate(sz);
55  m_data_sz = sz;
56  }
57  }
58  void free() {
59  BC_ASSERT(m_curr_index==0,
60  "Stack_Allocator free called while memory is still allocated");
61  m_allocator.deallocate(m_data, m_data_sz);
62  m_data_sz = 0;
63  m_data = nullptr;
64  }
65 
66  size_t available_bytes() const {
67  return m_data_sz - m_curr_index;
68  }
69  size_t allocated_bytes() const {
70  return m_curr_index;
71  }
72  size_t reserved_bytes() const {
73  return m_data_sz;
74  }
75 
76  template<class Allocator>
77  void set_allocator(Allocator alloc) {
78  BC_ASSERT(m_curr_index==0,
79  "Stack_Allocator set_allocator called while memory is still allocated");
80  m_allocator.set_allocator(alloc);
81  }
82 
84  if (m_curr_index + sz > m_data_sz) {
85  std::cout << "BC_Memory Allocation failure: \n" <<
86  "\tcurrent_stack_index == " << m_curr_index << " out of " << m_data_sz
87  <<"\n\t attempting to allocate " << sz << " bytes, error: " << (m_curr_index + sz <= m_data_sz) << std::endl;
88  }
89 
90  BC_ASSERT(!(m_curr_index + sz > m_data_sz),
91  "BC_Memory Allocation failure, attempting to allocate memory larger that workspace size");
92 
93  Byte* mem = m_data + m_curr_index;
94  m_curr_index += sz;
95  return mem;
96  }
97 
98  void deallocate(Byte* data, std::size_t sz) {
99  BC_ASSERT(m_curr_index != 0,
100  "BC_Memory Deallocation failure, attempting to deallocate already deallocated memory.");
101 
102  BC_ASSERT(data == (m_data + m_curr_index - sz),
103  "BC_Memory Deallocation failure, attempting to deallocate memory out of order,"
104  "\nStack_Allocator memory functions as a stack, deallocations must be in reverse order of allocations.");
105 
106 
107  m_curr_index -= sz;
108  }
109 
110 
112  m_curr_index = 0;
113  }
114 
115  template<class T>
117  return reinterpret_cast<T*>(allocate(sz * sizeof(T)));
118  }
119 
120  template<class T>
121  void deallocate(T* data, std::size_t sz) {
122  deallocate(reinterpret_cast<Byte*>(data), sz * sizeof(T));
123  }
124 
126  BC_ASSERT(m_curr_index==0,
127  "Stack_Allocator Destructor called while memory is still allocated, Memory Leak Detected");
128  m_allocator.deallocate(m_data, m_data_sz);
129  }
130 };
131 } //end of ns detail
132 
133 
137 template<class ValueType, class SystemTag>
139 
140  template<class, class>
141  friend class Stack_Allocator;
142 
144  std::shared_ptr<ws_base_t> ws_ptr;
145 
146 public:
147 
148  using system_tag = SystemTag;
149  using value_type = ValueType;
151 
152  template<class T>
153  struct rebind {
155  };
156 
157  Stack_Allocator(int sz=0):
158  ws_ptr(new ws_base_t(sz)) {}
159 
160  template<class T>
162  ws_ptr(ws.ws_ptr) {}
163 
164  Stack_Allocator(const Stack_Allocator&)=default;
165  Stack_Allocator(Stack_Allocator&&)=default;
166 
168  void reserve(std::size_t sz) {
169  ws_ptr->reserve(sz * sizeof(value_type));
170  }
171 
173  void free() {
174  ws_ptr->free();
175  }
176 
178  ws_ptr->force_deallocate();
179  }
180 
181  size_t available_bytes() const {
182  return ws_ptr->available_bytes();
183  }
184 
185  size_t allocated_bytes() const {
186  return ws_ptr->allocated_bytes();
187  }
188 
189  size_t reserved_bytes() const {
190  return ws_ptr->reserved_bytes();
191  }
192 
193  template<class Allocator>
194  void set_allocator(Allocator alloc) {
195  ws_ptr->set_allocator(alloc);
196  }
197 
198  ValueType* allocate(std::size_t sz) {
199  return ws_ptr->template allocate<ValueType>(sz);
200  }
201  void deallocate(ValueType* data, std::size_t sz) {
202  ws_ptr->template deallocate<ValueType>(data, sz);
203  }
204 };
205 
206 }
207 }
208 
209 
210 #endif /* WORKSPACE_H_ */
std::false_type propagate_on_container_copy_construction
Definition: stack_allocator.h:150
size_t available_bytes() const
Definition: stack_allocator.h:181
void free()
Definition: stack_allocator.h:58
~Stack_Allocator_Base()
Definition: stack_allocator.h:125
T * allocate(std::size_t sz)
Definition: stack_allocator.h:116
void set_allocator(const Allocator &alloc)
Definition: polymorphic_allocator.h:179
Byte * allocate(std::size_t sz)
Definition: stack_allocator.h:83
ValueType * allocate(std::size_t sz)
Definition: stack_allocator.h:198
void deallocate(Byte *data, std::size_t sz)
Definition: stack_allocator.h:98
Stack_Allocator(const Stack_Allocator< T, SystemTag > &ws)
Definition: stack_allocator.h:161
size_t reserved_bytes() const
Definition: stack_allocator.h:72
int size_t
Definition: common.h:283
void deallocate(ValueType *data, std::size_t sz)
Definition: stack_allocator.h:201
Definition: allocators.h:20
An unsynced memory pool implemented as a stack.
Definition: stack_allocator.h:138
void set_allocator(Allocator alloc)
Definition: stack_allocator.h:77
void force_deallocate()
Definition: stack_allocator.h:177
size_t allocated_bytes() const
Definition: stack_allocator.h:185
size_t allocated_bytes() const
Definition: stack_allocator.h:69
void free()
Delete all reserved memory, if memory is currently allocated an error is thrown.
Definition: stack_allocator.h:173
Definition: stack_allocator.h:19
size_t reserved_bytes() const
Definition: stack_allocator.h:189
void force_deallocate()
Definition: stack_allocator.h:111
void reserve(std::size_t sz)
Definition: stack_allocator.h:36
Definition: allocators.h:17
Stack_Allocator_Base(std::size_t sz=0)
Definition: stack_allocator.h:32
void deallocate(value_type *data, std::size_t sz)
Definition: polymorphic_allocator.h:174
#define BC_ASSERT(condition, message)
Definition: common.h:185
value_type * allocate(std::size_t sz)
Definition: polymorphic_allocator.h:170
void deallocate(T *data, std::size_t sz)
Definition: stack_allocator.h:121
Definition: common.h:26
Definition: stack_allocator.h:153
size_t available_bytes() const
Definition: stack_allocator.h:66
void set_allocator(Allocator alloc)
Definition: stack_allocator.h:194
Stack_Allocator(int sz=0)
Definition: stack_allocator.h:157
void reserve(std::size_t sz)
Reserve an amount of memory in bytes.
Definition: stack_allocator.h:168
The Evaluator determines if an expression needs to be greedily optimized.
Definition: algorithms.h:22