summaryrefslogtreecommitdiff
path: root/src/hash_table.c
diff options
context:
space:
mode:
authorPreston Pan <preston@nullring.xyz>2023-01-02 22:31:49 -0800
committerPreston Pan <preston@nullring.xyz>2023-01-02 22:31:49 -0800
commit64feef1b9ea72adf7ba32998e9dca7d507607498 (patch)
treea409e61877bb51aa6fb2477175dabbf3dbccf298 /src/hash_table.c
a lot of stuff.
Diffstat (limited to 'src/hash_table.c')
-rw-r--r--src/hash_table.c121
1 files changed, 121 insertions, 0 deletions
diff --git a/src/hash_table.c b/src/hash_table.c
new file mode 100644
index 0000000..9384945
--- /dev/null
+++ b/src/hash_table.c
@@ -0,0 +1,121 @@
+#include "./include/hash_table.h"
+#include "./include/ast.h"
+#include "./include/macros.h"
+#include <stdbool.h>
+#include <stdlib.h>
+#include <string.h>
+
+pair_t *init_pair(char *key, ast_t *value) {
+ pair_t *p = (pair_t *)malloc(sizeof(pair_t));
+ if (p == NULL)
+ die("malloc on pair");
+ p->key = key;
+ p->value = value;
+ return p;
+}
+
+sl_node_t *init_sl_node(char *key, ast_t *value) {
+ sl_node_t *n = (sl_node_t *)malloc(sizeof(sl_node_t));
+ if (n == NULL)
+ die("malloc on node");
+ n->value = init_pair(key, value);
+ n->next = NULL;
+ return n;
+}
+
+/*** SINGLY LINKED LIST FUNCTIONS ***/
+sl_list_t *init_sl_list() {
+ sl_list_t *l = (sl_list_t *)malloc(sizeof(sl_list_t));
+ if (l == NULL)
+ die("malloc on list");
+ l->size = 0;
+ l->head = NULL;
+ return l;
+}
+
+void sl_list_add(sl_list_t *l, char *key, ast_t *value) {
+ sl_node_t *cur = l->head;
+ bool modified = false;
+ if (l->head == NULL) {
+ l->head = init_sl_node(key, value);
+ l->size++;
+ }
+
+ for (int i = 0; i < l->size - 1; i++) {
+ if (strcmp(cur->value->key, key) == 0) {
+ cur->value->value = value;
+ modified = true;
+ break;
+ }
+ cur = cur->next;
+ }
+ if (strcmp(cur->value->key, key) == 0) {
+ cur->value->value = value;
+ modified = true;
+ }
+
+ if (!modified) {
+ cur->next = init_sl_node(key, value);
+ l->size++;
+ }
+}
+
+ast_t *sl_list_get(sl_list_t *l, char *key) {
+ sl_node_t *cur = l->head;
+ for (int i = 0; i < l->size; i++) {
+ if (strcmp(cur->value->key, key) == 0)
+ return cur->value->value;
+ cur = cur->next;
+ }
+ return NULL;
+}
+
+void sl_list_free(sl_list_t *l) {
+ sl_node_t *cur = l->head;
+ sl_node_t *tmp;
+ for (int i = 0; i < l->size; i++) {
+ tmp = cur;
+ cur = cur->next;
+ free(tmp);
+ }
+ free(l);
+}
+
+/*** HASH TABLE FUNCTIONS ***/
+hash_table_t *init_hash_table(int size) {
+ hash_table_t *h = (hash_table_t *)malloc(sizeof(hash_table_t));
+ if (h == NULL)
+ die("malloc on hash table");
+ h->size = size;
+ h->buckets = malloc(sizeof(sl_list_t *));
+ if (h->buckets == NULL)
+ die("malloc on buckets");
+ for (int i = 0; i < h->size; i++)
+ h->buckets[i] = init_sl_list();
+ return h;
+}
+
+void hash_table_add(hash_table_t *h, char *key, ast_t *value) {
+ sl_list_t *l = h->buckets[hash(key, h->size)];
+ sl_list_add(l, key, value);
+}
+
+ast_t *hash_table_get(hash_table_t *h, char *key) {
+ sl_list_t *l = h->buckets[hash(key, h->size)];
+ return sl_list_get(l, key);
+}
+
+void hash_table_free(hash_table_t *h) {
+ for (int i = 0; i < h->size; i++)
+ sl_list_free(h->buckets[i]);
+ free(h);
+}
+
+/* DJB2 HASH FUNCTION */
+unsigned long hash(char *key, int size) {
+ unsigned long hash = 5381;
+ int c;
+ while ((c = *key++))
+ hash = ((hash << 5) + hash) + c; /* hash * 33 + c */
+ return hash % size;
+}