diff options
Diffstat (limited to 'linden/indra/llcommon/llstringtable.h')
-rw-r--r-- | linden/indra/llcommon/llstringtable.h | 227 |
1 files changed, 227 insertions, 0 deletions
diff --git a/linden/indra/llcommon/llstringtable.h b/linden/indra/llcommon/llstringtable.h new file mode 100644 index 0000000..d53c0e1 --- /dev/null +++ b/linden/indra/llcommon/llstringtable.h | |||
@@ -0,0 +1,227 @@ | |||
1 | /** | ||
2 | * @file llstringtable.h | ||
3 | * @brief The LLStringTable class provides a _fast_ method for finding | ||
4 | * unique copies of strings. | ||
5 | * | ||
6 | * Copyright (c) 2001-2007, Linden Research, Inc. | ||
7 | * | ||
8 | * The source code in this file ("Source Code") is provided by Linden Lab | ||
9 | * to you under the terms of the GNU General Public License, version 2.0 | ||
10 | * ("GPL"), unless you have obtained a separate licensing agreement | ||
11 | * ("Other License"), formally executed by you and Linden Lab. Terms of | ||
12 | * the GPL can be found in doc/GPL-license.txt in this distribution, or | ||
13 | * online at http://secondlife.com/developers/opensource/gplv2 | ||
14 | * | ||
15 | * There are special exceptions to the terms and conditions of the GPL as | ||
16 | * it is applied to this Source Code. View the full text of the exception | ||
17 | * in the file doc/FLOSS-exception.txt in this software distribution, or | ||
18 | * online at http://secondlife.com/developers/opensource/flossexception | ||
19 | * | ||
20 | * By copying, modifying or distributing this software, you acknowledge | ||
21 | * that you have read and understood your obligations described above, | ||
22 | * and agree to abide by those obligations. | ||
23 | * | ||
24 | * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO | ||
25 | * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, | ||
26 | * COMPLETENESS OR PERFORMANCE. | ||
27 | */ | ||
28 | |||
29 | #ifndef LL_STRING_TABLE_H | ||
30 | #define LL_STRING_TABLE_H | ||
31 | |||
32 | #include "llstl.h" | ||
33 | #include <list> | ||
34 | #include <set> | ||
35 | |||
36 | #if LL_WINDOWS | ||
37 | # if (_MSC_VER >= 1300) | ||
38 | # define STRING_TABLE_HASH_MAP 1 | ||
39 | # pragma warning(disable : 4996) | ||
40 | # endif | ||
41 | #else | ||
42 | //# define STRING_TABLE_HASH_MAP 1 | ||
43 | #endif | ||
44 | |||
45 | #if STRING_TABLE_HASH_MAP | ||
46 | #include <hash_map> | ||
47 | #endif | ||
48 | |||
49 | // string_table.h | ||
50 | // LLStringTable class header file | ||
51 | // Provides a _fast_ method for finding unique copies of strings | ||
52 | // | ||
53 | // Copyright 2001-2002, Linden Research, Inc. | ||
54 | |||
55 | const U32 MAX_STRINGS_LENGTH = 256; | ||
56 | |||
57 | class LLStringTableEntry | ||
58 | { | ||
59 | public: | ||
60 | LLStringTableEntry(const char *str) | ||
61 | : mString(NULL), mCount(1) | ||
62 | { | ||
63 | // Copy string | ||
64 | U32 length = (U32)strlen(str) + 1; /*Flawfinder: ignore*/ | ||
65 | length = llmin(length, MAX_STRINGS_LENGTH); | ||
66 | mString = new char[length]; | ||
67 | strncpy(mString, str, length); /*Flawfinder: ignore*/ | ||
68 | mString[length - 1] = 0; | ||
69 | } | ||
70 | ~LLStringTableEntry() | ||
71 | { | ||
72 | delete [] mString; | ||
73 | mCount = 0; | ||
74 | } | ||
75 | void incCount() { mCount++; } | ||
76 | BOOL decCount() { return --mCount; } | ||
77 | |||
78 | char *mString; | ||
79 | S32 mCount; | ||
80 | }; | ||
81 | |||
82 | class LLStringTable | ||
83 | { | ||
84 | public: | ||
85 | LLStringTable(int tablesize); | ||
86 | ~LLStringTable(); | ||
87 | |||
88 | char *checkString(const char *str); | ||
89 | char *checkString(const std::string& str); | ||
90 | LLStringTableEntry *checkStringEntry(const char *str); | ||
91 | LLStringTableEntry *checkStringEntry(const std::string& str); | ||
92 | |||
93 | char *addString(const char *str); | ||
94 | char *addString(const std::string& str); | ||
95 | LLStringTableEntry *addStringEntry(const char *str); | ||
96 | LLStringTableEntry *addStringEntry(const std::string& str); | ||
97 | void removeString(const char *str); | ||
98 | |||
99 | S32 mMaxEntries; | ||
100 | S32 mUniqueEntries; | ||
101 | |||
102 | #if STRING_TABLE_HASH_MAP | ||
103 | typedef std::hash_multimap<U32, LLStringTableEntry *> string_hash_t; | ||
104 | string_hash_t mStringHash; | ||
105 | #else | ||
106 | typedef std::list<LLStringTableEntry *> string_list_t; | ||
107 | typedef string_list_t * string_list_ptr_t; | ||
108 | string_list_ptr_t *mStringList; | ||
109 | #endif | ||
110 | }; | ||
111 | |||
112 | extern LLStringTable gStringTable; | ||
113 | |||
114 | //============================================================================ | ||
115 | |||
116 | // This class is designed to be used locally, | ||
117 | // e.g. as a member of an LLXmlTree | ||
118 | // Strings can be inserted only, then quickly looked up | ||
119 | |||
120 | typedef const std::string* LLStdStringHandle; | ||
121 | |||
122 | class LLStdStringTable | ||
123 | { | ||
124 | public: | ||
125 | LLStdStringTable(S32 tablesize = 0) | ||
126 | { | ||
127 | if (tablesize == 0) | ||
128 | { | ||
129 | tablesize = 256; // default | ||
130 | } | ||
131 | // Make sure tablesize is power of 2 | ||
132 | for (S32 i = 31; i>0; i--) | ||
133 | { | ||
134 | if (tablesize & (1<<i)) | ||
135 | { | ||
136 | if (tablesize >= (3<<(i-1))) | ||
137 | tablesize = (1<<(i+1)); | ||
138 | else | ||
139 | tablesize = (1<<i); | ||
140 | break; | ||
141 | } | ||
142 | } | ||
143 | mTableSize = tablesize; | ||
144 | mStringList = new string_set_t[tablesize]; | ||
145 | } | ||
146 | ~LLStdStringTable() | ||
147 | { | ||
148 | cleanup(); | ||
149 | delete[] mStringList; | ||
150 | } | ||
151 | void cleanup() | ||
152 | { | ||
153 | // remove strings | ||
154 | for (S32 i = 0; i<mTableSize; i++) | ||
155 | { | ||
156 | string_set_t& stringset = mStringList[i]; | ||
157 | for (string_set_t::iterator iter = stringset.begin(); iter != stringset.end(); iter++) | ||
158 | { | ||
159 | delete *iter; | ||
160 | } | ||
161 | stringset.clear(); | ||
162 | } | ||
163 | } | ||
164 | |||
165 | LLStdStringHandle lookup(const std::string& s) | ||
166 | { | ||
167 | U32 hashval = makehash(s); | ||
168 | return lookup(hashval, s); | ||
169 | } | ||
170 | |||
171 | LLStdStringHandle checkString(const std::string& s) | ||
172 | { | ||
173 | U32 hashval = makehash(s); | ||
174 | return lookup(hashval, s); | ||
175 | } | ||
176 | |||
177 | LLStdStringHandle insert(const std::string& s) | ||
178 | { | ||
179 | U32 hashval = makehash(s); | ||
180 | LLStdStringHandle result = lookup(hashval, s); | ||
181 | if (result == NULL) | ||
182 | { | ||
183 | result = new std::string(s); | ||
184 | mStringList[hashval].insert(result); | ||
185 | } | ||
186 | return result; | ||
187 | } | ||
188 | LLStdStringHandle addString(const std::string& s) | ||
189 | { | ||
190 | return insert(s); | ||
191 | } | ||
192 | |||
193 | private: | ||
194 | U32 makehash(const std::string& s) | ||
195 | { | ||
196 | S32 len = (S32)s.size(); | ||
197 | const char* c = s.c_str(); | ||
198 | U32 hashval = 0; | ||
199 | for (S32 i=0; i<len; i++) | ||
200 | { | ||
201 | hashval = ((hashval<<5) + hashval) + *c++; | ||
202 | } | ||
203 | return hashval & (mTableSize-1); | ||
204 | } | ||
205 | LLStdStringHandle lookup(U32 hashval, const std::string& s) | ||
206 | { | ||
207 | string_set_t& stringset = mStringList[hashval]; | ||
208 | LLStdStringHandle handle = &s; | ||
209 | string_set_t::iterator iter = stringset.find(handle); // compares actual strings | ||
210 | if (iter != stringset.end()) | ||
211 | { | ||
212 | return *iter; | ||
213 | } | ||
214 | else | ||
215 | { | ||
216 | return NULL; | ||
217 | } | ||
218 | } | ||
219 | |||
220 | private: | ||
221 | S32 mTableSize; | ||
222 | typedef std::set<LLStdStringHandle, compare_pointer_contents<std::string> > string_set_t; | ||
223 | string_set_t* mStringList; // [mTableSize] | ||
224 | }; | ||
225 | |||
226 | |||
227 | #endif | ||