Added timer system implementation
--HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%40593
This commit is contained in:
		
							parent
							
								
									bda756f3ff
								
							
						
					
					
						commit
						5fe8bed256
					
				
							
								
								
									
										199
									
								
								core/CTimerSys.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								core/CTimerSys.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,199 @@ | ||||
| /**
 | ||||
| * vim: set ts=4 : | ||||
| * =============================================================== | ||||
| * SourceMod (C)2004-2007 AlliedModders LLC.  All rights reserved. | ||||
| * =============================================================== | ||||
| * | ||||
| * This file is not open source and may not be copied without explicit | ||||
| * written permission of AlliedModders LLC.  This file may not be redistributed  | ||||
| * in whole or significant part. | ||||
| * For information, see LICENSE.txt or http://www.sourcemod.net/license.php
 | ||||
| * | ||||
| * Version: $Id$ | ||||
| */ | ||||
| 
 | ||||
| #include "CTimerSys.h" | ||||
| 
 | ||||
| CTimerSystem g_Timers; | ||||
| 
 | ||||
| void ITimer::Initialize(ITimedEvent *pCallbacks, float fInterval, float fToExec, void *pData, int flags) | ||||
| { | ||||
| 	m_Listener = pCallbacks; | ||||
| 	m_Interval = fInterval; | ||||
| 	m_ToExec = fToExec; | ||||
| 	m_pData = pData; | ||||
| 	m_Flags = flags; | ||||
| 	m_InExec = false; | ||||
| 	m_KillMe = false; | ||||
| } | ||||
| 
 | ||||
| void CTimerSystem::OnSourceModAllInitialized() | ||||
| { | ||||
| 	g_ShareSys.AddInterface(NULL, this); | ||||
| } | ||||
| 
 | ||||
