1
mirror of https://github.com/rapid7/metasploit-payloads synced 2025-01-20 20:37:27 +01:00
metasploit-payloads/c/meterpreter/source/common/list.c

419 lines
8.3 KiB
C
Raw Normal View History

2013-11-06 02:38:22 +01:00
/*!
* @file list.c
* @brief Definitions for functions that operate on lists.
* @details An implementation of a simple thread safe double linked list structure. Can be used as either
* a stack (via pop/push), a queue (via push/shift) or an array (via get/add/insert/remove). If
* performing a group of actions on a list based on results from list actions, acquire the list
* lock before the group of actions and release lock when done.
*/
2013-11-06 02:38:22 +01:00
#include "common.h"
2013-11-06 02:38:22 +01:00
/*!
* @brief Create a thread-safe double linked list.
* @returns A new instance of a linked list.
* @retval NULL Indicates a memory allocation failure.
*/
PLIST list_create(VOID)
{
PLIST pList = (PLIST)malloc(sizeof(LIST));
if (pList != NULL)
{
pList->start = NULL;
pList->end = NULL;
pList->count = 0;
pList->lock = lock_create();
2013-11-06 02:38:22 +01:00
if (pList->lock == NULL)
{
list_destroy(pList);
return NULL;
}
}
return pList;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Destroy an existing linked list.
* @details This destroys all nodes and the list itself but not the data held in the
* linked list. This is the responsibility of the caller to destroy.
* @param list The \c LIST instance to destroy.
*/
VOID list_destroy(PLIST pList)
{
PNODE current_node;
PNODE next_node;
if (pList != NULL)
{
lock_acquire(pList->lock);
current_node = pList->start;
2013-11-06 02:38:22 +01:00
while (current_node != NULL)
{
2013-11-06 02:38:22 +01:00
next_node = current_node->next;
current_node->next = NULL;
2013-11-06 02:38:22 +01:00
current_node->prev = NULL;
2013-11-06 02:38:22 +01:00
free(current_node);
current_node = next_node;
}
pList->count = 0;
lock_release(pList->lock);
2013-11-06 02:38:22 +01:00
lock_destroy(pList->lock);
free(pList);
}
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Get the number of items in the list.
* @param pList The \c LIST to get a count of.
2013-11-06 02:38:22 +01:00
* @returns The number of elements in the list.
* @remark If using this coung value to itterate through the list with `list_get`, acquire
* the lists lock before the `list_count/list_get` block and release it afterwards.
*/
DWORD list_count(PLIST pList)
{
DWORD count = 0;
if (pList != NULL)
{
lock_acquire(pList->lock);
count = pList->count;
lock_release(pList->lock);
}
return count;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Get the data value held in the list and a specified index.
* @param pList Pointer to the \c LIST to get the element from.
2013-11-06 02:38:22 +01:00
* @param index Index of the element to get;
* @returns Pointer to the item in the list.
* @retval NULL Indicates the element doesn't exist in the list.
* @remark This will perform a linear search from the beginning of the list.
*/
LPVOID list_get(PLIST pList, DWORD index)
{
2013-11-06 02:38:22 +01:00
LPVOID data = NULL;
PNODE current_node = NULL;
if (pList == NULL)
return NULL;
lock_acquire(pList->lock);
if (pList->count <= index)
{
lock_release(pList->lock);
return NULL;
}
current_node = pList->start;
2013-11-06 02:38:22 +01:00
while (current_node != NULL)
{
2013-11-06 02:38:22 +01:00
if (index == 0)
{
break;
2013-11-06 02:38:22 +01:00
}
current_node = current_node->next;
index--;
}
2013-11-06 02:38:22 +01:00
if (current_node != NULL)
{
data = current_node->data;
2013-11-06 02:38:22 +01:00
}
lock_release(pList->lock);
return data;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Add a data item onto the end of the list.
* @param pList Pointer to the \c LIST to add the item to.
2013-11-06 02:38:22 +01:00
* @param data The data that is to be added to the list.
* @returns Indication of success or failure.
* @sa list_push
*/
BOOL list_add(PLIST pList, LPVOID data)
{
return list_push(pList, data);
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Internal function to remove a node from a list.
* @param pList Pointer to the \c LIST containing \c node.
* @param pNode Pointer to the \c NOTE to remove.
2013-11-06 02:38:22 +01:00
* @returns Indication of success or failure.
* @remark Assumes caller has aquired the appropriate lock first.
*/
BOOL list_remove_node(PLIST pList, PNODE pNode)
{
if (pList == NULL || pNode == NULL)
2013-11-06 02:38:22 +01:00
{
return FALSE;
2013-11-06 02:38:22 +01:00
}
if (pList->count - 1 == 0)
{
pList->start = NULL;
pList->end = NULL;
}
else
{
if (pList->start == pNode)
{
pList->start = pList->start->next;
pList->start->prev = NULL;
}
else if (pList->end == pNode)
{
pList->end = pList->end->prev;
pList->end->next = NULL;
}
2013-11-06 02:38:22 +01:00
else
{
pNode->next->prev = pNode->prev;
pNode->prev->next = pNode->next;
}
}
pList->count -= 1;
pNode->next = NULL;
2013-11-06 02:38:22 +01:00
pNode->prev = NULL;
free(pNode);
return TRUE;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Remove a given data item from the list.
* @param pList Pointer to the \c LIST to remove the item from.
2013-11-06 02:38:22 +01:00
* @param data The data that is to be removed from the list.
* @remark Assumes data items are unqique as only the first occurrence is removed.
* @returns Indication of success or failure.
* @sa list_remove_node
*/
BOOL list_remove(PLIST pList, LPVOID data)
{
2013-11-06 02:38:22 +01:00
BOOL result = FALSE;
PNODE current_node = NULL;
if (pList == NULL || data == NULL)
2013-11-06 02:38:22 +01:00
{
return FALSE;
2013-11-06 02:38:22 +01:00
}
lock_acquire(pList->lock);
current_node = pList->start;
2013-11-06 02:38:22 +01:00
while (current_node != NULL)
{
2013-11-06 02:38:22 +01:00
if (current_node->data == data)
{
break;
2013-11-06 02:38:22 +01:00
}
current_node = current_node->next;
}
result = list_remove_node(pList, current_node);
2013-11-06 02:38:22 +01:00
lock_release(pList->lock);
return result;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Remove a list item at the specified index.
* @param pList Pointer to the \c LIST to remove the item from.
2013-11-06 02:38:22 +01:00
* @param index Index of the item to remove.
* @returns Indication of success or failure.
*/
BOOL list_delete(PLIST pList, DWORD index)
{
2013-11-06 02:38:22 +01:00
BOOL result = FALSE;
LPVOID data = NULL;
PNODE current_node = NULL;
if (pList == NULL)
2013-11-06 02:38:22 +01:00
{
return FALSE;
2013-11-06 02:38:22 +01:00
}
lock_acquire(pList->lock);
if (pList->count > index)
{
current_node = pList->start;
2013-11-06 02:38:22 +01:00
while (current_node != NULL)
{
2013-11-06 02:38:22 +01:00
if (index == 0)
{
result = list_remove_node(pList, current_node);
break;
}
current_node = current_node->next;
index--;
}
}
lock_release(pList->lock);
2013-11-06 02:38:22 +01:00
return result;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Push a data item onto the end of the list.
* @param pList Pointer to the \c LIST to append the data to.
2013-11-06 02:38:22 +01:00
* @param data Pointer to the data to append.
* @returns Indication of success or failure.
*/
BOOL list_push(PLIST pList, LPVOID data)
{
PNODE pNode = NULL;
2013-11-06 02:38:22 +01:00
if (pList == NULL)
return FALSE;
pNode = (PNODE)malloc(sizeof(NODE));
if (pNode == NULL)
2013-11-06 02:38:22 +01:00
{
return FALSE;
2013-11-06 02:38:22 +01:00
}
pNode->data = data;
pNode->next = NULL;
pNode->prev = NULL;
lock_acquire(pList->lock);
if (pList->end != NULL)
2013-11-06 02:38:22 +01:00
{
pList->end->next = pNode;
pNode->prev = pList->end;
pList->end = pNode;
}
else
{
pList->start = pNode;
pList->end = pNode;
}
pList->count += 1;
lock_release(pList->lock);
return TRUE;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Pop a data value off the end of the list.
* @param pList Pointer to the \c LIST to pop the value from.
2013-11-06 02:38:22 +01:00
* @returns The popped value.
* @retval NULL Indicates no data in the list.
*/
LPVOID list_pop(PLIST pList)
{
LPVOID data = NULL;
if (pList == NULL)
2013-11-06 02:38:22 +01:00
{
return NULL;
2013-11-06 02:38:22 +01:00
}
lock_acquire(pList->lock);
if (pList->end != NULL)
{
data = pList->end->data;
list_remove_node(pList, pList->end);
}
lock_release(pList->lock);
return data;
}
2013-11-06 02:38:22 +01:00
/*!
* @brief Pop a data value off the start of the list.
* @param pList Pointer to the \c LIST to shift the value from.
2013-11-06 02:38:22 +01:00
* @returns The shifted value.
* @retval NULL Indicates no data in the list.
*/
LPVOID list_shift(PLIST pList)
{
LPVOID data = NULL;
if (pList == NULL)
2013-11-06 02:38:22 +01:00
{
return NULL;
2013-11-06 02:38:22 +01:00
}
lock_acquire(pList->lock);
if (pList->start != NULL)
{
data = pList->start->data;
list_remove_node(pList, pList->start);
}
lock_release(pList->lock);
return data;
2013-11-06 02:38:22 +01:00
}
/*!
* @brief Iterate over the list and call a function callback on each element.
* @param pList Pointer to the \c LIST to enumerate.
* @param pCallback Callback function to invoke for each element in the list.
* @param pState Pointer to the state to pass with each function call.
*/
BOOL list_enumerate(PLIST pList, PLISTENUMCALLBACK pCallback, LPVOID pState)
{
if (pList == NULL || pCallback == NULL)
{
return FALSE;
}
lock_acquire(pList->lock);
PNODE pCurrent = pList->start;
BOOL bResult = FALSE;
while (pCurrent != NULL)
{
bResult = pCallback(pState, pCurrent->data) || bResult;
pCurrent = pCurrent->next;
}
lock_release(pList->lock);
return bResult;
}