-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtable.c
240 lines (189 loc) · 5.28 KB
/
table.c
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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
/**
* Melon Software Framework is Copyright (C) 2021 - 2025 Knot126
*
* =============================================================================
*
* Table/Dictionary type
*
* @note This is implemented using an ordered hash table - that is, references
* to the actual values are made in the hash map, which are then stored in an
* array in the order they were inserted.
*/
#include "memory.h"
#include "error.h"
#include "log.h"
#include "value.h"
#include "table.h"
#define NOT_FOUND ((size_t) -1)
DgError DgTableInit(DgTable *this) {
/**
* Initialise a table
*
* @param this Table object
* @return Error code
*/
DgError error;
// Zero it all! (not needed ?)
DgMemoryZero(this, sizeof *this);
// Prepare the main array
if ((error = DgArrayInit(&this->array))) {
return error;
}
return DG_ERROR_SUCCESSFUL;
}
DgError DgTableFree(DgTable *this, bool deep) {
/**
* Free a table
*
* @param this Table object
* @return Error code
*/
DgError error;
if ((error = DgArrayFree(&this->array, deep))) {
return error;
}
return error;
}
static size_t DgTableIndexForKey(DgTable * restrict this, const DgValue * restrict key) {
/**
* TODO: optimising this using some kind of hash table, preferably better
* than my first attempt at doing so...
*/
size_t len = DgTableLength(this);
for (size_t i = 0; i < len; i++) {
if (DgValueEqual(key, DgArrayAt(&this->array, 2 * i))) {
return i;
}
}
return NOT_FOUND;
}
bool DgTableHas(DgTable * restrict this, DgValue * restrict key) {
/**
* Check if the table has an entry with the given key.
*
* @param this Table to check in
* @param key Key to check for
* @return true if the table has an entry with `key`, false if not
*/
return DgTableIndexForKey(this, key) != NOT_FOUND;
}
DgError DgTablePut(DgTable * restrict this, DgValue * restrict key, DgValue * restrict value) {
/**
* Put a key -> value pair into the table
*
* @note This effectively takes ownership of the key and value. If you want
* them to be copied instead use DgTableSet().
*
* @param this Table object
* @param key Key
* @param value Value
*/
DgError error;
size_t index = DgTableIndexForKey(this, key);
// Handle the case where key/value already exists
if (index != NOT_FOUND) {
// Place the value into its slot
error = DgArrayPut(&this->array, 2 * index + 1, value);
// Free the key, since we'll never be using it
DgValueFree(key);
// Return our error from earlier, if any
if (error) {
return error;
}
}
// Handle the case where the key does not exist yet
else {
// Append the new key/value pair
if ((error = DgArrayAppend(&this->array, key))) {
return error;
}
if ((error = DgArrayAppend(&this->array, value))) {
return error;
}
}
return DG_ERROR_SUCCESS;
}
DgValue *DgTableAt(DgTable * restrict this, DgValue * restrict key) {
/**
* Get a direct pointer to the value assocaited with a key
*
* @note Automatically frees the key (regardless if success or failure)
*
* @param this Table object
* @param key Key (in)
* @param value Value (out)
*/
DgValue *value = NULL;
size_t index = DgTableIndexForKey(this, key);
if (index != NOT_FOUND) {
value = DgArrayAt(&this->array, 2 * index + 1);
}
DgValueFree(key);
return value;
}
DgError DgTableRemove(DgTable * restrict this, DgValue * restrict key) {
/**
* Remove an element from the table
*
* @note This will return DG_ERROR_NOT_FOUND if the entry does not exist.
* Make sure to handle this correctly!
*
* @param this Table object
* @param key The key assocaited with the entry to remove
* @return Error status
*/
size_t index = DgTableIndexForKey(this, key);
DgValueFree(key);
if (index == NOT_FOUND) {
return DG_ERROR_NOT_FOUND;
}
return DgArrayRemoveND(&this->array, 2 * index, 2, true);
}
DgError DgTablePairAt(DgTable * restrict this, size_t index, DgValue ** const restrict key, DgValue ** const restrict value) {
/**
* Get direct pointers to the key and value entries at index. This is good
* for iteration.
*
* @note This function this most useful for serialisation or iteration,
* since it will show you the preserved order of pairs.
*
* @param this Table object
* @param index The index to get
* @param key The key for the index (or NULL to ignore)
* @param value The value for the index (or NULL to ignore)
*/
if (index >= DgTableLength(this)) {
return DG_ERROR_OUT_OF_RANGE;
}
if (key) {
key[0] = DgArrayAt(&this->array, 2 * index);
}
if (value) {
value[0] = DgArrayAt(&this->array, 2 * index + 1);
}
return DG_ERROR_SUCCESSFUL;
}
size_t DgTableLength(DgTable * restrict this) {
/**
* Get the length of the table
*
* @param this Table to find the length of
* @return Length of the table
*/
return DgArrayLength(&this->array) / 2;
}
DgError DgTableSetPointer(DgTable * restrict this, const char *key, void *value) {
DgValue k = DgMakeString(key);
DgValue v = DgMakePointer(value);
return DgTablePut(this, &k, &v);
}
void *DgTableGetPointer(DgTable * restrict this, const char *key) {
DgValue k = DgMakeStaticString(key);
DgValue *v = DgTableAt(this, &k);
if (!v) { return NULL; }
DgValueType t = DgValueGetType(v);
if (t != DG_TYPE_POINTER) {
return NULL;
}
return v->data.asPointer;
}