-
Notifications
You must be signed in to change notification settings - Fork 10
/
strings.c
113 lines (91 loc) · 2.13 KB
/
strings.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
/*
Copyright (C) 2008 Arnaldo Carvalho de Melo <[email protected]>
This program is free software; you can redistribute it and/or modify it
under the terms of version 2 of the GNU General Public License as
published by the Free Software Foundation.
*/
#include "strings.h"
#include "gobuffer.h"
#include <search.h>
#include <stdint.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <zlib.h>
#include "dutil.h"
struct strings *strings__new(void)
{
struct strings *self = malloc(sizeof(*self));
if (self != NULL) {
self->tree = NULL;
gobuffer__init(&self->gb);
}
return self;
}
static void do_nothing(void *ptr __unused)
{
}
void strings__delete(struct strings *self)
{
if (self == NULL)
return;
tdestroy(self->tree, do_nothing);
__gobuffer__delete(&self->gb);
free(self);
}
static strings_t strings__insert(struct strings *self, const char *s)
{
return gobuffer__add(&self->gb, s, strlen(s) + 1);
}
struct search_key {
struct strings *self;
const char *str;
};
static int strings__compare(const void *a, const void *b)
{
const struct search_key *key = a;
return strcmp(key->str, key->self->gb.entries + (unsigned long)b);
}
strings_t strings__add(struct strings *self, const char *str)
{
unsigned long *s;
strings_t index;
struct search_key key = {
.self = self,
.str = str,
};
if (str == NULL)
return 0;
s = tsearch(&key, &self->tree, strings__compare);
if (s != NULL) {
if (*(struct search_key **)s == (void *)&key) { /* Not found, replace with the right key */
index = strings__insert(self, str);
if (index != 0)
*s = (unsigned long)index;
else {
tdelete(&key, &self->tree, strings__compare);
return 0;
}
} else /* Found! */
index = *s;
} else
return 0;
return index;
}
strings_t strings__find(struct strings *self, const char *str)
{
strings_t *s;
struct search_key key = {
.self = self,
.str = str,
};
if (str == NULL)
return 0;
s = tfind(&key, &self->tree, strings__compare);
return s ? *s : 0;
}
int strings__cmp(const struct strings *self, strings_t a, strings_t b)
{
return a == b ? 0 : strcmp(strings__ptr(self, a),
strings__ptr(self, b));
}