|
|
/*++
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; }
|