Files
godot/thirdparty/openxr/src/loader/loader_init_data.cpp
2025-09-22 21:25:45 +10:00

134 lines
5.8 KiB
C++

// Copyright (c) 2017-2025 The Khronos Group Inc.
// Copyright (c) 2017-2019 Valve Corporation
// Copyright (c) 2017-2019 LunarG, Inc.
//
// SPDX-License-Identifier: Apache-2.0 OR MIT
//
// Initial Author: Mark Young <marky@lunarg.com>
//
#include "loader_logger.hpp"
#include "runtime_interface.hpp"
#include "loader_instance.hpp"
#include "loader_init_data.hpp"
#include "loader_properties.hpp"
XrResult LoaderInitData::initialize(const XrLoaderInitInfoBaseHeaderKHR* info) {
// We iterate the chain per struct type, so we only pick the first of each type in the chain.
XrResult result = initializeProperties(info);
if (result != XR_SUCCESS) {
return result;
}
#if defined(XR_HAS_REQUIRED_PLATFORM_LOADER_INIT_STRUCT)
result = initializePlatform(info);
if (result != XR_SUCCESS) {
return result;
}
#endif // defined(XR_HAS_REQUIRED_PLATFORM_LOADER_INIT_STRUCT)
_initialized = true;
return XR_SUCCESS;
}
XrResult LoaderInitData::initializeProperties(const XrLoaderInitInfoBaseHeaderKHR* info) {
while (info != nullptr) {
if (info->type == XR_TYPE_LOADER_INIT_INFO_PROPERTIES_EXT) {
const auto* propertyInfo = reinterpret_cast<XrLoaderInitInfoPropertiesEXT const*>(info);
// Validate the inputs first.
for (uint32_t i = 0; i < propertyInfo->propertyValueCount; i++) {
if (propertyInfo->propertyValues[i].name == nullptr) {
return XR_ERROR_VALIDATION_FAILURE;
}
if (propertyInfo->propertyValues[i].value == nullptr) {
return XR_ERROR_VALIDATION_FAILURE;
}
std::string view{propertyInfo->propertyValues[i].name};
if (view.size() == 0) {
return XR_ERROR_VALIDATION_FAILURE;
}
}
// Inject provided properties into the loader property store.
LoaderProperty::ClearOverrides();
for (uint32_t i = 0; i < propertyInfo->propertyValueCount; i++) {
LoaderProperty::SetOverride(propertyInfo->propertyValues[i].name, propertyInfo->propertyValues[i].value);
}
// Take only the first such struct.
return XR_SUCCESS;
}
info = reinterpret_cast<const XrLoaderInitInfoBaseHeaderKHR*>(info->next);
}
// fine if we don't find this.
return XR_SUCCESS;
}
#if defined(XR_USE_PLATFORM_ANDROID) && defined(XR_HAS_REQUIRED_PLATFORM_LOADER_INIT_STRUCT)
XrResult LoaderInitData::initializePlatform(const XrLoaderInitInfoBaseHeaderKHR* info) {
// Check and copy the Android-specific init data.
while (info != nullptr) {
if (info->type == XR_TYPE_LOADER_INIT_INFO_ANDROID_KHR) {
auto cast_info = reinterpret_cast<XrLoaderInitInfoAndroidKHR const*>(info);
if (cast_info->applicationVM == nullptr) {
return XR_ERROR_VALIDATION_FAILURE;
}
if (cast_info->applicationContext == nullptr) {
return XR_ERROR_VALIDATION_FAILURE;
}
// Copy and store the JVM pointer and Android Context, ensuring the JVM is initialised.
_platform_info = *cast_info;
_platform_info.next = nullptr; // Not safe to store next pointer since the memory may not exist later.
if (_platform_info.applicationVM == nullptr) {
return XR_ERROR_VALIDATION_FAILURE;
}
if (_platform_info.applicationContext == nullptr) {
return XR_ERROR_VALIDATION_FAILURE;
}
jni::init(static_cast<jni::JavaVM*>(_platform_info.applicationVM));
const jni::Object context = jni::Object{static_cast<jni::jobject>(_platform_info.applicationContext)};
// Retrieve a reference to the Android AssetManager.
const auto assetManager = context.call<jni::Object>("getAssets()Landroid/content/res/AssetManager;");
_android_asset_manager = AAssetManager_fromJava(jni::env(), assetManager.getHandle());
// Retrieve the path to the native libraries.
const auto applicationContext = context.call<jni::Object>("getApplicationContext()Landroid/content/Context;");
const auto applicationInfo = context.call<jni::Object>("getApplicationInfo()Landroid/content/pm/ApplicationInfo;");
_android_native_library_path = applicationInfo.get<std::string>("nativeLibraryDir");
// Take only the first such struct.
return XR_SUCCESS;
}
info = reinterpret_cast<const XrLoaderInitInfoBaseHeaderKHR*>(info->next);
}
// We didn't find one.
return XR_ERROR_VALIDATION_FAILURE;
}
#endif // defined(XR_USE_PLATFORM_ANDROID) && defined(XR_HAS_REQUIRED_PLATFORM_LOADER_INIT_STRUCT)
XrResult InitializeLoaderInitData(const XrLoaderInitInfoBaseHeaderKHR* loaderInitInfo) {
if (!ActiveLoaderInstance::IsAvailable()) {
LoaderLogger::LogVerboseMessage("InitializeLoaderInitData", "Unloading any previously loaded runtime");
// This will not shutdown the runtime, only unload the library.
RuntimeInterface::UnloadRuntime("InitializeLoaderInitData");
} else {
LoaderLogger::LogErrorMessage("InitializeLoaderInitData",
"An active instance currently exists while trying to reinitialize the loader");
return XR_ERROR_INITIALIZATION_FAILED;
}
return LoaderInitData::instance().initialize(loaderInitInfo);
}
#if defined(XR_USE_PLATFORM_ANDROID) && defined(XR_HAS_REQUIRED_PLATFORM_LOADER_INIT_STRUCT)
std::string GetAndroidNativeLibraryDir() { return LoaderInitData::instance()._android_native_library_path; }
void* GetAndroidAssetManager() { return LoaderInitData::instance()._android_asset_manager; }
#endif // defined(XR_USE_PLATFORM_ANDROID) && defined(XR_HAS_REQUIRED_PLATFORM_LOADER_INIT_STRUCT)