BlackCat_Tensors
A GPU-supported autograd and linear algebra library, designed for neural network construction
array.h
Go to the documentation of this file.
1 /* Project: BlackCat_Tensors
2  * Author: JosephJaspers
3  * Copyright 2018
4  *
5  * This Source Code Form is subject to the terms of the Mozilla Public
6  * License, v. 2.0. If a copy of the MPL was not distributed with this
7  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
8 
9 #ifndef BC_EXPRESSION_TEMPLATES_ARRAY_H_
10 #define BC_EXPRESSION_TEMPLATES_ARRAY_H_
11 
13 #include "array_kernel_array.h"
14 #include "expression_binary.h"
15 
16 namespace bc {
17 namespace tensors {
18 namespace exprs {
19 
20 template<class, class, class, class...>
22 
23 
24 template<class Shape, class Scalar, class AllocatorType, class... Tags>
25 struct Array:
26  private AllocatorType,
27  public Kernel_Array<
28  Shape,
29  Scalar,
30  typename bc::allocator_traits<AllocatorType>::system_tag,
31  Tags...>
32 {
34  using value_type = Scalar;
35  using allocator_type = AllocatorType;
37  using shape_type = Shape;
38 
39 private:
40 
41  using parent_type = Kernel_Array<Shape, Scalar, system_tag, Tags...>;
43 
44  stream_type m_stream;
45 
46 public:
47 
48  const stream_type& get_stream() const { return m_stream; }
49  stream_type& get_stream() { return m_stream; }
50 
52  return static_cast<const allocator_type&>(*this);
53  }
54 
55  Array() {
56  if (Shape::tensor_dim == 0) {
57  this->m_data = get_allocator().allocate(1);
58  }
59  }
60 
61  Array(const Array& array):
63  select_on_container_copy_construction(array)),
65  m_stream(array.get_stream())
66  {
67  evaluate(
68  make_bin_expr<bc::oper::Assign>(
69  this->expression_template(),
70  array.expression_template()), get_stream());
71  }
72 
73  Array(Array&& array):
75  parent_type(array),
76  m_stream(array.get_stream())
77  {
78  array.m_data = nullptr;
79  }
80 
82  parent_type(shape, get_allocator()) {}
83 
85  allocator_type(allocator),
86  parent_type(shape, get_allocator()) {}
87 
88 
90  parent_type(shape, get_allocator()) {}
91 
92  template<
93  class... ShapeDims,
94  class=std::enable_if_t<
95  traits::sequence_of_v<bc::size_t, ShapeDims...> &&
96  sizeof...(ShapeDims) == Shape::tensor_dim>>
97  Array(const ShapeDims&... shape_dims):
98  parent_type(shape_type(shape_dims...), get_allocator()) {}
99 
100  //Shape-like object with maybe allocator
101  template<
102  class Expression,
103  class=std::enable_if_t<
106  Array(const Expression& expression, allocator_type allocator=allocator_type()):
107  allocator_type(allocator),
108  parent_type(
109  Shape(expression.inner_shape()),
110  get_allocator())
111  {
112  evaluate(
113  make_bin_expr<bc::oper::Assign>(
114  this->expression_template(),
115  expression.expression_template()), get_stream());
116  }
117 
118  //If Copy-constructing from a slice, attempt to query the allocator
119  //Restrict to same value_type (obviously), same dims (for fast-copy)
120  template<class AltShape, class... SliceTags>
122  const Array_Slice<
123  AltShape,
124  value_type,
126  SliceTags...>& expression):
128  select_on_container_copy_construction(expression.get_allocator())),
129  parent_type(Shape(expression.inner_shape()), get_allocator()),
130  m_stream(expression.get_stream())
131  {
132  evaluate(make_bin_expr<bc::oper::Assign>(
133  this->expression_template(), expression.expression_template()), get_stream());
134  }
135 
136 public:
137 
138  Array& operator = (Array&& array) {
139  if (allocator_traits_t::is_always_equal::value ||
140  array.get_allocator() == this->get_allocator())
141  {
142  std::swap((shape_type&)(*this), (shape_type&)array);
143  std::swap(this->m_data, array.m_data);
144 
145  if (allocator_traits_t::
146  propagate_on_container_move_assignment::value) {
147  (allocator_type&)(*this) = (allocator_type&&)(array);
148  }
149  } else {
150  get_allocator().deallocate(this->data(), this->size());
151  (shape_type&)(*this) = (shape_type&)array;
152  this->m_data = get_allocator().allocate(this->size());
153  evaluate(make_bin_expr<bc::oper::Assign>(
154  this->expression_template(), array.expression_template()), get_stream());
155  }
156  return *this;
157  }
158 
159 protected:
160 
161  void deallocate()
162  {
163  if (this->m_data) {
164  AllocatorType::deallocate(this->data(), this->size());
165  this->m_data= nullptr;
166  }
167  }
168 };
169 
170 
171 template<class Shape, class Allocator>
173 {
174  using value_type = typename Allocator::value_type;
176 }
177 
178 } //ns BC
179 } //ns exprs
180 } //ns tensors
181 
182 #endif /* SHAPE_H_ */
stream_type & get_stream()
Definition: array.h:49
Array(const Array &array)
Definition: array.h:61
Definition: shape.h:17
size_t value_type
Definition: shape.h:26
static constexpr int tensor_dim
Definition: shape.h:24
Definition: array_kernel_array.h:41
Shape shape_type
Definition: array.h:37
AllocatorType allocator_type
Definition: array.h:35
allocator_type get_allocator() const
Definition: array.h:51
void deallocate()
Definition: array.h:161
BCINLINE size_t size() const
Definition: shape.h:79
class::::::Args static auto swap(bc::streams::Stream< bc::host_tag > stream, Begin begin, End end, Args... args)
Definition: algorithms.h:140
BCINLINE const auto & inner_shape() const
Definition: shape.h:76
bc::traits::conditional_detected_t< bc::traits::query_system_tag, Allocator, host_tag > system_tag
Definition: allocator_traits.h:23
Array(const ShapeDims &... shape_dims)
Definition: array.h:97
Definition: dim.h:17
Array()
Definition: array.h:55
Definition: array.h:25
Definition: allocator_traits.h:20
Array(const Array_Slice< AltShape, value_type, allocator_type, SliceTags... > &expression)
Definition: array.h:121
int size_t
Definition: common.h:283
Array(bc::Dim< shape_type::tensor_dim > shape, allocator_type allocator)
Definition: array.h:84
Definition: expression_template_traits.h:76
BCINLINE Shape()
Definition: shape.h:35
Definition: allocators.h:20
const stream_type & get_stream() const
Definition: array.h:48
Array(shape_type shape)
Definition: array.h:89
Definition: array.h:21
BCINLINE const Kernel_Array< Shape, Scalar, bc::allocator_traits< AllocatorType >::system_tag, Tags... > & expression_template() const
Definition: expression_template_base.h:22
typename bc::allocator_traits< Allocator >::system_tag system_tag
Definition: array.h:33
Array(const Expression &expression, allocator_type allocator=allocator_type())
Definition: array.h:106
Tensor< 0, ValueType, Allocator > Scalar
Definition: tensors.h:44
Scalar value_type
Definition: array.h:34
BCINLINE auto shape(Integers... ints)
Definition: shape.h:264
Array(bc::Dim< shape_type::tensor_dim > shape)
Definition: array.h:81
Array(Array &&array)
Definition: array.h:73
Definition: device.h:27
Array & operator=(Array &&array)
Definition: array.h:138
The Evaluator determines if an expression needs to be greedily optimized.
Definition: algorithms.h:22
auto make_tensor_array(Shape shape, Allocator alloc)
Definition: array.h:172