diff options
Diffstat (limited to 'vcs-svn/string_pool.c')
| -rw-r--r-- | vcs-svn/string_pool.c | 102 | 
1 files changed, 102 insertions, 0 deletions
| diff --git a/vcs-svn/string_pool.c b/vcs-svn/string_pool.c new file mode 100644 index 0000000000..f5b1da836e --- /dev/null +++ b/vcs-svn/string_pool.c @@ -0,0 +1,102 @@ +/* + * 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(); +} | 
