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

801 lines
15 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1994 Microsoft Corporation
Module Name:
serialst.cxx
Abstract:
Functions to deal with a serialized list. These are replaced by macros in
the retail version, except for functions no longer inlined due to critsec
wrapper
Contents:
[InitializeSerializedList]
[TerminateSerializedList]
[LockSerializedList]
[UnlockSerializedList]
[InsertAtHeadOfSerializedList]
[InsertAtTailOfSerializedList]
[RemoveFromSerializedList]
[IsSerializedListEmpty]
[HeadOfSerializedList]
[TailOfSerializedList]
[CheckEntryOnSerializedList]
[(CheckEntryOnList)]
SlDequeueHead
SlDequeueTail
IsOnSerializedList
Author:
Richard L Firth (rfirth) 16-Feb-1995
Environment:
Win-32 user level
Revision History:
16-Feb-1995 rfirth
Created
--*/
#include <wininetp.h>
#if INET_DEBUG
//
// manifests
//
#define SERIALIZED_LIST_SIGNATURE 'tslS'
//
// private prototypes
//
PRIVATE
DEBUG_FUNCTION
BOOL
CheckEntryOnList(
IN PLIST_ENTRY List,
IN PLIST_ENTRY Entry,
IN BOOL ExpectedResult
);
//
// data
//
BOOL fCheckEntryOnList = FALSE;
BOOL ReportCheckEntryOnListErrors = FALSE;
//
// functions
//
DEBUG_FUNCTION
BOOL
InitializeSerializedList(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
initializes a serialized list
Arguments:
SerializedList - pointer to SERIALIZED_LIST
Return Value:
None.
--*/
{
INET_ASSERT(SerializedList != NULL);
SerializedList->Signature = SERIALIZED_LIST_SIGNATURE;
SerializedList->LockCount = 0;
INITIALIZE_RESOURCE_INFO(&SerializedList->ResourceInfo);
InitializeListHead(&SerializedList->List);
SerializedList->ElementCount = 0;
return SerializedList->Lock.Init();
}
DEBUG_FUNCTION
VOID
TerminateSerializedList(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Undoes InitializeSerializeList
Arguments:
SerializedList - pointer to serialized list to terminate
Return Value:
None.
--*/
{
INET_ASSERT(SerializedList != NULL);
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
INET_ASSERT(SerializedList->ElementCount == 0);
if (SerializedList->ElementCount != 0) {
DEBUG_PRINT(SERIALST,
ERROR,
("list @ %#x has %d elements, first is %#x\n",
SerializedList,
SerializedList->ElementCount,
SerializedList->List.Flink
));
} else {
INET_ASSERT(IsListEmpty(&SerializedList->List));
}
SerializedList->Lock.FreeLock();
}
DEBUG_FUNCTION
BOOL
LockSerializedList(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Acquires a serialized list locks
Arguments:
SerializedList - SERIALIZED_LIST to lock
Return Value:
Success if able to acquire a lock.
--*/
{
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
INET_ASSERT(SerializedList->LockCount >= 0);
if ((!SerializedList->Lock.IsInitialized() && !SerializedList->Lock.Init()) ||
!SerializedList->Lock.Lock())
{
return FALSE;
}
else
{
if (SerializedList->LockCount != 0)
{
INET_ASSERT(SerializedList->ResourceInfo.Tid == GetCurrentThreadId());
}
}
++SerializedList->LockCount;
SerializedList->ResourceInfo.Tid = GetCurrentThreadId();
return TRUE;
}
DEBUG_FUNCTION
VOID
UnlockSerializedList(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Releases a serialized list lock
Arguments:
SerializedList - SERIALIZED_LIST to unlock
Return Value:
None.
--*/
{
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
INET_ASSERT(SerializedList->ResourceInfo.Tid == GetCurrentThreadId());
INET_ASSERT(SerializedList->LockCount > 0);
--SerializedList->LockCount;
SerializedList->Lock.Unlock();
}
DEBUG_FUNCTION
BOOL
InsertAtHeadOfSerializedList(
IN LPSERIALIZED_LIST SerializedList,
IN PLIST_ENTRY Entry
)
/*++
Routine Description:
Adds an item to the head of a serialized list
Arguments:
SerializedList - SERIALIZED_LIST to update
Entry - thing to update it with
Return Value:
FALSE - only if unable to acquire the lock
--*/
{
INET_ASSERT(Entry != &SerializedList->List);
if (LockSerializedList(SerializedList))
{
if (fCheckEntryOnList)
{
CheckEntryOnList(&SerializedList->List, Entry, FALSE);
}
InsertHeadList(&SerializedList->List, Entry);
++SerializedList->ElementCount;
INET_ASSERT(SerializedList->ElementCount > 0);
UnlockSerializedList(SerializedList);
return TRUE;
}
else
{
return FALSE;
}
}
DEBUG_FUNCTION
BOOL
InsertAtTailOfSerializedList(
IN LPSERIALIZED_LIST SerializedList,
IN PLIST_ENTRY Entry
)
/*++
Routine Description:
Adds an item to the head of a serialized list
Arguments:
SerializedList - SERIALIZED_LIST to update
Entry - thing to update it with
Return Value:
FALSE - only if not enough memory was available to insert an item.
--*/
{
INET_ASSERT(Entry != &SerializedList->List);
if (LockSerializedList(SerializedList))
{
if (fCheckEntryOnList) {
CheckEntryOnList(&SerializedList->List, Entry, FALSE);
}
InsertTailList(&SerializedList->List, Entry);
++SerializedList->ElementCount;
INET_ASSERT(SerializedList->ElementCount > 0);
UnlockSerializedList(SerializedList);
return TRUE;
}
return FALSE;
}
BOOL
DEBUG_FUNCTION
RemoveFromSerializedList(
IN LPSERIALIZED_LIST SerializedList,
IN PLIST_ENTRY Entry
)
/*++
Routine Description:
Removes the entry from a serialized list
Arguments:
SerializedList - SERIALIZED_LIST to remove entry from
Entry - pointer to entry to remove
Return Value:
FALSE if unable sycnhronize access to the list due to low-memory.
--*/
{
INET_ASSERT((Entry->Flink != NULL) && (Entry->Blink != NULL));
if (LockSerializedList(SerializedList))
{
if (fCheckEntryOnList)
{
CheckEntryOnList(&SerializedList->List, Entry, TRUE);
}
INET_ASSERT(SerializedList->ElementCount > 0);
RemoveEntryList(Entry);
--SerializedList->ElementCount;
Entry->Flink = NULL;
Entry->Blink = NULL;
UnlockSerializedList(SerializedList);
return TRUE;
}
else
{
return FALSE;
}
}
DEBUG_FUNCTION
BOOL
IsSerializedListEmpty(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Checks if a serialized list contains any elements
Arguments:
SerializedList - pointer to list to check
Return Value:
BOOL
--*/
{
// For simplicity, don't worry about returning additional status.
// Due to this always being tied to additional manipulation,
// the lock has already been acquired in all current cases.
if (!LockSerializedList(SerializedList))
return TRUE;
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
BOOL empty;
if (IsListEmpty(&SerializedList->List)) {
INET_ASSERT(SerializedList->ElementCount == 0);
empty = TRUE;
} else {
INET_ASSERT(SerializedList->ElementCount != 0);
empty = FALSE;
}
UnlockSerializedList(SerializedList);
return empty;
}
DEBUG_FUNCTION
PLIST_ENTRY
HeadOfSerializedList(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Returns the element at the tail of the list, without taking the lock
Arguments:
SerializedList - pointer to SERIALIZED_LIST
Return Value:
PLIST_ENTRY
pointer to element at tail of list
--*/
{
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
return SerializedList->List.Flink;
}
DEBUG_FUNCTION
PLIST_ENTRY
TailOfSerializedList(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Returns the element at the tail of the list, without taking the lock
Arguments:
SerializedList - pointer to SERIALIZED_LIST
Return Value:
PLIST_ENTRY
pointer to element at tail of list
--*/
{
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
return SerializedList->List.Blink;
}
DEBUG_FUNCTION
BOOL
CheckEntryOnSerializedList(
IN LPSERIALIZED_LIST SerializedList,
IN PLIST_ENTRY Entry,
IN BOOL ExpectedResult
)
/*++
Routine Description:
Checks an entry exists (or doesn't exist) on a list
Arguments:
SerializedList - pointer to serialized list
Entry - pointer to entry
ExpectedResult - TRUE if expected on list, else FALSE
Return Value:
BOOL
TRUE - expected result
FALSE - unexpected result
--*/
{
INET_ASSERT(SerializedList->Signature == SERIALIZED_LIST_SIGNATURE);
if (!LockSerializedList(SerializedList))
return FALSE;
BOOL result;
__try {
result = CheckEntryOnList(&SerializedList->List, Entry, ExpectedResult);
} __except(EXCEPTION_EXECUTE_HANDLER) {
DEBUG_PRINT(SERIALST,
FATAL,
("List @ %#x (%d elements) is bad\n",
SerializedList,
SerializedList->ElementCount
));
result = FALSE;
}
ENDEXCEPT
UnlockSerializedList(SerializedList);
return result;
}
PRIVATE
DEBUG_FUNCTION
BOOL
CheckEntryOnList(
IN PLIST_ENTRY List,
IN PLIST_ENTRY Entry,
IN BOOL ExpectedResult
)
{
BOOLEAN found = FALSE;
PLIST_ENTRY p;
if (!IsListEmpty(List)) {
for (p = List->Flink; p != List; p = p->Flink) {
if (p == Entry) {
found = TRUE;
break;
}
}
}
if (found != ExpectedResult) {
if (ReportCheckEntryOnListErrors) {
LPSTR description;
description = found
? "Entry %#x already on list %#x\n"
: "Entry %#x not found on list %#x\n"
;
DEBUG_PRINT(SERIALST,
ERROR,
(description,
Entry,
List
));
DEBUG_BREAK(SERIALST);
}
return FALSE;
}
return TRUE;
}
#else // else !INET_DEBUG
BOOL
InitializeSerializedList(LPSERIALIZED_LIST pList)
{
InitializeListHead(&(pList)->List);
(pList)->ElementCount = 0;
return (pList->Lock).Init();
}
BOOL
InsertAtHeadOfSerializedList(LPSERIALIZED_LIST list, PLIST_ENTRY entry)
{
if (LockSerializedList(list))
{
InsertHeadList(&(list)->List, entry);
++(list)->ElementCount;
UnlockSerializedList(list);
return TRUE;
}
else
{
return FALSE;
}
}
BOOL
InsertAtTailOfSerializedList(LPSERIALIZED_LIST list, PLIST_ENTRY entry)
{
if (LockSerializedList(list))
{
InsertTailList(&(list)->List, entry);
++(list)->ElementCount;
UnlockSerializedList(list);
return TRUE;
}
else
{
return FALSE;
}
}
BOOL
RemoveFromSerializedList(LPSERIALIZED_LIST list, PLIST_ENTRY entry)
{
if (LockSerializedList(list))
{
RemoveEntryList(entry);
--(list)->ElementCount;
UnlockSerializedList(list);
return TRUE;
}
else
{
return FALSE;
}
}
#endif // INET_DEBUG
//
// functions that are always functions
//
LPVOID
SlDequeueHead(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Dequeues the element at the head of the queue and returns its address or
NULL if the queue is empty
Arguments:
SerializedList - pointer to SERIALIZED_LIST to dequeue from
Return Value:
LPVOID
--*/
{
LPVOID entry = NULL;
if (!IsSerializedListEmpty(SerializedList)) {
if (LockSerializedList(SerializedList)) {
if (!IsSerializedListEmpty(SerializedList)) {
entry = (LPVOID)HeadOfSerializedList(SerializedList);
if (!RemoveFromSerializedList(SerializedList, (PLIST_ENTRY)entry))
entry = NULL;
}
UnlockSerializedList(SerializedList);
}
}
return entry;
}
LPVOID
SlDequeueTail(
IN LPSERIALIZED_LIST SerializedList
)
/*++
Routine Description:
Dequeues the element at the tail of the queue and returns its address or
NULL if the queue is empty
Arguments:
SerializedList - pointer to SERIALIZED_LIST to dequeue from
Return Value:
LPVOID
--*/
{
LPVOID entry = NULL;
if (!IsSerializedListEmpty(SerializedList)) {
if (LockSerializedList(SerializedList)) {
if (!IsSerializedListEmpty(SerializedList)) {
entry = (LPVOID)TailOfSerializedList(SerializedList);
if (!RemoveFromSerializedList(SerializedList, (PLIST_ENTRY)entry))
entry = NULL;
}
UnlockSerializedList(SerializedList);
}
} else {
entry = NULL;
}
return entry;
}
BOOL
IsOnSerializedList(
IN LPSERIALIZED_LIST SerializedList,
IN PLIST_ENTRY Entry
)
/*++
Routine Description:
Checks if an entry is on a serialized list. Useful to call before
RemoveFromSerializedList() if multiple threads can remove the element
Arguments:
SerializedList - pointer to SERIALIZED_LIST
Entry - pointer to element to check
Return Value:
BOOL
TRUE - Entry is on SerializedList
FALSE - " " not on "
--*/
{
BOOL onList = FALSE;
LPVOID entry;
if (!IsSerializedListEmpty(SerializedList)) {
if (LockSerializedList(SerializedList)) {
if (!IsSerializedListEmpty(SerializedList)) {
for (PLIST_ENTRY entry = HeadOfSerializedList(SerializedList);
entry != (PLIST_ENTRY)SlSelf(SerializedList);
entry = entry->Flink) {
if (entry == Entry) {
onList = TRUE;
break;
}
}
}
UnlockSerializedList(SerializedList);
}
}
return onList;
}