| 
									
										
										
										
											2019-11-05 12:01:00 +01:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*  rid_owner.h                                                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							|  |  |  | /*                      https://godotengine.org                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2020-02-11 14:01:43 +01:00
										 |  |  | /* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							|  |  |  | /* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md).   */ | 
					
						
							| 
									
										
										
										
											2019-11-05 12:01:00 +01: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.                */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | #ifndef RID_OWNER_H
 | 
					
						
							|  |  |  | #define RID_OWNER_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 00:05:21 -03:00
										 |  |  | #include "core/list.h"
 | 
					
						
							|  |  |  | #include "core/oa_hash_map.h"
 | 
					
						
							|  |  |  | #include "core/os/memory.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | #include "core/print_string.h"
 | 
					
						
							|  |  |  | #include "core/rid.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-14 00:05:21 -03:00
										 |  |  | #include "core/safe_refcount.h"
 | 
					
						
							|  |  |  | #include "core/set.h"
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | #include "core/spin_lock.h"
 | 
					
						
							| 
									
										
										
										
											2019-07-29 18:19:31 -03:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:13:06 -03:00
										 |  |  | #include <typeinfo>
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | class RID_AllocBase { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	static volatile uint64_t base_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  | 	static RID _make_from_id(uint64_t p_id) { | 
					
						
							|  |  |  | 		RID rid; | 
					
						
							|  |  |  | 		rid._id = p_id; | 
					
						
							|  |  |  | 		return rid; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	static uint64_t _gen_id() { | 
					
						
							|  |  |  | 		return atomic_increment(&base_id); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	static RID _gen_rid() { | 
					
						
							|  |  |  | 		return _make_from_id(_gen_id()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	virtual ~RID_AllocBase() {} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | template <class T, bool THREAD_SAFE = false> | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | class RID_Alloc : public RID_AllocBase { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	T **chunks; | 
					
						
							|  |  |  | 	uint32_t **free_list_chunks; | 
					
						
							|  |  |  | 	uint32_t **validator_chunks; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uint32_t elements_in_chunk; | 
					
						
							|  |  |  | 	uint32_t max_alloc; | 
					
						
							|  |  |  | 	uint32_t alloc_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const char *description; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 	SpinLock spin_lock; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | public: | 
					
						
							|  |  |  | 	RID make_rid(const T &p_value) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		if (alloc_count == max_alloc) { | 
					
						
							|  |  |  | 			//allocate a new chunk
 | 
					
						
							| 
									
										
										
										
											2019-06-19 17:03:19 -03:00
										 |  |  | 			uint32_t chunk_count = alloc_count == 0 ? 0 : (max_alloc / elements_in_chunk); | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			//grow chunks
 | 
					
						
							|  |  |  | 			chunks = (T **)memrealloc(chunks, sizeof(T *) * (chunk_count + 1)); | 
					
						
							|  |  |  | 			chunks[chunk_count] = (T *)memalloc(sizeof(T) * elements_in_chunk); //but don't initialize
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			//grow validators
 | 
					
						
							|  |  |  | 			validator_chunks = (uint32_t **)memrealloc(validator_chunks, sizeof(uint32_t *) * (chunk_count + 1)); | 
					
						
							|  |  |  | 			validator_chunks[chunk_count] = (uint32_t *)memalloc(sizeof(uint32_t) * elements_in_chunk); | 
					
						
							|  |  |  | 			//grow free lists
 | 
					
						
							|  |  |  | 			free_list_chunks = (uint32_t **)memrealloc(free_list_chunks, sizeof(uint32_t *) * (chunk_count + 1)); | 
					
						
							|  |  |  | 			free_list_chunks[chunk_count] = (uint32_t *)memalloc(sizeof(uint32_t) * elements_in_chunk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			//initialize
 | 
					
						
							|  |  |  | 			for (uint32_t i = 0; i < elements_in_chunk; i++) { | 
					
						
							|  |  |  | 				//dont initialize chunk
 | 
					
						
							|  |  |  | 				validator_chunks[chunk_count][i] = 0xFFFFFFFF; | 
					
						
							|  |  |  | 				free_list_chunks[chunk_count][i] = alloc_count + i; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			max_alloc += elements_in_chunk; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t free_index = free_list_chunks[alloc_count / elements_in_chunk][alloc_count % elements_in_chunk]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t free_chunk = free_index / elements_in_chunk; | 
					
						
							|  |  |  | 		uint32_t free_element = free_index % elements_in_chunk; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		T *ptr = &chunks[free_chunk][free_element]; | 
					
						
							|  |  |  | 		memnew_placement(ptr, T(p_value)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-20 21:31:21 +02:00
										 |  |  | 		uint32_t validator = (uint32_t)(_gen_id() & 0xFFFFFFFF); | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		uint64_t id = validator; | 
					
						
							|  |  |  | 		id <<= 32; | 
					
						
							|  |  |  | 		id |= free_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		validator_chunks[free_chunk][free_element] = validator; | 
					
						
							|  |  |  | 		alloc_count++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		return _make_from_id(id); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ T *getornull(const RID &p_rid) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		uint64_t id = p_rid.get_id(); | 
					
						
							|  |  |  | 		uint32_t idx = uint32_t(id & 0xFFFFFFFF); | 
					
						
							| 
									
										
										
										
											2019-06-24 16:13:06 -03:00
										 |  |  | 		if (unlikely(idx >= max_alloc)) { | 
					
						
							| 
									
										
										
										
											2019-08-05 14:18:13 +02:00
										 |  |  | 			if (THREAD_SAFE) { | 
					
						
							|  |  |  | 				spin_lock.unlock(); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 			return nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t idx_chunk = idx / elements_in_chunk; | 
					
						
							|  |  |  | 		uint32_t idx_element = idx % elements_in_chunk; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t validator = uint32_t(id >> 32); | 
					
						
							| 
									
										
										
										
											2019-08-05 14:18:13 +02:00
										 |  |  | 		if (unlikely(validator_chunks[idx_chunk][idx_element] != validator)) { | 
					
						
							|  |  |  | 			if (THREAD_SAFE) { | 
					
						
							|  |  |  | 				spin_lock.unlock(); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 			return nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		T *ptr = &chunks[idx_chunk][idx_element]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return ptr; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ bool owns(const RID &p_rid) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		uint64_t id = p_rid.get_id(); | 
					
						
							|  |  |  | 		uint32_t idx = uint32_t(id & 0xFFFFFFFF); | 
					
						
							| 
									
										
										
										
											2019-06-24 16:13:06 -03:00
										 |  |  | 		if (unlikely(idx >= max_alloc)) { | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 			if (THREAD_SAFE) { | 
					
						
							|  |  |  | 				spin_lock.unlock(); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t idx_chunk = idx / elements_in_chunk; | 
					
						
							|  |  |  | 		uint32_t idx_element = idx % elements_in_chunk; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t validator = uint32_t(id >> 32); | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		bool owned = validator_chunks[idx_chunk][idx_element] == validator; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return owned; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ void free(const RID &p_rid) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		uint64_t id = p_rid.get_id(); | 
					
						
							|  |  |  | 		uint32_t idx = uint32_t(id & 0xFFFFFFFF); | 
					
						
							| 
									
										
										
										
											2019-08-05 14:18:13 +02:00
										 |  |  | 		if (unlikely(idx >= max_alloc)) { | 
					
						
							|  |  |  | 			if (THREAD_SAFE) { | 
					
						
							|  |  |  | 				spin_lock.unlock(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			ERR_FAIL(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t idx_chunk = idx / elements_in_chunk; | 
					
						
							|  |  |  | 		uint32_t idx_element = idx % elements_in_chunk; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint32_t validator = uint32_t(id >> 32); | 
					
						
							| 
									
										
										
										
											2019-08-05 14:18:13 +02:00
										 |  |  | 		if (unlikely(validator_chunks[idx_chunk][idx_element] != validator)) { | 
					
						
							|  |  |  | 			if (THREAD_SAFE) { | 
					
						
							|  |  |  | 				spin_lock.unlock(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			ERR_FAIL(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		chunks[idx_chunk][idx_element].~T(); | 
					
						
							|  |  |  | 		validator_chunks[idx_chunk][idx_element] = 0xFFFFFFFF; // go invalid
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		alloc_count--; | 
					
						
							| 
									
										
										
										
											2019-06-19 17:03:19 -03:00
										 |  |  | 		free_list_chunks[alloc_count / elements_in_chunk][alloc_count % elements_in_chunk] = idx; | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 22:54:32 -03:00
										 |  |  | 	_FORCE_INLINE_ uint32_t get_rid_count() const { | 
					
						
							|  |  |  | 		return alloc_count; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 18:19:31 -03:00
										 |  |  | 	_FORCE_INLINE_ T *get_ptr_by_index(uint32_t p_index) { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		ERR_FAIL_INDEX_V(p_index, alloc_count, nullptr); | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-07-04 22:54:32 -03:00
										 |  |  | 		uint64_t idx = free_list_chunks[p_index / elements_in_chunk][p_index % elements_in_chunk]; | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 		T *ptr = &chunks[idx / elements_in_chunk][idx % elements_in_chunk]; | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return ptr; | 
					
						
							| 
									
										
										
										
											2019-07-04 22:54:32 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 18:19:31 -03:00
										 |  |  | 	_FORCE_INLINE_ RID get_rid_by_index(uint32_t p_index) { | 
					
						
							|  |  |  | 		ERR_FAIL_INDEX_V(p_index, alloc_count, RID()); | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		uint64_t idx = free_list_chunks[p_index / elements_in_chunk][p_index % elements_in_chunk]; | 
					
						
							|  |  |  | 		uint64_t validator = validator_chunks[idx / elements_in_chunk][idx % elements_in_chunk]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		RID rid = _make_from_id((validator << 32) | idx); | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return rid; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 	void get_owned_list(List<RID> *p_owned) { | 
					
						
							| 
									
										
										
										
											2019-10-20 21:31:21 +02:00
										 |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.lock(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for (size_t i = 0; i < max_alloc; i++) { | 
					
						
							|  |  |  | 			uint64_t validator = validator_chunks[i / elements_in_chunk][i % elements_in_chunk]; | 
					
						
							|  |  |  | 			if (validator != 0xFFFFFFFF) { | 
					
						
							|  |  |  | 				p_owned->push_back(_make_from_id((validator << 32) | i)); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (THREAD_SAFE) { | 
					
						
							|  |  |  | 			spin_lock.unlock(); | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void set_description(const char *p_descrption) { | 
					
						
							|  |  |  | 		description = p_descrption; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	RID_Alloc(uint32_t p_target_chunk_byte_size = 4096) { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		chunks = nullptr; | 
					
						
							|  |  |  | 		free_list_chunks = nullptr; | 
					
						
							|  |  |  | 		validator_chunks = nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		elements_in_chunk = sizeof(T) > p_target_chunk_byte_size ? 1 : (p_target_chunk_byte_size / sizeof(T)); | 
					
						
							|  |  |  | 		max_alloc = 0; | 
					
						
							|  |  |  | 		alloc_count = 0; | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		description = nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	~RID_Alloc() { | 
					
						
							|  |  |  | 		if (alloc_count) { | 
					
						
							|  |  |  | 			if (description) { | 
					
						
							| 
									
										
										
										
											2019-06-24 16:13:06 -03:00
										 |  |  | 				print_error("ERROR: " + itos(alloc_count) + " RID allocations of type '" + description + "' were leaked at exit."); | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2020-02-29 19:40:24 +01:00
										 |  |  | #ifdef NO_SAFE_CAST
 | 
					
						
							|  |  |  | 				print_error("ERROR: " + itos(alloc_count) + " RID allocations of type 'unknown' were leaked at exit."); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:13:06 -03:00
										 |  |  | 				print_error("ERROR: " + itos(alloc_count) + " RID allocations of type '" + typeid(T).name() + "' were leaked at exit."); | 
					
						
							| 
									
										
										
										
											2020-02-29 19:40:24 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-20 21:31:21 +02:00
										 |  |  | 			for (size_t i = 0; i < max_alloc; i++) { | 
					
						
							|  |  |  | 				uint64_t validator = validator_chunks[i / elements_in_chunk][i % elements_in_chunk]; | 
					
						
							|  |  |  | 				if (validator != 0xFFFFFFFF) { | 
					
						
							|  |  |  | 					chunks[i / elements_in_chunk][i % elements_in_chunk].~T(); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:13:06 -03:00
										 |  |  | 		uint32_t chunk_count = max_alloc / elements_in_chunk; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		for (uint32_t i = 0; i < chunk_count; i++) { | 
					
						
							|  |  |  | 			memfree(chunks[i]); | 
					
						
							|  |  |  | 			memfree(validator_chunks[i]); | 
					
						
							|  |  |  | 			memfree(free_list_chunks[i]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (chunks) { | 
					
						
							|  |  |  | 			memfree(chunks); | 
					
						
							|  |  |  | 			memfree(free_list_chunks); | 
					
						
							|  |  |  | 			memfree(validator_chunks); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | template <class T, bool THREAD_SAFE = false> | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | class RID_PtrOwner { | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 	RID_Alloc<T *, THREAD_SAFE> alloc; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	_FORCE_INLINE_ RID make_rid(T *p_ptr) { | 
					
						
							|  |  |  | 		return alloc.make_rid(p_ptr); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ T *getornull(const RID &p_rid) { | 
					
						
							|  |  |  | 		T **ptr = alloc.getornull(p_rid); | 
					
						
							|  |  |  | 		if (unlikely(!ptr)) { | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 			return nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return *ptr; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ bool owns(const RID &p_rid) { | 
					
						
							|  |  |  | 		return alloc.owns(p_rid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ void free(const RID &p_rid) { | 
					
						
							|  |  |  | 		alloc.free(p_rid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ void get_owned_list(List<RID> *p_owned) { | 
					
						
							|  |  |  | 		return alloc.get_owned_list(p_owned); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void set_description(const char *p_descrption) { | 
					
						
							|  |  |  | 		alloc.set_description(p_descrption); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RID_PtrOwner(uint32_t p_target_chunk_byte_size = 4096) : | 
					
						
							|  |  |  | 			alloc(p_target_chunk_byte_size) {} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | template <class T, bool THREAD_SAFE = false> | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | class RID_Owner { | 
					
						
							| 
									
										
										
										
											2019-07-29 12:59:18 -03:00
										 |  |  | 	RID_Alloc<T, THREAD_SAFE> alloc; | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	_FORCE_INLINE_ RID make_rid(const T &p_ptr) { | 
					
						
							|  |  |  | 		return alloc.make_rid(p_ptr); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ T *getornull(const RID &p_rid) { | 
					
						
							|  |  |  | 		return alloc.getornull(p_rid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ bool owns(const RID &p_rid) { | 
					
						
							|  |  |  | 		return alloc.owns(p_rid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_FORCE_INLINE_ void free(const RID &p_rid) { | 
					
						
							|  |  |  | 		alloc.free(p_rid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-04 22:54:32 -03:00
										 |  |  | 	_FORCE_INLINE_ uint32_t get_rid_count() const { | 
					
						
							|  |  |  | 		return alloc.get_rid_count(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 18:19:31 -03:00
										 |  |  | 	_FORCE_INLINE_ RID get_rid_by_index(uint32_t p_index) { | 
					
						
							| 
									
										
										
										
											2019-07-04 22:54:32 -03:00
										 |  |  | 		return alloc.get_rid_by_index(p_index); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-29 18:19:31 -03:00
										 |  |  | 	_FORCE_INLINE_ T *get_ptr_by_index(uint32_t p_index) { | 
					
						
							|  |  |  | 		return alloc.get_ptr_by_index(p_index); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:38:51 -03:00
										 |  |  | 	_FORCE_INLINE_ void get_owned_list(List<RID> *p_owned) { | 
					
						
							|  |  |  | 		return alloc.get_owned_list(p_owned); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void set_description(const char *p_descrption) { | 
					
						
							|  |  |  | 		alloc.set_description(p_descrption); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RID_Owner(uint32_t p_target_chunk_byte_size = 4096) : | 
					
						
							|  |  |  | 			alloc(p_target_chunk_byte_size) {} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #endif // RID_OWNER_H
 |