wrap fowardPixels for hardware decoder

This commit is contained in:
halx99 2020-08-31 23:09:42 +08:00
parent dffbad412a
commit 5e0e017b2a
2 changed files with 177 additions and 166 deletions

View File

@ -502,13 +502,71 @@ backend::PixelFormat getDevicePixelFormat(backend::PixelFormat format)
case backend::PixelFormat::ETC: case backend::PixelFormat::ETC:
if(Configuration::getInstance()->supportsETC()) if(Configuration::getInstance()->supportsETC())
return format; return format;
case backend::PixelFormat::ETC2_RGB:
case backend::PixelFormat::ETC2_RGBA:
if(Configuration::getInstance()->supportsETC2())
return format;
else else
return backend::PixelFormat::RGB888; return backend::PixelFormat::RGBA8888;
case backend::PixelFormat::ASTC4:
case backend::PixelFormat::ASTC8:
if(Configuration::getInstance()->supportsASTC())
return format;
else
return backend::PixelFormat::RGBA8888;
default: default:
return format; return format;
} }
} }
namespace
{
bool testFormatForPvr2TCSupport(PVR2TexturePixelFormat /*format*/)
{
return true;
}
bool testFormatForPvr3TCSupport(PVR3TexturePixelFormat format)
{
switch (format) {
case PVR3TexturePixelFormat::DXT1:
case PVR3TexturePixelFormat::DXT3:
case PVR3TexturePixelFormat::DXT5:
return Configuration::getInstance()->supportsS3TC();
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;
}
}
}
namespace
{
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;
}
}
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
// Implement Image // Implement Image
////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////
@ -621,22 +679,22 @@ bool Image::initWithImageData(const unsigned char* data, ssize_t dataLen, bool o
ret = initWithWebpData(unpackedData, unpackedLen); ret = initWithWebpData(unpackedData, unpackedLen);
break; break;
case Format::PVR: case Format::PVR:
ret = initWithPVRData(unpackedData, unpackedLen); ret = initWithPVRData(unpackedData, unpackedLen, ownData);
break; break;
case Format::ETC: case Format::ETC:
ret = initWithETCData(unpackedData, unpackedLen); ret = initWithETCData(unpackedData, unpackedLen, ownData);
break; break;
case Format::ETC2: case Format::ETC2:
ret = initWithETC2Data(unpackedData, unpackedLen); ret = initWithETC2Data(unpackedData, unpackedLen, ownData);
break; break;
case Format::S3TC: case Format::S3TC:
ret = initWithS3TCData(unpackedData, unpackedLen); ret = initWithS3TCData(unpackedData, unpackedLen, ownData);
break; break;
case Format::ATITC: case Format::ATITC:
ret = initWithATITCData(unpackedData, unpackedLen); ret = initWithATITCData(unpackedData, unpackedLen, ownData);
break; break;
case Format::ASTC: case Format::ASTC:
ret = initWithASTCData(unpackedData, unpackedLen); ret = initWithASTCData(unpackedData, unpackedLen, ownData);
break; break;
case Format::BMP: case Format::BMP:
ret = initWithBmpData(unpackedData, unpackedLen); ret = initWithBmpData(unpackedData, unpackedLen);
@ -1141,46 +1199,82 @@ bool Image::initWithBmpData(unsigned char* data, ssize_t dataLen)
return true; return true;
} }
namespace bool Image::initWithTGAData(tImageTGA* tgaData)
{ {
bool testFormatForPvr2TCSupport(PVR2TexturePixelFormat /*format*/) bool ret = false;
do
{ {
return true; CC_BREAK_IF(tgaData == nullptr);
}
bool testFormatForPvr3TCSupport(PVR3TexturePixelFormat format) // tgaLoadBuffer only support type 2, 3, 10
if (2 == tgaData->type || 10 == tgaData->type)
{ {
switch (format) { // true color
case PVR3TexturePixelFormat::DXT1: // unsupported RGB555
case PVR3TexturePixelFormat::DXT3: if (tgaData->pixelDepth == 16)
case PVR3TexturePixelFormat::DXT5: {
return Configuration::getInstance()->supportsS3TC(); _pixelFormat = backend::PixelFormat::RGB5A1;
}
case PVR3TexturePixelFormat::BGRA8888: else if(tgaData->pixelDepth == 24)
return Configuration::getInstance()->supportsBGRA8888(); {
_pixelFormat = backend::PixelFormat::RGB888;
case PVR3TexturePixelFormat::PVRTC2BPP_RGB: }
case PVR3TexturePixelFormat::PVRTC2BPP_RGBA: else if(tgaData->pixelDepth == 32)
case PVR3TexturePixelFormat::PVRTC4BPP_RGB: {
case PVR3TexturePixelFormat::PVRTC4BPP_RGBA: _pixelFormat = backend::PixelFormat::RGBA8888;
case PVR3TexturePixelFormat::ETC1: }
case PVR3TexturePixelFormat::RGBA8888: else
case PVR3TexturePixelFormat::RGBA4444: {
case PVR3TexturePixelFormat::RGBA5551: CCLOG("Image WARNING: unsupported true color tga data pixel format. FILE: %s", _filePath.c_str());
case PVR3TexturePixelFormat::RGB565: break;
case PVR3TexturePixelFormat::RGB888:
case PVR3TexturePixelFormat::A8:
case PVR3TexturePixelFormat::L8:
case PVR3TexturePixelFormat::LA88:
return true;
default:
return false;
} }
} }
else if(3 == tgaData->type)
{
// gray
if (8 == tgaData->pixelDepth)
{
_pixelFormat = backend::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;
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;
} }
bool Image::initWithPVRv2Data(unsigned char * data, ssize_t dataLen) bool Image::initWithPVRv2Data(unsigned char * data, ssize_t dataLen, bool ownData)
{ {
int blockSize = 0, widthBlocks = 0, heightBlocks = 0; int blockSize = 0, widthBlocks = 0, heightBlocks = 0;
int width = 0, height = 0; int width = 0, height = 0;
@ -1329,9 +1423,7 @@ bool Image::initWithPVRv2Data(unsigned char * data, ssize_t dataLen)
} }
if (!_unpack) { // hardware decoder, hold data directly if (!_unpack) { // hardware decoder, hold data directly
_data = data; forwardPixels(data, dataLen, pixelOffset, ownData);
_dataLen = dataLen;
_offset = pixelOffset;
} }
else { else {
_data = _mipmaps[0].address; _data = _mipmaps[0].address;
@ -1341,7 +1433,7 @@ bool Image::initWithPVRv2Data(unsigned char * data, ssize_t dataLen)
return true; return true;
} }
bool Image::initWithPVRv3Data(unsigned char * data, ssize_t dataLen) bool Image::initWithPVRv3Data(unsigned char * data, ssize_t dataLen, bool ownData)
{ {
if (static_cast<size_t>(dataLen) < sizeof(PVRv3TexHeader)) if (static_cast<size_t>(dataLen) < sizeof(PVRv3TexHeader))
{ {
@ -1506,9 +1598,7 @@ bool Image::initWithPVRv3Data(unsigned char * data, ssize_t dataLen)
} }
if (!_unpack) { if (!_unpack) {
_data = data; forwardPixels(data, dataLen, pixelOffset, ownData);
_dataLen = dataLen;
_offset = pixelOffset;
} }
else { else {
_data = _mipmaps[0].address; _data = _mipmaps[0].address;
@ -1518,7 +1608,7 @@ bool Image::initWithPVRv3Data(unsigned char * data, ssize_t dataLen)
return true; return true;
} }
bool Image::initWithETCData(unsigned char* data, ssize_t dataLen) bool Image::initWithETCData(unsigned char* data, ssize_t dataLen, bool ownData)
{ {
const etc1_byte* header = static_cast<const etc1_byte*>(data); const etc1_byte* header = static_cast<const etc1_byte*>(data);
@ -1548,14 +1638,9 @@ bool Image::initWithETCData(unsigned char* data, ssize_t dataLen)
if (compressedFormat != backend::PixelFormat::NONE) if (compressedFormat != backend::PixelFormat::NONE)
{ {
//old opengl version has no define for GL_ETC1_RGB8_OES, add macro to make compiler happy.
#if defined(GL_ETC1_RGB8_OES) || defined(CC_USE_METAL)
_pixelFormat = compressedFormat; _pixelFormat = compressedFormat;
_data = data; forwardPixels(data, dataLen, ETC_PKM_HEADER_SIZE, ownData);
_dataLen = dataLen;
_offset = ETC_PKM_HEADER_SIZE;
return true; return true;
#endif
} }
else else
{ {
@ -1585,7 +1670,7 @@ bool Image::initWithETCData(unsigned char* data, ssize_t dataLen)
return false; return false;
} }
bool Image::initWithETC2Data(unsigned char* data, ssize_t dataLen) bool Image::initWithETC2Data(unsigned char* data, ssize_t dataLen, bool ownData)
{ {
const etc2_byte* header = static_cast<const etc2_byte*>(data); const etc2_byte* header = static_cast<const etc2_byte*>(data);
@ -1611,9 +1696,7 @@ bool Image::initWithETC2Data(unsigned char* data, ssize_t dataLen)
if (Configuration::getInstance()->supportsETC2()) { if (Configuration::getInstance()->supportsETC2()) {
_pixelFormat = format == ETC2_RGBA_NO_MIPMAPS ? backend::PixelFormat::ETC2_RGBA : backend::PixelFormat::ETC2_RGB; _pixelFormat = format == ETC2_RGBA_NO_MIPMAPS ? backend::PixelFormat::ETC2_RGBA : backend::PixelFormat::ETC2_RGB;
_data = data; forwardPixels(data, dataLen, ETC2_PKM_HEADER_SIZE, ownData);
_dataLen = dataLen;
_offset = ETC2_PKM_HEADER_SIZE;
return true; return true;
} }
else { else {
@ -1641,7 +1724,7 @@ bool Image::initWithETC2Data(unsigned char* data, ssize_t dataLen)
} }
} }
bool Image::initWithASTCData(unsigned char* data, ssize_t dataLen) bool Image::initWithASTCData(unsigned char* data, ssize_t dataLen, bool ownData)
{ {
ASTCTexHeader* header = (ASTCTexHeader*)data; ASTCTexHeader* header = (ASTCTexHeader*)data;
@ -1670,9 +1753,7 @@ bool Image::initWithASTCData(unsigned char* data, ssize_t dataLen)
_pixelFormat = backend::PixelFormat::ASTC8; _pixelFormat = backend::PixelFormat::ASTC8;
} }
_data = data; forwardPixels(data, dataLen, ASTC_HEAD_SIZE, ownData);
_dataLen = dataLen;
_offset = ASTC_HEAD_SIZE;
return true; return true;
} }
else else
@ -1702,92 +1783,7 @@ bool Image::initWithASTCData(unsigned char* data, ssize_t dataLen)
return false; return false;
} }
bool Image::initWithS3TCData(unsigned char * data, ssize_t dataLen, bool ownData)
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)
{
_pixelFormat = backend::PixelFormat::RGB5A1;
}
else if(tgaData->pixelDepth == 24)
{
_pixelFormat = backend::PixelFormat::RGB888;
}
else if(tgaData->pixelDepth == 32)
{
_pixelFormat = backend::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)
{
_pixelFormat = backend::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;
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;
}
namespace
{
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(unsigned char * data, ssize_t dataLen)
{ {
const uint32_t FOURCC_DXT1 = makeFourCC('D', 'X', 'T', '1'); const uint32_t FOURCC_DXT1 = makeFourCC('D', 'X', 'T', '1');
const uint32_t FOURCC_DXT3 = makeFourCC('D', 'X', 'T', '3'); const uint32_t FOURCC_DXT3 = makeFourCC('D', 'X', 'T', '3');
@ -1897,16 +1893,14 @@ bool Image::initWithS3TCData(unsigned char * data, ssize_t dataLen)
/* end load the mipmaps */ /* end load the mipmaps */
if (hardware) { if (hardware) {
_data = data; forwardPixels(data, dataLen, pixelOffset, ownData);
_dataLen = dataLen;
_offset = pixelOffset;
} }
return true; return true;
} }
bool Image::initWithATITCData(unsigned char *data, ssize_t dataLen) bool Image::initWithATITCData(unsigned char *data, ssize_t dataLen, bool ownData)
{ {
/* load the .ktx file */ /* load the .ktx file */
ATITCTexHeader *header = (ATITCTexHeader *)data; ATITCTexHeader *header = (ATITCTexHeader *)data;
@ -2030,17 +2024,29 @@ bool Image::initWithATITCData(unsigned char *data, ssize_t dataLen)
/* end load the mipmaps */ /* end load the mipmaps */
if (hardware) { if (hardware) {
_data = data; forwardPixels(data, dataLen, pixelOffset, ownData);
_dataLen = dataLen;
_offset = pixelOffset;
} }
return true; return true;
} }
bool Image::initWithPVRData(unsigned char * data, ssize_t dataLen) bool Image::initWithPVRData(unsigned char * data, ssize_t dataLen, bool ownData)
{ {
return initWithPVRv2Data(data, dataLen) || initWithPVRv3Data(data, dataLen); return initWithPVRv2Data(data, dataLen, ownData) || initWithPVRv3Data(data, dataLen, ownData);
}
void Image::forwardPixels(unsigned char* data, ssize_t dataLen, int offset, bool ownData)
{
if(ownData) {
_data = data;
_dataLen = dataLen;
_offset = offset;
}
else {
_dataLen = dataLen - offset;
_data = (unsigned char*)malloc(_dataLen);
memcpy(_data, data + offset, _dataLen);
}
} }
bool Image::initWithWebpData(unsigned char * data, ssize_t dataLen) bool Image::initWithWebpData(unsigned char * data, ssize_t dataLen)

View File

@ -165,23 +165,28 @@ public:
void reversePremultipliedAlpha(); void reversePremultipliedAlpha();
protected: protected:
typedef struct sImageTGA tImageTGA;
// all internal init function have chance to own the data for fast forward data to hardware decoder // all internal init function have chance to own the data for fast forward data to hardware decoder
// see: initWithImageData // see: initWithImageData
bool initWithJpgData(unsigned char * data, ssize_t dataLen); bool initWithJpgData(unsigned char * data, ssize_t dataLen);
bool initWithPngData(unsigned char * data, ssize_t dataLen); bool initWithPngData(unsigned char * data, ssize_t dataLen);
bool initWithBmpData(unsigned char *data, ssize_t dataLen); bool initWithBmpData(unsigned char *data, ssize_t dataLen);
bool initWithWebpData(unsigned char * data, ssize_t dataLen); bool initWithWebpData(unsigned char * data, ssize_t dataLen);
bool initWithPVRData(unsigned char * data, ssize_t dataLen);
bool initWithPVRv2Data(unsigned char * data, ssize_t dataLen);
bool initWithPVRv3Data(unsigned char * data, ssize_t dataLen);
bool initWithETCData(unsigned char* data, ssize_t dataLen);
bool initWithETC2Data(unsigned char* data, ssize_t dataLen);
bool initWithS3TCData(unsigned char * data, ssize_t dataLen);
bool initWithATITCData(unsigned char *data, ssize_t dataLen);
bool initWithASTCData(unsigned char* data, ssize_t dataLen);
typedef struct sImageTGA tImageTGA;
bool initWithTGAData(tImageTGA* tgaData); bool initWithTGAData(tImageTGA* tgaData);
bool initWithPVRData(unsigned char * data, ssize_t dataLen, bool ownData);
bool initWithPVRv2Data(unsigned char * data, ssize_t dataLen, bool ownData);
bool initWithPVRv3Data(unsigned char * data, ssize_t dataLen, bool ownData);
bool initWithETCData(unsigned char* data, ssize_t dataLen, bool ownData);
bool initWithETC2Data(unsigned char* data, ssize_t dataLen, bool ownData);
bool initWithASTCData(unsigned char* data, ssize_t dataLen, bool ownData);
bool initWithS3TCData(unsigned char * data, ssize_t dataLen, bool ownData);
bool initWithATITCData(unsigned char *data, ssize_t dataLen, bool ownData);
// fast forward pixels to GPU if ownData
void forwardPixels(unsigned char* data, ssize_t dataLen, int offset, bool ownData);
bool saveImageToPNG(const std::string& filePath, bool isToRGB = true); bool saveImageToPNG(const std::string& filePath, bool isToRGB = true);
bool saveImageToJPG(const std::string& filePath); bool saveImageToJPG(const std::string& filePath);