1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-09 21:42:13 +00:00
OpenMW/apps/openmw/mwrender/occlusionquery.cpp

207 lines
6.6 KiB
C++
Raw Normal View History

#include "occlusionquery.hpp"
#include <OgreRenderSystem.h>
#include <OgreRoot.h>
#include <OgreBillboardSet.h>
2012-03-24 19:41:23 +00:00
#include <OgreHardwareOcclusionQuery.h>
#include <OgreEntity.h>
#include <OgreSubEntity.h>
#include <OgreMeshManager.h>
2012-04-05 09:19:51 +00:00
#include <OgreMaterialManager.h>
#include <OgreCamera.h>
#include "renderconst.hpp"
using namespace MWRender;
using namespace Ogre;
OcclusionQuery::OcclusionQuery(OEngine::Render::OgreRenderer* renderer, SceneNode* sunNode) :
mSunTotalAreaQuery(0), mSunVisibleAreaQuery(0), mActiveQuery(0),
mDoQuery(0), mSunVisibility(0),
mWasVisible(false),
mActive(false),
mFirstFrame(true)
{
mRendering = renderer;
mSunNode = sunNode;
try {
RenderSystem* renderSystem = Root::getSingleton().getRenderSystem();
mSunTotalAreaQuery = renderSystem->createHardwareOcclusionQuery();
mSunVisibleAreaQuery = renderSystem->createHardwareOcclusionQuery();
mSupported = (mSunTotalAreaQuery != 0) && (mSunVisibleAreaQuery != 0);
}
2013-07-31 16:46:32 +00:00
catch (Ogre::Exception& e)
{
mSupported = false;
}
if (!mSupported)
{
std::cout << "Hardware occlusion queries not supported." << std::endl;
return;
}
2012-03-31 17:08:05 +00:00
mBBNodeReal = mRendering->getScene()->getRootSceneNode()->createChildSceneNode();
2012-03-24 19:41:23 +00:00
2013-05-07 17:15:28 +00:00
static Ogre::Mesh* plane = MeshManager::getSingleton().createPlane("occlusionbillboard",
ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::Plane(Ogre::Vector3(0,0,1), 0), 1, 1, 1, 1, true, 1, 1, 1, Vector3::UNIT_Y).get();
plane->_setBounds(Ogre::AxisAlignedBox::BOX_INFINITE);
2013-05-07 17:15:28 +00:00
mBBQueryTotal = mRendering->getScene()->createEntity("occlusionbillboard");
2012-04-11 16:53:13 +00:00
mBBQueryTotal->setCastShadows(false);
2012-04-18 23:10:55 +00:00
mBBQueryTotal->setVisibilityFlags(RV_OcclusionQuery);
mBBQueryTotal->setRenderQueueGroup(RQG_OcclusionQuery+1);
mBBQueryTotal->setMaterialName("QueryTotalPixels");
2012-03-31 17:08:05 +00:00
mBBNodeReal->attachObject(mBBQueryTotal);
2013-05-07 17:15:28 +00:00
mBBQueryVisible = mRendering->getScene()->createEntity("occlusionbillboard");
2012-04-11 16:53:13 +00:00
mBBQueryVisible->setCastShadows(false);
2012-04-18 23:10:55 +00:00
mBBQueryVisible->setVisibilityFlags(RV_OcclusionQuery);
mBBQueryVisible->setRenderQueueGroup(RQG_OcclusionQuery+1);
mBBQueryVisible->setMaterialName("QueryVisiblePixels");
2012-03-31 17:08:05 +00:00
mBBNodeReal->attachObject(mBBQueryVisible);
mRendering->getScene()->addRenderObjectListener(this);
2012-03-25 18:52:56 +00:00
mRendering->getScene()->addRenderQueueListener(this);
mDoQuery = true;
}
OcclusionQuery::~OcclusionQuery()
{
mRendering->getScene()->removeRenderObjectListener (this);
mRendering->getScene()->removeRenderQueueListener(this);
RenderSystem* renderSystem = Root::getSingleton().getRenderSystem();
if (mSunTotalAreaQuery)
renderSystem->destroyHardwareOcclusionQuery(mSunTotalAreaQuery);
if (mSunVisibleAreaQuery)
renderSystem->destroyHardwareOcclusionQuery(mSunVisibleAreaQuery);
}
bool OcclusionQuery::supported()
{
2012-03-28 22:05:48 +00:00
return mSupported;
}
void OcclusionQuery::notifyRenderSingleObject(Renderable* rend, const Pass* pass, const AutoParamDataSource* source,
const LightList* pLightList, bool suppressRenderStateChanges)
{
if (!mActive) return;
// The following code activates and deactivates the occlusion queries
// so that the queries only include the rendering of the intended meshes
// Close the last occlusion query
// Each occlusion query should only last a single rendering
if (mActiveQuery != NULL)
{
mActiveQuery->endOcclusionQuery();
mActiveQuery = NULL;
}
// Open a new occlusion query
if (mDoQuery == true)
{
if (rend == mBBQueryTotal->getSubEntity(0))
2012-03-25 18:52:56 +00:00
{
mActiveQuery = mSunTotalAreaQuery;
2012-03-25 18:52:56 +00:00
mWasVisible = true;
}
else if (rend == mBBQueryVisible->getSubEntity(0))
2012-03-24 19:41:23 +00:00
{
2012-03-25 18:52:56 +00:00
mActiveQuery = mSunVisibleAreaQuery;
2012-03-24 19:41:23 +00:00
}
}
2012-03-25 18:52:56 +00:00
if (mActiveQuery != NULL)
mActiveQuery->beginOcclusionQuery();
}
void OcclusionQuery::renderQueueEnded(uint8 queueGroupId, const String& invocation, bool& repeatThisInvocation)
2012-03-26 19:52:38 +00:00
{
if (!mActive) return;
2012-03-31 18:18:20 +00:00
if (mActiveQuery != NULL)
{
mActiveQuery->endOcclusionQuery();
mActiveQuery = NULL;
}
2012-03-26 19:52:38 +00:00
/**
* for every beginOcclusionQuery(), we want a respective pullOcclusionQuery() and vice versa
* this also means that results can be wrong at other places if we pull, but beginOcclusionQuery() was never called
* this can happen for example if the object that is tested is outside of the view frustum
* to prevent this, check if the queries have been performed after everything has been rendered and if not, start them manually
*/
2012-04-04 16:53:40 +00:00
if (queueGroupId == RQG_SkiesLate)
2012-03-25 18:52:56 +00:00
{
2012-03-26 19:52:38 +00:00
if (mWasVisible == false && mDoQuery)
{
mSunTotalAreaQuery->beginOcclusionQuery();
mSunTotalAreaQuery->endOcclusionQuery();
mSunVisibleAreaQuery->beginOcclusionQuery();
mSunVisibleAreaQuery->endOcclusionQuery();
}
2012-03-25 20:53:00 +00:00
}
}
2012-03-25 22:31:03 +00:00
void OcclusionQuery::update(float duration)
{
if (mFirstFrame)
{
// GLHardwareOcclusionQuery::isStillOutstanding doesn't seem to like getting called when nothing has been rendered yet
mFirstFrame = false;
return;
}
if (!mSupported) return;
2012-03-25 18:52:56 +00:00
mWasVisible = false;
2012-03-24 17:38:58 +00:00
// Adjust the position of the sun billboards according to camera viewing distance
// we need to do this to make sure that _everything_ can occlude the sun
float dist = mRendering->getCamera()->getFarClipDistance();
if (dist==0) dist = 10000000;
dist -= 1000; // bias
dist /= 1000.f;
if (mSunNode)
{
mBBNodeReal->setPosition(mSunNode->getPosition() * dist);
mBBNodeReal->setOrientation(Ogre::Vector3::UNIT_Z.getRotationTo(-mBBNodeReal->getPosition().normalisedCopy()));
mBBNodeReal->setScale(150.f*dist, 150.f*dist, 150.f*dist);
}
2012-03-24 17:38:58 +00:00
// Stop occlusion queries until we get their information
// (may not happen on the same frame they are requested in)
mDoQuery = false;
2012-03-24 19:41:23 +00:00
if (!mSunTotalAreaQuery->isStillOutstanding()
&& !mSunVisibleAreaQuery->isStillOutstanding())
{
unsigned int totalPixels;
unsigned int visiblePixels;
mSunTotalAreaQuery->pullOcclusionQuery(&totalPixels);
mSunVisibleAreaQuery->pullOcclusionQuery(&visiblePixels);
if (totalPixels == 0)
{
// probably outside of the view frustum
mSunVisibility = 0;
}
else
{
mSunVisibility = float(visiblePixels) / float(totalPixels);
if (mSunVisibility > 1) mSunVisibility = 1;
}
2012-03-31 18:05:23 +00:00
mDoQuery = true;
}
}
void OcclusionQuery::setSunNode(Ogre::SceneNode* node)
{
mSunNode = node;
}