Fix SMaterialLayer operator!= and optimize operator=

SMaterialLayers are now identical when both have identity matrices.
Before it didn't consider them identical when one layer had set the identity matrix explicitly and the other didn't.
Generally didn't matter, just caused very rarely some extra state switches in the drivers. And just as rarely had a cheaper comparison. Just seems more correct this way.

operator= no longer releases texture memory which was allocated at one point. 
Unless explicitly requested such memory is now always released later in the destructor.
This can avoid quite a few memory allocations/released in the driver. Usually not a noticeable performance difference on most platforms. But it can help avoid memory fragmentation.
We instead use an extra bool now to tell if the texture memory is used. So slight increase in SMaterialLayer and SMaterial size. But I did a quick performance test and this had no negative influence here, while it did improve speed in the case where it switched between material layers using/not using texture matrices a bit.

git-svn-id: svn://svn.code.sf.net/p/irrlicht/code/trunk@6488 dfc29bdd-3216-0410-991c-e03cc46cb475
This commit is contained in:
cutealien 2023-05-07 14:51:09 +00:00
parent c57da57edc
commit 9ce63bc7d3
4 changed files with 67 additions and 22 deletions

@ -1,6 +1,10 @@
-------------------------- --------------------------
Changes in 1.9 (not yet released) Changes in 1.9 (not yet released)
- Bugfix: SMaterialLayer::operator!= no longer returns true when comparing a layer without texture matrix with one with a set identity texture matrix. Those are the same.
- SMaterialLayer no longer releases allocated texture memory before destructor unless explicitly requested.
This can avoid constantly allocation/releasing memory in the active driver material when setting materials.
Note that it does not copy texture memory when it's unused. This is still optimized.
- Add IMeshSceneNode::setNodeRegistration to allow registering MeshSceneNodes to the SceneManager per buffer instead of per node - Add IMeshSceneNode::setNodeRegistration to allow registering MeshSceneNodes to the SceneManager per buffer instead of per node
- Add SMaterialLayer::hasSetTextureMatrix and SMaterialLayer::resetTextureMatrix - Add SMaterialLayer::hasSetTextureMatrix and SMaterialLayer::resetTextureMatrix
- Add IShaderConstantSetCallBack::OnCreate to allow earlier access to IMaterialRendererServices - Add IShaderConstantSetCallBack::OnCreate to allow earlier access to IMaterialRendererServices

