#include "stock.h" #pragma hdrstop #include "runtask.h" #define SUPERCLASS // #define TF_RUNTASK TF_GENERAL #define TF_RUNTASK 0 // #define TF_RUNTASKV TF_CUSTOM1 // verbose version #define TF_RUNTASKV 0 // constructor CRunnableTask::CRunnableTask(DWORD dwFlags) { _lState = IRTIR_TASK_NOT_RUNNING; _dwFlags = dwFlags; ASSERT(NULL == _hDone); if (_dwFlags & RTF_SUPPORTKILLSUSPEND) { // we signal this on suspend or kill // Explicitly call the ANSI version so we don't need to worry // about whether we're being built UNICODE and have to switch // to a wrapper function... _hDone = CreateEventA(NULL, TRUE, FALSE, NULL); } #ifdef DEBUG _dwTaskID = GetTickCount(); TraceMsg(TF_RUNTASK, "CRunnableTask (%#lx): creating task", _dwTaskID); #endif _cRef = 1; } // destructor CRunnableTask::~CRunnableTask() { DEBUG_CODE( TraceMsg(TF_RUNTASK, "CRunnableTask (%#lx): deleting task", _dwTaskID); ) if (_hDone) CloseHandle(_hDone); } STDMETHODIMP CRunnableTask::QueryInterface( REFIID riid, LPVOID * ppvObj ) { if ( ppvObj == NULL ) { return E_INVALIDARG; } if ( riid == IID_IRunnableTask ) { *ppvObj = SAFECAST( this, IRunnableTask *); AddRef(); } else return E_NOINTERFACE; return NOERROR; } STDMETHODIMP_(ULONG) CRunnableTask::AddRef() { InterlockedIncrement(&_cRef); return _cRef; } STDMETHODIMP_ (ULONG) CRunnableTask::Release() { if (0 == _cRef) { AssertMsg(0, TEXT("CRunnableTask::Release called too many times!")); return 0; } if ( InterlockedDecrement(&_cRef) == 0) { delete this; return 0; } return _cRef; } /*---------------------------------------------------------- Purpose: IRunnableTask::Run method This does a lot of the state-related work, and then calls the derived-class's RunRT() method. */ STDMETHODIMP CRunnableTask::Run(void) { HRESULT hr = E_FAIL; // Are we already running? if (_lState == IRTIR_TASK_RUNNING) { // Yes; nothing to do hr = S_FALSE; } else if ( _lState == IRTIR_TASK_PENDING ) { hr = E_FAIL; } else if ( _lState == IRTIR_TASK_NOT_RUNNING ) { // Say we're running LONG lRes = InterlockedExchange(&_lState, IRTIR_TASK_RUNNING); if ( lRes == IRTIR_TASK_PENDING ) { _lState = IRTIR_TASK_FINISHED; return NOERROR; } if (_lState == IRTIR_TASK_RUNNING) { // Prepare to run DEBUG_CODE( TraceMsg(TF_RUNTASKV, "CRunnableTask (%#lx): initialize to run", _dwTaskID); ) hr = RunInitRT(); ASSERT(E_PENDING != hr); } if (SUCCEEDED(hr)) { if (_lState == IRTIR_TASK_RUNNING) { // Continue to do the work hr = InternalResumeRT(); } else if (_lState == IRTIR_TASK_SUSPENDED) { // it is possible that RunInitRT took a little longer to complete and our state changed // from running to suspended with _hDone signaled, which would cause us to not call // internal resume. We simulate internal resume here if (_hDone) ResetEvent(_hDone); hr = E_PENDING; } } if (FAILED(hr) && E_PENDING != hr) { DEBUG_CODE( TraceMsg(TF_WARNING, "CRunnableTask (%#lx): task failed to run: %#lx", _dwTaskID, hr); ) } // Are we finished? if (_lState != IRTIR_TASK_SUSPENDED || hr != E_PENDING) { // Yes _lState = IRTIR_TASK_FINISHED; } } return hr; } /*---------------------------------------------------------- Purpose: IRunnableTask::Kill method */ STDMETHODIMP CRunnableTask::Kill(BOOL fWait) { if ( !(_dwFlags & RTF_SUPPORTKILLSUSPEND) ) return E_NOTIMPL; if (_lState != IRTIR_TASK_RUNNING) return S_FALSE; DEBUG_CODE( TraceMsg(TF_RUNTASKV, "CRunnableTask (%#lx): killing task", _dwTaskID); ) LONG lRes = InterlockedExchange(&_lState, IRTIR_TASK_PENDING); if (lRes == IRTIR_TASK_FINISHED) { DEBUG_CODE( TraceMsg(TF_RUNTASKV, "CRunnableTask (%#lx): task already finished", _dwTaskID); ) _lState = lRes; } else if (_hDone) { // signal the event it is likely to be waiting on SetEvent(_hDone); } return KillRT(fWait); } /*---------------------------------------------------------- Purpose: IRunnableTask::Suspend method */ STDMETHODIMP CRunnableTask::Suspend( void ) { if ( !(_dwFlags & RTF_SUPPORTKILLSUSPEND) ) return E_NOTIMPL; if (_lState != IRTIR_TASK_RUNNING) return E_FAIL; DEBUG_CODE( TraceMsg(TF_RUNTASKV, "CRunnableTask (%#lx): suspending task", _dwTaskID); ) LONG lRes = InterlockedExchange(&_lState, IRTIR_TASK_SUSPENDED); if (IRTIR_TASK_FINISHED == lRes) { // we finished before we could suspend DEBUG_CODE( TraceMsg(TF_RUNTASKV, "CRunnableTask (%#lx): task already finished", _dwTaskID); ) _lState = lRes; return NOERROR; } if (_hDone) SetEvent(_hDone); return SuspendRT(); } /*---------------------------------------------------------- Purpose: IRunnableTask::Resume method */ STDMETHODIMP CRunnableTask::Resume(void) { if (_lState != IRTIR_TASK_SUSPENDED) return E_FAIL; DEBUG_CODE( TraceMsg(TF_RUNTASKV, "CRunnableTask (%#lx): resuming task", _dwTaskID); ) _lState = IRTIR_TASK_RUNNING; if (_hDone) ResetEvent(_hDone); return ResumeRT(); } /*---------------------------------------------------------- Purpose: IRunnableTask::IsRunning method */ STDMETHODIMP_( ULONG ) CRunnableTask:: IsRunning ( void ) { return _lState; }