From 8b4ca54c744f54ea07894692623876a39c08ce81 Mon Sep 17 00:00:00 2001 From: Timo Koch <timo.koch@iws.uni-stuttgart.de> Date: Wed, 1 Aug 2018 12:21:25 +0200 Subject: [PATCH] [timeloop] Fix negative time indices. Add timeloop test. --- dumux/common/timeloop.hh | 116 ++++++++++++++----- test/common/CMakeLists.txt | 1 + test/common/timeloop/CMakeLists.txt | 1 + test/common/timeloop/test_timeloop.cc | 153 ++++++++++++++++++++++++++ 4 files changed, 242 insertions(+), 29 deletions(-) create mode 100644 test/common/timeloop/CMakeLists.txt create mode 100644 test/common/timeloop/test_timeloop.cc diff --git a/dumux/common/timeloop.hh b/dumux/common/timeloop.hh index 446866cfaa..1c31e6ecf8 100644 --- a/dumux/common/timeloop.hh +++ b/dumux/common/timeloop.hh @@ -31,6 +31,7 @@ #include <dune/common/timer.hh> #include <dune/common/parallel/collectivecommunication.hh> #include <dune/common/parallel/mpihelper.hh> +#include <dune/common/exceptions.hh> #include <dumux/common/parameters.hh> @@ -134,9 +135,13 @@ public: endTime_ = tEnd; timeStepSize_ = dt; + lastTimeStepSize_ = 0.0; maxTimeStepSize_ = std::numeric_limits<Scalar>::max(); + userSetMaxTimeStepSize_ = maxTimeStepSize_; timeStepIdx_ = 0; finished_ = false; + timeAfterLastTimeStep_ = 0.0; + timeStepWallClockTime_ = 0.0; timer_.stop(); timer_.reset(); @@ -149,6 +154,12 @@ public: { timeStepIdx_++; time_ += timeStepSize_; + lastTimeStepSize_ = timeStepSize_; + + // compute how long the last time step took + const auto cpuTime = wallClockTime(); + timeStepWallClockTime_ = cpuTime - timeAfterLastTimeStep_; + timeAfterLastTimeStep_ = cpuTime; } /*! @@ -189,7 +200,11 @@ public: * \param t The time \f$\mathrm{[s]}\f$ at which the simulation is finished */ void setEndTime(Scalar t) - { endTime_ = t; } + { + endTime_ = t; + if (verbose_) + std::cout << "Set new end time to t = " << t << " seconds." << std::endl; + } /*! * \brief Returns the current wall clock time (cpu time) spend in this time loop @@ -223,8 +238,9 @@ public: void setMaxTimeStepSize(Scalar maxDt) { using std::min; - maxTimeStepSize_ = maxDt; - timeStepSize_ = min(timeStepSize_, maxDt); + userSetMaxTimeStepSize_ = maxDt; + computeMaxTimeStepSize_(); + timeStepSize_ = min(timeStepSize_, maxTimeStepSize_); } /*! @@ -280,27 +296,28 @@ public: * and other possible check points */ Scalar maxTimeStepSize() const - { return maxTimeStepSize_; } + { + using std::min; + return min(userSetMaxTimeStepSize_, maxTimeStepSize_); + } /*! * \brief State info on cpu time. + * \note Always call this after TimeLoop::advanceTimeStep() */ - void reportTimeStep() + void reportTimeStep() const { - auto timeStepCpuTime = timer_.elapsed(); - cpuTime_ += timeStepCpuTime; + const auto cpuTime = wallClockTime(); if (verbose_) { std::cout << "Time step " << timeStepIdx_ << " done in " - << timeStepCpuTime << " seconds. " - << "Wall time: " << cpuTime_ + << timeStepWallClockTime_ << " seconds. " + << "Wall clock time: " << cpuTime << ", time: " << time_ - << ", time step size: " << timeStepSize_ + << ", time step size: " << lastTimeStepSize_ << std::endl; } - - timer_.reset(); } /*! @@ -309,21 +326,20 @@ public: template< class Communicator = Dune::CollectiveCommunication<typename Dune::MPIHelper::MPICommunicator> > void finalize(const Communicator& comm = Dune::MPIHelper::getCollectiveCommunication()) { - timer_.stop(); - cpuTime_ += timer_.elapsed(); + auto cpuTime = timer_.stop(); if (verbose_) { - std::cout << "Simulation took " << cpuTime_ << " seconds on " + std::cout << "Simulation took " << cpuTime << " seconds on " << comm.size() << " processes.\n"; } if (comm.size() > 1) - cpuTime_ = comm.sum(cpuTime_); + cpuTime = comm.sum(cpuTime); if (verbose_) { - std::cout << "The cumulative CPU time was " << cpuTime_ << " seconds.\n"; + std::cout << "The cumulative CPU time was " << cpuTime << " seconds.\n"; } } @@ -347,17 +363,17 @@ private: using std::max; using std::min; - - maxTimeStepSize_ = min(maxTimeStepSize_, max<Scalar>(0.0, endTime_ - time_)); + maxTimeStepSize_ = min(userSetMaxTimeStepSize_, max<Scalar>(0.0, endTime_ - time_)); } Dune::Timer timer_; Scalar time_; Scalar endTime_; - double cpuTime_; Scalar timeStepSize_; - Scalar maxTimeStepSize_; + Scalar lastTimeStepSize_; + Scalar maxTimeStepSize_, userSetMaxTimeStepSize_; + Scalar timeAfterLastTimeStep_, timeStepWallClockTime_; int timeStepIdx_; bool finished_; bool verbose_; @@ -405,6 +421,9 @@ public: { isCheckPoint_ = false; } + + // make sure to respect future check check points + this->setTimeStepSize(this->timeStepSize()); } /*! @@ -428,16 +447,46 @@ public: * \brief Set a periodic check point * \note You can query if we are at a time check point with isCheckPoint() * \param interval Set a periodic checkout every [interal] seconds + * \param offset time from which the periodic check points are supposed to start (simulation time) + * the first checkpoint will be at time = offset. + * \note If offset is in the past the first checkpoint will be at the next + * periodic heck point greater or equal than time + * \note This also updates the time step size and potentially reduces the time step size to meet the next check point */ - void setPeriodicCheckPoint(Scalar interval) + void setPeriodicCheckPoint(Scalar interval, Scalar offset = 0.0) { + using std::signbit; + if (signbit(interval)) + DUNE_THROW(Dune::InvalidStateException, "Interval has to be positive!"); + periodicCheckPoints_ = true; deltaPeriodicCheckPoint_ = interval; + lastPeriodicCheckPoint_ = offset; + while (lastPeriodicCheckPoint_ + interval - this->time() < 1e-14*interval) + lastPeriodicCheckPoint_ += interval; + if (this->verbose()) - std::cout << "Enabled periodic check points every " << interval << " seconds." << std::endl; + std::cout << "Enabled periodic check points every " << interval + << " seconds with the next check point at " << lastPeriodicCheckPoint_ + interval << " seconds." << std::endl; // make sure we respect this check point on the next time step setTimeStepSize(this->timeStepSize()); + + // check if the current time point is a check point + if (Dune::FloatCmp::eq(this->time(), lastPeriodicCheckPoint_, 1e-8*interval)) + isCheckPoint_ = true; + } + + //! disable periodic check points + void disablePeriodicCheckPoints() + { periodicCheckPoints_ = false; } + + //! remove all check points + void removeAllCheckPoints() + { + periodicCheckPoints_ = false; + while (!checkPoints_.empty()) + checkPoints_.pop(); } //! Whether now is a time checkpoint @@ -488,19 +537,28 @@ private: //! Adds a check point to the queue void setCheckPoint_(Scalar t) { + if (t < this->time()) + { + if (this->verbose()) + std::cerr << "Couldn't insert checkpoint at t = " << t + << " because that's in the past! (current simulation time is " << this->time() << ")" << std::endl; + return; + } + if (!checkPoints_.empty()) { if (t < checkPoints_.back()) { if (this->verbose()) - std::cerr << "--- Couldn't insert checkpoint as it is earlier than the last check point in the queue.\n" - << "--- Checkpoints can only be inserted in ascending order." << std::endl; + std::cerr << "Couldn't insert checkpoint as it is earlier than the last check point in the queue.\n" + << "Checkpoints can only be inserted in ascending order." << std::endl; + return; } - else - checkPoints_.push(t); } - else - checkPoints_.push(t); + + checkPoints_.push(t); + if (this->verbose()) + std::cout << "Set check point at t = " << t << " seconds." << std::endl; } /*! diff --git a/test/common/CMakeLists.txt b/test/common/CMakeLists.txt index 6d8f4f2ea9..ad388d297d 100644 --- a/test/common/CMakeLists.txt +++ b/test/common/CMakeLists.txt @@ -4,4 +4,5 @@ add_subdirectory(math) add_subdirectory(parameters) add_subdirectory(propertysystem) add_subdirectory(spline) +add_subdirectory(timeloop) add_subdirectory(typetraits) diff --git a/test/common/timeloop/CMakeLists.txt b/test/common/timeloop/CMakeLists.txt new file mode 100644 index 0000000000..9bfd94b5b6 --- /dev/null +++ b/test/common/timeloop/CMakeLists.txt @@ -0,0 +1 @@ +dune_add_test(SOURCES test_timeloop.cc) diff --git a/test/common/timeloop/test_timeloop.cc b/test/common/timeloop/test_timeloop.cc new file mode 100644 index 0000000000..c7294cc9f3 --- /dev/null +++ b/test/common/timeloop/test_timeloop.cc @@ -0,0 +1,153 @@ +// time loop test +#include <config.h> + +#include <iostream> +#include <cmath> +#include <chrono> +#include <thread> +#include <limits> + +#include <dune/common/parallel/mpihelper.hh> +#include <dune/common/exceptions.hh> + +#include <dumux/common/timeloop.hh> + +int main(int argc, char* argv[]) try +{ + const auto& mpiHelper = Dune::MPIHelper::instance(argc, argv); + + //! Standard time loop + double tStart = 0; double tEnd = 1; double dt = 0.1; + + //! Standard time loop + { + if (mpiHelper.rank() == 0) std::cout << "------- Test time loop ----------" << std::endl; + Dumux::TimeLoop<double> timeLoop(tStart, dt, tEnd); + + if (std::abs(timeLoop.time()-tStart) > 1e-15) + DUNE_THROW(Dune::InvalidStateException, "Wrong time at start!"); + if (std::abs(timeLoop.endTime()-tEnd) > 1e-15) + DUNE_THROW(Dune::InvalidStateException, "Wrong end time: " << timeLoop.endTime() << " should be " << tEnd); + if (std::abs(timeLoop.timeStepSize()-dt) > 1e-15) + DUNE_THROW(Dune::InvalidStateException, "Wrong time step size!"); + + timeLoop.setMaxTimeStepSize(0.033333333333333); + if (std::abs(timeLoop.timeStepSize()-0.033333333333333) > 1e-15) + DUNE_THROW(Dune::InvalidStateException, "Wrong time step size!"); + + timeLoop.start(); // starts the timer + timeLoop.stop(); // stops the timer + timeLoop.resetTimer(); // resets the timer + timeLoop.start(); // starts the timer again + + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + if (timeLoop.timeStepIndex() != 1) + DUNE_THROW(Dune::InvalidStateException, "Wrong time step index!"); + + timeLoop.reportTimeStep(); + + while (!timeLoop.finished()) + { + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + timeLoop.reportTimeStep(); + timeLoop.setTimeStepSize(timeLoop.timeStepSize()); + } + + timeLoop.finalize(); + + if (std::abs(timeLoop.time()-tEnd) > 1e-15) + DUNE_THROW(Dune::InvalidStateException, "Ended with wrong end time!"); + } + + + // check point timeLoop + { + if (mpiHelper.rank() == 0) std::cout << std::endl << "------- Test check point time loop ----------" << std::endl; + Dumux::CheckPointTimeLoop<double> timeLoop(tStart, dt, tEnd); + timeLoop.setPeriodicCheckPoint(0.03, -0.03); + if (!timeLoop.isCheckPoint()) + DUNE_THROW(Dune::InvalidStateException, "This is supposed to be a check point! t = " << timeLoop.time()); + + timeLoop.start(); + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + timeLoop.reportTimeStep(); + timeLoop.setPeriodicCheckPoint(0.03); + + if (std::signbit(timeLoop.timeStepSize())) + DUNE_THROW(Dune::InvalidStateException, "Time step size is negative! dt = " << timeLoop.timeStepSize()); + + if (!timeLoop.isCheckPoint()) + DUNE_THROW(Dune::InvalidStateException, "This is supposed to be a check point! t = " << timeLoop.time()); + + timeLoop.setCheckPoint(0.02); // can't be set because it's in the past + timeLoop.setCheckPoint(0.08); + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + timeLoop.reportTimeStep(); + + if (std::signbit(timeLoop.timeStepSize())) + DUNE_THROW(Dune::InvalidStateException, "Time step size is negative! dt = " << timeLoop.timeStepSize()); + + if (!timeLoop.isCheckPoint()) + DUNE_THROW(Dune::InvalidStateException, "This is supposed to be a check point! t = " << timeLoop.time()); + + timeLoop.setEndTime(1e9); + timeLoop.setTimeStepSize(1e7); + + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + timeLoop.reportTimeStep(); + + if (std::signbit(timeLoop.timeStepSize())) + DUNE_THROW(Dune::InvalidStateException, "Time step size is negative! dt = " << timeLoop.timeStepSize()); + + if (!timeLoop.isCheckPoint()) + DUNE_THROW(Dune::InvalidStateException, "This is supposed to be a check point! t = " << timeLoop.time()); + + timeLoop.removeAllCheckPoints(); + timeLoop.setTimeStepSize(0.5e-6); + timeLoop.setPeriodicCheckPoint(1.0e6, 1.0e6); + + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + timeLoop.reportTimeStep(); + + if (std::signbit(timeLoop.timeStepSize())) + DUNE_THROW(Dune::InvalidStateException, "Time step size is negative! dt = " << timeLoop.timeStepSize()); + + while (!timeLoop.finished()) + { + std::this_thread::sleep_for(std::chrono::milliseconds(5)); + timeLoop.advanceTimeStep(); + timeLoop.reportTimeStep(); + timeLoop.setTimeStepSize(std::numeric_limits<double>::max()); + + if (std::abs(timeLoop.time() - 2.0e6) < 1e-15 && !timeLoop.isCheckPoint()) + DUNE_THROW(Dune::InvalidStateException, "This is supposed to be a check point! t = " << timeLoop.time()); + + if (timeLoop.timeStepIndex() == 10) + timeLoop.setPeriodicCheckPoint(1.0e8); + + if (std::signbit(timeLoop.timeStepSize())) + DUNE_THROW(Dune::InvalidStateException, "Time step size is negative! dt = " << timeLoop.timeStepSize()); + } + + timeLoop.finalize(); + + if (std::abs(timeLoop.time()-1e9) > 1e-15) + DUNE_THROW(Dune::InvalidStateException, "Ended with wrong end time!"); + + } + + return 0; +} +// ////////////////////////////////// +// Error handler +// ///////////////////////////////// +catch (const Dune::Exception& e) { + std::cout << e << std::endl; + return 1; +} -- GitLab