2020-09-30 16:53:55 +02:00

300 lines
5.2 KiB
C++

/*++
rwnew.h
This file defines several variations of Reader/Writer locks
with different properties regarding handles used, and other
implementation details.
Also defined are some variations of CRITICAL_SECTIONS which use
fewer or no handles.
--*/
#ifndef _RWNEW_H
#define _RWNEW_H
#ifdef _RW_IMPLEMENTATION_
#define _RW_INTERFACE_ __declspec( dllexport )
#else
#define _RW_INTERFACE_ __declspec( dllimport )
#endif
#pragma warning( disable:4251 )
#include <limits.h>
#include "lockq.h"
#include "rwintrnl.h"
class _RW_INTERFACE_ CCritSection {
private :
//
// Handle of thread owning the lock !
//
HANDLE m_hOwner ;
//
// Count of Recursive Calls
//
long m_RecursionCount ;
//
// Count used to see who gets the lock next !
//
long m_lock ;
//
// Queue of waiting threads
//
CSingleReleaseQueue m_queue ;
//
// Copying of these objects is not allowed !!!!
//
CCritSection( CCritSection& ) ;
CCritSection& operator=( CCritSection& ) ;
public :
#ifdef DEBUG
DWORD m_dwThreadOwner ;
#endif
//
// Construct a critical section object
//
CCritSection( ) :
m_queue( FALSE ),
m_hOwner( INVALID_HANDLE_VALUE ),
m_RecursionCount( 0 ),
m_lock( -1 ) {
}
//
// Acquire the critical section
//
void
Enter(
CWaitingThread& myself
) ;
//
// Another version which acquires the critical section -
// creates its own CWaitingThread object !
//
void
Enter() ;
//
// REturns TRUE if the lock is available right now !
//
BOOL
TryEnter(
CWaitingThread& myself
) ;
//
// Returns TRUE if we can get the lock right now !
//
BOOL
TryEnter() {
CWaitingThread myself ;
return TryEnter( myself ) ;
}
//
// Release the critical section !
//
void
Leave() ;
} ;
//
// This version of critical section is more like an event - doesn't
// care who releases locks - and doesn't handle recursive grabs !
//
class _RW_INTERFACE_ CSimpleCritSection {
private :
//
// Count used to see who gets the lock next !
//
long m_lock ;
//
// Queue of waiting threads
//
CSingleReleaseQueue m_queue ;
//
// Copying of these objects is not allowed !!!!
//
CSimpleCritSection( CCritSection& ) ;
CSimpleCritSection& operator=( CCritSection& ) ;
public :
//
// Construct a critical section object
//
CSimpleCritSection( ) :
m_queue( FALSE ),
m_lock( -1 ) {
}
//
// Acquire the critical section
//
void
Enter(
CWaitingThread& myself
) ;
//
// Another version which acquires the critical section -
// creates its own CWaitingThread object !
//
void
Enter() ;
//
// REturns TRUE if the lock is available right now !
//
BOOL
TryEnter(
CWaitingThread& myself
) ;
//
// Returns TRUE if we can get the lock right now !
//
BOOL
TryEnter() {
CWaitingThread myself ;
return TryEnter( myself ) ;
}
//
// Release the critical section !
//
void
Leave() ;
} ;
//
// Another class which tries to create Reader/Write locks with
// no handles !!
//
class _RW_INTERFACE_ CShareLockNH {
private :
//
// Lock grabbed by writers to have exclusive access
//
CSimpleCritSection m_lock ;
//
// Number of readers who have grabbed the Read Lock -
// Negative if a writer is waiting !
//
volatile long m_cReadLock ;
//
// Number of Readers who have left the lock since a
// writer tried to grab it !
//
volatile long m_cOutReaders ;
//
// Number of readers who are entering the lock after
// being blocked !!!
//
volatile long m_cOutAcquiringReaders ;
//
// Handle that all the readers who are waiting try to grab !
//
volatile HANDLE m_hWaitingReaders ;
//
// Handle that the single writer waiting for the lock is trying
// to grab !
//
volatile HANDLE m_hWaitingWriters ;
void inline
WakeReaders() ;
//
// The internal work of ShareLock - does a lot more of the stuff required
// when a writer is present !!!
//
void
ShareLockInternal() ;
//
// The internal work of ShareLock - does a lot more of the stuff required
// when a writer is present !!!
//
void
ShareUnlockInternal() ;
public :
//
// Construction of CShareLockNH() objects always succeeds and there
// are no error cases !
//
CShareLockNH() ;
//
// Grab the lock Shared - other threads may pass through ShareLock() as well
//
void ShareLock() ;
//
// Releases the lock - if we are the last reader to leave writers may
// start to enter the lock !
//
void ShareUnlock() ;
//
// Grab the lock Exclusively - no other readers or writers may enter !!
//
void ExclusiveLock() ;
//
// Release the Exclusive Locks - if there are readers waiting they
// will enter before other waiting writers !
//
void ExclusiveUnlock() ;
//
// Convert an ExclusiveLock to a Shared - this cannot fail !
//
void ExclusiveToShared() ;
//
// Convert a Shared Lock to an Exclusive one - this can fail - returns
// TRUE if successfull !
//
BOOL SharedToExclusive() ;
BOOL TryShareLock() ;
BOOL TryExclusiveLock() ;
} ;
#endif // _RWNEW_H_