| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390 | /* * Copyright 2015 The Etc2Comp Authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * *  http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */#ifdef _WIN32#define _CRT_SECURE_NO_WARNINGS (1)#endif#include "EtcConfig.h"#include "EtcFile.h"#include "EtcFileHeader.h"#include "EtcColor.h"#include "Etc.h"#include "EtcBlock4x4EncodingBits.h"#include <stdio.h>#include <string.h>#include <assert.h>#include <stdlib.h>using namespace Etc;// ----------------------------------------------------------------------------------------------------//File::File(const char *a_pstrFilename, Format a_fileformat, Image::Format a_imageformat,			unsigned char *a_paucEncodingBits, unsigned int a_uiEncodingBitsBytes,			unsigned int a_uiSourceWidth, unsigned int a_uiSourceHeight,			unsigned int a_uiExtendedWidth, unsigned int a_uiExtendedHeight){	if (a_pstrFilename == nullptr)	{		m_pstrFilename = const_cast<char *>("");	}	else	{		m_pstrFilename = new char[strlen(a_pstrFilename) + 1];		strcpy(m_pstrFilename, a_pstrFilename);	}	m_fileformat = a_fileformat;	if (m_fileformat == Format::INFER_FROM_FILE_EXTENSION)	{		// ***** TODO: add this later *****		m_fileformat = Format::KTX;	}	m_imageformat = a_imageformat;	m_uiNumMipmaps = 1;	m_pMipmapImages = new RawImage[m_uiNumMipmaps];	m_pMipmapImages[0].paucEncodingBits = std::shared_ptr<unsigned char>(a_paucEncodingBits, [](unsigned char *p) { delete[] p; } );	m_pMipmapImages[0].uiEncodingBitsBytes = a_uiEncodingBitsBytes;	m_pMipmapImages[0].uiExtendedWidth = a_uiExtendedWidth;	m_pMipmapImages[0].uiExtendedHeight = a_uiExtendedHeight;	m_uiSourceWidth = a_uiSourceWidth;	m_uiSourceHeight = a_uiSourceHeight;	switch (m_fileformat)	{	case Format::PKM:		m_pheader = new FileHeader_Pkm(this);		break;	case Format::KTX:		m_pheader = new FileHeader_Ktx(this);		break;	default:		assert(0);		break;	}}// ----------------------------------------------------------------------------------------------------//File::File(const char *a_pstrFilename, Format a_fileformat, Image::Format a_imageformat,	unsigned int a_uiNumMipmaps, RawImage *a_pMipmapImages,	unsigned int a_uiSourceWidth, unsigned int a_uiSourceHeight){	if (a_pstrFilename == nullptr)	{		m_pstrFilename = const_cast<char *>("");	}	else	{		m_pstrFilename = new char[strlen(a_pstrFilename) + 1];		strcpy(m_pstrFilename, a_pstrFilename);	}	m_fileformat = a_fileformat;	if (m_fileformat == Format::INFER_FROM_FILE_EXTENSION)	{		// ***** TODO: add this later *****		m_fileformat = Format::KTX;	}	m_imageformat = a_imageformat;	m_uiNumMipmaps = a_uiNumMipmaps;	m_pMipmapImages = new RawImage[m_uiNumMipmaps];	for(unsigned int mip = 0; mip < m_uiNumMipmaps; mip++)	{		m_pMipmapImages[mip] = a_pMipmapImages[mip];	}	m_uiSourceWidth = a_uiSourceWidth;	m_uiSourceHeight = a_uiSourceHeight;	switch (m_fileformat)	{	case Format::PKM:		m_pheader = new FileHeader_Pkm(this);		break;	case Format::KTX:		m_pheader = new FileHeader_Ktx(this);		break;	default:		assert(0);		break;	}}// ----------------------------------------------------------------------------------------------------//File::File(const char *a_pstrFilename, Format a_fileformat){	if (a_pstrFilename == nullptr)	{		return;	}	else	{		m_pstrFilename = new char[strlen(a_pstrFilename) + 1];		strcpy(m_pstrFilename, a_pstrFilename);	}	m_fileformat = a_fileformat;	if (m_fileformat == Format::INFER_FROM_FILE_EXTENSION)	{		// ***** TODO: add this later *****		m_fileformat = Format::KTX;	}	FILE *pfile = fopen(m_pstrFilename, "rb");	if (pfile == nullptr)	{		printf("ERROR: Couldn't open %s", m_pstrFilename);		exit(1);	}	fseek(pfile, 0, SEEK_END);	unsigned int fileSize = ftell(pfile);	fseek(pfile, 0, SEEK_SET);	size_t szResult;	m_pheader = new FileHeader_Ktx(this);	szResult = fread( ((FileHeader_Ktx*)m_pheader)->GetData(), 1, sizeof(FileHeader_Ktx::Data), pfile);	assert(szResult > 0);	m_uiNumMipmaps = 1;	m_pMipmapImages = new RawImage[m_uiNumMipmaps];	if (((FileHeader_Ktx*)m_pheader)->GetData()->m_u32BytesOfKeyValueData > 0)		fseek(pfile, ((FileHeader_Ktx*)m_pheader)->GetData()->m_u32BytesOfKeyValueData, SEEK_CUR);	szResult = fread(&m_pMipmapImages->uiEncodingBitsBytes, 1, sizeof(unsigned int), pfile);	assert(szResult > 0);	m_pMipmapImages->paucEncodingBits = std::shared_ptr<unsigned char>(new unsigned char[m_pMipmapImages->uiEncodingBitsBytes], [](unsigned char *p) { delete[] p; } );	assert(ftell(pfile) + m_pMipmapImages->uiEncodingBitsBytes <= fileSize);	szResult = fread(m_pMipmapImages->paucEncodingBits.get(), 1, m_pMipmapImages->uiEncodingBitsBytes, pfile);	assert(szResult == m_pMipmapImages->uiEncodingBitsBytes);	uint32_t uiInternalFormat = ((FileHeader_Ktx*)m_pheader)->GetData()->m_u32GlInternalFormat;	uint32_t uiBaseInternalFormat = ((FileHeader_Ktx*)m_pheader)->GetData()->m_u32GlBaseInternalFormat;		if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC1_RGB8 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC1_RGB8)	{		m_imageformat = Image::Format::ETC1;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_RGB8 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_RGB8)	{		m_imageformat = Image::Format::RGB8;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_RGB8A1 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_RGB8A1)	{		m_imageformat = Image::Format::RGB8A1;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_RGBA8 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_RGBA8)	{		m_imageformat = Image::Format::RGBA8;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_R11 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_R11)	{		m_imageformat = Image::Format::R11;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_SIGNED_R11 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_R11)	{		m_imageformat = Image::Format::SIGNED_R11;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_RG11 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_RG11)	{		m_imageformat = Image::Format::RG11;	}	else if (uiInternalFormat == (uint32_t)FileHeader_Ktx::InternalFormat::ETC2_SIGNED_RG11 && uiBaseInternalFormat == (uint32_t)FileHeader_Ktx::BaseInternalFormat::ETC2_RG11)	{		m_imageformat = Image::Format::SIGNED_RG11;	}	else	{		m_imageformat = Image::Format::UNKNOWN;	}	m_uiSourceWidth = ((FileHeader_Ktx*)m_pheader)->GetData()->m_u32PixelWidth;	m_uiSourceHeight = ((FileHeader_Ktx*)m_pheader)->GetData()->m_u32PixelHeight;	m_pMipmapImages->uiExtendedWidth = Image::CalcExtendedDimension((unsigned short)m_uiSourceWidth);	m_pMipmapImages->uiExtendedHeight = Image::CalcExtendedDimension((unsigned short)m_uiSourceHeight);	unsigned int uiBlocks = m_pMipmapImages->uiExtendedWidth * m_pMipmapImages->uiExtendedHeight / 16;	Block4x4EncodingBits::Format encodingbitsformat = Image::DetermineEncodingBitsFormat(m_imageformat);	unsigned int expectedbytes = uiBlocks * Block4x4EncodingBits::GetBytesPerBlock(encodingbitsformat);	assert(expectedbytes == m_pMipmapImages->uiEncodingBitsBytes);	fclose(pfile);}File::~File(){	if (m_pMipmapImages != nullptr)	{		delete [] m_pMipmapImages;	}	if(m_pstrFilename != nullptr)	{		delete[] m_pstrFilename;		m_pstrFilename = nullptr;	}	if (m_pheader != nullptr)	{		delete m_pheader;		m_pheader = nullptr;	}}void File::UseSingleBlock(int a_iPixelX, int a_iPixelY){	if (a_iPixelX <= -1 || a_iPixelY <= -1)		return;	if (a_iPixelX >(int) m_uiSourceWidth)	{		//if we are using a ktx thats the size of a single block or less		//then make sure we use the 4x4 image as the single block		if (m_uiSourceWidth <= 4)		{			a_iPixelX = 0;		}		else		{			printf("blockAtHV: H coordinate out of range, capped to image width\n");			a_iPixelX = m_uiSourceWidth - 1;		}	}	if (a_iPixelY >(int) m_uiSourceHeight)	{		//if we are using a ktx thats the size of a single block or less		//then make sure we use the 4x4 image as the single block		if (m_uiSourceHeight <= 4)		{			a_iPixelY= 0;		}		else		{			printf("blockAtHV: V coordinate out of range, capped to image height\n");			a_iPixelY = m_uiSourceHeight - 1;		}	}	unsigned int origWidth = m_uiSourceWidth;	unsigned int origHeight = m_uiSourceHeight;	m_uiSourceWidth = 4;	m_uiSourceHeight = 4;	Block4x4EncodingBits::Format encodingbitsformat = Image::DetermineEncodingBitsFormat(m_imageformat);	unsigned int uiEncodingBitsBytesPerBlock = Block4x4EncodingBits::GetBytesPerBlock(encodingbitsformat);	int numMipmaps = 1;	RawImage* pMipmapImages = new RawImage[numMipmaps];	pMipmapImages[0].uiExtendedWidth = Image::CalcExtendedDimension((unsigned short)m_uiSourceWidth);	pMipmapImages[0].uiExtendedHeight = Image::CalcExtendedDimension((unsigned short)m_uiSourceHeight);	pMipmapImages[0].uiEncodingBitsBytes = 0;	pMipmapImages[0].paucEncodingBits = std::shared_ptr<unsigned char>(new unsigned char[uiEncodingBitsBytesPerBlock], [](unsigned char *p) { delete[] p; });	//block position in pixels	// remove the bottom 2 bits to get the block coordinates 	unsigned int iBlockPosX = (a_iPixelX & 0xFFFFFFFC);	unsigned int iBlockPosY = (a_iPixelY & 0xFFFFFFFC);	int numXBlocks = (origWidth / 4);	int numYBlocks = (origHeight / 4);		// block location 	//int iBlockX = (a_iPixelX % 4) == 0 ? a_iPixelX / 4.0f : (a_iPixelX / 4) + 1;	//int iBlockY = (a_iPixelY % 4) == 0 ? a_iPixelY / 4.0f : (a_iPixelY / 4) + 1;	//m_paucEncodingBits += ((iBlockY * numXBlocks) + iBlockX) * uiEncodingBitsBytesPerBlock;		unsigned int num = numXBlocks*numYBlocks;	unsigned int uiH = 0, uiV = 0;	unsigned char* pEncodingBits = m_pMipmapImages[0].paucEncodingBits.get();	for (unsigned int uiBlock = 0; uiBlock < num; uiBlock++)	{		if (uiH == iBlockPosX && uiV == iBlockPosY)		{			memcpy(pMipmapImages[0].paucEncodingBits.get(),pEncodingBits, uiEncodingBitsBytesPerBlock);			break;		}		pEncodingBits += uiEncodingBitsBytesPerBlock;		uiH += 4;		if (uiH >= origWidth)		{			uiH = 0;			uiV += 4;		}	}	delete [] m_pMipmapImages;	m_pMipmapImages = pMipmapImages;}// ----------------------------------------------------------------------------------------------------//void File::Write(){	FILE *pfile = fopen(m_pstrFilename, "wb");	if (pfile == nullptr)	{		printf("Error: couldn't open Etc file (%s)\n", m_pstrFilename);		exit(1);	}	m_pheader->Write(pfile);	for(unsigned int mip = 0; mip < m_uiNumMipmaps; mip++)	{		if(m_fileformat == Format::KTX)		{			// Write u32 image size			uint32_t u32ImageSize = m_pMipmapImages[mip].uiEncodingBitsBytes;			uint32_t szBytesWritten = fwrite(&u32ImageSize, 1, sizeof(u32ImageSize), pfile);			assert(szBytesWritten == sizeof(u32ImageSize));		}		unsigned int iResult = (int)fwrite(m_pMipmapImages[mip].paucEncodingBits.get(), 1, m_pMipmapImages[mip].uiEncodingBitsBytes, pfile);		if (iResult != m_pMipmapImages[mip].uiEncodingBitsBytes)	{		printf("Error: couldn't write Etc file (%s)\n", m_pstrFilename);		exit(1);		}	}	fclose(pfile);}// ----------------------------------------------------------------------------------------------------//
 |