// Copyright 2018-2020, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Contains the instance struct that a lot of things hang from. * @author Jakob Bornecrantz * @ingroup oxr_main */ #pragma once #include "xrt/xrt_device.h" #include "xrt/xrt_tracking.h" #include "xrt/xrt_compositor.h" #include "xrt/xrt_vulkan_includes.h" #include "xrt/xrt_openxr_includes.h" #include "os/os_threading.h" #include "util/u_index_fifo.h" #include "util/u_hashset.h" #include "util/u_hashmap.h" #include "util/u_device.h" #include "oxr_extension_support.h" #include "oxr_subaction.h" #ifdef XRT_FEATURE_RENDERDOC #include "renderdoc_app.h" #include #endif #ifdef __cplusplus extern "C" { #endif /*! * @defgroup oxr OpenXR state tracker * * Client application facing code. * * @ingroup xrt */ /*! * @brief Cast a pointer to an OpenXR handle in such a way as to avoid warnings. * * Avoids -Wpointer-to-int-cast by first casting to the same size int, then * promoting to the 64-bit int, then casting to the handle type. That's a lot of * no-ops on 64-bit, but a widening conversion on 32-bit. * * @ingroup oxr */ #define XRT_CAST_PTR_TO_OXR_HANDLE(HANDLE_TYPE, PTR) ((HANDLE_TYPE)(uint64_t)(uintptr_t)(PTR)) /*! * @brief Cast an OpenXR handle to a pointer in such a way as to avoid warnings. * * Avoids -Wint-to-pointer-cast by first casting to a 64-bit int, then to a * pointer-sized int, then to the desired pointer type. That's a lot of no-ops * on 64-bit, but a narrowing (!) conversion on 32-bit. * * @ingroup oxr */ #define XRT_CAST_OXR_HANDLE_TO_PTR(PTR_TYPE, HANDLE) ((PTR_TYPE)(uintptr_t)(uint64_t)(HANDLE)) /*! * @defgroup oxr_main OpenXR main code * * Gets called from @ref oxr_api functions and talks to devices and * @ref comp using @ref xrt_iface. * * @ingroup oxr * @{ */ // For corruption and layer checking. // clang-format off #define OXR_XR_DEBUG_INSTANCE (*(uint64_t *)"oxrinst\0") #define OXR_XR_DEBUG_SESSION (*(uint64_t *)"oxrsess\0") #define OXR_XR_DEBUG_SPACE (*(uint64_t *)"oxrspac\0") #define OXR_XR_DEBUG_PATH (*(uint64_t *)"oxrpath\0") #define OXR_XR_DEBUG_ACTION (*(uint64_t *)"oxracti\0") #define OXR_XR_DEBUG_SWAPCHAIN (*(uint64_t *)"oxrswap\0") #define OXR_XR_DEBUG_ACTIONSET (*(uint64_t *)"oxraset\0") #define OXR_XR_DEBUG_MESSENGER (*(uint64_t *)"oxrmess\0") #define OXR_XR_DEBUG_SOURCESET (*(uint64_t *)"oxrsrcs\0") #define OXR_XR_DEBUG_SOURCE (*(uint64_t *)"oxrsrc_\0") #define OXR_XR_DEBUG_HTRACKER (*(uint64_t *)"oxrhtra\0") // clang-format on /* * * Forward declare structs. * */ struct xrt_instance; struct oxr_logger; struct oxr_extension_status; struct oxr_instance; struct oxr_system; struct oxr_session; struct oxr_event; struct oxr_swapchain; struct oxr_space; struct oxr_action_set; struct oxr_action; struct oxr_debug_messenger; struct oxr_handle_base; struct oxr_subaction_paths; struct oxr_action_attachment; struct oxr_action_set_attachment; struct oxr_action_input; struct oxr_action_output; struct oxr_binding; struct oxr_interaction_profile; struct oxr_action_set_ref; struct oxr_action_ref; struct oxr_hand_tracker; #define XRT_MAX_HANDLE_CHILDREN 256 #define OXR_MAX_SWAPCHAIN_IMAGES 8 #define OXR_MAX_BINDINGS_PER_ACTION 16 struct time_state; /*! * Function pointer type for a handle destruction function. * * @relates oxr_handle_base */ typedef XrResult (*oxr_handle_destroyer)(struct oxr_logger *log, struct oxr_handle_base *hb); /*! * State of a handle base, to reduce likelihood of going "boom" on * out-of-order destruction or other unsavory behavior. */ enum oxr_handle_state { /*! State during/before oxr_handle_init, or after failure */ OXR_HANDLE_STATE_UNINITIALIZED = 0, /*! State after successful oxr_handle_init */ OXR_HANDLE_STATE_LIVE, /*! State after successful oxr_handle_destroy */ OXR_HANDLE_STATE_DESTROYED, }; /*! * Sub action paths. */ enum oxr_subaction_path { OXR_SUB_ACTION_PATH_USER, OXR_SUB_ACTION_PATH_HEAD, OXR_SUB_ACTION_PATH_LEFT, OXR_SUB_ACTION_PATH_RIGHT, OXR_SUB_ACTION_PATH_GAMEPAD, }; /*! * Tracks the state of a image that belongs to a @ref oxr_swapchain. */ enum oxr_image_state { OXR_IMAGE_STATE_READY, OXR_IMAGE_STATE_ACQUIRED, OXR_IMAGE_STATE_WAITED, }; /* * * oxr_handle_base.c * */ /*! * Destroy the handle's object, as well as all child handles recursively. * * This should be how all handle-associated objects are destroyed. * * @public @memberof oxr_handle_base */ XrResult oxr_handle_destroy(struct oxr_logger *log, struct oxr_handle_base *hb); /*! * Returns a human-readable label for a handle state. * * @relates oxr_handle_base */ const char * oxr_handle_state_to_string(enum oxr_handle_state state); /*! * * @name oxr_instance.c * @{ * */ /*! * To go back to a OpenXR object. * * @relates oxr_instance */ static inline XrInstance oxr_instance_to_openxr(struct oxr_instance *inst) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrInstance, inst); } /*! * Creates a instance, does minimal validation of @p createInfo. * * @param[in] log Logger * @param[in] createInfo OpenXR creation info. * @param[in] extensions Parsed extension list to be enabled. * @param[out] out_inst Pointer to pointer to a instance, returned instance. * * @public @static @memberof oxr_instance */ XrResult oxr_instance_create(struct oxr_logger *log, const XrInstanceCreateInfo *createInfo, const struct oxr_extension_status *extensions, struct oxr_instance **out_inst); /*! * @public @memberof oxr_instance */ XrResult oxr_instance_get_properties(struct oxr_logger *log, struct oxr_instance *inst, XrInstanceProperties *instanceProperties); #if XR_USE_TIMESPEC /*! * @public @memberof oxr_instance */ XrResult oxr_instance_convert_time_to_timespec(struct oxr_logger *log, struct oxr_instance *inst, XrTime time, struct timespec *timespecTime); /*! * @public @memberof oxr_instance */ XrResult oxr_instance_convert_timespec_to_time(struct oxr_logger *log, struct oxr_instance *inst, const struct timespec *timespecTime, XrTime *time); #endif // XR_USE_TIMESPEC /*! * @} */ /*! * * @name oxr_path.c * @{ * */ /*! * Initialize the path system. * @private @memberof oxr_instance */ XrResult oxr_path_init(struct oxr_logger *log, struct oxr_instance *inst); /*! * @public @memberof oxr_instance */ bool oxr_path_is_valid(struct oxr_logger *log, struct oxr_instance *inst, XrPath path); /*! * @public @memberof oxr_instance */ void * oxr_path_get_attached(struct oxr_logger *log, struct oxr_instance *inst, XrPath path); /*! * Get the path for the given string if it exists, or create it if it does not. * * @public @memberof oxr_instance */ XrResult oxr_path_get_or_create( struct oxr_logger *log, struct oxr_instance *inst, const char *str, size_t length, XrPath *out_path); /*! * Only get the path for the given string if it exists. * * @public @memberof oxr_instance */ XrResult oxr_path_only_get(struct oxr_logger *log, struct oxr_instance *inst, const char *str, size_t length, XrPath *out_path); /*! * Get a pointer and length of the internal string. * * The pointer has the same life time as the instance. The length is the number * of valid characters, not including the null termination character (but an * extra null byte is always reserved at the end so can strings can be given * to functions expecting null terminated strings). * * @public @memberof oxr_instance */ XrResult oxr_path_get_string( struct oxr_logger *log, struct oxr_instance *inst, XrPath path, const char **out_str, size_t *out_length); /*! * Destroy the path system and all paths that the instance has created. * * @private @memberof oxr_instance */ void oxr_path_destroy(struct oxr_logger *log, struct oxr_instance *inst); /*! * @} */ /*! * To go back to a OpenXR object. * * @relates oxr_action_set */ static inline XrActionSet oxr_action_set_to_openxr(struct oxr_action_set *act_set) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrActionSet, act_set); } /*! * To go back to a OpenXR object. * * @relates oxr_hand_tracker */ static inline XrHandTrackerEXT oxr_hand_tracker_to_openxr(struct oxr_hand_tracker *hand_tracker) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrHandTrackerEXT, hand_tracker); } /*! * To go back to a OpenXR object. * * @relates oxr_action */ static inline XrAction oxr_action_to_openxr(struct oxr_action *act) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrAction, act); } /*! * * @name oxr_input.c * @{ * */ /*! * Helper function to classify subaction_paths. * * Sets all members of @p subaction_paths ( @ref oxr_subaction_paths ) as * appropriate based on the subaction paths found in the list. * * If no paths are provided, @p sub_paths->any will be true. * * @return false if an invalid subaction path is provided. * * @public @memberof oxr_instance * @see oxr_sub_paths */ bool oxr_classify_sub_action_paths(struct oxr_logger *log, struct oxr_instance *inst, uint32_t subaction_path_count, const XrPath *subaction_paths, struct oxr_subaction_paths *subaction_paths_out); /*! * Find the pose input for the set of subaction_paths * * @public @memberof oxr_session */ XrResult oxr_action_get_pose_input(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, const struct oxr_subaction_paths *subaction_paths_ptr, struct oxr_action_input **out_input); /*! * @public @memberof oxr_instance */ XrResult oxr_action_set_create(struct oxr_logger *log, struct oxr_instance *inst, const XrActionSetCreateInfo *createInfo, struct oxr_action_set **out_act_set); /*! * @public @memberof oxr_action */ XrResult oxr_action_create(struct oxr_logger *log, struct oxr_action_set *act_set, const XrActionCreateInfo *createInfo, struct oxr_action **out_act); /*! * @public @memberof oxr_session * @see oxr_action_set */ XrResult oxr_session_attach_action_sets(struct oxr_logger *log, struct oxr_session *sess, const XrSessionActionSetsAttachInfo *bindInfo); /*! * @public @memberof oxr_session */ XrResult oxr_action_sync_data(struct oxr_logger *log, struct oxr_session *sess, uint32_t countActionSets, const XrActiveActionSet *actionSets); /*! * @public @memberof oxr_session */ XrResult oxr_action_enumerate_bound_sources(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, uint32_t sourceCapacityInput, uint32_t *sourceCountOutput, XrPath *sources); /*! * @public @memberof oxr_session */ XrResult oxr_action_get_boolean(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, struct oxr_subaction_paths subaction_paths, XrActionStateBoolean *data); /*! * @public @memberof oxr_session */ XrResult oxr_action_get_vector1f(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, struct oxr_subaction_paths subaction_paths, XrActionStateFloat *data); /*! * @public @memberof oxr_session */ XrResult oxr_action_get_vector2f(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, struct oxr_subaction_paths subaction_paths, XrActionStateVector2f *data); /*! * @public @memberof oxr_session */ XrResult oxr_action_get_pose(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, struct oxr_subaction_paths subaction_paths, XrActionStatePose *data); /*! * @public @memberof oxr_session */ XrResult oxr_action_apply_haptic_feedback(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, struct oxr_subaction_paths subaction_paths, const XrHapticBaseHeader *hapticEvent); /*! * @public @memberof oxr_session */ XrResult oxr_action_stop_haptic_feedback(struct oxr_logger *log, struct oxr_session *sess, uint32_t act_key, struct oxr_subaction_paths subaction_paths); /*! * @public @memberof oxr_instance */ XrResult oxr_hand_tracker_create(struct oxr_logger *log, struct oxr_session *sess, const XrHandTrackerCreateInfoEXT *createInfo, struct oxr_hand_tracker **out_hand_tracker); /*! * @} */ /*! * * @name oxr_binding.c * @{ * */ /*! * Find the best matching profile for the given @ref xrt_device. * * @param log Logger. * @param inst Instance. * @param xdev Can be null. * @param[out] out_p Returned interaction profile. * * @public @memberof oxr_instance */ void oxr_find_profile_for_device(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, struct oxr_interaction_profile **out_p); /*! * Free all memory allocated by the binding system. * * @public @memberof oxr_instance */ void oxr_binding_destroy_all(struct oxr_logger *log, struct oxr_instance *inst); /*! * Find all bindings that is the given action key is bound to. * @public @memberof oxr_interaction_profile */ void oxr_binding_find_bindings_from_key(struct oxr_logger *log, struct oxr_interaction_profile *profile, uint32_t key, struct oxr_binding *bindings[OXR_MAX_BINDINGS_PER_ACTION], size_t *binding_count); /*! * @public @memberof oxr_instance */ XrResult oxr_action_suggest_interaction_profile_bindings(struct oxr_logger *log, struct oxr_instance *inst, const XrInteractionProfileSuggestedBinding *suggestedBindings); /*! * @public @memberof oxr_instance */ XrResult oxr_action_get_current_interaction_profile(struct oxr_logger *log, struct oxr_session *sess, XrPath topLevelUserPath, XrInteractionProfileState *interactionProfile); /*! * @public @memberof oxr_session */ XrResult oxr_action_get_input_source_localized_name(struct oxr_logger *log, struct oxr_session *sess, const XrInputSourceLocalizedNameGetInfo *getInfo, uint32_t bufferCapacityInput, uint32_t *bufferCountOutput, char *buffer); /*! * @} */ /*! * * @name oxr_session.c * @{ * */ /*! * To go back to a OpenXR object. * * @relates oxr_session */ static inline XrSession oxr_session_to_openxr(struct oxr_session *sess) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrSession, sess); } XrResult oxr_session_create(struct oxr_logger *log, struct oxr_system *sys, const XrSessionCreateInfo *createInfo, struct oxr_session **out_session); XrResult oxr_session_enumerate_formats(struct oxr_logger *log, struct oxr_session *sess, uint32_t formatCapacityInput, uint32_t *formatCountOutput, int64_t *formats); /*! * Change the state of the session, queues a event. */ void oxr_session_change_state(struct oxr_logger *log, struct oxr_session *sess, XrSessionState state); XrResult oxr_session_begin(struct oxr_logger *log, struct oxr_session *sess, const XrSessionBeginInfo *beginInfo); XrResult oxr_session_end(struct oxr_logger *log, struct oxr_session *sess); XrResult oxr_session_request_exit(struct oxr_logger *log, struct oxr_session *sess); void oxr_session_poll(struct oxr_logger *log, struct oxr_session *sess); XrResult oxr_session_locate_views(struct oxr_logger *log, struct oxr_session *sess, const XrViewLocateInfo *viewLocateInfo, XrViewState *viewState, uint32_t viewCapacityInput, uint32_t *viewCountOutput, XrView *views); XrResult oxr_session_frame_wait(struct oxr_logger *log, struct oxr_session *sess, XrFrameState *frameState); XrResult oxr_session_frame_begin(struct oxr_logger *log, struct oxr_session *sess); XrResult oxr_session_frame_end(struct oxr_logger *log, struct oxr_session *sess, const XrFrameEndInfo *frameEndInfo); XrResult oxr_session_hand_joints(struct oxr_logger *log, struct oxr_hand_tracker *hand_tracker, const XrHandJointsLocateInfoEXT *locateInfo, XrHandJointLocationsEXT *locations); /* * * oxr_space.c * */ /*! * To go back to a OpenXR object. */ static inline XrSpace oxr_space_to_openxr(struct oxr_space *spc) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrSpace, spc); } XrResult oxr_space_action_create(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, const XrActionSpaceCreateInfo *createInfo, struct oxr_space **out_space); XrResult oxr_space_reference_create(struct oxr_logger *log, struct oxr_session *sess, const XrReferenceSpaceCreateInfo *createInfo, struct oxr_space **out_space); /*! * Transforms a relation given in pure global space into the oxr_space @p spc. * If @p apply_space_pose is true, the pose offset of @p spc will be included in @p out_relation. */ bool oxr_space_pure_relation_in_space(struct oxr_logger *log, XrTime time, struct xrt_space_relation *relation, struct oxr_space *spc, bool apply_space_pose, struct xrt_space_relation *out_relation); /*! * Transforms a pose given in pure global space into a relation in the oxr_space @p spc. * If @p apply_space_pose is true, the pose offset of @p spc will be included in @p out_relation. */ bool oxr_space_pure_pose_in_space(struct oxr_logger *log, XrTime time, struct xrt_pose *pose, struct oxr_space *spc, bool apply_space_pose, struct xrt_space_relation *out_relation); /*! * Transforms a relation in an given oxr_space @p spc into pure global space, taking the pose offset of @p spc into * account. */ bool oxr_space_pure_relation_from_space(struct oxr_logger *log, XrTime time, struct xrt_space_relation *relation, struct oxr_space *spc, struct xrt_space_relation *out_relation); /*! * Transforms a posen in a given oxr_space @p spc into a relation in "pure" global space, taking the pose offset of @p * spc into account. */ bool oxr_space_pure_pose_from_space(struct oxr_logger *log, XrTime time, struct xrt_pose *pose, struct oxr_space *spc, struct xrt_space_relation *out_relation); /*! * Returns the pure relation in global space of an oxr_space, meaning the tracking_origin offsets are already applied * and sets @p out_xdev to the device the space is associated with. * * @todo: This function currently assumes all reference spaces are associated with the HMD. */ bool oxr_space_get_pure_relation(struct oxr_logger *log, struct oxr_space *spc, XrTime time, struct xrt_space_relation *out_relation, struct xrt_device **out_xdev); XrResult oxr_space_locate( struct oxr_logger *log, struct oxr_space *spc, struct oxr_space *baseSpc, XrTime time, XrSpaceLocation *location); bool is_local_space_set_up(struct oxr_session *sess); XrSpaceLocationFlags xrt_to_xr_space_location_flags(enum xrt_space_relation_flags relation_flags); bool global_to_local_space(struct oxr_logger *log, struct oxr_session *sess, XrTime time, struct xrt_space_relation *rel); /* * * oxr_swapchain.c * */ /*! * To go back to a OpenXR object. */ static inline XrSwapchain oxr_swapchain_to_openxr(struct oxr_swapchain *sc) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrSwapchain, sc); } XrResult oxr_create_swapchain(struct oxr_logger *, struct oxr_session *sess, const XrSwapchainCreateInfo *, struct oxr_swapchain **out_swapchain); /* * * oxr_messenger.c * */ /*! * To go back to a OpenXR object. */ static inline XrDebugUtilsMessengerEXT oxr_messenger_to_openxr(struct oxr_debug_messenger *mssngr) { return XRT_CAST_PTR_TO_OXR_HANDLE(XrDebugUtilsMessengerEXT, mssngr); } XrResult oxr_create_messenger(struct oxr_logger *, struct oxr_instance *inst, const XrDebugUtilsMessengerCreateInfoEXT *, struct oxr_debug_messenger **out_mssngr); XrResult oxr_destroy_messenger(struct oxr_logger *log, struct oxr_debug_messenger *mssngr); /* * * oxr_system.c * */ XrResult oxr_system_select(struct oxr_logger *log, struct oxr_system **systems, uint32_t system_count, XrFormFactor form_factor, struct oxr_system **out_selected); XrResult oxr_system_fill_in(struct oxr_logger *log, struct oxr_instance *inst, XrSystemId systemId, struct oxr_system *sys); XrResult oxr_system_verify_id(struct oxr_logger *log, const struct oxr_instance *inst, XrSystemId systemId); XrResult oxr_system_get_by_id(struct oxr_logger *log, struct oxr_instance *inst, XrSystemId systemId, struct oxr_system **system); XrResult oxr_system_get_properties(struct oxr_logger *log, struct oxr_system *sys, XrSystemProperties *properties); XrResult oxr_system_enumerate_view_confs(struct oxr_logger *log, struct oxr_system *sys, uint32_t viewConfigurationTypeCapacityInput, uint32_t *viewConfigurationTypeCountOutput, XrViewConfigurationType *viewConfigurationTypes); XrResult oxr_system_enumerate_blend_modes(struct oxr_logger *log, struct oxr_system *sys, XrViewConfigurationType viewConfigurationType, uint32_t environmentBlendModeCapacityInput, uint32_t *environmentBlendModeCountOutput, XrEnvironmentBlendMode *environmentBlendModes); XrResult oxr_system_get_view_conf_properties(struct oxr_logger *log, struct oxr_system *sys, XrViewConfigurationType viewConfigurationType, XrViewConfigurationProperties *configurationProperties); XrResult oxr_system_enumerate_view_conf_views(struct oxr_logger *log, struct oxr_system *sys, XrViewConfigurationType viewConfigurationType, uint32_t viewCapacityInput, uint32_t *viewCountOutput, XrViewConfigurationView *views); bool oxr_system_get_hand_tracking_support(struct oxr_logger *log, struct oxr_instance *inst); /* * * oxr_event.cpp * */ XrResult oxr_poll_event(struct oxr_logger *log, struct oxr_instance *inst, XrEventDataBuffer *eventData); XrResult oxr_event_push_XrEventDataSessionStateChanged(struct oxr_logger *log, struct oxr_session *sess, XrSessionState state, XrTime time); XrResult oxr_event_push_XrEventDataMainSessionVisibilityChangedEXTX(struct oxr_logger *log, struct oxr_session *sess, bool visible); XrResult oxr_event_push_XrEventDataInteractionProfileChanged(struct oxr_logger *log, struct oxr_session *sess); /*! * This clears all pending events refers to the given session. */ XrResult oxr_event_remove_session_events(struct oxr_logger *log, struct oxr_session *sess); /* * * oxr_xdev.c * */ void oxr_xdev_destroy(struct xrt_device **xdev_ptr); void oxr_xdev_update(struct xrt_device *xdev); /*! * Return true if it finds an input of that name on this device. */ bool oxr_xdev_find_input(struct xrt_device *xdev, enum xrt_input_name name, struct xrt_input **out_input); /*! * Return true if it finds an output of that name on this device. */ bool oxr_xdev_find_output(struct xrt_device *xdev, enum xrt_output_name name, struct xrt_output **out_output); void oxr_xdev_get_relation_chain(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, enum xrt_input_name name, XrTime at_time, struct xrt_relation_chain *xrc); void oxr_xdev_get_space_relation(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, enum xrt_input_name name, XrTime at_time, struct xrt_space_relation *out_relation); /*! * Returns the hand tracking value of the named input from the device. * Does NOT apply tracking origin offset to each joint. */ void oxr_xdev_get_hand_tracking_at(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, enum xrt_input_name name, XrTime at_time, struct xrt_hand_joint_set *out_value); /* * * OpenGL, located in various files. * */ #ifdef XR_USE_GRAPHICS_API_OPENGL #ifdef XR_USE_PLATFORM_XLIB XrResult oxr_session_populate_gl_xlib(struct oxr_logger *log, struct oxr_system *sys, XrGraphicsBindingOpenGLXlibKHR const *next, struct oxr_session *sess); #endif // XR_USE_PLATFORM_XLIB #endif // XR_USE_GRAPHICS_API_OPENGL #if defined(XR_USE_GRAPHICS_API_OPENGL) || defined(XR_USE_GRAPHICS_API_OPENGL_ES) XrResult oxr_swapchain_gl_create(struct oxr_logger *, struct oxr_session *sess, const XrSwapchainCreateInfo *, struct oxr_swapchain **out_swapchain); #endif // XR_USE_GRAPHICS_API_OPENGL || XR_USE_GRAPHICS_API_OPENGL_ES #if defined(XR_USE_GRAPHICS_API_OPENGL_ES) #if defined(XR_USE_PLATFORM_ANDROID) XrResult oxr_session_populate_gles_android(struct oxr_logger *log, struct oxr_system *sys, XrGraphicsBindingOpenGLESAndroidKHR const *next, struct oxr_session *sess); #endif // XR_USE_PLATFORM_ANDROID #endif // XR_USE_GRAPHICS_API_OPENGL_ES /* * * Vulkan, located in various files. * */ #ifdef XR_USE_GRAPHICS_API_VULKAN XrResult oxr_vk_get_instance_exts(struct oxr_logger *log, struct oxr_system *sys, uint32_t namesCapacityInput, uint32_t *namesCountOutput, char *namesString); XrResult oxr_vk_get_device_exts(struct oxr_logger *log, struct oxr_system *sys, uint32_t namesCapacityInput, uint32_t *namesCountOutput, char *namesString); XrResult oxr_vk_get_requirements(struct oxr_logger *log, struct oxr_system *sys, XrGraphicsRequirementsVulkanKHR *graphicsRequirements); XrResult oxr_vk_create_vulkan_instance(struct oxr_logger *log, struct oxr_system *sys, const XrVulkanInstanceCreateInfoKHR *createInfo, VkInstance *vulkanInstance, VkResult *vulkanResult); XrResult oxr_vk_create_vulkan_device(struct oxr_logger *log, struct oxr_system *sys, const XrVulkanDeviceCreateInfoKHR *createInfo, VkDevice *vulkanDevice, VkResult *vulkanResult); XrResult oxr_vk_get_physical_device(struct oxr_logger *log, struct oxr_instance *inst, struct oxr_system *sys, VkInstance vkInstance, PFN_vkGetInstanceProcAddr getProc, VkPhysicalDevice *vkPhysicalDevice); XrResult oxr_session_populate_vk(struct oxr_logger *log, struct oxr_system *sys, XrGraphicsBindingVulkanKHR const *next, struct oxr_session *sess); XrResult oxr_swapchain_vk_create(struct oxr_logger *, struct oxr_session *sess, const XrSwapchainCreateInfo *, struct oxr_swapchain **out_swapchain); #endif /* * * EGL, located in various files. * */ #ifdef XR_USE_PLATFORM_EGL XrResult oxr_session_populate_egl(struct oxr_logger *log, struct oxr_system *sys, XrGraphicsBindingEGLMNDX const *next, struct oxr_session *sess); #endif /* * * Structs * */ /*! * Used to hold diverse child handles and ensure orderly destruction. * * Each object referenced by an OpenXR handle should have one of these as its * first element, thus "extending" this class. */ struct oxr_handle_base { //! Magic (per-handle-type) value for debugging. uint64_t debug; /*! * Pointer to this object's parent handle holder, if any. */ struct oxr_handle_base *parent; /*! * Array of children, if any. */ struct oxr_handle_base *children[XRT_MAX_HANDLE_CHILDREN]; /*! * Current handle state. */ enum oxr_handle_state state; /*! * Destroy the object this handle refers to. */ oxr_handle_destroyer destroy; }; /*! * Single or multiple devices grouped together to form a system that sessions * can be created from. Might need to open devices in order to get all * properties from it, but shouldn't. * * Not strictly an object, but an atom. * * Valid only within a XrInstance (@ref oxr_instance) * * @obj{XrSystemId} */ struct oxr_system { struct oxr_instance *inst; //! System compositor, used to create session compositors. struct xrt_system_compositor *xsysc; struct xrt_device *xdevs[16]; size_t xdev_count; /* index for xdevs array */ struct { #define OXR_ROLE_FIELD(X) int X; OXR_FOR_EACH_VALID_SUBACTION_PATH(OXR_ROLE_FIELD) #undef OXR_ROLE_FIELD } role; XrSystemId systemId; //! Have the client application called the gfx api requirements func? bool gotten_requirements; XrFormFactor form_factor; XrViewConfigurationType view_config_type; XrViewConfigurationView views[2]; uint32_t blend_mode_count; XrEnvironmentBlendMode blend_modes[3]; #ifdef XR_USE_GRAPHICS_API_VULKAN //! The instance/device we create when vulkan_enable2 is used VkInstance vulkan_enable2_instance; //! The device returned with the last xrGetVulkanGraphicsDeviceKHR or xrGetVulkanGraphicsDevice2KHR call. //! XR_NULL_HANDLE if neither has been called. VkPhysicalDevice suggested_vulkan_physical_device; struct { // No better place to keep this state. bool timeline_semaphore_enabled; } vk; #endif }; #define GET_XDEV_BY_ROLE(SYS, ROLE) SYS->role.ROLE == XRT_DEVICE_ROLE_UNASSIGNED ? NULL : SYS->xdevs[SYS->role.ROLE] #define MAKE_EXT_STATUS(mixed_case, all_caps) bool mixed_case; /*! * Structure tracking which extensions are enabled for a given instance. * * Names are systematic: the extension name with the XR_ prefix removed. */ struct oxr_extension_status { OXR_EXTENSION_SUPPORT_GENERATE(MAKE_EXT_STATUS) }; /*! * Main object that ties everything together. * * No parent type/handle: this is the root handle. * * @obj{XrInstance} * @extends oxr_handle_base */ struct oxr_instance { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; /* ---- HACK ---- */ void *hack; /* ---- HACK ---- */ struct xrt_instance *xinst; //! Enabled extensions struct oxr_extension_status extensions; // Hardcoded single system. struct oxr_system system; struct time_state *timekeeping; struct { struct u_hashset *name_store; struct u_hashset *loc_store; } action_sets; //! Path store, for looking up paths. struct u_hashset *path_store; //! Mapping from ID to path. struct oxr_path **path_array; //! Total length of path array. size_t path_array_length; //! Number of paths in the array (0 is always null). size_t path_num; // Event queue. struct { struct os_mutex mutex; struct oxr_event *last; struct oxr_event *next; } event; //! Interaction profile bindings that have been suggested by the client. struct oxr_interaction_profile **profiles; size_t profile_count; struct oxr_session *sessions; struct { #define SUBACTION_PATH_MEMBER(X) XrPath X; OXR_FOR_EACH_SUBACTION_PATH(SUBACTION_PATH_MEMBER) #undef SUBACTION_PATH_MEMBER XrPath khr_simple_controller; XrPath google_daydream_controller; XrPath htc_vive_controller; XrPath htc_vive_pro; XrPath microsoft_motion_controller; XrPath microsoft_xbox_controller; XrPath oculus_go_controller; XrPath oculus_touch_controller; XrPath valve_index_controller; XrPath mndx_ball_on_a_stick_controller; XrPath msft_hand_interaction; } path_cache; struct { struct { struct { uint32_t major; uint32_t minor; uint32_t patch; const char *name; //< Engine name, not freed. } engine; } detected; } appinfo; struct { //! Unreal has a bug in the VulkanRHI backend. bool disable_vulkan_format_depth_stencil; } quirks; //! Debug messengers struct oxr_debug_messenger *messengers[XRT_MAX_HANDLE_CHILDREN]; bool lifecycle_verbose; bool debug_views; bool debug_spaces; bool debug_bindings; #ifdef XRT_FEATURE_RENDERDOC RENDERDOC_API_1_4_1 *rdoc_api; #endif }; /*! * Object that client program interact with. * * Parent type/handle is @ref oxr_instance * * @obj{XrSession} * @extends oxr_handle_base */ struct oxr_session { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; struct oxr_system *sys; //! Native compositor that is wrapped by client compositors. struct xrt_compositor_native *xcn; struct xrt_compositor *compositor; struct oxr_session *next; XrSessionState state; bool has_begun; /*! * There is a extra state between xrBeginSession has been called and * the first xrEndFrame has been called. These are to track this. */ bool has_ended_once; bool compositor_visible; bool compositor_focused; // the number of xrWaitFrame calls that did not yet have a corresponding // xrEndFrame or xrBeginFrame (discarded frame) call int active_wait_frames; struct os_mutex active_wait_frames_lock; bool frame_started; bool exiting; struct { int64_t waited; int64_t begun; } frame_id; struct os_semaphore sem; /*! * An array of action set attachments that this session owns. * * If non-null, this means action sets have been attached to this * session. */ struct oxr_action_set_attachment *act_set_attachments; /*! * Length of @ref oxr_session::act_set_attachments. */ size_t action_set_attachment_count; /*! * A map of action set key to action set attachments. * * If non-null, this means action sets have been attached to this * session, since this map points to elements of * oxr_session::act_set_attachments */ struct u_hashmap_int *act_sets_attachments_by_key; /*! * A map of action key to action attachment. * * The action attachments are actually owned by the action set * attachments, but we own the action set attachments, so this is OK. * * If non-null, this means action sets have been attached to this * session, since this map points to @p oxr_action_attachment members of * oxr_session::act_set_attachments elements. */ struct u_hashmap_int *act_attachments_by_key; /*! * Currently bound interaction profile. * @{ */ #define OXR_PATH_MEMBER(X) XrPath X; OXR_FOR_EACH_VALID_SUBACTION_PATH(OXR_PATH_MEMBER) #undef OXR_PATH_MEMBER /*! * @} */ /*! * IPD, to be expanded to a proper 3D relation. */ float ipd_meters; /*! * Frame timing debug output. */ bool frame_timing_spew; //! Extra sleep in wait frame. uint32_t frame_timing_wait_sleep_ms; /*! * To pipe swapchain creation to right code. */ XrResult (*create_swapchain)(struct oxr_logger *, struct oxr_session *sess, const XrSwapchainCreateInfo *, struct oxr_swapchain **); /*! initial relation of head in "global" space. * Used as reference for local space. */ struct xrt_space_relation local_space_pure_relation; }; /*! * Returns XR_SUCCESS or XR_SESSION_LOSS_PENDING as appropriate. * * @public @memberof oxr_session */ static inline XrResult oxr_session_success_result(struct oxr_session *session) { switch (session->state) { case XR_SESSION_STATE_LOSS_PENDING: return XR_SESSION_LOSS_PENDING; default: return XR_SUCCESS; } } /*! * Returns XR_SUCCESS, XR_SESSION_LOSS_PENDING, or XR_SESSION_NOT_FOCUSED, as * appropriate. * * @public @memberof oxr_session */ static inline XrResult oxr_session_success_focused_result(struct oxr_session *session) { switch (session->state) { case XR_SESSION_STATE_LOSS_PENDING: return XR_SESSION_LOSS_PENDING; case XR_SESSION_STATE_FOCUSED: return XR_SUCCESS; default: return XR_SESSION_NOT_FOCUSED; } } /*! * A single interaction profile. */ struct oxr_interaction_profile { XrPath path; //! Used to lookup @ref xrt_binding_profile for fallback. enum xrt_device_name xname; //! Name presented to the user. const char *localized_name; struct oxr_binding *bindings; size_t binding_count; }; /*! * Interaction profile binding state. */ struct oxr_binding { XrPath *paths; size_t path_count; //! Name presented to the user. const char *localized_name; enum oxr_subaction_path subaction_path; size_t key_count; uint32_t *keys; //! store which entry in paths was suggested, for each action key uint32_t *preferred_binding_path_index; enum xrt_input_name input; enum xrt_output_name output; }; /*! * @defgroup oxr_input OpenXR input * @ingroup oxr_main * * @brief The action-set/action-based input subsystem of OpenXR. * * * Action sets are created as children of the Instance, but are primarily used * with one or more Sessions. They may be used with multiple sessions at a time, * so we can't just put the per-session information directly in the action set * or action. Instead, we have the `_attachment `structures, which mirror the * action sets and actions but are rooted under the Session: * * - For every action set attached to a session, that session owns a @ref * oxr_action_set_attachment. * - For each action in those attached action sets, the action set attachment * owns an @ref oxr_action_attachment. * * We go from the public handle to the `_attachment` structure by using a `key` * value and a hash map: specifically, we look up the * oxr_action_set::act_set_key and oxr_action::act_key in the session. * * ![](monado-input-class-relationships.drawio.svg) */ /*! * A parsed equivalent of a list of sub-action paths. * * If @p any is true, then no paths were provided, which typically means any * input is acceptable. * * @ingroup oxr_main * @ingroup oxr_input */ struct oxr_subaction_paths { bool any; #define OXR_SUBPATH_MEMBER(X) bool X; OXR_FOR_EACH_SUBACTION_PATH(OXR_SUBPATH_MEMBER) #undef OXR_SUBPATH_MEMBER }; /*! * The data associated with the attachment of an Action Set (@ref * oxr_action_set) to as Session (@ref oxr_session). * * This structure has no pointer to the @ref oxr_action_set that created it * because the application is allowed to destroy an action before the session, * which should change nothing except not allow the application to access the * corresponding data anymore. * * @ingroup oxr_input * * @see oxr_action_set */ struct oxr_action_set_attachment { //! Owning session. struct oxr_session *sess; //! Action set refcounted data struct oxr_action_set_ref *act_set_ref; //! Unique key for the session hashmap. uint32_t act_set_key; //! Which sub-action paths are requested on the latest sync. struct oxr_subaction_paths requested_subaction_paths; //! An array of action attachments we own. struct oxr_action_attachment *act_attachments; /*! * Length of @ref oxr_action_set_attachment::act_attachments. */ size_t action_attachment_count; }; /*! * De-initialize an action set attachment and its action attachments. * * Frees the action attachments, but does not de-allocate the action set * attachment. * * @public @memberof oxr_action_set_attachment */ void oxr_action_set_attachment_teardown(struct oxr_action_set_attachment *act_set_attached); /*! * The state of a action input. * * @ingroup oxr_input * * @see oxr_action_attachment */ struct oxr_action_state { /*! * The actual value - must interpret using action type */ union xrt_input_value value; //! Is this active (bound and providing input)? bool active; // Was this changed. bool changed; //! When was this last changed. XrTime timestamp; }; /*! * A input action pair of a @ref xrt_input and a @ref xrt_device, along with the * required transform. * * @ingroup oxr_input * * @see xrt_device * @see xrt_input */ struct oxr_action_input { struct xrt_device *xdev; struct xrt_input *input; struct oxr_input_transform *transforms; size_t transform_count; XrPath bound_path; }; /*! * A output action pair of a @ref xrt_output_name and a @ref xrt_device. * * @ingroup oxr_input * * @see xrt_device * @see xrt_output_name */ struct oxr_action_output { struct xrt_device *xdev; enum xrt_output_name name; XrPath bound_path; }; /*! * The set of inputs/outputs for a single sub-action path for an action. * * Each @ref oxr_action_attachment has one of these for every known sub-action * path in the spec. Many, or even most, will be "empty". * * A single action will either be input or output, not both. * * @ingroup oxr_input * * @see oxr_action_attachment */ struct oxr_action_cache { struct oxr_action_state current; size_t input_count; struct oxr_action_input *inputs; int64_t stop_output_time; size_t output_count; struct oxr_action_output *outputs; }; /*! * Data associated with an Action that has been attached to a Session. * * More information on the action vs action attachment and action set vs action * set attachment parallel is in the docs for @ref oxr_input * * @ingroup oxr_input * * @see oxr_action */ struct oxr_action_attachment { //! The owning action set attachment struct oxr_action_set_attachment *act_set_attached; //! This action's refcounted data struct oxr_action_ref *act_ref; /*! * The corresponding session. * * This will always be valid: the session outlives this object because * it owns act_set_attached. */ struct oxr_session *sess; //! Unique key for the session hashmap. uint32_t act_key; /*! * For pose actions any subaction paths are special treated, at bind * time we pick one subaction path and stick to it as long as the action * lives. */ struct oxr_subaction_paths any_pose_subaction_path; struct oxr_action_state any_state; #define OXR_CACHE_MEMBER(X) struct oxr_action_cache X; OXR_FOR_EACH_SUBACTION_PATH(OXR_CACHE_MEMBER) #undef OXR_CACHE_MEMBER }; /*! * @} */ /*! * Can be one of several reference space types, or a space that is bound to an * action. * * Parent type/handle is @ref oxr_session * * @obj{XrSpace} * @extends oxr_handle_base */ struct oxr_space { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Owner of this space. struct oxr_session *sess; //! Pose that was given during creation. struct xrt_pose pose; //! What kind of reference space is this, if any. XrReferenceSpaceType type; //! Action key from which action this space was created from. uint32_t act_key; //! Is this a reference space? bool is_reference; //! Which sub action path is this? struct oxr_subaction_paths subaction_paths; }; /*! * A set of images used for rendering. * * Parent type/handle is @ref oxr_session * * @obj{XrSwapchain} * @extends oxr_handle_base */ struct oxr_swapchain { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Owner of this swapchain. struct oxr_session *sess; //! Compositor swapchain. struct xrt_swapchain *swapchain; //! Swapchain size. uint32_t width, height; //! For 1 is 2D texture, greater then 1 2D array texture. uint32_t array_layer_count; struct { enum oxr_image_state state; } images[OXR_MAX_SWAPCHAIN_IMAGES]; struct { size_t num; struct u_index_fifo fifo; } acquired; struct { bool yes; int index; } waited; struct { bool yes; int index; } released; // Is this a static swapchain, needed for acquire semantics. bool is_static; XrResult (*destroy)(struct oxr_logger *, struct oxr_swapchain *); XrResult (*enumerate_images)(struct oxr_logger *, struct oxr_swapchain *, uint32_t, XrSwapchainImageBaseHeader *); XrResult (*acquire_image)(struct oxr_logger *, struct oxr_swapchain *, const XrSwapchainImageAcquireInfo *, uint32_t *); XrResult (*wait_image)(struct oxr_logger *, struct oxr_swapchain *, const XrSwapchainImageWaitInfo *); XrResult (*release_image)(struct oxr_logger *, struct oxr_swapchain *, const XrSwapchainImageReleaseInfo *); }; struct oxr_refcounted { struct xrt_reference base; //! Destruction callback void (*destroy)(struct oxr_refcounted *); }; /*! * Increase the reference count of @p orc. */ static inline void oxr_refcounted_ref(struct oxr_refcounted *orc) { xrt_reference_inc(&orc->base); } /*! * Decrease the reference count of @p orc, destroying it if it reaches 0. */ static inline void oxr_refcounted_unref(struct oxr_refcounted *orc) { if (xrt_reference_dec(&orc->base)) { orc->destroy(orc); } } /*! * The reference-counted data of an action set. * * One or more sessions may still need this data after the application destroys * its XrActionSet handle, so this data is refcounted. * * @ingroup oxr_input * * @see oxr_action_set * @extends oxr_refcounted */ struct oxr_action_set_ref { struct oxr_refcounted base; //! Application supplied name of this action. char name[XR_MAX_ACTION_SET_NAME_SIZE]; /*! * Has this action set even been attached to any session, marking it as * immutable. */ bool ever_attached; //! Unique key for the session hashmap. uint32_t act_set_key; //! Application supplied action set priority. uint32_t priority; struct { struct u_hashset *name_store; struct u_hashset *loc_store; } actions; }; /*! * A group of actions. * * Parent type/handle is @ref oxr_instance * * Note, however, that an action set must be "attached" to a session * ( @ref oxr_session ) to be used and not just configured. * The corresponding data is in @ref oxr_action_set_attachment. * * @ingroup oxr_input * * @obj{XrActionSet} * @extends oxr_handle_base */ struct oxr_action_set { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Owner of this action set. struct oxr_instance *inst; /*! * The data for this action set that must live as long as any session we * are attached to. */ struct oxr_action_set_ref *data; /*! * Unique key for the session hashmap. * * Duplicated from oxr_action_set_ref::act_set_key for efficiency. */ uint32_t act_set_key; //! The item in the name hashset. struct u_hashset_item *name_item; //! The item in the localized hashset. struct u_hashset_item *loc_item; }; /*! * The reference-counted data of an action. * * One or more sessions may still need this data after the application destroys * its XrAction handle, so this data is refcounted. * * @ingroup oxr_input * * @see oxr_action * @extends oxr_refcounted */ struct oxr_action_ref { struct oxr_refcounted base; //! Application supplied name of this action. char name[XR_MAX_ACTION_NAME_SIZE]; //! Unique key for the session hashmap. uint32_t act_key; //! Type this action was created with. XrActionType action_type; //! Which sub action paths that this action was created with. struct oxr_subaction_paths subaction_paths; }; /*! * A single action. * * Parent type/handle is @ref oxr_action_set * * For actual usage, an action is attached to a session: the corresponding data * is in @ref oxr_action_attachment * * @ingroup oxr_input * * @obj{XrAction} * @extends oxr_handle_base */ struct oxr_action { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Owner of this action. struct oxr_action_set *act_set; //! The data for this action that must live as long as any session we //! are attached to. struct oxr_action_ref *data; /*! * Unique key for the session hashmap. * * Duplicated from oxr_action_ref::act_key for efficiency. */ uint32_t act_key; //! The item in the name hashset. struct u_hashset_item *name_item; //! The item in the localized hashset. struct u_hashset_item *loc_item; }; /*! * Debug object created by the client program. * * Parent type/handle is @ref oxr_instance * * @obj{XrDebugUtilsMessengerEXT} */ struct oxr_debug_messenger { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Owner of this messenger. struct oxr_instance *inst; //! Severities to submit to this messenger XrDebugUtilsMessageSeverityFlagsEXT message_severities; //! Types to submit to this messenger XrDebugUtilsMessageTypeFlagsEXT message_types; //! Callback function PFN_xrDebugUtilsMessengerCallbackEXT user_callback; //! Opaque user data void *XR_MAY_ALIAS user_data; }; /*! * A hand tracker. * * Parent type/handle is @ref oxr_instance * * * @obj{XrHandTrackerEXT} * @extends oxr_handle_base */ struct oxr_hand_tracker { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Owner of this hand tracker. struct oxr_session *sess; //! xrt_device backing this hand tracker struct xrt_device *xdev; //! the input name associated with this hand tracker enum xrt_input_name input_name; XrHandEXT hand; XrHandJointSetEXT hand_joint_set; }; /*! * @} */ #ifdef __cplusplus } #endif