Skip to content
Snippets Groups Projects
Commit bb332424 authored by Tomáš Oberhuber's avatar Tomáš Oberhuber
Browse files

Writting documentation on solver monitor.

parent 86ae79d4
No related branches found
No related tags found
1 merge request!116Documentation for linear solvers and preconditioners
...@@ -16,108 +16,143 @@ ...@@ -16,108 +16,143 @@
#include <TNL/Timer.h> #include <TNL/Timer.h>
namespace TNL { namespace TNL {
namespace Solvers { namespace Solvers {
/**
* \brief Base class for solver monitors.
*
* The solver monitors serve for monitoring a convergence and status of various solvers.
* The solver monitor uses separate thread for monitoring the solver status in preset time period.
*/
class SolverMonitor class SolverMonitor
{ {
public: public:
SolverMonitor()
: timeout_milliseconds( 500 ),
started( false ),
stopped( false ),
timer( nullptr )
{}
virtual void refresh() = 0;
void setRefreshRate( const int& refreshRate )
{
timeout_milliseconds = refreshRate;
}
void setTimer( Timer& timer )
{
this->timer = &timer;
}
void runMainLoop()
{
// We need to use both 'started' and 'stopped' to avoid a deadlock
// when the loop thread runs this method delayed after the
// SolverMonitorThread's destructor has already called stopMainLoop()
// from the main thread.
started = true;
const int timeout_base = 100;
const std::chrono::milliseconds timeout( timeout_base );
while( ! stopped ) {
refresh();
// make sure to detect changes to refresh rate
int steps = timeout_milliseconds / timeout_base;
if( steps <= 0 )
steps = 1;
int i = 0;
while( ! stopped && i++ < steps ) {
std::this_thread::sleep_for( timeout );
}
}
// reset to initial state
started = false;
stopped = false;
}
void stopMainLoop() /**
{ * \brief Basic construct with no arguments
stopped = true; */
} SolverMonitor()
: timeout_milliseconds( 500 ),
started( false ),
stopped( false ),
timer( nullptr )
{}
/**
* \brief This abstract method is responsible for printing or visualizing the status of the solver.
*/
virtual void refresh() = 0;
/**
* \brief Set the time interval between two consecutive calls of \ref SolverMonitor::refresh.
*
* \param refreshRate refresh rate in miliseconds.
*/
void setRefreshRate( const int& refreshRate ) { timeout_milliseconds = refreshRate; }
/**
* \brief Set a timer object for the solver monitor.
*
* If a timer is set, the monitor can measure real elapsed time since the start of the solver.
*
* \param timer is an instance of \ref TNL::Timer.
*/
void setTimer( Timer& timer ) { this->timer = &timer; }
/**
* \brief Starts the main loop from which the method \ref SolverMonitor::refresh is called in given time periods.
*/
void runMainLoop()
{
// We need to use both 'started' and 'stopped' to avoid a deadlock
// when the loop thread runs this method delayed after the
// SolverMonitorThread's destructor has already called stopMainLoop()
// from the main thread.
started = true;
const int timeout_base = 100;
const std::chrono::milliseconds timeout( timeout_base );
while( ! stopped ) {
refresh();
// make sure to detect changes to refresh rate
int steps = timeout_milliseconds / timeout_base;
if( steps <= 0 )
steps = 1;
int i = 0;
while( ! stopped && i++ < steps ) {
std::this_thread::sleep_for( timeout );
}
}
bool isStopped() const // reset to initial state
{ started = false;
return stopped; stopped = false;
} }
protected: /**
double getElapsedTime() * \brief Stops the main loop of the monitor. See \ref SolverMonitor::runMainLoop.
{ */
if( ! timer ) void stopMainLoop() { stopped = true; }
return 0.0;
return timer->getRealTime(); /**
} * \brief Checks whether the main loop was stopped.
*
* \return true if the main loop was stopped.
* \return false if the mian loop was not stopped yet.
*/
bool isStopped() const { return stopped; }
protected:
double getElapsedTime()
{
if( ! timer )
return 0.0;
return timer->getRealTime();
}
std::atomic_int timeout_milliseconds; std::atomic_int timeout_milliseconds;
std::atomic_bool started, stopped; std::atomic_bool started, stopped;
Timer* timer; Timer* timer;
}; };
// a RAII wrapper for launching the SolverMonitor's main loop in a separate thread /**
* \brief A RAII wrapper for launching the SolverMonitor's main loop in a separate thread.
*/
class SolverMonitorThread class SolverMonitorThread
{ {
public: public:
SolverMonitorThread( SolverMonitor& solverMonitor ) /**
: solverMonitor( solverMonitor ), * \brief Constructor with instance of solver monitor.
t( &SolverMonitor::runMainLoop, &solverMonitor ) *
{} * \param solverMonitor is a reference to an instance of a solver monitor.
*/
~SolverMonitorThread() SolverMonitorThread( SolverMonitor& solverMonitor )
{ : solverMonitor( solverMonitor ),
solverMonitor.stopMainLoop(); t( &SolverMonitor::runMainLoop, &solverMonitor )
if( t.joinable() ) {}
t.join();
} /**
* \brief Destructor.
*/
~SolverMonitorThread()
{
solverMonitor.stopMainLoop();
if( t.joinable() )
t.join();
}
private: private:
SolverMonitor& solverMonitor; SolverMonitor& solverMonitor;
std::thread t; std::thread t;
}; };
} // namespace Solvers } // namespace Solvers
} // namespace TNL } // namespace TNL
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment