aboutsummaryrefslogtreecommitdiff
path: root/dep/CascLib/src/common/DynamicArray.cpp
blob: e744fbe39122e746e40c85e52cd4752a5c6ba011 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
/*****************************************************************************/
/* DynamicArray.cpp                       Copyright (c) Ladislav Zezula 2015 */
/*---------------------------------------------------------------------------*/
/* Description:                                                              */
/*---------------------------------------------------------------------------*/
/*   Date    Ver   Who  Comment                                              */
/* --------  ----  ---  -------                                              */
/* 30.10.15  1.00  Lad  The first version of DynamicArray.cpp                */
/*****************************************************************************/

#define __CASCLIB_SELF__
#include "../CascLib.h"
#include "../CascCommon.h"

//-----------------------------------------------------------------------------
// Local functions

static bool EnlargeArray(PDYNAMIC_ARRAY pArray, size_t NewItemCount)
{
    char * NewItemArray;
    size_t ItemCountMax;

    // We expect the array to be already allocated
    assert(pArray->ItemArray != NULL);
    assert(pArray->ItemCountMax != 0);
    
    // Shall we enlarge the table?
    if(NewItemCount > pArray->ItemCountMax)
    {
        // Calculate new table size
        ItemCountMax = pArray->ItemCountMax;
        while(ItemCountMax < NewItemCount)
            ItemCountMax = ItemCountMax << 1;

        // Allocate new table
        NewItemArray = CASC_REALLOC(char, pArray->ItemArray, pArray->ItemSize * ItemCountMax);
        if(NewItemArray == NULL)
            return false;

        // Set the new table size
        pArray->ItemCountMax = ItemCountMax;
        pArray->ItemArray = NewItemArray;
    }

    return true;
}

//-----------------------------------------------------------------------------
// Public functions

int Array_Create_(PDYNAMIC_ARRAY pArray, size_t ItemSize, size_t ItemCountMax)
{
    pArray->ItemArray = CASC_ALLOC(char, (ItemSize * ItemCountMax));
    if(pArray->ItemArray == NULL)
        return ERROR_NOT_ENOUGH_MEMORY;

    pArray->ItemCountMax = ItemCountMax;
    pArray->ItemCount = 0;
    pArray->ItemSize = ItemSize;
    return ERROR_SUCCESS;
}

void * Array_Insert(PDYNAMIC_ARRAY pArray, const void * NewItems, size_t NewItemCount)
{
    char * NewItemPtr;

    // Try to enlarge the buffer, if needed
    if(!EnlargeArray(pArray, pArray->ItemCount + NewItemCount))
        return NULL;
    NewItemPtr = pArray->ItemArray + (pArray->ItemCount * pArray->ItemSize);

    // Copy the old item(s), if any
    if(NewItems != NULL)
        memcpy(NewItemPtr, NewItems, (NewItemCount * pArray->ItemSize));

    // Increment the size of the array
    pArray->ItemCount += NewItemCount;
    return NewItemPtr;
}

void * Array_ItemAt(PDYNAMIC_ARRAY pArray, size_t ItemIndex)
{
    assert(ItemIndex < pArray->ItemCount);
    return pArray->ItemArray + (ItemIndex * pArray->ItemSize);
}

size_t Array_IndexOf(PDYNAMIC_ARRAY pArray, const void * ArrayPtr)
{
    char * ArrayItem = (char *)ArrayPtr;

    assert(pArray->ItemArray <= ArrayItem && ArrayItem <= pArray->ItemArray + (pArray->ItemCount * pArray->ItemSize));
    return ((ArrayItem - pArray->ItemArray) / pArray->ItemSize);
}

void Array_Free(PDYNAMIC_ARRAY pArray)
{
    if(pArray != NULL && pArray->ItemArray != NULL)
    {
        CASC_FREE(pArray->ItemArray);
    }
}