BlackCat_Tensors
A GPU-supported autograd and linear algebra library, designed for neural network construction
dim.h
Go to the documentation of this file.
1 /*
2  * Dim.h
3  *
4  * Created on: Sep 26, 2019
5  * Author: joseph
6  */
7 
8 #ifndef BLACKCATTENSORS_SHAPE_DIM_H_
9 #define BLACKCATTENSORS_SHAPE_DIM_H_
10 
11 #include <string>
12 #include <ostream>
13 
14 namespace bc {
15 
16 template<int N>
17 struct Dim {
18 
19  static_assert(N>=0, "Dim<N>: ASSERT 'N>=0'");
20 
21  static constexpr int tensor_dim = N;
23  using size_t = bc::size_t;
24 
25  bc::size_t m_index[N] = { 0 };
26 
27  BCINLINE
28  value_type size() const {
29  return prod();
30  }
31 
32  BCINLINE
33  Dim& fill(bc::size_t value) {
34  for (int i = 0; i < N; ++i)
35  m_index[i] = value;
36  return *this;
37  }
38 
39  BCINLINE
40  const value_type* data() const {
41  return m_index;
42  }
43 
44  BCINLINE
46  return m_index;
47  }
48 
50  BCINLINE
51  const value_type& operator [](int i) const {
52  return m_index[i];
53  }
54 
55  BCINLINE
57  return m_index[i];
58  }
59 
60  BCINLINE
61  value_type dim(size_t i, size_t default_value=1) const {
62  //casting to unsigned ensures that
63  //negative values do not lead to undefined behavior
64  return static_cast<const unsigned&>(i) < N
65  ? m_index[i]
66  : default_value;
67  }
68 
69  BCINLINE
71  return this->dim(N-1);
72  }
73 
74  BCINLINE
75  bool operator == (const Dim& other) const {
76  for (int i = 0; i < N; ++i) {
77  if (other[i] != this->m_index[i])
78  return false;
79  }
80  return true;
81  }
82 
83  BCINLINE
84  bool operator != (const Dim& other) const {
85  return !(*this == other);
86  }
87 
88  template<int X> BCINLINE
89  bool operator == (const Dim<X>& other) const {
90  return false;
91  }
92 
93  template<int X> BCINLINE
94  bool operator != (const Dim<X>& other) const {
95  return true;
96  }
97 
98 
99 
100  template<
101  class... Ints,
102  class=std::enable_if_t<
103  bc::traits::sequence_of_v<bc::size_t, Ints...>>>
104  BCINLINE
105  auto concat(Ints... value) const {
106  return concat(bc::Dim<sizeof...(Ints)> { value... });
107  }
108 
109  template <int X> BCINLINE
110  Dim<N+X> concat(const Dim<X>& other) const {
111  Dim<N+X> concat_dim;
112 
113  for (int i = 0; i < N; ++i)
114  concat_dim[i] = m_index[i];
115 
116  for (int i = 0; i < X; ++i)
117  concat_dim[i+N] = other[i];
118 
119  return concat_dim;
120  }
121 
122  BCINLINE auto begin() const { return m_index; }
123  BCINLINE auto end() const { return m_index + N; }
124  BCINLINE auto begin() { return m_index; }
125  BCINLINE auto end() { return m_index + N; }
126 
127 private:
128 
129  template<class Operator> BCINLINE
130  Dim op_impl(Operator op, const Dim& other) const {
131  Dim<N> result;
132  for (int i = 0; i < N; ++i) {
133  result[i] = op(m_index[i], other[i]);
134  }
135  return result;
136  }
137 
138  template<class Operator> BCINLINE
139  Dim& inplace_op_impl(Operator op, const Dim& other) {
140  for (int i = 0; i < N; ++i) {
141  m_index[i] = op(m_index[i], other[i]);
142  }
143  return *this;
144  }
145 
146  template<class Operator> BCINLINE
147  Dim scalar_op_impl(Operator op, const value_type& other) const {
148  Dim<N> result;
149  for (int i = 0; i < N; ++i) {
150  result[i] = op(m_index[i], other);
151  }
152  return result;
153  }
154 
155  template<class Operator> BCINLINE
156  Dim& inplace_scalar_op_impl(Operator op, const value_type& other) {
157  for (int i = 0; i < N; ++i) {
158  m_index[i] = op(m_index[i], other);
159  }
160  return *this;
161  }
162 
163 public:
164 
165 
166 #define BC_DIM_OP(op, functor)\
167  Dim operator op(const Dim& other) const { \
168  return this->op_impl(bc::oper::functor(), other); \
169  }
170 
171 #define BC_DIM_INPLACE_OP(op, functor)\
172  Dim operator op##=(const Dim& other) { \
173  return this->inplace_op_impl(bc::oper::functor(), other); \
174  }
175 
176 #define BC_DIM_INPLACE_SCALAR_OP(op, functor) \
177  friend Dim operator op##=(Dim &dim, const value_type& scalar) { \
178  return dim.inplace_scalar_op_impl(bc::oper::functor(), scalar); \
179  } \
180 
181 #define BC_DIM_SCALAR_OP(op, functor) \
182  friend Dim operator op(const Dim &dim, const value_type& scalar) { \
183  return dim.scalar_op_impl(bc::oper::functor(), scalar); \
184  } \
185  \
186  friend Dim operator op(const value_type& scalar, const Dim &dim) { \
187  return dim.scalar_op_impl(bc::oper::functor(), scalar); \
188  }
189 
190 #define BC_DIM_OP_FACTORY(op, functor) \
191 BC_DIM_OP(op, functor) \
192 BC_DIM_SCALAR_OP(op, functor)
193 
194 #define BC_DIM_OP_BOTH(op, functor) \
195 BC_DIM_OP_FACTORY(op, functor) \
196 BC_DIM_INPLACE_OP(op, functor) \
197 BC_DIM_INPLACE_SCALAR_OP(op, functor)
198 
204 BC_DIM_OP_FACTORY(<=, Lesser_Equal)
205 BC_DIM_OP_FACTORY(>, Greater)
206 BC_DIM_OP_FACTORY(>=, Greater_Equal)
207 
208 #undef BC_DIM_OP
209 #undef BC_DIM_INPLACE_OP
210 #undef BC_DIM_INPLACE_SCALAR_OP
211 #undef BC_DIM_SCALAR_OP
212 #undef BC_DIM_OP_FACTORY
213 #undef BC_DIM_OP_BOTH
214 
215  BCINLINE
216  bool all(size_t start, size_t end) const {
217  for (; start<end; ++start)
218  if (m_index[start] == 0)
219  return false;
220  return true;
221  }
222 
223  BCINLINE
224  bool all(size_t end=N) const {
225  return all(0, end);
226  }
227 
228 
229  BCINLINE
230  value_type sum(size_t start, size_t end) const {
231  value_type s = 0;
232  for (; start<end; ++start)
233  s *= m_index[start];
234  return s;
235  }
236 
237  BCINLINE
238  value_type sum(size_t end=N) const {
239  return sum(0, end);
240  }
241 
242 
243  BCINLINE
244  value_type prod(size_t start, size_t end) const {
245  value_type p = 1;
246  for (; start<end; ++start)
247  p *= m_index[start];
248  return p;
249  }
250 
251  BCINLINE
252  value_type prod(size_t end=N) const {
253  return prod(0, end);
254  }
255 
256  BCINLINE
257  Dim reversed() const {
258  Dim rev;
259  for (int i = 0; i < N; ++i)
260  rev[i] = m_index[N-1-i];
261 
262  return rev;
263  }
264 
265  template<int Start, int End=N> BCINLINE
266  Dim<End-Start> subdim() const {
267  return *(reinterpret_cast<const Dim<End-Start>*>(m_index + Start));
268  }
269 
270  std::string to_string(int begin, int end) const {
271  std::string str= "[";
272  while (begin < end) {
273  str += std::to_string(m_index[begin++]);
274 
275  if (begin != end)
276  str += ", ";
277  }
278 
279  return str + "]";
280  }
281 
282  std::string to_string(int end=N) const {
283  return to_string(0, end);
284  }
285 
286  friend std::ostream& operator << (std::ostream& os, const Dim& self) {
287  return os << self.to_string();
288  }
289 };
290 
291 
292 
293 template<>
294 struct Dim<0> {
295  static constexpr int tensor_dim = 0;
297 
298  BCINLINE
299  value_type size(value_type base_sz=1, value_type base_index=0) const {
300  return 1;
301  }
302 
304  BCINLINE
305  value_type operator [](int i) const {
306  return 1;
307  }
308 
309  BCINLINE
310  value_type dim(bc::size_t i) const {
311  return 1;
312  }
313 
314  BCINLINE
316  return 1;
317  }
318 
319  BCINLINE bool operator == (const Dim& other) const {
320  return true;
321  }
322  BCINLINE bool operator != (const Dim& other) const {
323  return false;
324  }
325 
326  std::string to_string() const {
327  return "[0]";
328  }
329 
330  friend std::ostream& operator << (std::ostream& os, const Dim& self) {
331  return os << self.to_string();
332  }
333 };
334 
335 template<class... Integers> BCINLINE
336 auto dim(const Integers&... ints) {
337  return Dim<sizeof...(Integers)> { bc::size_t(ints)... };
338 }
339 
340 
341 }
342 
343 
344 
345 #endif /* DIM_H_ */
bc::size_t m_index[N]
Definition: dim.h:25
#define BCINLINE
Definition: common.h:96
static constexpr int tensor_dim
Definition: dim.h:21
BCINLINE value_type outer_dim() const
Definition: dim.h:315
BCINLINE Dim< N+X > concat(const Dim< X > &other) const
Definition: dim.h:110
std::string to_string(int begin, int end) const
Definition: dim.h:270
BCINLINE value_type outer_dim() const
Definition: dim.h:70
BCINLINE value_type dim(size_t i, size_t default_value=1) const
Definition: dim.h:61
#define BC_DIM_OP_FACTORY(op, functor)
Definition: dim.h:190
BCINLINE Dim< End-Start > subdim() const
Definition: dim.h:266
BCINLINE Dim & fill(bc::size_t value)
Definition: dim.h:33
BCINLINE value_type sum(size_t start, size_t end) const
Definition: dim.h:230
#define BC_DIM_OP_BOTH(op, functor)
Definition: dim.h:194
std::string to_string(int precision=8, bool pretty=true, bool sparse=false) const
Definition: tensor_utility.h:34
BCINLINE bool operator==(const Dim &other) const
Definition: dim.h:75
BCINLINE value_type sum(size_t end=N) const
Definition: dim.h:238
BCINLINE auto end() const
Definition: dim.h:123
Definition: dim.h:17
bc::size_t value_type
Definition: dim.h:296
BCINLINE value_type size() const
Definition: dim.h:28
BCINLINE Dim reversed() const
Definition: dim.h:257
BCINLINE value_type size(value_type base_sz=1, value_type base_index=0) const
Definition: dim.h:299
int size_t
Definition: common.h:283
BCINLINE const value_type & operator[](int i) const
unchecked version of dim
Definition: dim.h:51
BCINLINE bool all(size_t start, size_t end) const
Definition: dim.h:216
BCINLINE bool all(size_t end=N) const
Definition: dim.h:224
BCINLINE auto begin() const
Definition: dim.h:122
Definition: common.h:24
BCINLINE value_type prod(size_t end=N) const
Definition: dim.h:252
BCINLINE auto begin()
Definition: dim.h:124
BCINLINE auto end()
Definition: dim.h:125
bc::size_t value_type
Definition: dim.h:22
std::string to_string() const
Definition: dim.h:326
BCINLINE value_type * data()
Definition: dim.h:45
std::string to_string(int end=N) const
Definition: dim.h:282
friend std::ostream & operator<<(std::ostream &os, const Dim &self)
Definition: dim.h:286
BCINLINE value_type dim(bc::size_t i) const
Definition: dim.h:310
BCINLINE const value_type * data() const
Definition: dim.h:40
BCINLINE auto concat(Ints... value) const
Definition: dim.h:105
The Evaluator determines if an expression needs to be greedily optimized.
Definition: algorithms.h:22
BCINLINE bool operator!=(const Dim &other) const
Definition: dim.h:84
BCINLINE value_type prod(size_t start, size_t end) const
Definition: dim.h:244