2020-05-24 16:00:42 +00:00
|
|
|
#include "vrviewer.hpp"
|
2020-05-01 19:37:01 +00:00
|
|
|
#include "vrtexture.hpp"
|
2020-01-09 23:10:09 +00:00
|
|
|
#include <osg/Texture2D>
|
|
|
|
#include <osgViewer/Renderer>
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
#include <osgDB/Registry>
|
|
|
|
#include <sstream>
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
#ifndef GL_TEXTURE_MAX_LEVEL
|
|
|
|
#define GL_TEXTURE_MAX_LEVEL 0x813D
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace MWVR
|
|
|
|
{
|
2020-06-24 19:26:11 +00:00
|
|
|
|
|
|
|
VRTexture::VRTexture(osg::ref_ptr<osg::State> state, std::size_t width, std::size_t height, uint32_t msaaSamples, uint32_t colorBuffer, uint32_t depthBuffer)
|
2020-01-23 23:14:23 +00:00
|
|
|
: mState(state)
|
2020-01-09 23:10:09 +00:00
|
|
|
, mWidth(width)
|
|
|
|
, mHeight(height)
|
2020-01-26 19:06:47 +00:00
|
|
|
, mSamples(msaaSamples)
|
2020-06-24 19:26:11 +00:00
|
|
|
, mColorBuffer(colorBuffer)
|
|
|
|
, mDepthBuffer(depthBuffer)
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
|
|
|
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
|
|
|
|
|
2020-01-26 19:06:47 +00:00
|
|
|
gl->glGenFramebuffers(1, &mBlitFBO);
|
|
|
|
gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, mBlitFBO);
|
2020-01-09 23:10:09 +00:00
|
|
|
|
|
|
|
gl->glGenFramebuffers(1, &mFBO);
|
|
|
|
|
2020-06-24 19:26:11 +00:00
|
|
|
if (mSamples <= 1)
|
2020-01-26 19:06:47 +00:00
|
|
|
mTextureTarget = GL_TEXTURE_2D;
|
2020-01-09 23:10:09 +00:00
|
|
|
else
|
2020-01-26 19:06:47 +00:00
|
|
|
mTextureTarget = GL_TEXTURE_2D_MULTISAMPLE;
|
|
|
|
|
2020-06-24 19:26:11 +00:00
|
|
|
if (mColorBuffer == 0)
|
|
|
|
{
|
|
|
|
glGenTextures(1, &mColorBuffer);
|
|
|
|
glBindTexture(mTextureTarget, mColorBuffer);
|
|
|
|
if (mSamples <= 1)
|
|
|
|
glTexImage2D(mTextureTarget, 0, GL_RGBA, mWidth, mHeight, 0, GL_RGBA, GL_UNSIGNED_INT, nullptr);
|
|
|
|
else
|
|
|
|
gl->glTexImage2DMultisample(mTextureTarget, mSamples, GL_RGBA, mWidth, mHeight, false);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_BORDER_ARB);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_BORDER_ARB);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_MAX_LEVEL, 0);
|
|
|
|
}
|
2020-01-26 19:06:47 +00:00
|
|
|
|
2020-06-24 19:26:11 +00:00
|
|
|
if (mDepthBuffer == 0)
|
|
|
|
{
|
|
|
|
glGenTextures(1, &mDepthBuffer);
|
|
|
|
glBindTexture(mTextureTarget, mDepthBuffer);
|
|
|
|
if (mSamples <= 1)
|
|
|
|
glTexImage2D(mTextureTarget, 0, GL_DEPTH_COMPONENT24, mWidth, mHeight, 0, GL_DEPTH_COMPONENT, GL_UNSIGNED_INT, nullptr);
|
|
|
|
else
|
|
|
|
gl->glTexImage2DMultisample(mTextureTarget, mSamples, GL_DEPTH_COMPONENT, mWidth, mHeight, false);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_WRAP_S, GL_REPEAT);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_WRAP_T, GL_REPEAT);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri(mTextureTarget, GL_TEXTURE_MAX_LEVEL, 0);
|
|
|
|
}
|
2020-01-26 19:06:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, mFBO);
|
|
|
|
gl->glFramebufferTexture2D(GL_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, mTextureTarget, mColorBuffer, 0);
|
|
|
|
gl->glFramebufferTexture2D(GL_FRAMEBUFFER_EXT, GL_DEPTH_ATTACHMENT_EXT, mTextureTarget, mDepthBuffer, 0);
|
|
|
|
if (gl->glCheckFramebufferStatus(GL_FRAMEBUFFER_EXT) != GL_FRAMEBUFFER_COMPLETE_EXT)
|
|
|
|
throw std::runtime_error("Failed to create OpenXR framebuffer");
|
2020-01-09 23:10:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, 0);
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
VRTexture::~VRTexture()
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
|
|
|
destroy(nullptr);
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void VRTexture::destroy(osg::State* state)
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
|
|
|
if (!state)
|
|
|
|
{
|
|
|
|
// Try re-using the state received during construction
|
|
|
|
state = mState.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state)
|
|
|
|
{
|
|
|
|
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
|
|
|
|
if (mFBO)
|
|
|
|
gl->glDeleteFramebuffers(1, &mFBO);
|
|
|
|
}
|
2020-01-26 19:06:47 +00:00
|
|
|
else if(mFBO)
|
2020-01-09 23:10:09 +00:00
|
|
|
// Without access to opengl methods, i'll just let the FBOs leak.
|
2020-01-26 19:06:47 +00:00
|
|
|
Log(Debug::Warning) << "destroy() called without a State. Leaking FBO";
|
2020-01-09 23:10:09 +00:00
|
|
|
|
|
|
|
if (mDepthBuffer)
|
|
|
|
glDeleteTextures(1, &mDepthBuffer);
|
2020-01-26 19:06:47 +00:00
|
|
|
if (mColorBuffer)
|
|
|
|
glDeleteTextures(1, &mColorBuffer);
|
2020-01-09 23:10:09 +00:00
|
|
|
|
2020-01-26 19:06:47 +00:00
|
|
|
mFBO = mDepthBuffer = mColorBuffer;
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void VRTexture::beginFrame(osg::GraphicsContext* gc)
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
2020-01-23 23:14:23 +00:00
|
|
|
auto state = gc->getState();
|
2020-01-09 23:10:09 +00:00
|
|
|
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
|
2020-01-26 19:06:47 +00:00
|
|
|
gl->glBindFramebuffer(GL_FRAMEBUFFER_EXT, mFBO);
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void VRTexture::endFrame(osg::GraphicsContext* gc, uint32_t blitTarget)
|
2020-01-09 23:10:09 +00:00
|
|
|
{
|
2020-01-23 23:14:23 +00:00
|
|
|
auto* state = gc->getState();
|
2020-01-09 23:10:09 +00:00
|
|
|
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
|
2020-01-26 19:06:47 +00:00
|
|
|
gl->glBindFramebuffer(GL_DRAW_FRAMEBUFFER_EXT, mBlitFBO);
|
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER_EXT, mFBO);
|
|
|
|
gl->glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, blitTarget, 0);
|
|
|
|
gl->glBlitFramebuffer(0, 0, mWidth, mHeight, 0, 0, mWidth, mHeight, GL_COLOR_BUFFER_BIT, GL_NEAREST);
|
|
|
|
gl->glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, 0, 0);
|
|
|
|
gl->glBindFramebuffer(GL_DRAW_FRAMEBUFFER_EXT, 0);
|
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER_EXT, 0);
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|
2020-01-23 23:14:23 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void VRTexture::blit(osg::GraphicsContext* gc, int x, int y, int w, int h)
|
2020-01-23 23:14:23 +00:00
|
|
|
{
|
|
|
|
auto* state = gc->getState();
|
|
|
|
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
|
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER_EXT, mFBO);
|
2020-03-04 23:04:23 +00:00
|
|
|
gl->glBlitFramebuffer(0, 0, mWidth, mHeight, x, y, w, h, GL_COLOR_BUFFER_BIT, GL_LINEAR);
|
2020-01-23 23:14:23 +00:00
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER_EXT, 0);
|
|
|
|
}
|
2020-03-01 22:05:38 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void VRTexture::blit(osg::GraphicsContext* gc, int x, int y, int w, int h, int blitTarget)
|
2020-03-01 22:05:38 +00:00
|
|
|
{
|
|
|
|
auto* state = gc->getState();
|
|
|
|
auto* gl = osg::GLExtensions::Get(state->getContextID(), false);
|
|
|
|
gl->glBindFramebuffer(GL_DRAW_FRAMEBUFFER_EXT, mBlitFBO);
|
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER_EXT, mFBO);
|
|
|
|
gl->glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, blitTarget, 0);
|
2020-03-04 23:04:23 +00:00
|
|
|
gl->glBlitFramebuffer(0, 0, mWidth, mHeight, x, y, w, h, GL_COLOR_BUFFER_BIT, GL_LINEAR);
|
2020-03-01 22:05:38 +00:00
|
|
|
gl->glFramebufferTexture2D(GL_DRAW_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, 0, 0);
|
|
|
|
gl->glBindFramebuffer(GL_DRAW_FRAMEBUFFER_EXT, 0);
|
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER_EXT, 0);
|
|
|
|
}
|
2020-01-09 23:10:09 +00:00
|
|
|
}
|