#ifndef OPENXR_MANAGER_IMPL_HPP #define OPENXR_MANAGER_IMPL_HPP #include "openxrmanager.hpp" #include "../mwinput/inputmanagerimp.hpp" #include #include #include #include #include #include #include #include #include #include namespace MWVR { // Error management macros and functions. Should be used on every openxr call. #define CHK_STRINGIFY(x) #x #define TOSTRING(x) CHK_STRINGIFY(x) #define FILE_AND_LINE __FILE__ ":" TOSTRING(__LINE__) #define CHECK_XRCMD(cmd) CheckXrResult(cmd, #cmd, FILE_AND_LINE); #define CHECK_XRRESULT(res, cmdStr) CheckXrResult(res, cmdStr, FILE_AND_LINE); XrResult CheckXrResult(XrResult res, const char* originator = nullptr, const char* sourceLocation = nullptr); std::string XrResultString(XrResult res); /// Conversion methods from openxr types to osg/mwvr types. Includes managing the differing conventions. MWVR::Pose fromXR(XrPosef pose); MWVR::FieldOfView fromXR(XrFovf fov); osg::Vec3 fromXR(XrVector3f); osg::Quat fromXR(XrQuaternionf quat); /// Conversion methods from osg/mwvr types to openxr types. Includes managing the differing conventions. XrPosef toXR(MWVR::Pose pose); XrFovf toXR(MWVR::FieldOfView fov); XrVector3f toXR(osg::Vec3 v); XrQuaternionf toXR(osg::Quat quat); XrCompositionLayerProjectionView toXR(MWVR::CompositionLayerProjectionView layer); /// \brief Implementation of OpenXRManager struct OpenXRManagerImpl { OpenXRManagerImpl(void); ~OpenXRManagerImpl(void); FrameInfo waitFrame(); void beginFrame(); void endFrame(FrameInfo frameInfo, const std::array* layerStack); bool appShouldSyncFrameLoop() const { return mAppShouldSyncFrameLoop; } bool appShouldRender() const { return mAppShouldRender; } bool appShouldReadInput() const { return mAppShouldReadInput; } std::array getPredictedViews(int64_t predictedDisplayTime, ReferenceSpace space); MWVR::Pose getPredictedHeadPose(int64_t predictedDisplayTime, ReferenceSpace space); void handleEvents(); void enablePredictions(); void disablePredictions(); long long getLastPredictedDisplayTime(); long long getLastPredictedDisplayPeriod(); std::array getRecommendedSwapchainConfig() const; XrSpace getReferenceSpace(ReferenceSpace space); XrSession xrSession() const { return mSession; }; XrInstance xrInstance() const { return mInstance; }; bool xrExtensionIsEnabled(const char* extensionName) const; void xrResourceAcquired(); void xrResourceReleased(); void xrUpdateNames(); PFN_xrVoidFunction xrGetFunction(const std::string& name); protected: void setupExtensionsAndLayers(); void enableExtension(const std::string& extension, bool optional); void setupDebugMessenger(void); void logLayersAndExtensions(); void LogInstanceInfo(); void LogReferenceSpaces(); void LogSwapchainFormats(); bool xrNextEvent(XrEventDataBuffer& eventBuffer); void xrQueueEvents(); const XrEventDataBaseHeader* nextEvent(); bool processEvent(const XrEventDataBaseHeader* header); void popEvent(); bool handleSessionStateChanged(const XrEventDataSessionStateChanged& stateChangedEvent); bool checkStopCondition(); private: bool initialized = false; bool mPredictionsEnabled = false; XrInstance mInstance = XR_NULL_HANDLE; XrSession mSession = XR_NULL_HANDLE; XrSpace mSpace = XR_NULL_HANDLE; XrFormFactor mFormFactor = XR_FORM_FACTOR_HEAD_MOUNTED_DISPLAY; XrViewConfigurationType mViewConfigType = XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO; XrEnvironmentBlendMode mEnvironmentBlendMode = XR_ENVIRONMENT_BLEND_MODE_OPAQUE; XrSystemId mSystemId = XR_NULL_SYSTEM_ID; XrSystemProperties mSystemProperties{ XR_TYPE_SYSTEM_PROPERTIES }; std::array mConfigViews{ { {XR_TYPE_VIEW_CONFIGURATION_VIEW}, {XR_TYPE_VIEW_CONFIGURATION_VIEW} } }; XrSpace mReferenceSpaceView = XR_NULL_HANDLE; XrSpace mReferenceSpaceStage = XR_NULL_HANDLE; XrFrameState mFrameState{}; XrSessionState mSessionState = XR_SESSION_STATE_UNKNOWN; XrDebugUtilsMessengerEXT mDebugMessenger{ nullptr }; bool mXrSessionShouldStop = false; bool mAppShouldSyncFrameLoop = false; bool mAppShouldRender = false; bool mAppShouldReadInput = false; uint32_t mAcquiredResources = 0; std::mutex mFrameStateMutex{}; std::mutex mEventMutex{}; std::set mAvailableExtensions; std::set mEnabledExtensions; std::queue mEventQueue; std::array mLayerDepth; }; } #endif