2020-07-31 11:31:32 +08:00
|
|
|
//===============================================================================
|
|
|
|
// Copyright (c) 2007-2016 Advanced Micro Devices, Inc. All rights reserved.
|
|
|
|
// Copyright (c) 2004-2006 ATI Technologies Inc.
|
|
|
|
//===============================================================================
|
|
|
|
//
|
|
|
|
// 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 :
|
2021-09-08 10:54:22 +08:00
|
|
|
//
|
2020-07-31 11:31:32 +08:00
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
2021-09-08 10:54:22 +08:00
|
|
|
//
|
2020-07-31 11:31:32 +08:00
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// File Name: Codec_DXT1.cpp
|
|
|
|
// Description: implementation of the CCodec_DXT1 class
|
|
|
|
//
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
#pragma warning(disable:4100)
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
#include "common.h"
|
|
|
|
#include "compressonator.h"
|
|
|
|
#include "codec_dxt1.h"
|
2020-07-31 11:31:32 +08:00
|
|
|
|
|
|
|
#ifdef TEST_CMP_CORE_DECODER
|
2021-09-08 10:54:22 +08:00
|
|
|
#include "cmp_core.h"
|
2020-07-31 11:31:32 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Construction/Destruction
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
CCodec_DXT1::CCodec_DXT1() :
|
2021-09-08 10:54:22 +08:00
|
|
|
CCodec_DXTC(CT_DXT1) {
|
2020-07-31 11:31:32 +08:00
|
|
|
m_bDXT1UseAlpha = false;
|
2021-09-08 10:54:22 +08:00
|
|
|
m_nAlphaThreshold = 0;
|
2020-07-31 11:31:32 +08:00
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
CCodec_DXT1::~CCodec_DXT1() {
|
2020-07-31 11:31:32 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
bool CCodec_DXT1::SetParameter(const CMP_CHAR* pszParamName, CMP_CHAR* sValue) {
|
|
|
|
if(strcmp(pszParamName, "DXT1UseAlpha") == 0) {
|
2020-07-31 11:31:32 +08:00
|
|
|
m_bDXT1UseAlpha = (std::stoi(sValue) > 0) ? true : false;
|
2021-09-08 10:54:22 +08:00
|
|
|
m_BC15Options.m_bUseAlpha = m_bDXT1UseAlpha;
|
|
|
|
}
|
|
|
|
else if(strcmp(pszParamName, "AlphaThreshold") == 0) {
|
2020-07-31 11:31:32 +08:00
|
|
|
m_nAlphaThreshold = (CMP_BYTE) (std::stoi(sValue) & 0xFF);
|
2021-09-08 10:54:22 +08:00
|
|
|
m_BC15Options.m_nAlphaThreshold = m_nAlphaThreshold;
|
|
|
|
}
|
|
|
|
else
|
2020-07-31 11:31:32 +08:00
|
|
|
return CCodec_DXTC::SetParameter(pszParamName, sValue);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
bool CCodec_DXT1::SetParameter(const CMP_CHAR* pszParamName, CMP_DWORD dwValue) {
|
|
|
|
if(strcmp(pszParamName, "DXT1UseAlpha") == 0) {
|
2020-07-31 11:31:32 +08:00
|
|
|
m_bDXT1UseAlpha = dwValue ? true : false;
|
2021-09-08 10:54:22 +08:00
|
|
|
m_BC15Options.m_bUseAlpha = m_bDXT1UseAlpha;
|
|
|
|
}
|
|
|
|
else if(strcmp(pszParamName, "AlphaThreshold") == 0) {
|
2020-07-31 11:31:32 +08:00
|
|
|
m_nAlphaThreshold = (CMP_BYTE) dwValue;
|
2021-09-08 10:54:22 +08:00
|
|
|
m_BC15Options.m_nAlphaThreshold = m_nAlphaThreshold;
|
|
|
|
}
|
2020-07-31 11:31:32 +08:00
|
|
|
else
|
|
|
|
return CCodec_DXTC::SetParameter(pszParamName, dwValue);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
bool CCodec_DXT1::GetParameter(const CMP_CHAR* pszParamName, CMP_DWORD& dwValue) {
|
|
|
|
if(strcmp(pszParamName, "DXT1UseAlpha") == 0) {
|
|
|
|
dwValue = m_BC15Options.m_bUseAlpha;
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(strcmp(pszParamName, "AlphaThreshold") == 0) {
|
|
|
|
dwValue = m_BC15Options.m_nAlphaThreshold;
|
|
|
|
}
|
2020-07-31 11:31:32 +08:00
|
|
|
else
|
|
|
|
return CCodec_DXTC::SetParameter(pszParamName, dwValue);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
CCodecBuffer* CCodec_DXT1::CreateBuffer(
|
2021-09-08 10:54:22 +08:00
|
|
|
CMP_BYTE nBlockWidth, CMP_BYTE nBlockHeight, CMP_BYTE nBlockDepth,
|
|
|
|
CMP_DWORD dwWidth, CMP_DWORD dwHeight, CMP_DWORD dwPitch, CMP_BYTE* pData,CMP_DWORD dwDataSize) const {
|
2020-07-31 11:31:32 +08:00
|
|
|
return CreateCodecBuffer(CBT_4x4Block_4BPP, nBlockWidth, nBlockHeight, nBlockDepth, dwWidth, dwHeight, dwPitch, pData,dwDataSize);
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
CodecError CCodec_DXT1::Compress(CCodecBuffer& bufferIn, CCodecBuffer& bufferOut, Codec_Feedback_Proc pFeedbackProc, CMP_DWORD_PTR pUser1, CMP_DWORD_PTR pUser2) {
|
2020-07-31 11:31:32 +08:00
|
|
|
assert(bufferIn.GetWidth() == bufferOut.GetWidth());
|
|
|
|
assert(bufferIn.GetHeight() == bufferOut.GetHeight());
|
|
|
|
|
|
|
|
if(bufferIn.GetWidth() != bufferOut.GetWidth() || bufferIn.GetHeight() != bufferOut.GetHeight())
|
|
|
|
return CE_Unknown;
|
|
|
|
|
|
|
|
const CMP_DWORD dwBlocksX = ((bufferIn.GetWidth() + 3) >> 2);
|
|
|
|
const CMP_DWORD dwBlocksY = ((bufferIn.GetHeight() + 3) >> 2);
|
|
|
|
|
|
|
|
bool bUseFixed = (!bufferIn.IsFloat() && bufferIn.GetChannelDepth() == 8 && !m_bUseFloat);
|
|
|
|
|
|
|
|
float fAlphaThreshold = CONVERT_BYTE_TO_FLOAT(m_nAlphaThreshold);
|
2021-09-08 10:54:22 +08:00
|
|
|
for(CMP_DWORD j = 0; j < dwBlocksY; j++) {
|
|
|
|
for(CMP_DWORD i = 0; i < dwBlocksX; i++) {
|
2020-07-31 11:31:32 +08:00
|
|
|
CMP_DWORD compressedBlock[2];
|
2021-09-08 10:54:22 +08:00
|
|
|
if(bUseFixed) {
|
2020-07-31 11:31:32 +08:00
|
|
|
CMP_BYTE srcBlock[BLOCK_SIZE_4X4X4];
|
|
|
|
bufferIn.ReadBlockRGBA(i*4, j*4, 4, 4, srcBlock);
|
|
|
|
CompressRGBBlock(srcBlock, compressedBlock, CalculateColourWeightings(srcBlock), true, m_bDXT1UseAlpha, m_nAlphaThreshold);
|
2021-09-08 10:54:22 +08:00
|
|
|
} else {
|
2020-07-31 11:31:32 +08:00
|
|
|
float srcBlock[BLOCK_SIZE_4X4X4];
|
|
|
|
bufferIn.ReadBlockRGBA(i*4, j*4, 4, 4, srcBlock);
|
|
|
|
CompressRGBBlock(srcBlock, compressedBlock, CalculateColourWeightings(srcBlock), true, m_bDXT1UseAlpha, fAlphaThreshold);
|
|
|
|
}
|
|
|
|
bufferOut.WriteBlock(i*4, j*4, compressedBlock, 2);
|
|
|
|
}
|
2021-09-08 10:54:22 +08:00
|
|
|
if(pFeedbackProc) {
|
2020-07-31 11:31:32 +08:00
|
|
|
float fProgress = 100.f * (j * dwBlocksX) / (dwBlocksX * dwBlocksY);
|
|
|
|
if(pFeedbackProc(fProgress, pUser1, pUser2))
|
|
|
|
return CE_Aborted;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CE_OK;
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
CodecError CCodec_DXT1::Compress_Fast(CCodecBuffer& bufferIn, CCodecBuffer& bufferOut, Codec_Feedback_Proc pFeedbackProc, CMP_DWORD_PTR pUser1, CMP_DWORD_PTR pUser2) {
|
2020-07-31 11:31:32 +08:00
|
|
|
assert(bufferIn.GetWidth() == bufferOut.GetWidth());
|
|
|
|
assert(bufferIn.GetHeight() == bufferOut.GetHeight());
|
|
|
|
|
|
|
|
if(bufferIn.GetWidth() != bufferOut.GetWidth() || bufferIn.GetHeight() != bufferOut.GetHeight())
|
|
|
|
return CE_Unknown;
|
|
|
|
|
|
|
|
const CMP_DWORD dwBlocksX = ((bufferIn.GetWidth() + 3) >> 2);
|
|
|
|
const CMP_DWORD dwBlocksY = ((bufferIn.GetHeight() + 3) >> 2);
|
|
|
|
|
|
|
|
CMP_DWORD compressedBlock[2];
|
|
|
|
CMP_BYTE srcBlock[BLOCK_SIZE_4X4X4];
|
2021-09-08 10:54:22 +08:00
|
|
|
for(CMP_DWORD j = 0; j < dwBlocksY; j++) {
|
|
|
|
for(CMP_DWORD i = 0; i < dwBlocksX; i++) {
|
2020-07-31 11:31:32 +08:00
|
|
|
bufferIn.ReadBlockRGBA(i*4, j*4, 4, 4, srcBlock);
|
|
|
|
CompressRGBBlock_Fast(srcBlock, compressedBlock);
|
|
|
|
bufferOut.WriteBlock(i*4, j*4, compressedBlock, 2);
|
|
|
|
}
|
2021-09-08 10:54:22 +08:00
|
|
|
if(pFeedbackProc) {
|
2020-07-31 11:31:32 +08:00
|
|
|
float fProgress = 100.f * (j * dwBlocksX) / (dwBlocksX * dwBlocksY);
|
|
|
|
if(pFeedbackProc(fProgress, pUser1, pUser2))
|
|
|
|
return CE_Aborted;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return CE_OK;
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
CodecError CCodec_DXT1::Compress_SuperFast(CCodecBuffer& bufferIn, CCodecBuffer& bufferOut, Codec_Feedback_Proc pFeedbackProc, CMP_DWORD_PTR pUser1, CMP_DWORD_PTR pUser2) {
|
2020-07-31 11:31:32 +08:00
|
|
|
assert(bufferIn.GetWidth() == bufferOut.GetWidth());
|
|
|
|
assert(bufferIn.GetHeight() == bufferOut.GetHeight());
|
|
|
|
|
|
|
|
if(bufferIn.GetWidth() != bufferOut.GetWidth() || bufferIn.GetHeight() != bufferOut.GetHeight())
|
|
|
|
return CE_Unknown;
|
|
|
|
|
|
|
|
const CMP_DWORD dwBlocksX = ((bufferIn.GetWidth() + 3) >> 2);
|
|
|
|
const CMP_DWORD dwBlocksY = ((bufferIn.GetHeight() + 3) >> 2);
|
|
|
|
|
|
|
|
CMP_DWORD compressedBlock[2];
|
|
|
|
CMP_BYTE srcBlock[BLOCK_SIZE_4X4X4];
|
2021-09-08 10:54:22 +08:00
|
|
|
for(CMP_DWORD j = 0; j < dwBlocksY; j++) {
|
|
|
|
for(CMP_DWORD i = 0; i < dwBlocksX; i++) {
|
2020-07-31 11:31:32 +08:00
|
|
|
bufferIn.ReadBlockRGBA(i*4, j*4, 4, 4, srcBlock);
|
|
|
|
CompressRGBBlock_SuperFast(srcBlock, compressedBlock);
|
|
|
|
bufferOut.WriteBlock(i*4, j*4, compressedBlock, 2);
|
|
|
|
}
|
2021-09-08 10:54:22 +08:00
|
|
|
if(pFeedbackProc) {
|
2020-07-31 11:31:32 +08:00
|
|
|
float fProgress = 100.f * (j * dwBlocksX) / (dwBlocksX * dwBlocksY);
|
|
|
|
if(pFeedbackProc(fProgress, pUser1, pUser2))
|
|
|
|
return CE_Aborted;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return CE_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
CodecError CCodec_DXT1::Decompress(CCodecBuffer& bufferIn, CCodecBuffer& bufferOut, Codec_Feedback_Proc pFeedbackProc, CMP_DWORD_PTR pUser1, CMP_DWORD_PTR pUser2) {
|
2020-07-31 11:31:32 +08:00
|
|
|
assert(bufferIn.GetWidth() == bufferOut.GetWidth());
|
|
|
|
assert(bufferIn.GetHeight() == bufferOut.GetHeight());
|
|
|
|
|
|
|
|
if(bufferIn.GetWidth() != bufferOut.GetWidth() || bufferIn.GetHeight() != bufferOut.GetHeight())
|
|
|
|
return CE_Unknown;
|
|
|
|
|
|
|
|
const CMP_DWORD dwBlocksX = ((bufferIn.GetWidth() + 3) >> 2);
|
|
|
|
const CMP_DWORD dwBlocksY = ((bufferIn.GetHeight() + 3) >> 2);
|
|
|
|
const CMP_DWORD dwBlocksXY = dwBlocksX*dwBlocksY;
|
|
|
|
|
|
|
|
bool bUseFixed = (!bufferOut.IsFloat() && bufferOut.GetChannelDepth() == 8 && !m_bUseFloat);
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
for(CMP_DWORD j = 0; j < dwBlocksY; j++) {
|
|
|
|
for(CMP_DWORD i = 0; i < dwBlocksX; i++) {
|
2020-07-31 11:31:32 +08:00
|
|
|
CMP_DWORD compressedBlock[2];
|
|
|
|
bufferIn.ReadBlock(i*4, j*4, compressedBlock, 2);
|
2021-09-08 10:54:22 +08:00
|
|
|
if(bUseFixed) {
|
2020-07-31 11:31:32 +08:00
|
|
|
CMP_BYTE destBlock[BLOCK_SIZE_4X4X4];
|
2021-09-08 10:54:22 +08:00
|
|
|
#ifdef TEST_CMP_CORE_DECODER
|
|
|
|
DecompressBlockBC1((CMP_BYTE *)compressedBlock,destBlock);
|
|
|
|
#else
|
|
|
|
DecompressRGBBlock(destBlock, compressedBlock, true);
|
|
|
|
#endif
|
2020-07-31 11:31:32 +08:00
|
|
|
bufferOut.WriteBlockRGBA(i*4, j*4, 4, 4, destBlock);
|
2021-09-08 10:54:22 +08:00
|
|
|
} else {
|
2020-07-31 11:31:32 +08:00
|
|
|
float destBlock[BLOCK_SIZE_4X4X4];
|
|
|
|
DecompressRGBBlock(destBlock, compressedBlock, true);
|
|
|
|
bufferOut.WriteBlockRGBA(i*4, j*4, 4, 4, destBlock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-08 10:54:22 +08:00
|
|
|
if (pFeedbackProc) {
|
2020-07-31 11:31:32 +08:00
|
|
|
float fProgress = 100.f * (j * dwBlocksX) / dwBlocksXY;
|
2021-09-08 10:54:22 +08:00
|
|
|
if (pFeedbackProc(fProgress, pUser1, pUser2)) {
|
2020-07-31 11:31:32 +08:00
|
|
|
return CE_Aborted;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return CE_OK;
|
|
|
|
}
|