Windows2000/private/inet/wininet/inc/reslock.hxx
2020-09-30 17:12:32 +02:00

191 lines
3.6 KiB
C++

/*++
Copyright (c) 1996 Microsoft Corporation
Module Name:
reslock.hxx
Abstract:
Contains RESOURCE_LOCK class
Author:
Richard L Firth (rfirth) 18-Jun-1996
Revision History:
18-Jun-1996 rfirth
Created
--*/
// manifests
#if INET_DEBUG
#define DEFAULT_RESOURCE_LOCK_TIMEOUT 30000 // 30 seconds
#else
#define DEFAULT_RESOURCE_LOCK_TIMEOUT INFINITE
#endif
// class definition
class RESOURCE_LOCK {
private:
//
// _Initialized - TRUE when the critical section & event have been created
//
BOOL _Initialized;
//
// _WriteCount - number of times writer has re-entered
//
LONG _WriteCount;
//
// _ThreadId - ID of the owning writer thread
//
DWORD _ThreadId;
//
// _Readers - number of reader threads. Used as a switch in concert with
// InterlockedIncrement/InterlockedDecrement: -1 => 0 means this is the
// first reader, and we reset the writer wait event; 0 => -1 means this is
// the last reader and we set the writer wait event
//
LONG _Readers;
//
// _Timeout - number of milliseconds to wait for one of the event handles
//
DWORD _Timeout;
//
// _hWriteEvent - signalled when the last reader thread exits. At this time
// the writer thread has exclusive access
//
HANDLE _hWriteEvent;
//
// _CritSect - used to serialize access. Writer keeps this until Releas()ed.
// Readers just keep it long enough to update _Readers variable
//
CRITICAL_SECTION _CritSect;
//
// Wait - wait for an event to become signalled
//
BOOL Wait(HANDLE hEvent) {
DWORD error = WaitForSingleObject(hEvent, _Timeout);
if (error == WAIT_OBJECT_0) {
return TRUE;
} else {
DEBUG_PRINT(UTIL,
ERROR,
("RESOURCE_LOCK::Wait(): WaitForSingleObject() returns %d\n",
error
));
if (error == WAIT_TIMEOUT) {
error = ERROR_INTERNET_TIMEOUT;
} else {
error = ERROR_INTERNET_INTERNAL_ERROR;
}
SetLastError(error);
return FALSE;
}
}
VOID SetTimeout(DWORD Timeout) {
_Timeout = Timeout;
}
DWORD GetTimeout(VOID) const {
return _Timeout;
}
public:
RESOURCE_LOCK() {
_WriteCount = 0;
_ThreadId = 0;
_Readers = -1;
_Timeout = DEFAULT_RESOURCE_LOCK_TIMEOUT;
_hWriteEvent = NULL;
_Initialized = FALSE;
}
~RESOURCE_LOCK() {
INET_ASSERT(_WriteCount == 0);
INET_ASSERT(_ThreadId == 0);
INET_ASSERT(_Readers == -1);
if (_Initialized) {
DeleteCriticalSection(&_CritSect);
if (_hWriteEvent != NULL) {
CloseHandle(_hWriteEvent);
}
}
}
VOID Initialize(VOID) {
_WriteCount = 0;
_ThreadId = 0;
_Readers = -1;
_Timeout = DEFAULT_RESOURCE_LOCK_TIMEOUT;
InitializeCriticalSection(&_CritSect);
//
// _hWriteEvent is an auto-reset, initially-signalled event
//
_hWriteEvent = CreateEvent(NULL, FALSE, TRUE, NULL);
_Initialized = TRUE;
}
BOOL IsInitialized(VOID) const {
return _Initialized;
}
BOOL IsValid(VOID) const {
return (_hWriteEvent != NULL) ? TRUE : FALSE;
}
BOOL
Acquire(
IN BOOL bExclusiveMode = FALSE
);
BOOL AcquireExclusive(VOID) {
return Acquire(TRUE);
}
VOID
Release(
VOID
);
};