BlackCat_Tensors
A GPU-supported autograd and linear algebra library, designed for neural network construction
layer_chain.h
Go to the documentation of this file.
1 #ifndef BLACKCAT_TUPLE
2 #define BLACKCAT_TUPLE
3 
4 #include "layer_manager.h"
5 
6 namespace bc {
7 namespace nn {
8 namespace detail {
9 
10 template<class T>
11 using query_layer_type = typename T::layer_type;
12 
13 }
14 
32 template<class Index, class Derived, class...>
33 struct LayerChain {};
34 
35 template<
36  class Index,
37  class Derived,
38  class CurrentLayer,
39  class... Layers>
40 struct LayerChain<
41  Index,
42  Derived,
43  CurrentLayer,
44  Layers...>:
45  LayerChain<
46  bc::traits::Integer<Index::value + 1>,
47  LayerChain<Index, Derived, CurrentLayer, Layers...>,
48  Layers...>,
50  LayerChain<Index, Derived, CurrentLayer, Layers...>,
51  CurrentLayer>
52 {
53  using self_type = LayerChain<
54  Index,
55  Derived,
56  CurrentLayer,
57  Layers...>;
58 
59  using parent_type = LayerChain<
61  self_type,
62  Layers...>;
63 
64  using layer_type = Layer_Manager<
65  self_type,
66  CurrentLayer>;
67 
70 
71  using type = CurrentLayer;
72  using value_type = typename CurrentLayer::value_type;
73  using system_tag = typename CurrentLayer::system_tag;
74 
76  using is_output_layer = bc::traits::truth_type<sizeof...(Layers)==0>;
79 
80  LayerChain(CurrentLayer f, Layers... layers):
81  parent_type(layers...),
82  layer_type(f) {}
83 
84  const auto& layer() const { return static_cast<const layer_type&>(*this); }
85  auto& layer() { return static_cast< layer_type&>(*this); }
86 
87  const auto& get(bc::traits::Integer<0>) const { return layer(); }
88  auto& get(bc::traits::Integer<0>) { return layer(); }
89 
90  template<int X>
91  const auto& get(bc::traits::Integer<X>) const {
92  return next().get(bc::traits::Integer<X-1>());
93  }
94 
95  template<int X>
97  return next().get(bc::traits::Integer<X-1>());
98  }
99 
100  const auto& head() const { return head_impl(is_input_layer()); }
101  auto& head() { return head_impl(is_input_layer()); }
102  const auto& tail() const { return tail_impl(is_output_layer()); }
103  auto& tail() { return tail_impl(is_output_layer()); }
104 
105  const auto& next() const { return next_impl(is_not_output_layer()); }
106  auto& next() { return next_impl(is_not_output_layer()); }
107  const auto& prev() const { return prev_impl(is_not_input_layer()); }
108  auto& prev() { return prev_impl(is_not_input_layer()); }
109 
110  template<class function> void for_each(function f) {
111  for_each_impl(f, is_not_output_layer());
112  }
113 
114  template<class function> void for_each(function f) const {
115  for_each_impl(f, is_not_output_layer());
116  }
117 
118  template<class function, class Arg>
119  auto for_each_propagate(function f, Arg&& arg) {
120  return for_each_propagate_impl(f, arg, is_not_output_layer());
121  }
122 
123  template<class function, class Arg>
124  auto for_each_propagate(function f, Arg&& arg) const {
125  return for_each_propagate_impl(f, arg, is_not_output_layer());
126  }
127 
128  template<class function, class Arg>
129  auto reverse_for_each_propagate(function f, Arg&& arg) {
130  return reverse_for_each_propagate_impl(f, arg, is_not_input_layer());
131  }
132 
133  template<class function, class Arg>
134  auto reverse_for_each_propagate(function f, Arg&& arg) const {
135  return reverse_for_each_propagate_impl(f, arg, is_not_input_layer());
136  }
137 
138  //------------------------ implementation --------------------------- //
139 private:
140 
141  template<int ADL=0> const auto& head_impl(std::true_type ) const { return *this; }
142  template<int ADL=0> auto& head_impl(std::true_type ) { return *this; }
143  template<int ADL=0> const auto& head_impl(std::false_type) const { return prev().head(); }
144  template<int ADL=0> auto& head_impl(std::false_type) { return prev().head(); }
145 
146  template<int ADL=0> const auto& tail_impl(std::true_type ) const { return *this; }
147  template<int ADL=0> auto& tail_impl(std::true_type ) { return *this; }
148  template<int ADL=0> const auto& tail_impl(std::false_type) const { return next().tail(); }
149  template<int ADL=0> auto& tail_impl(std::false_type) { return next().tail(); }
150 
151  template<int ADL=0> const auto& next_impl(std::true_type ) const { return static_cast<const parent_type&>(*this); }
152  template<int ADL=0> auto& next_impl(std::true_type ) { return static_cast<parent_type&>(*this); }
153  template<int ADL=0> const auto& next_impl(std::false_type) const { return *this; }
154  template<int ADL=0> auto& next_impl(std::false_type) { return *this; }
155 
156  template<int ADL=0> const auto& prev_impl(std::true_type ) const { return static_cast<const Derived&>(*this); }
157  template<int ADL=0> auto& prev_impl(std::true_type ) { return static_cast<Derived&>(*this); }
158  template<int ADL=0> const auto& prev_impl(std::false_type) const { return *this; }
159  template<int ADL=0> auto& prev_impl(std::false_type) { return *this; }
160 
161  template<class Function>
162  void for_each_impl(Function f, std::true_type) const {
163  f(layer());
164  this->next().for_each(f);
165  }
166 
167  template<class Function>
168  void for_each_impl(Function f, std::true_type) {
169  f(layer());
170  this->next().for_each(f);
171  }
172 
173  template<class Function>
174  void for_each_impl(Function f, std::false_type) const {
175  f(layer());
176  }
177 
178  template<class Function>
179  void for_each_impl(Function f, std::false_type) {
180  f(layer());
181  }
182 
183  template<class Function, class Arg>
184  auto for_each_propagate_impl(Function f, Arg&& arg, std::true_type) const {
185  return next().for_each_propagate(f, f(layer(), arg));
186  }
187 
188  template<class Function, class Arg>
189  auto for_each_propagate_impl(Function f, Arg&& arg, std::false_type) const {
190  return f(layer(), arg);
191  }
192 
193  template<class Function, class Arg>
194  auto for_each_propagate_impl(Function f, Arg&& arg, std::true_type) {
195  return next().for_each_propagate(f, f(layer(), arg));
196  }
197 
198  template<class Function, class Arg>
199  auto for_each_propagate_impl(Function f, Arg&& arg, std::false_type) {
200  return f(layer(), arg);
201  }
202 
203  template<class Function, class Arg>
204  auto reverse_for_each_propagate_impl(Function f, Arg&& arg, std::true_type) const {
205  return prev().reverse_for_each_propagate(f, f(layer(), arg));
206  }
207 
208  template<class Function, class Arg>
209  auto reverse_for_each_propagate_impl(Function f, Arg&& arg, std::true_type) {
210  return prev().reverse_for_each_propagate(f, f(layer(), arg));
211  }
212 
213  template<class Function, class Arg>
214  auto reverse_for_each_propagate_impl(Function f, Arg&& arg, std::false_type) {
215  return f(layer(), arg);
216  }
217 
218  template<class Function, class Arg>
219  auto reverse_for_each_propagate_impl(Function f, Arg&& arg, std::false_type) const {
220  return f(layer(), arg);
221  }
222 
223 };
224 }
225 }
226 #endif
LayerChain(CurrentLayer f, Layers... layers)
Definition: layer_chain.h:80
const auto & prev() const
Definition: layer_chain.h:107
CurrentLayer type
Definition: layer_chain.h:71
bc::traits::truth_type< sizeof...(Layers)==0 > is_output_layer
Definition: layer_chain.h:76
Definition: constexpr_int.h:14
const auto & head() const
Definition: layer_chain.h:100
typename CurrentLayer::value_type value_type
Definition: layer_chain.h:72
bc::traits::not_type< is_output_layer::value > is_not_output_layer
Definition: layer_chain.h:78
const auto & layer() const
Definition: layer_chain.h:84
bc::traits::truth_type< Index::value==0 > is_input_layer
Definition: layer_chain.h:75
bc::traits::not_type< is_input_layer::value > is_not_input_layer
Definition: layer_chain.h:77
conditional_t< cond, false_type, true_type > not_type
Definition: type_traits.h:52
auto for_each_propagate(function f, Arg &&arg) const
Definition: layer_chain.h:124
auto reverse_for_each_propagate(function f, Arg &&arg) const
Definition: layer_chain.h:134
Definition: type_traits.h:72
typename CurrentLayer::system_tag system_tag
Definition: layer_chain.h:73
const auto & tail() const
Definition: layer_chain.h:102
typename T::layer_type query_layer_type
Definition: layer_chain.h:11
auto for_each_propagate(function f, Arg &&arg)
Definition: layer_chain.h:119
Definition: layer_manager.h:17
auto reverse_for_each_propagate(function f, Arg &&arg)
Definition: layer_chain.h:129
Layer_Chain is an iterator-like object that connects different types of neural-network layers and def...
Definition: layer_chain.h:33
conditional_t< Bool, true_type, false_type > truth_type
Definition: type_traits.h:49
const auto & next() const
Definition: layer_chain.h:105
void for_each(function f) const
Definition: layer_chain.h:114
Definition: unaryfunction.h:16
void for_each(function f)
Definition: layer_chain.h:110
The Evaluator determines if an expression needs to be greedily optimized.
Definition: algorithms.h:22