1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 06:35:30 +00:00
OpenMW/apps/openmw/mwrender/occlusionquery.cpp

267 lines
9.2 KiB
C++
Raw Normal View History

#include "occlusionquery.hpp"
#include <OgreRenderSystem.h>
#include <OgreRoot.h>
#include <OgreBillboardSet.h>
2012-03-24 20:41:23 +01:00
#include <OgreHardwareOcclusionQuery.h>
#include <OgreEntity.h>
using namespace MWRender;
using namespace Ogre;
OcclusionQuery::OcclusionQuery(OEngine::Render::OgreRenderer* renderer, SceneNode* sunNode) :
2012-03-24 20:41:23 +01:00
mSunTotalAreaQuery(0), mSunVisibleAreaQuery(0), mSingleObjectQuery(0), mActiveQuery(0),
2012-03-27 20:59:58 +02:00
mDoQuery(0), mSunVisibility(0), mQuerySingleObjectStarted(false), mTestResult(false),
mQuerySingleObjectRequested(false), mWasVisible(false), mObjectWasVisible(false), mDoQuery2(false),
mBBNode(0)
{
mRendering = renderer;
mSunNode = sunNode;
try {
RenderSystem* renderSystem = Root::getSingleton().getRenderSystem();
mSunTotalAreaQuery = renderSystem->createHardwareOcclusionQuery();
mSunVisibleAreaQuery = renderSystem->createHardwareOcclusionQuery();
2012-03-24 20:41:23 +01:00
mSingleObjectQuery = renderSystem->createHardwareOcclusionQuery();
2012-03-24 20:41:23 +01:00
mSupported = (mSunTotalAreaQuery != 0) && (mSunVisibleAreaQuery != 0) && (mSingleObjectQuery != 0);
}
catch (Ogre::Exception e)
{
mSupported = false;
}
if (!mSupported)
{
std::cout << "Hardware occlusion queries not supported." << std::endl;
return;
}
// This means that everything up to RENDER_QUEUE_MAIN can occlude the objects that are tested
const int queue = RENDER_QUEUE_MAIN+1;
MaterialPtr matBase = MaterialManager::getSingleton().getByName("BaseWhiteNoLighting");
MaterialPtr matQueryArea = matBase->clone("QueryTotalPixels");
matQueryArea->setDepthWriteEnabled(false);
matQueryArea->setColourWriteEnabled(false);
matQueryArea->setDepthCheckEnabled(false); // Not occluded by objects
MaterialPtr matQueryVisible = matBase->clone("QueryVisiblePixels");
matQueryVisible->setDepthWriteEnabled(false);
2012-03-25 20:52:56 +02:00
matQueryVisible->setColourWriteEnabled(false); // Uncomment this to visualize the occlusion query
matQueryVisible->setDepthCheckEnabled(true); // Occluded by objects
2012-03-25 22:53:00 +02:00
matQueryVisible->setCullingMode(CULL_NONE);
matQueryVisible->setManualCullingMode(MANUAL_CULL_NONE);
if (sunNode)
mBBNode = mSunNode->getParentSceneNode()->createChildSceneNode();
2012-03-24 18:38:58 +01:00
2012-03-24 20:41:23 +01:00
mObjectNode = mRendering->getScene()->getRootSceneNode()->createChildSceneNode();
2012-03-31 19:08:05 +02:00
mBBNodeReal = mRendering->getScene()->getRootSceneNode()->createChildSceneNode();
2012-03-24 20:41:23 +01:00
mBBQueryTotal = mRendering->getScene()->createBillboardSet(1);
mBBQueryTotal->setDefaultDimensions(150, 150);
mBBQueryTotal->createBillboard(Vector3::ZERO);
mBBQueryTotal->setMaterialName("QueryTotalPixels");
2012-03-31 20:10:34 +02:00
mBBQueryTotal->setRenderQueueGroup(queue+1);
2012-03-31 19:08:05 +02:00
mBBNodeReal->attachObject(mBBQueryTotal);
mBBQueryVisible = mRendering->getScene()->createBillboardSet(1);
mBBQueryVisible->setDefaultDimensions(150, 150);
mBBQueryVisible->createBillboard(Vector3::ZERO);
mBBQueryVisible->setMaterialName("QueryVisiblePixels");
2012-03-31 20:10:34 +02:00
mBBQueryVisible->setRenderQueueGroup(queue+1);
2012-03-31 19:08:05 +02:00
mBBNodeReal->attachObject(mBBQueryVisible);
2012-03-24 20:41:23 +01:00
mBBQuerySingleObject = mRendering->getScene()->createBillboardSet(1);
2012-03-25 23:28:51 +02:00
/// \todo ideally this should occupy exactly 1 pixel on the screen
mBBQuerySingleObject->setDefaultDimensions(0.003, 0.003);
2012-03-24 20:41:23 +01:00
mBBQuerySingleObject->createBillboard(Vector3::ZERO);
mBBQuerySingleObject->setMaterialName("QueryVisiblePixels");
mBBQuerySingleObject->setRenderQueueGroup(queue);
mObjectNode->attachObject(mBBQuerySingleObject);
mRendering->getScene()->addRenderObjectListener(this);
2012-03-25 20:52:56 +02:00
mRendering->getScene()->addRenderQueueListener(this);
mDoQuery = true;
2012-03-31 19:12:02 +02:00
mDoQuery2 = true;
}
OcclusionQuery::~OcclusionQuery()
{
RenderSystem* renderSystem = Root::getSingleton().getRenderSystem();
if (mSunTotalAreaQuery) renderSystem->destroyHardwareOcclusionQuery(mSunTotalAreaQuery);
if (mSunVisibleAreaQuery) renderSystem->destroyHardwareOcclusionQuery(mSunVisibleAreaQuery);
2012-03-24 20:41:23 +01:00
if (mSingleObjectQuery) renderSystem->destroyHardwareOcclusionQuery(mSingleObjectQuery);
}
bool OcclusionQuery::supported()
{
2012-03-29 00:05:48 +02:00
return mSupported;
}
void OcclusionQuery::notifyRenderSingleObject(Renderable* rend, const Pass* pass, const AutoParamDataSource* source,
const LightList* pLightList, bool suppressRenderStateChanges)
{
// The following code activates and deactivates the occlusion queries
// so that the queries only include the rendering of their intended targets
// 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)
{
2012-03-25 20:52:56 +02:00
if (rend == mBBQueryTotal)
{
mActiveQuery = mSunTotalAreaQuery;
2012-03-25 20:52:56 +02:00
mWasVisible = true;
}
else if (rend == mBBQueryVisible)
2012-03-24 20:41:23 +01:00
{
2012-03-25 20:52:56 +02:00
mActiveQuery = mSunVisibleAreaQuery;
2012-03-24 20:41:23 +01:00
}
}
2012-03-31 20:05:23 +02:00
if (mDoQuery == true && rend == mBBQuerySingleObject)
2012-03-25 20:52:56 +02:00
{
mQuerySingleObjectStarted = true;
mQuerySingleObjectRequested = false;
mActiveQuery = mSingleObjectQuery;
2012-03-26 21:52:38 +02:00
mObjectWasVisible = true;
2012-03-25 20:52:56 +02:00
}
if (mActiveQuery != NULL)
mActiveQuery->beginOcclusionQuery();
}
void OcclusionQuery::renderQueueEnded(uint8 queueGroupId, const String& invocation, bool& repeatThisInvocation)
2012-03-26 21:52:38 +02:00
{
2012-03-31 20:18:20 +02:00
if (mActiveQuery != NULL)
{
mActiveQuery->endOcclusionQuery();
mActiveQuery = NULL;
}
2012-03-26 21:52:38 +02: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
*/
if (queueGroupId == RENDER_QUEUE_SKIES_LATE)
2012-03-25 20:52:56 +02:00
{
2012-03-26 21:52:38 +02:00
if (mWasVisible == false && mDoQuery)
{
mSunTotalAreaQuery->beginOcclusionQuery();
mSunTotalAreaQuery->endOcclusionQuery();
mSunVisibleAreaQuery->beginOcclusionQuery();
mSunVisibleAreaQuery->endOcclusionQuery();
}
2012-03-31 20:05:23 +02:00
if (mObjectWasVisible == false && mDoQuery)
2012-03-26 21:52:38 +02:00
{
mSingleObjectQuery->beginOcclusionQuery();
mSingleObjectQuery->endOcclusionQuery();
mQuerySingleObjectStarted = true;
mQuerySingleObjectRequested = false;
}
2012-03-25 22:53:00 +02:00
}
}
2012-03-26 00:31:03 +02:00
void OcclusionQuery::update(float duration)
{
if (!mSupported) return;
2012-03-25 20:52:56 +02:00
mWasVisible = false;
2012-03-26 21:52:38 +02:00
mObjectWasVisible = false;
2012-03-25 20:52:56 +02:00
2012-03-24 18:38:58 +01: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 (mBBNode)
{
mBBNode->setPosition(mSunNode->getPosition() * dist);
mBBNode->setScale(dist, dist, dist);
mBBNodeReal->setPosition(mBBNode->_getDerivedPosition());
mBBNodeReal->setScale(mBBNode->getScale());
}
2012-03-24 18:38:58 +01:00
// Stop occlusion queries until we get their information
// (may not happen on the same frame they are requested in)
mDoQuery = false;
2012-03-31 19:12:02 +02:00
mDoQuery2 = false;
2012-03-24 20:41:23 +01:00
if (!mSunTotalAreaQuery->isStillOutstanding()
2012-03-31 20:05:23 +02:00
&& !mSunVisibleAreaQuery->isStillOutstanding()
&& !mSingleObjectQuery->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-25 20:52:56 +02:00
unsigned int result;
mSingleObjectQuery->pullOcclusionQuery(&result);
2012-03-24 20:41:23 +01:00
2012-03-25 20:52:56 +02:00
mTestResult = (result != 0);
2012-03-24 20:41:23 +01:00
2012-03-25 20:52:56 +02:00
mQuerySingleObjectStarted = false;
mQuerySingleObjectRequested = false;
2012-03-31 19:12:02 +02:00
2012-03-31 20:05:23 +02:00
mDoQuery = true;
}
}
2012-03-25 20:52:56 +02:00
void OcclusionQuery::occlusionTest(const Ogre::Vector3& position, Ogre::SceneNode* object)
2012-03-24 20:41:23 +01:00
{
assert( !occlusionTestPending()
&& "Occlusion test still pending");
mBBQuerySingleObject->setVisible(true);
mObjectNode->setPosition(position);
2012-03-25 20:52:56 +02:00
// scale proportional to camera distance, in order to always give the billboard the same size in screen-space
mObjectNode->setScale( Vector3(1,1,1)*(position - mRendering->getCamera()->getRealPosition()).length() );
2012-03-24 20:41:23 +01:00
mQuerySingleObjectRequested = true;
}
bool OcclusionQuery::occlusionTestPending()
{
return (mQuerySingleObjectRequested || mQuerySingleObjectStarted);
}
void OcclusionQuery::setSunNode(Ogre::SceneNode* node)
{
mSunNode = node;
if (!mBBNode)
mBBNode = node->getParentSceneNode()->createChildSceneNode();
}
2012-03-24 20:41:23 +01:00
bool OcclusionQuery::getTestResult()
{
assert( !occlusionTestPending()
&& "Occlusion test still pending");
return mTestResult;
}