1
0
mirror of https://github.com/godotengine/godot.git synced 2025-11-19 14:31:59 +00:00
Files
godot/modules/openxr/extensions/openxr_android_thread_settings_extension.cpp
2025-11-17 22:15:03 +00:00

150 lines
5.9 KiB
C++

/**************************************************************************/
/* openxr_android_thread_settings_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_android_thread_settings_extension.h"
#include "core/object/callable_method_pointer.h"
#include "core/string/print_string.h"
#include "servers/rendering/rendering_server.h"
#ifdef XR_USE_PLATFORM_ANDROID
#include "../openxr_api.h"
#include <unistd.h>
#endif
OpenXRAndroidThreadSettingsExtension *OpenXRAndroidThreadSettingsExtension::singleton = nullptr;
OpenXRAndroidThreadSettingsExtension *OpenXRAndroidThreadSettingsExtension::get_singleton() {
return singleton;
}
OpenXRAndroidThreadSettingsExtension::OpenXRAndroidThreadSettingsExtension() {
singleton = this;
}
OpenXRAndroidThreadSettingsExtension::~OpenXRAndroidThreadSettingsExtension() {
singleton = nullptr;
}
void OpenXRAndroidThreadSettingsExtension::_bind_methods() {
ClassDB::bind_method(D_METHOD("set_application_thread_type", "thread_type", "thread_id"), &OpenXRAndroidThreadSettingsExtension::set_application_thread_type, DEFVAL(0));
BIND_ENUM_CONSTANT(THREAD_TYPE_APPLICATION_MAIN);
BIND_ENUM_CONSTANT(THREAD_TYPE_APPLICATION_WORKER);
BIND_ENUM_CONSTANT(THREAD_TYPE_RENDERER_MAIN);
BIND_ENUM_CONSTANT(THREAD_TYPE_RENDERER_WORKER);
}
HashMap<String, bool *> OpenXRAndroidThreadSettingsExtension::get_requested_extensions() {
HashMap<String, bool *> request_extensions;
#ifdef XR_USE_PLATFORM_ANDROID
request_extensions[XR_KHR_ANDROID_THREAD_SETTINGS_EXTENSION_NAME] = &available;
#endif
return request_extensions;
}
void OpenXRAndroidThreadSettingsExtension::on_instance_created(XrInstance p_instance) {
if (!available) {
return;
}
if (!_initialize_openxr_android_thread_settings_extension()) {
print_error("OpenXR: Failed to initialize android thread settings extension");
available = false;
}
}
void OpenXRAndroidThreadSettingsExtension::on_session_created(XrSession p_session) {
if (!available) {
return;
}
// Attempt to mark this thread as the "main thread".
set_application_thread_type(THREAD_TYPE_APPLICATION_MAIN);
// Attempt to mark the render thread too.
RenderingServer *rendering_server = RenderingServer::get_singleton();
ERR_FAIL_NULL(rendering_server);
rendering_server->call_on_render_thread(callable_mp(this, &OpenXRAndroidThreadSettingsExtension::_set_render_thread_type));
}
bool OpenXRAndroidThreadSettingsExtension::set_application_thread_type(ThreadType p_thread_type, uint32_t p_thread_id) {
if (!available) {
return false;
}
#ifdef XR_USE_PLATFORM_ANDROID
XrAndroidThreadTypeKHR thread_type{};
switch (p_thread_type) {
case THREAD_TYPE_APPLICATION_MAIN:
thread_type = XR_ANDROID_THREAD_TYPE_APPLICATION_MAIN_KHR;
break;
case THREAD_TYPE_APPLICATION_WORKER:
thread_type = XR_ANDROID_THREAD_TYPE_APPLICATION_WORKER_KHR;
break;
case THREAD_TYPE_RENDERER_MAIN:
thread_type = XR_ANDROID_THREAD_TYPE_RENDERER_MAIN_KHR;
break;
case THREAD_TYPE_RENDERER_WORKER:
thread_type = XR_ANDROID_THREAD_TYPE_RENDERER_WORKER_KHR;
break;
default:
print_error(vformat("OpenXR: Failed to set android application thread; invalid thread type %d", p_thread_type));
return false;
}
XrResult result = xrSetAndroidApplicationThreadKHR(OpenXRAPI::get_singleton()->get_session(), thread_type, p_thread_id == 0 ? gettid() : p_thread_id);
if (result != XR_SUCCESS) {
print_error(vformat("OpenXR: Failed to set android application thread; %s", OpenXRAPI::get_singleton()->get_error_string(result)));
return false;
}
#endif
return true;
}
bool OpenXRAndroidThreadSettingsExtension::_initialize_openxr_android_thread_settings_extension() {
#ifdef XR_USE_PLATFORM_ANDROID
EXT_INIT_XR_FUNC_V(xrSetAndroidApplicationThreadKHR);
#endif
return true;
}
void OpenXRAndroidThreadSettingsExtension::_set_render_thread_type() {
// Skip when the render thread == the main thread
if (Thread::is_main_thread()) {
return;
}
set_application_thread_type(THREAD_TYPE_RENDERER_MAIN);
}