1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-19 20:23:54 +00:00
openmw/components/terrain/compositemaprenderer.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

182 lines
5.7 KiB
C++
Raw Normal View History

#include "compositemaprenderer.hpp"
#include <osg/FrameBufferObject>
#include <osg/RenderInfo>
#include <osg/Texture2D>
#include <algorithm>
namespace Terrain
{
CompositeMapRenderer::CompositeMapRenderer()
: mTargetFrameRate(120)
, mMinimumTimeAvailable(0.0025)
{
setSupportsDisplayList(false);
setCullingActive(false);
mFBO = new osg::FrameBufferObject;
2020-06-25 19:46:07 +00:00
getOrCreateStateSet()->setMode(GL_LIGHTING, osg::StateAttribute::OFF);
2022-09-22 18:26:05 +00:00
}
CompositeMapRenderer::~CompositeMapRenderer() {}
void CompositeMapRenderer::drawImplementation(osg::RenderInfo& renderInfo) const
{
double dt = mTimer.time_s();
dt = std::min(dt, 0.2);
mTimer.setStartTick();
double targetFrameTime = 1.0 / static_cast<double>(mTargetFrameRate);
double conservativeTimeRatio(0.75);
double availableTime = std::max((targetFrameTime - dt) * conservativeTimeRatio, mMinimumTimeAvailable);
std::lock_guard<std::mutex> lock(mMutex);
if (mImmediateCompileSet.empty() && mCompileSet.empty())
return;
while (!mImmediateCompileSet.empty())
{
osg::ref_ptr<CompositeMap> node = *mImmediateCompileSet.begin();
mImmediateCompileSet.erase(node);
2022-09-22 18:26:05 +00:00
mMutex.unlock();
2023-12-28 04:30:04 +00:00
compile(*node, renderInfo);
mMutex.lock();
2022-09-22 18:26:05 +00:00
}
2023-11-29 14:43:33 +00:00
const auto deadline = std::chrono::steady_clock::now() + std::chrono::duration<double>(availableTime);
while (!mCompileSet.empty() && std::chrono::steady_clock::now() < deadline)
{
osg::ref_ptr<CompositeMap> node = *mCompileSet.begin();
mCompileSet.erase(node);
2022-09-22 18:26:05 +00:00
mMutex.unlock();
2023-12-28 04:30:04 +00:00
compile(*node, renderInfo);
mMutex.lock();
2022-09-22 18:26:05 +00:00
if (node->mCompiled < node->mDrawables.size())
2022-09-22 18:26:05 +00:00
{
// We did not compile the map fully.
// Place it back to queue to continue work in the next time.
mCompileSet.insert(node);
2022-09-22 18:26:05 +00:00
}
}
2018-02-13 18:25:28 +00:00
mTimer.setStartTick();
}
2023-12-28 04:30:04 +00:00
void CompositeMapRenderer::compile(CompositeMap& compositeMap, osg::RenderInfo& renderInfo) const
{
// if there are no more external references we can assume the texture is no longer required
if (compositeMap.mTexture->referenceCount() <= 1)
2022-09-22 18:26:05 +00:00
{
compositeMap.mCompiled = compositeMap.mDrawables.size();
return;
}
osg::Timer timer;
osg::State& state = *renderInfo.getState();
osg::GLExtensions* ext = state.get<osg::GLExtensions>();
if (!mFBO)
return;
if (!ext->isFrameBufferObjectSupported)
return;
osg::FrameBufferAttachment attach(compositeMap.mTexture);
mFBO->setAttachment(osg::Camera::COLOR_BUFFER, attach);
mFBO->apply(state, osg::FrameBufferObject::DRAW_FRAMEBUFFER);
GLenum status = ext->glCheckFramebufferStatus(GL_FRAMEBUFFER_EXT);
if (status != GL_FRAMEBUFFER_COMPLETE_EXT)
{
GLuint fboId = state.getGraphicsContext() ? state.getGraphicsContext()->getDefaultFboId() : 0;
ext->glBindFramebuffer(GL_FRAMEBUFFER_EXT, fboId);
OSG_ALWAYS << "Error attaching FBO" << std::endl;
return;
}
// inform State that Texture attribute has changed due to compiling of FBO texture
// should OSG be doing this on its own?
state.haveAppliedTextureAttribute(state.getActiveTextureUnit(), osg::StateAttribute::TEXTURE);
for (unsigned int i = compositeMap.mCompiled; i < compositeMap.mDrawables.size(); ++i)
{
osg::Drawable* drw = compositeMap.mDrawables[i];
osg::StateSet* stateset = drw->getStateSet();
if (stateset)
renderInfo.getState()->pushStateSet(stateset);
renderInfo.getState()->apply();
glViewport(0, 0, compositeMap.mTexture->getTextureWidth(), compositeMap.mTexture->getTextureHeight());
drw->drawImplementation(renderInfo);
if (stateset)
renderInfo.getState()->popStateSet();
++compositeMap.mCompiled;
2019-02-20 13:37:00 +00:00
compositeMap.mDrawables[i] = nullptr;
}
2019-02-20 13:37:00 +00:00
if (compositeMap.mCompiled == compositeMap.mDrawables.size())
compositeMap.mDrawables = std::vector<osg::ref_ptr<osg::Drawable>>();
state.haveAppliedAttribute(osg::StateAttribute::VIEWPORT);
GLuint fboId = state.getGraphicsContext() ? state.getGraphicsContext()->getDefaultFboId() : 0;
ext->glBindFramebuffer(GL_FRAMEBUFFER_EXT, fboId);
}
void CompositeMapRenderer::setMinimumTimeAvailableForCompile(double time)
2017-03-09 18:28:02 +00:00
{
mMinimumTimeAvailable = time;
}
void CompositeMapRenderer::setTargetFrameRate(float framerate)
{
2020-06-25 19:46:07 +00:00
mTargetFrameRate = framerate;
}
void CompositeMapRenderer::addCompositeMap(CompositeMap* compositeMap, bool immediate)
2022-09-22 18:26:05 +00:00
{
2020-06-25 19:46:07 +00:00
std::lock_guard<std::mutex> lock(mMutex);
if (immediate)
mImmediateCompileSet.insert(compositeMap);
2022-09-22 18:26:05 +00:00
else
mCompileSet.insert(compositeMap);
}
void CompositeMapRenderer::setImmediate(CompositeMap* compositeMap)
2022-09-22 18:26:05 +00:00
{
2020-06-25 19:46:07 +00:00
std::lock_guard<std::mutex> lock(mMutex);
2017-03-09 18:28:02 +00:00
CompileSet::iterator found = mCompileSet.find(compositeMap);
if (found == mCompileSet.end())
2022-09-22 18:26:05 +00:00
return;
else
{
mImmediateCompileSet.insert(compositeMap);
2017-03-09 18:28:02 +00:00
mCompileSet.erase(found);
2022-09-22 18:26:05 +00:00
}
}
unsigned int CompositeMapRenderer::getCompileSetSize() const
2022-09-22 18:26:05 +00:00
{
2020-06-25 19:46:07 +00:00
std::lock_guard<std::mutex> lock(mMutex);
return mCompileSet.size();
}
CompositeMap::CompositeMap()
2017-03-09 18:28:02 +00:00
: mCompiled(0)
2022-09-22 18:26:05 +00:00
{
}
CompositeMap::~CompositeMap() {}
}