|  | @@ -35,10 +35,22 @@
 | 
											
												
													
														|  |  #include "core/object/class_db.h"
 |  |  #include "core/object/class_db.h"
 | 
											
												
													
														|  |  #include "core/object/script_language.h"
 |  |  #include "core/object/script_language.h"
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +#ifdef DEV_ENABLED
 | 
											
												
													
														|  | 
 |  | +// Includes sanity checks to ensure that a queue set as a thread singleton override
 | 
											
												
													
														|  | 
 |  | +// is only ever called from the thread it was set for.
 | 
											
												
													
														|  | 
 |  | +#define LOCK_MUTEX                                     \
 | 
											
												
													
														|  | 
 |  | +	if (this != MessageQueue::thread_singleton) {      \
 | 
											
												
													
														|  | 
 |  | +		DEV_ASSERT(!this->is_current_thread_override); \
 | 
											
												
													
														|  | 
 |  | +		mutex.lock();                                  \
 | 
											
												
													
														|  | 
 |  | +	} else {                                           \
 | 
											
												
													
														|  | 
 |  | +		DEV_ASSERT(this->is_current_thread_override);  \
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +#else
 | 
											
												
													
														|  |  #define LOCK_MUTEX                                \
 |  |  #define LOCK_MUTEX                                \
 | 
											
												
													
														|  |  	if (this != MessageQueue::thread_singleton) { \
 |  |  	if (this != MessageQueue::thread_singleton) { \
 | 
											
												
													
														|  |  		mutex.lock();                             \
 |  |  		mutex.lock();                             \
 | 
											
												
													
														|  |  	}
 |  |  	}
 | 
											
												
													
														|  | 
 |  | +#endif
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  #define UNLOCK_MUTEX                              \
 |  |  #define UNLOCK_MUTEX                              \
 | 
											
												
													
														|  |  	if (this != MessageQueue::thread_singleton) { \
 |  |  	if (this != MessageQueue::thread_singleton) { \
 | 
											
										
											
												
													
														|  | @@ -213,8 +225,8 @@ void CallQueue::_call_function(const Callable &p_callable, const Variant *p_args
 | 
											
												
													
														|  |  Error CallQueue::flush() {
 |  |  Error CallQueue::flush() {
 | 
											
												
													
														|  |  	LOCK_MUTEX;
 |  |  	LOCK_MUTEX;
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -	// Non-main threads are not meant to be flushed, but appended to the main one.
 |  | 
 | 
											
												
													
														|  | -	if (this != MessageQueue::main_singleton) {
 |  | 
 | 
											
												
													
														|  | 
 |  | +	// Thread overrides are not meant to be flushed, but appended to the main one.
 | 
											
												
													
														|  | 
 |  | +	if (this == MessageQueue::thread_singleton) {
 | 
											
												
													
														|  |  		if (pages.size() == 0) {
 |  |  		if (pages.size() == 0) {
 | 
											
												
													
														|  |  			return OK;
 |  |  			return OK;
 | 
											
												
													
														|  |  		}
 |  |  		}
 | 
											
										
											
												
													
														|  | @@ -237,6 +249,7 @@ Error CallQueue::flush() {
 | 
											
												
													
														|  |  				uint32_t dst_offset = mq->page_bytes[dst_page];
 |  |  				uint32_t dst_offset = mq->page_bytes[dst_page];
 | 
											
												
													
														|  |  				if (dst_offset + page_bytes[0] < uint32_t(PAGE_SIZE_BYTES)) {
 |  |  				if (dst_offset + page_bytes[0] < uint32_t(PAGE_SIZE_BYTES)) {
 | 
											
												
													
														|  |  					memcpy(mq->pages[dst_page]->data + dst_offset, pages[0]->data, page_bytes[0]);
 |  |  					memcpy(mq->pages[dst_page]->data + dst_offset, pages[0]->data, page_bytes[0]);
 | 
											
												
													
														|  | 
 |  | +					mq->page_bytes[dst_page] += page_bytes[0];
 | 
											
												
													
														|  |  					src_page++;
 |  |  					src_page++;
 | 
											
												
													
														|  |  				}
 |  |  				}
 | 
											
												
													
														|  |  			}
 |  |  			}
 | 
											
										
											
												
													
														|  | @@ -520,6 +533,10 @@ CallQueue::~CallQueue() {
 | 
											
												
													
														|  |  	if (!allocator_is_custom) {
 |  |  	if (!allocator_is_custom) {
 | 
											
												
													
														|  |  		memdelete(allocator);
 |  |  		memdelete(allocator);
 | 
											
												
													
														|  |  	}
 |  |  	}
 | 
											
												
													
														|  | 
 |  | +	// This is done here to avoid a circular dependency between the sanity checks and the thread singleton pointer.
 | 
											
												
													
														|  | 
 |  | +	if (this == MessageQueue::thread_singleton) {
 | 
											
												
													
														|  | 
 |  | +		MessageQueue::thread_singleton = nullptr;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  |  }
 |  |  }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  //////////////////////
 |  |  //////////////////////
 | 
											
										
											
												
													
														|  | @@ -528,7 +545,18 @@ CallQueue *MessageQueue::main_singleton = nullptr;
 | 
											
												
													
														|  |  thread_local CallQueue *MessageQueue::thread_singleton = nullptr;
 |  |  thread_local CallQueue *MessageQueue::thread_singleton = nullptr;
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  void MessageQueue::set_thread_singleton_override(CallQueue *p_thread_singleton) {
 |  |  void MessageQueue::set_thread_singleton_override(CallQueue *p_thread_singleton) {
 | 
											
												
													
														|  | 
 |  | +	DEV_ASSERT(p_thread_singleton); // To unset the thread singleton, don't call this with nullptr, but just memfree() it.
 | 
											
												
													
														|  | 
 |  | +#ifdef DEV_ENABLED
 | 
											
												
													
														|  | 
 |  | +	if (thread_singleton) {
 | 
											
												
													
														|  | 
 |  | +		thread_singleton->is_current_thread_override = false;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +#endif
 | 
											
												
													
														|  |  	thread_singleton = p_thread_singleton;
 |  |  	thread_singleton = p_thread_singleton;
 | 
											
												
													
														|  | 
 |  | +#ifdef DEV_ENABLED
 | 
											
												
													
														|  | 
 |  | +	if (thread_singleton) {
 | 
											
												
													
														|  | 
 |  | +		thread_singleton->is_current_thread_override = true;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +#endif
 | 
											
												
													
														|  |  }
 |  |  }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  MessageQueue::MessageQueue() :
 |  |  MessageQueue::MessageQueue() :
 |