// Copyright 2018-2019, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief File for negotiating with the loader. * @author Ryan Pavlik * @author Jakob Bornecrantz * @ingroup oxr_api */ #include #include #include "xrt/xrt_compiler.h" #include "util/u_debug.h" #include "oxr_objects.h" #include "oxr_logger.h" #include "oxr_api_funcs.h" #include "oxr_api_verify.h" DEBUG_GET_ONCE_BOOL_OPTION(negotiate, "OXR_DEBUG_NEGOTIATE", false) #define PRINT_NEGOTIATE(...) \ do { \ if (debug_get_bool_option_negotiate()) { \ fprintf(stderr, __VA_ARGS__); \ } \ } while (false) #ifdef _WIN32 __declspec(dllexport) XRAPI_ATTR XrResult XRAPI_CALL xrNegotiateLoaderRuntimeInterface( const XrNegotiateLoaderInfo *loaderInfo, XrNegotiateRuntimeRequest *runtimeRequest); #endif XRAPI_ATTR XrResult XRAPI_CALL xrNegotiateLoaderRuntimeInterface(const XrNegotiateLoaderInfo *loaderInfo, XrNegotiateRuntimeRequest *runtimeRequest) { PRINT_NEGOTIATE("xrNegotiateLoaderRuntimeInterface\n"); // Make sure that we understand the structs passed to this function. if (loaderInfo->structType != XR_LOADER_INTERFACE_STRUCT_LOADER_INFO || loaderInfo->structVersion != XR_LOADER_INFO_STRUCT_VERSION || loaderInfo->structSize != sizeof(XrNegotiateLoaderInfo)) { PRINT_NEGOTIATE("\tloaderInfo bad!\n"); return XR_ERROR_INITIALIZATION_FAILED; } // Make sure that we understand the structs passed to this function. if (runtimeRequest->structType != XR_LOADER_INTERFACE_STRUCT_RUNTIME_REQUEST || runtimeRequest->structVersion != XR_CURRENT_LOADER_RUNTIME_VERSION || runtimeRequest->structSize != sizeof(XrNegotiateRuntimeRequest)) { PRINT_NEGOTIATE("\truntimeRequest bad!\n"); return XR_ERROR_INITIALIZATION_FAILED; } // TODO: properly define what we support uint16_t supported_major = XR_VERSION_MAJOR(XR_CURRENT_API_VERSION); uint32_t requested_min_major = loaderInfo->minInterfaceVersion; uint32_t requested_max_major = loaderInfo->maxInterfaceVersion; if (supported_major > requested_max_major || supported_major < requested_min_major) { PRINT_NEGOTIATE( "\tXRT - OpenXR doesn't support requested version %d <= " "%d <= %d\n", requested_min_major, supported_major, requested_max_major); return XR_ERROR_INITIALIZATION_FAILED; } runtimeRequest->getInstanceProcAddr = oxr_xrGetInstanceProcAddr; runtimeRequest->runtimeInterfaceVersion = XR_CURRENT_LOADER_RUNTIME_VERSION; runtimeRequest->runtimeApiVersion = XR_CURRENT_API_VERSION; PRINT_NEGOTIATE("\tall ok!\n"); return XR_SUCCESS; } XrResult oxr_xrEnumerateApiLayerProperties(uint32_t propertyCapacityInput, uint32_t *propertyCountOutput, XrApiLayerProperties *properties) { struct oxr_logger log; oxr_log_init(&log, "xrEnumerateApiLayerProperties"); /* We have no layers inbuilt. */ if (propertyCountOutput != NULL) { *propertyCountOutput = 0; } return XR_SUCCESS; } /*! * @brief Helper define for generating that GetInstanceProcAddr function. * * Use for functions that should be unconditionally available. */ #define ENTRY(funcName) \ do { \ if (strcmp(name, #funcName) == 0) { \ PFN_##funcName ret = &oxr_##funcName; \ *out_function = (PFN_xrVoidFunction)(ret); \ return XR_SUCCESS; \ } \ } while (false) /*! * @brief Helper define for generating that GetInstanceProcAddr function for * conditionally-available functions. * * Checks the extra condition to e.g. find out if the extension is enabled */ #define ENTRY_IF(funcName, extraCondition, message) \ do { \ if (strcmp(name, #funcName) == 0) { \ if (extraCondition) { \ PFN_##funcName ret = &oxr_##funcName; \ *out_function = (PFN_xrVoidFunction)(ret); \ return XR_SUCCESS; \ } \ return XR_ERROR_FUNCTION_UNSUPPORTED; \ } \ } while (false) /*! * @brief Helper define for generating that GetInstanceProcAddr function for * extension-provided functions. * * Wraps ENTRY_IF for the common case. * * Pass the function name and the (mixed-case) extension name without the * leading XR_. */ #define ENTRY_IF_EXT(funcName, short_ext_name) \ ENTRY_IF(funcName, inst->extensions.short_ext_name, \ "Required extension XR_" #short_ext_name " not enabled") /*! * Handle a non-null instance pointer. */ static XrResult handle_non_null(struct oxr_instance *inst, struct oxr_logger *log, const char *name, PFN_xrVoidFunction *out_function) { ENTRY(xrGetInstanceProcAddr); ENTRY(xrEnumerateInstanceExtensionProperties); ENTRY(xrCreateInstance); ENTRY(xrDestroyInstance); ENTRY(xrGetInstanceProperties); ENTRY(xrPollEvent); ENTRY(xrResultToString); ENTRY(xrStructureTypeToString); ENTRY(xrGetSystem); ENTRY(xrGetSystemProperties); ENTRY(xrEnumerateEnvironmentBlendModes); ENTRY(xrCreateSession); ENTRY(xrDestroySession); ENTRY(xrEnumerateReferenceSpaces); ENTRY(xrCreateReferenceSpace); ENTRY(xrGetReferenceSpaceBoundsRect); ENTRY(xrCreateActionSpace); ENTRY(xrLocateSpace); ENTRY(xrDestroySpace); ENTRY(xrEnumerateViewConfigurations); ENTRY(xrGetViewConfigurationProperties); ENTRY(xrEnumerateViewConfigurationViews); ENTRY(xrEnumerateSwapchainFormats); ENTRY(xrCreateSwapchain); ENTRY(xrDestroySwapchain); ENTRY(xrEnumerateSwapchainImages); ENTRY(xrAcquireSwapchainImage); ENTRY(xrWaitSwapchainImage); ENTRY(xrReleaseSwapchainImage); ENTRY(xrBeginSession); ENTRY(xrEndSession); ENTRY(xrWaitFrame); ENTRY(xrBeginFrame); ENTRY(xrEndFrame); ENTRY(xrRequestExitSession); ENTRY(xrLocateViews); ENTRY(xrStringToPath); ENTRY(xrPathToString); ENTRY(xrCreateActionSet); ENTRY(xrDestroyActionSet); ENTRY(xrCreateAction); ENTRY(xrDestroyAction); ENTRY(xrSuggestInteractionProfileBindings); ENTRY(xrAttachSessionActionSets); ENTRY(xrGetCurrentInteractionProfile); ENTRY(xrGetActionStateBoolean); ENTRY(xrGetActionStateFloat); ENTRY(xrGetActionStateVector2f); ENTRY(xrGetActionStatePose); ENTRY(xrSyncActions); ENTRY(xrEnumerateBoundSourcesForAction); ENTRY(xrGetInputSourceLocalizedName); ENTRY(xrApplyHapticFeedback); ENTRY(xrStopHapticFeedback); #ifdef OXR_HAVE_KHR_visibility_mask ENTRY_IF_EXT(xrGetVisibilityMaskKHR, KHR_visibility_mask); #endif // OXR_HAVE_KHR_visibility_mask #ifdef OXR_HAVE_KHR_convert_timespec_time ENTRY_IF_EXT(xrConvertTimespecTimeToTimeKHR, KHR_convert_timespec_time); ENTRY_IF_EXT(xrConvertTimeToTimespecTimeKHR, KHR_convert_timespec_time); #endif // OXR_HAVE_KHR_convert_timespec_time #ifdef OXR_HAVE_EXT_performance_settings ENTRY_IF_EXT(xrPerfSettingsSetPerformanceLevelEXT, EXT_performance_settings); #endif // OXR_HAVE_EXT_performance_settings #ifdef OXR_HAVE_EXT_thermal_query ENTRY_IF_EXT(xrThermalGetTemperatureTrendEXT, EXT_thermal_query); #endif // OXR_HAVE_EXT_thermal_query ENTRY_IF_EXT(xrCreateHandTrackerEXT, EXT_hand_tracking); ENTRY_IF_EXT(xrDestroyHandTrackerEXT, EXT_hand_tracking); ENTRY_IF_EXT(xrLocateHandJointsEXT, EXT_hand_tracking); #if 0 #ifdef OXR_HAVE_EXT_debug_utils ENTRY_IF_EXT(xrSetDebugUtilsObjectNameEXT, EXT_debug_utils); ENTRY_IF_EXT(xrCreateDebugUtilsMessengerEXT, EXT_debug_utils); ENTRY_IF_EXT(xrDestroyDebugUtilsMessengerEXT, EXT_debug_utils); ENTRY_IF_EXT(xrSubmitDebugUtilsMessageEXT, EXT_debug_utils); ENTRY_IF_EXT(xrSessionBeginDebugUtilsLabelRegionEXT, EXT_debug_utils); ENTRY_IF_EXT(xrSessionEndDebugUtilsLabelRegionEXT, EXT_debug_utils); ENTRY_IF_EXT(xrSessionInsertDebugUtilsLabelEXT, EXT_debug_utils); #endif // OXR_HAVE_EXT_debug_utils #endif #ifdef OXR_HAVE_KHR_opengl_enable ENTRY_IF_EXT(xrGetOpenGLGraphicsRequirementsKHR, KHR_opengl_enable); #endif // OXR_HAVE_KHR_opengl_enable #ifdef OXR_HAVE_KHR_opengl_es_enable ENTRY_IF_EXT(xrGetOpenGLESGraphicsRequirementsKHR, KHR_opengl_es_enable); #endif // OXR_HAVE_KHR_opengl_es_enable #ifdef OXR_HAVE_KHR_vulkan_enable ENTRY_IF_EXT(xrGetVulkanInstanceExtensionsKHR, KHR_vulkan_enable); ENTRY_IF_EXT(xrGetVulkanDeviceExtensionsKHR, KHR_vulkan_enable); ENTRY_IF_EXT(xrGetVulkanGraphicsDeviceKHR, KHR_vulkan_enable); ENTRY_IF_EXT(xrGetVulkanGraphicsRequirementsKHR, KHR_vulkan_enable); #endif // OXR_HAVE_KHR_vulkan_enable /* * Not logging here because there's no need to loudly advertise * which extensions the loader knows about (it calls this on * every known function) that we don't implement. */ return XR_ERROR_FUNCTION_UNSUPPORTED; } /*! * Special case a null instance pointer. */ static XrResult handle_null(struct oxr_logger *log, const char *name, PFN_xrVoidFunction *out_function) { ENTRY(xrCreateInstance); ENTRY(xrEnumerateInstanceExtensionProperties); ENTRY(xrEnumerateApiLayerProperties); /* * This is fine to log, since there should not be other * null-instance calls. */ return oxr_error(log, XR_ERROR_FUNCTION_UNSUPPORTED, "(name = \"%s\")", name); } XrResult oxr_xrGetInstanceProcAddr(XrInstance instance, const char *name, PFN_xrVoidFunction *function) { struct oxr_logger log; // We need to set this unconditionally, per the spec. *function = NULL; if (instance == XR_NULL_HANDLE) { oxr_log_init(&log, "xrGetInstanceProcAddr"); return handle_null(&log, name, function); } struct oxr_instance *inst; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrGetInstanceProcAddr"); return handle_non_null(inst, &log, name, function); }