2023-10-03 20:37:00 +02: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 "CImageLoaderTGA.h"
# include "IReadFile.h"
# include "os.h"
# include "CColorConverter.h"
# include "CImage.h"
# include "irrString.h"
2024-02-11 18:28:32 +01:00
# define MAX(x, y) (((x) > (y)) ? (x) : (y))
2023-10-03 20:37:00 +02:00
namespace irr
{
namespace video
{
//! returns true if the file maybe is able to be loaded by this class
//! based on the file extension (e.g. ".tga")
2024-03-20 19:35:52 +01:00
bool CImageLoaderTGA : : isALoadableFileExtension ( const io : : path & filename ) const
2023-10-03 20:37:00 +02:00
{
2024-03-20 19:35:52 +01:00
return core : : hasFileExtension ( filename , " tga " ) ;
2023-10-03 20:37:00 +02:00
}
//! loads a compressed tga.
2024-03-20 19:35:52 +01:00
u8 * CImageLoaderTGA : : loadCompressedImage ( io : : IReadFile * file , const STGAHeader & header ) const
2023-10-03 20:37:00 +02:00
{
// This was written and sent in by Jon Pry, thank you very much!
// I only changed the formatting a little bit.
2024-03-20 19:35:52 +01:00
const u32 bytesPerPixel = header . PixelDepth / 8 ;
const u32 imageSize = header . ImageHeight * header . ImageWidth * bytesPerPixel ;
u8 * data = new u8 [ imageSize ] ;
2023-10-16 22:59:41 +02:00
u32 currentByte = 0 ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
while ( currentByte < imageSize ) {
2023-10-03 20:37:00 +02:00
u8 chunkheader = 0 ;
file - > read ( & chunkheader , sizeof ( u8 ) ) ; // Read The Chunk's Header
2024-03-20 19:35:52 +01:00
if ( chunkheader < 128 ) // If The Chunk Is A 'RAW' Chunk
2023-10-03 20:37:00 +02:00
{
chunkheader + + ; // Add 1 To The Value To Get Total Number Of Raw Pixels
2023-10-16 22:59:41 +02:00
const u32 bytesToRead = bytesPerPixel * chunkheader ;
2024-03-20 19:35:52 +01:00
if ( currentByte + bytesToRead < = imageSize ) {
2023-10-16 22:59:41 +02:00
file - > read ( & data [ currentByte ] , bytesToRead ) ;
currentByte + = bytesToRead ;
2024-03-20 19:35:52 +01:00
} else {
2023-10-16 22:59:41 +02:00
os : : Printer : : log ( " Compressed TGA file RAW chunk tries writing beyond buffer " , file - > getFileName ( ) , ELL_WARNING ) ;
break ;
}
2024-03-20 19:35:52 +01:00
} else {
2023-10-03 20:37:00 +02:00
// thnx to neojzs for some fixes with this code
// If It's An RLE Header
chunkheader - = 127 ; // Subtract 127 To Get Rid Of The ID Bit
2023-10-16 22:59:41 +02:00
u32 dataOffset = currentByte ;
2024-03-20 19:35:52 +01:00
if ( dataOffset + bytesPerPixel < imageSize ) {
2023-10-16 22:59:41 +02:00
file - > read ( & data [ dataOffset ] , bytesPerPixel ) ;
currentByte + = bytesPerPixel ;
2024-03-20 19:35:52 +01:00
} else {
2023-10-16 22:59:41 +02:00
os : : Printer : : log ( " Compressed TGA file RLE headertries writing beyond buffer " , file - > getFileName ( ) , ELL_WARNING ) ;
break ;
}
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
for ( u32 counter = 1 ; counter < chunkheader ; counter + + ) {
if ( currentByte + bytesPerPixel < = imageSize ) {
for ( u32 elementCounter = 0 ; elementCounter < bytesPerPixel ; elementCounter + + ) {
2023-09-23 20:33:46 +02:00
data [ currentByte + elementCounter ] = data [ dataOffset + elementCounter ] ;
}
}
2023-10-03 20:37:00 +02:00
currentByte + = bytesPerPixel ;
}
}
}
return data ;
}
//! returns true if the file maybe is able to be loaded by this class
2024-03-20 19:35:52 +01:00
bool CImageLoaderTGA : : isALoadableFileFormat ( io : : IReadFile * file ) const
2023-10-03 20:37:00 +02:00
{
if ( ! file )
return false ;
STGAFooter footer ;
memset ( & footer , 0 , sizeof ( STGAFooter ) ) ;
2024-03-20 19:35:52 +01:00
file - > seek ( file - > getSize ( ) - sizeof ( STGAFooter ) ) ;
2023-10-03 20:37:00 +02:00
file - > read ( & footer , sizeof ( STGAFooter ) ) ;
2024-03-20 19:35:52 +01:00
return ( ! strcmp ( footer . Signature , " TRUEVISION-XFILE. " ) ) ; // very old tgas are refused.
2023-10-03 20:37:00 +02:00
}
//! creates a surface from the file
2024-03-20 19:35:52 +01:00
IImage * CImageLoaderTGA : : loadImage ( io : : IReadFile * file ) const
2023-10-03 20:37:00 +02:00
{
STGAHeader header ;
u32 * palette = 0 ;
file - > read ( & header , sizeof ( STGAHeader ) ) ;
# ifdef __BIG_ENDIAN__
header . ColorMapLength = os : : Byteswap : : byteswap ( header . ColorMapLength ) ;
header . ImageWidth = os : : Byteswap : : byteswap ( header . ImageWidth ) ;
header . ImageHeight = os : : Byteswap : : byteswap ( header . ImageHeight ) ;
# endif
2024-03-20 19:35:52 +01:00
if ( ! checkImageDimensions ( header . ImageWidth , header . ImageHeight ) ) {
2023-10-03 20:37:00 +02:00
os : : Printer : : log ( " Image dimensions too large in file " , file - > getFileName ( ) , ELL_ERROR ) ;
return 0 ;
}
// skip image identification field
if ( header . IdLength )
file - > seek ( header . IdLength , true ) ;
2024-03-20 19:35:52 +01:00
if ( header . ColorMapType ) {
2023-09-23 18:34:42 +02:00
// Create 32 bit palette
2024-02-11 18:28:32 +01:00
// `core::max_()` is not used here because it takes its inputs as references. Since `header` is packed, use the macro `MAX()` instead:
const irr : : u16 paletteSize = MAX ( ( u16 ) 256u , header . ColorMapLength ) ; // ColorMapLength can lie, but so far we only use palette for 8-bit, so ensure it has 256 entries
2023-09-23 18:34:42 +02:00
palette = new u32 [ paletteSize ] ;
2024-03-20 19:35:52 +01:00
if ( paletteSize > header . ColorMapLength ) {
2023-09-23 18:34:42 +02:00
// To catch images using palette colors with invalid indices
2024-03-20 19:35:52 +01:00
const irr : : u32 errorCol = irr : : video : : SColor ( 255 , 255 , 0 , 205 ) . color ; // bright magenta
for ( irr : : u16 i = header . ColorMapLength ; i < paletteSize ; + + i )
2023-09-23 18:34:42 +02:00
palette [ i ] = errorCol ;
}
2023-10-03 20:37:00 +02:00
// read color map
2024-03-20 19:35:52 +01:00
u8 * colorMap = new u8 [ header . ColorMapEntrySize / 8 * header . ColorMapLength ] ;
file - > read ( colorMap , header . ColorMapEntrySize / 8 * header . ColorMapLength ) ;
2023-10-03 20:37:00 +02:00
// convert to 32-bit palette
2024-03-20 19:35:52 +01:00
switch ( header . ColorMapEntrySize ) {
case 16 :
CColorConverter : : convert_A1R5G5B5toA8R8G8B8 ( colorMap , header . ColorMapLength , palette ) ;
break ;
case 24 :
CColorConverter : : convert_B8G8R8toA8R8G8B8 ( colorMap , header . ColorMapLength , palette ) ;
break ;
case 32 :
CColorConverter : : convert_B8G8R8A8toA8R8G8B8 ( colorMap , header . ColorMapLength , palette ) ;
break ;
2023-10-03 20:37:00 +02:00
}
2024-03-20 19:35:52 +01:00
delete [ ] colorMap ;
2023-10-03 20:37:00 +02:00
}
// read image
2024-03-20 19:35:52 +01:00
u8 * data = 0 ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
if ( header . ImageType = = 1 | | // Uncompressed, color-mapped images.
2023-10-03 20:37:00 +02:00
header . ImageType = = 2 | | // Uncompressed, RGB images
2024-03-20 19:35:52 +01:00
header . ImageType = = 3 // Uncompressed, black and white images
) {
const s32 imageSize = header . ImageHeight * header . ImageWidth * ( header . PixelDepth / 8 ) ;
2023-10-03 20:37:00 +02:00
data = new u8 [ imageSize ] ;
2024-03-20 19:35:52 +01:00
file - > read ( data , imageSize ) ;
} else if ( header . ImageType = = 10 ) {
2023-10-03 20:37:00 +02:00
// Runlength encoded RGB images
data = loadCompressedImage ( file , header ) ;
2024-03-20 19:35:52 +01:00
} else {
2023-10-03 20:37:00 +02:00
os : : Printer : : log ( " Unsupported TGA file type " , file - > getFileName ( ) , ELL_ERROR ) ;
2024-03-20 19:35:52 +01:00
delete [ ] palette ;
2023-10-03 20:37:00 +02:00
return 0 ;
}
2024-03-20 19:35:52 +01:00
IImage * image = 0 ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
switch ( header . PixelDepth ) {
case 8 : {
if ( header . ImageType = = 3 ) // grey image
2023-10-03 20:37:00 +02:00
{
2024-03-20 19:35:52 +01:00
image = new CImage ( ECF_R8G8B8 ,
2023-10-03 20:37:00 +02:00
core : : dimension2d < u32 > ( header . ImageWidth , header . ImageHeight ) ) ;
2024-03-20 19:35:52 +01:00
if ( image )
CColorConverter : : convert8BitTo24Bit ( ( u8 * ) data ,
( u8 * ) image - > getData ( ) ,
header . ImageWidth , header . ImageHeight ,
0 , 0 , ( header . ImageDescriptor & 0x20 ) = = 0 ) ;
} else {
switch ( header . ColorMapEntrySize ) {
case 16 :
image = new CImage ( ECF_A1R5G5B5 , core : : dimension2d < u32 > ( header . ImageWidth , header . ImageHeight ) ) ;
2023-10-03 20:37:00 +02:00
if ( image )
2024-03-20 19:35:52 +01:00
CColorConverter : : convert8BitTo16Bit ( ( u8 * ) data ,
( s16 * ) image - > getData ( ) ,
header . ImageWidth , header . ImageHeight ,
( s32 * ) palette , 0 ,
( header . ImageDescriptor & 0x20 ) = = 0 ) ;
break ;
// Note: 24 bit with palette would need a 24 bit palette, too lazy doing that now (textures will prefer 32-bit later anyway)
default :
image = new CImage ( ECF_A8R8G8B8 , core : : dimension2d < u32 > ( header . ImageWidth , header . ImageHeight ) ) ;
if ( image )
CColorConverter : : convert8BitTo32Bit ( ( u8 * ) data ,
( u8 * ) image - > getData ( ) ,
header . ImageWidth , header . ImageHeight ,
( u8 * ) palette , 0 ,
( header . ImageDescriptor & 0x20 ) = = 0 ) ;
break ;
2023-10-03 20:37:00 +02:00
}
}
2024-03-20 19:35:52 +01:00
} break ;
2023-10-03 20:37:00 +02:00
case 16 :
image = new CImage ( ECF_A1R5G5B5 ,
2024-03-20 19:35:52 +01:00
core : : dimension2d < u32 > ( header . ImageWidth , header . ImageHeight ) ) ;
2023-10-03 20:37:00 +02:00
if ( image )
2024-03-20 19:35:52 +01:00
CColorConverter : : convert16BitTo16Bit ( ( s16 * ) data ,
( s16 * ) image - > getData ( ) , header . ImageWidth , header . ImageHeight , 0 , ( header . ImageDescriptor & 0x20 ) = = 0 ) ;
2023-10-03 20:37:00 +02:00
break ;
case 24 :
2024-03-20 19:35:52 +01:00
image = new CImage ( ECF_R8G8B8 ,
2023-10-03 20:37:00 +02:00
core : : dimension2d < u32 > ( header . ImageWidth , header . ImageHeight ) ) ;
2024-03-20 19:35:52 +01:00
if ( image )
CColorConverter : : convert24BitTo24Bit (
( u8 * ) data , ( u8 * ) image - > getData ( ) , header . ImageWidth , header . ImageHeight , 0 , ( header . ImageDescriptor & 0x20 ) = = 0 , true ) ;
2023-10-03 20:37:00 +02:00
break ;
case 32 :
2024-03-20 19:35:52 +01:00
image = new CImage ( ECF_A8R8G8B8 ,
2023-10-03 20:37:00 +02:00
core : : dimension2d < u32 > ( header . ImageWidth , header . ImageHeight ) ) ;
2024-03-20 19:35:52 +01:00
if ( image )
CColorConverter : : convert32BitTo32Bit ( ( s32 * ) data ,
( s32 * ) image - > getData ( ) , header . ImageWidth , header . ImageHeight , 0 , ( header . ImageDescriptor & 0x20 ) = = 0 ) ;
2023-10-03 20:37:00 +02:00
break ;
default :
os : : Printer : : log ( " Unsupported TGA format " , file - > getFileName ( ) , ELL_ERROR ) ;
break ;
}
2024-03-20 19:35:52 +01:00
delete [ ] data ;
delete [ ] palette ;
2023-10-03 20:37:00 +02:00
return image ;
}
//! creates a loader which is able to load tgas
2024-03-20 19:35:52 +01:00
IImageLoader * createImageLoaderTGA ( )
2023-10-03 20:37:00 +02:00
{
return new CImageLoaderTGA ( ) ;
}
} // end namespace video
} // end namespace irr