2020-05-24 16:00:42 +00:00
# include "vrenvironment.hpp"
# include "openxrmanager.hpp"
# include "openxrmanagerimpl.hpp"
# include "openxrinputmanager.hpp"
# include "openxrswapchain.hpp"
# include "../mwinput/inputmanagerimp.hpp"
# include "../mwbase/environment.hpp"
2020-06-07 18:02:03 +00:00
# include "../mwbase/statemanager.hpp"
2020-05-24 16:00:42 +00:00
# include <components/debug/debuglog.hpp>
# include <components/sdlutil/sdlgraphicswindow.hpp>
# include <Windows.h>
# include <openxr/openxr.h>
# include <openxr/openxr_platform.h>
# include <openxr/openxr_platform_defines.h>
# include <openxr/openxr_reflection.h>
# include <osg/Camera>
# include <vector>
# include <array>
# include <iostream>
# include "vrsession.hpp"
# include "vrgui.hpp"
# include <time.h>
2020-06-03 17:46:20 +00:00
# include <thread>
2020-05-24 16:00:42 +00:00
2020-06-07 18:02:03 +00:00
# ifdef max
# undef max
# endif
# ifdef min
# undef min
# endif
2020-05-24 16:00:42 +00:00
namespace MWVR
{
VRSession : : VRSession ( )
{
}
VRSession : : ~ VRSession ( )
{
}
osg : : Matrix VRSession : : projectionMatrix ( FramePhase phase , Side side )
{
assert ( ( ( int ) side ) < 2 ) ;
2020-06-03 17:46:20 +00:00
auto fov = predictedPoses ( VRSession : : FramePhase : : Update ) . view [ ( int ) side ] . fov ;
2020-05-24 16:00:42 +00:00
float near_ = Settings : : Manager : : getFloat ( " near clip " , " Camera " ) ;
float far_ = Settings : : Manager : : getFloat ( " viewing distance " , " Camera " ) ;
return fov . perspectiveMatrix ( near_ , far_ ) ;
}
osg : : Matrix VRSession : : viewMatrix ( FramePhase phase , Side side )
{
2020-06-07 18:02:03 +00:00
MWVR : : Pose pose { } ;
pose = predictedPoses ( phase ) . view [ ( int ) side ] . pose ;
if ( MWBase : : Environment : : get ( ) . getStateManager ( ) - > getState ( ) = = MWBase : : StateManager : : State_NoGame )
{
pose = predictedPoses ( phase ) . eye [ ( int ) side ] ;
osg : : Vec3 position = pose . position * Environment : : get ( ) . unitsPerMeter ( ) ;
osg : : Quat orientation = pose . orientation ;
osg : : Vec3d forward = orientation * osg : : Vec3d ( 0 , 1 , 0 ) ;
osg : : Vec3d up = orientation * osg : : Vec3d ( 0 , 0 , 1 ) ;
osg : : Matrix viewMatrix ;
viewMatrix . makeLookAt ( position , position + forward , up ) ;
return viewMatrix ;
}
2020-05-24 16:00:42 +00:00
osg : : Vec3 position = pose . position * Environment : : get ( ) . unitsPerMeter ( ) ;
osg : : Quat orientation = pose . orientation ;
float y = position . y ( ) ;
float z = position . z ( ) ;
position . y ( ) = z ;
position . z ( ) = - y ;
y = orientation . y ( ) ;
z = orientation . z ( ) ;
orientation . y ( ) = z ;
orientation . z ( ) = - y ;
osg : : Matrix viewMatrix ;
viewMatrix . setTrans ( - position ) ;
viewMatrix . postMultRotate ( orientation . conj ( ) ) ;
return viewMatrix ;
}
bool VRSession : : isRunning ( ) const {
auto * xr = Environment : : get ( ) . getManager ( ) ;
return xr - > sessionRunning ( ) ;
}
void VRSession : : swapBuffers ( osg : : GraphicsContext * gc , VRViewer & viewer )
{
Timer timer ( " VRSession::SwapBuffers " ) ;
auto * xr = Environment : : get ( ) . getManager ( ) ;
2020-06-03 17:46:20 +00:00
beginPhase ( FramePhase : : Swap ) ;
2020-05-24 16:00:42 +00:00
2020-06-08 21:54:24 +00:00
if ( getFrame ( FramePhase : : Swap ) - > mShouldRender & & isRunning ( ) )
2020-06-03 17:46:20 +00:00
{
auto leftView = viewer . mViews [ " LeftEye " ] ;
auto rightView = viewer . mViews [ " RightEye " ] ;
viewer . blitEyesToMirrorTexture ( gc ) ;
gc - > swapBuffersImplementation ( ) ;
leftView - > swapBuffers ( gc ) ;
rightView - > swapBuffers ( gc ) ;
std : : array < XrCompositionLayerProjectionView , 2 > compositionLayerProjectionViews { } ;
compositionLayerProjectionViews [ 0 ] . type = XR_TYPE_COMPOSITION_LAYER_PROJECTION_VIEW ;
compositionLayerProjectionViews [ 1 ] . type = XR_TYPE_COMPOSITION_LAYER_PROJECTION_VIEW ;
compositionLayerProjectionViews [ 0 ] . subImage = leftView - > swapchain ( ) . subImage ( ) ;
compositionLayerProjectionViews [ 1 ] . subImage = rightView - > swapchain ( ) . subImage ( ) ;
compositionLayerProjectionViews [ 0 ] . pose = toXR ( getFrame ( FramePhase : : Swap ) - > mPredictedPoses . eye [ ( int ) Side : : LEFT_HAND ] ) ;
compositionLayerProjectionViews [ 1 ] . pose = toXR ( getFrame ( FramePhase : : Swap ) - > mPredictedPoses . eye [ ( int ) Side : : RIGHT_HAND ] ) ;
compositionLayerProjectionViews [ 0 ] . fov = toXR ( getFrame ( FramePhase : : Swap ) - > mPredictedPoses . view [ ( int ) Side : : LEFT_HAND ] . fov ) ;
compositionLayerProjectionViews [ 1 ] . fov = toXR ( getFrame ( FramePhase : : Swap ) - > mPredictedPoses . view [ ( int ) Side : : RIGHT_HAND ] . fov ) ;
XrCompositionLayerProjection layer { } ;
layer . type = XR_TYPE_COMPOSITION_LAYER_PROJECTION ;
layer . space = xr - > impl ( ) . getReferenceSpace ( TrackedSpace : : STAGE ) ;
layer . viewCount = 2 ;
layer . views = compositionLayerProjectionViews . data ( ) ;
auto * layerStack = reinterpret_cast < XrCompositionLayerBaseHeader * > ( & layer ) ;
2020-06-07 18:02:03 +00:00
Log ( Debug : : Debug ) < < getFrame ( FramePhase : : Swap ) - > mFrameNo < < " : EndFrame " < < std : : this_thread : : get_id ( ) ;
2020-06-03 17:46:20 +00:00
xr - > endFrame ( getFrame ( FramePhase : : Swap ) - > mPredictedDisplayTime , 1 , & layerStack ) ;
2020-05-24 16:00:42 +00:00
}
2020-06-03 17:46:20 +00:00
{
2020-06-07 18:02:03 +00:00
std : : unique_lock < std : : mutex > lock ( mMutex ) ;
// Some of these values are useless until the prediction time bug is resolved by oculus.
//auto xrPredictionChange = xr->impl().frameState().predictedDisplayTime - mLastPredictedDisplayTime;
mLastPredictedDisplayTime = xr - > impl ( ) . frameState ( ) . predictedDisplayTime ;
mLastPredictedDisplayPeriod = xr - > impl ( ) . frameState ( ) . predictedDisplayPeriod ;
auto now = std : : chrono : : steady_clock : : now ( ) ;
mLastFrameInterval = std : : chrono : : duration_cast < std : : chrono : : nanoseconds > ( now - mLastRenderedFrameTimestamp ) ;
mLastRenderedFrameTimestamp = now ;
mLastRenderedFrame = getFrame ( FramePhase : : Swap ) - > mFrameNo ;
//Log(Debug::Debug) << getFrame(FramePhase::Swap)->mFrameNo << ": xrPrediction=" << xr->impl().frameState().predictedDisplayTime << ", ourPrediction=" << getFrame(FramePhase::Swap)->mPredictedDisplayTime << ", miss=" << miss << "ms";
//Log(Debug::Debug) << "xrPredictionChange=" << (xrPredictionChange / 1000000) << "ms";
//Log(Debug::Debug) << "xrPredictionPeriod=" << (mLastPredictedDisplayPeriod / 1000000) << "ms";
// Just a quick averaging fps over some time rather than just the instantaneous.
auto seconds = std : : chrono : : duration_cast < std : : chrono : : duration < double > > ( now - mStart ) . count ( ) ;
static int sBaseFrames = 0 ;
if ( seconds > 10.f )
{
Log ( Debug : : Debug ) < < " Fps: " < < ( static_cast < double > ( mLastRenderedFrame - sBaseFrames ) / seconds ) ;
mStart = now ;
sBaseFrames = mLastRenderedFrame ;
}
2020-06-03 17:46:20 +00:00
2020-06-07 18:02:03 +00:00
getFrame ( FramePhase : : Swap ) = nullptr ;
mFramesInFlight - - ;
}
2020-06-03 17:46:20 +00:00
mCondition . notify_one ( ) ;
2020-05-24 16:00:42 +00:00
}
2020-06-03 17:46:20 +00:00
void VRSession : : beginPhase ( FramePhase phase )
2020-05-24 16:00:42 +00:00
{
Timer timer ( " VRSession::advanceFrame " ) ;
2020-06-07 18:02:03 +00:00
Log ( Debug : : Debug ) < < " beginPhase( " < < ( ( int ) phase ) < < " ) " < < std : : this_thread : : get_id ( ) ;
2020-05-24 16:00:42 +00:00
2020-06-07 18:02:03 +00:00
if ( getFrame ( phase ) )
{
Log ( Debug : : Warning ) < < " advanceFramePhase called with a frame alreay in the target phase " ;
return ;
}
if ( phase = = FramePhase : : Update )
{
prepareFrame ( ) ;
}
else
2020-06-03 17:46:20 +00:00
{
std : : unique_lock < std : : mutex > lock ( mMutex ) ;
FramePhase previousPhase = static_cast < FramePhase > ( ( int ) phase - 1 ) ;
if ( ! getFrame ( previousPhase ) )
2020-06-07 18:02:03 +00:00
throw std : : logic_error ( " beginPhase called without a frame " ) ;
2020-06-03 17:46:20 +00:00
getFrame ( phase ) = std : : move ( getFrame ( previousPhase ) ) ;
}
2020-06-07 18:02:03 +00:00
// TODO: Invokation should depend on earliest render rather than necessarily phase.
// Specifically. Without shadows this is fine because nothing is being rendered
// during cull or earlier.
// Thought: Add an Shadowmapping phase and invoke it from the shadow code
// But with shadows rendering occurs during cull and we must do frame sync before those calls.
// If you want to pay the FPS toll and play with shadows, change FramePhase::Draw to FramePhase::Cull or enjoy your eyes getting torn apart by jitters.
if ( phase = = FramePhase : : Draw & & getFrame ( phase ) - > mShouldRender )
doFrameSync ( ) ;
}
void VRSession : : doFrameSync ( )
{
2020-06-03 17:46:20 +00:00
{
2020-06-07 18:02:03 +00:00
std : : unique_lock < std : : mutex > lock ( mMutex ) ;
while ( mLastRenderedFrame ! = mFrames - 1 )
2020-06-03 17:46:20 +00:00
{
mCondition . wait ( lock ) ;
}
}
2020-06-07 18:02:03 +00:00
auto * xr = Environment : : get ( ) . getManager ( ) ;
Log ( Debug : : Debug ) < < mFrames < < " : WaitFrame " < < std : : this_thread : : get_id ( ) ;
xr - > waitFrame ( ) ;
Log ( Debug : : Debug ) < < mFrames < < " : BeginFrame " < < std : : this_thread : : get_id ( ) ;
xr - > beginFrame ( ) ;
2020-06-03 17:46:20 +00:00
}
std : : unique_ptr < VRSession : : VRFrame > & VRSession : : getFrame ( FramePhase phase )
{
if ( ( unsigned int ) phase > = mFrame . size ( ) )
throw std : : logic_error ( " Invalid frame phase " ) ;
return mFrame [ ( int ) phase ] ;
2020-05-24 16:00:42 +00:00
}
2020-06-03 17:46:20 +00:00
void VRSession : : prepareFrame ( )
2020-05-24 16:00:42 +00:00
{
2020-06-07 18:02:03 +00:00
2020-05-24 16:00:42 +00:00
std : : unique_lock < std : : mutex > lock ( mMutex ) ;
2020-06-07 18:02:03 +00:00
mFrames + + ;
2020-05-24 16:00:42 +00:00
assert ( ! mPredrawFrame ) ;
Timer timer ( " VRSession::startFrame " ) ;
auto * xr = Environment : : get ( ) . getManager ( ) ;
2020-06-03 17:46:20 +00:00
xr - > handleEvents ( ) ;
2020-06-07 18:02:03 +00:00
2020-05-24 16:00:42 +00:00
auto frameState = xr - > impl ( ) . frameState ( ) ;
2020-06-07 18:02:03 +00:00
// auto predictedDisplayTime = frameState.predictedDisplayTime;
// if (predictedDisplayTime == 0)
// {
// // First time, need to invent a frame time since openxr won't help us without calling waitframe.
// predictedDisplayTime = std::chrono::duration_cast<std::chrono::nanoseconds>(std::chrono::steady_clock::now().time_since_epoch()).count();
// }
// else
// {
// // Predict display time based on real framerate
// float intervalsf = static_cast<double>(mLastFrameInterval.count()) / static_cast<double>(mLastPredictedDisplayPeriod);
// int intervals = std::max((int)std::roundf(intervalsf), 1);
// predictedDisplayTime = mLastPredictedDisplayTime + intervals * (mFrames - mLastRenderedFrame) * mLastPredictedDisplayPeriod;
// }
// TODO:
//////////////////////// OCULUS BUG
//////////////////// Oculus will suddenly start monotonically increasing their predicted display time by precisely 1 second
//////////////////// regardless of real time passed, causing predictions to go crazy due to the time difference.
//////////////////// Therefore, for the time being, i ignore oculus' predicted display time altogether.
long long predictedDisplayTime = std : : chrono : : duration_cast < std : : chrono : : nanoseconds > ( std : : chrono : : steady_clock : : now ( ) . time_since_epoch ( ) ) . count ( ) ;
if ( mFrames > 1 )
2020-05-24 16:00:42 +00:00
{
2020-06-07 18:02:03 +00:00
float intervalsf = static_cast < double > ( mLastFrameInterval . count ( ) ) / static_cast < double > ( frameState . predictedDisplayPeriod ) ;
int intervals = std : : max ( ( int ) std : : roundf ( intervalsf ) , 1 ) ;
predictedDisplayTime = predictedDisplayTime + intervals * ( mFrames - mLastRenderedFrame ) * frameState . predictedDisplayPeriod ;
2020-05-24 16:00:42 +00:00
}
2020-06-07 18:02:03 +00:00
2020-05-24 16:00:42 +00:00
PoseSet predictedPoses { } ;
2020-06-07 18:02:03 +00:00
xr - > impl ( ) . enablePredictions ( ) ;
predictedPoses . head = xr - > impl ( ) . getPredictedLimbPose ( predictedDisplayTime , TrackedLimb : : HEAD , TrackedSpace : : STAGE ) * mPlayerScale ;
auto hmdViews = xr - > impl ( ) . getPredictedViews ( predictedDisplayTime , TrackedSpace : : VIEW ) ;
predictedPoses . view [ ( int ) Side : : LEFT_HAND ] . pose = fromXR ( hmdViews [ ( int ) Side : : LEFT_HAND ] . pose ) * mPlayerScale ;
predictedPoses . view [ ( int ) Side : : RIGHT_HAND ] . pose = fromXR ( hmdViews [ ( int ) Side : : RIGHT_HAND ] . pose ) * mPlayerScale ;
predictedPoses . view [ ( int ) Side : : LEFT_HAND ] . fov = fromXR ( hmdViews [ ( int ) Side : : LEFT_HAND ] . fov ) ;
predictedPoses . view [ ( int ) Side : : RIGHT_HAND ] . fov = fromXR ( hmdViews [ ( int ) Side : : RIGHT_HAND ] . fov ) ;
auto stageViews = xr - > impl ( ) . getPredictedViews ( predictedDisplayTime , TrackedSpace : : STAGE ) ;
predictedPoses . eye [ ( int ) Side : : LEFT_HAND ] = fromXR ( stageViews [ ( int ) Side : : LEFT_HAND ] . pose ) * mPlayerScale ;
predictedPoses . eye [ ( int ) Side : : RIGHT_HAND ] = fromXR ( stageViews [ ( int ) Side : : RIGHT_HAND ] . pose ) * mPlayerScale ;
auto * input = Environment : : get ( ) . getInputManager ( ) ;
if ( input )
2020-05-24 16:00:42 +00:00
{
2020-06-07 18:02:03 +00:00
predictedPoses . hands [ ( int ) Side : : LEFT_HAND ] = input - > getHandPose ( predictedDisplayTime , TrackedSpace : : STAGE , Side : : LEFT_HAND ) * mPlayerScale ;
predictedPoses . hands [ ( int ) Side : : RIGHT_HAND ] = input - > getHandPose ( predictedDisplayTime , TrackedSpace : : STAGE , Side : : RIGHT_HAND ) * mPlayerScale ;
2020-06-03 17:46:20 +00:00
}
2020-06-07 18:02:03 +00:00
xr - > impl ( ) . disablePredictions ( ) ;
2020-05-24 16:00:42 +00:00
2020-06-03 17:46:20 +00:00
auto & frame = getFrame ( FramePhase : : Update ) ;
frame . reset ( new VRFrame ) ;
frame - > mPredictedDisplayTime = predictedDisplayTime ;
frame - > mFrameNo = mFrames ;
frame - > mPredictedPoses = predictedPoses ;
frame - > mShouldRender = isRunning ( ) ;
2020-06-07 18:02:03 +00:00
mFramesInFlight + + ;
2020-05-24 16:00:42 +00:00
}
const PoseSet & VRSession : : predictedPoses ( FramePhase phase )
{
2020-06-03 17:46:20 +00:00
auto & frame = getFrame ( phase ) ;
if ( phase = = FramePhase : : Update & & ! frame )
beginPhase ( FramePhase : : Update ) ;
2020-05-24 16:00:42 +00:00
2020-06-03 17:46:20 +00:00
if ( ! frame )
throw std : : logic_error ( " Attempted to get poses from a phase with no current pose " ) ;
return frame - > mPredictedPoses ;
2020-05-24 16:00:42 +00:00
}
// OSG doesn't provide API to extract euler angles from a quat, but i need it.
// Credits goes to Dennis Bunfield, i just copied his formula https://narkive.com/v0re6547.4
void getEulerAngles ( const osg : : Quat & quat , float & yaw , float & pitch , float & roll )
{
// Now do the computation
osg : : Matrixd m2 ( osg : : Matrixd : : rotate ( quat ) ) ;
double * mat = ( double * ) m2 . ptr ( ) ;
double angle_x = 0.0 ;
double angle_y = 0.0 ;
double angle_z = 0.0 ;
double D , C , tr_x , tr_y ;
angle_y = D = asin ( mat [ 2 ] ) ; /* Calculate Y-axis angle */
C = cos ( angle_y ) ;
if ( fabs ( C ) > 0.005 ) /* Test for Gimball lock? */
{
tr_x = mat [ 10 ] / C ; /* No, so get X-axis angle */
tr_y = - mat [ 6 ] / C ;
angle_x = atan2 ( tr_y , tr_x ) ;
tr_x = mat [ 0 ] / C ; /* Get Z-axis angle */
tr_y = - mat [ 1 ] / C ;
angle_z = atan2 ( tr_y , tr_x ) ;
}
else /* Gimball lock has occurred */
{
angle_x = 0 ; /* Set X-axis angle to zero
*/
tr_x = mat [ 5 ] ; /* And calculate Z-axis angle
*/
tr_y = mat [ 4 ] ;
angle_z = atan2 ( tr_y , tr_x ) ;
}
yaw = angle_z ;
pitch = angle_x ;
roll = angle_y ;
}
void VRSession : : movementAngles ( float & yaw , float & pitch )
{
assert ( mPredrawFrame ) ;
auto * input = Environment : : get ( ) . getInputManager ( ) ;
2020-06-03 17:46:20 +00:00
// TODO: This strictly speaking violates the rule of not making predictions outside of prepareFrame()
// I should either add VIEW hands to the predicted pose set, or compute this using STAGE poses
// It would likely suffice to compute euler angles for STAGE head and hand and return the difference?
2020-06-07 18:02:03 +00:00
if ( ! getFrame ( FramePhase : : Update ) )
beginPhase ( FramePhase : : Update ) ;
2020-06-03 17:46:20 +00:00
auto lhandquat = input - > getHandPose ( getFrame ( FramePhase : : Update ) - > mPredictedDisplayTime , TrackedSpace : : VIEW , Side : : LEFT_HAND ) . orientation ;
2020-05-24 16:00:42 +00:00
float roll = 0.f ;
getEulerAngles ( lhandquat , yaw , pitch , roll ) ;
}
}
std : : ostream & operator < < (
std : : ostream & os ,
const MWVR : : Pose & pose )
{
os < < " position= " < < pose . position < < " orientation= " < < pose . orientation < < " velocity= " < < pose . velocity ;
return os ;
}