mirror of
				https://github.com/godotengine/godot.git
				synced 2025-10-26 03:04:31 +00:00 
			
		
		
		
	
		
			
	
	
		
			255 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			255 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | /**************************************************************************/ | ||
|  | /*  openxr_future_extension.cpp                                           */ | ||
|  | /**************************************************************************/ | ||
|  | /*                         This file is part of:                          */ | ||
|  | /*                             GODOT ENGINE                               */ | ||
|  | /*                        https://godotengine.org                         */ | ||
|  | /**************************************************************************/ | ||
|  | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | ||
|  | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | ||
|  | /*                                                                        */ | ||
|  | /* 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.                 */ | ||
|  | /**************************************************************************/ | ||
|  | 
 | ||
|  | #include "openxr_future_extension.h"
 | ||
|  | 
 | ||
|  | #include "../openxr_api.h"
 | ||
|  | 
 | ||
|  | ////////////////////////////////////////////////////////////////////////////
 | ||
|  | // OpenXRFutureResult
 | ||
|  | 
 | ||
|  | void OpenXRFutureResult::_bind_methods() { | ||
|  | 	ClassDB::bind_method(D_METHOD("get_status"), &OpenXRFutureResult::get_status); | ||
|  | 	ClassDB::bind_method(D_METHOD("get_future"), &OpenXRFutureResult::_get_future); | ||
|  | 	ClassDB::bind_method(D_METHOD("cancel_future"), &OpenXRFutureResult::cancel_future); | ||
|  | 
 | ||
|  | 	ADD_SIGNAL(MethodInfo("completed", PropertyInfo(Variant::OBJECT, "result", PROPERTY_HINT_RESOURCE_TYPE, "OpenXRFutureResult"))); | ||
|  | 
 | ||
|  | 	BIND_ENUM_CONSTANT(RESULT_RUNNING); | ||
|  | 	BIND_ENUM_CONSTANT(RESULT_FINISHED); | ||
|  | 	BIND_ENUM_CONSTANT(RESULT_CANCELLED); | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureResult::_mark_as_finished() { | ||
|  | 	// Update our status
 | ||
|  | 	status = RESULT_FINISHED; | ||
|  | 
 | ||
|  | 	// Perform our callback
 | ||
|  | 	on_success_callback.call((uint64_t)future); | ||
|  | 
 | ||
|  | 	// Emit our signal
 | ||
|  | 	emit_signal(SNAME("completed"), this); | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureResult::_mark_as_cancelled() { | ||
|  | 	// Update our status
 | ||
|  | 	status = RESULT_CANCELLED; | ||
|  | 
 | ||
|  | 	// There is no point in doing a callback for cancellation as its always user invoked.
 | ||
|  | 
 | ||
|  | 	// But we do emit our signal to make sure any await finishes.
 | ||
|  | 	emit_signal(SNAME("completed"), this); | ||
|  | } | ||
|  | 
 | ||
|  | OpenXRFutureResult::ResultStatus OpenXRFutureResult::get_status() const { | ||
|  | 	return status; | ||
|  | } | ||
|  | 
 | ||
|  | XrFutureEXT OpenXRFutureResult::get_future() const { | ||
|  | 	return future; | ||
|  | } | ||
|  | 
 | ||
|  | uint64_t OpenXRFutureResult::_get_future() const { | ||
|  | 	return (uint64_t)future; | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureResult::cancel_future() { | ||
|  | 	ERR_FAIL_COND(status != RESULT_RUNNING); | ||
|  | 
 | ||
|  | 	OpenXRFutureExtension *future_extension = OpenXRFutureExtension::get_singleton(); | ||
|  | 	ERR_FAIL_NULL(future_extension); | ||
|  | 
 | ||
|  | 	future_extension->cancel_future(future); | ||
|  | } | ||
|  | 
 | ||
|  | OpenXRFutureResult::OpenXRFutureResult(XrFutureEXT p_future, const Callable &p_on_success) { | ||
|  | 	future = p_future; | ||
|  | 	on_success_callback = p_on_success; | ||
|  | } | ||
|  | 
 | ||
|  | ////////////////////////////////////////////////////////////////////////////
 | ||
|  | // OpenXRFutureExtension
 | ||
|  | 
 | ||
|  | OpenXRFutureExtension *OpenXRFutureExtension::singleton = nullptr; | ||
|  | 
 | ||
|  | OpenXRFutureExtension *OpenXRFutureExtension::get_singleton() { | ||
|  | 	return singleton; | ||
|  | } | ||
|  | 
 | ||
|  | OpenXRFutureExtension::OpenXRFutureExtension() { | ||
|  | 	singleton = this; | ||
|  | } | ||
|  | 
 | ||
|  | OpenXRFutureExtension::~OpenXRFutureExtension() { | ||
|  | 	singleton = nullptr; | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::_bind_methods() { | ||
|  | 	ClassDB::bind_method(D_METHOD("is_active"), &OpenXRFutureExtension::is_active); | ||
|  | 	ClassDB::bind_method(D_METHOD("register_future", "future", "on_success"), &OpenXRFutureExtension::_register_future, DEFVAL(Callable())); | ||
|  | 	ClassDB::bind_method(D_METHOD("cancel_future", "future"), &OpenXRFutureExtension::_cancel_future); | ||
|  | } | ||
|  | 
 | ||
|  | HashMap<String, bool *> OpenXRFutureExtension::get_requested_extensions() { | ||
|  | 	HashMap<String, bool *> request_extensions; | ||
|  | 
 | ||
|  | 	request_extensions[XR_EXT_FUTURE_EXTENSION_NAME] = &future_ext; | ||
|  | 
 | ||
|  | 	return request_extensions; | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::on_instance_created(const XrInstance p_instance) { | ||
|  | 	if (future_ext) { | ||
|  | 		EXT_INIT_XR_FUNC(xrPollFutureEXT); | ||
|  | 		EXT_INIT_XR_FUNC(xrCancelFutureEXT); | ||
|  | 
 | ||
|  | 		future_ext = xrPollFutureEXT_ptr && xrCancelFutureEXT_ptr; | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::on_instance_destroyed() { | ||
|  | 	xrPollFutureEXT_ptr = nullptr; | ||
|  | 	xrCancelFutureEXT_ptr = nullptr; | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::on_session_destroyed() { | ||
|  | 	if (!is_active()) { | ||
|  | 		return; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	OpenXRAPI *openxr_api = OpenXRAPI::get_singleton(); | ||
|  | 	ERR_FAIL_NULL(openxr_api); | ||
|  | 
 | ||
|  | 	// Cancel any running futures.
 | ||
|  | 	for (const KeyValue<XrFutureEXT, Ref<OpenXRFutureResult>> &element : futures) { | ||
|  | 		XrFutureCancelInfoEXT cancel_info = { | ||
|  | 			XR_TYPE_FUTURE_CANCEL_INFO_EXT, // type
 | ||
|  | 			nullptr, // next
 | ||
|  | 			element.key // future
 | ||
|  | 		}; | ||
|  | 		XrResult result = xrCancelFutureEXT_ptr(openxr_api->get_instance(), &cancel_info); | ||
|  | 		if (XR_FAILED(result)) { | ||
|  | 			WARN_PRINT("OpenXR: Failed to cancel future [" + openxr_api->get_error_string(result) + "]"); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		// Make sure we mark our future result as cancelled
 | ||
|  | 		element.value->_mark_as_cancelled(); | ||
|  | 	} | ||
|  | 	futures.clear(); | ||
|  | } | ||
|  | 
 | ||
|  | bool OpenXRFutureExtension::is_active() const { | ||
|  | 	return future_ext; | ||
|  | } | ||
|  | 
 | ||
|  | Ref<OpenXRFutureResult> OpenXRFutureExtension::register_future(XrFutureEXT p_future, const Callable &p_on_success) { | ||
|  | 	ERR_FAIL_COND_V(futures.has(p_future), nullptr); | ||
|  | 
 | ||
|  | 	Ref<OpenXRFutureResult> future_result; | ||
|  | 	future_result.instantiate(p_future, p_on_success); | ||
|  | 
 | ||
|  | 	futures[p_future] = future_result; | ||
|  | 
 | ||
|  | 	return future_result; | ||
|  | } | ||
|  | 
 | ||
|  | Ref<OpenXRFutureResult> OpenXRFutureExtension::_register_future(uint64_t p_future, const Callable &p_on_success) { | ||
|  | 	return register_future((XrFutureEXT)p_future, p_on_success); | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::cancel_future(XrFutureEXT p_future) { | ||
|  | 	ERR_FAIL_COND(!futures.has(p_future)); | ||
|  | 
 | ||
|  | 	OpenXRAPI *openxr_api = OpenXRAPI::get_singleton(); | ||
|  | 	ERR_FAIL_NULL(openxr_api); | ||
|  | 
 | ||
|  | 	XrFutureCancelInfoEXT cancel_info = { | ||
|  | 		XR_TYPE_FUTURE_CANCEL_INFO_EXT, // type
 | ||
|  | 		nullptr, // next
 | ||
|  | 		p_future // future
 | ||
|  | 	}; | ||
|  | 	XrResult result = xrCancelFutureEXT_ptr(openxr_api->get_instance(), &cancel_info); | ||
|  | 	if (XR_FAILED(result)) { | ||
|  | 		WARN_PRINT("OpenXR: Failed to cancel future [" + openxr_api->get_error_string(result) + "]"); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Make sure we mark our future result as cancelled
 | ||
|  | 	futures[p_future]->_mark_as_cancelled(); | ||
|  | 
 | ||
|  | 	// And erase it from the futures we track
 | ||
|  | 	futures.erase(p_future); | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::_cancel_future(uint64_t p_future) { | ||
|  | 	cancel_future((XrFutureEXT)p_future); | ||
|  | } | ||
|  | 
 | ||
|  | void OpenXRFutureExtension::on_process() { | ||
|  | 	if (!is_active()) { | ||
|  | 		return; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	OpenXRAPI *openxr_api = OpenXRAPI::get_singleton(); | ||
|  | 	ERR_FAIL_NULL(openxr_api); | ||
|  | 
 | ||
|  | 	// Process futures
 | ||
|  | 	Vector<XrFutureEXT> completed; | ||
|  | 	for (const KeyValue<XrFutureEXT, Ref<OpenXRFutureResult>> &element : futures) { | ||
|  | 		XrFuturePollInfoEXT poll_info = { | ||
|  | 			XR_TYPE_FUTURE_POLL_INFO_EXT, // type
 | ||
|  | 			nullptr, // next
 | ||
|  | 			element.key // future
 | ||
|  | 		}; | ||
|  | 		XrFuturePollResultEXT poll_result = { | ||
|  | 			XR_TYPE_FUTURE_POLL_RESULT_EXT, // type
 | ||
|  | 			nullptr, // next
 | ||
|  | 			XR_FUTURE_STATE_MAX_ENUM_EXT // state
 | ||
|  | 		}; | ||
|  | 		XrResult result = xrPollFutureEXT_ptr(openxr_api->get_instance(), &poll_info, &poll_result); | ||
|  | 		if (XR_FAILED(result)) { | ||
|  | 			ERR_PRINT("OpenXR: Failed to obtain future status [" + openxr_api->get_error_string(result) + "]"); | ||
|  | 			// Maybe remove this depending on the error?
 | ||
|  | 			continue; | ||
|  | 		} | ||
|  | 		if (poll_result.state == XR_FUTURE_STATE_READY_EXT) { | ||
|  | 			// Mark our future result as finished (this will invoke our callback).
 | ||
|  | 			element.value->_mark_as_finished(); | ||
|  | 
 | ||
|  | 			// Queue removing this
 | ||
|  | 			completed.push_back(element.key); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Now clean up completed futures.
 | ||
|  | 	for (const XrFutureEXT &future : completed) { | ||
|  | 		futures.erase(future); | ||
|  | 	} | ||
|  | } |