#include "vrviewer.hpp" #include "openxrmanagerimpl.hpp" #include "openxrswapchain.hpp" #include "vrenvironment.hpp" #include "vrsession.hpp" #include "vrframebuffer.hpp" #include "vrview.hpp" #include "../mwrender/vismask.hpp" #include #include #include #include #include #include #include #include #include #include namespace MWVR { const std::array VRViewer::sViewNames = { "LeftEye", "RightEye" }; // 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( osg::ref_ptr viewer) : mViewer(viewer) , mPreDraw(new PredrawCallback(this)) , mPostDraw(new PostdrawCallback(this)) , mFinalDraw(new FinaldrawCallback(this)) , mUpdateViewCallback(new UpdateViewCallback(this)) , mMsaaResolveTexture{} , mMirrorTexture{ nullptr } , mOpenXRConfigured(false) , mCallbacksConfigured(false) { mViewer->setRealizeOperation(new RealizeOperation()); } VRViewer::~VRViewer(void) { } 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; } return recommended; } static VRViewer::MirrorTextureEye mirrorTextureEyeFromString(const std::string& str) { if (Misc::StringUtils::ciEqual(str, "left")) return VRViewer::MirrorTextureEye::Left; if (Misc::StringUtils::ciEqual(str, "right")) return VRViewer::MirrorTextureEye::Right; if (Misc::StringUtils::ciEqual(str, "both")) return VRViewer::MirrorTextureEye::Both; return VRViewer::MirrorTextureEye::Both; } void VRViewer::configureXR(osg::GraphicsContext* gc) { std::unique_lock lock(mMutex); if (mOpenXRConfigured) { return; } auto* xr = Environment::get().getManager(); xr->realize(gc); // Run through initial events to start session // For the rest of runtime this is handled by vrsession xr->handleEvents(); // Set up swapchain config mSwapchainConfig = xr->getRecommendedSwapchainConfig(); std::array xConfString; std::array 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++) { auto name = sViewNames[i]; mSwapchainConfig[i].selectedWidth = parseResolution(xConfString[i], mSwapchainConfig[i].recommendedWidth, mSwapchainConfig[i].maxWidth); mSwapchainConfig[i].selectedHeight = parseResolution(yConfString[i], mSwapchainConfig[i].recommendedHeight, mSwapchainConfig[i].maxHeight); mSwapchainConfig[i].selectedSamples = std::max(1, // OpenXR requires a non-zero value std::min(mSwapchainConfig[i].maxSamples, Settings::Manager::getInt("antialiasing", "Video") ) ); Log(Debug::Verbose) << name << " resolution: Recommended x=" << mSwapchainConfig[i].recommendedWidth << ", y=" << mSwapchainConfig[i].recommendedHeight; Log(Debug::Verbose) << name << " resolution: Max x=" << mSwapchainConfig[i].maxWidth << ", y=" << mSwapchainConfig[i].maxHeight; Log(Debug::Verbose) << name << " resolution: Selected x=" << mSwapchainConfig[i].selectedWidth << ", y=" << mSwapchainConfig[i].selectedHeight; mSwapchainConfig[i].name = sViewNames[i]; if (i > 0) mSwapchainConfig[i].offsetWidth = mSwapchainConfig[i].selectedWidth + mSwapchainConfig[i].offsetWidth; mSwapchain[i].reset(new OpenXRSwapchain(gc->getState(), mSwapchainConfig[i])); mSubImages[i].width = mSwapchainConfig[i].selectedWidth; mSubImages[i].height = mSwapchainConfig[i].selectedHeight; mSubImages[i].x = mSubImages[i].y = 0; mSubImages[i].swapchain = mSwapchain[i].get(); } int width = mSubImages[0].width + mSubImages[1].width; int height = std::max(mSubImages[0].height, mSubImages[1].height); int samples = std::max(mSwapchainConfig[0].selectedSamples, mSwapchainConfig[1].selectedSamples); mFramebuffer.reset(new VRFramebuffer(gc->getState(), width, height, samples)); mFramebuffer->createColorBuffer(gc); mFramebuffer->createDepthBuffer(gc); mMsaaResolveTexture.reset(new VRFramebuffer(gc->getState(), width, height, 0)); mMsaaResolveTexture->createColorBuffer(gc); mGammaResolveTexture.reset(new VRFramebuffer(gc->getState(), width, height, 0)); mGammaResolveTexture->createColorBuffer(gc); mGammaResolveTexture->createDepthBuffer(gc); mViewer->setReleaseContextAtEndOfFrameHint(false); mViewer->getCamera()->getGraphicsContext()->setSwapCallback(new VRViewer::SwapBuffersCallback(this)); setupMirrorTexture(); Log(Debug::Verbose) << "XR configured"; mOpenXRConfigured = true; } void VRViewer::configureCallbacks() { if (mCallbacksConfigured) return; // Give the main camera an initial draw callback that disables camera setup (we don't want it) Misc::StereoView::instance().setUpdateViewCallback(mUpdateViewCallback); Misc::CallbackManager::instance().addCallback(Misc::CallbackManager::DrawStage::Initial, new InitialDrawCallback(this)); Misc::CallbackManager::instance().addCallback(Misc::CallbackManager::DrawStage::PreDraw, mPreDraw); Misc::CallbackManager::instance().addCallback(Misc::CallbackManager::DrawStage::PostDraw, mPostDraw); Misc::CallbackManager::instance().addCallback(Misc::CallbackManager::DrawStage::Final, mFinalDraw); auto cullMask = ~(MWRender::VisMask::Mask_UpdateVisitor | MWRender::VisMask::Mask_SimpleWater); cullMask &= ~MWRender::VisMask::Mask_GUI; cullMask |= MWRender::VisMask::Mask_3DGUI; Misc::StereoView::instance().setCullMask(cullMask); mCallbacksConfigured = true; } void VRViewer::setupMirrorTexture() { mMirrorTextureEnabled = Settings::Manager::getBool("mirror texture", "VR"); mMirrorTextureEye = mirrorTextureEyeFromString(Settings::Manager::getString("mirror texture eye", "VR")); mFlipMirrorTextureOrder = Settings::Manager::getBool("flip mirror texture order", "VR"); mMirrorTextureShouldBeCleanedUp = true; mMirrorTextureViews.clear(); if (mMirrorTextureEye == MirrorTextureEye::Left || mMirrorTextureEye == MirrorTextureEye::Both) mMirrorTextureViews.push_back(sViewNames[(int)Side::LEFT_SIDE]); if (mMirrorTextureEye == MirrorTextureEye::Right || mMirrorTextureEye == MirrorTextureEye::Both) mMirrorTextureViews.push_back(sViewNames[(int)Side::RIGHT_SIDE]); if (mFlipMirrorTextureOrder) std::reverse(mMirrorTextureViews.begin(), mMirrorTextureViews.end()); // 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 } void VRViewer::processChangedSettings(const std::set>& changed) { bool mirrorTextureChanged = false; for (Settings::CategorySettingVector::const_iterator it = changed.begin(); it != changed.end(); ++it) { if (it->first == "VR" && it->second == "mirror texture") { mirrorTextureChanged = true; } if (it->first == "VR" && it->second == "mirror texture eye") { mirrorTextureChanged = true; } if (it->first == "VR" && it->second == "flip mirror texture order") { mirrorTextureChanged = true; } } if (mirrorTextureChanged) setupMirrorTexture(); } SubImage VRViewer::subImage(Side side) { return mSubImages[static_cast(side)]; } static GLuint createShader(osg::GLExtensions* gl, const char* source, GLenum type) { GLint len = strlen(source); GLuint shader = gl->glCreateShader(type); gl->glShaderSource(shader, 1, &source, &len); gl->glCompileShader(shader); GLint isCompiled = 0; gl->glGetShaderiv(shader, GL_COMPILE_STATUS, &isCompiled); if (isCompiled == GL_FALSE) { GLint maxLength = 0; gl->glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &maxLength); std::vector infoLog(maxLength); gl->glGetShaderInfoLog(shader, maxLength, &maxLength, &infoLog[0]); gl->glDeleteShader(shader); Log(Debug::Error) << "Failed to compile shader: " << infoLog.data(); return 0; } return shader; } static bool applyGamma(osg::RenderInfo& info, VRFramebuffer& target, VRFramebuffer& source) { osg::State* state = info.getState(); static const char* vSource = "#version 120\n varying vec2 uv; void main(){ gl_Position = vec4(gl_Vertex.xy*2.0 - 1, 0, 1); uv = gl_Vertex.xy;}"; static const char* fSource = "#version 120\n varying vec2 uv; uniform sampler2D t; uniform float gamma; uniform float contrast;" "void main() {" "vec4 color1 = texture2D(t, uv);" "vec3 rgb = color1.rgb;" "rgb = (rgb - 0.5f) * contrast + 0.5f;" "rgb = pow(rgb, vec3(1.0/gamma));" "gl_FragColor = vec4(rgb, color1.a);" "}"; static bool first = true; static osg::ref_ptr program = nullptr; static osg::ref_ptr vShader = nullptr; static osg::ref_ptr fShader = nullptr; static osg::ref_ptr gammaUniform = nullptr; static osg::ref_ptr contrastUniform = nullptr; osg::Viewport* viewport = nullptr; static osg::ref_ptr stateset = nullptr; static osg::ref_ptr geometry = nullptr; static osg::ref_ptr texture = nullptr; static osg::ref_ptr textureObject = nullptr; static std::vector vertices = { {0, 0, 0, 0}, {1, 0, 0, 0}, {1, 1, 0, 0}, {0, 0, 0, 0}, {1, 1, 0, 0}, {0, 1, 0, 0} }; static osg::ref_ptr vertexArray = new osg::Vec4Array(vertices.begin(), vertices.end()); if (first) { geometry = new osg::Geometry(); geometry->setVertexArray(vertexArray); geometry->addPrimitiveSet(new osg::DrawArrays(GL_TRIANGLES, 0, 6)); geometry->setUseDisplayList(false); stateset = geometry->getOrCreateStateSet(); vShader = new osg::Shader(osg::Shader::Type::VERTEX, vSource); fShader = new osg::Shader(osg::Shader::Type::FRAGMENT, fSource); program = new osg::Program(); program->addShader(vShader); program->addShader(fShader); program->compileGLObjects(*state); stateset->setAttributeAndModes(program, osg::StateAttribute::ON); texture = new osg::Texture2D(); texture->setName("diffuseMap"); textureObject = new osg::Texture::TextureObject(texture, source.colorBuffer(), GL_TEXTURE_2D); texture->setTextureObject(state->getContextID(), textureObject); stateset->setTextureAttributeAndModes(0, texture, osg::StateAttribute::PROTECTED); stateset->setTextureMode(0, GL_TEXTURE_2D, osg::StateAttribute::PROTECTED); gammaUniform = new osg::Uniform("gamma", Settings::Manager::getFloat("gamma", "Video")); contrastUniform = new osg::Uniform("contrast", Settings::Manager::getFloat("contrast", "Video")); stateset->addUniform(gammaUniform); stateset->addUniform(contrastUniform); geometry->compileGLObjects(info); first = false; } target.bindFramebuffer(state->getGraphicsContext(), GL_FRAMEBUFFER_EXT); if (program != nullptr) { // OSG does not pop statesets until after the final draw callback. Unrelated statesets may therefore still be on the stack at this point. // Pop these to avoid inheriting arbitrary state from these. They will not be used more in this frame. state->popAllStateSets(); state->apply(); gammaUniform->set(Settings::Manager::getFloat("gamma", "Video")); contrastUniform->set(Settings::Manager::getFloat("contrast", "Video")); state->pushStateSet(stateset); state->apply(); if(!viewport) viewport = new osg::Viewport(0, 0, target.width(), target.height()); viewport->setViewport(0, 0, target.width(), target.height()); viewport->apply(*state); glClear(GL_DEPTH_BUFFER_BIT | GL_COLOR_BUFFER_BIT); geometry->draw(info); state->popStateSet(); return true; } return false; } void VRViewer::blit(osg::RenderInfo& info) { if (mMirrorTextureShouldBeCleanedUp) { mMirrorTexture = nullptr; mMirrorTextureShouldBeCleanedUp = false; } if (!mMirrorTextureEnabled) return; auto* state = info.getState(); auto* gc = state->getGraphicsContext(); auto* gl = osg::GLExtensions::Get(state->getContextID(), false); auto* traits = SDLUtil::GraphicsWindowSDL2::findContext(*mViewer)->getTraits(); int screenWidth = traits->width; int screenHeight = traits->height; if (!mMirrorTexture) { mMirrorTexture.reset(new VRFramebuffer(state, screenWidth, screenHeight, 0)); mMirrorTexture->createColorBuffer(gc); } int mirrorWidth = screenWidth / mMirrorTextureViews.size(); //// Since OpenXR does not include native support for mirror textures, we have to generate them ourselves mMsaaResolveTexture->bindFramebuffer(gc, GL_FRAMEBUFFER_EXT); mFramebuffer->blit(gc, 0, 0, mFramebuffer->width(), mFramebuffer->height(), 0, 0, mMsaaResolveTexture->width(), mMsaaResolveTexture->height(), GL_COLOR_BUFFER_BIT, GL_NEAREST); bool shouldDoGamma = Settings::Manager::getBool("gamma postprocessing", "VR Debug"); if (!shouldDoGamma || !applyGamma(info, *mGammaResolveTexture, *mMsaaResolveTexture)) { mGammaResolveTexture->bindFramebuffer(gc, GL_FRAMEBUFFER_EXT); mMsaaResolveTexture->blit(gc, 0, 0, mMsaaResolveTexture->width(), mMsaaResolveTexture->height(), 0, 0, mGammaResolveTexture->width(), mGammaResolveTexture->height(), GL_COLOR_BUFFER_BIT, GL_NEAREST); } mGammaResolveTexture->bindFramebuffer(gc, GL_FRAMEBUFFER_EXT); mFramebuffer->blit(gc, 0, 0, mFramebuffer->width(), mFramebuffer->height(), 0, 0, mGammaResolveTexture->width(), mGammaResolveTexture->height(), GL_DEPTH_BUFFER_BIT, GL_NEAREST); if (mMirrorTexture) { mMirrorTexture->bindFramebuffer(gc, GL_FRAMEBUFFER_EXT); mMsaaResolveTexture->blit(gc, 0, 0, mMsaaResolveTexture->width(), mMsaaResolveTexture->height(), 0, 0, screenWidth, screenHeight, GL_COLOR_BUFFER_BIT, GL_LINEAR); gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, 0); mMirrorTexture->blit(gc, 0, 0, screenWidth, screenHeight, 0, 0, screenWidth, screenHeight, GL_COLOR_BUFFER_BIT, GL_NEAREST); } mSwapchain[0]->endFrame(gc, *mGammaResolveTexture); mSwapchain[1]->endFrame(gc, *mGammaResolveTexture); gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, 0); } void VRViewer::SwapBuffersCallback::swapBuffersImplementation( osg::GraphicsContext* gc) { mViewer->swapBuffersCallback(gc); } void RealizeOperation::operator()( osg::GraphicsContext* gc) { if (Debug::shouldDebugOpenGL()) Debug::EnableGLDebugOperation()(gc); Environment::get().getViewer()->configureXR(gc); } bool RealizeOperation::realized() { return Environment::get().getViewer()->xrConfigured(); } void VRViewer::initialDrawCallback(osg::RenderInfo& info) { if (mRenderingReady) return; Environment::get().getSession()->beginPhase(VRSession::FramePhase::Draw); if (Environment::get().getSession()->getFrame(VRSession::FramePhase::Draw)->mShouldRender) { mSwapchain[0]->beginFrame(info.getState()->getGraphicsContext()); mSwapchain[1]->beginFrame(info.getState()->getGraphicsContext()); } mViewer->getCamera()->setViewport(0, 0, mFramebuffer->width(), mFramebuffer->height()); osg::GraphicsOperation* graphicsOperation = info.getCurrentCamera()->getRenderer(); osgViewer::Renderer* renderer = dynamic_cast(graphicsOperation); if (renderer != nullptr) { // Disable normal OSG FBO camera setup renderer->setCameraRequiresSetUp(false); } mRenderingReady = true; } void VRViewer::preDrawCallback(osg::RenderInfo& info) { if (Environment::get().getSession()->getFrame(VRSession::FramePhase::Draw)->mShouldRender) { mFramebuffer->bindFramebuffer(info.getState()->getGraphicsContext(), GL_FRAMEBUFFER_EXT); //mSwapchain->framebuffer()->bindFramebuffer(info.getState()->getGraphicsContext(), GL_FRAMEBUFFER_EXT); } } void VRViewer::postDrawCallback(osg::RenderInfo& info) { auto* camera = info.getCurrentCamera(); auto name = camera->getName(); } void VRViewer::finalDrawCallback(osg::RenderInfo& info) { auto* session = Environment::get().getSession(); auto* frameMeta = session->getFrame(VRSession::FramePhase::Draw).get(); if (frameMeta->mShouldSyncFrameLoop) { if (frameMeta->mShouldRender) { blit(info); } } } void VRViewer::swapBuffersCallback(osg::GraphicsContext* gc) { auto* session = Environment::get().getSession(); session->swapBuffers(gc, *this); mRenderingReady = false; } void VRViewer::updateView(Misc::View& left, Misc::View& right) { auto phase = VRSession::FramePhase::Update; auto session = Environment::get().getSession(); auto& frame = session->getFrame(phase); if (frame->mShouldRender) { left = frame->mPredictedPoses.view[static_cast(Side::LEFT_SIDE)]; right = frame->mPredictedPoses.view[static_cast(Side::RIGHT_SIDE)]; } } void VRViewer::UpdateViewCallback::updateView(Misc::View& left, Misc::View& right) { mViewer->updateView(left, right); } void VRViewer::FinaldrawCallback::operator()(osg::RenderInfo& info, Misc::StereoView::StereoDrawCallback::View view) const { if (view != Misc::StereoView::StereoDrawCallback::View::Left) mViewer->finalDrawCallback(info); } }