From 0339e32c0cae4fd29f922bc882ea35ee788100d3 Mon Sep 17 00:00:00 2001
From: psi29a <psi29a@gmail.com>
Date: Mon, 8 Aug 2022 21:10:40 +0000
Subject: [PATCH] Merge branch 'ubb_fix' into 'master'

Remove excessive allocations in lightmanager

See merge request OpenMW/openmw!2251
---
 components/sceneutil/lightmanager.cpp | 34 ++++++++++++++++++---------
 1 file changed, 23 insertions(+), 11 deletions(-)

diff --git a/components/sceneutil/lightmanager.cpp b/components/sceneutil/lightmanager.cpp
index be3fe4679b..e24cdcf6f3 100644
--- a/components/sceneutil/lightmanager.cpp
+++ b/components/sceneutil/lightmanager.cpp
@@ -476,16 +476,14 @@ namespace SceneUtil
         {
             osg::ref_ptr<osg::StateSet> stateset = new osg::StateSet;
 
-            osg::ref_ptr<osg::IntArray> indices = new osg::IntArray(mLightManager->getMaxLights());
-            osg::ref_ptr<osg::Uniform> indicesUni = new osg::Uniform(osg::Uniform::Type::INT, "PointLightIndex", indices->size());
+            osg::ref_ptr<osg::Uniform> indicesUni = new osg::Uniform(osg::Uniform::Type::INT, "PointLightIndex", mLightManager->getMaxLights());
             int pointCount = 0;
 
             for (size_t i = 0; i < lightList.size(); ++i)
             {
                 int bufIndex = mLightManager->getLightIndexMap(frameNum)[lightList[i]->mLightSource->getId()];
-                indices->at(pointCount++) = bufIndex;
+                indicesUni->setElement(pointCount++, bufIndex);
             }
-            indicesUni->setArray(indices);
             stateset->addUniform(indicesUni);
             stateset->addUniform(new osg::Uniform("PointLightCount", pointCount));
 
@@ -608,20 +606,32 @@ namespace SceneUtil
     class LightManagerCullCallback : public SceneUtil::NodeCallback<LightManagerCullCallback, LightManager*, osgUtil::CullVisitor*>
     {
     public:
-        void operator()(LightManager* node, osgUtil::CullVisitor* cv)
+        LightManagerCullCallback(LightManager* lightManager)
         {
-            osg::ref_ptr<osg::StateSet> stateset = new osg::StateSet;
+            if (!lightManager->getUBOManager())
+                return;
 
-            if (node->getLightingMethod() == LightingMethod::SingleUBO)
+            for (size_t i = 0; i < mStateSet.size(); ++i)
             {
-                auto buffer = node->getUBOManager()->getLightBuffer(cv->getTraversalNumber());
-
+                auto& buffer = lightManager->getUBOManager()->getLightBuffer(i);
 #if OSG_VERSION_GREATER_OR_EQUAL(3,5,7)
                 osg::ref_ptr<osg::UniformBufferBinding> ubb = new osg::UniformBufferBinding(static_cast<int>(Resource::SceneManager::UBOBinding::LightBuffer), buffer->getData(), 0, buffer->getData()->getTotalDataSize());
 #else
                 osg::ref_ptr<osg::UniformBufferBinding> ubb = new osg::UniformBufferBinding(static_cast<int>(Resource::SceneManager::UBOBinding::LightBuffer), buffer->getData()->getBufferObject(), 0, buffer->getData()->getTotalDataSize());
 #endif
-                stateset->setAttributeAndModes(ubb, osg::StateAttribute::ON);
+                mStateSet[i]->setAttributeAndModes(ubb, osg::StateAttribute::ON);
+            }
+        }
+
+        void operator()(LightManager* node, osgUtil::CullVisitor* cv)
+        {
+            const size_t frameId = cv->getTraversalNumber() % 2;
+
+            auto& stateset = mStateSet[frameId];
+
+            if (node->getLightingMethod() == LightingMethod::SingleUBO)
+            {
+                auto& buffer = node->getUBOManager()->getLightBuffer(cv->getTraversalNumber());
 
                 if (auto sun = node->getSunlight())
                 {
@@ -652,6 +662,8 @@ namespace SceneUtil
             if (node->getPPLightsBuffer() && cv->getCurrentCamera()->getName() == Constants::SceneCamera)
                 node->getPPLightsBuffer()->updateCount(cv->getTraversalNumber());
         }
+
+        std::array<osg::ref_ptr<osg::StateSet>, 2> mStateSet = { new osg::StateSet, new osg::StateSet };
     };
 
     UBOManager::UBOManager(int lightCount)
@@ -838,7 +850,7 @@ namespace SceneUtil
 
         getOrCreateStateSet()->addUniform(new osg::Uniform("PointLightCount", 0));
 
-        addCullCallback(new LightManagerCullCallback);
+        addCullCallback(new LightManagerCullCallback(this));
     }
 
     LightManager::LightManager(const LightManager &copy, const osg::CopyOp &copyop)