/*
* Copyright (C) 2010 Sebastian Held (sebastian.held@gmx.de)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#ifndef ENGINE_MULTITHREAD_H
#define ENGINE_MULTITHREAD_H
#include "operator_multithread.h"
#include "engine_sse_compressed.h"
#include
#include
#include
#include
#include "tools/useful.h"
#ifndef __GNUC__
#include // for struct timeval
#else
#include
#endif
#ifdef MPI_SUPPORT
#define ENGINE_MULTITHREAD_BASE Engine_MPI
#include "engine_mpi.h"
#else
#define ENGINE_MULTITHREAD_BASE Engine_SSE_Compressed
#endif
class Engine_Multithread;
namespace NS_Engine_Multithread
{
class DBG // debug
{
public:
DBG() {}
~DBG() { std::cout << os.str();}
std::ostringstream& cout() {return os;}
protected:
std::ostringstream os;
};
class Timer //debug
{
public:
Timer() {gettimeofday(&t1,NULL);}
double elapsed() {gettimeofday(&t2,NULL); return (t2.tv_sec-t1.tv_sec) + (t2.tv_usec-t1.tv_usec)*1e-6;}
protected:
timeval t1,t2;
};
class thread
{
public:
thread( Engine_Multithread* ptr, unsigned int start, unsigned int stop, unsigned int stop_h, unsigned int threadID );
void operator()();
protected:
unsigned int m_start, m_stop, m_stop_h, m_threadID;
Engine_Multithread *m_enginePtr;
};
} // namespace
class Engine_Multithread : public ENGINE_MULTITHREAD_BASE
{
friend class NS_Engine_Multithread::thread;
friend class Engine_CylinderMultiGrid;
public:
static Engine_Multithread* New(const Operator_Multithread* op, unsigned int numThreads = 0);
virtual ~Engine_Multithread();
virtual void setNumThreads( unsigned int numThreads );
virtual void Init();
virtual void Reset();
//! Iterate \a iterTS number of timesteps
virtual bool IterateTS(unsigned int iterTS);
virtual void DoPreVoltageUpdates(int threadID);
virtual void DoPostVoltageUpdates(int threadID);
virtual void Apply2Voltages(int threadID);
virtual void DoPreCurrentUpdates(int threadID);
virtual void DoPostCurrentUpdates(int threadID);
virtual void Apply2Current(int threadID);
protected:
Engine_Multithread(const Operator_Multithread* op);
const Operator_Multithread* m_Op_MT;
boost::thread_group m_thread_group;
boost::barrier *m_startBarrier, *m_stopBarrier;
boost::barrier *m_IterateBarrier;
volatile unsigned int m_iterTS;
unsigned int m_numThreads; //!< number of worker threads
volatile bool m_stopThreads;
#ifdef MPI_SUPPORT
/*! Workaround needed for subgridding scheme... (see Engine_CylinderMultiGrid)
Some engines may need an additional barrier for synchronizing MPI communication.
This engine will not initialize or cleanup this barrier, but check for it and wait before executing any MPI sync.
Make sure to cleanup (delete) this barriere before Engine_Multithread::Reset() is called.
*/
boost::barrier *m_MPI_Barrier;
#endif
#ifdef ENABLE_DEBUG_TIME
std::map > m_timer_list;
#endif
};
#endif // ENGINE_MULTITHREAD_H