initial commit, 4.5 stable
Some checks failed
🔗 GHA / 📊 Static checks (push) Has been cancelled
🔗 GHA / 🤖 Android (push) Has been cancelled
🔗 GHA / 🍏 iOS (push) Has been cancelled
🔗 GHA / 🐧 Linux (push) Has been cancelled
🔗 GHA / 🍎 macOS (push) Has been cancelled
🔗 GHA / 🏁 Windows (push) Has been cancelled
🔗 GHA / 🌐 Web (push) Has been cancelled
Some checks failed
🔗 GHA / 📊 Static checks (push) Has been cancelled
🔗 GHA / 🤖 Android (push) Has been cancelled
🔗 GHA / 🍏 iOS (push) Has been cancelled
🔗 GHA / 🐧 Linux (push) Has been cancelled
🔗 GHA / 🍎 macOS (push) Has been cancelled
🔗 GHA / 🏁 Windows (push) Has been cancelled
🔗 GHA / 🌐 Web (push) Has been cancelled
This commit is contained in:
266
modules/openxr/extensions/openxr_future_extension.cpp
Normal file
266
modules/openxr/extensions/openxr_future_extension.cpp
Normal file
@@ -0,0 +1,266 @@
|
||||
/**************************************************************************/
|
||||
/* 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);
|
||||
|
||||
ClassDB::bind_method(D_METHOD("set_result_value", "result_value"), &OpenXRFutureResult::set_result_value);
|
||||
ClassDB::bind_method(D_METHOD("get_result_value"), &OpenXRFutureResult::get_result_value);
|
||||
|
||||
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(this); // Note, `this` will be converted to a variant that will be refcounted!
|
||||
|
||||
// Emit our signal, we assume our callback has provided us with the correct result value by calling set_result_value.
|
||||
emit_signal(SNAME("completed"), result_value);
|
||||
}
|
||||
|
||||
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.
|
||||
Variant no_result;
|
||||
emit_signal(SNAME("completed"), no_result);
|
||||
}
|
||||
|
||||
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::set_result_value(const Variant &p_result_value) {
|
||||
result_value = p_result_value;
|
||||
}
|
||||
|
||||
Variant OpenXRFutureResult::get_result_value() const {
|
||||
return result_value;
|
||||
}
|
||||
|
||||
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);
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user