// Copyright 2018-2019, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Holds instance related entrypoints. * @author Jakob Bornecrantz * @ingroup oxr_api */ #include #include #include #include "xrt/xrt_compiler.h" #include "xrt/xrt_prober.h" #include "util/u_debug.h" #include "oxr_objects.h" #include "oxr_logger.h" #include "oxr_two_call.h" #include "oxr_api_funcs.h" #include "oxr_api_verify.h" static const XrExtensionProperties extension_properties[] = { #ifdef XR_USE_GRAPHICS_API_OPENGL {XR_TYPE_EXTENSION_PROPERTIES, NULL, XR_KHR_OPENGL_ENABLE_EXTENSION_NAME, XR_KHR_opengl_enable_SPEC_VERSION}, #endif #ifdef XR_USE_GRAPHICS_API_VULKAN {XR_TYPE_EXTENSION_PROPERTIES, NULL, XR_KHR_VULKAN_ENABLE_EXTENSION_NAME, XR_KHR_vulkan_enable_SPEC_VERSION}, #endif {XR_TYPE_EXTENSION_PROPERTIES, NULL, XR_KHR_HEADLESS_EXTENSION_NAME, XR_KHR_headless_SPEC_VERSION}, #ifdef XR_USE_TIMESPEC {XR_TYPE_EXTENSION_PROPERTIES, NULL, XR_KHR_CONVERT_TIMESPEC_TIME_EXTENSION_NAME, XR_KHR_convert_timespec_time_SPEC_VERSION}, #endif }; XrResult oxr_xrEnumerateInstanceExtensionProperties(const char* layerName, uint32_t propertyCapacityInput, uint32_t* propertyCountOutput, XrExtensionProperties* properties) { struct oxr_logger log; oxr_log_init(&log, "xrEnumerateInstanceExtensionProperties"); OXR_TWO_CALL_HELPER(&log, propertyCapacityInput, propertyCountOutput, properties, ARRAY_SIZE(extension_properties), extension_properties); } XrResult oxr_xrCreateInstance(const XrInstanceCreateInfo* createInfo, XrInstance* out_instance) { XrResult ret; struct oxr_logger log; oxr_log_init(&log, "xrCreateInstance"); OXR_VERIFY_ARG_TYPE_AND_NULL(&log, createInfo, XR_TYPE_INSTANCE_CREATE_INFO); const uint32_t major = XR_VERSION_MAJOR(XR_CURRENT_API_VERSION); const uint32_t minor = XR_VERSION_MINOR(XR_CURRENT_API_VERSION); #if 0 const uint32_t patch = XR_VERSION_PATCH(XR_CURRENT_API_VERSION); #endif if (createInfo->applicationInfo.apiVersion < XR_MAKE_VERSION(major, minor, 0)) { return oxr_error( &log, XR_ERROR_API_VERSION_UNSUPPORTED, "(createInfo->applicationInfo.apiVersion) " "Cannot satisfy request for version less than %d.%d.%d", major, minor, 0); } /* * This is a slight fib, to let us approximately run things between 1.0 * and 2.0 */ if (createInfo->applicationInfo.apiVersion >= XR_MAKE_VERSION(2, 0, 0)) { return oxr_error( &log, XR_ERROR_API_VERSION_UNSUPPORTED, "(createInfo->applicationInfo.apiVersion) " "Cannot satisfy request for version: too high"); } struct oxr_instance* inst; ret = oxr_instance_create(&log, createInfo, &inst); if (ret != XR_SUCCESS) { return ret; } *out_instance = oxr_instance_to_openxr(inst); return XR_SUCCESS; } XrResult oxr_xrDestroyInstance(XrInstance instance) { struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrDestroyInstance"); return oxr_handle_destroy(&log, &inst->handle); } XrResult oxr_xrGetInstanceProperties(XrInstance instance, XrInstanceProperties* instanceProperties) { struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrGetInstanceProperties"); return oxr_instance_get_properties(&log, inst, instanceProperties); } XrResult oxr_xrPollEvent(XrInstance instance, XrEventDataBuffer* eventData) { struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrPollEvent"); OXR_VERIFY_ARG_NOT_NULL(&log, eventData); return oxr_poll_event(&log, inst, eventData); } XrResult oxr_xrResultToString(XrInstance instance, XrResult value, char buffer[XR_MAX_RESULT_STRING_SIZE]) { struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrResultToString"); OXR_WARN_ONCE(&log, "fill in properly"); buffer[0] = '\0'; return XR_SUCCESS; } XrResult oxr_xrStructureTypeToString(XrInstance instance, XrStructureType value, char buffer[XR_MAX_STRUCTURE_NAME_SIZE]) { struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrStructureTypeToString"); OXR_WARN_ONCE(&log, "fill in properly"); buffer[0] = '\0'; return XR_SUCCESS; } XrResult oxr_xrStringToPath(XrInstance instance, const char* pathString, XrPath* out_path) { struct oxr_instance* inst; struct oxr_logger log; XrResult ret; XrPath path; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrStringToPath"); ret = oxr_verify_full_path_c(&log, pathString, "pathString"); if (ret != XR_SUCCESS) { return ret; } ret = oxr_path_get_or_create(&log, inst, pathString, strlen(pathString), &path); if (ret != XR_SUCCESS) { return ret; } *out_path = path; return XR_SUCCESS; } XrResult oxr_xrPathToString(XrInstance instance, XrPath path, uint32_t bufferCapacityInput, uint32_t* bufferCountOutput, char* buffer) { struct oxr_instance* inst; struct oxr_logger log; const char* str; size_t length; XrResult ret; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrPathToString"); if (path == XR_NULL_PATH) { return oxr_error(&log, XR_ERROR_PATH_INVALID, "(path == XR_NULL_PATH)"); } ret = oxr_path_get_string(&log, inst, path, &str, &length); if (ret != XR_SUCCESS) { return ret; } // Length is the number of valid characters, not including the null // termination character (but a extra null byte is always reserved). OXR_TWO_CALL_HELPER(&log, bufferCapacityInput, bufferCountOutput, buffer, length + 1, str); return XR_SUCCESS; } // ---- XR_KHR_convert_timespec_time extension #ifdef XR_USE_TIMESPEC XrResult oxr_xrConvertTimespecTimeToTimeKHR(XrInstance instance, const struct timespec* timespecTime, XrTime* time) { //! @todo do we need to check and see if this extension was enabled //! first? struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrConvertTimespecTimeToTimeKHR"); OXR_VERIFY_ARG_NOT_NULL(&log, timespecTime); OXR_VERIFY_ARG_NOT_NULL(&log, time); return oxr_instance_convert_timespec_to_time(&log, inst, timespecTime, time); } XrResult oxr_xrConvertTimeToTimespecTimeKHR(XrInstance instance, XrTime time, struct timespec* timespecTime) { struct oxr_instance* inst; struct oxr_logger log; OXR_VERIFY_INSTANCE_AND_INIT_LOG(&log, instance, inst, "xrConvertTimeToTimespecTimeKHR"); OXR_VERIFY_ARG_NOT_NULL(&log, timespecTime); return oxr_instance_convert_time_to_timespec(&log, inst, time, timespecTime); } #endif // XR_USE_TIMESPEC