ATOM-16320 Remove PVRTC and ETC compressor (#4557)
Signed-off-by: Qing Tao <qingtao@amazon.com>monroegm-disable-blank-issue-2
parent
2496bc38b5
commit
e64b9d3536
@ -1,233 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
||||||
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
||||||
*
|
|
||||||
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
#include <Atom/ImageProcessing/ImageObject.h>
|
|
||||||
#include <Processing/PixelFormatInfo.h>
|
|
||||||
#include <Processing/ImageFlags.h>
|
|
||||||
#include <Converters/PixelOperation.h>
|
|
||||||
#include <Compressors/ETC2.h>
|
|
||||||
|
|
||||||
#include <EtcConfig.h>
|
|
||||||
#include <Etc.h>
|
|
||||||
#include <EtcImage.h>
|
|
||||||
#include <EtcColorFloatRGBA.h>
|
|
||||||
|
|
||||||
namespace ImageProcessingAtom
|
|
||||||
{
|
|
||||||
//limited to 1 thread because AP requires so. We may change to n when AP allocate n thread to a job in the furture
|
|
||||||
static const int MAX_COMP_JOBS = 1;
|
|
||||||
static const int MIN_COMP_JOBS = 1;
|
|
||||||
static const float ETC_LOW_EFFORT_LEVEL = 25.0f;
|
|
||||||
static const float ETC_MED_EFFORT_LEVEL = 40.0f;
|
|
||||||
static const float ETC_HIGH_EFFORT_LEVEL = 80.0f;
|
|
||||||
|
|
||||||
//Grab the Etc2Comp specific pixel format enum
|
|
||||||
static Etc::Image::Format FindEtc2PixelFormat(EPixelFormat fmt)
|
|
||||||
{
|
|
||||||
switch (fmt)
|
|
||||||
{
|
|
||||||
case ePixelFormat_EAC_RG11:
|
|
||||||
return Etc::Image::Format::RG11;
|
|
||||||
case ePixelFormat_EAC_R11:
|
|
||||||
return Etc::Image::Format::R11;
|
|
||||||
case ePixelFormat_ETC2:
|
|
||||||
return Etc::Image::Format::RGB8;
|
|
||||||
case ePixelFormat_ETC2a1:
|
|
||||||
return Etc::Image::Format::RGB8A1;
|
|
||||||
case ePixelFormat_ETC2a:
|
|
||||||
return Etc::Image::Format::RGBA8;
|
|
||||||
default:
|
|
||||||
return Etc::Image::Format::FORMATS;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//Get the errmetric required for the compression
|
|
||||||
static Etc::ErrorMetric FindErrMetric(Etc::Image::Format fmt)
|
|
||||||
{
|
|
||||||
switch (fmt)
|
|
||||||
{
|
|
||||||
case Etc::Image::Format::RG11:
|
|
||||||
return Etc::ErrorMetric::NORMALXYZ;
|
|
||||||
case Etc::Image::Format::R11:
|
|
||||||
return Etc::ErrorMetric::NUMERIC;
|
|
||||||
case Etc::Image::Format::RGB8:
|
|
||||||
return Etc::ErrorMetric::RGBX;
|
|
||||||
case Etc::Image::Format::RGBA8:
|
|
||||||
case Etc::Image::Format::RGB8A1:
|
|
||||||
return Etc::ErrorMetric::RGBA;
|
|
||||||
default:
|
|
||||||
return Etc::ErrorMetric::ERROR_METRICS;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//Convert to sRGB format
|
|
||||||
static Etc::Image::Format FindGammaEtc2PixelFormat(Etc::Image::Format fmt)
|
|
||||||
{
|
|
||||||
switch (fmt)
|
|
||||||
{
|
|
||||||
case Etc::Image::Format::RGB8:
|
|
||||||
return Etc::Image::Format::SRGB8;
|
|
||||||
case Etc::Image::Format::RGBA8:
|
|
||||||
return Etc::Image::Format::SRGBA8;
|
|
||||||
case Etc::Image::Format::RGB8A1:
|
|
||||||
return Etc::Image::Format::SRGB8A1;
|
|
||||||
default:
|
|
||||||
return Etc::Image::Format::FORMATS;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ETC2Compressor::IsCompressedPixelFormatSupported(EPixelFormat fmt)
|
|
||||||
{
|
|
||||||
return (FindEtc2PixelFormat(fmt) != Etc::Image::Format::FORMATS);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ETC2Compressor::IsUncompressedPixelFormatSupported(EPixelFormat fmt)
|
|
||||||
{
|
|
||||||
//for uncompress format
|
|
||||||
if (fmt == ePixelFormat_R8G8B8A8)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
EPixelFormat ETC2Compressor::GetSuggestedUncompressedFormat([[maybe_unused]] EPixelFormat compressedfmt, [[maybe_unused]] EPixelFormat uncompressedfmt) const
|
|
||||||
{
|
|
||||||
return ePixelFormat_R8G8B8A8;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ETC2Compressor::DoesSupportDecompress([[maybe_unused]] EPixelFormat fmtDst)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
ColorSpace ETC2Compressor::GetSupportedColorSpace([[maybe_unused]] EPixelFormat compressFormat) const
|
|
||||||
{
|
|
||||||
return ColorSpace::autoSelect;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* ETC2Compressor::GetName() const
|
|
||||||
{
|
|
||||||
return "ETC2Compressor";
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr ETC2Compressor::CompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst,
|
|
||||||
const CompressOption* compressOption) const
|
|
||||||
{
|
|
||||||
//validate input
|
|
||||||
EPixelFormat fmtSrc = srcImage->GetPixelFormat();
|
|
||||||
|
|
||||||
//src format need to be uncompressed and dst format need to compressed.
|
|
||||||
if (!IsUncompressedPixelFormatSupported(fmtSrc) || !IsCompressedPixelFormatSupported(fmtDst))
|
|
||||||
{
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr dstImage(srcImage->AllocateImage(fmtDst));
|
|
||||||
|
|
||||||
//determinate compression quality
|
|
||||||
ICompressor::EQuality quality = ICompressor::eQuality_Normal;
|
|
||||||
//get setting from compression option
|
|
||||||
if (compressOption)
|
|
||||||
{
|
|
||||||
quality = compressOption->compressQuality;
|
|
||||||
}
|
|
||||||
|
|
||||||
float qualityEffort = 0.0f;
|
|
||||||
switch (quality)
|
|
||||||
{
|
|
||||||
case eQuality_Preview:
|
|
||||||
case eQuality_Fast:
|
|
||||||
{
|
|
||||||
qualityEffort = ETC_LOW_EFFORT_LEVEL;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case eQuality_Normal:
|
|
||||||
{
|
|
||||||
qualityEffort = ETC_MED_EFFORT_LEVEL;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
{
|
|
||||||
qualityEffort = ETC_HIGH_EFFORT_LEVEL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Etc::Image::Format dstEtc2Format = FindEtc2PixelFormat(fmtDst);
|
|
||||||
if (srcImage->GetImageFlags() & EIF_SRGBRead)
|
|
||||||
{
|
|
||||||
dstEtc2Format = FindGammaEtc2PixelFormat(dstEtc2Format);
|
|
||||||
}
|
|
||||||
|
|
||||||
//use to read pixel data from src image
|
|
||||||
IPixelOperationPtr pixelOp = CreatePixelOperation(fmtSrc);
|
|
||||||
//get count of bytes per pixel for images
|
|
||||||
AZ::u32 pixelBytes = CPixelFormats::GetInstance().GetPixelFormatInfo(fmtSrc)->bitsPerBlock / 8;
|
|
||||||
|
|
||||||
const AZ::u32 mipCount = dstImage->GetMipCount();
|
|
||||||
for (AZ::u32 mip = 0; mip < mipCount; ++mip)
|
|
||||||
{
|
|
||||||
const AZ::u32 width = srcImage->GetWidth(mip);
|
|
||||||
const AZ::u32 height = srcImage->GetHeight(mip);
|
|
||||||
|
|
||||||
// Prepare source data
|
|
||||||
AZ::u8* srcMem;
|
|
||||||
AZ::u32 srcPitch;
|
|
||||||
srcImage->GetImagePointer(mip, srcMem, srcPitch);
|
|
||||||
const AZ::u32 pixelCount = srcImage->GetPixelCount(mip);
|
|
||||||
|
|
||||||
Etc::ColorFloatRGBA* rgbaPixels = new Etc::ColorFloatRGBA[pixelCount];
|
|
||||||
Etc::ColorFloatRGBA* rgbaPixelPtr = rgbaPixels;
|
|
||||||
float r, g, b, a;
|
|
||||||
for (AZ::u32 pixelIdx = 0; pixelIdx < pixelCount; pixelIdx++, srcMem += pixelBytes, rgbaPixelPtr++)
|
|
||||||
{
|
|
||||||
pixelOp->GetRGBA(srcMem, r, g, b, a);
|
|
||||||
rgbaPixelPtr->fA = a;
|
|
||||||
rgbaPixelPtr->fR = r;
|
|
||||||
rgbaPixelPtr->fG = g;
|
|
||||||
rgbaPixelPtr->fB = b;
|
|
||||||
}
|
|
||||||
|
|
||||||
//Call into etc2Comp lib to compress. https://medium.com/@duhroach/building-a-blazing-fast-etc2-compressor-307f3e9aad99
|
|
||||||
Etc::ErrorMetric errMetric = FindErrMetric(dstEtc2Format);
|
|
||||||
unsigned char* paucEncodingBits;
|
|
||||||
unsigned int uiEncodingBitsBytes;
|
|
||||||
unsigned int uiExtendedWidth;
|
|
||||||
unsigned int uiExtendedHeight;
|
|
||||||
int iEncodingTime_ms;
|
|
||||||
|
|
||||||
Etc::Encode(reinterpret_cast<float*>(rgbaPixels),
|
|
||||||
width, height,
|
|
||||||
dstEtc2Format,
|
|
||||||
errMetric,
|
|
||||||
qualityEffort,
|
|
||||||
MIN_COMP_JOBS,
|
|
||||||
MAX_COMP_JOBS,
|
|
||||||
&paucEncodingBits, &uiEncodingBitsBytes,
|
|
||||||
&uiExtendedWidth, &uiExtendedHeight,
|
|
||||||
&iEncodingTime_ms);
|
|
||||||
|
|
||||||
AZ::u8* dstMem;
|
|
||||||
AZ::u32 dstPitch;
|
|
||||||
dstImage->GetImagePointer(mip, dstMem, dstPitch);
|
|
||||||
|
|
||||||
memcpy(dstMem, paucEncodingBits, uiEncodingBitsBytes);
|
|
||||||
delete[] rgbaPixels;
|
|
||||||
}
|
|
||||||
|
|
||||||
return dstImage;
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr ETC2Compressor::DecompressImage(IImageObjectPtr srcImage, [[maybe_unused]] EPixelFormat fmtDst) const
|
|
||||||
{
|
|
||||||
//etc2Comp doesn't support decompression
|
|
||||||
//Since PVRTexLib support ETC formats too. It may take over the decompression.
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
} // namespace ImageProcessingAtom
|
|
||||||
@ -1,31 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
||||||
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
||||||
*
|
|
||||||
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <Compressors/Compressor.h>
|
|
||||||
|
|
||||||
namespace ImageProcessingAtom
|
|
||||||
{
|
|
||||||
class ETC2Compressor
|
|
||||||
: public ICompressor
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
static bool IsCompressedPixelFormatSupported(EPixelFormat fmt);
|
|
||||||
static bool IsUncompressedPixelFormatSupported(EPixelFormat fmt);
|
|
||||||
static bool DoesSupportDecompress(EPixelFormat fmtDst);
|
|
||||||
|
|
||||||
IImageObjectPtr CompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst, const CompressOption* compressOption) const override;
|
|
||||||
IImageObjectPtr DecompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst) const override;
|
|
||||||
|
|
||||||
EPixelFormat GetSuggestedUncompressedFormat(EPixelFormat compressedfmt, EPixelFormat uncompressedfmt) const override;
|
|
||||||
ColorSpace GetSupportedColorSpace(EPixelFormat compressFormat) const final;
|
|
||||||
const char* GetName() const final;
|
|
||||||
|
|
||||||
};
|
|
||||||
} // namespace ImageProcessingAtom
|
|
||||||
@ -1,338 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
||||||
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
||||||
*
|
|
||||||
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <ImageProcessing_Traits_Platform.h>
|
|
||||||
#include <AzCore/PlatformIncl.h>
|
|
||||||
#include <Atom/ImageProcessing/ImageObject.h>
|
|
||||||
#include <Processing/ImageFlags.h>
|
|
||||||
#include <Processing/PixelFormatInfo.h>
|
|
||||||
#include <Compressors/PVRTC.h>
|
|
||||||
|
|
||||||
#include <PVRTexture.h>
|
|
||||||
#include <PVRTextureUtilities.h>
|
|
||||||
|
|
||||||
|
|
||||||
namespace ImageProcessingAtom
|
|
||||||
{
|
|
||||||
// Note: PVRTexLib supports ETC formats, PVRTC formats and BC formats
|
|
||||||
// We haven't tested the performace to compress BC formats compare to CTSquisher
|
|
||||||
// For PVRTC formats, we only added PVRTC 1 support for now
|
|
||||||
// The compression for ePVRTPF_EAC_R11 and ePVRTPF_EAC_RG11 are very slow. It takes 7 and 14 minutes for a 2048x2048 texture.
|
|
||||||
EPVRTPixelFormat FindPvrPixelFormat(EPixelFormat fmt)
|
|
||||||
{
|
|
||||||
switch (fmt)
|
|
||||||
{
|
|
||||||
case ePixelFormat_PVRTC2:
|
|
||||||
return ePVRTPF_PVRTCI_2bpp_RGBA;
|
|
||||||
case ePixelFormat_PVRTC4:
|
|
||||||
return ePVRTPF_PVRTCI_4bpp_RGBA;
|
|
||||||
case ePixelFormat_EAC_R11:
|
|
||||||
return ePVRTPF_EAC_R11;
|
|
||||||
case ePixelFormat_EAC_RG11:
|
|
||||||
return ePVRTPF_EAC_RG11;
|
|
||||||
case ePixelFormat_ETC2:
|
|
||||||
return ePVRTPF_ETC2_RGB;
|
|
||||||
case ePixelFormat_ETC2a1:
|
|
||||||
return ePVRTPF_ETC2_RGB_A1;
|
|
||||||
case ePixelFormat_ETC2a:
|
|
||||||
return ePVRTPF_ETC2_RGBA;
|
|
||||||
default:
|
|
||||||
return ePVRTPF_NumCompressedPFs;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool PVRTCCompressor::IsCompressedPixelFormatSupported(EPixelFormat fmt)
|
|
||||||
{
|
|
||||||
return (FindPvrPixelFormat(fmt) != ePVRTPF_NumCompressedPFs);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool PVRTCCompressor::IsUncompressedPixelFormatSupported(EPixelFormat fmt)
|
|
||||||
{
|
|
||||||
//for uncompress format
|
|
||||||
if (fmt == ePixelFormat_R8G8B8A8)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
EPixelFormat PVRTCCompressor::GetSuggestedUncompressedFormat([[maybe_unused]] EPixelFormat compressedfmt, [[maybe_unused]] EPixelFormat uncompressedfmt) const
|
|
||||||
{
|
|
||||||
return ePixelFormat_R8G8B8A8;
|
|
||||||
}
|
|
||||||
|
|
||||||
ColorSpace PVRTCCompressor::GetSupportedColorSpace([[maybe_unused]] EPixelFormat compressFormat) const
|
|
||||||
{
|
|
||||||
return ColorSpace::autoSelect;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* PVRTCCompressor::GetName() const
|
|
||||||
{
|
|
||||||
return "PVRTCCompressor";
|
|
||||||
}
|
|
||||||
|
|
||||||
bool PVRTCCompressor::DoesSupportDecompress([[maybe_unused]] EPixelFormat fmtDst)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr PVRTCCompressor::CompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst,
|
|
||||||
const CompressOption* compressOption) const
|
|
||||||
{
|
|
||||||
//validate input
|
|
||||||
EPixelFormat fmtSrc = srcImage->GetPixelFormat();
|
|
||||||
|
|
||||||
//src format need to be uncompressed and dst format need to compressed.
|
|
||||||
if (!IsUncompressedPixelFormatSupported(fmtSrc) || !IsCompressedPixelFormatSupported(fmtDst))
|
|
||||||
{
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr dstImage(srcImage->AllocateImage(fmtDst));
|
|
||||||
|
|
||||||
//determinate compression quality
|
|
||||||
pvrtexture::ECompressorQuality internalQuality = pvrtexture::eETCFast;
|
|
||||||
ICompressor::EQuality quality = ICompressor::eQuality_Normal;
|
|
||||||
AZ::Vector3 uniformWeights = AZ::Vector3(0.3333f, 0.3334f, 0.3333f);
|
|
||||||
AZ::Vector3 weights = uniformWeights;
|
|
||||||
bool isUniform = true;
|
|
||||||
//get setting from compression option
|
|
||||||
if (compressOption)
|
|
||||||
{
|
|
||||||
quality = compressOption->compressQuality;
|
|
||||||
weights = compressOption->rgbWeight;
|
|
||||||
isUniform = (weights == uniformWeights);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (IsETCFormat(fmtDst))
|
|
||||||
{
|
|
||||||
if ((quality <= eQuality_Normal) && isUniform)
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::eETCFast;
|
|
||||||
}
|
|
||||||
else if (quality <= eQuality_Normal)
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::eETCNormal;
|
|
||||||
}
|
|
||||||
else if (isUniform)
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::eETCSlow;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::eETCSlow;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (quality == eQuality_Preview)
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::ePVRTCFastest;
|
|
||||||
}
|
|
||||||
else if (quality == eQuality_Fast)
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::ePVRTCFast;
|
|
||||||
}
|
|
||||||
else if (quality == eQuality_Normal)
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::ePVRTCNormal;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
internalQuality = pvrtexture::ePVRTCHigh;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// setup color space
|
|
||||||
EPVRTColourSpace cspace = ePVRTCSpacelRGB;
|
|
||||||
if (srcImage->GetImageFlags() & EIF_SRGBRead)
|
|
||||||
{
|
|
||||||
cspace = ePVRTCSpacesRGB;
|
|
||||||
}
|
|
||||||
|
|
||||||
//setup src texture for compression
|
|
||||||
const pvrtexture::PixelType srcPixelType('r', 'g', 'b', 'a', 8, 8, 8, 8);
|
|
||||||
const AZ::u32 dstMips = dstImage->GetMipCount();
|
|
||||||
for (AZ::u32 mip = 0; mip < dstMips; ++mip)
|
|
||||||
{
|
|
||||||
const AZ::u32 width = srcImage->GetWidth(mip);
|
|
||||||
const AZ::u32 height = srcImage->GetHeight(mip);
|
|
||||||
|
|
||||||
// Prepare source data
|
|
||||||
AZ::u8* srcMem;
|
|
||||||
uint32 srcPitch;
|
|
||||||
srcImage->GetImagePointer(mip, srcMem, srcPitch);
|
|
||||||
|
|
||||||
const pvrtexture::CPVRTextureHeader srcHeader(
|
|
||||||
srcPixelType.PixelTypeID, // AZ::u64 u64PixelFormat,
|
|
||||||
width, // uint32 u32Height=1,
|
|
||||||
height, // uint32 u32Width=1,
|
|
||||||
1, // uint32 u32Depth=1,
|
|
||||||
1, // uint32 u32NumMipMaps=1,
|
|
||||||
1, // uint32 u32NumArrayMembers=1,
|
|
||||||
1, // uint32 u32NumFaces=1,
|
|
||||||
cspace, // EPVRTColourSpace eColourSpace=ePVRTCSpacelRGB,
|
|
||||||
ePVRTVarTypeUnsignedByteNorm, // EPVRTVariableType eChannelType=ePVRTVarTypeUnsignedByteNorm,
|
|
||||||
false); // bool bPreMultiplied=false);
|
|
||||||
|
|
||||||
pvrtexture::CPVRTexture compressTexture(srcHeader, srcMem);
|
|
||||||
|
|
||||||
//compressing
|
|
||||||
bool isSuccess = false;
|
|
||||||
#if AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
try
|
|
||||||
#endif // AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
{
|
|
||||||
isSuccess = pvrtexture::Transcode(
|
|
||||||
compressTexture,
|
|
||||||
pvrtexture::PixelType(FindPvrPixelFormat(fmtDst)),
|
|
||||||
ePVRTVarTypeUnsignedByteNorm,
|
|
||||||
cspace,
|
|
||||||
internalQuality);
|
|
||||||
}
|
|
||||||
#if AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
catch (...)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Unknown exception in PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
#endif // AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
|
|
||||||
if (!isSuccess)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Failed to compress image with PVRTexLib.");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Getting compressed data
|
|
||||||
const void* const compressedData = compressTexture.getDataPtr();
|
|
||||||
if (!compressedData)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Failed to obtain compressed image data by using PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
const AZ::u32 compressedDataSize = compressTexture.getDataSize();
|
|
||||||
if (dstImage->GetMipBufSize(mip) != compressedDataSize)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Compressed image data size mismatch while using PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
//save compressed data to dst image
|
|
||||||
AZ::u8* dstMem;
|
|
||||||
AZ::u32 dstPitch;
|
|
||||||
dstImage->GetImagePointer(mip, dstMem, dstPitch);
|
|
||||||
memcpy(dstMem, compressedData, compressedDataSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
return dstImage;
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr PVRTCCompressor::DecompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst) const
|
|
||||||
{
|
|
||||||
//validate input
|
|
||||||
EPixelFormat fmtSrc = srcImage->GetPixelFormat(); //compressed
|
|
||||||
|
|
||||||
if (!IsCompressedPixelFormatSupported(fmtSrc) || !IsUncompressedPixelFormatSupported(fmtDst))
|
|
||||||
{
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
EPVRTColourSpace colorSpace = ePVRTCSpacelRGB;
|
|
||||||
if (srcImage->GetImageFlags() & EIF_SRGBRead)
|
|
||||||
{
|
|
||||||
colorSpace = ePVRTCSpacesRGB;
|
|
||||||
}
|
|
||||||
|
|
||||||
IImageObjectPtr dstImage(srcImage->AllocateImage(fmtDst));
|
|
||||||
|
|
||||||
const AZ::u32 mipCount = dstImage->GetMipCount();
|
|
||||||
for (AZ::u32 mip = 0; mip < mipCount; ++mip)
|
|
||||||
{
|
|
||||||
const AZ::u32 width = srcImage->GetWidth(mip);
|
|
||||||
const AZ::u32 height = srcImage->GetHeight(mip);
|
|
||||||
|
|
||||||
// Preparing source compressed data
|
|
||||||
const pvrtexture::CPVRTextureHeader compressedHeader(
|
|
||||||
FindPvrPixelFormat(fmtSrc), // AZ::u64 u64PixelFormat,
|
|
||||||
width, // uint32 u32Height=1,
|
|
||||||
height, // uint32 u32Width=1,
|
|
||||||
1, // uint32 u32Depth=1,
|
|
||||||
1, // uint32 u32NumMipMaps=1,
|
|
||||||
1, // uint32 u32NumArrayMembers=1,
|
|
||||||
1, // uint32 u32NumFaces=1,
|
|
||||||
colorSpace, // EPVRTColourSpace eColourSpace=ePVRTCSpacelRGB,
|
|
||||||
ePVRTVarTypeUnsignedByteNorm, // EPVRTVariableType eChannelType=ePVRTVarTypeUnsignedByteNorm,
|
|
||||||
false); // bool bPreMultiplied=false);
|
|
||||||
|
|
||||||
const AZ::u32 compressedDataSize = compressedHeader.getDataSize();
|
|
||||||
if (srcImage->GetMipBufSize(mip) != compressedDataSize)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Decompressed image data size mismatch while using PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
AZ::u8* srcMem;
|
|
||||||
AZ::u32 srcPitch;
|
|
||||||
srcImage->GetImagePointer(mip, srcMem, srcPitch);
|
|
||||||
pvrtexture::CPVRTexture cTexture(compressedHeader, srcMem);
|
|
||||||
|
|
||||||
// Decompress
|
|
||||||
bool bOk = false;
|
|
||||||
#if AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
try
|
|
||||||
{
|
|
||||||
#endif // AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
bOk = pvrtexture::Transcode(
|
|
||||||
cTexture,
|
|
||||||
pvrtexture::PVRStandard8PixelType,
|
|
||||||
ePVRTVarTypeUnsignedByteNorm,
|
|
||||||
colorSpace,
|
|
||||||
pvrtexture::ePVRTCHigh);
|
|
||||||
|
|
||||||
#if AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
}
|
|
||||||
catch (...)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Unknown exception in PVRTexLib when decompressing");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
#endif // AZ_TRAIT_IMAGEPROCESSING_SUPPORT_TRY_CATCH
|
|
||||||
|
|
||||||
if (!bOk)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Failed to decompress an image by using PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Getting decompressed data
|
|
||||||
const void* const pDecompressedData = cTexture.getDataPtr();
|
|
||||||
if (!pDecompressedData)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Failed to obtain decompressed image data by using PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
const AZ::u32 decompressedDataSize = cTexture.getDataSize();
|
|
||||||
if (dstImage->GetMipBufSize(mip) != decompressedDataSize)
|
|
||||||
{
|
|
||||||
AZ_Error("Image Processing", false, "Decompressed image data size mismatch while using PVRTexLib");
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
//save decompressed image to dst image
|
|
||||||
AZ::u8* dstMem;
|
|
||||||
AZ::u32 dstPitch;
|
|
||||||
dstImage->GetImagePointer(mip, dstMem, dstPitch);
|
|
||||||
memcpy(dstMem, pDecompressedData, decompressedDataSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
return dstImage;
|
|
||||||
}
|
|
||||||
} //namespace ImageProcessingAtom
|
|
||||||
@ -1,30 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
||||||
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
||||||
*
|
|
||||||
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <Compressors/Compressor.h>
|
|
||||||
|
|
||||||
namespace ImageProcessingAtom
|
|
||||||
{
|
|
||||||
class PVRTCCompressor
|
|
||||||
: public ICompressor
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
static bool IsCompressedPixelFormatSupported(EPixelFormat fmt);
|
|
||||||
static bool IsUncompressedPixelFormatSupported(EPixelFormat fmt);
|
|
||||||
static bool DoesSupportDecompress(EPixelFormat fmtDst);
|
|
||||||
|
|
||||||
IImageObjectPtr CompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst, const CompressOption* compressOption) const override;
|
|
||||||
IImageObjectPtr DecompressImage(IImageObjectPtr srcImage, EPixelFormat fmtDst) const override;
|
|
||||||
|
|
||||||
EPixelFormat GetSuggestedUncompressedFormat(EPixelFormat compressedfmt, EPixelFormat uncompressedfmt) const override;
|
|
||||||
ColorSpace GetSupportedColorSpace(EPixelFormat compressFormat) const final;
|
|
||||||
const char* GetName() const final;
|
|
||||||
};
|
|
||||||
} // namespace ImageProcessingAtom
|
|
||||||
Loading…
Reference in New Issue