Source code of Windows XP (NT5)
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
|
|
#ifndef __LOCKS_H__
#define __LOCKS_H__
#include <windows.h>
#include <dothrow.h>
# pragma once
const bool THROW_LOCK = true; const bool NOTHROW_LOCK = false;
class CriticalSection { public: CriticalSection (bool can_throw , DWORD count = 0 ); // Initialize the CRITICAL_SECTION.
~CriticalSection (void); // Implicitly destroy the CRITICAL_SECTION.
bool close (void); // dummy call
bool acquire (void); // Acquire lock ownership ( block if necessary).
bool tryacquire (void); // Conditionally acquire lock ( non blocking ). Returns
// false on failure
bool release (void);
// Release lock
bool acquire_read (void); bool acquire_write (void);
bool tryacquire_read (void); bool tryacquire_write (void);
bool valid() const { return initialized_;}
const CRITICAL_SECTION &lock (void) const; // Return the underlying mutex.
void dump( void ) const ;
private:
CRITICAL_SECTION lock_; bool initialized_; bool can_throw_; void raise_exception();
private: // = Prevent assignment and initialization.
void operator= (CriticalSection &); CriticalSection (const CriticalSection &); };
template <class Lock> bool SleepAndLock( Lock& lock, int timeout ) { if (lock.valid()) for (;!lock.locked();) { Sleep(timeout); lock.acquire(); } return lock.locked(); }
inline CriticalSection::CriticalSection (bool can_throw,DWORD count):initialized_(false),can_throw_(can_throw) { __try { if(!count) InitializeCriticalSection(&lock_); else #ifdef _WIN32_WINNT
#if _WIN32_WINNT > 0x0400
InitializeCriticalSectionAndSpinCount(&lock_,count); #else
InitializeCriticalSection(&lock_); #endif
#else
InitializeCriticalSection(&lock_); #endif
initialized_ = true; } __except(GetExceptionCode() == STATUS_NO_MEMORY) { } if (initialized_ != true) raise_exception(); };
inline CriticalSection::~CriticalSection () { if (initialized_) DeleteCriticalSection(&lock_); }
inline void CriticalSection::raise_exception() { if (can_throw_) throw CX_MemoryException(); };
inline bool CriticalSection::acquire(void) { return acquire_write(); };
inline bool CriticalSection::acquire_read(void) { return acquire_write(); };
#ifndef STATUS_POSSIBLE_DEADLOCK
#define STATUS_POSSIBLE_DEADLOCK (0xC0000194L)
#endif
inline bool CriticalSection::acquire_write(void) { if (initialized_) { bool bBreak = false; __try { EnterCriticalSection(&lock_); return true; } __except((GetExceptionCode() == STATUS_NO_MEMORY)? EXCEPTION_EXECUTE_HANDLER: ((bBreak = (GetExceptionCode() == STATUS_POSSIBLE_DEADLOCK))? EXCEPTION_EXECUTE_HANDLER: EXCEPTION_CONTINUE_SEARCH)) { if (bBreak) DebugBreak(); } } return false; };
inline bool CriticalSection::release(void) { if (initialized_) { LeaveCriticalSection(&lock_); return true; } return false; }
inline bool CriticalSection::tryacquire() { return tryacquire_write(); };
inline bool CriticalSection::tryacquire_read() { return tryacquire_write(); };
inline bool CriticalSection::tryacquire_write() { __try { #ifdef _WIN32_WINNT
#if _WIN32_WINNT > 0x0400
return TRUE == TryEnterCriticalSection(&lock_); #else
return false; #endif
#else
return false; #endif
} __except(GetExceptionCode() == STATUS_NO_MEMORY) { }; return false; };
inline bool CriticalSection::close() { return false; };
#ifdef _WIN32_WINNT
#if _WIN32_WINNT > 0x0400
class ReaderWriter { struct RTL_RESOURCE { struct RTL_RESOURCE_DEBUG; RTL_CRITICAL_SECTION CriticalSection; HANDLE SharedSemaphore; ULONG NumberOfWaitingShared; HANDLE ExclusiveSemaphore; ULONG NumberOfWaitingExclusive; LONG NumberOfActive; HANDLE ExclusiveOwnerThread; ULONG Flags; // See RTL_RESOURCE_FLAG_ equates below.
RTL_RESOURCE_DEBUG* DebugInfo; };
public: ReaderWriter (bool can_throw);
~ReaderWriter (void);
bool close (void);
bool acquire (void);
bool tryacquire (void);
bool release (void);
// Release lock
bool acquire_read (void); bool acquire_write (void);
bool tryacquire_read (void); bool tryacquire_write (void);
void upgrade(); void downgrade();
bool valid() const { return initialized_;}
const ReaderWriter &lock (void) const; // Return the underlying mutex.
void dump( void ) const;
private:
RTL_RESOURCE lock_; bool initialized_; bool can_throw_; void raise_exception();
private: // = Prevent assignment and initialization.
void operator= (ReaderWriter &); ReaderWriter (const ReaderWriter &); };
inline void ReaderWriter::raise_exception() { if (can_throw_) throw CX_MemoryException(); }; #endif
#endif
#endif
|