Added an implementation of MRSW locks.

This commit is contained in:
Alex Korobka 1999-01-31 09:24:44 +00:00 committed by Alexandre Julliard
parent 17a9e349ff
commit 6479f0fab2
3 changed files with 211 additions and 13 deletions

View File

@ -4,6 +4,7 @@
* This file contains the Rtl* API functions. These should be implementable. * This file contains the Rtl* API functions. These should be implementable.
* *
* Copyright 1996-1998 Marcus Meissner * Copyright 1996-1998 Marcus Meissner
* 1999 Alex Korobka
*/ */
#include <stdlib.h> #include <stdlib.h>
@ -833,13 +834,6 @@ DWORD WINAPI RtlTimeToTimeFields(DWORD x1,DWORD x2) {
FIXME(ntdll,"(0x%08lx,0x%08lx),stub!\n",x1,x2); FIXME(ntdll,"(0x%08lx,0x%08lx),stub!\n",x1,x2);
return 0; return 0;
} }
/******************************************************************************
* RtlInitializeResource [NTDLL]
*/
NTSTATUS WINAPI RtlInitializeResource(DWORD x1) {
FIXME(ntdll,"(0x%08lx),stub!\n",x1);
return 0;
}
/****************************************************************************** /******************************************************************************
* RtlCompareUnicodeString [NTDLL] * RtlCompareUnicodeString [NTDLL]
*/ */
@ -858,3 +852,189 @@ void __cdecl DbgPrint(LPCSTR fmt,LPVOID args) {
MSG("DbgPrint says: %s",buf); MSG("DbgPrint says: %s",buf);
/* hmm, raise exception? */ /* hmm, raise exception? */
} }
/***********************************************************************
* RtlInitializeResource (NTDLL.409)
*
* xxxResource() functions implement multiple-reader-single-writer lock.
* The code is based on information published in WDJ January 1999 issue.
*/
void WINAPI RtlInitializeResource(LPRTL_RWLOCK rwl)
{
if( rwl )
{
rwl->iNumberActive = 0;
rwl->uExclusiveWaiters = 0;
rwl->uSharedWaiters = 0;
rwl->hOwningThreadId = 0;
rwl->dwTimeoutBoost = 0; /* no info on this one, default value is 0 */
InitializeCriticalSection( &rwl->rtlCS );
rwl->hExclusiveReleaseSemaphore = CreateSemaphore32A( NULL, 0, 65535, NULL );
rwl->hSharedReleaseSemaphore = CreateSemaphore32A( NULL, 0, 65535, NULL );
}
}
/***********************************************************************
* RtlDeleteResource (NTDLL.330)
*/
void WINAPI RtlDeleteResource(LPRTL_RWLOCK rwl)
{
if( rwl )
{
EnterCriticalSection( &rwl->rtlCS );
if( rwl->iNumberActive || rwl->uExclusiveWaiters || rwl->uSharedWaiters )
MSG("Deleting active MRSW lock (%p), expect failure\n", rwl );
rwl->hOwningThreadId = 0;
rwl->uExclusiveWaiters = rwl->uSharedWaiters = 0;
rwl->iNumberActive = 0;
CloseHandle( rwl->hExclusiveReleaseSemaphore );
CloseHandle( rwl->hSharedReleaseSemaphore );
LeaveCriticalSection( &rwl->rtlCS );
DeleteCriticalSection( &rwl->rtlCS );
}
}
/***********************************************************************
* RtlAcquireResourceExclusive (NTDLL.256)
*/
BYTE WINAPI RtlAcquireResourceExclusive(LPRTL_RWLOCK rwl, BYTE fWait)
{
BYTE retVal = 0;
if( !rwl ) return 0;
start:
EnterCriticalSection( &rwl->rtlCS );
if( rwl->iNumberActive == 0 ) /* lock is free */
{
rwl->iNumberActive = -1;
retVal = 1;
}
else if( rwl->iNumberActive < 0 ) /* exclusive lock in progress */
{
if( rwl->hOwningThreadId == GetCurrentThreadId() )
{
retVal = 1;
rwl->iNumberActive--;
goto done;
}
wait:
if( fWait )
{
rwl->uExclusiveWaiters++;
LeaveCriticalSection( &rwl->rtlCS );
if( WaitForSingleObject( rwl->hExclusiveReleaseSemaphore, INFINITE32 ) == WAIT_FAILED )
goto done;
goto start; /* restart the acquisition to avoid deadlocks */
}
}
else /* one or more shared locks are in progress */
if( fWait )
goto wait;
if( retVal == 1 )
rwl->hOwningThreadId = GetCurrentThreadId();
done:
LeaveCriticalSection( &rwl->rtlCS );
return retVal;
}
/***********************************************************************
* RtlAcquireResourceShared (NTDLL.257)
*/
BYTE WINAPI RtlAcquireResourceShared(LPRTL_RWLOCK rwl, BYTE fWait)
{
DWORD dwWait = WAIT_FAILED;
BYTE retVal = 0;
if( !rwl ) return 0;
start:
EnterCriticalSection( &rwl->rtlCS );
if( rwl->iNumberActive < 0 )
{
if( rwl->hOwningThreadId == GetCurrentThreadId() )
{
rwl->iNumberActive--;
retVal = 1;
goto done;
}
if( fWait )
{
rwl->uSharedWaiters++;
LeaveCriticalSection( &rwl->rtlCS );
if( (dwWait = WaitForSingleObject( rwl->hSharedReleaseSemaphore, INFINITE32 )) == WAIT_FAILED )
goto done;
goto start;
}
}
else
{
if( dwWait != WAIT_OBJECT_0 ) /* otherwise RtlReleaseResource() has already done it */
rwl->iNumberActive++;
retVal = 1;
}
done:
LeaveCriticalSection( &rwl->rtlCS );
return retVal;
}
/***********************************************************************
* RtlReleaseResource (NTDLL.471)
*/
void WINAPI RtlReleaseResource(LPRTL_RWLOCK rwl)
{
EnterCriticalSection( &rwl->rtlCS );
if( rwl->iNumberActive > 0 ) /* have one or more readers */
{
if( --rwl->iNumberActive == 0 )
{
if( rwl->uExclusiveWaiters )
{
wake_exclusive:
rwl->uExclusiveWaiters--;
ReleaseSemaphore( rwl->hExclusiveReleaseSemaphore, 1, NULL );
}
}
}
else
if( rwl->iNumberActive < 0 ) /* have a writer, possibly recursive */
{
if( ++rwl->iNumberActive == 0 )
{
rwl->hOwningThreadId = 0;
if( rwl->uExclusiveWaiters )
goto wake_exclusive;
else
if( rwl->uSharedWaiters )
{
UINT32 n = rwl->uSharedWaiters;
rwl->iNumberActive = rwl->uSharedWaiters; /* prevent new writers from joining until
* all queued readers have done their thing */
rwl->uSharedWaiters = 0;
ReleaseSemaphore( rwl->hSharedReleaseSemaphore, n, NULL );
}
}
}
LeaveCriticalSection( &rwl->rtlCS );
}
/***********************************************************************
* RtlDumpResource (NTDLL.340)
*/
void WINAPI RtlDumpResource(LPRTL_RWLOCK rwl)
{
if( rwl )
{
MSG("RtlDumpResource(%p):\n\tactive count = %i\n\twaiting readers = %i\n\twaiting writers = %i\n",
rwl, rwl->iNumberActive, rwl->uSharedWaiters, rwl->uExclusiveWaiters );
if( rwl->iNumberActive )
MSG("\towner thread = %08x\n", rwl->hOwningThreadId );
}
}

