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.
610 lines
12 KiB
610 lines
12 KiB
/*++
|
|
|
|
Copyright (c) 1996 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
growlist.c
|
|
|
|
Abstract:
|
|
|
|
Simple buffer management functions that maintenence of a list of binary
|
|
objects.
|
|
|
|
Author:
|
|
|
|
08-Aug-1997 jimschm Created
|
|
|
|
Revision History:
|
|
|
|
marcw 2-Sep-1999 Moved over from Win9xUpg project.
|
|
|
|
--*/
|
|
|
|
#include "pch.h"
|
|
|
|
#define INSERT_LAST 0xffffffff
|
|
|
|
PBYTE
|
|
pGlAdd (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN UINT InsertBefore,
|
|
IN PBYTE DataToAdd, OPTIONAL
|
|
IN UINT SizeOfData,
|
|
IN UINT NulBytesToAdd
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
pGlAdd allocates memory for a binary block by using a pool, and then expands
|
|
an array of pointers, maintaining a quick-access list.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to add the entry to
|
|
|
|
InsertBefore - Specifies the index of the array element to insert
|
|
before, or INSERT_LIST to append.
|
|
|
|
DataToAdd - Specifies the binary block of data to add.
|
|
|
|
SizeOfData - Specifies the size of data.
|
|
|
|
NulBytesToAdd - Specifies the number of nul bytes to add to the buffer
|
|
|
|
Return Value:
|
|
|
|
A pointer to the binary block if data was copied into the list, 1 if a list
|
|
item was created but no data was set for the item, or NULL if an error
|
|
occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
PBYTE *Item;
|
|
PBYTE *InsertAt;
|
|
PBYTE Data;
|
|
UINT OldEnd;
|
|
UINT Size;
|
|
UINT TotalSize;
|
|
|
|
TotalSize = SizeOfData + NulBytesToAdd;
|
|
|
|
MYASSERT (TotalSize || !DataToAdd);
|
|
|
|
//
|
|
// Allocate pool if necessary
|
|
//
|
|
|
|
if (!GrowList->ListData) {
|
|
GrowList->ListData = PmCreateNamedPool ("GrowList");
|
|
if (!GrowList->ListData) {
|
|
DEBUGMSG ((DBG_WARNING, "GrowList: Could not allocate pool"));
|
|
return NULL;
|
|
}
|
|
|
|
PmDisableTracking (GrowList->ListData);
|
|
}
|
|
|
|
//
|
|
// Expand list array
|
|
//
|
|
|
|
OldEnd = GrowList->ListArray.End;
|
|
Item = (PBYTE *) GbGrow (&GrowList->ListArray, sizeof (PBYTE));
|
|
if (!Item) {
|
|
DEBUGMSG ((DBG_WARNING, "GrowList: Could not allocate array item"));
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Copy data
|
|
//
|
|
|
|
if (DataToAdd || NulBytesToAdd) {
|
|
Data = PmGetAlignedMemory (GrowList->ListData, TotalSize);
|
|
if (!Data) {
|
|
GrowList->ListArray.End = OldEnd;
|
|
DEBUGMSG ((DBG_WARNING, "GrowList: Could not allocate data block"));
|
|
return NULL;
|
|
}
|
|
|
|
if (DataToAdd) {
|
|
CopyMemory (Data, DataToAdd, SizeOfData);
|
|
}
|
|
if (NulBytesToAdd) {
|
|
ZeroMemory (Data + SizeOfData, NulBytesToAdd);
|
|
}
|
|
} else {
|
|
Data = NULL;
|
|
}
|
|
|
|
//
|
|
// Adjust array
|
|
//
|
|
|
|
Size = GlGetSize (GrowList);
|
|
|
|
if (InsertBefore >= Size) {
|
|
//
|
|
// Append mode
|
|
//
|
|
|
|
*Item = Data;
|
|
|
|
} else {
|
|
//
|
|
// Insert mode
|
|
//
|
|
|
|
InsertAt = (PBYTE *) (GrowList->ListArray.Buf) + InsertBefore;
|
|
MoveMemory (&InsertAt[1], InsertAt, (Size - InsertBefore) * sizeof (PBYTE));
|
|
*InsertAt = Data;
|
|
}
|
|
|
|
return Data ? Data : (PBYTE) 1;
|
|
}
|
|
|
|
|
|
VOID
|
|
GlFree (
|
|
IN PGROWLIST GrowList
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlFree frees the resources allocated by a GROWLIST.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to clean up
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
|
|
{
|
|
GbFree (&GrowList->ListArray);
|
|
if (GrowList->ListData) {
|
|
PmDestroyPool (GrowList->ListData);
|
|
}
|
|
|
|
ZeroMemory (GrowList, sizeof (GROWLIST));
|
|
}
|
|
|
|
|
|
VOID
|
|
GlReset (
|
|
IN OUT PGROWLIST GrowList
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlReset empties the grow list but does not destroy the index buffer or pool.
|
|
It is used to efficiently reuse a list.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to clean up
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
|
|
{
|
|
GrowList->ListArray.End = 0;;
|
|
if (GrowList->ListData) {
|
|
PmEmptyPool (GrowList->ListData);
|
|
}
|
|
}
|
|
|
|
|
|
PBYTE
|
|
GlGetItem (
|
|
IN PGROWLIST GrowList,
|
|
IN UINT Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlGetItem returns a pointer to the block of data for item specified by
|
|
Index.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to access
|
|
|
|
Index - Specifies zero-based index of item in list to access
|
|
|
|
Return Value:
|
|
|
|
A pointer to the item's data, or NULL if the Index does not
|
|
represent an actual item.
|
|
|
|
--*/
|
|
|
|
{
|
|
PBYTE *ItemPtr;
|
|
UINT Size;
|
|
|
|
Size = GlGetSize (GrowList);
|
|
if (Index >= Size) {
|
|
return NULL;
|
|
}
|
|
|
|
ItemPtr = (PBYTE *) (GrowList->ListArray.Buf);
|
|
MYASSERT(ItemPtr);
|
|
|
|
return ItemPtr[Index];
|
|
}
|
|
|
|
|
|
UINT
|
|
GlGetSize (
|
|
IN PGROWLIST GrowList
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlGetSize calculates the number of items in the list.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to calculate the size of
|
|
|
|
Return Value:
|
|
|
|
The number of items in the list, or zero if the list is empty.
|
|
|
|
--*/
|
|
|
|
{
|
|
return GrowList->ListArray.End / sizeof (PBYTE);
|
|
}
|
|
|
|
|
|
PBYTE
|
|
RealGlAppend (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN PBYTE DataToAppend, OPTIONAL
|
|
IN UINT SizeOfData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlAppend appends a black of data as a new list item.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
DataToAppend - Specifies a block of data to be copied
|
|
|
|
SizeOfData - Specifies the number of bytes in DataToAppend
|
|
|
|
Return Value:
|
|
|
|
A pointer to the binary block if data was copied into the list, 1 if a list
|
|
item was created but no data was set for the item, or NULL if an error
|
|
occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
return pGlAdd (GrowList, INSERT_LAST, DataToAppend, SizeOfData, 0);
|
|
}
|
|
|
|
|
|
PBYTE
|
|
RealGlAppendAddNul (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN PBYTE DataToAppend, OPTIONAL
|
|
IN UINT SizeOfData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlAppend appends a black of data as a new list item and appends two zero
|
|
bytes (used for string termination).
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
DataToAppend - Specifies a block of data to be copied
|
|
|
|
SizeOfData - Specifies the number of bytes in DataToAppend
|
|
|
|
Return Value:
|
|
|
|
A pointer to the binary block if data was copied into the list, 1 if a list
|
|
item was created but no data was set for the item, or NULL if an error
|
|
occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
return pGlAdd (GrowList, INSERT_LAST, DataToAppend, SizeOfData, 2);
|
|
}
|
|
|
|
|
|
PBYTE
|
|
RealGlInsert (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN UINT Index,
|
|
IN PBYTE DataToInsert, OPTIONAL
|
|
IN UINT SizeOfData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlAppend inserts a black of data as a new list item, before the specified
|
|
Index.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
Index - Specifies the zero-based index of item to insert ahead of.
|
|
|
|
DataToInsert - Specifies a block of data to be copied
|
|
|
|
SizeOfData - Specifies the number of bytes in DataToInsert
|
|
|
|
Return Value:
|
|
|
|
A pointer to the binary block if data was copied into the list, 1 if a list
|
|
item was created but no data was set for the item, or NULL if an error
|
|
occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Size;
|
|
|
|
Size = GlGetSize (GrowList);
|
|
if (Index > Size) {
|
|
return NULL;
|
|
}
|
|
|
|
return pGlAdd (GrowList, Index, DataToInsert, SizeOfData, 0);
|
|
}
|
|
|
|
|
|
PBYTE
|
|
RealGlInsertAddNul (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN UINT Index,
|
|
IN PBYTE DataToInsert, OPTIONAL
|
|
IN UINT SizeOfData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlAppend inserts a block of data as a new list item, before the specified
|
|
Index. Two zero bytes are appended to the block of data (used for string
|
|
termination).
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
Index - Specifies the zero-based index of item to insert ahead of.
|
|
|
|
DataToInsert - Specifies a block of data to be copied
|
|
|
|
SizeOfData - Specifies the number of bytes in DataToInsert
|
|
|
|
Return Value:
|
|
|
|
A pointer to the binary block if data was copied into the list, 1 if a list
|
|
item was created but no data was set for the item, or NULL if an error
|
|
occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Size;
|
|
|
|
Size = GlGetSize (GrowList);
|
|
if (Index > Size) {
|
|
return NULL;
|
|
}
|
|
|
|
return pGlAdd (GrowList, Index, DataToInsert, SizeOfData, 2);
|
|
}
|
|
|
|
|
|
BOOL
|
|
GlDeleteItem (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN UINT Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlDeleteItem removes an item from the list.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
Index - Specifies the zero-based index of the item to remove.
|
|
|
|
Return Value:
|
|
|
|
TRUE if the data block was removed from the list, or FALSE if
|
|
Index is invalid.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Size;
|
|
PBYTE *DeleteAt;
|
|
|
|
Size = GlGetSize (GrowList);
|
|
if (Size <= Index) {
|
|
return FALSE;
|
|
}
|
|
|
|
DeleteAt = (PBYTE *) (GrowList->ListArray.Buf) + Index;
|
|
if (*DeleteAt) {
|
|
PmReleaseMemory (GrowList->ListData, (PVOID) (*DeleteAt));
|
|
}
|
|
|
|
Size--;
|
|
if (Size > Index) {
|
|
MoveMemory (DeleteAt, &DeleteAt[1], (Size - Index) * sizeof (PBYTE));
|
|
}
|
|
|
|
GrowList->ListArray.End = Size * sizeof (PBYTE);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOL
|
|
GlResetItem (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN UINT Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlResetItem sets the list pointer of the specified item to NULL, freeing the
|
|
memory associated with the item's data.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
Index - Specifies the zero-based index of the item to reset.
|
|
|
|
Return Value:
|
|
|
|
TRUE if the data block was freed and the list element was nulled,
|
|
or FALSE if Index is invalid.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Size;
|
|
PBYTE *ResetAt;
|
|
|
|
Size = GlGetSize (GrowList);
|
|
if (Size <= Index) {
|
|
return FALSE;
|
|
}
|
|
|
|
ResetAt = (PBYTE *) (GrowList->ListArray.Buf) + Index;
|
|
if (*ResetAt) {
|
|
PmReleaseMemory (GrowList->ListData, (PVOID) (*ResetAt));
|
|
}
|
|
|
|
*ResetAt = NULL;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
PBYTE
|
|
RealGlSetItem (
|
|
IN OUT PGROWLIST GrowList,
|
|
IN UINT Index,
|
|
IN PCBYTE DataToCopy,
|
|
IN UINT DataSize
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
GlSetItem replaces the data associated with a list item.
|
|
|
|
Arguments:
|
|
|
|
GrowList - Specifies the list to modify
|
|
|
|
Index - Specifies the zero-based index of the item to remove.
|
|
|
|
DataToCopy - Specifies data to associate with the list item
|
|
|
|
DataSize - Specifies the size of Data
|
|
|
|
Return Value:
|
|
|
|
A pointer to the binary block if data was copied into the list, 1 if a list
|
|
item was created but no data was set for the item, or NULL if an error
|
|
occurred.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT Size;
|
|
PBYTE *ReplaceAt;
|
|
PBYTE Data;
|
|
|
|
MYASSERT (DataSize || !DataToCopy);
|
|
|
|
Size = GlGetSize (GrowList);
|
|
if (Size <= Index) {
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Copy data
|
|
//
|
|
|
|
if (DataToCopy) {
|
|
Data = PmGetAlignedMemory (GrowList->ListData, DataSize);
|
|
if (!Data) {
|
|
DEBUGMSG ((DBG_WARNING, "GrowList: Could not allocate data block (2)"));
|
|
return NULL;
|
|
}
|
|
|
|
CopyMemory (Data, DataToCopy, DataSize);
|
|
} else {
|
|
Data = NULL;
|
|
}
|
|
|
|
//
|
|
// Update list pointer
|
|
//
|
|
|
|
ReplaceAt = (PBYTE *) (GrowList->ListArray.Buf) + Index;
|
|
if (*ReplaceAt) {
|
|
PmReleaseMemory (GrowList->ListData, (PVOID) (*ReplaceAt));
|
|
}
|
|
*ReplaceAt = Data;
|
|
|
|
return Data ? Data : (PBYTE) 1;
|
|
}
|
|
|
|
|
|
|