// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdpch.h" #include "nel/misc/debug.h" #include "nel/misc/hierarchical_timer.h" #include "module_manager.h" using namespace NLMISC; using namespace std; /* * Static variables */ uint CModuleManager::_MaxModules = 0; NLMISC::CMutex *CModuleManager::_ModMutexes = NULL; vector CModuleManager::_RegisteredManagers; // Constructor CModuleManager::CModuleManager(const char *name, bool independent) { _Independent = independent; _Thread = NULL; _StopThread = false; _ThreadStopped = false; _StackName = (name != NULL) ? string(name) : ""; _Id = _RegisteredManagers.size(); _Cycle = 0; _CompleteCycle = false; _RegisteredManagers.push_back(this); } // Destructor CModuleManager::~CModuleManager() { if (_Thread != NULL) { nlwarning("FEMMAN: [%s] Execution is not finished yet. Brutal thread killing", _StackName.c_str()); _Thread->terminate(); delete _Thread; } } // void CModuleManager::init(uint maxModules) { nlassert(_MaxModules == 0); nlassert(_ModMutexes == NULL); _MaxModules = maxModules; _ModMutexes = new CMutex[maxModules]; } // void CModuleManager::release() { _MaxModules = 0; delete [] _ModMutexes; _ModMutexes = NULL; } // void CModuleManager::startAll() { // first reset cycle resetCycle(); uint i; // and start all managers at once for (i=0; i<_RegisteredManagers.size(); ++i) if (_RegisteredManagers[i]->_Independent) _RegisteredManagers[i]->start(); } void CModuleManager::stopAll(TTime timeout) { uint i; // send soft stop for (i=0; i<_RegisteredManagers.size(); ++i) _RegisteredManagers[i]->stop(false, 0); // wait for all managers to stop or timeout TTime before = CTime::getLocalTime(); while (!allStopped() && CTime::getLocalTime()-before < timeout) nlSleep(10); // and hard stop for (i=0; i<_RegisteredManagers.size(); ++i) _RegisteredManagers[i]->stop(true, 0); } void CModuleManager::resetCycle() { uint i; // reset all managers cycle counter for (i=0; i<_RegisteredManagers.size(); ++i) _RegisteredManagers[i]->_Cycle = 0; } bool CModuleManager::allReady() { uint i; // checks if all managers are at the same cycle for (i=0; i<_RegisteredManagers.size()-1; ++i) if (_RegisteredManagers[i]->_Cycle != _RegisteredManagers[i+1]->_Cycle) return false; return true; } bool CModuleManager::allComplete() { uint i; // checks if all managers have set the stop flag for (i=0; i<_RegisteredManagers.size()-1; ++i) if (_RegisteredManagers[i]->_CompleteCycle != _RegisteredManagers[i+1]->_CompleteCycle) return false; return true; } bool CModuleManager::allStopped() { uint i; // checks if all managers have set the stop flag for (i=0; i<_RegisteredManagers.size(); ++i) if (!_RegisteredManagers[i]->_ThreadStopped) return false; return true; } void CModuleManager::resetManagers() { // clear all registered managers _RegisteredManagers.clear(); } // void CModuleManager::addModule(uint id, TModuleExecCallback cb) { nlassert(id < _MaxModules); nlassert(cb != NULL); nldebug("FEMMAN: [%s] Added module %d (Cb=%p) to stack", _StackName.c_str(), id, cb); _ExecutionStack.push_back(CExecutionItem()); _ExecutionStack.back().Type = Module; _ExecutionStack.back().Id = id; _ExecutionStack.back().Cb = cb; _ExecutedModules.push_back(id); } // void CModuleManager::addWait(uint id) { nlassert(id < _MaxModules); nldebug("FEMMAN: [%s] Added wait %d to stack", _StackName.c_str(), id); _ExecutionStack.push_back(CExecutionItem()); _ExecutionStack.back().Type = Wait; _ExecutionStack.back().Id = id; _ExecutionStack.back().Cb = NULL; } // void CModuleManager::start() { _Thread = IThread::create(this); _StopThread = false; _ThreadStopped = false; _Thread->start(); //nlinfo("FEMMAN: [%s] Start", _StackName.c_str()); } // void CModuleManager::runOnce() { _StopThread = false; _ThreadStopped = false; //nlinfo("FEMMAN: [%s] Running one time", _StackName.c_str()); // step cycle stepCycle(); // wait for all managers to sync on the same cycle before entering mutexes waitAllReady(); // lock mutexes enterMutexes(); completeCycle(); // and wait for all managers to finish entering mutexes waitAllComplete(); //H_BEFORE(MMExecuteStack); executeStack(); //H_AFTER(MMExecuteStack); _ThreadStopped = true; } // void CModuleManager::stop(bool blockingMode, TTime timeout) { // non independent modules (called by main thread) are always stopped at this point, no need to force stop if (!_Independent) { _ThreadStopped = true; return; } if (!blockingMode) { // if soft stop, just send stop message and leave nlinfo("FEMMAN: [%s] soft stop", _StackName.c_str()); _StopThread = true; return; } if (!_StopThread) { // if not yet called stop, send message stop nlinfo("FEMMAN: [%s] hard stop", _StackName.c_str()); _StopThread = true; // wait for stop or timeout TTime before = CTime::getLocalTime(); while (!_ThreadStopped && CTime::getLocalTime()-before < timeout) nlSleep(10); } if ( _Thread ) { // if timeout, terminate thread if (!_ThreadStopped) { nlwarning("FEMMAN: [%s] Can't stop. Brutal thread killing", _StackName.c_str()); _Thread->terminate(); } delete _Thread; _Thread = NULL; } } // void CModuleManager::run() { nldebug("FEMMAN: [%s] attached thread loop start", _StackName.c_str()); while (true) { // step cycle stepCycle(); // wait for all managers to sync on the same cycle before entering mutexes waitAllReady(); // lock mutexes enterMutexes(); completeCycle(); // and wait for all managers to finish entering mutexes waitAllComplete(); //H_BEFORE(MMExecuteStack); executeStack(); //H_AFTER(MMExecuteStack); // if stop sent, just leave if (_StopThread) break; } nldebug("FEMMAN: [%s] attached thread loop end", _StackName.c_str()); _ThreadStopped = true; } // void CModuleManager::executeStack() { // for each item, // if a module, // calls associated callback // leaves mutex // else // enters mutex // leaves mutex //nldebug("FEMMAN: [%s] execute stack", _StackName.c_str()); uint i; for (i=0; i<_ExecutionStack.size(); ++i) { CExecutionItem &item = _ExecutionStack[i]; if (item.Type == Module) { //nldebug("FEMMAN: [%s] execute module %d at %p", _StackName.c_str(), item.Id, item.Cb); //H_BEFORE(MMCall); item.Cb(); //H_AFTER(MMCall); _ModMutexes[item.Id].leave(); } else if (item.Type == Wait) { //nldebug("FEMMAN: [%s] wait for module %d to finish", _StackName.c_str(), item.Id); //H_BEFORE(MMWait); //TTime t = CTime::getLocalTime(); _ModMutexes[item.Id].enter(); //nlinfo( "Waited %u ms", (uint32)(CTime::getLocalTime()-t) ); _ModMutexes[item.Id].leave(); //H_AFTER(MMWait); } else { nlwarning("FEMMAN: Unexpected ExecutionItem type (%d) at item %d of the execution stack %s", item.Type, i, _StackName.c_str()); uint j; for (j=0; j<_ExecutionStack.size(); ++j) nlwarning("FEMMAN: > %d [%s] Id=%d Cb=%p", j, (item.Type == Module) ? "MOD" : (item.Type == Wait) ? "WAIT" : "ERR", item.Id, item.Cb); nlerror("FEMMAN: Error in execution stack %s", _StackName.c_str()); } } //nldebug("FEMMAN: [%s] stack executed", _StackName.c_str()); } void CModuleManager::enterMutexes() { // enters all controlled mutexes //nldebug("FEMMAN: [%s] Entering all controlled mutexes", _StackName.c_str()); uint i; for (i=0; i<_ExecutedModules.size(); ++i) { //nldebug("FEMMAN: [%s] Entering mutex %d", _StackName.c_str(), _ExecutedModules[i]); //TTime t = CTime::getLocalTime(); _ModMutexes[_ExecutedModules[i]].enter(); //nlinfo( "Waited %u ms", (uint32)(CTime::getLocalTime()-t) ); } //nldebug("FEMMAN: [%s] All controlled mutexes successfully entered", _StackName.c_str()); }