LCOV - code coverage report
Current view: top level - models/sde_sirs - simulation.h (source / functions) Hit Total Coverage
Test: coverage.info Lines: 32 38 84.2 %
Date: 2025-01-17 12:16:22 Functions: 6 7 85.7 %

          Line data    Source code
       1             : /* 
       2             : * Copyright (C) 2020-2025 MEmilio
       3             : *
       4             : * Authors: Nils Wassmuth, Rene Schmieding, Martin J. Kuehn
       5             : *
       6             : * Contact: Martin J. Kuehn <Martin.Kuehn@DLR.de>
       7             : *
       8             : * Licensed under the Apache License, Version 2.0 (the "License");
       9             : * you may not use this file except in compliance with the License.
      10             : * You may obtain a copy of the License at
      11             : *
      12             : *     http://www.apache.org/licenses/LICENSE-2.0
      13             : *
      14             : * Unless required by applicable law or agreed to in writing, software
      15             : * distributed under the License is distributed on an "AS IS" BASIS,
      16             : * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
      17             : * See the License for the specific language governing permissions and
      18             : * limitations under the License.
      19             : */
      20             : #ifndef MIO_SDE_SIRS_SIMULATION_H
      21             : #define MIO_SDE_SIRS_SIMULATION_H
      22             : 
      23             : #include "memilio/compartments/flow_simulation.h"
      24             : #include "memilio/compartments/simulation.h"
      25             : #include "memilio/math/euler.h"
      26             : #include "sde_sirs/model.h"
      27             : 
      28             : namespace mio
      29             : {
      30             : namespace ssirs
      31             : {
      32             : 
      33             : /// @brief A specialized Simulation for mio::ssirs::Model.
      34             : class Simulation : public mio::Simulation<ScalarType, Model>
      35             : {
      36             : protected:
      37             :     using mio::Simulation<ScalarType, Model>::set_integrator;
      38             : 
      39             : public:
      40             :     /**
      41             :      * @brief Set up the simulation with an ODE solver.
      42             :      * @param[in] model An instance of mio::ssirs::Model.
      43             :      * @param[in] t0 Start time.
      44             :      * @param[in] dt Initial step size of integration.
      45             :      */
      46           9 :     Simulation(Model const& model, ScalarType t0 = 0., ScalarType dt = 0.1)
      47           9 :         : mio::Simulation<ScalarType, Model>(model, t0, dt)
      48             :     {
      49           9 :         auto integrator = std::make_shared<mio::EulerIntegratorCore<ScalarType>>();
      50           9 :         set_integrator(integrator);
      51          18 :     }
      52             : 
      53             :     using mio::Simulation<ScalarType, Model>::Simulation;
      54             :     /**
      55             :      * @brief advance simulation to tmax
      56             :      * tmax must be greater than get_result().get_last_time_point()
      57             :      * @param tmax next stopping point of simulation
      58             :      */
      59          18 :     Eigen::Ref<Eigen::VectorX<ScalarType>> advance(ScalarType tmax)
      60             :     {
      61          18 :         return get_ode_integrator().advance(
      62          18 :             [this](auto&& y, auto&& t, auto&& dydt) {
      63          18 :                 get_model().step_size = get_dt();
      64          18 :                 get_model().get_derivatives(y, y, t, dydt);
      65          18 :             },
      66          18 :             tmax, get_dt(), get_result());
      67             :     }
      68             : };
      69             : 
      70             : /**
      71             :  * @brief Run a Simulation of a mio::ssirs::Model.
      72             :  * @param[in] t0 Start time.
      73             :  * @param[in] tmax End time.
      74             :  * @param[in] dt Initial step size of integration.
      75             :  * @param[in] model An instance of mio::ssirs::Model.
      76             :  * @return A TimeSeries to represent the final simulation result
      77             :  */
      78           0 : inline TimeSeries<ScalarType> simulate(ScalarType t0, ScalarType tmax, ScalarType dt, Model const& model)
      79             : {
      80           0 :     model.check_constraints();
      81           0 :     Simulation sim(model, t0, dt);
      82           0 :     sim.advance(tmax);
      83           0 :     return sim.get_result();
      84           0 : }
      85             : 
      86             : /// @brief A specialized FlowSimulation for mio::ssirs::Model.
      87             : class FlowSimulation : public mio::FlowSimulation<ScalarType, Model>
      88             : {
      89             : protected:
      90             :     using mio::FlowSimulation<ScalarType, Model>::set_integrator;
      91             : 
      92             : public:
      93             :     /**
      94             :      * @brief Set up the simulation with an ODE solver.
      95             :      * @param[in] model An instance of mio::ssirs::Model.
      96             :      * @param[in] t0 Start time.
      97             :      * @param[in] dt Initial step size of integration.
      98             :      */
      99           9 :     FlowSimulation(Model const& model, ScalarType t0 = 0., ScalarType dt = 0.1)
     100           9 :         : mio::FlowSimulation<ScalarType, Model>(model, t0, dt)
     101             :     {
     102           9 :         auto integrator = std::make_shared<mio::EulerIntegratorCore<ScalarType>>();
     103           9 :         set_integrator(integrator);
     104          18 :     }
     105             : 
     106             :     /**
     107             :      * @brief Advance the simulation to tmax.
     108             :      * tmax must be greater than get_result().get_last_time_point().
     109             :      * @param[in] tmax Next stopping time of the simulation.
     110             :      */
     111          18 :     Eigen::Ref<Eigen::VectorX<ScalarType>> advance(ScalarType tmax)
     112             :     {
     113          18 :         assert(get_flows().get_num_time_points() == get_result().get_num_time_points());
     114          18 :         auto result = this->get_ode_integrator().advance(
     115             :             // see the general mio::FlowSimulation for more details on this DerivFunction
     116         126 :             [this](auto&& flows, auto&& t, auto&& dflows_dt) {
     117          18 :                 const auto& pop_result = get_result();
     118          18 :                 auto& model            = get_model();
     119             :                 // compute current population
     120          54 :                 model.get_derivatives(flows - get_flows().get_value(pop_result.get_num_time_points() - 1), m_pop);
     121          36 :                 m_pop += pop_result.get_last_value();
     122             :                 // compute the current change in flows with respect to the current population
     123          18 :                 dflows_dt.setZero();
     124          18 :                 model.step_size = get_dt(); // set the current step size
     125          54 :                 model.get_flows(m_pop, m_pop, t, dflows_dt);
     126          18 :             },
     127          18 :             tmax, get_dt(), get_flows());
     128          18 :         compute_population_results();
     129          36 :         return result;
     130             :     }
     131             : };
     132             : 
     133             : /**
     134             :  * @brief Run a FlowSimulation of mio::ssirs::Model.
     135             :  * @param[in] t0 Start time.
     136             :  * @param[in] tmax End time.
     137             :  * @param[in] dt Initial step size of integration.
     138             :  * @param[in] model An instance of mio::ssirs::Model.
     139             :  * @return The simulation result as two TimeSeries. The first describes the compartments at each time point,
     140             :  *         the second gives the corresponding flows that lead from t0 to each time point.
     141             :  */
     142             : inline std::vector<TimeSeries<ScalarType>> simulate_flows(ScalarType t0, ScalarType tmax, ScalarType dt,
     143             :                                                           Model const& model)
     144             : {
     145             :     model.check_constraints();
     146             :     FlowSimulation sim(model, t0, dt);
     147             :     sim.advance(tmax);
     148             :     return {sim.get_result(), sim.get_flows()};
     149             : }
     150             : 
     151             : } // namespace ssirs
     152             : } // namespace mio
     153             : 
     154             : #endif // MIO_SDE_SIRS_SIMULATION_H

Generated by: LCOV version 1.14