ProcessHacker/phlib/fastlock.c
2025-05-13 19:45:22 +03:00

385 lines
9.6 KiB
C

/*
* Process Hacker -
* fast resource lock
*
* Copyright (C) 2009-2010 wj32
*
* This file is part of Process Hacker.
*
* Process Hacker is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Process Hacker is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Process Hacker. If not, see <http://www.gnu.org/licenses/>.
*/
#include <phbase.h>
#include <fastlock.h>
// FastLock is a port of FastResourceLock from PH 1.x.
//
// The code contains no comments because it is a direct port. Please see FastResourceLock.cs in PH
// 1.x for details.
// The fast lock is around 7% faster than the critical section when there is no contention, when
// used solely for mutual exclusion. It is also much smaller than the critical section.
#define PH_LOCK_OWNED 0x1
#define PH_LOCK_EXCLUSIVE_WAKING 0x2
#define PH_LOCK_SHARED_OWNERS_SHIFT 2
#define PH_LOCK_SHARED_OWNERS_MASK 0x3ff
#define PH_LOCK_SHARED_OWNERS_INC 0x4
#define PH_LOCK_SHARED_WAITERS_SHIFT 12
#define PH_LOCK_SHARED_WAITERS_MASK 0x3ff
#define PH_LOCK_SHARED_WAITERS_INC 0x1000
#define PH_LOCK_EXCLUSIVE_WAITERS_SHIFT 22
#define PH_LOCK_EXCLUSIVE_WAITERS_MASK 0x3ff
#define PH_LOCK_EXCLUSIVE_WAITERS_INC 0x400000
#define PH_LOCK_EXCLUSIVE_MASK \
(PH_LOCK_EXCLUSIVE_WAKING | \
(PH_LOCK_EXCLUSIVE_WAITERS_MASK << PH_LOCK_EXCLUSIVE_WAITERS_SHIFT))
VOID PhInitializeFastLock(
_Out_ PPH_FAST_LOCK FastLock
)
{
FastLock->Value = 0;
FastLock->ExclusiveWakeEvent = NULL;
FastLock->SharedWakeEvent = NULL;
}
VOID PhDeleteFastLock(
_Inout_ PPH_FAST_LOCK FastLock
)
{
if (FastLock->ExclusiveWakeEvent)
{
NtClose(FastLock->ExclusiveWakeEvent);
FastLock->ExclusiveWakeEvent = NULL;
}
if (FastLock->SharedWakeEvent)
{
NtClose(FastLock->SharedWakeEvent);
FastLock->SharedWakeEvent = NULL;
}
}
FORCEINLINE VOID PhpEnsureEventCreated(
_Inout_ PHANDLE Handle
)
{
HANDLE handle;
if (*Handle != NULL)
return;
NtCreateSemaphore(&handle, SEMAPHORE_ALL_ACCESS, NULL, 0, MAXLONG);
if (_InterlockedCompareExchangePointer(
Handle,
handle,
NULL
) != NULL)
{
NtClose(handle);
}
}
FORCEINLINE ULONG PhpGetSpinCount(
VOID
)
{
if ((ULONG)PhSystemBasicInformation.NumberOfProcessors > 1)
return 4000;
else
return 0;
}
_May_raise_
_Acquires_exclusive_lock_(*FastLock)
VOID FASTCALL PhfAcquireFastLockExclusive(
_Inout_ PPH_FAST_LOCK FastLock
)
{
ULONG value;
ULONG i = 0;
ULONG spinCount;
spinCount = PhpGetSpinCount();
while (TRUE)
{
value = FastLock->Value;
if (!(value & (PH_LOCK_OWNED | PH_LOCK_EXCLUSIVE_WAKING)))
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_OWNED,
value
) == value)
break;
}
else if (i >= spinCount)
{
PhpEnsureEventCreated(&FastLock->ExclusiveWakeEvent);
if (_InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_EXCLUSIVE_WAITERS_INC,
value
) == value)
{
if (NtWaitForSingleObject(
FastLock->ExclusiveWakeEvent,
FALSE,
NULL
) != STATUS_WAIT_0)
PhRaiseStatus(STATUS_UNSUCCESSFUL);
do
{
value = FastLock->Value;
} while (_InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_OWNED - PH_LOCK_EXCLUSIVE_WAKING,
value
) != value);
break;
}
}
i++;
YieldProcessor();
}
}
_May_raise_
_Acquires_shared_lock_(*FastLock)
VOID FASTCALL PhfAcquireFastLockShared(
_Inout_ PPH_FAST_LOCK FastLock
)
{
ULONG value;
ULONG i = 0;
ULONG spinCount;
spinCount = PhpGetSpinCount();
while (TRUE)
{
value = FastLock->Value;
if (!(value & (
PH_LOCK_OWNED |
(PH_LOCK_SHARED_OWNERS_MASK << PH_LOCK_SHARED_OWNERS_SHIFT) |
PH_LOCK_EXCLUSIVE_MASK
)))
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_OWNED + PH_LOCK_SHARED_OWNERS_INC,
value
) == value)
break;
}
else if (
(value & PH_LOCK_OWNED) &&
((value >> PH_LOCK_SHARED_OWNERS_SHIFT) & PH_LOCK_SHARED_OWNERS_MASK) > 0 &&
!(value & PH_LOCK_EXCLUSIVE_MASK)
)
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_SHARED_OWNERS_INC,
value
) == value)
break;
}
else if (i >= spinCount)
{
PhpEnsureEventCreated(&FastLock->SharedWakeEvent);
if (_InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_SHARED_WAITERS_INC,
value
) == value)
{
if (NtWaitForSingleObject(
FastLock->SharedWakeEvent,
FALSE,
NULL
) != STATUS_WAIT_0)
PhRaiseStatus(STATUS_UNSUCCESSFUL);
continue;
}
}
i++;
YieldProcessor();
}
}
_Releases_exclusive_lock_(*FastLock)
VOID FASTCALL PhfReleaseFastLockExclusive(
_Inout_ PPH_FAST_LOCK FastLock
)
{
ULONG value;
while (TRUE)
{
value = FastLock->Value;
if ((value >> PH_LOCK_EXCLUSIVE_WAITERS_SHIFT) & PH_LOCK_EXCLUSIVE_WAITERS_MASK)
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value - PH_LOCK_OWNED + PH_LOCK_EXCLUSIVE_WAKING - PH_LOCK_EXCLUSIVE_WAITERS_INC,
value
) == value)
{
NtReleaseSemaphore(FastLock->ExclusiveWakeEvent, 1, NULL);
break;
}
}
else
{
ULONG sharedWaiters;
sharedWaiters = (value >> PH_LOCK_SHARED_WAITERS_SHIFT) & PH_LOCK_SHARED_WAITERS_MASK;
if (_InterlockedCompareExchange(
&FastLock->Value,
value & ~(PH_LOCK_OWNED | (PH_LOCK_SHARED_WAITERS_MASK << PH_LOCK_SHARED_WAITERS_SHIFT)),
value
) == value)
{
if (sharedWaiters)
NtReleaseSemaphore(FastLock->SharedWakeEvent, sharedWaiters, 0);
break;
}
}
YieldProcessor();
}
}
_Releases_shared_lock_(*FastLock)
VOID FASTCALL PhfReleaseFastLockShared(
_Inout_ PPH_FAST_LOCK FastLock
)
{
ULONG value;
while (TRUE)
{
value = FastLock->Value;
if (((value >> PH_LOCK_SHARED_OWNERS_SHIFT) & PH_LOCK_SHARED_OWNERS_MASK) > 1)
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value - PH_LOCK_SHARED_OWNERS_INC,
value
) == value)
break;
}
else if ((value >> PH_LOCK_EXCLUSIVE_WAITERS_SHIFT) & PH_LOCK_EXCLUSIVE_WAITERS_MASK)
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value - PH_LOCK_OWNED + PH_LOCK_EXCLUSIVE_WAKING -
PH_LOCK_SHARED_OWNERS_INC - PH_LOCK_EXCLUSIVE_WAITERS_INC,
value
) == value)
{
NtReleaseSemaphore(FastLock->ExclusiveWakeEvent, 1, NULL);
break;
}
}
else
{
if (_InterlockedCompareExchange(
&FastLock->Value,
value - PH_LOCK_OWNED - PH_LOCK_SHARED_OWNERS_INC,
value
) == value)
break;
}
YieldProcessor();
}
}
_When_(return != 0, _Acquires_exclusive_lock_(*FastLock))
BOOLEAN FASTCALL PhfTryAcquireFastLockExclusive(
_Inout_ PPH_FAST_LOCK FastLock
)
{
ULONG value;
value = FastLock->Value;
if (value & (PH_LOCK_OWNED | PH_LOCK_EXCLUSIVE_WAKING))
return FALSE;
return _InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_OWNED,
value
) == value;
}
_When_(return != 0, _Acquires_shared_lock_(*FastLock))
BOOLEAN FASTCALL PhfTryAcquireFastLockShared(
_Inout_ PPH_FAST_LOCK FastLock
)
{
ULONG value;
value = FastLock->Value;
if (value & PH_LOCK_EXCLUSIVE_MASK)
return FALSE;
if (!(value & PH_LOCK_OWNED))
{
return _InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_OWNED + PH_LOCK_SHARED_OWNERS_INC,
value
) == value;
}
else if ((value >> PH_LOCK_SHARED_OWNERS_SHIFT) & PH_LOCK_SHARED_OWNERS_MASK)
{
return _InterlockedCompareExchange(
&FastLock->Value,
value + PH_LOCK_SHARED_OWNERS_INC,
value
) == value;
}
else
{
return FALSE;
}
}