Darwin Neuroevolution Framework
evolution.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 "darwin.h"
18 #include "pubsub.h"
19 #include "thread_pool.h"
20 
21 #include <third_party/json/json.h>
22 using nlohmann::json;
23 
24 #include <chrono>
25 #include <memory>
26 #include <mutex>
27 #include <string>
28 #include <utility>
29 #include <vector>
30 using namespace std;
31 
32 namespace darwin {
33 
34 class Evolution;
35 
39  int generation = 0;
40 
42  float best_fitness = 0;
43 
45  float median_fitness = 0;
46 
48  float worst_fitness = 0;
49 
51  shared_ptr<core::PropertySet> calibration_fitness;
52 
54  shared_ptr<Genotype> champion;
55 
56  GenerationSummary() = default;
57 
58  GenerationSummary(const Population* population,
59  shared_ptr<core::PropertySet> calibration_fitness);
60 };
61 
65  int index = -1;
66 
68  float value = 0;
69 
70  CompressedFitnessValue(int index, float value) : index(index), value(value) {}
71 };
72 
74 enum class FitnessInfoKind {
75  SamplesOnly,
77  FullRaw,
78 };
79 
80 inline auto customStringify(core::TypeTag<FitnessInfoKind>) {
81  static auto stringify = new core::StringifyKnownValues<FitnessInfoKind>{
82  { FitnessInfoKind::SamplesOnly, "samples_only" },
83  { FitnessInfoKind::FullCompressed, "full_compressed" },
84  { FitnessInfoKind::FullRaw, "full_raw" },
85  };
86  return stringify;
87 }
88 
90 enum class ProfileInfoKind {
92  AllStages,
93 };
94 
95 inline auto customStringify(core::TypeTag<ProfileInfoKind>) {
96  static auto stringify = new core::StringifyKnownValues<ProfileInfoKind>{
97  { ProfileInfoKind::GenerationOnly, "generation_only" },
98  { ProfileInfoKind::AllStages, "all_stages" },
99  };
100  return stringify;
101 }
102 
105  PROPERTY(max_generations,
106  int,
107  1000000,
108  "Automatically stop the experiment after the max number of generations");
109 
110  PROPERTY(save_champion_genotype,
111  bool,
112  true,
113  "Save the best genotype from each generation");
114 
115  PROPERTY(fitness_information,
117  FitnessInfoKind::FullCompressed,
118  "What kind of fitness information to save");
119 
120  PROPERTY(save_genealogy,
121  bool,
122  false,
123  "Save the genealogy information (can be very large!)");
124 
125  PROPERTY(profile_information,
127  ProfileInfoKind::GenerationOnly,
128  "Performance trace (counters/timings)");
129 };
130 
131 vector<CompressedFitnessValue> compressFitness(const Population* population);
132 
135  using Clock = std::chrono::steady_clock;
136 
137  public:
139  enum Annotation : uint32_t {
140  Paused = 1,
141  Canceled = 2,
142  Generation = 4,
143  };
144 
145  public:
146  EvolutionStage() = default;
147 
149  EvolutionStage(const string& name_, size_t size_, uint32_t annotations);
150 
152  void start() { start_timestamp_ = Clock::now(); }
153 
155  void finish() { finish_timestamp_ = Clock::now(); }
156 
158  const string& name() const { return name_; }
159 
161  double elapsed() const;
162 
164  int progressPercent() const;
165 
167  const vector<EvolutionStage>& subStages() const { return sub_stages_; }
168 
169  void recordSubStage(const EvolutionStage& stage);
170  void advanceProgress(size_t increment);
171  void addAnnotations(uint32_t annotations);
172  uint32_t annotations() const { return annotations_; }
173 
174  friend void to_json(json& json_obj, const EvolutionStage& stage);
175 
176  private:
177  string name_;
178  size_t size_ = 0;
179  size_t progress_ = 0;
180  Clock::time_point start_timestamp_;
181  Clock::time_point finish_timestamp_;
182  uint32_t annotations_ = 0;
183  vector<EvolutionStage> sub_stages_;
184 };
185 
188  public:
189  EvolutionTrace(shared_ptr<const Experiment> experiment, const EvolutionConfig& config);
190 
192  int size() const;
193 
195  GenerationSummary generationSummary(int generation) const;
196 
197  GenerationSummary addGeneration(const Population* population,
198  shared_ptr<core::PropertySet> calibration_fitness,
199  const EvolutionStage& top_stage);
200 
201  private:
202  mutable mutex lock_;
203 
204  // a brief history of the generations
205  vector<GenerationSummary> generations_;
206 
207  shared_ptr<const Experiment> experiment_;
208  EvolutionConfig config_;
209 
210  unique_ptr<DbEvolutionTrace> db_trace_;
211 };
212 
220  public:
221  virtual ~ProgressMonitor() = default;
222 
230  virtual void beginStage(const string& name, size_t size, uint32_t annotations) = 0;
231 
233  virtual void finishStage(const string& name) = 0;
234 
239  virtual void reportProgress(size_t increment) = 0;
240 };
241 
251  public:
253  static void beginStage(const string& name, size_t size, uint32_t annotations) {
254  if (progress_monitor_ != nullptr) {
255  progress_monitor_->beginStage(name, size, annotations);
256  }
257  }
258 
260  static void finishStage(const string& name) {
261  if (progress_monitor_ != nullptr) {
262  progress_monitor_->finishStage(name);
263  }
264  }
265 
267  static void reportProgress(size_t increment = 1) {
268  if (progress_monitor_ != nullptr) {
269  progress_monitor_->reportProgress(increment);
270  }
271  }
272 
274  static void registerMonitor(ProgressMonitor* monitor) {
275  CHECK(progress_monitor_ == nullptr);
276  progress_monitor_ = monitor;
277  }
278 
279  private:
280  static ProgressMonitor* progress_monitor_;
281 };
282 
285  public pp::Controller,
286  public ProgressMonitor {
287  friend Evolution* evolution();
288 
289  public:
291  enum class State {
292  Invalid,
293  Initializing,
294  Running,
295  Pausing,
296  Paused,
297  Canceling,
298  Stopped
299  };
300 
302  enum EventFlag : uint32_t {
303  StateChanged = 1 << 0,
304  ProgressUpdate = 1 << 1,
305  EndGeneration = 1 << 2,
306  EndEvolution = 1 << 3,
307  NewExperiment = 1 << 4,
308  Reset = 1 << 5,
309  All = EventFlag(-1),
310  };
311 
314 
317 
320 
323  struct Snapshot {
325  State state = State::Invalid;
326 
328  int generation = 0;
329 
333 
336  shared_ptr<const Experiment> experiment;
337 
340  shared_ptr<const EvolutionTrace> trace;
341 
347  const Population* population = nullptr;
348 
354  const Domain* domain = nullptr;
355  };
356 
357  public:
358  static void init();
359 
367  bool newExperiment(shared_ptr<Experiment> experiment, const EvolutionConfig& config);
368 
370  Snapshot snapshot() const;
371 
373  const Experiment& experiment() const { return *experiment_; }
374 
376  const EvolutionConfig& config() const { return config_; }
377 
380  void run();
381 
386  void pause();
387 
392  bool reset();
393 
396  void waitForState(State target_state) const;
397 
398  private:
399  Evolution() {
400  pp::ParallelForSupport::init(this);
401  ProgressManager::registerMonitor(this);
402  }
403 
404  void mainThread();
405 
406  void evolutionCycle();
407 
408  // pp::Controller interface
409  void checkpoint() override;
410 
411  // ProgressMonitor interface
412  void beginStage(const string& name, size_t size, uint32_t annotations) override;
413  void finishStage(const string& name) override;
414  void reportProgress(size_t increment = 1) override;
415 
416  private:
417  std::thread::id main_thread_id_;
418 
419  mutable mutex lock_;
420  mutable condition_variable state_cv_;
421 
422  State state_ = State::Initializing;
423  vector<EvolutionStage> stage_stack_;
424 
425  EvolutionConfig config_;
426 
427  // population & domain
428  unique_ptr<Population> population_;
429  unique_ptr<Domain> domain_;
430 
431  shared_ptr<Experiment> experiment_;
432  shared_ptr<EvolutionTrace> trace_;
433 };
434 
436 inline Evolution* evolution() {
437  static Evolution* instance = new Evolution();
438  return instance;
439 }
440 
442 class StageScope {
443  public:
446  StageScope(const string& name, size_t size = 0, uint32_t annotations = 0)
447  : name_(name) {
448  ProgressManager::beginStage(name, size, annotations);
449  }
450 
452  ~StageScope() { ProgressManager::finishStage(name_); }
453 
454  private:
455  string name_;
456 };
457 
458 } // namespace darwin
static void finishStage(const string &name)
Reports the finish of a stage.
Definition: evolution.h:260
FitnessInfoKind
The kind of captured fitness data.
Definition: evolution.h:74
A population implementation encapsulates the fixed-size set of genotypes, together with the rules for...
Definition: darwin.h:161
~StageScope()
Finishes the stage.
Definition: evolution.h:452
Summary of a generation (fitness samples, best genotype, ...)
Definition: evolution.h:37
State snapshot of an evolution run.
Definition: evolution.h:323
EventFlag
Event hints (as bit flags)
Definition: evolution.h:302
void finish()
Marks the finish of the stage.
Definition: evolution.h:155
const vector< EvolutionStage > & subStages() const
List of sub-stages, if any.
Definition: evolution.h:167
STL namespace.
One fitness data point.
Definition: evolution.h:63
const string & name() const
Stage name.
Definition: evolution.h:158
Just the best/median/worst/calibration fitness values.
const EvolutionConfig & config() const
Accessor to the associated EvolutionConfig instance.
Definition: evolution.h:376
The controller for running evolution experiments.
Definition: evolution.h:284
void start()
Marks the start of the stage.
Definition: evolution.h:152
Interface for monitoring evolution progress.
Definition: evolution.h:219
Timings for the full stages tree for a generation.
A scope-based Stage wrapper.
Definition: evolution.h:442
static void reportProgress(size_t increment=1)
Reports stage progress.
Definition: evolution.h:267
core::PubSub< EvolutionStage > top_stages
Channel for publishing the completition of a generation&#39;s top stage.
Definition: evolution.h:319
State
Evolution state.
Definition: evolution.h:291
EvolutionStage stage
Currently running stage (most inner stage if there are nested stages)
Definition: evolution.h:332
Encapsulates the runtime experiment state.
Definition: darwin.h:328
Evolution * evolution()
Accessor to the Evolution singleton instance.
Definition: evolution.h:436
Annotation
Stage bit flags.
Definition: evolution.h:139
All the fitness values, raw.
The foundation for data structures supporting runtime reflection.
Definition: properties.h:388
Recording of a evolution experiment run.
Definition: evolution.h:187
core::PubSub< GenerationSummary > generation_summary
Channel for publishing generation summaries.
Definition: evolution.h:316
Key Darwin Neuroevolution Framework interfaces.
static void beginStage(const string &name, size_t size, uint32_t annotations)
Reports the start of a stage.
Definition: evolution.h:253
Classes derived from this are not copyable or movable.
Definition: utils.h:69
Connects the progress updates with a registered progress monitor.
Definition: evolution.h:250
shared_ptr< Genotype > champion
Best genotype in the generation.
Definition: evolution.h:54
shared_ptr< core::PropertySet > calibration_fitness
Calibration fitness values.
Definition: evolution.h:51
Interface to a domain implementation.
Definition: darwin.h:229
ProfileInfoKind
The kind of captured profile data.
Definition: evolution.h:90
static void registerMonitor(ProgressMonitor *monitor)
Registers a ProgressMonitor implementation.
Definition: evolution.h:274
core::PubSub< uint32_t > events
Evolution events notifications.
Definition: evolution.h:313
Handles types with a fixed, known set of values (enumerations for example)
Definition: stringify.h:85
Settings for an evolution experiment run.
Definition: evolution.h:104
All the fitness values, compressed.
Tracks the execution of an execution (sub)stage.
Definition: evolution.h:134
const Experiment & experiment() const
Accessor to the associted Experiment instance.
Definition: evolution.h:373
Just the per-generation elapsed timings.
An optional thread pool controller, which can be used to pause/resume/cancel the queued work items...
Definition: thread_pool.h:107
const Stringify< T > * stringify()
Returns the stringifier for type T.
Definition: stringify.h:166
shared_ptr< const Experiment > experiment
The associated Experiment instance.
Definition: evolution.h:336
shared_ptr< const EvolutionTrace > trace
The associated EvolutionTrace instance.
Definition: evolution.h:340
StageScope(const string &name, size_t size=0, uint32_t annotations=0)
Starts a new stage, which will be automatically completed at the end of the current scope...
Definition: evolution.h:446