2020-03-15 14:31:38 +00:00
|
|
|
#include "vrenvironment.hpp"
|
2020-01-09 23:10:09 +00:00
|
|
|
#include "openxrmanager.hpp"
|
|
|
|
#include "openxrmanagerimpl.hpp"
|
|
|
|
#include "../mwinput/inputmanagerimp.hpp"
|
|
|
|
|
|
|
|
#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>
|
2020-02-02 12:12:53 +00:00
|
|
|
#include <map>
|
2020-01-09 23:10:09 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
namespace MWVR
|
|
|
|
{
|
2020-02-02 12:12:53 +00:00
|
|
|
|
2020-01-09 23:10:09 +00:00
|
|
|
|
2020-02-02 12:12:53 +00:00
|
|
|
static std::vector<Timer::MeasurementContext> stats;
|
|
|
|
static std::mutex statsMutex;
|
|
|
|
static std::thread statsThread;
|
|
|
|
static bool statsThreadRunning = false;
|
|
|
|
|
|
|
|
static void statsThreadRun()
|
|
|
|
{
|
2020-05-31 11:19:26 +00:00
|
|
|
//while (statsThreadRunning)
|
|
|
|
//{
|
|
|
|
// std::stringstream ss;
|
|
|
|
// for (auto& context : stats)
|
|
|
|
// {
|
|
|
|
// for (auto& measurement : *context.second)
|
|
|
|
// {
|
|
|
|
// double ms = static_cast<double>(measurement.second) / 1000000.;
|
|
|
|
// Log(Debug::Verbose) << context.first << "." << measurement.first << ": " << ms << "ms";
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// Log(Debug::Verbose) << ss.str();
|
|
|
|
|
|
|
|
// std::this_thread::sleep_for(std::chrono::seconds(1));
|
|
|
|
//}
|
2020-02-02 12:12:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Timer::Timer(const char* name) : mName(name)
|
|
|
|
{
|
|
|
|
mLastCheckpoint = mBegin = std::chrono::steady_clock::now();
|
|
|
|
|
|
|
|
std::unique_lock<std::mutex> lock(statsMutex);
|
|
|
|
for (auto& m : stats)
|
|
|
|
{
|
|
|
|
if (m.first == mName)
|
|
|
|
mContext = m.second;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mContext == nullptr)
|
|
|
|
{
|
|
|
|
mContext = new Measures();
|
|
|
|
mContext->reserve(32);
|
|
|
|
stats.emplace_back(MeasurementContext(mName, mContext));
|
|
|
|
}
|
|
|
|
|
2020-05-31 11:19:26 +00:00
|
|
|
//if (!statsThreadRunning)
|
|
|
|
//{
|
|
|
|
// statsThreadRunning = true;
|
|
|
|
// statsThread = std::thread([] { statsThreadRun(); });
|
|
|
|
//}
|
2020-02-02 12:12:53 +00:00
|
|
|
}
|
|
|
|
Timer::~Timer()
|
|
|
|
{
|
|
|
|
//statsThreadRunning = false;
|
|
|
|
checkpoint("~");
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timer::checkpoint(const char* name)
|
|
|
|
{
|
|
|
|
auto now = std::chrono::steady_clock::now();
|
|
|
|
auto elapsed = std::chrono::duration_cast<std::chrono::nanoseconds>(now - mLastCheckpoint);
|
|
|
|
mLastCheckpoint = now;
|
|
|
|
|
|
|
|
Measure* measure = nullptr;
|
|
|
|
for (auto& m : *mContext)
|
|
|
|
{
|
|
|
|
if (m.first == name)
|
|
|
|
{
|
|
|
|
measure = &m;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!measure)
|
|
|
|
{
|
|
|
|
mContext->push_back(Measure(name, elapsed.count()));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
measure->second = measure->second * 0.95 + elapsed.count() * 0.05;
|
|
|
|
}
|
|
|
|
}
|
2020-01-09 23:10:09 +00:00
|
|
|
|
|
|
|
OpenXRManager::OpenXRManager()
|
|
|
|
: mPrivate(nullptr)
|
|
|
|
, mMutex()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
OpenXRManager::~OpenXRManager()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-02-02 12:12:53 +00:00
|
|
|
bool
|
2020-01-09 23:10:09 +00:00
|
|
|
OpenXRManager::realized()
|
|
|
|
{
|
|
|
|
return !!mPrivate;
|
|
|
|
}
|
|
|
|
|
|
|
|
long long
|
|
|
|
OpenXRManager::frameIndex()
|
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-25 13:27:13 +00:00
|
|
|
return impl().mFrameIndex;
|
2020-01-23 23:14:23 +00:00
|
|
|
return -1;
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool OpenXRManager::sessionRunning()
|
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-23 23:14:23 +00:00
|
|
|
return impl().mSessionRunning;
|
2020-01-09 23:10:09 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenXRManager::handleEvents()
|
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-23 23:14:23 +00:00
|
|
|
return impl().handleEvents();
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void OpenXRManager::waitFrame()
|
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-23 23:14:23 +00:00
|
|
|
return impl().waitFrame();
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
2020-01-26 19:06:47 +00:00
|
|
|
void OpenXRManager::beginFrame()
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-26 19:06:47 +00:00
|
|
|
return impl().beginFrame();
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
2020-05-24 16:00:42 +00:00
|
|
|
void OpenXRManager::endFrame(int64_t displayTime, int layerCount, XrCompositionLayerBaseHeader** layerStack)
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
|
|
|
if (realized())
|
2020-05-24 16:00:42 +00:00
|
|
|
return impl().endFrame(displayTime, layerCount, layerStack);
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void OpenXRManager::updateControls()
|
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-23 23:14:23 +00:00
|
|
|
return impl().updateControls();
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
OpenXRManager::realize(
|
|
|
|
osg::GraphicsContext* gc)
|
|
|
|
{
|
|
|
|
lock_guard lock(mMutex);
|
|
|
|
if (!realized())
|
|
|
|
{
|
|
|
|
gc->makeCurrent();
|
|
|
|
try {
|
|
|
|
mPrivate = std::make_shared<OpenXRManagerImpl>();
|
|
|
|
}
|
2020-02-02 12:12:53 +00:00
|
|
|
catch (std::exception & e)
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
|
|
|
Log(Debug::Error) << "Exception thrown by OpenXR: " << e.what();
|
|
|
|
osg::ref_ptr<osg::State> state = gc->getState();
|
2020-02-02 12:12:53 +00:00
|
|
|
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int OpenXRManager::eyes()
|
|
|
|
{
|
|
|
|
if (realized())
|
2020-01-23 23:14:23 +00:00
|
|
|
return impl().eyes();
|
2020-01-09 23:10:09 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
OpenXRManager::RealizeOperation::operator()(
|
|
|
|
osg::GraphicsContext* gc)
|
|
|
|
{
|
2020-03-15 14:31:38 +00:00
|
|
|
auto* xr = Environment::get().getManager();
|
2020-02-29 22:53:56 +00:00
|
|
|
xr->realize(gc);
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
2020-02-02 12:12:53 +00:00
|
|
|
bool
|
2020-01-09 23:10:09 +00:00
|
|
|
OpenXRManager::RealizeOperation::realized()
|
|
|
|
{
|
2020-03-15 14:31:38 +00:00
|
|
|
auto* xr = Environment::get().getManager();
|
2020-02-29 22:53:56 +00:00
|
|
|
return xr->realized();
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
2020-02-02 12:12:53 +00:00
|
|
|
void
|
2020-01-09 23:10:09 +00:00
|
|
|
OpenXRManager::CleanupOperation::operator()(
|
|
|
|
osg::GraphicsContext* gc)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
2020-02-14 21:11:19 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
Pose Pose::operator+(const Pose& rhs)
|
|
|
|
{
|
|
|
|
Pose pose = *this;
|
|
|
|
pose.position += this->orientation * rhs.position;
|
|
|
|
pose.orientation = rhs.orientation * this->orientation;
|
|
|
|
return pose;
|
|
|
|
}
|
2020-05-24 16:00:42 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
const Pose& Pose::operator+=(const Pose& rhs)
|
|
|
|
{
|
|
|
|
*this = *this + rhs;
|
|
|
|
return *this;
|
|
|
|
}
|
2020-05-24 16:00:42 +00:00
|
|
|
|
|
|
|
Pose Pose::operator*(float scalar)
|
|
|
|
{
|
|
|
|
Pose pose = *this;
|
|
|
|
pose.position *= scalar;
|
|
|
|
pose.velocity *= scalar;
|
|
|
|
return pose;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Pose& Pose::operator*=(float scalar)
|
|
|
|
{
|
|
|
|
*this = *this * scalar;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// near and far named with an underscore because of galaxy brain defines in included windows headers.
|
|
|
|
osg::Matrix FieldOfView::perspectiveMatrix(float near_, float far_)
|
|
|
|
{
|
|
|
|
const float tanLeft = tanf(angleLeft);
|
|
|
|
const float tanRight = tanf(angleRight);
|
|
|
|
const float tanDown = tanf(angleDown);
|
|
|
|
const float tanUp = tanf(angleUp);
|
|
|
|
|
|
|
|
const float tanWidth = tanRight - tanLeft;
|
|
|
|
const float tanHeight = tanUp - tanDown;
|
|
|
|
|
|
|
|
const float offset = near_;
|
|
|
|
|
|
|
|
float matrix[16] = {};
|
|
|
|
|
|
|
|
matrix[0] = 2 / tanWidth;
|
|
|
|
matrix[4] = 0;
|
|
|
|
matrix[8] = (tanRight + tanLeft) / tanWidth;
|
|
|
|
matrix[12] = 0;
|
|
|
|
|
|
|
|
matrix[1] = 0;
|
|
|
|
matrix[5] = 2 / tanHeight;
|
|
|
|
matrix[9] = (tanUp + tanDown) / tanHeight;
|
|
|
|
matrix[13] = 0;
|
|
|
|
|
|
|
|
if (far_ <= near_) {
|
|
|
|
matrix[2] = 0;
|
|
|
|
matrix[6] = 0;
|
|
|
|
matrix[10] = -1;
|
|
|
|
matrix[14] = -(near_ + offset);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
matrix[2] = 0;
|
|
|
|
matrix[6] = 0;
|
|
|
|
matrix[10] = -(far_ + offset) / (far_ - near_);
|
|
|
|
matrix[14] = -(far_ * (near_ + offset)) / (far_ - near_);
|
|
|
|
}
|
|
|
|
|
|
|
|
matrix[3] = 0;
|
|
|
|
matrix[7] = 0;
|
|
|
|
matrix[11] = -1;
|
|
|
|
matrix[15] = 0;
|
|
|
|
|
|
|
|
return osg::Matrix(matrix);
|
|
|
|
}
|
2020-01-23 23:14:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream& operator <<(
|
|
|
|
std::ostream& os,
|
|
|
|
const MWVR::Pose& pose)
|
|
|
|
{
|
|
|
|
os << "position=" << pose.position << " orientation=" << pose.orientation << " velocity=" << pose.velocity;
|
|
|
|
return os;
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
2020-01-23 23:14:23 +00:00
|
|
|
|