View File

@ -168,6 +168,24 @@ typedef struct _UNICODE_STRING {
LPWSTR Buffer; LPWSTR Buffer;
} UNICODE_STRING,*LPUNICODE_STRING; } UNICODE_STRING,*LPUNICODE_STRING;
typedef struct _RTL_RWLOCK {
CRITICAL_SECTION rtlCS;
HANDLE32 hSharedReleaseSemaphore;
UINT32 uSharedWaiters;
HANDLE32 hExclusiveReleaseSemaphore;
UINT32 uExclusiveWaiters;
INT32 iNumberActive;
HANDLE32 hOwningThreadId;
DWORD dwTimeoutBoost;
PVOID pDebugInfo;
} RTL_RWLOCK, *LPRTL_RWLOCK;
VOID WINAPI RtlInitializeResource(LPRTL_RWLOCK);
VOID WINAPI RtlDeleteResource(LPRTL_RWLOCK);
BYTE WINAPI RtlAcquireResourceExclusive(LPRTL_RWLOCK, BYTE fWait);
BYTE WINAPI RtlAcquireResourceShared(LPRTL_RWLOCK, BYTE fWait);
VOID WINAPI RtlReleaseResource(LPRTL_RWLOCK);
VOID WINAPI RtlDumpResource(LPRTL_RWLOCK);
BOOL32 WINAPI IsValidSid(LPSID); BOOL32 WINAPI IsValidSid(LPSID);
BOOL32 WINAPI EqualSid(LPSID,LPSID); BOOL32 WINAPI EqualSid(LPSID,LPSID);

View File

