mirror of
https://github.com/minetest/irrlicht.git
synced 2025-01-26 01:30:23 +01:00
Add IImage::checkDataSizeLimit and make IImage getDataSizeFromFormat return size_t
It's to allow image loader to check for sane limits for image sizes.
Idea came from this patch from sfan5 for Minetest: dbd39120e7
Thought solution is a bit different.
Image loader checks not yet added (will come soon).
Also note that limit is to s32. While u32 might work mostly it will run into some troubles with color converter for now (which maybe could be changes). Also 2GB ought to be enough for anybody, right?
git-svn-id: svn://svn.code.sf.net/p/irrlicht/code/trunk@6386 dfc29bdd-3216-0410-991c-e03cc46cb475
This commit is contained in:
parent
76d013d9d6
commit
72b1522083
@ -1,6 +1,7 @@
|
|||||||
--------------------------
|
--------------------------
|
||||||
Changes in 1.9 (not yet released)
|
Changes in 1.9 (not yet released)
|
||||||
|
|
||||||
|
- Add IImage::checkDataSizeLimit and make IImage getDataSizeFromFormat return size_t so image loaders can check if sizes are sane.
|
||||||
- Fix crash with large jpg files. Based somewhat on a patch in Minetest from sfan5 https://github.com/minetest/irrlicht/commit/594de9915346a87f67cd94e28c7933993efb5d3b
|
- Fix crash with large jpg files. Based somewhat on a patch in Minetest from sfan5 https://github.com/minetest/irrlicht/commit/594de9915346a87f67cd94e28c7933993efb5d3b
|
||||||
- COSOperator::getSystemMemory now returns some value on OSX (thought same for total and available).
|
- COSOperator::getSystemMemory now returns some value on OSX (thought same for total and available).
|
||||||
Thanks @sfan5 for patch https://irrlicht.sourceforge.io/forum/viewtopic.php?f=2&t=52819 and https://github.com/minetest/irrlicht/commit/e469c54f76f6d24f92389b4e8a27b9cce7152888
|
Thanks @sfan5 for patch https://irrlicht.sourceforge.io/forum/viewtopic.php?f=2&t=52819 and https://github.com/minetest/irrlicht/commit/e469c54f76f6d24f92389b4e8a27b9cce7152888
|
||||||
|
@ -84,7 +84,7 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
//! Returns image data size in bytes
|
//! Returns image data size in bytes
|
||||||
u32 getImageDataSizeInBytes() const
|
size_t getImageDataSizeInBytes() const
|
||||||
{
|
{
|
||||||
return getDataSizeFromFormat(Format, Size.Width, Size.Height);
|
return getDataSizeFromFormat(Format, Size.Width, Size.Height);
|
||||||
}
|
}
|
||||||
@ -295,7 +295,7 @@ public:
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
u32 dataSize = 0;
|
size_t dataSize = 0;
|
||||||
u32 width = Size.Width;
|
u32 width = Size.Width;
|
||||||
u32 height = Size.Height;
|
u32 height = Size.Height;
|
||||||
|
|
||||||
@ -445,43 +445,57 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//! calculate image data size in bytes for selected format, width and height.
|
//! You should not create images where the result of getDataSizeFromFormat doesn't pass this function
|
||||||
static u32 getDataSizeFromFormat(ECOLOR_FORMAT format, u32 width, u32 height)
|
/** Note that CImage does not yet check for this, but going beyond this limit is not supported well.
|
||||||
|
Image loaders should check for this.
|
||||||
|
If you don't have the format yet then checking width*height*bytes_per_pixel is mostly fine, but make
|
||||||
|
sure to work with size_t so it doesn't clip the result to u32 too early.
|
||||||
|
\return true when dataSize is small enough that it should be fine. */
|
||||||
|
static bool checkDataSizeLimit(size_t dataSize)
|
||||||
{
|
{
|
||||||
u32 imageSize = 0;
|
// 2gb for now. Could be we could do more on some platforms, but we still will run into
|
||||||
|
// problems right now then for example in then color converter (which currently still uses
|
||||||
|
// s32 for sizes).
|
||||||
|
return (size_t)(s32)(dataSize) == dataSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
//! calculate image data size in bytes for selected format, width and height.
|
||||||
|
static size_t getDataSizeFromFormat(ECOLOR_FORMAT format, u32 width, u32 height)
|
||||||
|
{
|
||||||
|
size_t imageSize = 0;
|
||||||
|
|
||||||
switch (format)
|
switch (format)
|
||||||
{
|
{
|
||||||
case ECF_DXT1:
|
case ECF_DXT1:
|
||||||
imageSize = ((width + 3) / 4) * ((height + 3) / 4) * 8;
|
imageSize = (size_t)((width + 3) / 4) * ((height + 3) / 4) * 8;
|
||||||
break;
|
break;
|
||||||
case ECF_DXT2:
|
case ECF_DXT2:
|
||||||
case ECF_DXT3:
|
case ECF_DXT3:
|
||||||
case ECF_DXT4:
|
case ECF_DXT4:
|
||||||
case ECF_DXT5:
|
case ECF_DXT5:
|
||||||
imageSize = ((width + 3) / 4) * ((height + 3) / 4) * 16;
|
imageSize = (size_t)((width + 3) / 4) * ((height + 3) / 4) * 16;
|
||||||
break;
|
break;
|
||||||
case ECF_PVRTC_RGB2:
|
case ECF_PVRTC_RGB2:
|
||||||
case ECF_PVRTC_ARGB2:
|
case ECF_PVRTC_ARGB2:
|
||||||
imageSize = (core::max_<u32>(width, 16) * core::max_<u32>(height, 8) * 2 + 7) / 8;
|
imageSize = ((size_t)core::max_<u32>(width, 16) * core::max_<u32>(height, 8) * 2 + 7) / 8;
|
||||||
break;
|
break;
|
||||||
case ECF_PVRTC_RGB4:
|
case ECF_PVRTC_RGB4:
|
||||||
case ECF_PVRTC_ARGB4:
|
case ECF_PVRTC_ARGB4:
|
||||||
imageSize = (core::max_<u32>(width, 8) * core::max_<u32>(height, 8) * 4 + 7) / 8;
|
imageSize = ((size_t)core::max_<u32>(width, 8) * core::max_<u32>(height, 8) * 4 + 7) / 8;
|
||||||
break;
|
break;
|
||||||
case ECF_PVRTC2_ARGB2:
|
case ECF_PVRTC2_ARGB2:
|
||||||
imageSize = core::ceil32(width / 8.0f) * core::ceil32(height / 4.0f) * 8;
|
imageSize = (size_t)core::ceil32(width / 8.0f) * core::ceil32(height / 4.0f) * 8;
|
||||||
break;
|
break;
|
||||||
case ECF_PVRTC2_ARGB4:
|
case ECF_PVRTC2_ARGB4:
|
||||||
case ECF_ETC1:
|
case ECF_ETC1:
|
||||||
case ECF_ETC2_RGB:
|
case ECF_ETC2_RGB:
|
||||||
imageSize = core::ceil32(width / 4.0f) * core::ceil32(height / 4.0f) * 8;
|
imageSize = (size_t)core::ceil32(width / 4.0f) * core::ceil32(height / 4.0f) * 8;
|
||||||
break;
|
break;
|
||||||
case ECF_ETC2_ARGB:
|
case ECF_ETC2_ARGB:
|
||||||
imageSize = core::ceil32(width / 4.0f) * core::ceil32(height / 4.0f) * 16;
|
imageSize = (size_t)core::ceil32(width / 4.0f) * core::ceil32(height / 4.0f) * 16;
|
||||||
break;
|
break;
|
||||||
default: // uncompressed formats
|
default: // uncompressed formats
|
||||||
imageSize = getBitsPerPixelFromFormat(format) / 8 * width;
|
imageSize = (size_t)getBitsPerPixelFromFormat(format) / 8 * width;
|
||||||
imageSize *= height;
|
imageSize *= height;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -335,7 +335,7 @@ void CD3D9Texture::regenerateMipMapLevels(void* data, u32 layer)
|
|||||||
u32 width = Size.Width;
|
u32 width = Size.Width;
|
||||||
u32 height = Size.Height;
|
u32 height = Size.Height;
|
||||||
u8* tmpData = static_cast<u8*>(data);
|
u8* tmpData = static_cast<u8*>(data);
|
||||||
u32 dataSize = 0;
|
size_t dataSize = 0;
|
||||||
u32 level = 0;
|
u32 level = 0;
|
||||||
|
|
||||||
do
|
do
|
||||||
@ -714,7 +714,7 @@ void CD3D9Texture::uploadTexture(void* data, u32 mipmapLevel, u32 layer)
|
|||||||
u32 width = Size.Width >> mipmapLevel;
|
u32 width = Size.Width >> mipmapLevel;
|
||||||
u32 height = Size.Height >> mipmapLevel;
|
u32 height = Size.Height >> mipmapLevel;
|
||||||
|
|
||||||
u32 dataSize = IImage::getDataSizeFromFormat(ColorFormat, width, height);
|
size_t dataSize = IImage::getDataSizeFromFormat(ColorFormat, width, height);
|
||||||
|
|
||||||
HRESULT hr = 0;
|
HRESULT hr = 0;
|
||||||
|
|
||||||
|
@ -24,8 +24,7 @@ CImage::CImage(ECOLOR_FORMAT format, const core::dimension2d<u32>& size, void* d
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
const u32 dataSize = getDataSizeFromFormat(Format, Size.Width, Size.Height);
|
const size_t dataSize = getDataSizeFromFormat(Format, Size.Width, Size.Height);
|
||||||
|
|
||||||
Data = new u8[align_next(dataSize,16)];
|
Data = new u8[align_next(dataSize,16)];
|
||||||
memcpy(Data, data, dataSize);
|
memcpy(Data, data, dataSize);
|
||||||
DeleteMemory = true;
|
DeleteMemory = true;
|
||||||
@ -36,7 +35,8 @@ CImage::CImage(ECOLOR_FORMAT format, const core::dimension2d<u32>& size, void* d
|
|||||||
//! Constructor of empty image
|
//! Constructor of empty image
|
||||||
CImage::CImage(ECOLOR_FORMAT format, const core::dimension2d<u32>& size) : IImage(format, size, true)
|
CImage::CImage(ECOLOR_FORMAT format, const core::dimension2d<u32>& size) : IImage(format, size, true)
|
||||||
{
|
{
|
||||||
Data = new u8[align_next(getDataSizeFromFormat(Format, Size.Width, Size.Height),16)];
|
const size_t dataSize = getDataSizeFromFormat(Format, Size.Width, Size.Height);
|
||||||
|
Data = new u8[align_next(dataSize,16)];
|
||||||
DeleteMemory = true;
|
DeleteMemory = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -332,8 +332,8 @@ void CImage::fill(const SColor &color)
|
|||||||
{
|
{
|
||||||
u8 rgb[3];
|
u8 rgb[3];
|
||||||
CColorConverter::convert_A8R8G8B8toR8G8B8(&color, 1, rgb);
|
CColorConverter::convert_A8R8G8B8toR8G8B8(&color, 1, rgb);
|
||||||
const u32 size = getImageDataSizeInBytes();
|
const size_t size = getImageDataSizeInBytes();
|
||||||
for (u32 i=0; i<size; i+=3)
|
for (size_t i=0; i<size; i+=3)
|
||||||
{
|
{
|
||||||
memcpy(Data+i, rgb, 3);
|
memcpy(Data+i, rgb, 3);
|
||||||
}
|
}
|
||||||
|
@ -693,8 +693,8 @@ IImage* CImageLoaderDDS::loadImage(io::IReadFile* file) const
|
|||||||
s32 width, height;
|
s32 width, height;
|
||||||
eDDSPixelFormat pixelFormat;
|
eDDSPixelFormat pixelFormat;
|
||||||
ECOLOR_FORMAT format = ECF_UNKNOWN;
|
ECOLOR_FORMAT format = ECF_UNKNOWN;
|
||||||
u32 dataSize = 0;
|
size_t dataSize = 0;
|
||||||
u32 mipMapsDataSize = 0;
|
size_t mipMapsDataSize = 0;
|
||||||
bool is3D = false;
|
bool is3D = false;
|
||||||
bool useAlpha = false;
|
bool useAlpha = false;
|
||||||
u32 mipMapCount = 0;
|
u32 mipMapCount = 0;
|
||||||
|
@ -69,7 +69,7 @@ core::array<IImage*> CImageLoaderPVR::loadImages(io::IReadFile* file, E_TEXTURE_
|
|||||||
core::array<u8*> mipMapsDataArray;
|
core::array<u8*> mipMapsDataArray;
|
||||||
|
|
||||||
ECOLOR_FORMAT format = ECF_UNKNOWN;
|
ECOLOR_FORMAT format = ECF_UNKNOWN;
|
||||||
u32 dataSize = 0;
|
size_t dataSize = 0;
|
||||||
|
|
||||||
file->seek(0);
|
file->seek(0);
|
||||||
file->read(&header, sizeof(SPVRHeader));
|
file->read(&header, sizeof(SPVRHeader));
|
||||||
@ -216,7 +216,7 @@ core::array<IImage*> CImageLoaderPVR::loadImages(io::IReadFile* file, E_TEXTURE_
|
|||||||
// read texture
|
// read texture
|
||||||
|
|
||||||
dataSize = 0;
|
dataSize = 0;
|
||||||
long offset = 0;
|
size_t offset = 0;
|
||||||
|
|
||||||
for (u32 i = 0; i < header.MipMapCount; ++i)
|
for (u32 i = 0; i < header.MipMapCount; ++i)
|
||||||
{
|
{
|
||||||
|
@ -401,7 +401,7 @@ public:
|
|||||||
u32 width = Size.Width;
|
u32 width = Size.Width;
|
||||||
u32 height = Size.Height;
|
u32 height = Size.Height;
|
||||||
u8* tmpData = static_cast<u8*>(data);
|
u8* tmpData = static_cast<u8*>(data);
|
||||||
u32 dataSize = 0;
|
size_t dataSize = 0;
|
||||||
u32 level = 0;
|
u32 level = 0;
|
||||||
|
|
||||||
do
|
do
|
||||||
@ -600,7 +600,7 @@ protected:
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
u32 dataSize = IImage::getDataSizeFromFormat(ColorFormat, width, height);
|
GLsizei dataSize = (GLsizei)IImage::getDataSizeFromFormat(ColorFormat, width, height);
|
||||||
|
|
||||||
switch (TextureType)
|
switch (TextureType)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user