// Copyright 2018-2019, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Holds session related functions. * @author Jakob Bornecrantz * @ingroup oxr_main */ #include #include #include #include #include "util/u_debug.h" #include "util/u_time.h" #include "util/u_misc.h" #include "xrt/xrt_compiler.h" #include "xrt/xrt_prober.h" #include "oxr_objects.h" #include "oxr_logger.h" #include "oxr_handle.h" DEBUG_GET_ONCE_FLOAT_OPTION(lfov_left, "OXR_OVERRIDE_LFOV_LEFT", 0.0f) DEBUG_GET_ONCE_FLOAT_OPTION(lfov_right, "OXR_OVERRIDE_LFOV_RIGHT", 0.0f) DEBUG_GET_ONCE_FLOAT_OPTION(lfov_up, "OXR_OVERRIDE_LFOV_UP", 0.0f) DEBUG_GET_ONCE_FLOAT_OPTION(lfov_down, "OXR_OVERRIDE_LFOV_DOWN", 0.0f) static inline int32_t radtodeg_for_display(float radians) { return (int32_t)(radians * 180 * M_1_PI); } static inline void xdev_destroy(struct xrt_device **xdev_ptr) { struct xrt_device *xdev = *xdev_ptr; if (xdev == NULL) { return; } xdev->destroy(xdev); *xdev_ptr = NULL; } static XrResult oxr_instance_destroy(struct oxr_logger *log, struct oxr_handle_base *hb) { struct oxr_instance *inst = (struct oxr_instance *)hb; oxr_path_destroy_all(log, inst); if (inst->path_store != NULL) { u_hashset_destroy(&inst->path_store); } xdev_destroy(&inst->system.head); xdev_destroy(&inst->system.left); xdev_destroy(&inst->system.right); xrt_prober_destroy(&inst->prober); time_state_destroy(inst->timekeeping); inst->timekeeping = NULL; free(inst); return XR_SUCCESS; } static void cache_path(struct oxr_logger *log, struct oxr_instance *inst, const char *str, XrPath *out_path) { oxr_path_get_or_create(log, inst, str, strlen(str), out_path); } XrResult oxr_instance_create(struct oxr_logger *log, const XrInstanceCreateInfo *createInfo, struct oxr_instance **out_instance) { struct oxr_instance *inst = NULL; struct xrt_device *xdevs[3] = {0}; int h_ret, p_ret; OXR_ALLOCATE_HANDLE_OR_RETURN(log, inst, OXR_XR_DEBUG_INSTANCE, oxr_instance_destroy, NULL); h_ret = u_hashset_create(&inst->path_store); if (h_ret != 0) { free(inst); return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to create hashset"); } // Cache certain often looked up paths. cache_path(log, inst, "/user", &inst->path_cache.user); cache_path(log, inst, "/user/hand/head", &inst->path_cache.head); cache_path(log, inst, "/user/hand/left", &inst->path_cache.left); cache_path(log, inst, "/user/hand/right", &inst->path_cache.right); cache_path(log, inst, "/user/hand/gamepad", &inst->path_cache.gamepad); p_ret = xrt_prober_create(&inst->prober); if (p_ret != 0) { xrt_prober_destroy(&inst->prober); return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to create prober"); } p_ret = xrt_prober_probe(inst->prober); if (p_ret != 0) { xrt_prober_destroy(&inst->prober); return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to probe device(s)"); } p_ret = xrt_prober_select(inst->prober, xdevs, 3); if (p_ret != 0) { xrt_prober_destroy(&inst->prober); return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to select device"); } struct xrt_device *dev = xdevs[0]; const float left_override = debug_get_float_option_lfov_left(); if (left_override != 0.0f) { printf( "Overriding left eye angle_left with %f radians (%i°), " "and right eye angle_right with %f radians (%i°)\n", left_override, radtodeg_for_display(left_override), -left_override, radtodeg_for_display(-left_override)); dev->hmd->views[0].fov.angle_left = left_override; dev->hmd->views[1].fov.angle_right = -left_override; } const float right_override = debug_get_float_option_lfov_right(); if (right_override != 0.0f) { printf( "Overriding left eye angle_right with %f radians (%i°), " "and right eye angle_left with %f radians (%i°)\n", right_override, radtodeg_for_display(right_override), -right_override, radtodeg_for_display(-right_override)); dev->hmd->views[0].fov.angle_right = right_override; dev->hmd->views[1].fov.angle_left = -right_override; } const float up_override = debug_get_float_option_lfov_up(); if (up_override != 0.0f) { printf("Overriding both eyes angle_up with %f radians (%i°)\n", up_override, radtodeg_for_display(up_override)); dev->hmd->views[0].fov.angle_up = up_override; dev->hmd->views[1].fov.angle_up = up_override; } const float down_override = debug_get_float_option_lfov_down(); if (down_override != 0.0f) { printf( "Overriding both eyes angle_down with %f radians (%i°)\n", down_override, radtodeg_for_display(down_override)); dev->hmd->views[0].fov.angle_down = down_override; dev->hmd->views[1].fov.angle_down = down_override; } oxr_system_fill_in(log, inst, 1, &inst->system, xdevs[0], xdevs[1], xdevs[2]); inst->timekeeping = time_state_create(); inst->headless = false; inst->opengl_enable = false; inst->vulkan_enable = false; for (uint32_t i = 0; i < createInfo->enabledExtensionCount; ++i) { if (strcmp(createInfo->enabledExtensionNames[i], XR_KHR_HEADLESS_EXTENSION_NAME) == 0) { inst->headless = true; } else if (strcmp(createInfo->enabledExtensionNames[i], XR_KHR_OPENGL_ENABLE_EXTENSION_NAME) == 0) { inst->opengl_enable = true; } else if (strcmp(createInfo->enabledExtensionNames[i], XR_KHR_VULKAN_ENABLE_EXTENSION_NAME) == 0) { inst->vulkan_enable = true; } } //! @todo check if this (and other creates) failed? *out_instance = inst; return XR_SUCCESS; } XrResult oxr_instance_get_properties(struct oxr_logger *log, struct oxr_instance *inst, XrInstanceProperties *instanceProperties) { instanceProperties->runtimeVersion = XR_MAKE_VERSION(0, 0, 42); strncpy(instanceProperties->runtimeName, "Monado(XRT) by Collabora et al", XR_MAX_RUNTIME_NAME_SIZE - 1); return XR_SUCCESS; } #ifdef XR_USE_TIMESPEC XrResult oxr_instance_convert_time_to_timespec(struct oxr_logger *log, struct oxr_instance *inst, XrTime time, struct timespec *timespecTime) { time_state_to_timespec(inst->timekeeping, time, timespecTime); return XR_SUCCESS; } XrResult oxr_instance_convert_timespec_to_time(struct oxr_logger *log, struct oxr_instance *inst, const struct timespec *timespecTime, XrTime *time) { *time = time_state_from_timespec(inst->timekeeping, timespecTime); return XR_SUCCESS; } #endif // XR_USE_TIMESPEC