Version: SMASH-3.2
clock.h
Go to the documentation of this file.
1 /*
2  *
3  * Copyright (c) 2014-2020,2022-2024
4  * SMASH Team
5  *
6  * GNU General Public License (GPLv3 or later)
7  *
8  */
9 
10 #ifndef SRC_INCLUDE_SMASH_CLOCK_H_
11 #define SRC_INCLUDE_SMASH_CLOCK_H_
12 
13 #include <algorithm>
14 #include <cmath>
15 #include <cstdint>
16 #include <cstdio>
17 #include <limits>
18 #include <stdexcept>
19 #include <vector>
20 
21 #include "logging.h"
22 #include "numeric_cast.h"
23 
24 namespace smash {
25 
26 static constexpr int LClock = LogArea::Clock::id;
27 
87 class Clock {
88  public:
90  using Representation = std::int64_t;
92  virtual double timestep_duration() const = 0;
94  virtual double current_time() const = 0;
96  virtual double next_time() const = 0;
104  virtual void reset(double start_time, bool is_output_clock) = 0;
111  virtual void remove_times_in_past(double start_time) = 0;
120  // guard against overflow:
121  if (counter_ >= std::numeric_limits<Representation>::max() - 1) {
122  throw std::overflow_error("Too many timesteps, clock overflow imminent");
123  }
124  ++counter_;
125  return *this;
126  }
127 
136  Clock& operator+=(Representation advance_several_timesteps) {
137  if (counter_ >= std::numeric_limits<Representation>::max() -
138  advance_several_timesteps) {
139  throw std::overflow_error("Too many timesteps, clock overflow imminent");
140  }
141  counter_ += advance_several_timesteps;
142  return *this;
143  }
144 
150  bool operator<(const Clock& rhs) const {
152  }
153 
159  bool operator<(double time) const { return present_internal_time() < time; }
160 
166  bool operator>(double time) const { return present_internal_time() > time; }
167 
168  virtual ~Clock() = default;
169 
170  protected:
182  virtual double present_internal_time() const = 0;
187 };
188 
210 class UniformClock : public Clock {
223  static constexpr double resolution = 0.000001;
224 
225  public:
227  UniformClock() = default;
235  UniformClock(double time, double dt, double time_end)
237  reset_time_(convert(time)),
238  time_end_(convert(time_end)) {
239  if (dt <= 0.) {
240  throw std::range_error("Time increment must be positive and non-zero");
241  }
242  if (reset_time_ >= time_end_) {
243  throw std::range_error(
244  "The initial time of UniformClock must be smaller than the end time. "
245  "(Attempt to set initial time to " +
246  std::to_string(time) + " and end time to " +
247  std::to_string(time_end) + " not possible)");
248  }
249  }
253  double current_time() const override {
254  auto present_time = present_internal_time();
255  // Do comparison in internal representation unit and return converted values
256  if (convert(present_time) > time_end_) {
257  return convert(time_end_);
258  } else {
259  return present_time;
260  }
261  }
270  double next_time() const override {
272  std::numeric_limits<Representation>::max() - timestep_duration_) {
273  throw std::overflow_error("Too many timesteps, clock overflow imminent");
274  }
275  auto next_point_in_time = reset_time_ + timestep_duration_ * (counter_ + 1);
276  if (next_point_in_time > time_end_) {
277  return convert(time_end_);
278  } else {
279  return convert(next_point_in_time);
280  }
281  }
282 
289  double timestep_duration() const override {
290  auto present_time = convert(present_internal_time());
291  if (present_time > time_end_) {
292  logg[LClock].warn() << "UniformClock asked for timestep duration beyond "
293  "end of simulation, returning 0.";
294  return 0.0;
295  } else if (present_time + timestep_duration_ > time_end_) {
296  return convert(time_end_ - present_time);
297  } else {
298  return convert(timestep_duration_);
299  }
300  }
306  void set_timestep_duration(double dt) {
307  if (dt <= 0.) {
308  throw std::range_error("Time increment must be positive and non-zero!");
309  }
311  counter_ = 0;
313  }
314 
321  void reset(double start_time, bool is_output_clock) override {
322  double reset_time;
323  if (is_output_clock) {
324  auto delta_t = convert(timestep_duration_);
325  reset_time = std::floor(start_time / delta_t) * delta_t;
326  } else {
327  reset_time = start_time;
328  }
329  if (reset_time < current_time()) {
330  logg[LClock].debug("Resetting clock from", current_time(), " fm to ",
331  reset_time, " fm");
332  }
333  reset_time_ = convert(reset_time);
334  counter_ = 0;
335  }
336 
337  void remove_times_in_past(double) override{};
338 
347  template <typename T>
348  typename std::enable_if<std::is_floating_point<T>::value, Clock&>::type
349  operator+=(T big_timestep) {
350  if (big_timestep < 0.) {
351  throw std::range_error("The clock cannot be turned back.");
352  }
353  reset_time_ += convert(big_timestep);
354  return *this;
355  }
364  Clock& operator+=(Representation advance_several_timesteps) {
365  if (counter_ >= std::numeric_limits<Representation>::max() -
366  advance_several_timesteps) {
367  throw std::overflow_error("Too many timesteps, clock overflow imminent");
368  }
369  counter_ += advance_several_timesteps;
370  return *this;
371  }
372 
373  protected:
379  double present_internal_time() const override {
381  }
382 
383  private:
385  static constexpr double to_double = resolution;
387  static constexpr double from_double = 1. / resolution;
388 
390  static Representation convert(double x) {
391  return numeric_cast<Representation>(std::round(x * from_double));
392  }
394  static double convert(Representation x) { return x * to_double; }
395 
402 };
403 
405 class CustomClock : public Clock {
406  public:
412  explicit CustomClock(std::vector<double> times) : custom_times_(times) {
413  std::sort(custom_times_.begin(), custom_times_.end());
414  counter_ = -1;
415  }
416 
423  double current_time() const override {
424  if (counter_ == -1) {
425  // current time before the first output should be the starting time
426  return start_time_;
427  } else if (counter_ < -1) {
428  throw std::runtime_error(
429  "Trying to access time of clock in invalid state.");
430  } else {
431  return custom_times_.at(counter_);
432  }
433  }
434 
439  double next_time() const override { return custom_times_.at(counter_ + 1); }
440 
442  double timestep_duration() const override {
443  return next_time() - current_time();
444  }
445 
453  void reset(double start_time, bool) override {
454  counter_ = -1;
455  start_time_ = start_time;
456  }
457 
463  void remove_times_in_past(double start_time) override {
464  custom_times_.erase(
465  std::remove_if(custom_times_.begin(), custom_times_.end(),
466  [start_time](double t) {
467  if (t <= start_time) {
468  logg[LClock].warn("Removing custom output time ", t,
469  " fm since it is earlier than the "
470  "starting time of the simulation");
471  return true;
472  } else {
473  return false;
474  }
475  }),
476  custom_times_.end());
477  }
478 
479  protected:
486  double present_internal_time() const override { return current_time(); }
487 
488  private:
490  std::vector<double> custom_times_;
492  double start_time_ = 0.;
493 };
494 } // namespace smash
495 
496 #endif // SRC_INCLUDE_SMASH_CLOCK_H_
Clock tracks the time in the simulation.
Definition: clock.h:87
bool operator<(double time) const
Compares the internal time of the clock against a fixed time.
Definition: clock.h:159
virtual ~Clock()=default
bool operator<(const Clock &rhs) const
Compares the internal times of two clocks.
Definition: clock.h:150
std::int64_t Representation
The type used for counting ticks/time.
Definition: clock.h:90
virtual double current_time() const =0
virtual double timestep_duration() const =0
virtual double present_internal_time() const =0
This function always returns the clock time, even if children might attribute a different behaviour t...
virtual void reset(double start_time, bool is_output_clock)=0
Reset the clock to the starting time of the simulation.
Representation counter_
Internally used to count the number of time steps.
Definition: clock.h:186
bool operator>(double time) const
Compares the internal time of the clock against a fixed time.
Definition: clock.h:166
Clock & operator++()
Advances the clock by one tick.
Definition: clock.h:119
virtual double next_time() const =0
virtual void remove_times_in_past(double start_time)=0
Remove output times before the starting time of the simulation if this is a custom clock.
Clock & operator+=(Representation advance_several_timesteps)
Advances the clock by an arbitrary number of ticks.
Definition: clock.h:136
Clock with explicitly defined time steps.
Definition: clock.h:405
double next_time() const override
Definition: clock.h:439
CustomClock(std::vector< double > times)
Initialises a custom clock with explicitly given output times.
Definition: clock.h:412
double start_time_
Starting time of the simulation.
Definition: clock.h:492
void remove_times_in_past(double start_time) override
Remove all custom times before start_time.
Definition: clock.h:463
std::vector< double > custom_times_
Vector of times where output is generated.
Definition: clock.h:490
void reset(double start_time, bool) override
Reset the clock to the starting time of the simulation.
Definition: clock.h:453
double present_internal_time() const override
For the CustomClock, the internal time is basically by design the same as what the current_time() met...
Definition: clock.h:486
double timestep_duration() const override
Definition: clock.h:442
double current_time() const override
Definition: clock.h:423
Clock with uniformly spaced time steps.
Definition: clock.h:210
void reset(double start_time, bool is_output_clock) override
Resets the time to the starting time of an event.
Definition: clock.h:321
static constexpr double from_double
A multiplier transferring the real time to the internal integer.
Definition: clock.h:387
void set_timestep_duration(double dt)
Sets the time step size (and resets the counter).
Definition: clock.h:306
static constexpr double to_double
A multiplier transferring the internal integer to the real time.
Definition: clock.h:385
Representation timestep_duration_
The time step size in .
Definition: clock.h:397
void remove_times_in_past(double) override
Remove output times before the starting time of the simulation if this is a custom clock.
Definition: clock.h:337
UniformClock(double time, double dt, double time_end)
Initialize with base time and time step size.
Definition: clock.h:235
double next_time() const override
Definition: clock.h:270
static constexpr double resolution
Defines the resolution of the clock (namely the smallest representable time difference).
Definition: clock.h:223
Representation time_end_
The end time of the particle propagation.
Definition: clock.h:401
static Representation convert(double x)
Convert a double x into the internal int representation.
Definition: clock.h:390
Representation reset_time_
The time of last reset (when counter_ was set to 0).
Definition: clock.h:399
UniformClock()=default
default initializer: Timestep size is set to 0!
double timestep_duration() const override
Definition: clock.h:289
std::enable_if< std::is_floating_point< T >::value, Clock & >::type operator+=(T big_timestep)
Advances the clock by an arbitrary timestep (multiple of 0.000001 fm).
Definition: clock.h:349
Clock & operator+=(Representation advance_several_timesteps)
advances the clock by an arbitrary number of ticks.
Definition: clock.h:364
static double convert(Representation x)
Convert an internal int value x into the double representation.
Definition: clock.h:394
double current_time() const override
Definition: clock.h:253
double present_internal_time() const override
Access the internal time of the clock, independently from the end time.
Definition: clock.h:379
std::array< einhard::Logger<>, std::tuple_size< LogArea::AreaTuple >::value > logg
An array that stores all pre-configured Logger objects.
Definition: logging.cc:40
Definition: action.h:24
static constexpr int LClock
Definition: clock.h:26