//========= Copyright 1996-2004, Valve LLC, All rights reserved. ============ // // Purpose: // // $NoKeywords: $ //============================================================================= #ifndef GC_JOB_H #define GC_JOB_H #ifdef _WIN32 #pragma once #endif #include "tier0/memdbgon.h" #include "tier1/functors.h" #include "workthreadpool.h" class GCConVar; namespace GCSDK { class CJobMgr; class CLock; class CJob; class IMsgNetPacket; // job creation function responsible for allocating the job of the specific type typedef CJob *(*JobCreationFunc_t)( void *pvServerParent, void * pvStartParam ); // job routing function, which allows for controlling message routing through the system. The GCs that should have the message sent to it should be added to the vector. It is fine to add the actual self GC typedef void (*JobRoutingFunc_t)( CUtlVector< uint32 >& vecGCsToSendTo, IMsgNetPacket *pNetPacket ); //----------------------------------------------------------------------------- // Purpose: static job information // contains information relevant to one type of CJob //----------------------------------------------------------------------------- struct JobType_t { const char *m_pchName; // name of this type of job MsgType_t m_eCreationMsg; // network message that creates this job uint32 m_nValidContexts; // a bit field indicating which contexts this message can be called from (i.e. from a server, or from a client, etc) JobCreationFunc_t m_pJobFactory; // virtual constructor JobRoutingFunc_t m_pJobRouter; // routing function for this job const GCConVar* m_pControlCV; // optional console variable that can be used to disable this message }; //----------------------------------------------------------------------------- // Purpose: reason as to why the current job has yielded to the main thread (paused) // if this is updated, k_prgchJobPauseReason[] in job.cpp also needs to be updated //----------------------------------------------------------------------------- enum EJobPauseReason { k_EJobPauseReasonNone, k_EJobPauseReasonNotStarted, k_EJobPauseReasonNetworkMsg, k_EJobPauseReasonSleepForTime, k_EJobPauseReasonWaitingForLock, k_EJobPauseReasonYield, k_EJobPauseReasonSQL, k_EJobPauseReasonWorkItem, k_EJobPauseReasonCount }; //----------------------------------------------------------------------------- // Purpose: contains information used to route a message to a job, or to // create a new job from that message type //----------------------------------------------------------------------------- struct JobMsgInfo_t { MsgType_t m_eMsg; JobID_t m_JobIDSource; JobID_t m_JobIDTarget; JobMsgInfo_t() { m_eMsg = (MsgType_t)0; m_JobIDSource = k_GIDNil; m_JobIDTarget = k_GIDNil; } JobMsgInfo_t( MsgType_t eMsg, JobID_t jobIDSource, JobID_t jobIDTarget ) { m_eMsg = eMsg; m_JobIDSource = jobIDSource; m_JobIDTarget = jobIDTarget; } }; typedef void (CJob::*JobThreadFunc_t)(); #define BYieldingAcquireLock( lock ) _BYieldingAcquireLock( lock, __FILE__, __LINE__ ) #define BAcquireLockImmediate( lock ) _BAcquireLockImmediate( lock, __FILE__, __LINE__ ) #define ReleaseLock( lock ) _ReleaseLock( lock, false, __FILE__, __LINE__ ) //----------------------------------------------------------------------------- // Purpose: A job is any server operation that requires state. Typically, we use jobs for // operations that need to pause waiting for responses from other servers. The // job object persists the state of the operation while it waits, and the reply // from the remote server re-activates the job. //----------------------------------------------------------------------------- class CJob { public: // Constructors & destructors, when overriding job name a static string pointer must be used explicit CJob( CJobMgr &jobMgr, char const *pchJobName = NULL ); virtual ~CJob(); // starts the job, storing off the network msg and calling it's Run() function void StartJobFromNetworkMsg( IMsgNetPacket *pNetPacket, const JobID_t &gidJobIDSrc, uint32 nContextMask ); // accessors JobID_t GetJobID() const { return m_JobID; } // called to start a job. The default behavior of starting a job is to start it scheduled (i.e. delayed). This is largely deprecated, and the more explicit versions // below should be used instead void StartJob( void * pvStartParam ); // schedules the job for execution, but does not interrupt the currently running job. Effectively starts the job on the yielding list as if it had immediately yielded // although is more efficient than actually doing so void StartJobDelayed( void * pvStartParam ); // starts a job immediately, interrupting the current job if one is already running. This should only be used in special cases, and in a lot of ways should be considered // yielding in that another job can run and perform modifications, although the caller will receive attention back the first time that the inner job itself yields void StartJobImmediate( void * pvStartParam ); // string name of the job const char *GetName() const; // return reason why we're paused EJobPauseReason GetPauseReason() const { return m_ePauseReason; } // string description of why we're paused const char *GetPauseReasonDescription() const; // return time at which this job was last paused or continued const CJobTime& GetTimeSwitched() const { return m_STimeSwitched; } // return microseconds run since we were last continued uint64 GetMicrosecondsRun() const { return m_FastTimerDelta.GetDurationInProgress().GetUlMicroseconds(); } bool BJobNeedsToHeartbeat() const { return ( m_STimeNextHeartbeat.CServerMicroSecsPassed() >= 0 ); } // --- locking pointers bool _BYieldingAcquireLock( CLock *pLock, const char *filename = "unknown", int line = 0 ); bool _BAcquireLockImmediate( CLock *pLock, const char *filename = "unknown", int line = 0 ); void _ReleaseLock( CLock *pLock, bool bForce = false, const char *filename = "unknown", int line = 0 ); void ReleaseLocks(); bool BJobHoldsLock( uint16 nType, uint64 nSubType ) const; bool BJobHoldsLock( const CLock* pLock ) const; /// If we hold any locks, spew about them and release them. /// This is useful for long running jobs, to make sure they don't leak /// locks that never get cleaned up void ShouldNotHoldAnyLocks(); // --- general methods for waiting for events // Simple yield to other jobs until Run() called again bool BYield(); // Yield IF JobMgr thinks we need to based on how long we've run and our priority bool BYieldIfNeeded( bool *pbYielded = NULL ); // waits for a set amount of time bool BYieldingWaitTime( uint32 m_cMicrosecondsToSleep ); bool BYieldingWaitOneFrame(); // waits for another network msg, returning false if none returns bool BYieldingWaitForMsg( IMsgNetPacket **ppNetPacket ); bool BYieldingWaitForMsg( CGCMsgBase *pMsg, MsgType_t eMsg ); bool BYieldingWaitForMsg( CProtoBufMsgBase *pMsg, MsgType_t eMsg ); // waits for another job(s) to complete bool BYieldingWaitForJob( JobID_t jobToWaitFor ); bool BYieldingWaitForJobs( const CUtlVector &vecJobsToWaitFor ); #ifdef GC void SOVALIDATE_SetSteamID( const CSteamID steamID ) { m_SOCacheValidSteamID = steamID; } CSteamID SOVALIDATE_GetSteamID() const { return m_SOCacheValidSteamID; } void VALIDATE_SetJobAccessType( uint32 nAccess) { m_nGCJobAccessType = nAccess; } uint32 VALIDATE_GetJobAccessType() const { return m_nGCJobAccessType; } bool BYieldingWaitForMsg( CGCMsgBase *pMsg, MsgType_t eMsg, const CSteamID &expectedID ); bool BYieldingWaitForMsg( CProtoBufMsgBase *pMsg, MsgType_t eMsg, const CSteamID &expectedID ); bool BYieldingRunQuery( CGCSQLQueryGroup *pQueryGroup, ESchemaCatalog eSchemaCatalog ); #endif void RecordWaitTimeout() { m_flags.m_bits.m_bWaitTimeout = true; } // wait for pending work items before deleting job void WaitForThreadFuncWorkItemBlocking(); // waits for a work item completion callback // You can pass a string that describes what sort of work item you are waiting on. // WARNING: This function saves the pointer to the string, it doesn't copy the string bool BYieldingWaitForWorkItem( const char *pszWorkItemName = NULL ); // adds this work item to threaded work pool and waits for it bool BYieldingWaitForThreadFuncWorkItem( CWorkItem * ); // calls a local function in a thread, and yields until it's done bool BYieldingWaitForThreadFunc( CFunctor *jobFunctor ); // creates a job template static JOB_TYPE *AllocateJob( PARAM_TYPE *pParam ) { return new JOB_TYPE( pParam ); } // delete a job (the job knows what allocator to use) static void DeleteJob( CJob *pJob ); void SetStartParam( void * pvStartParam ) { Assert( NULL == m_pvStartParam ); m_pvStartParam = pvStartParam; } void SetFromFromMsg( bool bRunFromMsg ) { m_bRunFromMsg = true; } void AddPacketToList( IMsgNetPacket *pNetPacket, const JobID_t gidJobIDSrc ); // marks a packet as being finished with, releases the packet and frees the memory void ReleaseNetPacket( IMsgNetPacket *pNetPacket ); void EndPause( EJobPauseReason eExpectedState ); // Generate an assertion in the coroutine of this job // (creating a minidump). Useful for inspecting stuck jobs void GenerateAssert( const char *pchMsg = NULL ); //called to determine if the requested context is valid. If multiple contexts are provided, this will return true only if ALL the contexts are valid bool BHasContext( uint32 nContext ) const { return ( m_nContextMask & nContext ) == nContext; } uint32 GetContexts() const { return m_nContextMask; } //called to control the default behavior for starting jobs, immediate, or delayed static void SetStartDefaultJobsDelayed( bool bStartJobsDelayed ) { s_bStartDefaultJobsDelayed = bStartJobsDelayed; } // accessor to get access to the JobMgr from the server we belong to CJobMgr &GetJobMgr(); protected: // main job implementation, in the coroutine. Every job must implement at least one of these methods. virtual bool BYieldingRunJob( void * pvStartParam ) { Assert( false ); return true; } // implement this if your job can be started directly virtual bool BYieldingRunJobFromMsg( IMsgNetPacket * pNetPacket ) { Assert( false ); return true; } // implement this if your job can be started by a network message // Can be overridden to return a different timeout per job class virtual uint32 CHeartbeatsBeforeTimeout(); // Called by CJobMgr to send heartbeat message to our listeners during long operations void Heartbeat(); uint32 m_bRunFromMsg:1, m_bWorkItemCanceled:1, // true if the work item we were waiting on was canceled m_bIsTest:1, m_bIsLongRunning:1; private: // starts the coroutine that activates the job void InitCoroutine(); // continues the current job void Continue(); // break into this coroutine - can only be called from OUTSIDE this coroutine void Debug(); // pauses the current job - can only be called from inside a coroutine void Pause( EJobPauseReason eReason, const char *pszResourceName ); static void BRunProxy( void *pvThis ); JobID_t m_JobID; // Our unique identifier. HCoroutine m_hCoroutine; void * m_pvStartParam; // Start params for our job, if any // all these flags indicate some kind of failure and we will want to report them union { struct { uint32 m_bJobFailed:1, // true if BYieldingRunJob returned false m_bLocksFailed:1, m_bLocksLongHeld:1, m_bLocksLongWait:1, m_bWaitTimeout:1, m_bLongInterYield:1, m_bTimeoutNetMsg:1, m_bTimeoutOther:1, m_uUnused:24; } m_bits; uint32 m_uFlags; } m_flags; int m_cLocksAttempted; int m_cLocksWaitedFor; EJobPauseReason m_ePauseReason; const char *m_pszPauseResourceName; MsgType_t m_unWaitMsgType; CJobTime m_STimeStarted; // time (frame) at which this job started CJobTime m_STimeSwitched; // time (frame) at which we were last paused or continued CJobTime m_STimeNextHeartbeat; // Time at which next heartbeat should be performed CFastTimer m_FastTimerDelta; // How much time we've been running for without yielding CCycleCount m_cyclecountTotal; // Total runtime uint32 m_nContextMask; // The context that this job was created in. Typically only used for message jobs to indicate the initiator of the message CJob *m_pJobPrev; // the job that launched us // lock manipulation void _SetLock( CLock *pLock, const char *filename, int line ); void UnsetLock( CLock *pLock ); void PassLockToJob( CJob *pNewJob, CLock *pLock ); void OnLockDeleted( CLock *pLock ); void AddJobToNotifyOnLockRelease( CJob *pJob ); CUtlVectorFixedGrowable< CLock *, 2 > m_vecLocks; CLock *m_pWaitingOnLock; // lock we're waiting on, if any const char *m_pWaitingOnLockFilename; int m_waitingOnLockLine; CJob *m_pJobToNotifyOnLockRelease; // other job that wants this lock CWorkItem *m_pWaitingOnWorkItem; // set if job is waiting for this work item #ifdef GC //context flags indicating what this job can access. Temporary and only for validating access on the GC uint32 m_nGCJobAccessType; //the steam ID that we are stating is safe to access. This is temporary to validate jobs during the split of the GC CSteamID m_SOCacheValidSteamID; #endif CJobMgr &m_JobMgr; // our job manager CUtlVectorFixedGrowable< IMsgNetPacket *, 1 > m_vecNetPackets; // list of tcp packets currently held by this job (ie, needing release on job exit) // pointer to our own static job info struct JobType_t const *m_pJobType; // Name of the job for when it's not registered const char *m_pchJobName; // setting the job info friend void Job_SetJobType( CJob &job, const JobType_t *pJobType ); friend class CJobMgr; friend class CLock; // used to store the memory allocation stack CUtlMemory< unsigned char > m_memAllocStack; static bool s_bStartDefaultJobsDelayed; }; // Only one job can be running at a time. We keep a global accessor to it. extern CJob *g_pJobCur; inline CJob &GJobCur() { Assert( g_pJobCur != NULL ); return *g_pJobCur; } #define AssertRunningJob() { Assert( NULL != g_pJobCur ); } #define AssertRunningThisJob() { Assert( this == g_pJobCur ); } #define AssertNotRunningThisJob() { Assert( this != g_pJobCur ); } #define AssertNotRunningJob() { Assert( NULL == g_pJobCur ); } //----------------------------------------------------------------------------- // Purpose: simple locking class // add this object to any classes you want jobs to be able to lock //----------------------------------------------------------------------------- #if defined( GC ) #include "tier0/memdbgoff.h" #endif class CLock { #if defined( GC ) DECLARE_CLASS_MEMPOOL( CLock ); #endif public: CLock( ); ~CLock(); bool BIsLocked() const { return m_pJob != NULL; } CJob *GetJobLocking() { return m_pJob; } CJob *GetJobWaitingQueueHead() { return m_pJobToNotifyOnLockRelease; } CJob *GetJobWaitingQueueTail() { return m_pJobWaitingQueueTail; } void AddToWaitingQueue( CJob *pJob ); const char *GetName() const; void SetName( const char *pchName ); int16 GetLockType() const { return m_nsLockType; } void SetLockType( int16 nsLockType ) { m_nsLockType = nsLockType; } uint64 GetLockSubType() const { return m_unLockSubType; } void SetLockSubType( uint64 unLockSubType ) { m_unLockSubType = unLockSubType; } int32 GetWaitingCount() const { return m_nWaitingCount; } int64 GetMicroSecondsSinceLock() const { return m_sTimeAcquired.CServerMicroSecsPassed(); } void IncrementReference(); int DecrementReference(); void ClearReference() { m_nRefCount = 0; } int32 GetReferenceCount() const { return m_nRefCount; } void Dump( const char *pszPrefix = "\t\t", int nPrintMax = 1, bool bPrintWaiting = true ) const; private: CJob *m_pJob; // the job that's currently locking us CJob *m_pJobToNotifyOnLockRelease; // Pointer to the first job waiting on us CJob *m_pJobWaitingQueueTail; // Pointer to the last job waiting on us const char *m_pchConstStr; // Prefix part of the lock's name int32 m_nRefCount; // # of times locked int32 m_nWaitingCount; // Count of jobs waiting on the lock CJobTime m_sTimeAcquired; // Time the lock was last locked uint64 m_unLockSubType; const char *m_pFilename; // Filename of the source file who acquired this lock int m_line; // Line number of the filename int16 m_nsLockType; // Lock priority for safely waiting on multiple locks friend class CJob; }; #if defined( GC ) #include "tier0/memdbgon.h" #endif //----------------------------------------------------------------------------------------- // An auto lock class which handles auto lifetime management of a lock //----------------------------------------------------------------------------------------- class CGCAutoLock { public: CGCAutoLock() : m_pLock( NULL ) {} CGCAutoLock( const CGCAutoLock& rhs ) { Acquire( rhs.m_pLock ); } ~CGCAutoLock() { Release(); } //determines if this lock is currently locked or not bool IsLocked() const { return ( m_pLock != NULL ); } //swaps two locks (faster than doing reassignments due to not needing all the reference counting) void Swap( CGCAutoLock& rhs ) { std::swap( m_pLock, rhs.m_pLock ); } CGCAutoLock& operator=( const CGCAutoLock& rhs ); //called to acquire a lock (the odd naming convention is to match the macro format to automatically provide the file and line of the call site) bool _BYieldingAcquireLock( CLock& lock, const char* pszFile, uint32 nLine ); //called to release the current lock void Release(); private: void Acquire( CLock* pLock ); CLock* m_pLock; }; //----------------------------------------------------------------------------- // Purpose: Use these macros to declare blocks where it is unsafe to yield. // The job will assert if it pauses within the block //----------------------------------------------------------------------------- #define DO_NOT_YIELD_THIS_SCOPE() CDoNotYieldScopeImpl doNotYieldScope_##line( FILE_AND_LINE ) #define BEGIN_DO_NOT_YIELD() CDoNotYieldScopeImpl::InternalPush( FILE_AND_LINE ) #define END_DO_NOT_YIELD() CDoNotYieldScopeImpl::InternalPop() class CDoNotYieldScopeImpl { public: explicit CDoNotYieldScopeImpl( const char *pchLocation ) { InternalPush( pchLocation ); } ~CDoNotYieldScopeImpl() { InternalPop(); } static void InternalPush( const char *pchLocation ); static void InternalPop(); private: // Disallow these constructors and operators CDoNotYieldScopeImpl( const CDoNotYieldScopeImpl &that ); CDoNotYieldScopeImpl &operator=( const CDoNotYieldScopeImpl &that ); }; } // namespace GCSDK #include "tier0/memdbgoff.h" #endif // GC_JOB_H