// Copyright 2018-2019, 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 "util/u_hashset.h" #include "util/u_hashmap.h" #ifdef __cplusplus extern "C" { #endif /*! * @defgroup oxr OpenXR state tracker * * Client application facing code. * * @ingroup xrt */ /*! * @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 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_source; struct oxr_source_set; struct oxr_source_input; struct oxr_source_output; #define XRT_MAX_HANDLE_CHILDREN 256 struct time_state; 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, }; /* * * 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. */ XrResult oxr_handle_destroy(struct oxr_logger *log, struct oxr_handle_base *hb); /*! * Returns a human-readable label for a handle state. */ const char * oxr_handle_state_to_string(enum oxr_handle_state state); /* * * oxr_instance.c * */ /*! * To go back to a OpenXR object. */ XRT_MAYBE_UNUSED static inline XrInstance oxr_instance_to_openxr(struct oxr_instance *inst) { return (XrInstance)inst; } XrResult oxr_instance_create(struct oxr_logger *log, const XrInstanceCreateInfo *createInfo, struct oxr_instance **out_inst); XrResult oxr_instance_get_properties(struct oxr_logger *log, struct oxr_instance *inst, XrInstanceProperties *instanceProperties); #if XR_USE_TIMESPEC XrResult oxr_instance_convert_time_to_timespec(struct oxr_logger *log, struct oxr_instance *inst, XrTime time, struct timespec *timespecTime); 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 /* * * oxr_path.c * */ 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. */ 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. */ 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). */ XrResult oxr_path_get_string(struct oxr_logger *log, struct oxr_instance *inst, XrPath path, const char **out_str, size_t *out_length); /*! * Destroy all paths that the instance has created. */ void oxr_path_destroy_all(struct oxr_logger *log, struct oxr_instance *inst); /* * * oxr_input.c * */ /*! * Helper function to classify sub_paths. */ void 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 */ XrResult oxr_source_get_pose_input(struct oxr_logger *log, struct oxr_session *sess, uint32_t key, const struct oxr_sub_paths *sub_paths, struct oxr_source_input **out_input); /*! * To go back to a OpenXR object. */ XRT_MAYBE_UNUSED static inline XrActionSet oxr_action_set_to_openxr(struct oxr_action_set *act_set) { return (XrActionSet)act_set; } XrResult oxr_action_set_create(struct oxr_logger *log, struct oxr_session *sess, const XrActionSetCreateInfo *createInfo, struct oxr_action_set **out_act_set); /*! * To go back to a OpenXR object. */ XRT_MAYBE_UNUSED static inline XrAction oxr_action_to_openxr(struct oxr_action *act) { return (XrAction)act; } XrResult oxr_action_create(struct oxr_logger *log, struct oxr_action_set *act_set, const XrActionCreateInfo *createInfo, struct oxr_action **out_act); XrResult oxr_action_sync_data(struct oxr_logger *log, struct oxr_session *sess, uint32_t countActionSets, const XrActiveActionSet *actionSets); XrResult oxr_action_set_interaction_profile_suggested_bindings( struct oxr_logger *log, struct oxr_session *sess, const XrInteractionProfileSuggestedBinding *suggestedBindings); XrResult oxr_action_get_current_interaction_profile( struct oxr_logger *log, struct oxr_session *sess, XrPath topLevelUserPath, XrInteractionProfileInfo *interactionProfile); XrResult oxr_action_get_input_source_localized_name( struct oxr_logger *log, struct oxr_session *sess, XrPath source, XrInputSourceLocalizedNameFlags whichComponents, uint32_t bufferCapacityInput, uint32_t *bufferCountOutput, char *buffer); XrResult oxr_action_get_boolean(struct oxr_logger *log, struct oxr_action *act, struct oxr_sub_paths sub_paths, XrActionStateBoolean *data); XrResult oxr_action_get_vector1f(struct oxr_logger *log, struct oxr_action *act, struct oxr_sub_paths sub_paths, XrActionStateVector1f *data); XrResult oxr_action_get_vector2f(struct oxr_logger *log, struct oxr_action *act, struct oxr_sub_paths sub_paths, XrActionStateVector2f *data); XrResult oxr_action_get_pose(struct oxr_logger *log, struct oxr_action *act, struct oxr_sub_paths sub_paths, XrActionStatePose *data); XrResult oxr_action_get_bound_sources(struct oxr_logger *log, struct oxr_action *act, uint32_t sourceCapacityInput, uint32_t *sourceCountOutput, XrPath *sources); XrResult oxr_action_apply_haptic_feedback(struct oxr_logger *log, struct oxr_action *act, uint32_t countSubactionPaths, const XrPath *subactionPaths, const XrHapticBaseHeader *hapticEvent); XrResult oxr_action_stop_haptic_feedback(struct oxr_logger *log, struct oxr_action *act, uint32_t countSubactionPaths, const XrPath *subactionPaths); /* * * oxr_session.c * */ /*! * To go back to a OpenXR object. */ XRT_MAYBE_UNUSED static inline XrSession oxr_session_to_openxr(struct oxr_session *sess) { return (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); /*! * 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. */ XRT_MAYBE_UNUSED static inline XrSpace oxr_space_to_openxr(struct oxr_space *spc) { return (XrSpace)spc; } XrResult oxr_space_action_create(struct oxr_logger *log, struct oxr_action *act, 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, XrSpaceRelation *relation); 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); /* * * oxr_swapchain.c * */ /*! * To go back to a OpenXR object. */ XRT_MAYBE_UNUSED static inline XrSwapchain oxr_swapchain_to_openxr(struct oxr_swapchain *sc) { return (XrSwapchain)sc; } XrResult oxr_create_swapchain(struct oxr_logger *, struct oxr_session *sess, const XrSwapchainCreateInfo *, struct oxr_swapchain **sc); /* * * oxr_messenger.c * */ #ifdef XR_EXT_debug_utils /*! * To go back to a OpenXR object. */ XRT_MAYBE_UNUSED static inline XrDebugUtilsMessengerEXT oxr_messenger_to_openxr(struct oxr_debug_messenger *mssngr) { return (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); #endif // XR_EXT_debug_utils /* * * 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 *head, struct xrt_device *left, struct xrt_device *right); 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, 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); /* * * oxr_xdev.c * */ void oxr_xdev_update(struct xrt_device* xdev, struct time_state* timekeeping); void oxr_xdev_find_input(struct xrt_device *xdev, enum xrt_input_name name, struct xrt_input **out_input); void oxr_xdev_find_output(struct xrt_device *xdev, enum xrt_output_name name, struct xrt_output **out_output); void oxr_xdev_get_pose_at(struct oxr_logger *log, struct oxr_instance *inst, struct xrt_device *xdev, enum xrt_input_name name, struct xrt_pose *pose, int64_t *timestamp); /* * * 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 XrResult oxr_swapchain_gl_create(struct oxr_logger *, struct oxr_session *sess, const XrSwapchainCreateInfo *, struct oxr_swapchain **out_swapchain); #endif /* * * 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 /* * * 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. */ 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. * * @obj{XrSystemId} */ struct oxr_system { struct oxr_instance *inst; struct xrt_device *head; struct xrt_device *left; struct xrt_device *right; XrSystemId systemId; XrFormFactor form_factor; XrViewConfigurationType view_config_type; XrViewConfigurationView views[2]; uint32_t num_blend_modes; XrEnvironmentBlendMode blend_modes[3]; }; /*! * Main object that ties everything together. * * @obj{XrInstance} */ struct oxr_instance { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; struct xrt_prober *prober; // Enabled extensions bool headless; bool opengl_enable; bool vulkan_enable; // Hardcoded single system. struct oxr_system system; struct time_state *timekeeping; //! Path store, for looking up paths. struct u_hashset *path_store; // Event queue. struct oxr_event *last_event; struct oxr_event *next_event; struct { XrPath user; XrPath head; XrPath left; XrPath right; XrPath gamepad; } path_cache; #ifdef XR_EXT_debug_utils //! Debug messengers struct oxr_debug_messenger *messengers[XRT_MAX_HANDLE_CHILDREN]; #endif // XR_EXT_debug_utils }; /*! * Object that client program interact with. * * @obj{XrSession} */ 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; XrSessionState state; bool frame_started; struct u_hashmap_int *act_sets; struct u_hashmap_int *sources; /*! * 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 **); }; /*! * To carry around a sementic selection of sub action paths. */ struct oxr_sub_paths { bool any; bool user; bool head; bool left; bool right; bool gamepad; }; /*! * Session input source. * * @see oxr_action_set */ struct oxr_source_set { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Which generation of the XrActionSet was this created from. uint32_t generation; }; /*! * The state of a action input source. * * @see oxr_source */ struct oxr_source_state { union { struct { float x; } vec1; struct { float x; float y; } vec2; bool boolean; }; bool active; // Was this changed. bool changed; //! When was this last changed. XrTime timestamp; }; /*! * A input source pair of a @ref xrt_input and a @ref xrt_device. * * @see xrt_device * @see xrt_input */ struct oxr_source_input { struct xrt_device *xdev; struct xrt_input *input; }; /*! * A output source pair of a @ref xrt_output_name and a @ref xrt_device. * * @see xrt_device * @see xrt_output_name */ struct oxr_source_output { struct xrt_device *xdev; enum xrt_output_name name; }; /*! * A set of inputs for a single sub action path. * * @see oxr_source */ struct oxr_source_cache { struct oxr_source_state current; size_t num_inputs; struct oxr_source_input *inputs; int64_t stop_output_time; size_t num_outputs; struct oxr_source_output *outputs; }; /*! * Session input source. * * @see oxr_action */ struct oxr_source { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; struct oxr_source_cache user; struct oxr_source_cache head; struct oxr_source_cache left; struct oxr_source_cache right; struct oxr_source_cache gamepad; }; /*! * Can be one of 3 references or a space that are bound to actions. * * @obj{XrSpace} */ struct oxr_space { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Onwer 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. * * @obj{XrSwapchain} */ struct oxr_swapchain { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Onwer of this swapchain. struct oxr_session *sess; //! Compositor swapchain. struct xrt_swapchain *swapchain; //! Actual state tracked! :D int acquired_index; int released_index; 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 *); }; /*! * A group of actions. * * @obj{XrActionSet} */ struct oxr_action_set { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Onwer of this messenger. struct oxr_session *sess; /*! * Every change that is done to a action set will increment this * counter and trigger a rebinding of inputs when syncing actions. */ uint32_t generation; //! Unique key for the session hashmap. uint32_t key; }; /*! * A single action. * * @obj{XrAction} */ struct oxr_action { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Onwer of this messenger. struct oxr_action_set *act_set; //! Application supplied name of this action. char name[XR_MAX_ACTION_NAME_SIZE]; //! Unique key for the session hashmap. uint32_t 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; }; /*! * Debug object created by the client program. * * @obj{XrDebugUtilsMessengerEXT} */ struct oxr_debug_messenger { //! Common structure for things referred to by OpenXR handles. struct oxr_handle_base handle; //! Onwer of this messenger. struct oxr_instance *inst; #ifdef XR_EXT_debug_utils //! 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; #endif // XR_EXT_debug_utils }; /*! * @} */ #ifdef __cplusplus } #endif