// 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 "oxr_extension_support.h" #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") // clang-format on /* * * Forward declare structs. * */ struct xrt_instance; struct oxr_logger; 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_sub_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; #define XRT_MAX_HANDLE_CHILDREN 256 #define OXR_MAX_SWAPCHAIN_IMAGES 8 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_sub_action_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); } /*! * @public @static @memberof oxr_instance */ XrResult oxr_instance_create(struct oxr_logger *log, const XrInstanceCreateInfo *createInfo, 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 a * 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_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 sub_paths. * * Sets all members of @p sub_paths ( @ref oxr_sub_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 * @relatesalso oxr_sub_paths */ bool oxr_classify_sub_action_paths(struct oxr_logger *log, struct oxr_instance *inst, uint32_t num_subaction_paths, const XrPath *subaction_paths, struct oxr_sub_paths *sub_paths); /*! * Find the pose input for the set of sub_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_sub_paths *sub_paths, 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 * @relatesalso 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_sub_paths sub_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_sub_paths sub_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_sub_paths sub_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_sub_paths sub_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_sub_paths sub_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_sub_paths sub_paths); /*! * @} */ /*! * * @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[32], size_t *num_bindings); /*! * @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); 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); /*! * Get the view space position at the given time in relation to the * local or stage space. */ XrResult oxr_session_get_view_pose_at(struct oxr_logger *, struct oxr_session *sess, XrTime at_time, struct xrt_pose *); XrResult oxr_session_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); /* * * 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); XrResult oxr_space_locate(struct oxr_logger *log, struct oxr_space *spc, struct oxr_space *baseSpc, XrTime time, XrSpaceLocation *location); XrResult oxr_space_ref_relation(struct oxr_logger *log, struct oxr_session *sess, XrReferenceSpaceType space, XrReferenceSpaceType baseSpc, XrTime time, struct xrt_space_relation *out_relation); bool initial_head_relation_valid(struct oxr_session *sess); bool global_to_local_space(struct oxr_session *sess, struct xrt_pose *pose); /* * * 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 num_systems, 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, struct xrt_device **xdevs, size_t num_xdevs); 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); /* * * 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); /*! * Returns the pose of the named input from the device, if the pose isn't valid * uses the device offset instead. */ void oxr_xdev_get_pose_at(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, enum xrt_input_name name, XrTime at_time, uint64_t *out_pose_timestamp_ns, struct xrt_space_relation *out_relation); /*! * Returns the relation of the named input from the device, always ensures * that position and orientation is valid by using the device offset. */ void oxr_xdev_get_relation_at(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, enum xrt_input_name name, XrTime at_time, uint64_t *out_relation_timestamp_ns, struct xrt_space_relation *out_relation); /* * * 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 /* * * 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_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; }; #define XRT_DEVICE_ROLE_UNASSIGNED (-1) /*! * 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; struct xrt_device *xdevs[16]; size_t num_xdevs; /* index for xdevs array */ struct { int head; int left; int right; } 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 num_blend_modes; XrEnvironmentBlendMode blend_modes[3]; }; #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; struct oxr_interaction_profile **profiles; size_t num_profiles; struct oxr_session *sessions; struct { XrPath user; XrPath head; XrPath left; XrPath right; XrPath gamepad; XrPath treadmill; 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; } path_cache; //! Debug messengers struct oxr_debug_messenger *messengers[XRT_MAX_HANDLE_CHILDREN]; bool lifecycle_verbose; bool debug_views; bool debug_spaces; bool debug_bindings; }; /*! * 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; 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 xrWaitFrame has been called. These are to track this. */ bool has_waited_once; 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 num_action_set_attachments; /*! * 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. * @{ */ XrPath left; XrPath right; XrPath head; XrPath gamepad; /*! * @} */ /*! * IPD, to be expanded to a proper 3D relation. */ float ipd_meters; float static_prediction_s; /*! * 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 initial_head_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; struct oxr_binding *bindings; size_t num_bindings; }; /*! * Interaction profile binding state. */ struct oxr_binding { XrPath *paths; size_t num_paths; enum oxr_sub_action_path sub_path; size_t num_keys; uint32_t *keys; //! store which entry in paths was suggested, for each action key uint32_t *preferred_binding_path_index; enum xrt_input_name *inputs; size_t num_inputs; enum xrt_output_name *outputs; size_t num_outputs; }; /*! * 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. */ struct oxr_sub_paths { bool any; bool user; bool head; bool left; bool right; bool gamepad; bool treadmill; }; /*! * The data associated with the attachment of an Action Set (@ref * oxr_action_set) to as Session (@ref oxr_session). * * 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::key and * oxr_action::key in the 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. * * @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_sub_paths requested_sub_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 num_action_attachments; }; /*! * 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. * * @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. * * @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 num_transforms; }; /*! * A output action pair of a @ref xrt_output_name and a @ref xrt_device. * * @see xrt_device * @see xrt_output_name */ struct oxr_action_output { struct xrt_device *xdev; enum xrt_output_name name; }; /*! * 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". * * @see oxr_action_attachment */ struct oxr_action_cache { struct oxr_action_state current; //! Which action is proving the binding. XrPath bound_path; size_t num_inputs; struct oxr_action_input *inputs; int64_t stop_output_time; size_t num_outputs; 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_action_set_attachment. * * @see oxr_action * @see oxr_action_set_attachment */ 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; struct oxr_action_state any_state; struct oxr_action_cache user; struct oxr_action_cache head; struct oxr_action_cache left; struct oxr_action_cache right; struct oxr_action_cache gamepad; }; /*! * @} */ /*! * 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_sub_paths sub_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; struct { //! Swapchain size. uint32_t width, height; //! For 1 is 2D texture, greater then 1 2D array texture. uint32_t num_array_layers; }; 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. * * @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; 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. * * @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. * * @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_sub_paths sub_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 * * @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; }; /*! * @} */ #ifdef __cplusplus } #endif