/* * Licensed under a two-clause BSD-style license. * See LICENSE for details. */ #include "git-compat-util.h" #include "trp.h" #include "obj_pool.h" #include "string_pool.h" static struct trp_root tree = { ~0 }; struct node { uint32_t offset; struct trp_node children; }; /* Two memory pools: one for struct node, and another for strings */ obj_pool_gen(node, struct node, 4096) obj_pool_gen(string, char, 4096) static char *node_value(struct node *node) { return node ? string_pointer(node->offset) : NULL; } static int node_cmp(struct node *a, struct node *b) { return strcmp(node_value(a), node_value(b)); } /* Build a Treap from the node structure (a trp_node w/ offset) */ trp_gen(static, tree_, struct node, children, node, node_cmp) const char *pool_fetch(uint32_t entry) { return node_value(node_pointer(entry)); } uint32_t pool_intern(const char *key) { /* Canonicalize key */ struct node *match = NULL, *node; uint32_t key_len; if (key == NULL) return ~0; key_len = strlen(key) + 1; node = node_pointer(node_alloc(1)); node->offset = string_alloc(key_len); strcpy(node_value(node), key); match = tree_search(&tree, node); if (!match) { tree_insert(&tree, node); } else { node_free(1); string_free(key_len); node = match; } return node_offset(node); } uint32_t pool_tok_r(char *str, const char *delim, char **saveptr) { char *token = strtok_r(str, delim, saveptr); return token ? pool_intern(token) : ~0; } void pool_print_seq(uint32_t len, uint32_t *seq, char delim, FILE *stream) { uint32_t i; for (i = 0; i < len && ~seq[i]; i++) { fputs(pool_fetch(seq[i]), stream); if (i < len - 1 && ~seq[i + 1]) fputc(delim, stream); } } uint32_t pool_tok_seq(uint32_t sz, uint32_t *seq, const char *delim, char *str) { char *context = NULL; uint32_t token = ~0; uint32_t length; if (sz == 0) return ~0; if (str) token = pool_tok_r(str, delim, &context); for (length = 0; length < sz; length++) { seq[length] = token; if (token == ~0) return length; token = pool_tok_r(NULL, delim, &context); } seq[sz - 1] = ~0; return sz; } void pool_reset(void) { node_reset(); string_reset(); }