|
|
|
@ -474,11 +474,17 @@ namespace MWVR
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
OpenXRManagerImpl::endFrame(FrameInfo frameInfo, int layerCount, const std::array<CompositionLayerProjectionView, 2>& layerStack)
|
|
|
|
|
OpenXRManagerImpl::endFrame(FrameInfo frameInfo, const std::array<CompositionLayerProjectionView, 2>* layerStack)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
XrFrameEndInfo frameEndInfo{ XR_TYPE_FRAME_END_INFO };
|
|
|
|
|
frameEndInfo.displayTime = frameInfo.runtimePredictedDisplayTime;
|
|
|
|
|
frameEndInfo.environmentBlendMode = mEnvironmentBlendMode;
|
|
|
|
|
if (layerStack)
|
|
|
|
|
{
|
|
|
|
|
std::array<XrCompositionLayerProjectionView, 2> compositionLayerProjectionViews{};
|
|
|
|
|
compositionLayerProjectionViews[(int)Side::LEFT_SIDE] = toXR(layerStack[(int)Side::LEFT_SIDE]);
|
|
|
|
|
compositionLayerProjectionViews[(int)Side::RIGHT_SIDE] = toXR(layerStack[(int)Side::RIGHT_SIDE]);
|
|
|
|
|
compositionLayerProjectionViews[(int)Side::LEFT_SIDE] = toXR((*layerStack)[(int)Side::LEFT_SIDE]);
|
|
|
|
|
compositionLayerProjectionViews[(int)Side::RIGHT_SIDE] = toXR((*layerStack)[(int)Side::RIGHT_SIDE]);
|
|
|
|
|
XrCompositionLayerProjection layer{};
|
|
|
|
|
layer.type = XR_TYPE_COMPOSITION_LAYER_PROJECTION;
|
|
|
|
|
layer.space = getReferenceSpace(ReferenceSpace::STAGE);
|
|
|
|
@ -493,8 +499,8 @@ namespace MWVR
|
|
|
|
|
// All values not set here are set previously as they are constant
|
|
|
|
|
compositionLayerDepth[(int)Side::LEFT_SIDE].farZ = farClip;
|
|
|
|
|
compositionLayerDepth[(int)Side::RIGHT_SIDE].farZ = farClip;
|
|
|
|
|
compositionLayerDepth[(int)Side::LEFT_SIDE].subImage = layerStack[(int)Side::LEFT_SIDE].swapchain->impl().xrSubImageDepth();
|
|
|
|
|
compositionLayerDepth[(int)Side::RIGHT_SIDE].subImage = layerStack[(int)Side::RIGHT_SIDE].swapchain->impl().xrSubImageDepth();
|
|
|
|
|
compositionLayerDepth[(int)Side::LEFT_SIDE].subImage = (*layerStack)[(int)Side::LEFT_SIDE].swapchain->impl().xrSubImageDepth();
|
|
|
|
|
compositionLayerDepth[(int)Side::RIGHT_SIDE].subImage = (*layerStack)[(int)Side::RIGHT_SIDE].swapchain->impl().xrSubImageDepth();
|
|
|
|
|
if (compositionLayerDepth[(int)Side::LEFT_SIDE].subImage.swapchain != XR_NULL_HANDLE
|
|
|
|
|
&& compositionLayerDepth[(int)Side::RIGHT_SIDE].subImage.swapchain != XR_NULL_HANDLE)
|
|
|
|
|
{
|
|
|
|
@ -502,13 +508,7 @@ namespace MWVR
|
|
|
|
|
compositionLayerProjectionViews[(int)Side::RIGHT_SIDE].next = &compositionLayerDepth[(int)Side::RIGHT_SIDE];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
XrFrameEndInfo frameEndInfo{ XR_TYPE_FRAME_END_INFO };
|
|
|
|
|
frameEndInfo.displayTime = frameInfo.runtimePredictedDisplayTime;
|
|
|
|
|
frameEndInfo.environmentBlendMode = mEnvironmentBlendMode;
|
|
|
|
|
//if (frameInfo.runtimeRequestsRender)
|
|
|
|
|
{
|
|
|
|
|
frameEndInfo.layerCount = layerCount;
|
|
|
|
|
frameEndInfo.layerCount = 1;
|
|
|
|
|
frameEndInfo.layers = &xrLayerStack;
|
|
|
|
|
}
|
|
|
|
|
CHECK_XRCMD(xrEndFrame(mSession, &frameEndInfo));
|
|
|
|
@ -602,6 +602,15 @@ namespace MWVR
|
|
|
|
|
}
|
|
|
|
|
popEvent();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mXrSessionShouldStop)
|
|
|
|
|
{
|
|
|
|
|
if (checkStopCondition())
|
|
|
|
|
{
|
|
|
|
|
CHECK_XRCMD(xrEndSession(mSession));
|
|
|
|
|
mXrSessionShouldStop = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const XrEventDataBaseHeader* OpenXRManagerImpl::nextEvent()
|
|
|
|
@ -638,50 +647,70 @@ namespace MWVR
|
|
|
|
|
OpenXRManagerImpl::handleSessionStateChanged(
|
|
|
|
|
const XrEventDataSessionStateChanged& stateChangedEvent)
|
|
|
|
|
{
|
|
|
|
|
auto oldState = mSessionState;
|
|
|
|
|
auto newState = stateChangedEvent.state;
|
|
|
|
|
Log(Debug::Verbose) << "XrEventDataSessionStateChanged: state " << to_string(oldState) << "->" << to_string(newState);
|
|
|
|
|
bool success = true;
|
|
|
|
|
Log(Debug::Verbose) << "XrEventDataSessionStateChanged: state " << to_string(mSessionState) << "->" << to_string(stateChangedEvent.state);
|
|
|
|
|
mSessionState = stateChangedEvent.state;
|
|
|
|
|
|
|
|
|
|
switch (newState)
|
|
|
|
|
// Ref: https://www.khronos.org/registry/OpenXR/specs/1.0/html/xrspec.html#session-states
|
|
|
|
|
switch (mSessionState)
|
|
|
|
|
{
|
|
|
|
|
case XR_SESSION_STATE_IDLE:
|
|
|
|
|
{
|
|
|
|
|
mAppShouldSyncFrameLoop = false;
|
|
|
|
|
mAppShouldRender = false;
|
|
|
|
|
mAppShouldReadInput = false;
|
|
|
|
|
mXrSessionShouldStop = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case XR_SESSION_STATE_READY:
|
|
|
|
|
{
|
|
|
|
|
mAppShouldSyncFrameLoop = true;
|
|
|
|
|
mAppShouldRender = false;
|
|
|
|
|
mAppShouldReadInput = false;
|
|
|
|
|
mXrSessionShouldStop = false;
|
|
|
|
|
|
|
|
|
|
XrSessionBeginInfo beginInfo{ XR_TYPE_SESSION_BEGIN_INFO };
|
|
|
|
|
beginInfo.primaryViewConfigurationType = mViewConfigType;
|
|
|
|
|
CHECK_XRCMD(xrBeginSession(mSession, &beginInfo));
|
|
|
|
|
mSessionRunning = true;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case XR_SESSION_STATE_STOPPING:
|
|
|
|
|
{
|
|
|
|
|
if (checkStopCondition())
|
|
|
|
|
{
|
|
|
|
|
CHECK_XRCMD(xrEndSession(mSession));
|
|
|
|
|
mSessionStopRequested = false;
|
|
|
|
|
mSessionRunning = false;
|
|
|
|
|
mAppShouldSyncFrameLoop = false;
|
|
|
|
|
mAppShouldRender = false;
|
|
|
|
|
mAppShouldReadInput = false;
|
|
|
|
|
mXrSessionShouldStop = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
case XR_SESSION_STATE_SYNCHRONIZED:
|
|
|
|
|
{
|
|
|
|
|
mSessionStopRequested = true;
|
|
|
|
|
success = false;
|
|
|
|
|
}
|
|
|
|
|
mAppShouldSyncFrameLoop = true;
|
|
|
|
|
mAppShouldRender = false;
|
|
|
|
|
mAppShouldReadInput = false;
|
|
|
|
|
mXrSessionShouldStop = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
Log(Debug::Verbose) << "XrEventDataSessionStateChanged: Ignoring new state " << to_string(newState);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
|
case XR_SESSION_STATE_VISIBLE:
|
|
|
|
|
{
|
|
|
|
|
mSessionState = newState;
|
|
|
|
|
mAppShouldSyncFrameLoop = true;
|
|
|
|
|
mAppShouldRender = true;
|
|
|
|
|
mAppShouldReadInput = false;
|
|
|
|
|
mXrSessionShouldStop = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
case XR_SESSION_STATE_FOCUSED:
|
|
|
|
|
{
|
|
|
|
|
Log(Debug::Verbose) << "XrEventDataSessionStateChanged: Conditions for state " << to_string(newState) << " not met, retrying next frame";
|
|
|
|
|
mAppShouldSyncFrameLoop = true;
|
|
|
|
|
mAppShouldRender = true;
|
|
|
|
|
mAppShouldReadInput = true;
|
|
|
|
|
mXrSessionShouldStop = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
Log(Debug::Warning) << "XrEventDataSessionStateChanged: Ignoring new state " << to_string(mSessionState);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return success;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool OpenXRManagerImpl::checkStopCondition()
|
|
|
|
@ -760,11 +789,6 @@ namespace MWVR
|
|
|
|
|
return mEnabledExtensions.count(extensionName) != 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool OpenXRManagerImpl::xrSessionCanRender()
|
|
|
|
|
{
|
|
|
|
|
return xrSessionRunning() && !xrSessionStopRequested();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OpenXRManagerImpl::xrResourceAcquired()
|
|
|
|
|
{
|
|
|
|
|
mAcquiredResources++;
|
|
|
|
@ -792,11 +816,6 @@ namespace MWVR
|
|
|
|
|
return function;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool OpenXRManagerImpl::xrSessionStopRequested()
|
|
|
|
|
{
|
|
|
|
|
return mSessionStopRequested;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OpenXRManagerImpl::enablePredictions()
|
|
|
|
|
{
|
|
|
|
|
mPredictionsEnabled = true;
|
|
|
|
|