You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ryzom-core/nel/src/3d/driver.cpp

461 lines
13 KiB
C++

// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
// Copyright (C) 2010 Winch Gate Property Limited
//
// This source file has been modified by the following contributors:
// Copyright (C) 2013-2020 Jan BOON (Kaetemi) <jan.boon@kaetemi.be>
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
#include "std3d.h"
#include <string>
#include "nel/misc/types_nl.h"
#include "nel/3d/driver.h"
#include "nel/3d/vertex_buffer.h"
#include "nel/misc/algo.h"
//#include <cstdio>
using namespace NLMISC;
using namespace std;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D
{
// ***************************************************************************
const uint32 IDriver::InterfaceVersion = 0x70; // total video memory
// ***************************************************************************
IDriver::IDriver() : _SyncTexDrvInfos( "IDriver::_SyncTexDrvInfos" )
{
_PolygonMode= Filled;
_StaticMemoryToVRAM=false;
_ResetCounter=0;
}
// ***************************************************************************
IDriver::~IDriver()
{
// Must clean up everything before closing driver.
// Must doing this in release(), so assert here if not done...
{
CSynchronized<TTexDrvInfoPtrMap>::CAccessor access(&_SyncTexDrvInfos);
TTexDrvInfoPtrMap &rTexDrvInfos = access.value();
nlassert(rTexDrvInfos.empty());
}
nlassert(_TexDrvShares.empty());
nlassert(_MatDrvInfos.empty());
nlassert(_VBDrvInfos.empty());
nlassert(_IBDrvInfos.empty());
nlassert(_GPUPrgDrvInfos.empty());
}
// ***************************************************************************
bool IDriver::release(void)
{
// Called by derived classes.
// DO THIS FIRST => to auto kill real textures (by smartptr).
// First, Because must not kill a pointer owned by a CSmartPtr.
// Release Textures drv.
ItTexDrvSharePtrList ittex;
while( (ittex = _TexDrvShares.begin()) !=_TexDrvShares.end() )
{
// NB: at CTextureDrvShare deletion, this->_TexDrvShares is updated (entry deleted);
delete *ittex;
}
// Release refptr of TextureDrvInfos. Should be all null (because of precedent pass).
{
CSynchronized<TTexDrvInfoPtrMap>::CAccessor access(&_SyncTexDrvInfos);
TTexDrvInfoPtrMap &rTexDrvInfos = access.value();
// must be empty, because precedent pass should have deleted all.
nlassert(rTexDrvInfos.empty());
}
// Release material drv.
ItMatDrvInfoPtrList itmat;
while( (itmat = _MatDrvInfos.begin()) != _MatDrvInfos.end() )
{
// NB: at IShader deletion, this->_MatDrvInfos is updated (entry deleted);
delete *itmat;
}
// Release VBs drv.
ItVBDrvInfoPtrList itvb;
while( (itvb = _VBDrvInfos.begin()) != _VBDrvInfos.end() )
{
// NB: at IVBDrvInfo deletion, this->_VBDrvInfos is updated (entry deleted);
delete *itvb;
}
// Release IBs drv.
ItIBDrvInfoPtrList itib;
while( (itib = _IBDrvInfos.begin()) != _IBDrvInfos.end() )
{
// NB: at IIBDrvInfo deletion, this->_IBDrvInfos is updated (entry deleted);
delete *itib;
}
// Release GPUPrg drv.
ItGPUPrgDrvInfoPtrList itGPUPrg;
while( (itGPUPrg = _GPUPrgDrvInfos.begin()) != _GPUPrgDrvInfos.end() )
{
// NB: at IVertexProgramDrvInfos deletion, this->_GPUPrgDrvInfos is updated (entry deleted);
delete *itGPUPrg;
}
return true;
}
// ***************************************************************************
GfxMode::GfxMode(uint16 w, uint16 h, uint8 d, bool windowed, bool offscreen, uint frequency, sint8 aa, const std::string &displayDevice)
{
DisplayDevice = displayDevice;
Windowed = windowed;
Width = w;
Height = h;
Depth = d;
OffScreen = offscreen;
Frequency = frequency;
AntiAlias = aa;
}
// ***************************************************************************
IDriver::TMessageBoxId IDriver::systemMessageBox (const char* message, const char* title, IDriver::TMessageBoxType type, IDriver::TMessageBoxIcon icon)
{
static const char* icons[iconCount]=
{
"",
"WAIT:\n",
"QUESTION:\n",
"HEY!\n",
"",
"WARNING!\n",
"ERROR!\n",
"INFORMATION:\n",
"STOP:\n"
};
static const char* messages[typeCount]=
{
"Press any key...",
"(O)k or (C)ancel ?",
"(Y)es or (N)o ?",
"(A)bort (R)etry (I)gnore ?",
"(Y)es (N)o (C)ancel ?",
"(R)etry (C)ancel ?"
};
printf ("%s%s\n%s", icons[icon], title, message);
for(;;)
{
printf ("\n%s", messages[type]);
int c=getchar();
if (type==okType)
return okId;
switch (c)
{
case 'O':
case 'o':
if ((type==okType)||(type==okCancelType))
return okId;
break;
case 'C':
case 'c':
if ((type==yesNoCancelType)||(type==okCancelType)||(type==retryCancelType))
return cancelId;
break;
case 'Y':
case 'y':
if ((type==yesNoCancelType)||(type==yesNoType))
return yesId;
break;
case 'N':
case 'n':
if ((type==yesNoCancelType)||(type==yesNoType))
return noId;
break;
case 'A':
case 'a':
if (type==abortRetryIgnoreType)
return abortId;
break;
case 'R':
case 'r':
if (type==abortRetryIgnoreType)
return retryId;
break;
case 'I':
case 'i':
if (type==abortRetryIgnoreType)
return ignoreId;
break;
}
}
nlassert (0); // no!
return okId;
}
// ***************************************************************************
void IDriver::removeVBDrvInfoPtr(ItVBDrvInfoPtrList vbDrvInfoIt)
{
_VBDrvInfos.erase(vbDrvInfoIt);
}
// ***************************************************************************
void IDriver::removeIBDrvInfoPtr(ItIBDrvInfoPtrList ibDrvInfoIt)
{
_IBDrvInfos.erase(ibDrvInfoIt);
}
// ***************************************************************************
void IDriver::removeTextureDrvInfoPtr(ItTexDrvInfoPtrMap texDrvInfoIt)
{
CSynchronized<TTexDrvInfoPtrMap>::CAccessor access(&_SyncTexDrvInfos);
TTexDrvInfoPtrMap &rTexDrvInfos = access.value();
rTexDrvInfos.erase(texDrvInfoIt);
}
// ***************************************************************************
void IDriver::removeTextureDrvSharePtr(ItTexDrvSharePtrList texDrvShareIt)
{
_TexDrvShares.erase(texDrvShareIt);
}
// ***************************************************************************
void IDriver::removeMatDrvInfoPtr(ItMatDrvInfoPtrList shaderIt)
{
_MatDrvInfos.erase(shaderIt);
}
// ***************************************************************************
void IDriver::removeGPUPrgDrvInfoPtr(ItGPUPrgDrvInfoPtrList gpuPrgDrvInfoIt)
{
_GPUPrgDrvInfos.erase(gpuPrgDrvInfoIt);
}
// ***************************************************************************
bool IDriver::invalidateShareTexture (ITexture &texture)
{
// Create the shared Name.
std::string name;
getTextureShareName (texture, name);
// Look for the driver info for this share name
CSynchronized<TTexDrvInfoPtrMap>::CAccessor access(&_SyncTexDrvInfos);
TTexDrvInfoPtrMap &rTexDrvInfos = access.value();
TTexDrvInfoPtrMap::iterator iteDrvInfo = rTexDrvInfos.find (name);
if (iteDrvInfo != rTexDrvInfos.end())
{
// Now parse all shared info
TTexDrvSharePtrList::iterator shareIte = _TexDrvShares.begin ();
while (shareIte != _TexDrvShares.end ())
{
// Good one ?
if ((*shareIte)->DrvTexture == iteDrvInfo->second)
{
// Remove this one
TTexDrvSharePtrList::iterator toRemove = shareIte;
shareIte++;
delete (*toRemove);
}
else
shareIte++;
}
// Ok
return true;
}
return false;
}
// ***************************************************************************
void IDriver::getTextureShareName (const ITexture& tex, string &output)
{
// Create the shared Name.
output= toLowerAscii(tex.getShareName());
// append format Id of the texture.
static char fmt[256];
smprintf(fmt, 256, "@Fmt:%d", (uint32)tex.getUploadFormat());
output+= fmt;
// append mipmap info
if(tex.mipMapOn())
output+= "@MMp:On";
else
output+= "@MMp:Off";
}
// ***************************************************************************
void IDriver::setStaticMemoryToVRAM (bool staticMemoryToVRAM)
{
_StaticMemoryToVRAM=staticMemoryToVRAM;
}
// ***************************************************************************
class CTextureDebugInfo
{
public:
uint MemoryCost;
string Line;
bool operator<(const CTextureDebugInfo &o) const {return Line<o.Line;}
};
class CTextureDebugKey
{
public:
ITexture::TUploadFormat UpLoadFormat;
ITexture::CTextureCategory *Category;
bool operator<(const CTextureDebugKey &o) const
{
const string &s0= Category?Category->Name:_EmptyString;
const string &s1= o.Category?o.Category->Name:_EmptyString;
if(s0 == s1)
return UpLoadFormat<o.UpLoadFormat;
else
return s0<s1;
}
private:
static std::string _EmptyString;
};
std::string CTextureDebugKey::_EmptyString;
// ***************************************************************************
void IDriver::profileTextureUsage(std::vector<std::string> &result)
{
std::set<ITextureDrvInfos *> texSet;
// uint i;
// reserve result, sort by UploadFormat
map<CTextureDebugKey, vector<CTextureDebugInfo> > tempInfo;
// Parse all the DrvShare list
uint totalSize= 0;
ItTexDrvSharePtrList it= _TexDrvShares.begin();
for(;it!=_TexDrvShares.end();it++)
{
// get TexDrvInfos and owner
ITextureDrvInfos *gltext= (ITextureDrvInfos*)(ITextureDrvInfos*)(*it)->DrvTexture;
ITexture *text= (*it)->getOwnerTexture();
nlassert(gltext && text);
// sort by upload format and texture category
CTextureDebugKey infoKey;
infoKey.UpLoadFormat= text->getUploadFormat();
nlassert(infoKey.UpLoadFormat<ITexture::UploadFormatCount);
infoKey.Category= text->getTextureCategory();
// get the shareName
string shareName;
if(text->supportSharing())
shareName= toLowerAscii(text->getShareName());
else
shareName= "Not Shared";
// only if not already append to the set
if(texSet.insert(gltext).second)
{
uint memCost= gltext->getTextureMemoryUsed();
totalSize+= memCost;
string typeStr= typeid(*text).name();
strFindReplace(typeStr, "class NL3D::", string());
tempInfo[infoKey].push_back(CTextureDebugInfo());
tempInfo[infoKey].back().Line= toString("Type: %15s. ShareName: %s. Size: %d Ko",
typeStr.c_str(),
shareName.c_str(),
memCost/1024);
tempInfo[infoKey].back().MemoryCost= memCost;
}
}
// For convenience, sort
map<CTextureDebugKey, vector<CTextureDebugInfo> >::iterator itCat;
for(itCat= tempInfo.begin();itCat!= tempInfo.end();itCat++)
sort(itCat->second.begin(), itCat->second.end());
// Store into result, appending Tag for each Mo reached. +10* is for extra lines and security
result.clear();
result.reserve(texSet.size() + 10*(tempInfo.size()) + totalSize/(1024*1024));
// copy and add tags
for(itCat= tempInfo.begin();itCat!= tempInfo.end();itCat++)
{
const CTextureDebugKey &infoKey= itCat->first;
vector<CTextureDebugInfo> &infoVect= itCat->second;
string strUploadFormat;
switch(infoKey.UpLoadFormat)
{
case ITexture::Auto: strUploadFormat= ("Format: Auto"); break;
case ITexture::RGBA8888: strUploadFormat= ("Format: RGBA8888"); break;
case ITexture::RGBA4444: strUploadFormat= ("Format: RGBA4444"); break;
case ITexture::RGBA5551: strUploadFormat= ("Format: RGBA5551"); break;
case ITexture::RGB888: strUploadFormat= ("Format: RGB888"); break;
case ITexture::RGB565: strUploadFormat= ("Format: RGB565"); break;
case ITexture::DXTC1: strUploadFormat= ("Format: DXTC1"); break;
case ITexture::DXTC1Alpha: strUploadFormat= ("Format: DXTC1Alpha"); break;
case ITexture::DXTC3: strUploadFormat= ("Format: DXTC3"); break;
case ITexture::DXTC5: strUploadFormat= ("Format: DXTC5"); break;
case ITexture::Luminance: strUploadFormat= ("Format: Luminance"); break;
case ITexture::Alpha: strUploadFormat= ("Format: Alpha"); break;
case ITexture::AlphaLuminance: strUploadFormat= ("Format: AlphaLuminance"); break;
case ITexture::DsDt: strUploadFormat= ("Format: DsDt"); break;
default: strUploadFormat= toString("Format??: %d", infoKey.UpLoadFormat); break;
}
// header info
result.push_back(toString("**** %s. %s ****", infoKey.Category?infoKey.Category->Name.c_str():"",
strUploadFormat.c_str()) );
// display stats for this format
uint tagTotal= 0;
uint curTotal= 0;
for(uint j=0;j<infoVect.size();j++)
{
result.push_back(infoVect[j].Line);
tagTotal+= infoVect[j].MemoryCost;
curTotal+= infoVect[j].MemoryCost;
if(tagTotal>=1024*1024)
{
result.push_back(toString("---- %.1f Mo", float(curTotal)/(1024*1024)));
tagTotal= 0;
}
}
// append last line?
if(tagTotal!=0)
result.push_back(toString("---- %.1f Mo", float(curTotal)/(1024*1024)));
}
// append the total
result.push_back(toString("**** Total ****"));
result.push_back(toString("Total: %d Ko", totalSize/1024));
}
// ***************************************************************************
}