| void CTimerSystem::RunFrame() | ||||
| { | ||||
| 	ITimer *pTimer; | ||||
| 	TimerIter iter; | ||||
| 
 | ||||
| 	for (iter=m_SingleTimers.begin(); iter!=m_SingleTimers.end(); ) | ||||
| 	{ | ||||
| 		pTimer = (*iter); | ||||
| 		if (gpGlobals->curtime >= pTimer->m_ToExec) | ||||
| 		{ | ||||
| 			pTimer->m_InExec = true; | ||||
| 			pTimer->m_Listener->OnTimer(pTimer, pTimer->m_pData); | ||||
| 			if (pTimer->m_KillMe) | ||||
| 			{ | ||||
| 				pTimer->m_Listener->OnTimerEnd(pTimer, pTimer->m_pData); | ||||
| 			} | ||||
| 			iter = m_SingleTimers.erase(iter); | ||||
| 			m_FreeTimers.push(pTimer); | ||||
| 		} else { | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	ResultType res; | ||||
| 	for (iter=m_LoopTimers.begin(); iter!=m_LoopTimers.end(); ) | ||||
| 	{ | ||||
| 		pTimer = (*iter); | ||||
| 		if (gpGlobals->curtime >= pTimer->m_ToExec) | ||||
| 		{ | ||||
| 			pTimer->m_InExec = true; | ||||
| 			res = pTimer->m_Listener->OnTimer(pTimer, pTimer->m_pData); | ||||
| 			if (pTimer->m_KillMe || (res == Pl_Stop)) | ||||
| 			{ | ||||
| 				pTimer->m_Listener->OnTimerEnd(pTimer, pTimer->m_pData); | ||||
| 				iter = m_LoopTimers.erase(iter); | ||||
| 				m_FreeTimers.push(pTimer); | ||||
| 				continue; | ||||
| 			} | ||||
| 			pTimer->m_InExec = false; | ||||
| 			pTimer->m_ToExec = gpGlobals->curtime + pTimer->m_Interval; | ||||
| 		} | ||||
| 		iter++; | ||||
| 	} | ||||
| 
 | ||||
| 	m_LastExecTime = gpGlobals->curtime; | ||||
| } | ||||
| 
 | ||||
| ITimer *CTimerSystem::CreateTimer(ITimedEvent *pCallbacks, float fInterval, void *pData, int flags) | ||||
| { | ||||
| 	ITimer *pTimer; | ||||
| 	TimerIter iter; | ||||
| 	float to_exec = gpGlobals->curtime + fInterval; | ||||
| 
 | ||||
| 	if (m_FreeTimers.empty()) | ||||
| 	{ | ||||
| 		pTimer = new ITimer; | ||||
| 	} else { | ||||
| 		pTimer = m_FreeTimers.front(); | ||||
| 		m_FreeTimers.pop(); | ||||
| 	} | ||||
| 
 | ||||
| 	pTimer->Initialize(pCallbacks, fInterval, to_exec, pData, flags); | ||||
| 
 | ||||
| 	if (flags & TIMER_FLAG_REPEAT) | ||||
| 	{ | ||||
| 		m_LoopTimers.push_back(pTimer); | ||||
| 		goto return_timer; | ||||
| 	} | ||||
| 
 | ||||
| 	if (m_SingleTimers.size() >= 1) | ||||
| 	{ | ||||
| 		iter = --m_SingleTimers.end(); | ||||
| 		if ((*iter)->m_ToExec <= to_exec) | ||||
| 		{ | ||||
| 			goto insert_end; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	for (iter=m_SingleTimers.begin(); iter!=m_SingleTimers.end(); iter++) | ||||
| 	{ | ||||
| 		if ((*iter)->m_ToExec >= to_exec) | ||||
| 		{ | ||||
| 			m_SingleTimers.insert(iter, pTimer); | ||||
| 			goto return_timer; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| insert_end: | ||||
| 	m_SingleTimers.push_back(pTimer); | ||||
| 
 | ||||
| return_timer: | ||||
| 	return pTimer; | ||||
| } | ||||
| 
 | ||||
| void CTimerSystem::FireTimerOnce(ITimer *pTimer, bool delayExec) | ||||
| { | ||||
| 	ResultType res; | ||||
| 
 | ||||
| 	if (pTimer->m_InExec) | ||||
| 	{ | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	pTimer->m_InExec = true; | ||||
| 	res = pTimer->m_Listener->OnTimer(pTimer, pTimer->m_pData); | ||||
| 
 | ||||
| 	if (!(pTimer->m_Flags & TIMER_FLAG_REPEAT)) | ||||
| 	{ | ||||
| 		pTimer->m_Listener->OnTimerEnd(pTimer, pTimer->m_pData); | ||||
| 		m_SingleTimers.remove(pTimer); | ||||
| 		m_FreeTimers.push(pTimer); | ||||
| 	} else { | ||||
| 		if (delayExec && (res != Pl_Stop) && !pTimer->m_KillMe) | ||||
| 		{ | ||||
| 			pTimer->m_ToExec = gpGlobals->curtime + pTimer->m_Interval; | ||||
| 			pTimer->m_InExec = false; | ||||
| 			return; | ||||
| 		} | ||||
| 		pTimer->m_Listener->OnTimerEnd(pTimer, pTimer->m_pData); | ||||
| 		m_LoopTimers.remove(pTimer); | ||||
| 		m_FreeTimers.push(pTimer); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void CTimerSystem::KillTimer(ITimer *pTimer) | ||||
| { | ||||
| 	TimerList *pList; | ||||
| 
 | ||||
| 	if (pTimer->m_KillMe) | ||||
| 	{ | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	if (pTimer->m_InExec) | ||||
| 	{ | ||||
| 		pTimer->m_KillMe = true; | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	pTimer->m_InExec = true; /* The timer it's not really executed but this check needs to be done */ | ||||
| 	pTimer->m_Listener->OnTimerEnd(pTimer, pTimer->m_pData); | ||||
| 
 | ||||
| 	pList = (pTimer->m_Flags & TIMER_FLAG_REPEAT) ? &m_LoopTimers : &m_SingleTimers; | ||||
| 
 | ||||
| 	pList->remove(pTimer); | ||||
| 	m_FreeTimers.push(pTimer); | ||||
| } | ||||
| 
 | ||||
| void CTimerSystem::MapChange() | ||||
| { | ||||
| 	ITimer *pTimer; | ||||
| 	TimerIter iter; | ||||
| 
 | ||||
| 	for (iter=m_SingleTimers.begin(); iter!=m_SingleTimers.end(); iter++) | ||||
| 	{ | ||||
| 		pTimer = (*iter); | ||||
| 		pTimer->m_ToExec = pTimer->m_ToExec - m_LastExecTime + gpGlobals->curtime; | ||||
| 	} | ||||
| 
 | ||||
| 	for (iter=m_LoopTimers.begin(); iter!=m_LoopTimers.end(); iter++) | ||||
| 	{ | ||||
| 		pTimer = (*iter); | ||||
| 		pTimer->m_ToExec = pTimer->m_ToExec - m_LastExecTime + gpGlobals->curtime; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										65
									
								
								core/CTimerSys.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								core/CTimerSys.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,65 @@ | ||||
| /**
 | ||||
| * vim: set ts=4 : | ||||
| * =============================================================== | ||||
| * SourceMod (C)2004-2007 AlliedModders LLC.  All rights reserved. | ||||
| * =============================================================== | ||||
| * | ||||
| * This file is not open source and may not be copied without explicit | ||||
| * written permission of AlliedModders LLC.  This file may not be redistributed  | ||||
| * in whole or significant part. | ||||
| * For information, see LICENSE.txt or http://www.sourcemod.net/license.php
 | ||||
| * | ||||
| * Version: $Id$ | ||||
| */ | ||||
| 
 | ||||
| #ifndef _INCLUDE_SOURCEMOD_CTIMERSYS_H_ | ||||
| #define _INCLUDE_SOURCEMOD_CTIMERSYS_H_ | ||||
| 
 | ||||
| #include "ShareSys.h" | ||||
| #include <ITimerSystem.h> | ||||
| #include <sh_stack.h> | ||||
| #include <sh_list.h> | ||||
| #include "sourcemm_api.h" | ||||
| 
 | ||||
| using namespace SourceHook; | ||||
| using namespace SourceMod; | ||||
| 
 | ||||
| typedef List<ITimer *> TimerList; | ||||
| typedef List<ITimer *>::iterator TimerIter; | ||||
| 
 | ||||
| class SourceMod::ITimer | ||||
| { | ||||
| public: | ||||
| 	void Initialize(ITimedEvent *pCallbacks, float fInterval, float fToExec, void *pData, int flags); | ||||
| 	ITimedEvent *m_Listener; | ||||
| 	void *m_pData; | ||||
| 	float m_Interval; | ||||
| 	float m_ToExec; | ||||
| 	int m_Flags; | ||||
| 	bool m_InExec; | ||||
| 	bool m_KillMe; | ||||
| }; | ||||
| 
 | ||||
| class CTimerSystem :  | ||||
| 	public ITimerSystem, | ||||
| 	public SMGlobalClass | ||||
| { | ||||
| public: //SMGlobalClass
 | ||||
| 	void OnSourceModAllInitialized(); | ||||
| public: //ITimerSystem
 | ||||
| 	ITimer *CreateTimer(ITimedEvent *pCallbacks, float fInterval, void *pData, int flags); | ||||
| 	void KillTimer(ITimer *pTimer); | ||||
| 	void FireTimerOnce(ITimer *pTimer, bool delayExec=false); | ||||
| public: | ||||
| 	void RunFrame(); | ||||
| 	void MapChange(); | ||||
| private: | ||||
| 	List<ITimer *> m_SingleTimers; | ||||
| 	List<ITimer *> m_LoopTimers; | ||||
| 	CStack<ITimer *> m_FreeTimers; | ||||
| 	float m_LastExecTime; | ||||
| }; | ||||
| 
 | ||||
| extern CTimerSystem g_Timers; | ||||
| 
 | ||||
| #endif //_INCLUDE_SOURCEMOD_CTIMERSYS_H_
 | ||||
| @ -1,7 +1,7 @@ | ||||
| <?xml version="1.0" encoding="Windows-1252"?> | ||||
| <VisualStudioProject | ||||
| 	ProjectType="Visual C++" | ||||
| 	Version="8.00" | ||||
| 	Version="8,00" | ||||
| 	Name="sourcemod_mm" | ||||
| 	ProjectGUID="{E39527CD-7CAB-4420-97CC-DA1B93B260BC}" | ||||
| 	RootNamespace="sourcemod_mm" | ||||
| @ -219,6 +219,10 @@ | ||||
| 				RelativePath="..\CTextParsers.cpp" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath="..\CTimerSys.cpp" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath="..\CTranslator.cpp" | ||||
| 				> | ||||
| @ -313,6 +317,10 @@ | ||||
| 				RelativePath="..\CTextParsers.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath="..\CTimerSys.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath="..\CTranslator.h" | ||||
| 				> | ||||
| @ -421,6 +429,10 @@ | ||||
| 				RelativePath="..\..\public\ITextParsers.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath="..\..\public\ITimerSystem.h" | ||||
| 				> | ||||
| 			</File> | ||||
| 			<File | ||||
| 				RelativePath="..\..\public\IUserMessages.h" | ||||
| 				> | ||||
|  | ||||
| @ -26,6 +26,7 @@ | ||||
| #include "CPlayerManager.h" | ||||
| #include "CTranslator.h" | ||||
| #include "ForwardSys.h" | ||||
| #include "CTimerSys.h" | ||||
| 
 | ||||
| SH_DECL_HOOK6(IServerGameDLL, LevelInit, SH_NOATTRIB, false, bool, const char *, const char *, const char *, const char *, bool, bool); | ||||
| SH_DECL_HOOK0_void(IServerGameDLL, LevelShutdown, SH_NOATTRIB, false); | ||||
| @ -193,6 +194,7 @@ bool SourceModBase::LevelInit(char const *pMapName, char const *pMapEntities, ch | ||||
| 	g_LastAuthCheck = 0.0f; | ||||
| 
 | ||||
| 	g_Logger.MapChange(pMapName); | ||||
| 	g_Timers.MapChange(); | ||||
| 
 | ||||
| 	/* Refresh language stuff */ | ||||
| 	char path[PLATFORM_MAX_PATH]; | ||||
| @ -221,7 +223,7 @@ void SourceModBase::GameFrame(bool simulating) | ||||
| 	 * precious CPU cycles. | ||||
| 	 */ | ||||
| 	float curtime = gpGlobals->curtime; | ||||
| 	if (curtime - g_LastTime > 0.1f) | ||||
| 	if (curtime - g_LastTime >= 0.1f) | ||||
| 	{ | ||||
| 		if (m_CheckingAuth | ||||
| 			&& (gpGlobals->curtime - g_LastAuthCheck > 0.7f)) | ||||
| @ -229,9 +231,11 @@ void SourceModBase::GameFrame(bool simulating) | ||||
| 			g_LastAuthCheck = gpGlobals->curtime; | ||||
| 			g_Players.RunAuthChecks(); | ||||
| 		} | ||||
| 
 | ||||
| 		g_Timers.RunFrame(); | ||||
| 		g_LastTime = curtime; | ||||
| 	} | ||||
| 	 | ||||
| 
 | ||||
| 	if (g_pOnGameFrame && g_pOnGameFrame->GetFunctionCount()) | ||||
| 	{ | ||||
| 		g_pOnGameFrame->Execute(NULL); | ||||
|  | ||||
| @ -20,11 +20,10 @@ | ||||
| #define _INCLUDE_SOURCEMOD_TIMER_SYSTEM_H_ | ||||
| 
 | ||||
| #include <IShareSys.h> | ||||
| #include <IForwardSys.h> | ||||
| 
 | ||||
| //:TODO: this is a placeholder and not yet implemented
 | ||||
| //remove these lines and uncomment once we're done!
 | ||||
| //#define SMINTERFACE_TIMERSYS_NAME		"ITimerSys"
 | ||||
| //#define SMINTERFACE_TIMERSYS_VERSION	1
 | ||||
| #define SMINTERFACE_TIMERSYS_NAME		"ITimerSys" | ||||
| #define SMINTERFACE_TIMERSYS_VERSION	1 | ||||
| 
 | ||||
| namespace SourceMod | ||||
| { | ||||
| @ -41,9 +40,7 @@ namespace SourceMod | ||||
| 		 * | ||||
| 		 * @param pTimer		Pointer to the timer instance. | ||||
| 		 * @param pData			Private pointer passed from host. | ||||
| 		 * @return				Pl_Handle to stop timer, Pl_Continue to continue. | ||||
| 		 *						Passing Pl_Continue when a timer's repeat count | ||||
| 		 *						has been exhausted will not extend it. | ||||
| 		 * @return				Pl_Stop to stop timer, Pl_Continue to continue. | ||||
| 		 */ | ||||
| 		virtual ResultType OnTimer(ITimer *pTimer, void *pData) =0; | ||||
| 
 | ||||
| @ -53,7 +50,7 @@ namespace SourceMod | ||||
| 		 * @param pTimer		Pointer to the timer instance. | ||||
| 		 * @param pData			Private data pointer passed from host. | ||||
| 		 */ | ||||
| 		virtual void OnTimerEnd() =0; | ||||
| 		virtual void OnTimerEnd(ITimer *pTimer, void *pData) =0; | ||||
| 	}; | ||||
| 
 | ||||
| 	#define TIMER_FLAG_REPEAT		(1<<0) | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user