@ -51,7 +51,8 @@ namespace video
public: public:
//! Default constructor //! Default constructor
SMaterialLayer() : Texture(0), TextureWrapU(ETC_REPEAT), TextureWrapV(ETC_REPEAT), TextureWrapW(ETC_REPEAT), SMaterialLayer() : Texture(0), TextureWrapU(ETC_REPEAT), TextureWrapV(ETC_REPEAT), TextureWrapW(ETC_REPEAT),
BilinearFilter(true), TrilinearFilter(false), AnisotropicFilter(0), LODBias(0), TextureMatrix(0) BilinearFilter(true), TrilinearFilter(false), AnisotropicFilter(0), LODBias(0),
TextureMatrix(0), TextureMatrixUsed(false)
{ {
} }
@ -84,26 +85,22 @@ namespace video
return *this; return *this;
Texture = other.Texture; Texture = other.Texture;
if (TextureMatrix) if (other.TextureMatrixUsed)
{ {
if (other.TextureMatrix) if (TextureMatrix)
*TextureMatrix = *other.TextureMatrix;
else
{ {
MatrixAllocator.destruct(TextureMatrix); *TextureMatrix = *other.TextureMatrix;
MatrixAllocator.deallocate(TextureMatrix);
TextureMatrix = 0;
} }
} else
else
{
if (other.TextureMatrix)
{ {
TextureMatrix = MatrixAllocator.allocate(1); TextureMatrix = MatrixAllocator.allocate(1);
MatrixAllocator.construct(TextureMatrix,*other.TextureMatrix); MatrixAllocator.construct(TextureMatrix,*other.TextureMatrix);
} }
else TextureMatrixUsed = true;
TextureMatrix = 0; }
else
{
TextureMatrixUsed = false;
} }
TextureWrapU = other.TextureWrapU; TextureWrapU = other.TextureWrapU;
TextureWrapV = other.TextureWrapV; TextureWrapV = other.TextureWrapV;
@ -124,6 +121,12 @@ namespace video
{ {
TextureMatrix = MatrixAllocator.allocate(1); TextureMatrix = MatrixAllocator.allocate(1);
MatrixAllocator.construct(TextureMatrix,core::IdentityMatrix); MatrixAllocator.construct(TextureMatrix,core::IdentityMatrix);
TextureMatrixUsed = true;
}
else if ( !TextureMatrixUsed )
{
*TextureMatrix = core::IdentityMatrix;
TextureMatrixUsed = true;
} }
return *TextureMatrix; return *TextureMatrix;
} }
@ -132,7 +135,7 @@ namespace video
/** \return Texture matrix of this layer. */ /** \return Texture matrix of this layer. */
const core::matrix4& getTextureMatrix() const const core::matrix4& getTextureMatrix() const
{ {
if (TextureMatrix) if (TextureMatrixUsed)
return *TextureMatrix; return *TextureMatrix;
else else
return core::IdentityMatrix; return core::IdentityMatrix;
@ -151,25 +154,27 @@ namespace video
} }
else else
*TextureMatrix = mat; *TextureMatrix = mat;
TextureMatrixUsed = true;
} }
//! Check if we have set a custom texture matrix //! Check if we have set a custom texture matrix
//! Note that otherwise we get an IdentityMatrix as default //! Note that otherwise we get an IdentityMatrix as default
inline bool hasSetTextureMatrix() const inline bool hasSetTextureMatrix() const
{ {
return TextureMatrix != 0; return TextureMatrixUsed;
} }
//! Reset texture matrix to identity matrix //! Reset texture matrix to identity matrix
//! Releases memory, which is expensive, but ver rarely useful for optimizations /** \param releaseMemory Releases also texture memory. Otherwise done in destructor */
void resetTextureMatrix() void resetTextureMatrix(bool releaseMemory=true)
{ {
if ( TextureMatrix ) if ( TextureMatrix && releaseMemory)
{ {
MatrixAllocator.destruct(TextureMatrix); MatrixAllocator.destruct(TextureMatrix);
MatrixAllocator.deallocate(TextureMatrix); MatrixAllocator.deallocate(TextureMatrix);
TextureMatrix = 0; TextureMatrix = 0;
} }
TextureMatrixUsed = false;
} }
//! Inequality operator //! Inequality operator
@ -189,8 +194,11 @@ namespace video
if (different) if (different)
return true; return true;
else else
different |= (TextureMatrix != b.TextureMatrix) && {
(!TextureMatrix || !b.TextureMatrix || (*TextureMatrix != *(b.TextureMatrix))); different = (TextureMatrixUsed && b.TextureMatrixUsed && (*TextureMatrix != *b.TextureMatrix))
|| (TextureMatrixUsed && !b.TextureMatrixUsed && (*TextureMatrix != core::IdentityMatrix))
|| (!TextureMatrixUsed && b.TextureMatrixUsed && (core::IdentityMatrix != *b.TextureMatrix));
}
return different; return different;
} }
@ -241,6 +249,7 @@ namespace video
/** Do not access this element directly as the internal /** Do not access this element directly as the internal
resource management has to cope with Null pointers etc. */ resource management has to cope with Null pointers etc. */
core::matrix4* TextureMatrix; core::matrix4* TextureMatrix;
bool TextureMatrixUsed; // TextureMatrix memory stays until destructor even when unused to avoid unnecessary allocation/de-allocations
}; };
} // end namespace video } // end namespace video

@ -71,9 +71,41 @@ static bool polygonOffset(video::E_DRIVER_TYPE type)
return result; return result;
} }
static bool testSMaterial()
{
irr::video::SMaterial a;
irr::video::SMaterial b;
// Same by default?
if ( !(a == b) )
return false;
if ( a != b )
return false;
// getting (creating) one texture matrix shouldn't change things
b.TextureLayer[0].getTextureMatrix();
if ( a != b )
return false;
// no longer same now
b.TextureLayer[0].getTextureMatrix().setTextureScale(5.f, 0.5f);
if ( a == b )
return false;
return true;
}
bool material() bool material()
{ {
bool result = true; bool result = true;
TestWithAllDrivers(polygonOffset); TestWithAllDrivers(polygonOffset);
if ( !testSMaterial() )
{
logTestString("testSMaterial failed\n\n");
result = false;
}
return result; return result;
} }

@ -1,4 +1,4 @@
Tests finished. 72 tests of 72 passed. Tests finished. 72 tests of 72 passed.
Compiled as DEBUG Compiled as DEBUG
Test suite pass at GMT Fri May 05 18:39:44 2023 Test suite pass at GMT Sun May 07 14:26:39 2023