Darwin Neuroevolution Framework
genotype.h
1 // Copyright 2018 The Darwin Neuroevolution Framework Authors.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #pragma once
16 
17 #include "cne.h"
18 
19 #include <core/darwin.h>
20 
21 namespace cne {
22 
23 template <class TRAITS>
24 class Genotype : public darwin::Genotype {
25  using HiddenLayerGene = typename TRAITS::HiddenLayerGene;
26  using OutputLayerGene = typename TRAITS::OutputLayerGene;
27 
28  public:
29  vector<HiddenLayerGene> hidden_layers;
30  OutputLayerGene output_layer;
31 
32  public:
33  Genotype() {
34  CHECK(g_inputs > 0);
35  CHECK(g_outputs > 0);
36  size_t prev_size = g_inputs;
37  for (size_t size : g_config.hidden_layers) {
38  hidden_layers.emplace_back(prev_size, size);
39  prev_size = size;
40  }
41  output_layer = { prev_size, g_outputs };
42  }
43 
44  void reset() override {
45  darwin::Genotype::reset();
46  }
47 
48  unique_ptr<darwin::Genotype> clone() const override {
49  return make_unique<Genotype>(*this);
50  }
51 
52  json save() const override {
53  json json_obj;
54  json_obj["hidden_layers"] = hidden_layers;
55  json_obj["output_layer"] = output_layer;
56  return json_obj;
57  }
58 
59  void load(const json& json_obj) override {
60  // load the genotype
61  Genotype tmp_genotype;
62  tmp_genotype.hidden_layers =
63  json_obj.at("hidden_layers").get<vector<HiddenLayerGene>>();
64  tmp_genotype.output_layer = json_obj.at("output_layer");
65 
66  // validate the genotype topology
67  size_t prev_size = g_inputs;
68  for (const auto& layer : tmp_genotype.hidden_layers) {
69  if (prev_size + 1 != layer.w.rows)
70  throw core::Exception("Can't load genotype, invalid topology");
71  prev_size = layer.w.cols;
72  if (prev_size == 0)
73  throw core::Exception("Can't load genotype, invalid topology");
74  }
75  if (prev_size + 1 != tmp_genotype.output_layer.w.rows)
76  throw core::Exception("Can't load genotype, invalid topology");
77  if (tmp_genotype.output_layer.w.cols != g_outputs)
78  throw core::Exception("Can't load genotype, invalid topology");
79 
80  // if everything went well, replace the genotype with the loaded one
81  std::swap(*this, tmp_genotype);
82  }
83 
84  unique_ptr<darwin::Brain> grow() const override;
85 
86  void inherit(const Genotype& parent1, const Genotype& parent2, float preference) {
87  // hidden layers
88  const size_t layers_count = hidden_layers.size();
89  CHECK(layers_count == parent1.hidden_layers.size());
90  CHECK(layers_count == parent2.hidden_layers.size());
91  for (size_t i = 0; i < layers_count; ++i) {
92  hidden_layers[i].crossover(
93  parent1.hidden_layers[i], parent2.hidden_layers[i], preference);
94  }
95 
96  // output layer
97  output_layer.crossover(parent1.output_layer, parent2.output_layer, preference);
98  }
99 
100  void mutate() {
101  for (auto& layer : hidden_layers) {
102  layer.mutate(ann::g_config.mutation_std_dev);
103  }
104  output_layer.mutate(ann::g_config.mutation_std_dev);
105  }
106 
107  void createPrimordialSeed() {
108  reset();
109  for (auto& layer : hidden_layers) {
110  layer.randomize();
111  }
112  output_layer.randomize();
113  }
114 };
115 
116 } // namespace cne
The base for exception types in the Darwin framework.
Definition: exception.h:27
Conventional Neuroevolution (CNE) populations.
Definition: brain.h:26
void reset(std::vector< T > &v)
Reset the values in a vector to 0.
Definition: ann_dynamic.h:32
The interface to the population-specific "genetic material", the Genotype
Definition: darwin.h:126