@ -261,8 +261,8 @@ type win32
253 stub RtlAbortRXact 253 stub RtlAbortRXact
254 stub RtlAbsoluteToSelfRelativeSD 254 stub RtlAbsoluteToSelfRelativeSD
255 stdcall RtlAcquirePebLock() RtlAcquirePebLock 255 stdcall RtlAcquirePebLock() RtlAcquirePebLock
256 stub RtlAcquireResourceExclusive 256 stdcall RtlAcquireResourceExclusive(ptr long) RtlAcquireResourceExclusive
257 stub RtlAcquireResourceShared 257 stdcall RtlAcquireResourceShared(ptr long) RtlAcquireResourceShared
258 stdcall RtlAddAccessAllowedAce(long long long long) RtlAddAccessAllowedAce 258 stdcall RtlAddAccessAllowedAce(long long long long) RtlAddAccessAllowedAce
259 stub RtlAddAccessDeniedAce 259 stub RtlAddAccessDeniedAce
260 stdcall RtlAddAce(ptr long long ptr long) RtlAddAce 260 stdcall RtlAddAce(ptr long long ptr long) RtlAddAce
@ -335,7 +335,7 @@ type win32
327 stdcall RtlDeleteCriticalSection(ptr) DeleteCriticalSection 327 stdcall RtlDeleteCriticalSection(ptr) DeleteCriticalSection
328 stub RtlDeleteElementGenericTable 328 stub RtlDeleteElementGenericTable
329 stub RtlDeleteRegistryValue 329 stub RtlDeleteRegistryValue
330 stub RtlDeleteResource 330 stdcall RtlDeleteResource(ptr) RtlDeleteResource
331 stdcall RtlDeleteSecurityObject(long) RtlDeleteSecurityObject 331 stdcall RtlDeleteSecurityObject(long) RtlDeleteSecurityObject
332 stdcall RtlDestroyEnvironment(long) RtlDestroyEnvironment 332 stdcall RtlDestroyEnvironment(long) RtlDestroyEnvironment
333 stdcall RtlDestroyHeap(long) HeapDestroy 333 stdcall RtlDestroyHeap(long) HeapDestroy
@ -345,7 +345,7 @@ type win32
337 stub RtlDoesFileExists_U 337 stub RtlDoesFileExists_U
338 stdcall RtlDosPathNameToNtPathName_U(ptr ptr long long) RtlDosPathNameToNtPathName_U 338 stdcall RtlDosPathNameToNtPathName_U(ptr ptr long long) RtlDosPathNameToNtPathName_U
339 stub RtlDosSearchPath_U 339 stub RtlDosSearchPath_U
340 stub RtlDumpResource 340 stdcall RtlDumpResource(ptr) RtlDumpResource
341 stub RtlEnlargedIntegerMultiply 341 stub RtlEnlargedIntegerMultiply
342 stub RtlEnlargedUnsignedDivide 342 stub RtlEnlargedUnsignedDivide
343 stub RtlEnlargedUnsignedMultiply 343 stub RtlEnlargedUnsignedMultiply
@ -414,7 +414,7 @@ type win32
406 stdcall RtlInitializeCriticalSection(ptr) InitializeCriticalSection 406 stdcall RtlInitializeCriticalSection(ptr) InitializeCriticalSection
407 stub RtlInitializeGenericTable 407 stub RtlInitializeGenericTable
408 stub RtlInitializeRXact 408 stub RtlInitializeRXact
409 stdcall RtlInitializeResource(long) RtlInitializeResource 409 stdcall RtlInitializeResource(ptr) RtlInitializeResource
410 stdcall RtlInitializeSid(ptr ptr long) RtlInitializeSid 410 stdcall RtlInitializeSid(ptr ptr long) RtlInitializeSid
411 stub RtlInsertElementGenericTable 411 stub RtlInsertElementGenericTable
412 stdcall RtlIntegerToChar(long long long long) RtlIntegerToChar 412 stdcall RtlIntegerToChar(long long long long) RtlIntegerToChar
@ -476,7 +476,7 @@ type win32
468 stub RtlRealPredecessor 468 stub RtlRealPredecessor
469 stub RtlRealSuccessor 469 stub RtlRealSuccessor
470 stdcall RtlReleasePebLock() RtlReleasePebLock 470 stdcall RtlReleasePebLock() RtlReleasePebLock
471 stub RtlReleaseResource 471 stdcall RtlReleaseResource(ptr) RtlReleaseResource
472 stub RtlRemoteCall 472 stub RtlRemoteCall
473 stub RtlResetRtlTranslations 473 stub RtlResetRtlTranslations
474 stub RtlRunDecodeUnicodeString 474 stub RtlRunDecodeUnicodeString