| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*  hashfuncs.h                                                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							| 
									
										
										
										
											2017-08-27 14:16:55 +02:00
										 |  |  | /*                      https://godotengine.org                          */ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2022-01-13 09:45:09 +01:00
										 |  |  | /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							|  |  |  | /* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md).   */ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* Permission is hereby granted, free of charge, to any person obtaining */ | 
					
						
							|  |  |  | /* a copy of this software and associated documentation files (the       */ | 
					
						
							|  |  |  | /* "Software"), to deal in the Software without restriction, including   */ | 
					
						
							|  |  |  | /* without limitation the rights to use, copy, modify, merge, publish,   */ | 
					
						
							|  |  |  | /* distribute, sublicense, and/or sell copies of the Software, and to    */ | 
					
						
							|  |  |  | /* permit persons to whom the Software is furnished to do so, subject to */ | 
					
						
							|  |  |  | /* the following conditions:                                             */ | 
					
						
							|  |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* The above copyright notice and this permission notice shall be        */ | 
					
						
							|  |  |  | /* included in all copies or substantial portions of the Software.       */ | 
					
						
							|  |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,       */ | 
					
						
							|  |  |  | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF    */ | 
					
						
							|  |  |  | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ | 
					
						
							|  |  |  | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY  */ | 
					
						
							|  |  |  | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,  */ | 
					
						
							|  |  |  | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE     */ | 
					
						
							|  |  |  | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2018-01-05 00:50:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #ifndef HASHFUNCS_H
 | 
					
						
							|  |  |  | #define HASHFUNCS_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 18:13:45 +02:00
										 |  |  | #include "core/math/math_defs.h"
 | 
					
						
							|  |  |  | #include "core/math/math_funcs.h"
 | 
					
						
							|  |  |  | #include "core/node_path.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-12 13:30:56 +01:00
										 |  |  | #include "core/string_name.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-11 18:13:45 +02:00
										 |  |  | #include "core/typedefs.h"
 | 
					
						
							|  |  |  | #include "core/ustring.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Hashing functions | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * DJB2 Hash function | 
					
						
							|  |  |  |  * @param C String | 
					
						
							|  |  |  |  * @return 32-bits hashcode | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static inline uint32_t hash_djb2(const char *p_cstr) { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	const unsigned char *chr = (const unsigned char *)p_cstr; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	uint32_t hash = 5381; | 
					
						
							|  |  |  | 	uint32_t c; | 
					
						
							| 
									
										
										
										
											2016-03-09 00:00:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	while ((c = *chr++)) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		hash = ((hash << 5) + hash) + c; /* hash * 33 + c */ | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-03-09 00:00:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return hash; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | static inline uint32_t hash_djb2_buffer(const uint8_t *p_buff, int p_len, uint32_t p_prev = 5381) { | 
					
						
							| 
									
										
										
										
											2014-05-14 01:22:15 -03:00
										 |  |  | 	uint32_t hash = p_prev; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	for (int i = 0; i < p_len; i++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		hash = ((hash << 5) + hash) + p_buff[i]; /* hash * 33 + c */ | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return hash; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | static inline uint32_t hash_djb2_one_32(uint32_t p_in, uint32_t p_prev = 5381) { | 
					
						
							|  |  |  | 	return ((p_prev << 5) + p_prev) + p_in; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 14:41:16 +01:00
										 |  |  | static inline uint32_t hash_one_uint64(const uint64_t p_int) { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	uint64_t v = p_int; | 
					
						
							| 
									
										
										
										
											2017-02-15 14:41:16 +01:00
										 |  |  | 	v = (~v) + (v << 18); // v = (v << 18) - v - 1;
 | 
					
						
							|  |  |  | 	v = v ^ (v >> 31); | 
					
						
							|  |  |  | 	v = v * 21; // v = (v + (v << 2)) + (v << 4);
 | 
					
						
							|  |  |  | 	v = v ^ (v >> 11); | 
					
						
							|  |  |  | 	v = v + (v << 6); | 
					
						
							|  |  |  | 	v = v ^ (v >> 22); | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	return (int)v; | 
					
						
							| 
									
										
										
										
											2017-02-15 14:41:16 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | static inline uint32_t hash_djb2_one_float(double p_in, uint32_t p_prev = 5381) { | 
					
						
							| 
									
										
										
										
											2017-02-15 14:41:16 +01:00
										 |  |  | 	union { | 
					
						
							|  |  |  | 		double d; | 
					
						
							|  |  |  | 		uint64_t i; | 
					
						
							|  |  |  | 	} u; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Normalize +/- 0.0 and NaN values so they hash the same.
 | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	if (p_in == 0.0f) { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 		u.d = 0.0; | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	} else if (Math::is_nan(p_in)) { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 		u.d = Math_NAN; | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 		u.d = p_in; | 
					
						
							| 
									
										
										
										
											2021-05-05 12:44:11 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 14:41:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	return ((p_prev << 5) + p_prev) + hash_one_uint64(u.i); | 
					
						
							| 
									
										
										
										
											2017-02-15 14:41:16 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | template <class T> | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | static inline uint32_t make_uint32_t(T p_in) { | 
					
						
							|  |  |  | 	union { | 
					
						
							|  |  |  | 		T t; | 
					
						
							|  |  |  | 		uint32_t _u32; | 
					
						
							|  |  |  | 	} _u; | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	_u._u32 = 0; | 
					
						
							|  |  |  | 	_u.t = p_in; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return _u._u32; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | static inline uint64_t hash_djb2_one_64(uint64_t p_in, uint64_t p_prev = 5381) { | 
					
						
							|  |  |  | 	return ((p_prev << 5) + p_prev) + p_in; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | template <class T> | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | static inline uint64_t make_uint64_t(T p_in) { | 
					
						
							|  |  |  | 	union { | 
					
						
							|  |  |  | 		T t; | 
					
						
							|  |  |  | 		uint64_t _u64; | 
					
						
							|  |  |  | 	} _u; | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	_u._u64 = 0; // in case p_in is smaller
 | 
					
						
							| 
									
										
										
										
											2016-03-09 00:00:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | 	_u.t = p_in; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return _u._u64; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 02:40:13 +02:00
										 |  |  | struct HashMapHasherDefault { | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const String &p_string) { return p_string.hash(); } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const char *p_cstr) { return hash_djb2(p_cstr); } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const uint64_t p_int) { return hash_one_uint64(p_int); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const int64_t p_int) { return hash(uint64_t(p_int)); } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const float p_float) { return hash_djb2_one_float(p_float); } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const double p_double) { return hash_djb2_one_float(p_double); } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const uint32_t p_int) { return p_int; } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const int32_t p_int) { return (uint32_t)p_int; } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const uint16_t p_int) { return p_int; } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const int16_t p_int) { return (uint32_t)p_int; } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const uint8_t p_int) { return p_int; } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const int8_t p_int) { return (uint32_t)p_int; } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const wchar_t p_wchar) { return (uint32_t)p_wchar; } | 
					
						
							| 
									
										
										
										
											2018-07-02 15:08:35 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const StringName &p_string_name) { return p_string_name.hash(); } | 
					
						
							|  |  |  | 	static _FORCE_INLINE_ uint32_t hash(const NodePath &p_path) { return p_path.hash(); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 02:40:13 +02:00
										 |  |  | 	//static _FORCE_INLINE_ uint32_t hash(const void* p_ptr)  { return uint32_t(uint64_t(p_ptr))*(0x9e3779b1L); }
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | struct HashMapComparatorDefault { | 
					
						
							|  |  |  | 	static bool compare(const T &p_lhs, const T &p_rhs) { | 
					
						
							|  |  |  | 		return p_lhs == p_rhs; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool compare(const float &p_lhs, const float &p_rhs) { | 
					
						
							|  |  |  | 		return (p_lhs == p_rhs) || (Math::is_nan(p_lhs) && Math::is_nan(p_rhs)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool compare(const double &p_lhs, const double &p_rhs) { | 
					
						
							|  |  |  | 		return (p_lhs == p_rhs) || (Math::is_nan(p_lhs) && Math::is_nan(p_rhs)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #endif
 |