1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* ***** BEGIN LICENSE BLOCK *****
3 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
5 * The contents of this file are subject to the Mozilla Public License Version
6 * 1.1 (the "License"); you may not use this file except in compliance with
7 * the License. You may obtain a copy of the License at
8 * http://www.mozilla.org/MPL/
10 * Software distributed under the License is distributed on an "AS IS" basis,
11 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12 * for the specific language governing rights and limitations under the
15 * The Original Code is C++ hashtable templates.
17 * The Initial Developer of the Original Code is
19 * Portions created by the Initial Developer are Copyright (C) 2002
20 * the Initial Developer. All Rights Reserved.
24 * Alternatively, the contents of this file may be used under the terms of
25 * either the GNU General Public License Version 2 or later (the "GPL"), or
26 * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
27 * in which case the provisions of the GPL or the LGPL are applicable instead
28 * of those above. If you wish to allow use of your version of this file only
29 * under the terms of either the GPL or the LGPL, and not to allow others to
30 * use your version of this file under the terms of the MPL, indicate your
31 * decision by deleting the provisions above and replace them with the notice
32 * and other provisions required by the GPL or the LGPL. If you do not delete
33 * the provisions above, a recipient may use your version of this file under
34 * the terms of any one of the MPL, the GPL or the LGPL.
36 * ***** END LICENSE BLOCK ***** */
38 #ifndef nsTHashKeys_h__
39 #define nsTHashKeys_h__
42 #include "nsISupports.h"
43 #include "nsIHashable.h"
48 #include "nsStringGlue.h"
49 #include "nsCRTGlue.h"
54 /** @file nsHashKeys.h
55 * standard HashKey classes for nsBaseHashtable and relatives. Each of these
56 * classes follows the nsTHashtable::EntryType specification
58 * Lightweight keytypes provided here:
63 * nsClearingVoidPtrHashKey
72 NS_COM_GLUE PRUint32
HashString(const nsAString
& aStr
);
73 NS_COM_GLUE PRUint32
HashString(const nsACString
& aStr
);
74 NS_COM_GLUE PRUint32
HashString(const char* aKey
);
75 NS_COM_GLUE PRUint32
HashString(const PRUnichar
* aKey
);
78 * hashkey wrapper using nsAString KeyType
80 * @see nsTHashtable::EntryType for specification
82 class nsStringHashKey
: public PLDHashEntryHdr
85 typedef const nsAString
& KeyType
;
86 typedef const nsAString
* KeyTypePointer
;
88 nsStringHashKey(KeyTypePointer aStr
) : mStr(*aStr
) { }
89 nsStringHashKey(const nsStringHashKey
& toCopy
) : mStr(toCopy
.mStr
) { }
90 ~nsStringHashKey() { }
92 KeyType
GetKey() const { return mStr
; }
93 PRBool
KeyEquals(const KeyTypePointer aKey
) const
95 return mStr
.Equals(*aKey
);
98 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return &aKey
; }
99 static PLDHashNumber
HashKey(const KeyTypePointer aKey
)
101 return HashString(*aKey
);
103 enum { ALLOW_MEMMOVE
= PR_TRUE
};
110 * hashkey wrapper using nsACString KeyType
112 * @see nsTHashtable::EntryType for specification
114 class nsCStringHashKey
: public PLDHashEntryHdr
117 typedef const nsACString
& KeyType
;
118 typedef const nsACString
* KeyTypePointer
;
120 nsCStringHashKey(const nsACString
* aStr
) : mStr(*aStr
) { }
121 nsCStringHashKey(const nsCStringHashKey
& toCopy
) : mStr(toCopy
.mStr
) { }
122 ~nsCStringHashKey() { }
124 KeyType
GetKey() const { return mStr
; }
126 PRBool
KeyEquals(KeyTypePointer aKey
) const { return mStr
.Equals(*aKey
); }
128 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return &aKey
; }
129 static PLDHashNumber
HashKey(KeyTypePointer aKey
)
131 return HashString(*aKey
);
133 enum { ALLOW_MEMMOVE
= PR_TRUE
};
136 const nsCString mStr
;
140 * hashkey wrapper using PRUint32 KeyType
142 * @see nsTHashtable::EntryType for specification
144 class nsUint32HashKey
: public PLDHashEntryHdr
147 typedef const PRUint32
& KeyType
;
148 typedef const PRUint32
* KeyTypePointer
;
150 nsUint32HashKey(KeyTypePointer aKey
) : mValue(*aKey
) { }
151 nsUint32HashKey(const nsUint32HashKey
& toCopy
) : mValue(toCopy
.mValue
) { }
152 ~nsUint32HashKey() { }
154 KeyType
GetKey() const { return mValue
; }
155 PRBool
KeyEquals(KeyTypePointer aKey
) const { return *aKey
== mValue
; }
157 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return &aKey
; }
158 static PLDHashNumber
HashKey(KeyTypePointer aKey
) { return *aKey
; }
159 enum { ALLOW_MEMMOVE
= PR_TRUE
};
162 const PRUint32 mValue
;
166 * hashkey wrapper using nsISupports* KeyType
168 * @see nsTHashtable::EntryType for specification
170 class nsISupportsHashKey
: public PLDHashEntryHdr
173 typedef nsISupports
* KeyType
;
174 typedef const nsISupports
* KeyTypePointer
;
176 nsISupportsHashKey(const nsISupports
* key
) :
177 mSupports(const_cast<nsISupports
*>(key
)) { }
178 nsISupportsHashKey(const nsISupportsHashKey
& toCopy
) :
179 mSupports(toCopy
.mSupports
) { }
180 ~nsISupportsHashKey() { }
182 KeyType
GetKey() const { return mSupports
; }
184 PRBool
KeyEquals(KeyTypePointer aKey
) const { return aKey
== mSupports
; }
186 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return aKey
; }
187 static PLDHashNumber
HashKey(KeyTypePointer aKey
)
189 return NS_PTR_TO_INT32(aKey
) >>2;
191 enum { ALLOW_MEMMOVE
= PR_TRUE
};
194 nsCOMPtr
<nsISupports
> mSupports
;
198 * hashkey wrapper using void* KeyType
200 * @see nsTHashtable::EntryType for specification
202 class nsVoidPtrHashKey
: public PLDHashEntryHdr
205 typedef const void* KeyType
;
206 typedef const void* KeyTypePointer
;
208 nsVoidPtrHashKey(const void* key
) :
210 nsVoidPtrHashKey(const nsVoidPtrHashKey
& toCopy
) :
211 mKey(toCopy
.mKey
) { }
212 ~nsVoidPtrHashKey() { }
214 KeyType
GetKey() const { return mKey
; }
216 PRBool
KeyEquals(KeyTypePointer aKey
) const { return aKey
== mKey
; }
218 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return aKey
; }
219 static PLDHashNumber
HashKey(KeyTypePointer aKey
)
221 return NS_PTR_TO_INT32(aKey
) >>2;
223 enum { ALLOW_MEMMOVE
= PR_TRUE
};
230 * hashkey wrapper using void* KeyType, that sets key to NULL upon
231 * destruction. Relevant only in cases where a memory pointer-scanner
232 * like valgrind might get confused about stale references.
234 * @see nsTHashtable::EntryType for specification
237 class nsClearingVoidPtrHashKey
: public PLDHashEntryHdr
240 typedef const void* KeyType
;
241 typedef const void* KeyTypePointer
;
243 nsClearingVoidPtrHashKey(const void* key
) :
245 nsClearingVoidPtrHashKey(const nsClearingVoidPtrHashKey
& toCopy
) :
246 mKey(toCopy
.mKey
) { }
247 ~nsClearingVoidPtrHashKey() { mKey
= NULL
; }
249 KeyType
GetKey() const { return mKey
; }
251 PRBool
KeyEquals(KeyTypePointer aKey
) const { return aKey
== mKey
; }
253 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return aKey
; }
254 static PLDHashNumber
HashKey(KeyTypePointer aKey
)
256 return NS_PTR_TO_INT32(aKey
) >>2;
258 enum { ALLOW_MEMMOVE
= PR_TRUE
};
265 * hashkey wrapper using nsID KeyType
267 * @see nsTHashtable::EntryType for specification
269 class nsIDHashKey
: public PLDHashEntryHdr
272 typedef const nsID
& KeyType
;
273 typedef const nsID
* KeyTypePointer
;
275 nsIDHashKey(const nsID
* inID
) : mID(*inID
) { }
276 nsIDHashKey(const nsIDHashKey
& toCopy
) : mID(toCopy
.mID
) { }
279 KeyType
GetKey() const { return mID
; }
281 PRBool
KeyEquals(KeyTypePointer aKey
) const { return aKey
->Equals(mID
); }
283 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return &aKey
; }
284 static PLDHashNumber
HashKey(KeyTypePointer aKey
);
285 enum { ALLOW_MEMMOVE
= PR_TRUE
};
292 * hashkey wrapper for "dependent" const char*; this class does not "own"
293 * its string pointer.
295 * This class must only be used if the strings have a lifetime longer than
296 * the hashtable they occupy. This normally occurs only for static
297 * strings or strings that have been arena-allocated.
299 * @see nsTHashtable::EntryType for specification
301 class nsDepCharHashKey
: public PLDHashEntryHdr
304 typedef const char* KeyType
;
305 typedef const char* KeyTypePointer
;
307 nsDepCharHashKey(const char* aKey
) { mKey
= aKey
; }
308 nsDepCharHashKey(const nsDepCharHashKey
& toCopy
) { mKey
= toCopy
.mKey
; }
309 ~nsDepCharHashKey() { }
311 const char* GetKey() const { return mKey
; }
312 PRBool
KeyEquals(const char* aKey
) const
314 return !strcmp(mKey
, aKey
);
317 static const char* KeyToPointer(const char* aKey
) { return aKey
; }
318 static PLDHashNumber
HashKey(const char* aKey
) { return HashString(aKey
); }
319 enum { ALLOW_MEMMOVE
= PR_TRUE
};
326 * hashkey wrapper for const char*; at construction, this class duplicates
327 * a string pointed to by the pointer so that it doesn't matter whether or not
328 * the string lives longer than the hash table.
330 class nsCharPtrHashKey
: public PLDHashEntryHdr
333 typedef const char* KeyType
;
334 typedef const char* KeyTypePointer
;
336 nsCharPtrHashKey(const char* aKey
) : mKey(strdup(aKey
)) { }
337 nsCharPtrHashKey(const nsCharPtrHashKey
& toCopy
) : mKey(strdup(toCopy
.mKey
)) { }
338 ~nsCharPtrHashKey() { if (mKey
) free(const_cast<char *>(mKey
)); }
340 const char* GetKey() const { return mKey
; }
341 PRBool
KeyEquals(KeyTypePointer aKey
) const
343 return !strcmp(mKey
, aKey
);
346 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return aKey
; }
347 static PLDHashNumber
HashKey(KeyTypePointer aKey
) { return HashString(aKey
); }
349 enum { ALLOW_MEMMOVE
= PR_TRUE
};
356 * hashkey wrapper for const PRUnichar*; at construction, this class duplicates
357 * a string pointed to by the pointer so that it doesn't matter whether or not
358 * the string lives longer than the hash table.
360 class nsUnicharPtrHashKey
: public PLDHashEntryHdr
363 typedef const PRUnichar
* KeyType
;
364 typedef const PRUnichar
* KeyTypePointer
;
366 nsUnicharPtrHashKey(const PRUnichar
* aKey
) : mKey(NS_strdup(aKey
)) { }
367 nsUnicharPtrHashKey(const nsUnicharPtrHashKey
& toCopy
) : mKey(NS_strdup(toCopy
.mKey
)) { }
368 ~nsUnicharPtrHashKey() { if (mKey
) NS_Free(const_cast<PRUnichar
*>(mKey
)); }
370 const PRUnichar
* GetKey() const { return mKey
; }
371 PRBool
KeyEquals(KeyTypePointer aKey
) const
373 return !NS_strcmp(mKey
, aKey
);
376 static KeyTypePointer
KeyToPointer(KeyType aKey
) { return aKey
; }
377 static PLDHashNumber
HashKey(KeyTypePointer aKey
) { return HashString(aKey
); }
379 enum { ALLOW_MEMMOVE
= PR_TRUE
};
382 const PRUnichar
* mKey
;
386 * Hashtable key class to use with objects that support nsIHashable
388 class nsHashableHashKey
: public PLDHashEntryHdr
391 typedef nsIHashable
* KeyType
;
392 typedef const nsIHashable
* KeyTypePointer
;
394 nsHashableHashKey(const nsIHashable
* aKey
) :
395 mKey(const_cast<nsIHashable
*>(aKey
)) { }
396 nsHashableHashKey(const nsHashableHashKey
& toCopy
) :
397 mKey(toCopy
.mKey
) { }
398 ~nsHashableHashKey() { }
400 nsIHashable
* GetKey() const { return mKey
; }
402 PRBool
KeyEquals(const nsIHashable
* aKey
) const {
404 if (NS_SUCCEEDED(mKey
->Equals(const_cast<nsIHashable
*>(aKey
), &eq
))) {
410 static const nsIHashable
* KeyToPointer(nsIHashable
* aKey
) { return aKey
; }
411 static PLDHashNumber
HashKey(const nsIHashable
* aKey
) {
412 PRUint32 code
= 8888; // magic number if GetHashCode fails :-(
416 const_cast<nsIHashable
*>(aKey
)->GetHashCode(&code
);
417 NS_ASSERTION(NS_SUCCEEDED(rv
), "GetHashCode should not throw!");
421 enum { ALLOW_MEMMOVE
= PR_TRUE
};
424 nsCOMPtr
<nsIHashable
> mKey
;
427 #endif // nsTHashKeys_h__