1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-16 00:49:54 +00:00
openmw-tes3mp/apps/openmw/mwvr/vrviewer.cpp

274 lines
9.9 KiB
C++
Raw Normal View History

#include "vrviewer.hpp"
#include "openxrmanagerimpl.hpp"
2020-03-15 14:31:38 +00:00
#include "vrenvironment.hpp"
#include "vrsession.hpp"
2020-06-28 09:33:01 +00:00
#include "vrframebuffer.hpp"
#include "vrview.hpp"
#include "../mwrender/vismask.hpp"
#include <osgViewer/Renderer>
#include <components/sceneutil/mwshadowtechnique.hpp>
2020-08-07 21:33:21 +00:00
#include <components/misc/stringops.hpp>
namespace MWVR
{
const std::array<const char*, 2> VRViewer::sViewNames = {
"LeftEye",
"RightEye"
};
2020-06-28 09:33:01 +00:00
// Callback to do construction with a graphics context
class RealizeOperation : public osg::GraphicsOperation
{
public:
RealizeOperation() : osg::GraphicsOperation("VRRealizeOperation", false) {};
void operator()(osg::GraphicsContext* gc) override;
bool realized();
private:
};
VRViewer::VRViewer(
2020-02-23 10:02:38 +00:00
osg::ref_ptr<osgViewer::Viewer> viewer)
2020-06-28 09:33:01 +00:00
: mViewer(viewer)
, mPreDraw(new PredrawCallback(this))
, mPostDraw(new PostdrawCallback(this))
2020-07-26 11:12:36 +00:00
, mVrShadow()
2020-02-15 19:01:11 +00:00
, mConfigured(false)
{
2020-06-28 09:33:01 +00:00
mViewer->setRealizeOperation(new RealizeOperation());
}
VRViewer::~VRViewer(void)
{
}
void VRViewer::traversals()
{
mViewer->updateTraversal();
mViewer->renderingTraversals();
}
2020-08-07 21:33:21 +00:00
int parseResolution(std::string conf, int recommended, int max)
{
if (Misc::StringUtils::isNumber(conf))
{
int res = std::atoi(conf.c_str());
if (res <= 0)
return recommended;
if (res > max)
return max;
return res;
}
conf = Misc::StringUtils::lowerCase(conf);
if (conf == "auto" || conf == "recommended")
{
return recommended;
}
if (conf == "max")
{
return max;
}
}
void VRViewer::realize(osg::GraphicsContext* context)
{
std::unique_lock<std::mutex> lock(mMutex);
if (mConfigured)
{
return;
}
// Give the main camera an initial draw callback that disables camera setup (we don't want it)
auto mainCamera = mCameras["MainCamera"] = mViewer->getCamera();
mainCamera->setName("Main");
mainCamera->setInitialDrawCallback(new VRView::InitialDrawCallback());
2020-03-15 14:31:38 +00:00
auto* xr = Environment::get().getManager();
2020-06-28 09:33:01 +00:00
xr->realize(context);
2020-02-29 22:53:56 +00:00
// Run through initial events to start session
// For the rest of runtime this is handled by vrsession
xr->handleEvents();
// Small feature culling
bool smallFeatureCulling = Settings::Manager::getBool("small feature culling", "Camera");
auto smallFeatureCullingPixelSize = Settings::Manager::getFloat("small feature culling pixel size", "Camera");
osg::Camera::CullingMode cullingMode = osg::Camera::DEFAULT_CULLING | osg::Camera::FAR_PLANE_CULLING;
if (!smallFeatureCulling)
cullingMode &= ~osg::CullStack::SMALL_FEATURE_CULLING;
else
cullingMode |= osg::CullStack::SMALL_FEATURE_CULLING;
// Configure eyes, their cameras, and their enslavement.
osg::Vec4 clearColor = mainCamera->getClearColor();
auto config = xr->getRecommendedSwapchainConfig();
2020-06-28 09:33:01 +00:00
bool mirror = Settings::Manager::getBool("mirror texture", "VR");
2020-08-07 21:33:21 +00:00
std::string mirrorTextureEyeString = Settings::Manager::getString("mirror texture eye", "VR");
mirrorTextureEyeString = Misc::StringUtils::lowerCase(mirrorTextureEyeString);
if (mirrorTextureEyeString == "left" || mirrorTextureEyeString == "both")
mMirrorTextureViews.push_back(sViewNames[(int)Side::LEFT_SIDE]);
if (mirrorTextureEyeString == "right" || mirrorTextureEyeString == "both")
mMirrorTextureViews.push_back(sViewNames[(int)Side::RIGHT_SIDE]);
if (Settings::Manager::getBool("flip mirror texture order", "VR"))
std::reverse(mMirrorTextureViews.begin(), mMirrorTextureViews.end());
2020-06-28 09:33:01 +00:00
// TODO: If mirror is false either hide the window or paste something meaningful into it.
// E.g. Fanart of Dagoth UR wearing a VR headset
2020-08-07 21:33:21 +00:00
std::array<std::string, 2> xConfString;
std::array<std::string, 2> yConfString;
xConfString[0] = Settings::Manager::getString("left eye resolution x", "VR");
yConfString[0] = Settings::Manager::getString("left eye resolution y", "VR");
xConfString[1] = Settings::Manager::getString("right eye resolution x", "VR");
yConfString[1] = Settings::Manager::getString("right eye resolution y", "VR");
for (unsigned i = 0; i < sViewNames.size(); i++)
{
2020-08-07 21:33:21 +00:00
auto name = sViewNames[i];
config[i].selectedWidth = parseResolution(xConfString[i], config[i].recommendedWidth, config[i].maxWidth);
config[i].selectedHeight = parseResolution(yConfString[i], config[i].recommendedHeight, config[i].maxHeight);
config[i].selectedSamples = Settings::Manager::getInt("antialiasing", "Video");
// OpenXR requires a non-zero value
if (config[i].selectedSamples < 1)
config[i].selectedSamples = 1;
Log(Debug::Verbose) << name << " resolution: Recommended x=" << config[i].recommendedWidth << ", y=" << config[i].recommendedHeight;
Log(Debug::Verbose) << name << " resolution: Max x=" << config[i].maxWidth << ", y=" << config[i].maxHeight;
Log(Debug::Verbose) << name << " resolution: Selected x=" << config[i].selectedWidth << ", y=" << config[i].selectedHeight;
auto view = new VRView(name, config[i], context->getState());
mViews[name] = view;
auto camera = mCameras[name] = view->createCamera(i + 2, clearColor, context);
camera->setPreDrawCallback(mPreDraw);
camera->setFinalDrawCallback(mPostDraw);
camera->setCullMask(~MWRender::Mask_GUI & ~MWRender::Mask_SimpleWater & ~MWRender::Mask_UpdateVisitor);
2020-08-07 21:33:21 +00:00
camera->setName(name);
if (smallFeatureCulling)
camera->setSmallFeatureCullingPixelSize(smallFeatureCullingPixelSize);
camera->setCullingMode(cullingMode);
mViewer->addSlave(camera, true);
auto* slave = mViewer->findSlaveForCamera(camera);
2020-07-26 11:12:36 +00:00
slave->_updateSlaveCallback = new VRView::UpdateSlaveCallback(view);
2020-06-28 09:33:01 +00:00
if (mirror)
mMsaaResolveMirrorTexture[i].reset(new VRFramebuffer(context->getState(),
view->swapchain().width(),
view->swapchain().height(),
0));
2020-07-26 11:12:36 +00:00
mVrShadow.configureShadowsForCamera(camera, i == 0);
}
2020-06-28 09:33:01 +00:00
if (mirror)
mMirrorTexture.reset(new VRFramebuffer(context->getState(), mainCamera->getViewport()->width(), mainCamera->getViewport()->height(), 0));
mViewer->setReleaseContextAtEndOfFrameHint(false);
mMainCameraGC = mainCamera->getGraphicsContext();
mMainCameraGC->setSwapCallback(new VRViewer::SwapBuffersCallback(this));
mainCamera->setGraphicsContext(nullptr);
mConfigured = true;
Log(Debug::Verbose) << "Realized";
}
VRView* VRViewer::getView(std::string name)
{
auto it = mViews.find(name);
if (it != mViews.end())
return it->second.get();
return nullptr;
}
void VRViewer::enableMainCamera(void)
{
mCameras["MainCamera"]->setGraphicsContext(mMainCameraGC);
}
void VRViewer::disableMainCamera(void)
{
mCameras["MainCamera"]->setGraphicsContext(nullptr);
}
void VRViewer::blitEyesToMirrorTexture(osg::GraphicsContext* gc)
{
2020-06-28 09:33:01 +00:00
if (!mMirrorTexture)
return;
auto* state = gc->getState();
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
2020-06-24 19:26:11 +00:00
int screenWidth = mCameras["MainCamera"]->getViewport()->width();
2020-08-07 21:33:21 +00:00
int mirrorWidth = screenWidth / mMirrorTextureViews.size();
int screenHeight = mCameras["MainCamera"]->getViewport()->height();
2020-06-24 19:26:11 +00:00
2020-06-28 09:33:01 +00:00
// Since OpenXR does not include native support for mirror textures, we have to generate them ourselves
2020-08-07 21:33:21 +00:00
// which means resolving msaa twice.
for (unsigned i = 0; i < mMirrorTextureViews.size(); i++)
{
auto& resolveTexture = *mMsaaResolveMirrorTexture[i];
2020-06-28 09:33:01 +00:00
resolveTexture.bindFramebuffer(gc, GL_FRAMEBUFFER_EXT);
2020-08-07 21:33:21 +00:00
mViews[mMirrorTextureViews[i]]->swapchain().renderBuffer()->blit(gc, 0, 0, resolveTexture.width(), resolveTexture.height());
2020-06-28 09:33:01 +00:00
mMirrorTexture->bindFramebuffer(gc, GL_FRAMEBUFFER_EXT);
2020-08-07 21:33:21 +00:00
resolveTexture.blit(gc, i * mirrorWidth, 0, (i + 1) * mirrorWidth, screenHeight);
}
gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, 0);
mMirrorTexture->blit(gc, 0, 0, screenWidth, screenHeight);
}
void
VRViewer::SwapBuffersCallback::swapBuffersImplementation(
osg::GraphicsContext* gc)
{
2020-03-15 14:31:38 +00:00
auto* session = Environment::get().getSession();
session->swapBuffers(gc, *mViewer);
}
void
2020-06-28 09:33:01 +00:00
RealizeOperation::operator()(
osg::GraphicsContext* gc)
{
2020-06-28 09:33:01 +00:00
return Environment::get().getViewer()->realize(gc);
}
bool
2020-06-28 09:33:01 +00:00
RealizeOperation::realized()
{
2020-03-15 14:31:38 +00:00
return Environment::get().getViewer()->realized();
}
void VRViewer::preDrawCallback(osg::RenderInfo& info)
{
auto* camera = info.getCurrentCamera();
auto name = camera->getName();
mViews[name]->prerenderCallback(info);
}
void VRViewer::postDrawCallback(osg::RenderInfo& info)
{
auto* camera = info.getCurrentCamera();
auto name = camera->getName();
auto& view = mViews[name];
view->postrenderCallback(info);
// This happens sometimes, i've not been able to catch it when as happens
// to see why and how i can stop it.
if (camera->getPreDrawCallback() != mPreDraw)
{
camera->setPreDrawCallback(mPreDraw);
Log(Debug::Warning) << ("osg overwrote predraw");
}
}
}