2020-01-03 20:05:16 +01:00
|
|
|
// Copyright (C) 2002-2012 Nikolaus Gebhardt
|
|
|
|
// This file is part of the "Irrlicht Engine".
|
|
|
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
|
|
|
|
|
|
|
#include "CImageLoaderPNG.h"
|
|
|
|
|
|
|
|
#ifdef _IRR_COMPILE_WITH_PNG_LOADER_
|
|
|
|
|
|
|
|
#ifdef _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
#ifndef _IRR_USE_NON_SYSTEM_LIB_PNG_
|
|
|
|
#include <png.h> // use system lib png
|
|
|
|
#else // _IRR_USE_NON_SYSTEM_LIB_PNG_
|
|
|
|
#include "libpng/png.h" // use irrlicht included lib png
|
|
|
|
#endif // _IRR_USE_NON_SYSTEM_LIB_PNG_
|
|
|
|
#endif // _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
|
|
|
|
#include "CImage.h"
|
2023-10-16 23:17:56 +02:00
|
|
|
#include "IReadFile.h"
|
2020-01-03 20:05:16 +01:00
|
|
|
#include "os.h"
|
|
|
|
|
|
|
|
namespace irr
|
|
|
|
{
|
|
|
|
namespace video
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
// PNG function for error handling
|
|
|
|
static void png_cpexcept_error(png_structp png_ptr, png_const_charp msg)
|
|
|
|
{
|
|
|
|
os::Printer::log("PNG fatal error", msg, ELL_ERROR);
|
|
|
|
longjmp(png_jmpbuf(png_ptr), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// PNG function for warning handling
|
|
|
|
static void png_cpexcept_warn(png_structp png_ptr, png_const_charp msg)
|
|
|
|
{
|
|
|
|
os::Printer::log("PNG warning", msg, ELL_WARNING);
|
|
|
|
}
|
|
|
|
|
|
|
|
// PNG function for file reading
|
|
|
|
void PNGAPI user_read_data_fcn(png_structp png_ptr, png_bytep data, png_size_t length)
|
|
|
|
{
|
|
|
|
png_size_t check;
|
|
|
|
|
|
|
|
// changed by zola {
|
|
|
|
io::IReadFile* file=(io::IReadFile*)png_get_io_ptr(png_ptr);
|
|
|
|
check=(png_size_t) file->read((void*)data,(u32)length);
|
|
|
|
// }
|
|
|
|
|
|
|
|
if (check != length)
|
|
|
|
png_error(png_ptr, "Read Error");
|
|
|
|
}
|
|
|
|
#endif // _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
|
|
|
|
|
|
|
|
//! returns true if the file maybe is able to be loaded by this class
|
|
|
|
//! based on the file extension (e.g. ".tga")
|
|
|
|
bool CImageLoaderPng::isALoadableFileExtension(const io::path& filename) const
|
|
|
|
{
|
|
|
|
#ifdef _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
return core::hasFileExtension ( filename, "png" );
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif // _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//! returns true if the file maybe is able to be loaded by this class
|
|
|
|
bool CImageLoaderPng::isALoadableFileFormat(io::IReadFile* file) const
|
|
|
|
{
|
|
|
|
#ifdef _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
if (!file)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
png_byte buffer[8];
|
|
|
|
// Read the first few bytes of the PNG file
|
|
|
|
if (file->read(buffer, 8) != 8)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check if it really is a PNG file
|
|
|
|
return !png_sig_cmp(buffer, 0, 8);
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif // _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// load in the image data
|
|
|
|
IImage* CImageLoaderPng::loadImage(io::IReadFile* file) const
|
|
|
|
{
|
|
|
|
#ifdef _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
if (!file)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
//Used to point to image rows
|
|
|
|
u8** RowPointers = 0;
|
|
|
|
|
|
|
|
png_byte buffer[8];
|
|
|
|
// Read the first few bytes of the PNG file
|
|
|
|
if( file->read(buffer, 8) != 8 )
|
|
|
|
{
|
2022-09-22 23:55:03 +02:00
|
|
|
os::Printer::log("LOAD PNG: can't read file (filesize < 8)", file->getFileName(), ELL_ERROR);
|
2020-01-03 20:05:16 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if it really is a PNG file
|
|
|
|
if( png_sig_cmp(buffer, 0, 8) )
|
|
|
|
{
|
2022-09-22 23:55:03 +02:00
|
|
|
os::Printer::log("LOAD PNG: not really a png (wrong signature)", file->getFileName(), ELL_ERROR);
|
2020-01-03 20:05:16 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate the png read struct
|
|
|
|
png_structp png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING,
|
|
|
|
NULL, (png_error_ptr)png_cpexcept_error, (png_error_ptr)png_cpexcept_warn);
|
|
|
|
if (!png_ptr)
|
|
|
|
{
|
2022-09-22 23:55:03 +02:00
|
|
|
os::Printer::log("LOAD PNG: Internal PNG create read struct failure", file->getFileName(), ELL_ERROR);
|
2020-01-03 20:05:16 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate the png info struct
|
|
|
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
|
|
|
if (!info_ptr)
|
|
|
|
{
|
2022-09-22 23:55:03 +02:00
|
|
|
os::Printer::log("LOAD PNG: Internal PNG create info struct failure", file->getFileName(), ELL_ERROR);
|
2020-01-03 20:05:16 +01:00
|
|
|
png_destroy_read_struct(&png_ptr, NULL, NULL);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// for proper error handling
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr)))
|
|
|
|
{
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
delete [] RowPointers;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// changed by zola so we don't need to have public FILE pointers
|
|
|
|
png_set_read_fn(png_ptr, file, user_read_data_fcn);
|
|
|
|
|
|
|
|
png_set_sig_bytes(png_ptr, 8); // Tell png that we read the signature
|
|
|
|
|
|
|
|
png_read_info(png_ptr, info_ptr); // Read the info section of the png file
|
|
|
|
|
2022-09-22 23:55:03 +02:00
|
|
|
u32 Width=0;
|
|
|
|
u32 Height=0;
|
|
|
|
s32 BitDepth=0;
|
|
|
|
s32 ColorType=0;
|
2020-01-03 20:05:16 +01:00
|
|
|
{
|
|
|
|
// Use temporary variables to avoid passing cast pointers
|
|
|
|
png_uint_32 w,h;
|
|
|
|
// Extract info
|
|
|
|
png_get_IHDR(png_ptr, info_ptr,
|
|
|
|
&w, &h,
|
|
|
|
&BitDepth, &ColorType, NULL, NULL, NULL);
|
|
|
|
Width=w;
|
|
|
|
Height=h;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert palette color to true color
|
|
|
|
if (ColorType==PNG_COLOR_TYPE_PALETTE)
|
|
|
|
png_set_palette_to_rgb(png_ptr);
|
|
|
|
|
|
|
|
// Convert low bit colors to 8 bit colors
|
|
|
|
if (BitDepth < 8)
|
|
|
|
{
|
|
|
|
if (ColorType==PNG_COLOR_TYPE_GRAY || ColorType==PNG_COLOR_TYPE_GRAY_ALPHA)
|
|
|
|
png_set_expand_gray_1_2_4_to_8(png_ptr);
|
|
|
|
else
|
|
|
|
png_set_packing(png_ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS))
|
|
|
|
png_set_tRNS_to_alpha(png_ptr);
|
|
|
|
|
|
|
|
// Convert high bit colors to 8 bit colors
|
|
|
|
if (BitDepth == 16)
|
|
|
|
png_set_strip_16(png_ptr);
|
|
|
|
|
|
|
|
// Convert gray color to true color
|
|
|
|
if (ColorType==PNG_COLOR_TYPE_GRAY || ColorType==PNG_COLOR_TYPE_GRAY_ALPHA)
|
|
|
|
png_set_gray_to_rgb(png_ptr);
|
|
|
|
|
|
|
|
int intent;
|
|
|
|
const double screen_gamma = 2.2;
|
|
|
|
|
|
|
|
if (png_get_sRGB(png_ptr, info_ptr, &intent))
|
|
|
|
png_set_gamma(png_ptr, screen_gamma, 0.45455);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
double image_gamma;
|
|
|
|
if (png_get_gAMA(png_ptr, info_ptr, &image_gamma))
|
|
|
|
png_set_gamma(png_ptr, screen_gamma, image_gamma);
|
|
|
|
else
|
|
|
|
png_set_gamma(png_ptr, screen_gamma, 0.45455);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the changes in between, as we need to get the new color type
|
|
|
|
// for proper processing of the RGBA type
|
|
|
|
png_read_update_info(png_ptr, info_ptr);
|
|
|
|
{
|
|
|
|
// Use temporary variables to avoid passing cast pointers
|
|
|
|
png_uint_32 w,h;
|
|
|
|
// Extract info
|
|
|
|
png_get_IHDR(png_ptr, info_ptr,
|
|
|
|
&w, &h,
|
|
|
|
&BitDepth, &ColorType, NULL, NULL, NULL);
|
|
|
|
Width=w;
|
|
|
|
Height=h;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert RGBA to BGRA
|
|
|
|
if (ColorType==PNG_COLOR_TYPE_RGB_ALPHA)
|
|
|
|
{
|
|
|
|
#ifdef __BIG_ENDIAN__
|
|
|
|
png_set_swap_alpha(png_ptr);
|
|
|
|
#else
|
|
|
|
png_set_bgr(png_ptr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2022-09-22 23:55:03 +02:00
|
|
|
ECOLOR_FORMAT colorFormat = ColorType==PNG_COLOR_TYPE_RGB_ALPHA ? ECF_A8R8G8B8 : ECF_R8G8B8;
|
|
|
|
|
|
|
|
if (!IImage::checkDataSizeLimit(IImage::getDataSizeFromFormat(colorFormat, Width, Height)))
|
|
|
|
png_cpexcept_error(png_ptr, "Image dimensions too large");
|
|
|
|
|
2020-01-03 20:05:16 +01:00
|
|
|
// Create the image structure to be filled by png data
|
2022-09-22 23:55:03 +02:00
|
|
|
video::IImage* image = new CImage(colorFormat, core::dimension2du(Width, Height));
|
2020-01-03 20:05:16 +01:00
|
|
|
if (!image)
|
|
|
|
{
|
2022-09-22 23:55:03 +02:00
|
|
|
os::Printer::log("LOAD PNG: Internal PNG create image struct failure", file->getFileName(), ELL_ERROR);
|
2020-01-03 20:05:16 +01:00
|
|
|
png_destroy_read_struct(&png_ptr, NULL, NULL);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create array of pointers to rows in image data
|
|
|
|
RowPointers = new png_bytep[Height];
|
|
|
|
if (!RowPointers)
|
|
|
|
{
|
2022-09-22 23:55:03 +02:00
|
|
|
os::Printer::log("LOAD PNG: Internal PNG create row pointers failure", file->getFileName(), ELL_ERROR);
|
2020-01-03 20:05:16 +01:00
|
|
|
png_destroy_read_struct(&png_ptr, NULL, NULL);
|
|
|
|
delete image;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fill array of pointers to rows in image data
|
|
|
|
unsigned char* data = (unsigned char*)image->getData();
|
|
|
|
for (u32 i=0; i<Height; ++i)
|
|
|
|
{
|
|
|
|
RowPointers[i]=data;
|
|
|
|
data += image->getPitch();
|
|
|
|
}
|
|
|
|
|
|
|
|
// for proper error handling
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr)))
|
|
|
|
{
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
delete [] RowPointers;
|
|
|
|
delete image;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read data using the library function that handles all transformations including interlacing
|
|
|
|
png_read_image(png_ptr, RowPointers);
|
|
|
|
|
|
|
|
png_read_end(png_ptr, NULL);
|
|
|
|
delete [] RowPointers;
|
|
|
|
png_destroy_read_struct(&png_ptr,&info_ptr, 0); // Clean up memory
|
|
|
|
|
|
|
|
return image;
|
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif // _IRR_COMPILE_WITH_LIBPNG_
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IImageLoader* createImageLoaderPNG()
|
|
|
|
{
|
|
|
|
return new CImageLoaderPng();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}// end namespace irr
|
|
|
|
}//end namespace video
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|