2018-03-29 13:52:49 +00:00
|
|
|
#include <stdlib.h>
|
2018-03-28 19:22:07 +00:00
|
|
|
#include <string.h>
|
2018-03-29 13:52:49 +00:00
|
|
|
|
|
|
|
#include "mylist.h"
|
2018-03-28 19:22:07 +00:00
|
|
|
#include "mem_util.h"
|
|
|
|
|
|
|
|
void mylist_resize(MyList *list, int newSize, bool runConstructor)
|
|
|
|
{
|
|
|
|
int newCapacity;
|
|
|
|
int oldSize;
|
|
|
|
int i;
|
2018-03-29 13:13:33 +00:00
|
|
|
void *element = NULL;
|
|
|
|
if (newSize < 0)
|
|
|
|
newSize = 0;
|
|
|
|
if (!list)
|
|
|
|
return;
|
2018-03-28 19:22:07 +00:00
|
|
|
newCapacity = newSize;
|
2018-03-29 13:13:33 +00:00
|
|
|
oldSize = list->size;
|
|
|
|
|
|
|
|
if (newSize == oldSize)
|
|
|
|
return;
|
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
if (newSize > list->capacity)
|
|
|
|
{
|
|
|
|
if (newCapacity < list->capacity * 2)
|
|
|
|
newCapacity = list->capacity * 2;
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
/* try to realloc */
|
|
|
|
list->data = (void**)realloc((void*)list->data, newCapacity * sizeof(void*));
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
for (i = list->capacity; i < newCapacity; i++)
|
|
|
|
list->data[i] = NULL;
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
list->capacity = newCapacity;
|
|
|
|
}
|
|
|
|
if (newSize <= list->size)
|
|
|
|
{
|
|
|
|
for (i = newSize; i < list->size; i++)
|
|
|
|
{
|
|
|
|
element = list->data[i];
|
2018-03-29 13:13:33 +00:00
|
|
|
|
|
|
|
if (element)
|
2018-03-28 19:22:07 +00:00
|
|
|
{
|
|
|
|
list->Destructor(element);
|
|
|
|
list->data[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (i = list->size; i < newSize; i++)
|
|
|
|
{
|
|
|
|
if (runConstructor)
|
|
|
|
list->data[i] = list->Constructor();
|
|
|
|
else
|
|
|
|
list->data[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
list->size = newSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *mylist_add_element(MyList *list)
|
|
|
|
{
|
|
|
|
int oldSize;
|
2018-03-29 13:13:33 +00:00
|
|
|
|
|
|
|
if (!list)
|
|
|
|
return NULL;
|
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
oldSize = list->size;
|
|
|
|
mylist_resize(list, oldSize + 1, true);
|
|
|
|
return list->data[oldSize];
|
|
|
|
}
|
|
|
|
|
2018-03-29 13:13:33 +00:00
|
|
|
void mylist_create(MyList **list_p, int initialCapacity,
|
|
|
|
constructor_t constructor, destructor_t destructor)
|
2018-03-28 19:22:07 +00:00
|
|
|
{
|
2018-03-29 13:13:33 +00:00
|
|
|
MyList *list = NULL;
|
|
|
|
if (!list_p)
|
|
|
|
return;
|
|
|
|
if (initialCapacity < 0)
|
|
|
|
initialCapacity = 0;
|
|
|
|
list = *list_p;
|
|
|
|
if (list)
|
2018-03-28 19:22:07 +00:00
|
|
|
mylist_destroy(list_p);
|
2018-03-29 13:13:33 +00:00
|
|
|
|
|
|
|
list = (MyList*)malloc(sizeof(MyList));
|
|
|
|
*list_p = list;
|
|
|
|
list->size = 0;
|
|
|
|
list->Constructor = constructor;
|
|
|
|
list->Destructor = destructor;
|
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
if (initialCapacity > 0)
|
|
|
|
{
|
2018-03-29 18:22:20 +00:00
|
|
|
list->data = (void**)calloc(initialCapacity, sizeof(void*));
|
2018-03-29 13:13:33 +00:00
|
|
|
list->capacity = initialCapacity;
|
2018-03-28 19:22:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-03-29 13:13:33 +00:00
|
|
|
list->data = NULL;
|
|
|
|
list->capacity = 0;
|
2018-03-28 19:22:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void mylist_destroy(MyList **list_p)
|
|
|
|
{
|
2018-03-29 13:13:33 +00:00
|
|
|
MyList *list = NULL;
|
|
|
|
if (!list_p)
|
|
|
|
return;
|
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
list = *list_p;
|
2018-03-29 13:13:33 +00:00
|
|
|
|
|
|
|
if (list)
|
2018-03-28 19:22:07 +00:00
|
|
|
{
|
|
|
|
mylist_resize(list, 0, false);
|
|
|
|
free(list->data);
|
|
|
|
free(list);
|
|
|
|
*list_p = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void mylist_assign(MyList *list, int index, void *value)
|
|
|
|
{
|
2018-03-29 13:13:33 +00:00
|
|
|
void *oldElement = NULL;
|
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
if (index < 0 || index >= list->size)
|
|
|
|
return;
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
oldElement = list->data[index];
|
|
|
|
list->Destructor(oldElement);
|
|
|
|
list->data[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mylist_remove_at(MyList *list, int index)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (index < 0 || index >= list->size)
|
|
|
|
return;
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
mylist_assign(list, index, NULL);
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
for (i = index + 1; i < list->size; i++)
|
|
|
|
list->data[i - 1] = list->data[i];
|
2018-03-29 13:13:33 +00:00
|
|
|
|
2018-03-28 19:22:07 +00:00
|
|
|
list->size--;
|
|
|
|
list->data[list->size] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mylist_pop_front(MyList *list)
|
|
|
|
{
|
|
|
|
mylist_remove_at(list, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mylist_push_back(MyList *list, void *value)
|
|
|
|
{
|
|
|
|
int oldSize;
|
2018-03-29 13:13:33 +00:00
|
|
|
if (!list)
|
|
|
|
return;
|
2018-03-28 19:22:07 +00:00
|
|
|
oldSize = list->size;
|
|
|
|
mylist_resize(list, oldSize + 1, false);
|
|
|
|
list->data[oldSize] = value;
|
|
|
|
}
|