2022-06-25 00:23:03 +08:00
/****************************************************************************
Copyright ( c ) 2010 - 2012 cocos2d - x . org
Copyright ( c ) 2013 - 2016 Chukong Technologies Inc .
http : //www.cocos2d-x.org
Permission is hereby granted , free of charge , to any person obtaining a copy
of this software and associated documentation files ( the " Software " ) , to deal
in the Software without restriction , including without limitation the rights
to use , copy , modify , merge , publish , distribute , sublicense , and / or sell
copies of the Software , and to permit persons to whom the Software is
furnished to do so , subject to the following conditions :
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software .
THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM ,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# include "CCImage.h"
# include <string>
# include <ctype.h>
# include <assert.h>
# include "base/CCData.h"
# include "base/ccConfig.h" // CC_USE_JPEG, CC_USE_TIFF, CC_USE_WEBP
# include "base/ccUtils.h"
# ifndef MIN
# define MIN(x,y) (((x) > (y)) ? (y) : (x))
# endif // MIN
# ifndef MAX
# define MAX(x,y) (((x) < (y)) ? (y) : (x))
# endif // MAX
extern " C "
{
// To resolve link error when building 32bits with Xcode 6.
// More information please refer to the discussion in https://github.com/cocos2d/cocos2d-x/pull/6986
# if defined (__unix) || (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
# ifndef __ENABLE_COMPATIBILITY_WITH_UNIX_2003__
# define __ENABLE_COMPATIBILITY_WITH_UNIX_2003__
# include <stdio.h>
FILE * fopen $ UNIX2003 ( const char * filename , const char * mode )
{
return fopen ( filename , mode ) ;
}
size_t fwrite $ UNIX2003 ( const void * a , size_t b , size_t c , FILE * d )
{
return fwrite ( a , b , c , d ) ;
}
char * strerror $ UNIX2003 ( int errnum )
{
return strerror ( errnum ) ;
}
# endif
# endif
# if CC_USE_PNG
# include "png/png.h"
# endif //CC_USE_PNG
# if CC_USE_TIFF
# include "tiff/tiffio.h"
# endif //CC_USE_TIFF
# if CC_USE_JPEG
# include "jpeg/jpeglib.h"
# endif // CC_USE_JPEG
# include "base/etc1.h"
# include "base/etc2.h"
}
2023-10-30 22:32:32 +08:00
# include "base/astc.h"
2022-06-25 00:23:03 +08:00
# if CC_USE_WEBP
# include "webp/decode.h"
# endif // CC_USE_WEBP
# include "base/pvr.h"
# include "base/TGAlib.h"
# include "base/ccMacros.h"
# include "platform/CCStdC.h"
# include "platform/CCFileUtils.h"
# include "base/CCConfiguration.h"
# include "base/ZipUtils.h"
# if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
# include "platform/android/CCFileUtils-android.h"
2023-10-30 22:32:32 +08:00
# elif(CC_TARGET_PLATFORM == CC_PLATFORM_OPENHARMONY)
# include "platform/openharmony/FileUtils-openharmony.h"
2022-06-25 00:23:03 +08:00
# endif
# include <map>
# define CC_GL_ATC_RGB_AMD 0x8C92
# define CC_GL_ATC_RGBA_EXPLICIT_ALPHA_AMD 0x8C93
# define CC_GL_ATC_RGBA_INTERPOLATED_ALPHA_AMD 0x87EE
NS_CC_BEGIN
//////////////////////////////////////////////////////////////////////////
//struct and data for pvr structure
namespace
{
typedef Image : : PixelFormatInfoMap : : value_type PixelFormatInfoMapValue ;
static const PixelFormatInfoMapValue TexturePixelFormatInfoTablesValue [ ] =
{
PixelFormatInfoMapValue ( Image : : PixelFormat : : BGRA8888 , Image : : PixelFormatInfo ( GL_BGRA , GL_BGRA , GL_UNSIGNED_BYTE , 32 , false , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : RGBA8888 , Image : : PixelFormatInfo ( GL_RGBA , GL_RGBA , GL_UNSIGNED_BYTE , 32 , false , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : RGBA4444 , Image : : PixelFormatInfo ( GL_RGBA , GL_RGBA , GL_UNSIGNED_SHORT_4_4_4_4 , 16 , false , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : RGB5A1 , Image : : PixelFormatInfo ( GL_RGBA , GL_RGBA , GL_UNSIGNED_SHORT_5_5_5_1 , 16 , false , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : RGB565 , Image : : PixelFormatInfo ( GL_RGB , GL_RGB , GL_UNSIGNED_SHORT_5_6_5 , 16 , false , false ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : RGB888 , Image : : PixelFormatInfo ( GL_RGB , GL_RGB , GL_UNSIGNED_BYTE , 24 , false , false ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : A8 , Image : : PixelFormatInfo ( GL_ALPHA , GL_ALPHA , GL_UNSIGNED_BYTE , 8 , false , false ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : I8 , Image : : PixelFormatInfo ( GL_LUMINANCE , GL_LUMINANCE , GL_UNSIGNED_BYTE , 8 , false , false ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : AI88 , Image : : PixelFormatInfo ( GL_LUMINANCE_ALPHA , GL_LUMINANCE_ALPHA , GL_UNSIGNED_BYTE , 16 , false , true ) ) ,
# ifdef GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG
PixelFormatInfoMapValue ( Image : : PixelFormat : : PVRTC2 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG , 0xFFFFFFFF , 0xFFFFFFFF , 2 , true , false ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : PVRTC2A , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG , 0xFFFFFFFF , 0xFFFFFFFF , 2 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : PVRTC4 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG , 0xFFFFFFFF , 0xFFFFFFFF , 4 , true , false ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : PVRTC4A , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG , 0xFFFFFFFF , 0xFFFFFFFF , 4 , true , true ) ) ,
# endif
# ifdef GL_ETC1_RGB8_OES
PixelFormatInfoMapValue ( Image : : PixelFormat : : ETC , Image : : PixelFormatInfo ( GL_ETC1_RGB8_OES , 0xFFFFFFFF , 0xFFFFFFFF , 4 , true , false ) ) ,
# endif
# ifdef GL_COMPRESSED_RGB8_ETC2
PixelFormatInfoMapValue ( Image : : PixelFormat : : ETC2_RGB , Image : : PixelFormatInfo ( GL_COMPRESSED_RGB8_ETC2 , 0xFFFFFFFF , 0xFFFFFFFF , 4 , true , false ) ) ,
# endif
# ifdef GL_COMPRESSED_RGBA8_ETC2_EAC
PixelFormatInfoMapValue ( Image : : PixelFormat : : ETC2_RGBA , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA8_ETC2_EAC , 0xFFFFFFFF , 0xFFFFFFFF , 8 , true , true ) ) ,
# endif
2023-10-30 22:32:32 +08:00
# ifdef GL_COMPRESSED_RGBA_ASTC_4x4_KHR
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_4x4 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_4x4_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 8 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_5x4 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_5x4_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 6 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_5x5 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_5x5_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 5 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_6x5 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_6x5_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 4 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_6x6 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_6x6_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 5 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_8x5 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_8x5_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 3 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_8x6 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_8x6_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 3 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_8x8 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_8x8_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 2 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_10x5 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_10x5_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 3 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_10x6 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_10x6_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 2 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_10x8 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_10x8_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 2 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_10x10 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_10x10_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 1 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_12x10 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_12x10_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 1 , true , true ) ) ,
PixelFormatInfoMapValue ( Image : : PixelFormat : : ASTC_RGBA_12x12 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_ASTC_12x12_KHR , 0xFFFFFFFF , 0xFFFFFFFF , 1 , true , true ) ) ,
# endif
2022-06-25 00:23:03 +08:00
# ifdef GL_COMPRESSED_RGBA_S3TC_DXT1_EXT
PixelFormatInfoMapValue ( Image : : PixelFormat : : S3TC_DXT1 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_S3TC_DXT1_EXT , 0xFFFFFFFF , 0xFFFFFFFF , 4 , true , false ) ) ,
# endif
# ifdef GL_COMPRESSED_RGBA_S3TC_DXT3_EXT
PixelFormatInfoMapValue ( Image : : PixelFormat : : S3TC_DXT3 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_S3TC_DXT3_EXT , 0xFFFFFFFF , 0xFFFFFFFF , 8 , true , false ) ) ,
# endif
# ifdef GL_COMPRESSED_RGBA_S3TC_DXT5_EXT
PixelFormatInfoMapValue ( Image : : PixelFormat : : S3TC_DXT5 , Image : : PixelFormatInfo ( GL_COMPRESSED_RGBA_S3TC_DXT5_EXT , 0xFFFFFFFF , 0xFFFFFFFF , 8 , true , false ) ) ,
# endif
# ifdef GL_ATC_RGB_AMD
PixelFormatInfoMapValue ( Image : : PixelFormat : : ATC_RGB , Image : : PixelFormatInfo ( GL_ATC_RGB_AMD ,
0xFFFFFFFF , 0xFFFFFFFF , 4 , true , false ) ) ,
# endif
# ifdef GL_ATC_RGBA_EXPLICIT_ALPHA_AMD
PixelFormatInfoMapValue ( Image : : PixelFormat : : ATC_EXPLICIT_ALPHA , Image : : PixelFormatInfo ( GL_ATC_RGBA_EXPLICIT_ALPHA_AMD ,
0xFFFFFFFF , 0xFFFFFFFF , 8 , true , false ) ) ,
# endif
# ifdef GL_ATC_RGBA_INTERPOLATED_ALPHA_AMD
PixelFormatInfoMapValue ( Image : : PixelFormat : : ATC_INTERPOLATED_ALPHA , Image : : PixelFormatInfo ( GL_ATC_RGBA_INTERPOLATED_ALPHA_AMD ,
0xFFFFFFFF , 0xFFFFFFFF , 8 , true , false ) ) ,
# endif
} ;
//CLASS IMPLEMENTATIONS:
//The PixpelFormat corresponding information
const Image : : PixelFormatInfoMap _pixelFormatInfoTables ( TexturePixelFormatInfoTablesValue ,
TexturePixelFormatInfoTablesValue + sizeof ( TexturePixelFormatInfoTablesValue ) / sizeof ( TexturePixelFormatInfoTablesValue [ 0 ] ) ) ;
const Image : : PixelFormatInfoMap & getPixelFormatInfoMap ( )
{
return _pixelFormatInfoTables ;
}
static const int PVR_TEXTURE_FLAG_TYPE_MASK = 0xff ;
static bool _PVRHaveAlphaPremultiplied = false ;
// Values taken from PVRTexture.h from http://www.imgtec.com
enum class PVR2TextureFlag
{
Mipmap = ( 1 < < 8 ) , // has mip map levels
Twiddle = ( 1 < < 9 ) , // is twiddled
Bumpmap = ( 1 < < 10 ) , // has normals encoded for a bump map
Tiling = ( 1 < < 11 ) , // is bordered for tiled pvr
Cubemap = ( 1 < < 12 ) , // is a cubemap/skybox
FalseMipCol = ( 1 < < 13 ) , // are there false colored MIP levels
Volume = ( 1 < < 14 ) , // is this a volume texture
Alpha = ( 1 < < 15 ) , // v2.1 is there transparency info in the texture
VerticalFlip = ( 1 < < 16 ) , // v2.1 is the texture vertically flipped
} ;
enum class PVR3TextureFlag
{
PremultipliedAlpha = ( 1 < < 1 ) // has premultiplied alpha
} ;
static const char gPVRTexIdentifier [ 5 ] = " PVR! " ;
// v2
enum class PVR2TexturePixelFormat : unsigned char
{
RGBA4444 = 0x10 ,
RGBA5551 ,
RGBA8888 ,
RGB565 ,
RGB555 , // unsupported
RGB888 ,
I8 ,
AI88 ,
PVRTC2BPP_RGBA ,
PVRTC4BPP_RGBA ,
BGRA8888 ,
A8 ,
} ;
// v3
enum class PVR3TexturePixelFormat : uint64_t
{
PVRTC2BPP_RGB = 0ULL ,
PVRTC2BPP_RGBA = 1ULL ,
PVRTC4BPP_RGB = 2ULL ,
PVRTC4BPP_RGBA = 3ULL ,
PVRTC2_2BPP_RGBA = 4ULL ,
PVRTC2_4BPP_RGBA = 5ULL ,
ETC1 = 6ULL ,
DXT1 = 7ULL ,
DXT2 = 8ULL ,
DXT3 = 9ULL ,
DXT4 = 10ULL ,
DXT5 = 11ULL ,
BC1 = 7ULL ,
BC2 = 9ULL ,
BC3 = 11ULL ,
BC4 = 12ULL ,
BC5 = 13ULL ,
BC6 = 14ULL ,
BC7 = 15ULL ,
UYVY = 16ULL ,
YUY2 = 17ULL ,
BW1bpp = 18ULL ,
R9G9B9E5 = 19ULL ,
RGBG8888 = 20ULL ,
GRGB8888 = 21ULL ,
ETC2_RGB = 22ULL ,
ETC2_RGBA = 23ULL ,
ETC2_RGBA1 = 24ULL ,
EAC_R11_Unsigned = 25ULL ,
EAC_R11_Signed = 26ULL ,
EAC_RG11_Unsigned = 27ULL ,
EAC_RG11_Signed = 28ULL ,
BGRA8888 = 0x0808080861726762ULL ,
RGBA8888 = 0x0808080861626772ULL ,
RGBA4444 = 0x0404040461626772ULL ,
RGBA5551 = 0x0105050561626772ULL ,
RGB565 = 0x0005060500626772ULL ,
RGB888 = 0x0008080800626772ULL ,
A8 = 0x0000000800000061ULL ,
L8 = 0x000000080000006cULL ,
LA88 = 0x000008080000616cULL ,
} ;
// v2
typedef const std : : map < PVR2TexturePixelFormat , Image : : PixelFormat > _pixel2_formathash ;
static const _pixel2_formathash : : value_type v2_pixel_formathash_value [ ] =
{
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : BGRA8888 , Image : : PixelFormat : : BGRA8888 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : RGBA8888 , Image : : PixelFormat : : RGBA8888 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : RGBA4444 , Image : : PixelFormat : : RGBA4444 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : RGBA5551 , Image : : PixelFormat : : RGB5A1 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : RGB565 , Image : : PixelFormat : : RGB565 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : RGB888 , Image : : PixelFormat : : RGB888 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : A8 , Image : : PixelFormat : : A8 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : I8 , Image : : PixelFormat : : I8 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : AI88 , Image : : PixelFormat : : AI88 ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : PVRTC2BPP_RGBA , Image : : PixelFormat : : PVRTC2A ) ,
_pixel2_formathash : : value_type ( PVR2TexturePixelFormat : : PVRTC4BPP_RGBA , Image : : PixelFormat : : PVRTC4A ) ,
} ;
static const int PVR2_MAX_TABLE_ELEMENTS = sizeof ( v2_pixel_formathash_value ) / sizeof ( v2_pixel_formathash_value [ 0 ] ) ;
static const _pixel2_formathash v2_pixel_formathash ( v2_pixel_formathash_value , v2_pixel_formathash_value + PVR2_MAX_TABLE_ELEMENTS ) ;
// v3
typedef const std : : map < PVR3TexturePixelFormat , Image : : PixelFormat > _pixel3_formathash ;
static _pixel3_formathash : : value_type v3_pixel_formathash_value [ ] =
{
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : BGRA8888 , Image : : PixelFormat : : BGRA8888 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : RGBA8888 , Image : : PixelFormat : : RGBA8888 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : RGBA4444 , Image : : PixelFormat : : RGBA4444 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : RGBA5551 , Image : : PixelFormat : : RGB5A1 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : RGB565 , Image : : PixelFormat : : RGB565 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : RGB888 , Image : : PixelFormat : : RGB888 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : A8 , Image : : PixelFormat : : A8 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : L8 , Image : : PixelFormat : : I8 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : LA88 , Image : : PixelFormat : : AI88 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : PVRTC2BPP_RGB , Image : : PixelFormat : : PVRTC2 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : PVRTC2BPP_RGBA , Image : : PixelFormat : : PVRTC2A ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : PVRTC4BPP_RGB , Image : : PixelFormat : : PVRTC4 ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : PVRTC4BPP_RGBA , Image : : PixelFormat : : PVRTC4A ) ,
_pixel3_formathash : : value_type ( PVR3TexturePixelFormat : : ETC1 , Image : : PixelFormat : : ETC ) ,
} ;
static const int PVR3_MAX_TABLE_ELEMENTS = sizeof ( v3_pixel_formathash_value ) / sizeof ( v3_pixel_formathash_value [ 0 ] ) ;
static const _pixel3_formathash v3_pixel_formathash ( v3_pixel_formathash_value , v3_pixel_formathash_value + PVR3_MAX_TABLE_ELEMENTS ) ;
typedef struct _PVRTexHeader
{
unsigned int headerLength ;
unsigned int height ;
unsigned int width ;
unsigned int numMipmaps ;
unsigned int flags ;
unsigned int dataLength ;
unsigned int bpp ;
unsigned int bitmaskRed ;
unsigned int bitmaskGreen ;
unsigned int bitmaskBlue ;
unsigned int bitmaskAlpha ;
unsigned int pvrTag ;
unsigned int numSurfs ;
} PVRv2TexHeader ;
# ifdef _MSC_VER
# pragma pack(push,1)
# endif
typedef struct
{
uint32_t version ;
uint32_t flags ;
uint64_t pixelFormat ;
uint32_t colorSpace ;
uint32_t channelType ;
uint32_t height ;
uint32_t width ;
uint32_t depth ;
uint32_t numberOfSurfaces ;
uint32_t numberOfFaces ;
uint32_t numberOfMipmaps ;
uint32_t metadataLength ;
# ifdef _MSC_VER
} PVRv3TexHeader ;
# pragma pack(pop)
# else
} __attribute__ ( ( packed ) ) PVRv3TexHeader ;
# endif
}
//pvr structure end
//////////////////////////////////////////////////////////////////////////
//struct and data for s3tc(dds) struct
namespace
{
struct DDColorKey
{
uint32_t colorSpaceLowValue ;
uint32_t colorSpaceHighValue ;
} ;
struct DDSCaps
{
uint32_t caps ;
uint32_t caps2 ;
uint32_t caps3 ;
uint32_t caps4 ;
} ;
struct DDPixelFormat
{
uint32_t size ;
uint32_t flags ;
uint32_t fourCC ;
uint32_t RGBBitCount ;
uint32_t RBitMask ;
uint32_t GBitMask ;
uint32_t BBitMask ;
uint32_t ABitMask ;
} ;
struct DDSURFACEDESC2
{
uint32_t size ;
uint32_t flags ;
uint32_t height ;
uint32_t width ;
union
{
uint32_t pitch ;
uint32_t linearSize ;
} DUMMYUNIONNAMEN1 ;
union
{
uint32_t backBufferCount ;
uint32_t depth ;
} DUMMYUNIONNAMEN5 ;
union
{
uint32_t mipMapCount ;
uint32_t refreshRate ;
uint32_t srcVBHandle ;
} DUMMYUNIONNAMEN2 ;
uint32_t alphaBitDepth ;
uint32_t reserved ;
uint32_t surface ;
union
{
DDColorKey ddckCKDestOverlay ;
uint32_t emptyFaceColor ;
} DUMMYUNIONNAMEN3 ;
DDColorKey ddckCKDestBlt ;
DDColorKey ddckCKSrcOverlay ;
DDColorKey ddckCKSrcBlt ;
union
{
DDPixelFormat ddpfPixelFormat ;
uint32_t FVF ;
} DUMMYUNIONNAMEN4 ;
DDSCaps ddsCaps ;
uint32_t textureStage ;
} ;
# pragma pack(push,1)
struct S3TCTexHeader
{
char fileCode [ 4 ] ;
DDSURFACEDESC2 ddsd ;
} ;
# pragma pack(pop)
}
//s3tc struct end
namespace
{
typedef struct
{
const unsigned char * data ;
ssize_t size ;
int offset ;
} tImageSource ;
# ifdef CC_USE_PNG
static void pngReadCallback ( png_structp png_ptr , png_bytep data , png_size_t length )
{
tImageSource * isource = ( tImageSource * ) png_get_io_ptr ( png_ptr ) ;
if ( ( int ) ( isource - > offset + length ) < = isource - > size )
{
memcpy ( data , isource - > data + isource - > offset , length ) ;
isource - > offset + = length ;
}
else
{
png_error ( png_ptr , " pngReaderCallback failed " ) ;
}
}
# endif //CC_USE_PNG
}
Image : : PixelFormat getDevicePixelFormat ( Image : : PixelFormat format )
{
switch ( format ) {
case Image : : PixelFormat : : PVRTC4 :
case Image : : PixelFormat : : PVRTC4A :
case Image : : PixelFormat : : PVRTC2 :
case Image : : PixelFormat : : PVRTC2A :
if ( Configuration : : getInstance ( ) - > supportsPVRTC ( ) )
return format ;
else
return Image : : PixelFormat : : RGBA8888 ;
case Image : : PixelFormat : : ETC :
if ( Configuration : : getInstance ( ) - > supportsETC ( ) )
return format ;
else
return Image : : PixelFormat : : RGB888 ;
default :
return format ;
}
}
//////////////////////////////////////////////////////////////////////////
// Implement Image
//////////////////////////////////////////////////////////////////////////
bool Image : : PNG_PREMULTIPLIED_ALPHA_ENABLED = false ;
Image : : Image ( )
: _data ( nullptr )
, _dataLen ( 0 )
, _width ( 0 )
, _height ( 0 )
, _fileType ( Format : : UNKNOWN )
, _renderFormat ( Image : : PixelFormat : : NONE )
, _numberOfMipmaps ( 0 )
, _hasPremultipliedAlpha ( false )
{
}
Image : : ~ Image ( )
{
CC_SAFE_FREE ( _data ) ;
}
bool Image : : initWithImageFile ( const std : : string & path )
{
bool ret = false ;
//NOTE: fullPathForFilename isn't threadsafe. we should make sure the parameter is a full path.
// _filePath = FileUtils::getInstance()->fullPathForFilename(path);
_filePath = path ;
Data data = FileUtils : : getInstance ( ) - > getDataFromFile ( _filePath ) ;
if ( ! data . isNull ( ) )
{
ret = initWithImageData ( data . getBytes ( ) , data . getSize ( ) ) ;
}
return ret ;
}
bool Image : : initWithImageData ( const unsigned char * data , ssize_t dataLen )
{
bool ret = false ;
do
{
CC_BREAK_IF ( ! data | | dataLen < = 0 ) ;
unsigned char * unpackedData = nullptr ;
ssize_t unpackedLen = 0 ;
//detect and unzip the compress file
if ( ZipUtils : : isCCZBuffer ( data , dataLen ) )
{
unpackedLen = ZipUtils : : inflateCCZBuffer ( data , dataLen , & unpackedData ) ;
}
else if ( ZipUtils : : isGZipBuffer ( data , dataLen ) )
{
unpackedLen = ZipUtils : : inflateMemory ( const_cast < unsigned char * > ( data ) , dataLen , & unpackedData ) ;
}
else
{
unpackedData = const_cast < unsigned char * > ( data ) ;
unpackedLen = dataLen ;
}
_fileType = detectFormat ( unpackedData , unpackedLen ) ;
switch ( _fileType )
{
case Format : : PNG :
ret = initWithPngData ( unpackedData , unpackedLen ) ;
break ;
case Format : : JPG :
ret = initWithJpgData ( unpackedData , unpackedLen ) ;
break ;
case Format : : TIFF :
ret = initWithTiffData ( unpackedData , unpackedLen ) ;
break ;
case Format : : WEBP :
ret = initWithWebpData ( unpackedData , unpackedLen ) ;
break ;
case Format : : PVR :
ret = initWithPVRData ( unpackedData , unpackedLen ) ;
break ;
case Format : : ETC :
ret = initWithETCData ( unpackedData , unpackedLen ) ;
break ;
case Format : : ETC2 :
ret = initWithETC2Data ( unpackedData , unpackedLen ) ;
break ;
2023-10-30 22:32:32 +08:00
case Format : : ASTC :
ret = initWithASTCData ( unpackedData , unpackedLen ) ;
break ;
2022-06-25 00:23:03 +08:00
case Format : : S3TC :
ret = initWithS3TCData ( unpackedData , unpackedLen ) ;
break ;
default :
{
// load and detect image format
tImageTGA * tgaData = tgaLoadBuffer ( unpackedData , unpackedLen ) ;
if ( tgaData ! = nullptr & & tgaData - > status = = TGA_OK )
{
ret = initWithTGAData ( tgaData ) ;
}
else
{
CCLOG ( " Image: unsupported image format! " ) ;
}
free ( tgaData ) ;
break ;
}
}
if ( unpackedData ! = data )
{
free ( unpackedData ) ;
}
} while ( 0 ) ;
return ret ;
}
bool Image : : isPng ( const unsigned char * data , ssize_t dataLen )
{
if ( dataLen < = 8 )
{
return false ;
}
static const unsigned char PNG_SIGNATURE [ ] = { 0x89 , 0x50 , 0x4e , 0x47 , 0x0d , 0x0a , 0x1a , 0x0a } ;
return memcmp ( PNG_SIGNATURE , data , sizeof ( PNG_SIGNATURE ) ) = = 0 ;
}
bool Image : : isEtc ( const unsigned char * data , ssize_t dataLen )
{
return etc1_pkm_is_valid ( ( etc1_byte * ) data ) ? true : false ;
}
bool Image : : isEtc2 ( const unsigned char * data , ssize_t dataLen )
{
return etc2_pkm_is_valid ( ( etc2_byte * ) data ) ? true : false ;
}
bool Image : : isS3TC ( const unsigned char * data , ssize_t /*dataLen*/ )
{
S3TCTexHeader * header = ( S3TCTexHeader * ) data ;
if ( strncmp ( header - > fileCode , " DDS " , 3 ) ! = 0 )
{
return false ;
}
return true ;
}
2023-10-30 22:32:32 +08:00
bool Image : : isASTC ( const unsigned char * data , ssize_t dataLen )
{
return astcIsValid ( const_cast < astc_byte * > ( data ) ) ;
}
2022-06-25 00:23:03 +08:00
bool Image : : isJpg ( const unsigned char * data , ssize_t dataLen )
{
if ( dataLen < = 4 )
{
return false ;
}
static const unsigned char JPG_SOI [ ] = { 0xFF , 0xD8 } ;
return memcmp ( data , JPG_SOI , 2 ) = = 0 ;
}
bool Image : : isTiff ( const unsigned char * data , ssize_t dataLen )
{
if ( dataLen < = 4 )
{
return false ;
}
static const char * TIFF_II = " II " ;
static const char * TIFF_MM = " MM " ;
return ( memcmp ( data , TIFF_II , 2 ) = = 0 & & * ( static_cast < const unsigned char * > ( data ) + 2 ) = = 42 & & * ( static_cast < const unsigned char * > ( data ) + 3 ) = = 0 ) | |
( memcmp ( data , TIFF_MM , 2 ) = = 0 & & * ( static_cast < const unsigned char * > ( data ) + 2 ) = = 0 & & * ( static_cast < const unsigned char * > ( data ) + 3 ) = = 42 ) ;
}
bool Image : : isWebp ( const unsigned char * data , ssize_t dataLen )
{
if ( dataLen < = 12 )
{
return false ;
}
static const char * WEBP_RIFF = " RIFF " ;
static const char * WEBP_WEBP = " WEBP " ;
return memcmp ( data , WEBP_RIFF , 4 ) = = 0
& & memcmp ( static_cast < const unsigned char * > ( data ) + 8 , WEBP_WEBP , 4 ) = = 0 ;
}
bool Image : : isPvr ( const unsigned char * data , ssize_t dataLen )
{
if ( static_cast < size_t > ( dataLen ) < sizeof ( PVRv2TexHeader ) | | static_cast < size_t > ( dataLen ) < sizeof ( PVRv3TexHeader ) )
{
return false ;
}
const PVRv2TexHeader * headerv2 = static_cast < const PVRv2TexHeader * > ( static_cast < const void * > ( data ) ) ;
const PVRv3TexHeader * headerv3 = static_cast < const PVRv3TexHeader * > ( static_cast < const void * > ( data ) ) ;
return memcmp ( & headerv2 - > pvrTag , gPVRTexIdentifier , strlen ( gPVRTexIdentifier ) ) = = 0 | | CC_SWAP_INT32_BIG_TO_HOST ( headerv3 - > version ) = = 0x50565203 ;
}
Image : : Format Image : : detectFormat ( const unsigned char * data , ssize_t dataLen )
{
if ( isPng ( data , dataLen ) )
{
return Format : : PNG ;
}
else if ( isJpg ( data , dataLen ) )
{
return Format : : JPG ;
}
else if ( isTiff ( data , dataLen ) )
{
return Format : : TIFF ;
}
else if ( isWebp ( data , dataLen ) )
{
return Format : : WEBP ;
}
else if ( isPvr ( data , dataLen ) )
{
return Format : : PVR ;
}
else if ( isEtc ( data , dataLen ) )
{
return Format : : ETC ;
}
else if ( isEtc2 ( data , dataLen ) )
{
return Format : : ETC2 ;
}
2023-10-30 22:32:32 +08:00
else if ( isASTC ( data , dataLen ) )
{
return Format : : ASTC ;
}
2022-06-25 00:23:03 +08:00
else if ( isS3TC ( data , dataLen ) )
{
return Format : : S3TC ;
}
else
{
return Format : : UNKNOWN ;
}
}
int Image : : getBitPerPixel ( ) const
{
return getPixelFormatInfoMap ( ) . at ( _renderFormat ) . bpp ;
}
bool Image : : hasAlpha ( ) const
{
return getPixelFormatInfoMap ( ) . at ( _renderFormat ) . alpha ;
}
bool Image : : isCompressed ( ) const
{
return getPixelFormatInfoMap ( ) . at ( _renderFormat ) . compressed ;
}
const Image : : PixelFormatInfo & Image : : getPixelFormatInfo ( ) const
{
return getPixelFormatInfoMap ( ) . at ( _renderFormat ) ;
}
2023-10-30 22:32:32 +08:00
Image : : PixelFormat Image : : getASTCFormat ( const unsigned char * pHeader ) const
{
int xdim = pHeader [ ASTC_HEADER_MAGIC ] ;
int ydim = pHeader [ ASTC_HEADER_MAGIC + 1 ] ;
if ( xdim = = 4 ) return Image : : PixelFormat : : ASTC_RGBA_4x4 ;
if ( xdim = = 5 ) {
if ( ydim = = 4 ) return Image : : PixelFormat : : ASTC_RGBA_5x4 ;
return Image : : PixelFormat : : ASTC_RGBA_5x5 ;
}
if ( xdim = = 6 ) {
if ( ydim = = 5 ) return Image : : PixelFormat : : ASTC_RGBA_6x5 ;
return Image : : PixelFormat : : ASTC_RGBA_6x6 ;
}
if ( xdim = = 8 ) {
if ( ydim = = 5 ) return Image : : PixelFormat : : ASTC_RGBA_8x5 ;
if ( ydim = = 6 ) return Image : : PixelFormat : : ASTC_RGBA_8x6 ;
return Image : : PixelFormat : : ASTC_RGBA_8x8 ;
}
if ( xdim = = 10 ) {
if ( ydim = = 5 ) return Image : : PixelFormat : : ASTC_RGBA_10x5 ;
if ( ydim = = 6 ) return Image : : PixelFormat : : ASTC_RGBA_10x6 ;
if ( ydim = = 8 ) return Image : : PixelFormat : : ASTC_RGBA_10x8 ;
return Image : : PixelFormat : : ASTC_RGBA_10x10 ;
}
if ( ydim = = 10 ) return Image : : PixelFormat : : ASTC_RGBA_12x10 ;
return Image : : PixelFormat : : ASTC_RGBA_12x12 ;
}
2022-06-25 00:23:03 +08:00
namespace
{
/*
* ERROR HANDLING :
*
* The JPEG library ' s standard error handler ( jerror . c ) is divided into
* several " methods " which you can override individually . This lets you
* adjust the behavior without duplicating a lot of code , which you might
* have to update with each future release .
*
* We override the " error_exit " method so that control is returned to the
* library ' s caller when a fatal error occurs , rather than calling exit ( )
* as the standard error_exit method does .
*
* We use C ' s setjmp / longjmp facility to return control . This means that the
* routine which calls the JPEG library must first execute a setjmp ( ) call to
* establish the return point . We want the replacement error_exit to do a
* longjmp ( ) . But we need to make the setjmp buffer accessible to the
* error_exit routine . To do this , we make a private extension of the
* standard JPEG error handler object . ( If we were using C + + , we ' d say we
* were making a subclass of the regular error handler . )
*
* Here ' s the extended error handler struct :
*/
# if CC_USE_JPEG
struct MyErrorMgr
{
struct jpeg_error_mgr pub ; /* "public" fields */
jmp_buf setjmp_buffer ; /* for return to caller */
} ;
typedef struct MyErrorMgr * MyErrorPtr ;
/*
* Here ' s the routine that will replace the standard error_exit method :
*/
METHODDEF ( void )
myErrorExit ( j_common_ptr cinfo )
{
/* cinfo->err really points to a MyErrorMgr struct, so coerce pointer */
MyErrorPtr myerr = ( MyErrorPtr ) cinfo - > err ;
/* Always display the message. */
/* We could postpone this until after returning, if we chose. */
/* internal message function can't show error message in some platforms, so we rewrite it here.
* edit it if has version conflict .
*/
//(*cinfo->err->output_message) (cinfo);
char buffer [ JMSG_LENGTH_MAX ] ;
( * cinfo - > err - > format_message ) ( cinfo , buffer ) ;
CCLOG ( " jpeg error: %s " , buffer ) ;
/* Return control to the setjmp point */
longjmp ( myerr - > setjmp_buffer , 1 ) ;
}
# endif // CC_USE_JPEG
}
bool Image : : initWithJpgData ( const unsigned char * data , ssize_t dataLen )
{
# if CC_USE_WIC
return decodeWithWIC ( data , dataLen ) ;
# elif CC_USE_JPEG
/* these are standard libjpeg structures for reading(decompression) */
struct jpeg_decompress_struct cinfo ;
/* We use our private extension JPEG error handler.
* Note that this struct must live as long as the main JPEG parameter
* struct , to avoid dangling - pointer problems .
*/
struct MyErrorMgr jerr ;
/* libjpeg data structure for storing one row, that is, scanline of an image */
JSAMPROW row_pointer [ 1 ] = { 0 } ;
unsigned long location = 0 ;
bool ret = false ;
do
{
/* We set up the normal JPEG error routines, then override error_exit. */
cinfo . err = jpeg_std_error ( & jerr . pub ) ;
jerr . pub . error_exit = myErrorExit ;
/* Establish the setjmp return context for MyErrorExit to use. */
if ( setjmp ( jerr . setjmp_buffer ) )
{
/* If we get here, the JPEG code has signaled an error.
* We need to clean up the JPEG object , close the input file , and return .
*/
jpeg_destroy_decompress ( & cinfo ) ;
break ;
}
/* setup decompression process and source, then read JPEG header */
jpeg_create_decompress ( & cinfo ) ;
# ifndef CC_TARGET_QT5
jpeg_mem_src ( & cinfo , const_cast < unsigned char * > ( data ) , dataLen ) ;
# endif /* CC_TARGET_QT5 */
/* reading the image header which contains image information */
# if (JPEG_LIB_VERSION >= 90)
// libjpeg 0.9 adds stricter types.
jpeg_read_header ( & cinfo , TRUE ) ;
# else
jpeg_read_header ( & cinfo , TRUE ) ;
# endif
// we only support RGB or grayscale
if ( cinfo . jpeg_color_space = = JCS_GRAYSCALE )
{
_renderFormat = Image : : PixelFormat : : I8 ;
} else
{
cinfo . out_color_space = JCS_RGB ;
_renderFormat = Image : : PixelFormat : : RGB888 ;
}
/* Start decompression jpeg here */
jpeg_start_decompress ( & cinfo ) ;
/* init image info */
2023-10-30 22:32:32 +08:00
_isCompressed = false ;
2022-06-25 00:23:03 +08:00
_width = cinfo . output_width ;
_height = cinfo . output_height ;
_hasPremultipliedAlpha = false ;
_dataLen = cinfo . output_width * cinfo . output_height * cinfo . output_components ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
CC_BREAK_IF ( ! _data ) ;
/* now actually read the jpeg into the raw buffer */
/* read one scan line at a time */
while ( cinfo . output_scanline < cinfo . output_height )
{
row_pointer [ 0 ] = _data + location ;
location + = cinfo . output_width * cinfo . output_components ;
jpeg_read_scanlines ( & cinfo , row_pointer , 1 ) ;
}
/* When read image file with broken data, jpeg_finish_decompress() may cause error.
* Besides , jpeg_destroy_decompress ( ) shall deallocate and release all memory associated
* with the decompression object .
* So it doesn ' t need to call jpeg_finish_decompress ( ) .
*/
//jpeg_finish_decompress( &cinfo );
jpeg_destroy_decompress ( & cinfo ) ;
/* wrap up decompression, destroy objects, free pointers and close open files */
ret = true ;
} while ( 0 ) ;
return ret ;
# else
CCLOG ( " jpeg is not enabled, please enable it in ccConfig.h " ) ;
return false ;
# endif // CC_USE_JPEG
}
bool Image : : initWithPngData ( const unsigned char * data , ssize_t dataLen )
{
# if CC_USE_WIC
return decodeWithWIC ( data , dataLen ) ;
# elif CC_USE_PNG
// length of bytes to check if it is a valid png file
# define PNGSIGSIZE 8
bool ret = false ;
png_byte header [ PNGSIGSIZE ] = { 0 } ;
png_structp png_ptr = 0 ;
png_infop info_ptr = 0 ;
do
{
// png header len is 8 bytes
CC_BREAK_IF ( dataLen < PNGSIGSIZE ) ;
// check the data is png or not
memcpy ( header , data , PNGSIGSIZE ) ;
CC_BREAK_IF ( png_sig_cmp ( header , 0 , PNGSIGSIZE ) ) ;
// init png_struct
png_ptr = png_create_read_struct ( PNG_LIBPNG_VER_STRING , 0 , 0 , 0 ) ;
CC_BREAK_IF ( ! png_ptr ) ;
// init png_info
info_ptr = png_create_info_struct ( png_ptr ) ;
CC_BREAK_IF ( ! info_ptr ) ;
# if (CC_TARGET_PLATFORM != CC_PLATFORM_BADA && CC_TARGET_PLATFORM != CC_PLATFORM_NACL)
CC_BREAK_IF ( setjmp ( png_jmpbuf ( png_ptr ) ) ) ;
# endif
// set the read call back function
tImageSource imageSource ;
imageSource . data = ( unsigned char * ) data ;
imageSource . size = dataLen ;
imageSource . offset = 0 ;
png_set_read_fn ( png_ptr , & imageSource , pngReadCallback ) ;
// read png header info
// read png file info
png_read_info ( png_ptr , info_ptr ) ;
2023-10-30 22:32:32 +08:00
_isCompressed = false ;
2022-06-25 00:23:03 +08:00
_width = png_get_image_width ( png_ptr , info_ptr ) ;
_height = png_get_image_height ( png_ptr , info_ptr ) ;
png_byte bit_depth = png_get_bit_depth ( png_ptr , info_ptr ) ;
png_uint_32 color_type = png_get_color_type ( png_ptr , info_ptr ) ;
//CCLOG("color type %u", color_type);
// force palette images to be expanded to 24-bit RGB
// it may include alpha channel
if ( color_type = = PNG_COLOR_TYPE_PALETTE )
{
png_set_palette_to_rgb ( png_ptr ) ;
}
// low-bit-depth grayscale images are to be expanded to 8 bits
if ( color_type = = PNG_COLOR_TYPE_GRAY & & bit_depth < 8 )
{
bit_depth = 8 ;
png_set_expand_gray_1_2_4_to_8 ( png_ptr ) ;
}
// expand any tRNS chunk data into a full alpha channel
if ( png_get_valid ( png_ptr , info_ptr , PNG_INFO_tRNS ) )
{
png_set_tRNS_to_alpha ( png_ptr ) ;
}
// reduce images with 16-bit samples to 8 bits
if ( bit_depth = = 16 )
{
png_set_strip_16 ( png_ptr ) ;
}
// Expanded earlier for grayscale, now take care of palette and rgb
if ( bit_depth < 8 )
{
png_set_packing ( png_ptr ) ;
}
// update info
png_read_update_info ( png_ptr , info_ptr ) ;
bit_depth = png_get_bit_depth ( png_ptr , info_ptr ) ;
color_type = png_get_color_type ( png_ptr , info_ptr ) ;
switch ( color_type )
{
case PNG_COLOR_TYPE_GRAY :
_renderFormat = Image : : PixelFormat : : I8 ;
break ;
case PNG_COLOR_TYPE_GRAY_ALPHA :
_renderFormat = Image : : PixelFormat : : AI88 ;
break ;
case PNG_COLOR_TYPE_RGB :
_renderFormat = Image : : PixelFormat : : RGB888 ;
break ;
case PNG_COLOR_TYPE_RGB_ALPHA :
_renderFormat = Image : : PixelFormat : : RGBA8888 ;
break ;
default :
break ;
}
// read png data
png_size_t rowbytes ;
png_bytep * row_pointers = ( png_bytep * ) malloc ( sizeof ( png_bytep ) * _height ) ;
rowbytes = png_get_rowbytes ( png_ptr , info_ptr ) ;
_dataLen = rowbytes * _height ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
if ( ! _data )
{
if ( row_pointers ! = nullptr )
{
free ( row_pointers ) ;
}
break ;
}
for ( unsigned short i = 0 ; i < _height ; + + i )
{
row_pointers [ i ] = _data + i * rowbytes ;
}
png_read_image ( png_ptr , row_pointers ) ;
png_read_end ( png_ptr , nullptr ) ;
// premultiplied alpha for RGBA8888
if ( PNG_PREMULTIPLIED_ALPHA_ENABLED & & color_type = = PNG_COLOR_TYPE_RGB_ALPHA )
{
premultipliedAlpha ( ) ;
}
else
{
_hasPremultipliedAlpha = false ;
}
if ( row_pointers ! = nullptr )
{
free ( row_pointers ) ;
}
ret = true ;
} while ( 0 ) ;
if ( png_ptr )
{
png_destroy_read_struct ( & png_ptr , ( info_ptr ) ? & info_ptr : 0 , 0 ) ;
}
return ret ;
# else
CCLOG ( " png is not enabled, please enable it in ccConfig.h " ) ;
return false ;
# endif //CC_USE_PNG
}
# if CC_USE_TIFF
namespace
{
static tmsize_t tiffReadProc ( thandle_t fd , void * buf , tmsize_t size )
{
tImageSource * isource = ( tImageSource * ) fd ;
uint8 * ma ;
uint64 mb ;
unsigned long n ;
unsigned long o ;
tmsize_t p ;
ma = ( uint8 * ) buf ;
mb = size ;
p = 0 ;
while ( mb > 0 )
{
n = 0x80000000UL ;
if ( ( uint64 ) n > mb )
n = ( unsigned long ) mb ;
if ( ( int ) ( isource - > offset + n ) < = isource - > size )
{
memcpy ( ma , isource - > data + isource - > offset , n ) ;
isource - > offset + = n ;
o = n ;
}
else
{
return 0 ;
}
ma + = o ;
mb - = o ;
p + = o ;
if ( o ! = n )
{
break ;
}
}
return p ;
}
static tmsize_t tiffWriteProc ( thandle_t fd , void * buf , tmsize_t size )
{
CC_UNUSED_PARAM ( fd ) ;
CC_UNUSED_PARAM ( buf ) ;
CC_UNUSED_PARAM ( size ) ;
return 0 ;
}
static uint64 tiffSeekProc ( thandle_t fd , uint64 off , int whence )
{
tImageSource * isource = ( tImageSource * ) fd ;
uint64 ret = - 1 ;
do
{
if ( whence = = SEEK_SET )
{
CC_BREAK_IF ( off > = ( uint64 ) isource - > size ) ;
ret = isource - > offset = ( uint32 ) off ;
}
else if ( whence = = SEEK_CUR )
{
CC_BREAK_IF ( isource - > offset + off > = ( uint64 ) isource - > size ) ;
ret = isource - > offset + = ( uint32 ) off ;
}
else if ( whence = = SEEK_END )
{
CC_BREAK_IF ( off > = ( uint64 ) isource - > size ) ;
ret = isource - > offset = ( uint32 ) ( isource - > size - 1 - off ) ;
}
else
{
CC_BREAK_IF ( off > = ( uint64 ) isource - > size ) ;
ret = isource - > offset = ( uint32 ) off ;
}
} while ( 0 ) ;
return ret ;
}
static uint64 tiffSizeProc ( thandle_t fd )
{
tImageSource * imageSrc = ( tImageSource * ) fd ;
return imageSrc - > size ;
}
static int tiffCloseProc ( thandle_t fd )
{
CC_UNUSED_PARAM ( fd ) ;
return 0 ;
}
static int tiffMapProc ( thandle_t fd , void * * base , toff_t * size )
{
CC_UNUSED_PARAM ( fd ) ;
CC_UNUSED_PARAM ( base ) ;
CC_UNUSED_PARAM ( size ) ;
return 0 ;
}
static void tiffUnmapProc ( thandle_t fd , void * base , toff_t size )
{
CC_UNUSED_PARAM ( fd ) ;
CC_UNUSED_PARAM ( base ) ;
CC_UNUSED_PARAM ( size ) ;
}
}
# endif // CC_USE_TIFF
bool Image : : initWithTiffData ( const unsigned char * data , ssize_t dataLen )
{
# if CC_USE_WIC
return decodeWithWIC ( data , dataLen ) ;
# elif CC_USE_TIFF
bool ret = false ;
do
{
// set the read call back function
tImageSource imageSource ;
imageSource . data = data ;
imageSource . size = dataLen ;
imageSource . offset = 0 ;
TIFF * tif = TIFFClientOpen ( " file.tif " , " r " , ( thandle_t ) & imageSource ,
tiffReadProc , tiffWriteProc ,
tiffSeekProc , tiffCloseProc , tiffSizeProc ,
tiffMapProc ,
tiffUnmapProc ) ;
CC_BREAK_IF ( nullptr = = tif ) ;
uint32 w = 0 , h = 0 ;
uint16 bitsPerSample = 0 , samplePerPixel = 0 , planarConfig = 0 ;
size_t npixels = 0 ;
TIFFGetField ( tif , TIFFTAG_IMAGEWIDTH , & w ) ;
TIFFGetField ( tif , TIFFTAG_IMAGELENGTH , & h ) ;
TIFFGetField ( tif , TIFFTAG_BITSPERSAMPLE , & bitsPerSample ) ;
TIFFGetField ( tif , TIFFTAG_SAMPLESPERPIXEL , & samplePerPixel ) ;
TIFFGetField ( tif , TIFFTAG_PLANARCONFIG , & planarConfig ) ;
npixels = w * h ;
_renderFormat = Image : : PixelFormat : : RGBA8888 ;
_width = w ;
_height = h ;
_dataLen = npixels * sizeof ( uint32 ) ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
uint32 * raster = ( uint32 * ) _TIFFmalloc ( npixels * sizeof ( uint32 ) ) ;
if ( raster ! = nullptr )
{
if ( TIFFReadRGBAImageOriented ( tif , w , h , raster , ORIENTATION_TOPLEFT , 0 ) )
{
/* the raster data is pre-multiplied by the alpha component
after invoking TIFFReadRGBAImageOriented */
_hasPremultipliedAlpha = true ;
memcpy ( _data , raster , npixels * sizeof ( uint32 ) ) ;
}
_TIFFfree ( raster ) ;
}
TIFFClose ( tif ) ;
ret = true ;
} while ( 0 ) ;
return ret ;
# else
CCLOG ( " tiff is not enabled, please enable it in ccConfig.h " ) ;
return false ;
# endif //CC_USE_TIFF
}
namespace
{
bool testFormatForPvr2TCSupport ( PVR2TexturePixelFormat format )
{
return true ;
}
bool testFormatForPvr3TCSupport ( PVR3TexturePixelFormat format )
{
switch ( format ) {
case PVR3TexturePixelFormat : : BGRA8888 :
return Configuration : : getInstance ( ) - > supportsBGRA8888 ( ) ;
case PVR3TexturePixelFormat : : PVRTC2BPP_RGB :
case PVR3TexturePixelFormat : : PVRTC2BPP_RGBA :
case PVR3TexturePixelFormat : : PVRTC4BPP_RGB :
case PVR3TexturePixelFormat : : PVRTC4BPP_RGBA :
case PVR3TexturePixelFormat : : ETC1 :
case PVR3TexturePixelFormat : : RGBA8888 :
case PVR3TexturePixelFormat : : RGBA4444 :
case PVR3TexturePixelFormat : : RGBA5551 :
case PVR3TexturePixelFormat : : RGB565 :
case PVR3TexturePixelFormat : : RGB888 :
case PVR3TexturePixelFormat : : A8 :
case PVR3TexturePixelFormat : : L8 :
case PVR3TexturePixelFormat : : LA88 :
return true ;
default :
return false ;
}
}
}
bool Image : : initWithPVRv2Data ( const unsigned char * data , ssize_t dataLen )
{
int dataLength = 0 , dataOffset = 0 , dataSize = 0 ;
int blockSize = 0 , widthBlocks = 0 , heightBlocks = 0 ;
int width = 0 , height = 0 ;
//Cast first sizeof(PVRTexHeader) bytes of data stream as PVRTexHeader
const PVRv2TexHeader * header = static_cast < const PVRv2TexHeader * > ( static_cast < const void * > ( data ) ) ;
//Make sure that tag is in correct formatting
if ( memcmp ( & header - > pvrTag , gPVRTexIdentifier , strlen ( gPVRTexIdentifier ) ) ! = 0 )
{
return false ;
}
Configuration * configuration = Configuration : : getInstance ( ) ;
//can not detect the premultiplied alpha from pvr file, use _PVRHaveAlphaPremultiplied instead.
_hasPremultipliedAlpha = _PVRHaveAlphaPremultiplied ;
unsigned int flags = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > flags ) ;
PVR2TexturePixelFormat formatFlags = static_cast < PVR2TexturePixelFormat > ( flags & PVR_TEXTURE_FLAG_TYPE_MASK ) ;
bool flipped = ( flags & ( unsigned int ) PVR2TextureFlag : : VerticalFlip ) ? true : false ;
if ( flipped )
{
CCLOG ( " initWithPVRv2Data: WARNING: Image is flipped. Regenerate it using PVRTexTool " ) ;
}
if ( ! configuration - > supportsNPOT ( ) & &
( static_cast < int > ( header - > width ) ! = utils : : nextPOT ( header - > width )
| | static_cast < int > ( header - > height ) ! = utils : : nextPOT ( header - > height ) ) )
{
CCLOG ( " initWithPVRv2Data: ERROR: Loading an NPOT texture (%dx%d) but is not supported on this device " , header - > width , header - > height ) ;
return false ;
}
if ( ! testFormatForPvr2TCSupport ( formatFlags ) )
{
CCLOG ( " initWithPVRv2Data: WARNING: Unsupported PVR Pixel Format: 0x%02X. Re-encode it with a OpenGL pixel format variant " , ( int ) formatFlags ) ;
return false ;
}
if ( v2_pixel_formathash . find ( formatFlags ) = = v2_pixel_formathash . end ( ) )
{
CCLOG ( " initWithPVRv2Data: WARNING: Unsupported PVR Pixel Format: 0x%02X. Re-encode it with a OpenGL pixel format variant " , ( int ) formatFlags ) ;
return false ;
}
auto it = getPixelFormatInfoMap ( ) . find ( getDevicePixelFormat ( v2_pixel_formathash . at ( formatFlags ) ) ) ;
if ( it = = getPixelFormatInfoMap ( ) . end ( ) )
{
CCLOG ( " initWithPVRv2Data: WARNING: Unsupported PVR Pixel Format: 0x%02X. Re-encode it with a OpenGL pixel format variant " , ( int ) formatFlags ) ;
return false ;
}
_renderFormat = it - > first ;
int bpp = it - > second . bpp ;
//Reset num of mipmaps
_numberOfMipmaps = 0 ;
//Get size of mipmap
_width = width = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > width ) ;
_height = height = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > height ) ;
2023-10-30 22:32:32 +08:00
_isCompressed = true ;
2022-06-25 00:23:03 +08:00
//Get ptr to where data starts..
dataLength = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > dataLength ) ;
assert ( Configuration : : getInstance ( ) - > supportsPVRTC ( ) ) ;
2023-10-30 22:32:32 +08:00
if ( Configuration : : getInstance ( ) - > supportsPVRTC ( ) = = false )
{
CCLOG ( " initWithPVRv2Data: ERROR: Unsupported PVR Compress texture on this device " ) ;
return false ;
}
2022-06-25 00:23:03 +08:00
//Move by size of header
_dataLen = dataLen - sizeof ( PVRv2TexHeader ) ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
memcpy ( _data , ( unsigned char * ) data + sizeof ( PVRv2TexHeader ) , _dataLen ) ;
// Calculate the data size for each texture level and respect the minimum number of blocks
while ( dataOffset < dataLength )
{
switch ( formatFlags ) {
case PVR2TexturePixelFormat : : PVRTC2BPP_RGBA :
blockSize = 8 * 4 ; // Pixel by pixel block size for 2bpp
widthBlocks = width / 8 ;
heightBlocks = height / 4 ;
break ;
case PVR2TexturePixelFormat : : PVRTC4BPP_RGBA :
blockSize = 4 * 4 ; // Pixel by pixel block size for 4bpp
widthBlocks = width / 4 ;
heightBlocks = height / 4 ;
break ;
case PVR2TexturePixelFormat : : BGRA8888 :
if ( Configuration : : getInstance ( ) - > supportsBGRA8888 ( ) = = false )
{
CCLOG ( " initWithPVRv2Data: Image. BGRA8888 not supported on this device " ) ;
return false ;
}
default :
blockSize = 1 ;
widthBlocks = width ;
heightBlocks = height ;
break ;
}
// Clamp to minimum number of blocks
if ( widthBlocks < 2 )
{
widthBlocks = 2 ;
}
if ( heightBlocks < 2 )
{
heightBlocks = 2 ;
}
dataSize = widthBlocks * heightBlocks * ( ( blockSize * bpp ) / 8 ) ;
int packetLength = ( dataLength - dataOffset ) ;
packetLength = packetLength > dataSize ? dataSize : packetLength ;
//Make record to the mipmaps array and increment counter
_mipmaps [ _numberOfMipmaps ] . address = _data + dataOffset ;
_mipmaps [ _numberOfMipmaps ] . offset = dataOffset ;
_mipmaps [ _numberOfMipmaps ] . len = packetLength ;
_numberOfMipmaps + + ;
dataOffset + = packetLength ;
//Update width and height to the next lower power of two
width = std : : max ( width > > 1 , 1 ) ;
height = std : : max ( height > > 1 , 1 ) ;
}
return true ;
}
bool Image : : initWithPVRv3Data ( const unsigned char * data , ssize_t dataLen )
{
if ( static_cast < size_t > ( dataLen ) < sizeof ( PVRv3TexHeader ) )
{
return false ;
}
const PVRv3TexHeader * header = static_cast < const PVRv3TexHeader * > ( static_cast < const void * > ( data ) ) ;
// validate version
if ( CC_SWAP_INT32_BIG_TO_HOST ( header - > version ) ! = 0x50565203 )
{
CCLOG ( " initWithPVRv3Data: WARNING: pvr file version mismatch " ) ;
return false ;
}
// parse pixel format
PVR3TexturePixelFormat pixelFormat = static_cast < PVR3TexturePixelFormat > ( header - > pixelFormat ) ;
if ( ! testFormatForPvr3TCSupport ( pixelFormat ) )
{
CCLOG ( " initWithPVRv3Data: WARNING: Unsupported PVR Pixel Format: 0x%016llX. Re-encode it with a OpenGL pixel format variant " ,
static_cast < unsigned long long > ( pixelFormat ) ) ;
return false ;
}
if ( v3_pixel_formathash . find ( pixelFormat ) = = v3_pixel_formathash . end ( ) )
{
CCLOG ( " initWithPVRv3Data: WARNING: Unsupported PVR Pixel Format: 0x%016llX. Re-encode it with a OpenGL pixel format variant " ,
static_cast < unsigned long long > ( pixelFormat ) ) ;
return false ;
}
auto it = getPixelFormatInfoMap ( ) . find ( getDevicePixelFormat ( v3_pixel_formathash . at ( pixelFormat ) ) ) ;
if ( it = = getPixelFormatInfoMap ( ) . end ( ) )
{
CCLOG ( " initWithPVRv3Data: WARNING: Unsupported PVR Pixel Format: 0x%016llX. Re-encode it with a OpenGL pixel format variant " ,
static_cast < unsigned long long > ( pixelFormat ) ) ;
return false ;
}
_renderFormat = it - > first ;
int bpp = it - > second . bpp ;
// flags
int flags = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > flags ) ;
// PVRv3 specifies premultiply alpha in a flag -- should always respect this in PVRv3 files
if ( flags & ( unsigned int ) PVR3TextureFlag : : PremultipliedAlpha )
{
_hasPremultipliedAlpha = true ;
}
else
{
_hasPremultipliedAlpha = false ;
}
// sizing
int width = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > width ) ;
int height = CC_SWAP_INT32_LITTLE_TO_HOST ( header - > height ) ;
2023-10-30 22:32:32 +08:00
_isCompressed = true ;
2022-06-25 00:23:03 +08:00
_width = width ;
_height = height ;
int dataOffset = 0 , dataSize = 0 ;
int blockSize = 0 , widthBlocks = 0 , heightBlocks = 0 ;
assert ( Configuration : : getInstance ( ) - > supportsPVRTC ( ) ) ;
2023-10-30 22:32:32 +08:00
if ( Configuration : : getInstance ( ) - > supportsPVRTC ( ) = = false )
{
CCLOG ( " initWithPVRv3Data: ERROR: Unsupported PVR Compress texture on this device " ) ;
return false ;
}
2022-06-25 00:23:03 +08:00
_dataLen = dataLen - ( sizeof ( PVRv3TexHeader ) + header - > metadataLength ) ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
memcpy ( _data , static_cast < const unsigned char * > ( data ) + sizeof ( PVRv3TexHeader ) + header - > metadataLength , _dataLen ) ;
_numberOfMipmaps = header - > numberOfMipmaps ;
CCASSERT ( _numberOfMipmaps < MIPMAP_MAX , " Image: Maximum number of mimpaps reached. Increase the CC_MIPMAP_MAX value " ) ;
for ( int i = 0 ; i < _numberOfMipmaps ; i + + )
{
switch ( ( PVR3TexturePixelFormat ) pixelFormat )
{
case PVR3TexturePixelFormat : : PVRTC2BPP_RGB :
case PVR3TexturePixelFormat : : PVRTC2BPP_RGBA :
blockSize = 8 * 4 ; // Pixel by pixel block size for 2bpp
widthBlocks = width / 8 ;
heightBlocks = height / 4 ;
break ;
case PVR3TexturePixelFormat : : PVRTC4BPP_RGB :
case PVR3TexturePixelFormat : : PVRTC4BPP_RGBA :
blockSize = 4 * 4 ; // Pixel by pixel block size for 4bpp
widthBlocks = width / 4 ;
heightBlocks = height / 4 ;
break ;
case PVR3TexturePixelFormat : : ETC1 :
blockSize = 4 * 4 ; // Pixel by pixel block size for 4bpp
widthBlocks = width / 4 ;
heightBlocks = height / 4 ;
break ;
case PVR3TexturePixelFormat : : BGRA8888 :
if ( ! Configuration : : getInstance ( ) - > supportsBGRA8888 ( ) )
{
CCLOG ( " initWithPVRv3Data: Image. BGRA8888 not supported on this device " ) ;
return false ;
}
default :
blockSize = 1 ;
widthBlocks = width ;
heightBlocks = height ;
break ;
}
// Clamp to minimum number of blocks
if ( widthBlocks < 2 )
{
widthBlocks = 2 ;
}
if ( heightBlocks < 2 )
{
heightBlocks = 2 ;
}
dataSize = widthBlocks * heightBlocks * ( ( blockSize * bpp ) / 8 ) ;
auto packetLength = _dataLen - dataOffset ;
packetLength = packetLength > dataSize ? dataSize : packetLength ;
_mipmaps [ i ] . address = _data + dataOffset ;
_mipmaps [ i ] . offset = dataOffset ;
_mipmaps [ i ] . len = static_cast < int > ( packetLength ) ;
dataOffset + = packetLength ;
CCASSERT ( dataOffset < = _dataLen , " Image: Invalid length " ) ;
width = std : : max ( width > > 1 , 1 ) ;
height = std : : max ( height > > 1 , 1 ) ;
}
return true ;
}
bool Image : : initWithETCData ( const unsigned char * data , ssize_t dataLen )
{
const etc1_byte * header = static_cast < const etc1_byte * > ( data ) ;
//check the data
if ( ! etc1_pkm_is_valid ( header ) )
{
return false ;
}
_width = etc1_pkm_get_width ( header ) ;
_height = etc1_pkm_get_height ( header ) ;
2023-10-30 22:32:32 +08:00
_isCompressed = true ;
2022-06-25 00:23:03 +08:00
if ( 0 = = _width | | 0 = = _height )
{
return false ;
}
assert ( Configuration : : getInstance ( ) - > supportsETC ( ) ) ;
2023-10-30 22:32:32 +08:00
if ( Configuration : : getInstance ( ) - > supportsETC ( ) = = false )
{
CCLOG ( " initWithETCData: ERROR: Unsupported ETC Compress texture on this device " ) ;
return false ;
}
2022-06-25 00:23:03 +08:00
//old opengl version has no define for GL_ETC1_RGB8_OES, add macro to make compiler happy.
# ifdef GL_ETC1_RGB8_OES
_renderFormat = Image : : PixelFormat : : ETC ;
_dataLen = dataLen - ETC_PKM_HEADER_SIZE ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
memcpy ( _data , static_cast < const unsigned char * > ( data ) + ETC_PKM_HEADER_SIZE , _dataLen ) ;
return true ;
# endif
return false ;
}
bool Image : : initWithETC2Data ( const unsigned char * data , ssize_t dataLen )
{
const etc2_byte * header = static_cast < const etc2_byte * > ( data ) ;
//check the data
if ( ! etc2_pkm_is_valid ( header ) )
{
return false ;
}
_width = etc2_pkm_get_width ( header ) ;
_height = etc2_pkm_get_height ( header ) ;
2023-10-30 22:32:32 +08:00
_isCompressed = true ;
2022-06-25 00:23:03 +08:00
if ( 0 = = _width | | 0 = = _height )
{
return false ;
}
assert ( Configuration : : getInstance ( ) - > supportsETC2 ( ) ) ;
2023-10-30 22:32:32 +08:00
if ( Configuration : : getInstance ( ) - > supportsETC2 ( ) = = false )
{
CCLOG ( " initWithETC2Data: ERROR: Unsupported ETC2 Compress texture on this device " ) ;
return false ;
}
2022-06-25 00:23:03 +08:00
etc2_uint32 format = etc2_pkm_get_format ( header ) ;
if ( format = = ETC2_RGB_NO_MIPMAPS )
{
_renderFormat = Image : : PixelFormat : : ETC2_RGB ;
}
else
{
_renderFormat = Image : : PixelFormat : : ETC2_RGBA ;
}
_dataLen = dataLen - ETC2_PKM_HEADER_SIZE ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
memcpy ( _data , static_cast < const unsigned char * > ( data ) + ETC2_PKM_HEADER_SIZE , _dataLen ) ;
return true ;
}
bool Image : : initWithTGAData ( tImageTGA * tgaData )
{
bool ret = false ;
do
{
CC_BREAK_IF ( tgaData = = nullptr ) ;
// tgaLoadBuffer only support type 2, 3, 10
if ( 2 = = tgaData - > type | | 10 = = tgaData - > type )
{
// true color
// unsupported RGB555
if ( tgaData - > pixelDepth = = 16 )
{
_renderFormat = Image : : PixelFormat : : RGB5A1 ;
}
else if ( tgaData - > pixelDepth = = 24 )
{
_renderFormat = Image : : PixelFormat : : RGB888 ;
}
else if ( tgaData - > pixelDepth = = 32 )
{
_renderFormat = Image : : PixelFormat : : RGBA8888 ;
}
else
{
CCLOG ( " Image WARNING: unsupported true color tga data pixel format. FILE: %s " , _filePath . c_str ( ) ) ;
break ;
}
}
else if ( 3 = = tgaData - > type )
{
// gray
if ( 8 = = tgaData - > pixelDepth )
{
_renderFormat = Image : : PixelFormat : : I8 ;
}
else
{
// actually this won't happen, if it happens, maybe the image file is not a tga
CCLOG ( " Image WARNING: unsupported gray tga data pixel format. FILE: %s " , _filePath . c_str ( ) ) ;
break ;
}
}
_width = tgaData - > width ;
_height = tgaData - > height ;
_data = tgaData - > imageData ;
_dataLen = _width * _height * tgaData - > pixelDepth / 8 ;
_fileType = Format : : TGA ;
_hasPremultipliedAlpha = false ;
ret = true ;
} while ( false ) ;
if ( ret )
{
if ( FileUtils : : getInstance ( ) - > getFileExtension ( _filePath ) ! = " .tga " )
{
CCLOG ( " Image WARNING: the image file suffix is not tga, but parsed as a tga image file. FILE: %s " , _filePath . c_str ( ) ) ;
}
}
else
{
if ( tgaData & & tgaData - > imageData ! = nullptr )
{
free ( tgaData - > imageData ) ;
_data = nullptr ;
}
}
return ret ;
}
2023-10-30 22:32:32 +08:00
bool Image : : initWithASTCData ( const unsigned char * data , ssize_t dataLen )
{
const auto * header = static_cast < const astc_byte * > ( data ) ;
//check the data
if ( ! astcIsValid ( header ) )
{
return false ;
}
_width = astcGetWidth ( header ) ;
_height = astcGetHeight ( header ) ;
_isCompressed = true ;
if ( 0 = = _width | | 0 = = _height )
{
return false ;
}
assert ( Configuration : : getInstance ( ) - > supportsASTC ( ) ) ;
if ( Configuration : : getInstance ( ) - > supportsASTC ( ) = = false )
{
CCLOG ( " initWithASTCData: ERROR: Unsupported ASTC Compress texture on this device " ) ;
return false ;
}
_renderFormat = getASTCFormat ( header ) ;
_dataLen = dataLen - ASTC_HEADER_SIZE ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
if ( _data = = nullptr )
{
CCLOG ( " initWithASTCData: ERROR : Image _data is null! " ) ;
return false ;
}
memcpy ( _data , static_cast < const unsigned char * > ( data ) + ASTC_HEADER_SIZE , _dataLen ) ;
return true ;
}
2022-06-25 00:23:03 +08:00
static uint32_t makeFourCC ( char ch0 , char ch1 , char ch2 , char ch3 )
{
const uint32_t fourCC = ( ( uint32_t ) ( char ) ( ch0 ) | ( ( uint32_t ) ( char ) ( ch1 ) < < 8 ) | ( ( uint32_t ) ( char ) ( ch2 ) < < 16 ) | ( ( uint32_t ) ( char ) ( ch3 ) < < 24 ) ) ;
return fourCC ;
}
bool Image : : initWithS3TCData ( const unsigned char * data , ssize_t dataLen )
{
const uint32_t FOURCC_DXT1 = makeFourCC ( ' D ' , ' X ' , ' T ' , ' 1 ' ) ;
const uint32_t FOURCC_DXT3 = makeFourCC ( ' D ' , ' X ' , ' T ' , ' 3 ' ) ;
const uint32_t FOURCC_DXT5 = makeFourCC ( ' D ' , ' X ' , ' T ' , ' 5 ' ) ;
/* load the .dds file */
S3TCTexHeader * header = ( S3TCTexHeader * ) data ;
unsigned char * pixelData = static_cast < unsigned char * > ( malloc ( ( dataLen - sizeof ( S3TCTexHeader ) ) * sizeof ( unsigned char ) ) ) ;
memcpy ( ( void * ) pixelData , data + sizeof ( S3TCTexHeader ) , dataLen - sizeof ( S3TCTexHeader ) ) ;
_width = header - > ddsd . width ;
_height = header - > ddsd . height ;
_numberOfMipmaps = MAX ( 1 , header - > ddsd . DUMMYUNIONNAMEN2 . mipMapCount ) ; //if dds header reports 0 mipmaps, set to 1 to force correct software decoding (if needed).
_dataLen = 0 ;
int blockSize = ( FOURCC_DXT1 = = header - > ddsd . DUMMYUNIONNAMEN4 . ddpfPixelFormat . fourCC ) ? 8 : 16 ;
/* calculate the dataLen */
int width = _width ;
int height = _height ;
assert ( Configuration : : getInstance ( ) - > supportsS3TC ( ) ) ;
2023-10-30 22:32:32 +08:00
if ( Configuration : : getInstance ( ) - > supportsS3TC ( ) = = false )
{
CCLOG ( " initWithS3TCData: ERROR: Unsupported S3TC Compress texture on this device " ) ;
return false ;
}
2022-06-25 00:23:03 +08:00
_dataLen = dataLen - sizeof ( S3TCTexHeader ) ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
memcpy ( ( void * ) _data , ( void * ) pixelData , _dataLen ) ;
/* if hardware supports s3tc, set pixelformat before loading mipmaps, to support non-mipmapped textures */
//decode texture through hardware
if ( FOURCC_DXT1 = = header - > ddsd . DUMMYUNIONNAMEN4 . ddpfPixelFormat . fourCC )
{
_renderFormat = PixelFormat : : S3TC_DXT1 ;
}
else if ( FOURCC_DXT3 = = header - > ddsd . DUMMYUNIONNAMEN4 . ddpfPixelFormat . fourCC )
{
_renderFormat = PixelFormat : : S3TC_DXT3 ;
}
else if ( FOURCC_DXT5 = = header - > ddsd . DUMMYUNIONNAMEN4 . ddpfPixelFormat . fourCC )
{
_renderFormat = PixelFormat : : S3TC_DXT5 ;
}
/* load the mipmaps */
int encodeOffset = 0 ;
width = _width ;
height = _height ;
for ( int i = 0 ; i < _numberOfMipmaps & & ( width | | height ) ; + + i )
{
if ( width = = 0 ) width = 1 ;
if ( height = = 0 ) height = 1 ;
int size = ( ( width + 3 ) / 4 ) * ( ( height + 3 ) / 4 ) * blockSize ;
//decode texture through hardware
_mipmaps [ i ] . address = ( unsigned char * ) _data + encodeOffset ;
_mipmaps [ i ] . offset = encodeOffset ;
_mipmaps [ i ] . len = size ;
encodeOffset + = size ;
width > > = 1 ;
height > > = 1 ;
}
/* end load the mipmaps */
if ( pixelData ! = nullptr )
{
free ( pixelData ) ;
}
return true ;
}
bool Image : : initWithPVRData ( const unsigned char * data , ssize_t dataLen )
{
return initWithPVRv2Data ( data , dataLen ) | | initWithPVRv3Data ( data , dataLen ) ;
}
bool Image : : initWithWebpData ( const unsigned char * data , ssize_t dataLen )
{
# if CC_USE_WEBP
bool ret = false ;
# if (CC_TARGET_PLATFORM == CC_PLATFORM_WINRT)
CCLOG ( " WEBP image format not supported on WinRT or WP8 " ) ;
# else
do
{
WebPDecoderConfig config ;
if ( WebPInitDecoderConfig ( & config ) = = 0 ) break ;
if ( WebPGetFeatures ( static_cast < const uint8_t * > ( data ) , dataLen , & config . input ) ! = VP8_STATUS_OK ) break ;
if ( config . input . width = = 0 | | config . input . height = = 0 ) break ;
config . output . colorspace = config . input . has_alpha ? MODE_rgbA : MODE_RGB ;
_renderFormat = config . input . has_alpha ? Image : : PixelFormat : : RGBA8888 : Image : : PixelFormat : : RGB888 ;
_width = config . input . width ;
_height = config . input . height ;
2023-10-30 22:32:32 +08:00
_isCompressed = false ;
2022-06-25 00:23:03 +08:00
//we ask webp to give data with premultiplied alpha
_hasPremultipliedAlpha = ( config . input . has_alpha ! = 0 ) ;
_dataLen = _width * _height * ( config . input . has_alpha ? 4 : 3 ) ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
config . output . u . RGBA . rgba = static_cast < uint8_t * > ( _data ) ;
config . output . u . RGBA . stride = _width * ( config . input . has_alpha ? 4 : 3 ) ;
config . output . u . RGBA . size = _dataLen ;
config . output . is_external_memory = 1 ;
if ( WebPDecode ( static_cast < const uint8_t * > ( data ) , dataLen , & config ) ! = VP8_STATUS_OK )
{
free ( _data ) ;
_data = nullptr ;
break ;
}
ret = true ;
} while ( 0 ) ;
# endif // (CC_TARGET_PLATFORM == CC_PLATFORM_WINRT)
return ret ;
# else
CCLOG ( " webp is not enabled, please enable it in ccConfig.h " ) ;
return false ;
# endif // CC_USE_WEBP
}
bool Image : : initWithRawData ( const unsigned char * data , ssize_t dataLen , int width , int height , int bitsPerComponent , bool preMulti )
{
bool ret = false ;
do
{
CC_BREAK_IF ( 0 = = width | | 0 = = height ) ;
_height = height ;
_width = width ;
_hasPremultipliedAlpha = preMulti ;
_renderFormat = Image : : PixelFormat : : RGBA8888 ;
2023-10-30 22:32:32 +08:00
_isCompressed = false ;
2022-06-25 00:23:03 +08:00
// only RGBA8888 supported
int bytesPerComponent = 4 ;
_dataLen = height * width * bytesPerComponent ;
_data = static_cast < unsigned char * > ( malloc ( _dataLen * sizeof ( unsigned char ) ) ) ;
CC_BREAK_IF ( ! _data ) ;
memcpy ( _data , data , _dataLen ) ;
ret = true ;
} while ( 0 ) ;
return ret ;
}
# if (CC_TARGET_PLATFORM != CC_PLATFORM_IOS)
bool Image : : saveToFile ( const std : : string & filename , bool isToRGB )
{
//only support for Image::PixelFormat::RGB888 or Image::PixelFormat::RGBA8888 uncompressed data
if ( isCompressed ( ) | | ( _renderFormat ! = Image : : PixelFormat : : RGB888 & & _renderFormat ! = Image : : PixelFormat : : RGBA8888 ) )
{
CCLOG ( " saveToFile: Image: saveToFile is only support for Image::PixelFormat::RGB888 or Image::PixelFormat::RGBA8888 uncompressed data for now " ) ;
return false ;
}
std : : string fileExtension = FileUtils : : getInstance ( ) - > getFileExtension ( filename ) ;
if ( fileExtension = = " .png " )
{
return saveImageToPNG ( filename , isToRGB ) ;
}
else if ( fileExtension = = " .jpg " )
{
return saveImageToJPG ( filename ) ;
}
else
{
CCLOG ( " saveToFile: Image: saveToFile no support file extension(only .png or .jpg) for file: %s " , filename . c_str ( ) ) ;
return false ;
}
}
# endif // (CC_TARGET_PLATFORM != CC_PLATFORM_IOS)
bool Image : : saveImageToPNG ( const std : : string & filePath , bool isToRGB )
{
# if CC_USE_WIC
return encodeWithWIC ( filePath , isToRGB , GUID_ContainerFormatPng ) ;
# elif CC_USE_PNG
bool ret = false ;
do
{
FILE * fp ;
png_structp png_ptr ;
png_infop info_ptr ;
png_colorp palette ;
png_bytep * row_pointers ;
fp = fopen ( FileUtils : : getInstance ( ) - > getSuitableFOpen ( filePath ) . c_str ( ) , " wb " ) ;
CC_BREAK_IF ( nullptr = = fp ) ;
png_ptr = png_create_write_struct ( PNG_LIBPNG_VER_STRING , nullptr , nullptr , nullptr ) ;
if ( nullptr = = png_ptr )
{
fclose ( fp ) ;
break ;
}
info_ptr = png_create_info_struct ( png_ptr ) ;
if ( nullptr = = info_ptr )
{
fclose ( fp ) ;
png_destroy_write_struct ( & png_ptr , nullptr ) ;
break ;
}
# if (CC_TARGET_PLATFORM != CC_PLATFORM_BADA && CC_TARGET_PLATFORM != CC_PLATFORM_NACL)
if ( setjmp ( png_jmpbuf ( png_ptr ) ) )
{
fclose ( fp ) ;
png_destroy_write_struct ( & png_ptr , & info_ptr ) ;
break ;
}
# endif
png_init_io ( png_ptr , fp ) ;
if ( ! isToRGB & & hasAlpha ( ) )
{
png_set_IHDR ( png_ptr , info_ptr , _width , _height , 8 , PNG_COLOR_TYPE_RGB_ALPHA ,
PNG_INTERLACE_NONE , PNG_COMPRESSION_TYPE_BASE , PNG_FILTER_TYPE_BASE ) ;
}
else
{
png_set_IHDR ( png_ptr , info_ptr , _width , _height , 8 , PNG_COLOR_TYPE_RGB ,
PNG_INTERLACE_NONE , PNG_COMPRESSION_TYPE_BASE , PNG_FILTER_TYPE_BASE ) ;
}
palette = ( png_colorp ) png_malloc ( png_ptr , PNG_MAX_PALETTE_LENGTH * sizeof ( png_color ) ) ;
png_set_PLTE ( png_ptr , info_ptr , palette , PNG_MAX_PALETTE_LENGTH ) ;
png_write_info ( png_ptr , info_ptr ) ;
png_set_packing ( png_ptr ) ;
row_pointers = ( png_bytep * ) malloc ( _height * sizeof ( png_bytep ) ) ;
if ( row_pointers = = nullptr )
{
fclose ( fp ) ;
png_destroy_write_struct ( & png_ptr , & info_ptr ) ;
break ;
}
if ( ! hasAlpha ( ) )
{
for ( int i = 0 ; i < ( int ) _height ; i + + )
{
row_pointers [ i ] = ( png_bytep ) _data + i * _width * 3 ;
}
png_write_image ( png_ptr , row_pointers ) ;
free ( row_pointers ) ;
row_pointers = nullptr ;
}
else
{
if ( isToRGB )
{
unsigned char * tempData = static_cast < unsigned char * > ( malloc ( _width * _height * 3 * sizeof ( unsigned char ) ) ) ;
if ( nullptr = = tempData )
{
fclose ( fp ) ;
png_destroy_write_struct ( & png_ptr , & info_ptr ) ;
free ( row_pointers ) ;
row_pointers = nullptr ;
break ;
}
for ( int i = 0 ; i < _height ; + + i )
{
for ( int j = 0 ; j < _width ; + + j )
{
tempData [ ( i * _width + j ) * 3 ] = _data [ ( i * _width + j ) * 4 ] ;
tempData [ ( i * _width + j ) * 3 + 1 ] = _data [ ( i * _width + j ) * 4 + 1 ] ;
tempData [ ( i * _width + j ) * 3 + 2 ] = _data [ ( i * _width + j ) * 4 + 2 ] ;
}
}
for ( int i = 0 ; i < ( int ) _height ; i + + )
{
row_pointers [ i ] = ( png_bytep ) tempData + i * _width * 3 ;
}
png_write_image ( png_ptr , row_pointers ) ;
free ( row_pointers ) ;
row_pointers = nullptr ;
if ( tempData ! = nullptr )
{
free ( tempData ) ;
}
}
else
{
for ( int i = 0 ; i < ( int ) _height ; i + + )
{
row_pointers [ i ] = ( png_bytep ) _data + i * _width * 4 ;
}
png_write_image ( png_ptr , row_pointers ) ;
free ( row_pointers ) ;
row_pointers = nullptr ;
}
}
png_write_end ( png_ptr , info_ptr ) ;
png_free ( png_ptr , palette ) ;
palette = nullptr ;
png_destroy_write_struct ( & png_ptr , & info_ptr ) ;
fclose ( fp ) ;
ret = true ;
} while ( 0 ) ;
return ret ;
# else
CCLOG ( " png is not enabled, please enable it in ccConfig.h " ) ;
return false ;
# endif // CC_USE_PNG
}
bool Image : : saveImageToJPG ( const std : : string & filePath )
{
# if CC_USE_WIC
return encodeWithWIC ( filePath , false , GUID_ContainerFormatJpeg ) ;
# elif CC_USE_JPEG
bool ret = false ;
do
{
struct jpeg_compress_struct cinfo ;
struct jpeg_error_mgr jerr ;
FILE * outfile ; /* target file */
JSAMPROW row_pointer [ 1 ] ; /* pointer to JSAMPLE row[s] */
int row_stride ; /* physical row width in image buffer */
cinfo . err = jpeg_std_error ( & jerr ) ;
/* Now we can initialize the JPEG compression object. */
jpeg_create_compress ( & cinfo ) ;
CC_BREAK_IF ( ( outfile = fopen ( FileUtils : : getInstance ( ) - > getSuitableFOpen ( filePath ) . c_str ( ) , " wb " ) ) = = nullptr ) ;
jpeg_stdio_dest ( & cinfo , outfile ) ;
cinfo . image_width = _width ; /* image width and height, in pixels */
cinfo . image_height = _height ;
cinfo . input_components = 3 ; /* # of color components per pixel */
cinfo . in_color_space = JCS_RGB ; /* colorspace of input image */
jpeg_set_defaults ( & cinfo ) ;
jpeg_set_quality ( & cinfo , 90 , TRUE ) ;
jpeg_start_compress ( & cinfo , TRUE ) ;
row_stride = _width * 3 ; /* JSAMPLEs per row in image_buffer */
if ( hasAlpha ( ) )
{
unsigned char * tempData = static_cast < unsigned char * > ( malloc ( _width * _height * 3 * sizeof ( unsigned char ) ) ) ;
if ( nullptr = = tempData )
{
jpeg_finish_compress ( & cinfo ) ;
jpeg_destroy_compress ( & cinfo ) ;
fclose ( outfile ) ;
break ;
}
for ( int i = 0 ; i < _height ; + + i )
{
for ( int j = 0 ; j < _width ; + + j )
{
tempData [ ( i * _width + j ) * 3 ] = _data [ ( i * _width + j ) * 4 ] ;
tempData [ ( i * _width + j ) * 3 + 1 ] = _data [ ( i * _width + j ) * 4 + 1 ] ;
tempData [ ( i * _width + j ) * 3 + 2 ] = _data [ ( i * _width + j ) * 4 + 2 ] ;
}
}
while ( cinfo . next_scanline < cinfo . image_height )
{
row_pointer [ 0 ] = & tempData [ cinfo . next_scanline * row_stride ] ;
( void ) jpeg_write_scanlines ( & cinfo , row_pointer , 1 ) ;
}
if ( tempData ! = nullptr )
{
free ( tempData ) ;
}
}
else
{
while ( cinfo . next_scanline < cinfo . image_height ) {
row_pointer [ 0 ] = & _data [ cinfo . next_scanline * row_stride ] ;
( void ) jpeg_write_scanlines ( & cinfo , row_pointer , 1 ) ;
}
}
jpeg_finish_compress ( & cinfo ) ;
fclose ( outfile ) ;
jpeg_destroy_compress ( & cinfo ) ;
ret = true ;
} while ( 0 ) ;
return ret ;
# else
CCLOG ( " jpeg is not enabled, please enable it in ccConfig.h " ) ;
return false ;
# endif // CC_USE_JPEG
}
void Image : : premultipliedAlpha ( )
{
if ( PNG_PREMULTIPLIED_ALPHA_ENABLED & & _renderFormat = = Image : : PixelFormat : : RGBA8888 )
{
unsigned int * fourBytes = ( unsigned int * ) _data ;
for ( int i = 0 ; i < _width * _height ; i + + )
{
unsigned char * p = _data + i * 4 ;
fourBytes [ i ] = CC_RGB_PREMULTIPLY_ALPHA ( p [ 0 ] , p [ 1 ] , p [ 2 ] , p [ 3 ] ) ;
}
_hasPremultipliedAlpha = true ;
}
else
{
_hasPremultipliedAlpha = false ;
return ;
}
}
void Image : : setPVRImagesHavePremultipliedAlpha ( bool haveAlphaPremultiplied )
{
_PVRHaveAlphaPremultiplied = haveAlphaPremultiplied ;
}
NS_CC_END