2016-02-05 23:03:53 +01:00
|
|
|
#include "imagemanager.hpp"
|
2015-03-28 02:20:20 +01:00
|
|
|
|
2017-05-05 21:21:11 +02:00
|
|
|
#include <cassert>
|
2015-03-28 02:20:20 +01:00
|
|
|
#include <osgDB/Registry>
|
|
|
|
|
2018-08-14 19:42:41 +04:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2021-09-11 15:49:47 +02:00
|
|
|
#include <components/misc/pathhelpers.hpp>
|
2015-03-28 02:20:20 +01:00
|
|
|
#include <components/vfs/manager.hpp>
|
|
|
|
|
2016-02-05 23:31:59 +01:00
|
|
|
#include "objectcache.hpp"
|
|
|
|
|
2015-07-17 15:13:43 -05:00
|
|
|
#ifdef OSG_LIBRARY_STATIC
|
|
|
|
// This list of plugins should match with the list in the top-level CMakelists.txt.
|
|
|
|
USE_OSGPLUGIN(png)
|
|
|
|
USE_OSGPLUGIN(tga)
|
|
|
|
USE_OSGPLUGIN(dds)
|
|
|
|
USE_OSGPLUGIN(jpeg)
|
2016-10-08 19:20:44 +02:00
|
|
|
USE_OSGPLUGIN(bmp)
|
|
|
|
USE_OSGPLUGIN(osg)
|
|
|
|
USE_SERIALIZER_WRAPPER_LIBRARY(osg)
|
2015-07-17 15:13:43 -05:00
|
|
|
#endif
|
|
|
|
|
2015-03-28 02:57:39 +01:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
2016-02-05 23:05:37 +01:00
|
|
|
osg::ref_ptr<osg::Image> createWarningImage()
|
2015-03-28 02:57:39 +01:00
|
|
|
{
|
|
|
|
osg::ref_ptr<osg::Image> warningImage = new osg::Image;
|
|
|
|
|
2015-05-02 18:06:17 +02:00
|
|
|
int width = 8, height = 8;
|
|
|
|
warningImage->allocateImage(width, height, 1, GL_RGB, GL_UNSIGNED_BYTE);
|
|
|
|
assert (warningImage->isDataContiguous());
|
|
|
|
unsigned char* data = warningImage->data();
|
2015-03-28 02:57:39 +01:00
|
|
|
for (int i=0;i<width*height;++i)
|
|
|
|
{
|
2015-05-02 18:06:17 +02:00
|
|
|
data[3*i] = (255);
|
|
|
|
data[3*i+1] = (0);
|
|
|
|
data[3*i+2] = (255);
|
2015-03-28 02:57:39 +01:00
|
|
|
}
|
2016-02-05 23:05:37 +01:00
|
|
|
return warningImage;
|
2015-03-28 02:57:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-03-28 02:20:20 +01:00
|
|
|
namespace Resource
|
|
|
|
{
|
|
|
|
|
2016-02-05 23:03:53 +01:00
|
|
|
ImageManager::ImageManager(const VFS::Manager *vfs)
|
2016-02-06 16:57:54 +01:00
|
|
|
: ResourceManager(vfs)
|
2016-02-05 23:05:37 +01:00
|
|
|
, mWarningImage(createWarningImage())
|
2020-04-29 14:14:53 +03:00
|
|
|
, mOptions(new osgDB::Options("dds_flip dds_dxt1_detect_rgba ignoreTga2Fields"))
|
2015-03-28 02:20:20 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-02-05 23:03:53 +01:00
|
|
|
ImageManager::~ImageManager()
|
2015-05-08 17:52:35 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-07-28 23:35:10 +02:00
|
|
|
bool checkSupported(osg::Image* image, const std::string& filename)
|
|
|
|
{
|
|
|
|
switch(image->getPixelFormat())
|
|
|
|
{
|
|
|
|
case(GL_COMPRESSED_RGB_S3TC_DXT1_EXT):
|
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT1_EXT):
|
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT3_EXT):
|
|
|
|
case(GL_COMPRESSED_RGBA_S3TC_DXT5_EXT):
|
|
|
|
{
|
2015-07-30 11:45:10 +02:00
|
|
|
osg::GLExtensions* exts = osg::GLExtensions::Get(0, false);
|
|
|
|
if (exts && !exts->isTextureCompressionS3TCSupported
|
|
|
|
// This one works too. Should it be included in isTextureCompressionS3TCSupported()? Submitted as a patch to OSG.
|
|
|
|
&& !osg::isGLExtensionSupported(0, "GL_S3_s3tc"))
|
2015-07-28 23:35:10 +02:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// not bothering with checks for other compression formats right now, we are unlikely to ever use those anyway
|
|
|
|
default:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-02-05 23:03:53 +01:00
|
|
|
osg::ref_ptr<osg::Image> ImageManager::getImage(const std::string &filename)
|
2015-11-16 23:26:43 +01:00
|
|
|
{
|
2021-09-06 22:01:41 +02:00
|
|
|
const std::string normalized = mVFS->normalizeFilename(filename);
|
2016-02-05 23:31:59 +01:00
|
|
|
|
|
|
|
osg::ref_ptr<osg::Object> obj = mCache->getRefFromObjectCache(normalized);
|
|
|
|
if (obj)
|
|
|
|
return osg::ref_ptr<osg::Image>(static_cast<osg::Image*>(obj.get()));
|
2015-11-16 23:26:43 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
Files::IStreamPtr stream;
|
|
|
|
try
|
|
|
|
{
|
2021-06-23 22:19:08 +02:00
|
|
|
stream = mVFS->get(normalized);
|
2015-11-16 23:26:43 +01:00
|
|
|
}
|
|
|
|
catch (std::exception& e)
|
|
|
|
{
|
2018-08-14 19:42:41 +04:00
|
|
|
Log(Debug::Error) << "Failed to open image: " << e.what();
|
2016-02-05 23:31:59 +01:00
|
|
|
mCache->addEntryToObjectCache(normalized, mWarningImage);
|
2016-02-05 22:58:02 +01:00
|
|
|
return mWarningImage;
|
2015-11-16 23:26:43 +01:00
|
|
|
}
|
|
|
|
|
2021-09-11 15:49:47 +02:00
|
|
|
const std::string ext(Misc::getFileExtension(normalized));
|
2015-11-16 23:26:43 +01:00
|
|
|
osgDB::ReaderWriter* reader = osgDB::Registry::instance()->getReaderWriterForExtension(ext);
|
|
|
|
if (!reader)
|
|
|
|
{
|
2018-08-14 19:42:41 +04:00
|
|
|
Log(Debug::Error) << "Error loading " << filename << ": no readerwriter for '" << ext << "' found";
|
2016-02-05 23:31:59 +01:00
|
|
|
mCache->addEntryToObjectCache(normalized, mWarningImage);
|
2016-02-05 22:58:02 +01:00
|
|
|
return mWarningImage;
|
2015-11-16 23:26:43 +01:00
|
|
|
}
|
|
|
|
|
2020-09-12 00:20:44 +01:00
|
|
|
bool killAlpha = false;
|
|
|
|
if (reader->supportedExtensions().count("tga"))
|
|
|
|
{
|
|
|
|
// Morrowind ignores the alpha channel of 16bpp TGA files even when the header says not to
|
|
|
|
unsigned char header[18];
|
|
|
|
stream->read((char*)header, 18);
|
|
|
|
if (stream->gcount() != 18)
|
|
|
|
{
|
|
|
|
Log(Debug::Error) << "Error loading " << filename << ": couldn't read TGA header";
|
|
|
|
mCache->addEntryToObjectCache(normalized, mWarningImage);
|
|
|
|
return mWarningImage;
|
|
|
|
}
|
|
|
|
int type = header[2];
|
|
|
|
int depth;
|
|
|
|
if (type == 1 || type == 9)
|
|
|
|
depth = header[7];
|
|
|
|
else
|
|
|
|
depth = header[16];
|
|
|
|
int alphaBPP = header[17] & 0x0F;
|
|
|
|
killAlpha = depth == 16 && alphaBPP == 1;
|
|
|
|
stream->seekg(0);
|
|
|
|
}
|
|
|
|
|
2016-02-05 20:23:41 +01:00
|
|
|
osgDB::ReaderWriter::ReadResult result = reader->readImage(*stream, mOptions);
|
2015-11-16 23:26:43 +01:00
|
|
|
if (!result.success())
|
|
|
|
{
|
2018-08-14 19:42:41 +04:00
|
|
|
Log(Debug::Error) << "Error loading " << filename << ": " << result.message() << " code " << result.status();
|
2016-02-05 23:31:59 +01:00
|
|
|
mCache->addEntryToObjectCache(normalized, mWarningImage);
|
2016-02-05 22:58:02 +01:00
|
|
|
return mWarningImage;
|
2015-11-16 23:26:43 +01:00
|
|
|
}
|
|
|
|
|
2017-10-14 18:28:46 +00:00
|
|
|
osg::ref_ptr<osg::Image> image = result.getImage();
|
|
|
|
|
2016-02-20 17:57:19 +01:00
|
|
|
image->setFileName(normalized);
|
2015-11-16 23:26:43 +01:00
|
|
|
if (!checkSupported(image, filename))
|
|
|
|
{
|
2020-11-13 11:39:47 +04:00
|
|
|
static bool uncompress = (getenv("OPENMW_DECOMPRESS_TEXTURES") != nullptr);
|
2017-10-14 18:28:46 +00:00
|
|
|
if (!uncompress)
|
|
|
|
{
|
2018-08-14 19:42:41 +04:00
|
|
|
Log(Debug::Error) << "Error loading " << filename << ": no S3TC texture compression support installed";
|
2017-10-14 18:28:46 +00:00
|
|
|
mCache->addEntryToObjectCache(normalized, mWarningImage);
|
|
|
|
return mWarningImage;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// decompress texture in software if not supported by GPU
|
|
|
|
// requires update to getColor() to be released with OSG 3.6
|
|
|
|
osg::ref_ptr<osg::Image> newImage = new osg::Image;
|
|
|
|
newImage->setFileName(image->getFileName());
|
|
|
|
newImage->allocateImage(image->s(), image->t(), image->r(), image->isImageTranslucent() ? GL_RGBA : GL_RGB, GL_UNSIGNED_BYTE);
|
|
|
|
for (int s=0; s<image->s(); ++s)
|
|
|
|
for (int t=0; t<image->t(); ++t)
|
|
|
|
for (int r=0; r<image->r(); ++r)
|
|
|
|
newImage->setColor(image->getColor(s,t,r), s,t,r);
|
|
|
|
image = newImage;
|
|
|
|
}
|
2015-11-16 23:26:43 +01:00
|
|
|
}
|
2020-09-12 00:20:44 +01:00
|
|
|
else if (killAlpha)
|
|
|
|
{
|
|
|
|
osg::ref_ptr<osg::Image> newImage = new osg::Image;
|
|
|
|
newImage->setFileName(image->getFileName());
|
|
|
|
newImage->allocateImage(image->s(), image->t(), image->r(), GL_RGB, GL_UNSIGNED_BYTE);
|
|
|
|
// OSG just won't write the alpha as there's nowhere to put it.
|
|
|
|
for (int s = 0; s < image->s(); ++s)
|
|
|
|
for (int t = 0; t < image->t(); ++t)
|
|
|
|
for (int r = 0; r < image->r(); ++r)
|
|
|
|
newImage->setColor(image->getColor(s, t, r), s, t, r);
|
|
|
|
image = newImage;
|
|
|
|
}
|
2015-11-16 23:26:43 +01:00
|
|
|
|
2016-02-05 23:31:59 +01:00
|
|
|
mCache->addEntryToObjectCache(normalized, image);
|
2015-11-16 23:26:43 +01:00
|
|
|
return image;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-05 23:05:37 +01:00
|
|
|
osg::Image *ImageManager::getWarningImage()
|
2015-06-19 20:55:04 +02:00
|
|
|
{
|
2016-02-05 23:05:37 +01:00
|
|
|
return mWarningImage;
|
2015-06-19 20:55:04 +02:00
|
|
|
}
|
|
|
|
|
2017-03-07 04:02:06 +01:00
|
|
|
void ImageManager::reportStats(unsigned int frameNumber, osg::Stats *stats) const
|
2017-02-22 02:18:18 +01:00
|
|
|
{
|
|
|
|
stats->setAttribute(frameNumber, "Image", mCache->getCacheSize());
|
|
|
|
}
|
|
|
|
|
2015-03-28 02:20:20 +01:00
|
|
|
}
|