Darwin Neuroevolution Framework
genotype.h
1 // Copyright 2019 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 "cgp.h"
18 #include "functions.h"
19 
20 #include <core/darwin.h>
21 #include <core/properties.h>
22 #include <core/stringify.h>
23 
24 #include <array>
25 #include <cstdint>
26 #include <utility>
27 #include <vector>
28 using namespace std;
29 
30 namespace cgp {
31 
32 class Population;
33 
34 enum class MutationStrategy {
35  FixedCount,
36  Probabilistic,
37 };
38 
39 inline auto customStringify(core::TypeTag<MutationStrategy>) {
41  { MutationStrategy::FixedCount, "fixed_count" },
42  { MutationStrategy::Probabilistic, "probabilistic" },
43  };
44  return stringify;
45 }
46 
47 struct FixedCountMutation : public core::PropertySet {
48  PROPERTY(mutation_count, int, 2, "Number of mutations per genotype");
49 };
50 
51 struct ProbabilisticMutation : public core::PropertySet {
52  PROPERTY(connection_mutation_chance,
53  float,
54  0.05f,
55  "Probability of mutating a connection");
56  PROPERTY(function_mutation_chance,
57  float,
58  0.05f,
59  "Probability of mutating a node's function");
60  PROPERTY(output_mutation_chance,
61  float,
62  0.1f,
63  "Probability of mutating an output gene");
64  PROPERTY(constant_mutation_chance,
65  float,
66  0.1f,
67  "Probability of mutating an evolvable constant");
68 };
69 
70 struct MutationVariant : public core::PropertySetVariant<MutationStrategy> {
71  CASE(MutationStrategy::FixedCount, fixed_count, FixedCountMutation);
72  CASE(MutationStrategy::Probabilistic, probabilistic, ProbabilisticMutation);
73 };
74 
75 using IndexType = uint16_t;
76 
77 struct FunctionGene {
78  FunctionId function;
79  array<IndexType, kMaxFunctionArity> connections;
80 
81  friend void to_json(json& json_obj, const FunctionGene& gene);
82  friend void from_json(const json& json_obj, FunctionGene& gene);
83  friend bool operator==(const FunctionGene& a, const FunctionGene& b);
84 };
85 
86 struct OutputGene {
87  IndexType connection;
88 
89  friend void to_json(json& json_obj, const OutputGene& gene);
90  friend void from_json(const json& json_obj, OutputGene& gene);
91  friend bool operator==(const OutputGene& a, const OutputGene& b);
92 };
93 
94 class Genotype : public darwin::Genotype {
95  public:
96  explicit Genotype(const Population* population);
97 
98  unique_ptr<darwin::Brain> grow() const override;
99  unique_ptr<darwin::Genotype> clone() const override;
100 
101  json save() const override;
102  void load(const json& json_obj) override;
103  void reset() override;
104 
105  void createPrimordialSeed();
106  void probabilisticMutation(const ProbabilisticMutation& config);
107  void fixedCountMutation(const FixedCountMutation& config);
108  void inherit(const Genotype& parent1, const Genotype& parent2, float preference);
109 
110  const Population* population() const { return population_; }
111  const vector<FunctionGene>& functionGenes() const { return function_genes_; }
112  const vector<OutputGene>& outputGenes() const { return output_genes_; }
113 
114  float getEvolvableConstant(int function_id) const;
115 
116  friend bool operator==(const Genotype& a, const Genotype& b);
117 
118  private:
119  template <class PRED>
120  void mutationHelper(PRED& predicates);
121 
122  pair<IndexType, IndexType> connectionRange(int layer, int levels_back) const;
123 
124  private:
125  const Population* population_ = nullptr;
126 
127  vector<FunctionGene> function_genes_;
128  vector<OutputGene> output_genes_;
129  vector<float> constants_genes_;
130 };
131 
132 } // namespace cgp
void reset(std::vector< T > &v)
Reset the values in a vector to 0.
Definition: ann_dynamic.h:32
Cartesian Genetic Programming (CGP)
Definition: brain.cpp:25
STL namespace.
The foundation for data structures supporting runtime reflection.
Definition: properties.h:388
A variant type (tagged-union) with PropertySet fields.
Definition: properties.h:194
Handles types with a fixed, known set of values (enumerations for example)
Definition: stringify.h:85
const Stringify< T > * stringify()
Returns the stringifier for type T.
Definition: stringify.h:166
The interface to the population-specific "genetic material", the Genotype
Definition: darwin.h:126