aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2022-08-10 21:24:37 -0400
committerDavid Robillard <d@drobilla.net>2022-08-17 13:51:23 -0400
commit7dfdb1744c99d874f7672250cd90b1b9d8d904f2 (patch)
tree5c4f93b77b99e674cdb528d2bac7f28798f37666
parent565366ae88efcb0e384039f17c0253917ddb5d63 (diff)
downloadjalv-7dfdb1744c99d874f7672250cd90b1b9d8d904f2.tar.gz
jalv-7dfdb1744c99d874f7672250cd90b1b9d8d904f2.tar.bz2
jalv-7dfdb1744c99d874f7672250cd90b1b9d8d904f2.zip
Clean up symap implementation and run unit test
-rw-r--r--src/symap.c119
-rw-r--r--src/symap.h23
-rw-r--r--test/meson.build13
3 files changed, 93 insertions, 62 deletions
diff --git a/src/symap.c b/src/symap.c
index 0dd4688..b034bd0 100644
--- a/src/symap.c
+++ b/src/symap.c
@@ -25,47 +25,45 @@
*/
struct SymapImpl {
- /// Unsorted array of strings, so symbols[i] is the symbol for ID i
- char** symbols;
-
- /// Array of IDs, sorted by corresponding string in `symbols`
- uint32_t* index;
-
- /// Number of symbols (number of items in `symbols` and `index`)
- uint32_t size;
+ char** symbols; ///< String array where symbols[i] is the symbol for ID i
+ uint32_t* index; ///< ID array sorted by corresponding string in `symbols`
+ uint32_t size; ///< Number of symbols (length of both symbols and index)
+ uint32_t pad; ///< Unused padding
};
Symap*
symap_new(void)
{
- Symap* map = (Symap*)malloc(sizeof(Symap));
- map->symbols = NULL;
- map->index = NULL;
- map->size = 0;
+ Symap* const map = (Symap*)calloc(1, sizeof(Symap));
+
+ if (map) {
+ map->symbols = NULL;
+ map->index = NULL;
+ map->size = 0U;
+ }
+
return map;
}
void
-symap_free(Symap* map)
+symap_free(Symap* const map)
{
- if (!map) {
- return;
- }
+ if (map) {
+ for (uint32_t i = 0U; i < map->size; ++i) {
+ free(map->symbols[i]);
+ }
- for (uint32_t i = 0; i < map->size; ++i) {
- free(map->symbols[i]);
+ free(map->symbols);
+ free(map->index);
+ free(map);
}
-
- free(map->symbols);
- free(map->index);
- free(map);
}
static char*
-symap_strdup(const char* str)
+symap_strdup(const char* const str)
{
const size_t len = strlen(str);
- char* copy = (char*)malloc(len + 1);
+ char* const copy = (char*)malloc(len + 1);
memcpy(copy, str, len + 1);
return copy;
}
@@ -75,9 +73,10 @@ symap_strdup(const char* str)
or the index where a new entry for `sym` should be inserted.
*/
static uint32_t
-symap_search(const Symap* map, const char* sym, bool* exact)
+symap_search(const Symap* const map, const char* const sym, bool* const exact)
{
*exact = false;
+
if (map->size == 0) {
return 0; // Empty map, insert at 0
}
@@ -115,12 +114,11 @@ symap_search(const Symap* map, const char* sym, bool* exact)
}
uint32_t
-symap_try_map(Symap* map, const char* sym)
+symap_try_map(const Symap* const map, const char* const sym)
{
bool exact = false;
const uint32_t index = symap_search(map, sym, &exact);
if (exact) {
- assert(!strcmp(map->symbols[map->index[index]], sym));
return map->index[index];
}
@@ -128,8 +126,9 @@ symap_try_map(Symap* map, const char* sym)
}
uint32_t
-symap_map(Symap* map, const char* sym)
+symap_map(Symap* const map, const char* sym)
{
+ // Search for existing symbol
bool exact = false;
const uint32_t index = symap_search(map, sym, &exact);
if (exact) {
@@ -137,15 +136,28 @@ symap_map(Symap* map, const char* sym)
return map->index[index];
}
- const uint32_t id = ++map->size;
- char* const str = symap_strdup(sym);
+ // Claim a new highest ID
+ const uint32_t id = map->size + 1;
+
+ // Grow symbol array
+ char** new_symbols = (char**)realloc(map->symbols, id * sizeof(sym));
+ if (!new_symbols) {
+ return 0;
+ }
+
+ // Grow index array
+ uint32_t* new_index = (uint32_t*)realloc(map->index, id * sizeof(index));
+ if (!new_index) {
+ return 0;
+ }
// Append new symbol to symbols array
- map->symbols = (char**)realloc(map->symbols, map->size * sizeof(str));
- map->symbols[id - 1] = str;
+ map->size = id;
+ map->symbols = new_symbols;
+ map->symbols[id - 1] = symap_strdup(sym);
// Insert new index element into sorted index
- map->index = (uint32_t*)realloc(map->index, map->size * sizeof(uint32_t));
+ map->index = new_index;
if (index < map->size - 1) {
memmove(map->index + index + 1,
map->index + index,
@@ -158,7 +170,7 @@ symap_map(Symap* map, const char* sym)
}
const char*
-symap_unmap(Symap* map, uint32_t id)
+symap_unmap(const Symap* const map, const uint32_t id)
{
if (id == 0) {
return NULL;
@@ -171,12 +183,12 @@ symap_unmap(Symap* map, uint32_t id)
return NULL;
}
-#ifdef STANDALONE
+#ifdef SYMAP_STANDALONE
# include <stdio.h>
static void
-symap_dump(Symap* map)
+symap_dump(Symap* const map)
{
fprintf(stderr, "{\n");
for (uint32_t i = 0; i < map->size; ++i) {
@@ -186,35 +198,44 @@ symap_dump(Symap* map)
fprintf(stderr, "}\n");
}
-int
-main()
+static int
+symap_test(Symap* const map)
{
# define N_SYMS 5
- char* syms[N_SYMS] = {"hello", "bonjour", "goodbye", "aloha", "salut"};
- Symap* map = symap_new();
+ static const char* const syms[N_SYMS] = {
+ "hello", "bonjour", "goodbye", "aloha", "salut"};
+
for (int i = 0; i < N_SYMS; ++i) {
if (symap_try_map(map, syms[i])) {
- fprintf(stderr, "error: Symbol already mapped\n");
- return 1;
+ return fprintf(stderr, "error: Symbol already mapped\n");
}
const uint32_t id = symap_map(map, syms[i]);
- if (strcmp(map->symbols[id - 1], syms[i])) {
- fprintf(stderr, "error: Corrupt symbol table\n");
- return 1;
+ if (!!strcmp(map->symbols[id - 1], syms[i])) {
+ return fprintf(stderr, "error: Corrupt symbol table\n");
}
if (symap_map(map, syms[i]) != id) {
- fprintf(stderr, "error: Remapped symbol to a different ID\n");
- return 1;
+ return fprintf(stderr, "error: Remapped symbol to a different ID\n");
}
symap_dump(map);
}
- symap_free(map);
return 0;
+
+# undef N_SYMS
+}
+
+int
+main(void)
+{
+ Symap* const map = symap_new();
+ const int st = symap_test(map);
+
+ symap_free(map);
+ return st;
}
-#endif // STANDALONE
+#endif // SYMAP_STANDALONE
diff --git a/src/symap.h b/src/symap.h
index ed74157..74dc59a 100644
--- a/src/symap.h
+++ b/src/symap.h
@@ -12,11 +12,14 @@
#ifndef SYMAP_H
#define SYMAP_H
+#include "zix/attributes.h"
+
#include <stdint.h>
typedef struct SymapImpl Symap;
/// Create a new symbol map
+ZIX_MALLOC_FUNC
Symap*
symap_new(void);
@@ -24,24 +27,18 @@ symap_new(void);
void
symap_free(Symap* map);
-/// Map a string to a symbol ID if it is already mapped, otherwise return 0
+/// Map a string to a symbol if it is already mapped, otherwise return 0
+ZIX_PURE_FUNC
uint32_t
-symap_try_map(Symap* map, const char* sym);
-
-/**
- Map a string to a symbol ID.
+symap_try_map(const Symap* map, const char* sym);
- Note that 0 is never a valid symbol ID.
-*/
+/// Map a string to a symbol
uint32_t
symap_map(Symap* map, const char* sym);
-/**
- Unmap a symbol ID back to a symbol, or NULL if no such ID exists.
-
- Note that 0 is never a valid symbol ID.
-*/
+/// Unmap a symbol back to a string if possible, otherwise return NULL
+ZIX_PURE_FUNC
const char*
-symap_unmap(Symap* map, uint32_t id);
+symap_unmap(const Symap* map, uint32_t id);
#endif // SYMAP_H
diff --git a/test/meson.build b/test/meson.build
index d789d01..f696dba 100644
--- a/test/meson.build
+++ b/test/meson.build
@@ -22,3 +22,16 @@ if not meson.is_subproject()
)
endif
endif
+
+##############
+# Unit Tests #
+##############
+
+test(
+ 'test_symap',
+ executable(
+ 'test_symap',
+ files('../src/symap.c'),
+ c_args: ['-DSYMAP_STANDALONE'],
+ ),
+)