BlackCat_Tensors
A GPU-supported autograd and linear algebra library, designed for neural network construction
shape.h
Go to the documentation of this file.
1 /*
2  * Shape.h
3  *
4  * Created on: Sep 24, 2019
5  * Author: joseph
6  */
7 
8 #ifndef BLACKCATTENSORS_SHAPE_SHAPE_H_
9 #define BLACKCATTENSORS_SHAPE_SHAPE_H_
10 
11 #include "dim.h"
12 
13 namespace bc {
14 
15 
16 template<int N>
17 struct Shape {
18 
19  static_assert(N >= 0, "Shape<N>: ASSERT 'N >= 0'");
20 
21  template<int>
22  friend struct Shape;
23 
24  static constexpr int tensor_dim = N;
25  using size_t = bc::size_t;
26  using value_type = size_t;
27 
28 private:
29 
30  Dim<N> m_inner_shape = {0};
31  Dim<N> m_block_shape = {0};
32 
33 public:
34 
36 
37  template<
38  class... Integers,
39  class=std::enable_if_t<
40  bc::traits::sequence_of_v<size_t, Integers...> &&
41  (sizeof...(Integers) == N)>> BCINLINE
42  Shape(Integers... ints):
43  m_inner_shape {ints...} {
44 
45  m_block_shape[0] = 1;
46  for (int i = 1; i < N; ++i)
47  m_block_shape[i] = m_inner_shape[i-1] * m_block_shape[i-1];
48  }
49 
50  template<int X, class=std::enable_if_t<(X > N)>> BCINLINE
51  Shape(const Shape<X>& shape):
52  m_inner_shape(shape.m_inner_shape.template subdim<0, N>()),
53  m_block_shape(shape.m_block_shape.template subdim<0, N>()) {}
54 
55  BCINLINE
56  Shape(Dim<N> new_shape, const Shape<N>& parent_shape):
57  m_inner_shape(new_shape),
58  m_block_shape(parent_shape.m_block_shape) {}
59 
60  BCINLINE
61  Shape(const Shape<N>& new_shape, const Shape<N>& parent_shape):
62  m_inner_shape(new_shape.m_inner_shape),
63  m_block_shape(parent_shape.m_block_shape) {
64  }
65 
66  BCINLINE
67  Shape(Dim<N> dims):
68  m_inner_shape(dims.template subdim<0, N>()) {
69 
70  m_block_shape[0] = 1;
71  for (int i = 1; i < N; ++i) {
72  m_block_shape[i] = m_inner_shape[i-1] * m_block_shape[i-1];
73  }
74  }
75 
76  BCINLINE const auto& inner_shape() const { return m_inner_shape; }
77  BCINLINE const auto& outer_shape() const { return m_block_shape; }
78  BCINLINE size_t operator [] (size_t i) const { return m_inner_shape[i]; }
79  BCINLINE size_t size() const { return m_inner_shape.size(); }
80  BCINLINE size_t rows() const { return m_inner_shape[0]; }
81  BCINLINE size_t cols() const { return m_inner_shape[1]; }
82  BCINLINE size_t dim(int i) const { return m_inner_shape.dim(i); }
83  BCINLINE size_t outer_dim() const { return m_inner_shape.outer_dim(); }
84 
85  BCINLINE size_t leading_dim(int i=N-1) const {
86  return i < N ? m_block_shape[i] : 0;
87  }
88 
89  BCINLINE bool operator == (const Shape& other) const {
90  return m_inner_shape ==other.m_inner_shape;
91  }
92 
93  BCINLINE size_t coefficientwise_dims_to_index(size_t index) const {
94  return index;
95  }
96 
97  template<
98  class... Integers,
99  class=std::enable_if_t<
100  bc::traits::sequence_of_v<size_t, Integers...> &&
101  (sizeof...(Integers) >= N)>> BCINLINE
102  size_t dims_to_index(Integers... ints) const {
103  return dims_to_index(bc::dim(ints...));
104  }
105 
106  template<int D, class=std::enable_if_t<(D>=N)>> BCINLINE
107  size_t dims_to_index(const Dim<D>& var) const {
108  size_t index = var[D-1];
109  for(int i = 1; i < N; ++i) {
110  index += leading_dim(i) * var[D-1-i];
111  }
112  return index;
113  }
114 };
115 
116 template<>
117 struct Shape<0> {
118 
119  using size_t = bc::size_t;
121 
122  static constexpr int tensor_dim = 0;
123 
124  template<int>
125  friend struct Shape;
126 
128 
129  template<class... Args>
130  BCINLINE Shape<0>(const Args&...) {}
131 
132  BCINLINE Dim<0> inner_shape() const { return bc::Dim<0>(); }
133  BCINLINE size_t operator [] (size_t i) { return 1; }
134  BCINLINE size_t size() const { return 1; }
135  BCINLINE size_t rows() const { return 1; }
136  BCINLINE size_t cols() const { return 1; }
137  BCINLINE size_t dim(int i) const { return 1; }
138  BCINLINE size_t outer_dim() const { return 1; }
139  BCINLINE size_t leading_dim(int i=0) const { return 0; }
140  BCINLINE bool operator == (const Shape& other) const { return true; }
141 
142  BCINLINE
143  size_t coefficientwise_dims_to_index(size_t i) const {
144  return 0;
145  }
146 
147  template<class... Integers> BCINLINE
148  size_t dims_to_index(Integers... ints) const {
149  return 0;
150  }
151 };
152 
153 template<>
154 struct Shape<1> {
155 
156  using size_t = bc::size_t;
158 
159  static constexpr int tensor_dim = 1;
160 
161  template<int>
162  friend struct Shape;
163 
164 private:
165 
166  bc::Dim<1> m_inner_shape = {0};
167 
168 public:
169 
172  m_inner_shape {param} {}
173 
174  template<int X, class=std::enable_if_t<(X>=1)>> BCINLINE
175  Shape(const Shape<X>& shape) {
176  m_inner_shape[0] = shape.m_inner_shape[0];
177  }
178 
179  BCINLINE
180  Shape(int length):
181  m_inner_shape { length } {}
182 
183  BCINLINE size_t operator [] (size_t i) const { return dim(i); }
184  BCINLINE size_t size() const { return m_inner_shape[0]; }
185  BCINLINE size_t rows() const { return m_inner_shape[0]; }
186  BCINLINE size_t cols() const { return 1; }
187  BCINLINE size_t dim(size_t i) const { return i == 0 ? m_inner_shape[0] : 1; }
188  BCINLINE size_t outer_dim() const { return m_inner_shape[0]; }
189  BCINLINE size_t leading_dim(size_t i=0) const { return i == 0 ? 1 : 0; }
190  BCINLINE const auto& inner_shape() const { return m_inner_shape; }
191 
192  BCINLINE bool operator == (const Shape<1>& other) const {
193  return rows() == other.rows();
194  }
195 
196  template<class... Integers> BCINLINE
197  size_t dims_to_index(size_t i, Integers... ints) const {
198  return dims_to_index(ints...);
199  }
200 
201  template<class... Integers> BCINLINE
202  size_t dims_to_index(size_t i) const {
203  return i;
204  }
205 
206  BCINLINE size_t coefficientwise_dims_to_index(size_t i) const {
207  return i;
208  }
209 };
210 
211 
213 
214  static constexpr int tensor_dim = 1;
215 
216  bc::Dim<1> m_inner_shape = {0};
217  bc::Dim<1> m_block_shape = {1};
218 
219  BCINLINE Strided_Vector_Shape(size_t length, size_t leading_dim) {
220  m_inner_shape[0] = length;
221  m_block_shape[0] = leading_dim;
222  }
223 
224  BCINLINE bool operator == (const Strided_Vector_Shape& other) const {
225  return rows() == other.rows();
226  }
227 
228  BCINLINE friend bool operator == (
229  const Strided_Vector_Shape& shape, const Shape<1>& other) {
230  return shape.rows() == other.rows();
231  }
232 
233  BCINLINE size_t operator [] (size_t idx) const { return dim(idx); }
234  BCINLINE size_t size() const { return m_inner_shape[0]; }
235  BCINLINE size_t rows() const { return m_inner_shape[0]; }
236  BCINLINE size_t cols() const { return 1; }
237  BCINLINE size_t dim(int i) const { return i == 0 ? m_inner_shape[0] : 1; }
238  BCINLINE size_t outer_dim() const { return m_inner_shape[0]; }
239  BCINLINE size_t leading_dim(int i=0) const {
240  return i == 0 ? m_block_shape[0] : 0;
241  }
242 
243  BCINLINE const auto& inner_shape() const { return m_inner_shape; }
244 
245  template<class... Integers>
246  BCINLINE size_t dims_to_index(size_t i, Integers... ints) const {
247  return dims_to_index(ints...);
248  }
249 
250  template<class... Integers>
251  BCINLINE size_t dims_to_index(size_t index) const {
252  return m_block_shape[0] * index;
253  }
254 
255  BCINLINE size_t coefficientwise_dims_to_index(size_t index) const {
256  return m_block_shape[0] * index;
257  }
258 };
259 
260 template<
261  class... Integers,
262  typename=std::enable_if_t<
263  traits::sequence_of_v<size_t, Integers...>>>
264 BCINLINE auto shape(Integers... ints) {
265  return Shape<sizeof...(Integers)>(ints...);
266 }
267 
268 template<
269  class InnerShape,
270  typename=std::enable_if_t<!
271  traits::sequence_of_v<size_t, InnerShape>>>
272 BCINLINE auto shape(InnerShape is) {
274 }
275 
276 }
277 
278 #endif /* SHAPE_H_ */
BCINLINE size_t dim(int i) const
Definition: shape.h:82
BCINLINE size_t rows() const
Definition: shape.h:235
BCINLINE Strided_Vector_Shape(size_t length, size_t leading_dim)
Definition: shape.h:219
BCINLINE size_t outer_dim() const
Definition: shape.h:188
bc::size_t size_t
Definition: shape.h:25
BCINLINE size_t dims_to_index(const Dim< D > &var) const
Definition: shape.h:107
BCINLINE size_t rows() const
Definition: shape.h:80
#define BCINLINE
Definition: common.h:96
BCINLINE size_t leading_dim(int i=0) const
Definition: shape.h:239
Definition: shape.h:17
size_t value_type
Definition: shape.h:26
BCINLINE value_type outer_dim() const
Definition: dim.h:70
Definition: dim.h:294
BCINLINE value_type dim(size_t i, size_t default_value=1) const
Definition: dim.h:61
BCINLINE size_t dims_to_index(size_t index) const
Definition: shape.h:251
BCINLINE size_t dims_to_index(size_t i) const
Definition: shape.h:202
BCINLINE size_t size() const
Definition: shape.h:134
BCINLINE Shape(int length)
Definition: shape.h:180
BCINLINE size_t dims_to_index(Integers... ints) const
Definition: shape.h:102
static constexpr int tensor_dim
Definition: shape.h:24
BCINLINE size_t coefficientwise_dims_to_index(size_t i) const
Definition: shape.h:143
BCINLINE const auto & outer_shape() const
Definition: shape.h:77
size_t value_type
Definition: shape.h:120
Definition: shape.h:212
BCINLINE auto dim(const Integers &... ints)
Definition: dim.h:336
BCINLINE Shape(bc::Dim< 1 > param)
Definition: shape.h:171
BCINLINE size_t size() const
Definition: shape.h:79
BCINLINE const auto & inner_shape() const
Definition: shape.h:76
BCINLINE size_t outer_dim() const
Definition: shape.h:83
BCINLINE size_t size() const
Definition: shape.h:234
Definition: dim.h:17
BCINLINE Shape(Dim< N > dims)
Definition: shape.h:67
BCINLINE size_t outer_dim() const
Definition: shape.h:138
BCINLINE size_t coefficientwise_dims_to_index(size_t index) const
Definition: shape.h:255
Definition: shape.h:154
BCINLINE size_t dim(int i) const
Definition: shape.h:137
BCINLINE size_t leading_dim(int i=0) const
Definition: shape.h:139
BCINLINE value_type size() const
Definition: dim.h:28
int size_t
Definition: common.h:283
BCINLINE size_t dims_to_index(size_t i, Integers... ints) const
Definition: shape.h:246
BCINLINE Shape()
Definition: shape.h:35
BCINLINE Shape()
Definition: shape.h:170
BCINLINE Dim< 0 > inner_shape() const
Definition: shape.h:132
BCINLINE size_t rows() const
Definition: shape.h:135
BCINLINE Shape(const Shape< X > &shape)
Definition: shape.h:51
BCINLINE Shape(Integers... ints)
Definition: shape.h:42
BCINLINE size_t cols() const
Definition: shape.h:136
BCINLINE bool operator==(const Shape &other) const
Definition: shape.h:89
size_t value_type
Definition: shape.h:157
BCINLINE size_t cols() const
Definition: shape.h:186
BCINLINE size_t rows() const
Definition: shape.h:185
BCINLINE size_t dims_to_index(Integers... ints) const
Definition: shape.h:148
BCINLINE Shape(const Shape< N > &new_shape, const Shape< N > &parent_shape)
Definition: shape.h:61
BCINLINE const auto & inner_shape() const
Definition: shape.h:190
BCINLINE size_t size() const
Definition: shape.h:184
BCINLINE size_t operator[](size_t i) const
Definition: shape.h:78
BCINLINE size_t cols() const
Definition: shape.h:236
BCINLINE size_t coefficientwise_dims_to_index(size_t i) const
Definition: shape.h:206
BCINLINE size_t cols() const
Definition: shape.h:81
BCINLINE const auto & inner_shape() const
Definition: shape.h:243
BCINLINE Shape(Dim< N > new_shape, const Shape< N > &parent_shape)
Definition: shape.h:56
BCINLINE auto shape(Integers... ints)
Definition: shape.h:264
BCINLINE Shape(const Shape< X > &shape)
Definition: shape.h:175
BCINLINE size_t dim(size_t i) const
Definition: shape.h:187
BCINLINE size_t dims_to_index(size_t i, Integers... ints) const
Definition: shape.h:197
BCINLINE size_t coefficientwise_dims_to_index(size_t index) const
Definition: shape.h:93
BCINLINE size_t dim(int i) const
Definition: shape.h:237
BCINLINE size_t leading_dim(int i=N-1) const
Definition: shape.h:85
Definition: shape.h:117
BCINLINE size_t outer_dim() const
Definition: shape.h:238
BCINLINE size_t leading_dim(size_t i=0) const
Definition: shape.h:189
The Evaluator determines if an expression needs to be greedily optimized.
Definition: algorithms.h:22