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>
|
2023-05-31 23:11:03 +02:00
|
|
|
#include <components/vfs/pathutil.hpp>
|
2015-03-28 02:20:20 +01:00
|
|
|
|
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
|
|
|
|
{
|
|
|
|
|
2023-09-09 19:29:26 +02:00
|
|
|
ImageManager::ImageManager(const VFS::Manager* vfs, double expiryDelay)
|
|
|
|
: ResourceManager(vfs, expiryDelay)
|
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"))
|
2022-05-14 23:06:32 -07:00
|
|
|
, mOptionsNoFlip(new osgDB::Options("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
|
|
|
|
2023-05-21 16:39:32 +02:00
|
|
|
bool checkSupported(osg::Image* image)
|
2015-07-28 23:35:10 +02:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2023-05-21 16:39:32 +02:00
|
|
|
osg::ref_ptr<osg::Image> ImageManager::getImage(std::string_view filename, bool disableFlip)
|
2015-11-16 23:26:43 +01:00
|
|
|
{
|
2023-05-31 23:11:03 +02:00
|
|
|
const std::string normalized = VFS::Path::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);
|
|
|
|
}
|
|
|
|
|
2022-05-14 23:06:32 -07:00
|
|
|
osgDB::ReaderWriter::ReadResult result
|
|
|
|
= reader->readImage(*stream, disableFlip ? mOptionsNoFlip : 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);
|
2023-05-21 16:39:32 +02:00
|
|
|
if (!checkSupported(image))
|
2015-11-16 23:26:43 +01:00
|
|
|
{
|
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
|
|
|
}
|