2013-04-17 10:30:38 +00:00
|
|
|
/*
|
|
|
|
* Texture Filtering
|
|
|
|
* Version: 1.0
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 Hiroshi Morii All Rights Reserved.
|
|
|
|
* Email koolsmoky(at)users.sourceforge.net
|
|
|
|
* Web http://www.3dfxzone.it/koolsmoky
|
|
|
|
*
|
|
|
|
* this is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* this is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with GNU Make; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*/
|
|
|
|
|
2015-10-14 00:20:03 +00:00
|
|
|
#ifdef _WIN32
|
2013-04-17 10:30:38 +00:00
|
|
|
#pragma warning(disable: 4786)
|
|
|
|
#endif
|
|
|
|
|
2016-01-20 06:09:05 +00:00
|
|
|
#include <Common/path.h>
|
|
|
|
#include <Common/StdString.h>
|
2013-04-17 10:30:38 +00:00
|
|
|
#include "TxFilter.h"
|
|
|
|
#include "TextureFilters.h"
|
|
|
|
#include "TxDbg.h"
|
|
|
|
|
|
|
|
void TxFilter::clear()
|
|
|
|
{
|
|
|
|
/* clear hires texture cache */
|
|
|
|
delete _txHiResCache;
|
|
|
|
|
|
|
|
/* clear texture cache */
|
|
|
|
delete _txTexCache;
|
|
|
|
|
|
|
|
/* free memory */
|
|
|
|
TxMemBuf::getInstance()->shutdown();
|
|
|
|
|
|
|
|
/* clear other stuff */
|
|
|
|
delete _txImage;
|
|
|
|
delete _txQuantize;
|
|
|
|
delete _txUtil;
|
|
|
|
}
|
|
|
|
|
|
|
|
TxFilter::~TxFilter()
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
TxFilter::TxFilter(int maxwidth, int maxheight, int maxbpp, int options,
|
2013-04-28 05:49:14 +00:00
|
|
|
int cachesize, const wchar_t *path, const wchar_t *ident,
|
2013-04-17 10:30:38 +00:00
|
|
|
dispInfoFuncExt callback) :
|
|
|
|
_numcore(0),
|
|
|
|
_tex1(NULL),
|
|
|
|
_tex2(NULL),
|
|
|
|
_maxwidth(0),
|
|
|
|
_maxheight(0),
|
|
|
|
_maxbpp(0),
|
|
|
|
_options(0),
|
|
|
|
_cacheSize(0),
|
|
|
|
_txQuantize(NULL),
|
|
|
|
_txTexCache(NULL),
|
|
|
|
_txHiResCache(NULL),
|
|
|
|
_txUtil(NULL),
|
|
|
|
_txImage(NULL),
|
|
|
|
_initialized(false)
|
|
|
|
{
|
|
|
|
/* HACKALERT: the emulator misbehaves and sometimes forgets to shutdown */
|
|
|
|
if ((ident && wcscmp(ident, L"DEFAULT") != 0 && _ident.compare(ident) == 0) &&
|
|
|
|
_maxwidth == maxwidth &&
|
|
|
|
_maxheight == maxheight &&
|
|
|
|
_maxbpp == maxbpp &&
|
|
|
|
_options == options &&
|
|
|
|
_cacheSize == cachesize) return;
|
|
|
|
clear(); /* gcc does not allow the destructor to be called */
|
|
|
|
|
|
|
|
/* shamelessness :P this first call to the debug output message creates
|
|
|
|
* a file in the executable directory. */
|
|
|
|
INFO(0, L"------------------------------------------------------------------\n");
|
2015-10-14 00:20:03 +00:00
|
|
|
#ifdef GHQCHK
|
|
|
|
INFO(0, L" GlideHQ Hires Texture Checker 1.02.00.%d\n", 0);
|
|
|
|
#else
|
|
|
|
INFO(0, L" GlideHQ version 1.02.00.%d\n", 0);
|
|
|
|
#endif
|
2013-04-17 10:30:38 +00:00
|
|
|
INFO(0, L" Copyright (C) 2010 Hiroshi Morii All Rights Reserved\n");
|
|
|
|
INFO(0, L" email : koolsmoky(at)users.sourceforge.net\n");
|
|
|
|
INFO(0, L" website : http://www.3dfxzone.it/koolsmoky\n");
|
|
|
|
INFO(0, L"\n");
|
|
|
|
INFO(0, L" Glide64 official website : http://glide64.emuxhaven.net\n");
|
|
|
|
INFO(0, L"------------------------------------------------------------------\n");
|
|
|
|
|
|
|
|
_options = options;
|
|
|
|
|
|
|
|
_txImage = new TxImage();
|
|
|
|
_txQuantize = new TxQuantize();
|
|
|
|
_txUtil = new TxUtil();
|
|
|
|
|
|
|
|
/* get number of CPU cores. */
|
|
|
|
_numcore = _txUtil->getNumberofProcessors();
|
|
|
|
|
|
|
|
_initialized = 0;
|
|
|
|
|
|
|
|
_tex1 = NULL;
|
|
|
|
_tex2 = NULL;
|
|
|
|
|
|
|
|
/* XXX: anything larger than 1024 * 1024 is overkill */
|
|
|
|
_maxwidth = maxwidth > 1024 ? 1024 : maxwidth;
|
|
|
|
_maxheight = maxheight > 1024 ? 1024 : maxheight;
|
|
|
|
_maxbpp = maxbpp;
|
|
|
|
|
|
|
|
_cacheSize = cachesize;
|
|
|
|
|
|
|
|
/* TODO: validate options and do overrides here*/
|
|
|
|
|
|
|
|
/* save path name */
|
|
|
|
if (path)
|
|
|
|
_path.assign(path);
|
|
|
|
|
|
|
|
/* save ROM name */
|
|
|
|
if (ident && wcscmp(ident, L"DEFAULT") != 0)
|
|
|
|
_ident.assign(ident);
|
|
|
|
|
|
|
|
/* check for dxtn extensions */
|
|
|
|
if (!TxLoadLib::getInstance()->getdxtCompressTexFuncExt())
|
|
|
|
_options &= ~S3TC_COMPRESSION;
|
|
|
|
|
|
|
|
if (!TxLoadLib::getInstance()->getfxtCompressTexFuncExt())
|
|
|
|
_options &= ~FXT1_COMPRESSION;
|
|
|
|
|
|
|
|
switch (options & COMPRESSION_MASK) {
|
|
|
|
case FXT1_COMPRESSION:
|
|
|
|
case S3TC_COMPRESSION:
|
|
|
|
break;
|
|
|
|
case NCC_COMPRESSION:
|
|
|
|
default:
|
|
|
|
_options &= ~COMPRESSION_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (TxMemBuf::getInstance()->init(_maxwidth, _maxheight)) {
|
|
|
|
if (!_tex1)
|
|
|
|
_tex1 = TxMemBuf::getInstance()->get(0);
|
|
|
|
|
|
|
|
if (!_tex2)
|
|
|
|
_tex2 = TxMemBuf::getInstance()->get(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if !_16BPP_HACK
|
|
|
|
/* initialize hq4x filter */
|
|
|
|
hq4x_init();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* initialize texture cache in bytes. 128Mb will do nicely in most cases */
|
|
|
|
_txTexCache = new TxTexCache(_options, _cacheSize, _path.c_str(), _ident.c_str(), callback);
|
|
|
|
|
|
|
|
/* hires texture */
|
|
|
|
#if HIRES_TEXTURE
|
|
|
|
_txHiResCache = new TxHiResCache(_maxwidth, _maxheight, _maxbpp, _options, _path.c_str(), _ident.c_str(), callback);
|
|
|
|
|
|
|
|
if (_txHiResCache->empty())
|
|
|
|
_options &= ~HIRESTEXTURES_MASK;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!(_options & COMPRESS_TEX))
|
|
|
|
_options &= ~COMPRESSION_MASK;
|
|
|
|
|
|
|
|
if (_tex1 && _tex2)
|
|
|
|
_initialized = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean
|
|
|
|
TxFilter::filter(uint8 *src, int srcwidth, int srcheight, uint16 srcformat, uint64 g64crc, GHQTexInfo *info)
|
|
|
|
{
|
|
|
|
uint8 *texture = src;
|
|
|
|
uint8 *tmptex = _tex1;
|
|
|
|
uint16 destformat = srcformat;
|
|
|
|
|
|
|
|
/* We need to be initialized first! */
|
|
|
|
if (!_initialized) return 0;
|
|
|
|
|
|
|
|
/* find cached textures */
|
|
|
|
if (_cacheSize) {
|
|
|
|
|
|
|
|
/* calculate checksum of source texture */
|
|
|
|
if (!g64crc)
|
|
|
|
g64crc = (uint64)(_txUtil->checksumTx(texture, srcwidth, srcheight, srcformat));
|
|
|
|
|
|
|
|
DBG_INFO(80, L"filter: crc:%08X %08X %d x %d gfmt:%x\n",
|
|
|
|
(uint32)(g64crc >> 32), (uint32)(g64crc & 0xffffffff), srcwidth, srcheight, srcformat);
|
|
|
|
|
|
|
|
#if 0 /* use hirestex to retrieve cached textures. */
|
|
|
|
/* check if we have it in cache */
|
|
|
|
if (!(g64crc & 0xffffffff00000000) && /* we reach here only when there is no hires texture for this crc */
|
|
|
|
_txTexCache->get(g64crc, info)) {
|
|
|
|
DBG_INFO(80, L"cache hit: %d x %d gfmt:%x\n", info->width, info->height, info->format);
|
|
|
|
return 1; /* yep, we've got it */
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Leave small textures alone because filtering makes little difference.
|
|
|
|
* Moreover, some filters require at least 4 * 4 to work.
|
|
|
|
* Bypass _options to do ARGB8888->16bpp if _maxbpp=16 or forced color reduction.
|
|
|
|
*/
|
|
|
|
if ((srcwidth >= 4 && srcheight >= 4) &&
|
|
|
|
((_options & (FILTER_MASK|ENHANCEMENT_MASK|COMPRESSION_MASK)) ||
|
|
|
|
(srcformat == GR_TEXFMT_ARGB_8888 && (_maxbpp < 32 || _options & FORCE16BPP_TEX)))) {
|
|
|
|
|
|
|
|
#if !_16BPP_HACK
|
|
|
|
/* convert textures to a format that the compressor accepts (ARGB8888) */
|
|
|
|
if (_options & COMPRESSION_MASK) {
|
|
|
|
#endif
|
|
|
|
if (srcformat != GR_TEXFMT_ARGB_8888) {
|
|
|
|
if (!_txQuantize->quantize(texture, tmptex, srcwidth, srcheight, srcformat, GR_TEXFMT_ARGB_8888)) {
|
|
|
|
DBG_INFO(80, L"Error: unsupported format! gfmt:%x\n", srcformat);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
texture = tmptex;
|
|
|
|
destformat = GR_TEXFMT_ARGB_8888;
|
|
|
|
}
|
|
|
|
#if !_16BPP_HACK
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch (destformat) {
|
|
|
|
case GR_TEXFMT_ARGB_8888:
|
|
|
|
|
|
|
|
/*
|
|
|
|
* prepare texture enhancements (x2, x4 scalers)
|
|
|
|
*/
|
|
|
|
int scale_shift = 0, num_filters = 0;
|
|
|
|
uint32 filter = 0;
|
|
|
|
|
|
|
|
if ((_options & ENHANCEMENT_MASK) == HQ4X_ENHANCEMENT) {
|
|
|
|
if (srcwidth <= (_maxwidth >> 2) && srcheight <= (_maxheight >> 2)) {
|
|
|
|
filter |= HQ4X_ENHANCEMENT;
|
|
|
|
scale_shift = 2;
|
|
|
|
num_filters++;
|
|
|
|
} else if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
filter |= HQ2X_ENHANCEMENT;
|
|
|
|
scale_shift = 1;
|
|
|
|
num_filters++;
|
|
|
|
}
|
|
|
|
} else if (_options & ENHANCEMENT_MASK) {
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
filter |= (_options & ENHANCEMENT_MASK);
|
|
|
|
scale_shift = 1;
|
|
|
|
num_filters++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* prepare texture filters
|
|
|
|
*/
|
|
|
|
if (_options & (SMOOTH_FILTER_MASK|SHARP_FILTER_MASK)) {
|
|
|
|
filter |= (_options & (SMOOTH_FILTER_MASK|SHARP_FILTER_MASK));
|
|
|
|
num_filters++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* execute texture enhancements and filters
|
|
|
|
*/
|
|
|
|
while (num_filters > 0) {
|
|
|
|
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
|
|
|
|
uint8 *_texture = texture;
|
|
|
|
uint8 *_tmptex = tmptex;
|
|
|
|
|
|
|
|
unsigned int numcore = _numcore;
|
|
|
|
unsigned int blkrow = 0;
|
|
|
|
while (numcore > 1 && blkrow == 0) {
|
|
|
|
blkrow = (srcheight >> 2) / numcore;
|
|
|
|
numcore--;
|
|
|
|
}
|
|
|
|
filter_8888((uint32*)_texture, srcwidth, srcheight, (uint32*)_tmptex, filter);
|
|
|
|
|
|
|
|
if (filter & ENHANCEMENT_MASK) {
|
|
|
|
srcwidth <<= scale_shift;
|
|
|
|
srcheight <<= scale_shift;
|
|
|
|
filter &= ~ENHANCEMENT_MASK;
|
|
|
|
scale_shift = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
texture = tmptex;
|
|
|
|
num_filters--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* texture compression
|
|
|
|
*/
|
|
|
|
/* ignored if we only have texture compression option on.
|
|
|
|
* only done when texture enhancer is used. see constructor. */
|
|
|
|
if ((_options & COMPRESSION_MASK) &&
|
|
|
|
(srcwidth >= 64 && srcheight >= 64) /* Texture compression is not suitable for low pixel coarse detail
|
|
|
|
* textures. The assumption here is that textures larger than 64x64
|
|
|
|
* have enough detail to produce decent quality when compressed. The
|
|
|
|
* down side is that narrow stripped textures that the N64 often use
|
|
|
|
* for large background textures are also ignored. It would be more
|
|
|
|
* reasonable if decisions are made based on fourier-transform
|
|
|
|
* spectrum or RMS error.
|
|
|
|
*/
|
|
|
|
) {
|
|
|
|
int compressionType = _options & COMPRESSION_MASK;
|
|
|
|
int tmpwidth, tmpheight;
|
|
|
|
uint16 tmpformat;
|
|
|
|
/* XXX: textures that use 8bit alpha channel look bad with the current
|
|
|
|
* fxt1 library, so we substitute it with dxtn for now. afaik all gfx
|
|
|
|
* cards that support fxt1 also support dxtn. (3dfx and Intel) */
|
|
|
|
if ((destformat == GR_TEXFMT_ALPHA_INTENSITY_88) ||
|
|
|
|
(destformat == GR_TEXFMT_ARGB_8888) ||
|
|
|
|
(destformat == GR_TEXFMT_ALPHA_8)) {
|
|
|
|
compressionType = S3TC_COMPRESSION;
|
|
|
|
}
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
if (_txQuantize->compress(texture, tmptex,
|
|
|
|
srcwidth, srcheight, srcformat,
|
|
|
|
&tmpwidth, &tmpheight, &tmpformat,
|
|
|
|
compressionType)) {
|
|
|
|
srcwidth = tmpwidth;
|
|
|
|
srcheight = tmpheight;
|
|
|
|
destformat = tmpformat;
|
|
|
|
texture = tmptex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* texture (re)conversions
|
|
|
|
*/
|
|
|
|
if (destformat == GR_TEXFMT_ARGB_8888) {
|
|
|
|
if (srcformat == GR_TEXFMT_ARGB_8888 && (_maxbpp < 32 || _options & FORCE16BPP_TEX)) srcformat = GR_TEXFMT_ARGB_4444;
|
|
|
|
if (srcformat != GR_TEXFMT_ARGB_8888) {
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
if (!_txQuantize->quantize(texture, tmptex, srcwidth, srcheight, GR_TEXFMT_ARGB_8888, srcformat)) {
|
|
|
|
DBG_INFO(80, L"Error: unsupported format! gfmt:%x\n", srcformat);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
texture = tmptex;
|
|
|
|
destformat = srcformat;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
#if !_16BPP_HACK
|
|
|
|
case GR_TEXFMT_ARGB_4444:
|
|
|
|
|
|
|
|
int scale_shift = 0;
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
|
|
|
|
switch (_options & ENHANCEMENT_MASK) {
|
|
|
|
case HQ4X_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 2) && srcheight <= (_maxheight >> 2)) {
|
|
|
|
hq4x_4444((uint8*)texture, (uint8*)tmptex, srcwidth, srcheight, srcwidth, srcwidth * 4 * 2);
|
|
|
|
scale_shift = 2;
|
|
|
|
}/* else if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
hq2x_16((uint8*)texture, srcwidth * 2, (uint8*)tmptex, srcwidth * 2 * 2, srcwidth, srcheight);
|
|
|
|
scale_shift = 1;
|
|
|
|
}*/
|
|
|
|
break;
|
|
|
|
case HQ2X_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
hq2x_16((uint8*)texture, srcwidth * 2, (uint8*)tmptex, srcwidth * 2 * 2, srcwidth, srcheight);
|
|
|
|
scale_shift = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case HQ2XS_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
hq2xS_16((uint8*)texture, srcwidth * 2, (uint8*)tmptex, srcwidth * 2 * 2, srcwidth, srcheight);
|
|
|
|
scale_shift = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case LQ2X_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
lq2x_16((uint8*)texture, srcwidth * 2, (uint8*)tmptex, srcwidth * 2 * 2, srcwidth, srcheight);
|
|
|
|
scale_shift = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case LQ2XS_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
lq2xS_16((uint8*)texture, srcwidth * 2, (uint8*)tmptex, srcwidth * 2 * 2, srcwidth, srcheight);
|
|
|
|
scale_shift = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case X2SAI_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
Super2xSaI_4444((uint16*)texture, (uint16*)tmptex, srcwidth, srcheight, srcwidth);
|
|
|
|
scale_shift = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case X2_ENHANCEMENT:
|
|
|
|
if (srcwidth <= (_maxwidth >> 1) && srcheight <= (_maxheight >> 1)) {
|
|
|
|
Texture2x_16((uint8*)texture, srcwidth * 2, (uint8*)tmptex, srcwidth * 2 * 2, srcwidth, srcheight);
|
|
|
|
scale_shift = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (scale_shift) {
|
|
|
|
srcwidth <<= scale_shift;
|
|
|
|
srcheight <<= scale_shift;
|
|
|
|
texture = tmptex;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_options & SMOOTH_FILTER_MASK) {
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
SmoothFilter_4444((uint16*)texture, srcwidth, srcheight, (uint16*)tmptex, (_options & SMOOTH_FILTER_MASK));
|
|
|
|
texture = tmptex;
|
|
|
|
} else if (_options & SHARP_FILTER_MASK) {
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
SharpFilter_4444((uint16*)texture, srcwidth, srcheight, (uint16*)tmptex, (_options & SHARP_FILTER_MASK));
|
|
|
|
texture = tmptex;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case GR_TEXFMT_ARGB_1555:
|
|
|
|
break;
|
|
|
|
case GR_TEXFMT_RGB_565:
|
|
|
|
break;
|
|
|
|
case GR_TEXFMT_ALPHA_8:
|
|
|
|
break;
|
|
|
|
#endif /* _16BPP_HACK */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in the texture info. */
|
|
|
|
info->data = texture;
|
|
|
|
info->width = srcwidth;
|
|
|
|
info->height = srcheight;
|
|
|
|
info->format = destformat;
|
|
|
|
info->smallLodLog2 = _txUtil->grLodLog2(srcwidth, srcheight);
|
|
|
|
info->largeLodLog2 = info->smallLodLog2;
|
|
|
|
info->aspectRatioLog2 = _txUtil->grAspectRatioLog2(srcwidth, srcheight);
|
|
|
|
info->is_hires_tex = 0;
|
|
|
|
|
|
|
|
/* cache the texture. */
|
|
|
|
if (_cacheSize) _txTexCache->add(g64crc, info);
|
|
|
|
|
|
|
|
DBG_INFO(80, L"filtered texture: %d x %d gfmt:%x\n", info->width, info->height, info->format);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean
|
|
|
|
TxFilter::hirestex(uint64 g64crc, uint64 r_crc64, uint16 *palette, GHQTexInfo *info)
|
|
|
|
{
|
|
|
|
/* NOTE: Rice CRC32 sometimes return the same value for different textures.
|
|
|
|
* As a workaround, Glide64 CRC32 is used for the key for NON-hires
|
|
|
|
* texture cache.
|
|
|
|
*
|
|
|
|
* r_crc64 = hi:palette low:texture
|
|
|
|
* (separate crc. doesn't necessary have to be rice crc)
|
|
|
|
* g64crc = texture + palette glide64 crc32
|
|
|
|
* (can be any other crc if robust)
|
|
|
|
*/
|
|
|
|
|
|
|
|
DBG_INFO(80, L"hirestex: r_crc64:%08X %08X, g64crc:%08X %08X\n",
|
|
|
|
(uint32)(r_crc64 >> 32), (uint32)(r_crc64 & 0xffffffff),
|
|
|
|
(uint32)(g64crc >> 32), (uint32)(g64crc & 0xffffffff));
|
|
|
|
|
|
|
|
#if HIRES_TEXTURE
|
|
|
|
/* check if we have it in hires memory cache. */
|
|
|
|
if ((_options & HIRESTEXTURES_MASK) && r_crc64) {
|
|
|
|
if (_txHiResCache->get(r_crc64, info)) {
|
|
|
|
DBG_INFO(80, L"hires hit: %d x %d gfmt:%x\n", info->width, info->height, info->format);
|
|
|
|
|
|
|
|
/* TODO: Enable emulation for special N64 combiner modes. There are few ways
|
|
|
|
* to get this done. Also applies for CI textures below.
|
|
|
|
*
|
|
|
|
* Solution 1. Load the hiresolution textures in ARGB8888 (or A8, IA88) format
|
|
|
|
* to cache. When a cache is hit, then we take the modes passed in from Glide64
|
|
|
|
* (also TODO) and apply the modification. Then we do color reduction or format
|
|
|
|
* conversion or compression if desired and stuff it into the non-hires texture
|
|
|
|
* cache.
|
|
|
|
*
|
|
|
|
* Solution 2. When a cache is hit and if the combiner modes are present,
|
|
|
|
* convert the texture to ARGB4444 and pass it back to Glide64 to process.
|
|
|
|
* If a texture is compressed, it needs to be decompressed first. Then add
|
|
|
|
* the processed texture to the non-hires texture cache.
|
|
|
|
*
|
|
|
|
* Solution 3. Hybrid of the above 2. Load the textures in ARGB8888 (A8, IA88)
|
|
|
|
* format. Convert the texture to ARGB4444 and pass it back to Glide64 when
|
|
|
|
* the combiner modes are present. Get the processed texture back from Glide64
|
|
|
|
* and compress if desired and add it to the non-hires texture cache.
|
|
|
|
*
|
|
|
|
* Solution 4. Take the easy way out and forget about this whole thing.
|
|
|
|
*/
|
|
|
|
|
|
|
|
return 1; /* yep, got it */
|
|
|
|
}
|
|
|
|
if (_txHiResCache->get((r_crc64 & 0xffffffff), info)) {
|
|
|
|
DBG_INFO(80, L"hires hit: %d x %d gfmt:%x\n", info->width, info->height, info->format);
|
|
|
|
|
|
|
|
/* for true CI textures, we use the passed in palette to convert to
|
|
|
|
* ARGB1555 and add it to memory cache.
|
|
|
|
*
|
|
|
|
* NOTE: we do this AFTER all other texture cache searches because
|
|
|
|
* only a few texture packs actually use true CI textures.
|
|
|
|
*
|
|
|
|
* NOTE: the pre-converted palette from Glide64 is in RGBA5551 format.
|
|
|
|
* A comp comes before RGB comp.
|
|
|
|
*/
|
|
|
|
if (palette && info->format == GR_TEXFMT_P_8) {
|
|
|
|
DBG_INFO(80, L"found GR_TEXFMT_P_8 format. Need conversion!!\n");
|
|
|
|
|
|
|
|
int width = info->width;
|
|
|
|
int height = info->height;
|
|
|
|
uint16 format = info->format;
|
|
|
|
/* XXX: avoid collision with zlib compression buffer in TxHiResTexture::get */
|
|
|
|
uint8 *texture = info->data;
|
|
|
|
uint8 *tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
|
|
|
|
/* use palette and convert to 16bit format */
|
|
|
|
_txQuantize->P8_16BPP((uint32*)texture, (uint32*)tmptex, info->width, info->height, (uint32*)palette);
|
|
|
|
texture = tmptex;
|
|
|
|
format = GR_TEXFMT_ARGB_1555;
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
/* XXX: compressed if memory cache compression is ON */
|
|
|
|
if (_options & COMPRESSION_MASK) {
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
if (_txQuantize->quantize(texture, tmptex, info->width, info->height, format, GR_TEXFMT_ARGB_8888)) {
|
|
|
|
texture = tmptex;
|
|
|
|
format = GR_TEXFMT_ARGB_8888;
|
|
|
|
}
|
|
|
|
if (format == GR_TEXFMT_ARGB_8888) {
|
|
|
|
tmptex = (texture == _tex1) ? _tex2 : _tex1;
|
|
|
|
if (_txQuantize->compress(texture, tmptex,
|
|
|
|
info->width, info->height, GR_TEXFMT_ARGB_1555,
|
|
|
|
&width, &height, &format,
|
|
|
|
_options & COMPRESSION_MASK)) {
|
|
|
|
texture = tmptex;
|
|
|
|
} else {
|
|
|
|
/*if (!_txQuantize->quantize(texture, tmptex, info->width, info->height, GR_TEXFMT_ARGB_8888, GR_TEXFMT_ARGB_1555)) {
|
|
|
|
DBG_INFO(80, L"Error: unsupported format! gfmt:%x\n", format);
|
|
|
|
return 0;
|
|
|
|
}*/
|
|
|
|
texture = tmptex;
|
|
|
|
format = GR_TEXFMT_ARGB_1555;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* fill in the required info to return */
|
|
|
|
info->data = texture;
|
|
|
|
info->width = width;
|
|
|
|
info->height = height;
|
|
|
|
info->format = format;
|
|
|
|
info->smallLodLog2 = _txUtil->grLodLog2(width, height);
|
|
|
|
info->largeLodLog2 = info->smallLodLog2;
|
|
|
|
info->aspectRatioLog2 = _txUtil->grAspectRatioLog2(width, height);
|
|
|
|
info->is_hires_tex = 1;
|
|
|
|
|
|
|
|
/* XXX: add to hires texture cache!!! */
|
|
|
|
_txHiResCache->add(r_crc64, info);
|
|
|
|
|
|
|
|
DBG_INFO(80, L"GR_TEXFMT_P_8 loaded as gfmt:%x!\n", format);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* check if we have it in memory cache */
|
|
|
|
if (_cacheSize && g64crc) {
|
|
|
|
if (_txTexCache->get(g64crc, info)) {
|
|
|
|
DBG_INFO(80, L"cache hit: %d x %d gfmt:%x\n", info->width, info->height, info->format);
|
|
|
|
return 1; /* yep, we've got it */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DBG_INFO(80, L"no cache hits.\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64
|
|
|
|
TxFilter::checksum64(uint8 *src, int width, int height, int size, int rowStride, uint8 *palette)
|
|
|
|
{
|
|
|
|
if (_options & (HIRESTEXTURES_MASK|DUMP_TEX))
|
|
|
|
return _txUtil->checksum64(src, width, height, size, rowStride, palette);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean
|
|
|
|
TxFilter::dmptx(uint8 *src, int width, int height, int rowStridePixel, uint16 gfmt, uint16 n64fmt, uint64 r_crc64)
|
|
|
|
{
|
|
|
|
if (!_initialized)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!(_options & DUMP_TEX))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
DBG_INFO(80, L"gfmt = %02x n64fmt = %02x\n", gfmt, n64fmt);
|
|
|
|
DBG_INFO(80, L"hirestex: r_crc64:%08X %08X\n",
|
|
|
|
(uint32)(r_crc64 >> 32), (uint32)(r_crc64 & 0xffffffff));
|
|
|
|
|
|
|
|
if (!_txQuantize->quantize(src, _tex1, rowStridePixel, height, (gfmt & 0x00ff), GR_TEXFMT_ARGB_8888))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
src = _tex1;
|
|
|
|
|
|
|
|
if (!_path.empty() && !_ident.empty()) {
|
|
|
|
/* dump it to disk */
|
|
|
|
FILE *fp = NULL;
|
|
|
|
CPath tmpbuf(stdstr().FromUTF16(_path.c_str()).c_str(),"");
|
|
|
|
|
|
|
|
/* create directories */
|
|
|
|
tmpbuf.AppendDirectory("texture_dump");
|
|
|
|
|
2015-10-25 11:10:54 +00:00
|
|
|
if (!tmpbuf.DirectoryExists() && !tmpbuf.DirectoryCreate())
|
2013-04-17 10:30:38 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
tmpbuf.AppendDirectory(stdstr().FromUTF16(_ident.c_str()).c_str());
|
2015-10-25 11:10:54 +00:00
|
|
|
if (!tmpbuf.DirectoryExists() && !tmpbuf.DirectoryCreate())
|
2013-04-17 10:30:38 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
tmpbuf.AppendDirectory("GlideHQ");
|
2015-10-25 11:10:54 +00:00
|
|
|
if (!tmpbuf.DirectoryExists() && !tmpbuf.DirectoryCreate())
|
2013-04-17 10:30:38 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
if ((n64fmt >> 8) == 0x2) {
|
|
|
|
tmpbuf.SetNameExtension(stdstr_f("%ls#%08X#%01X#%01X#%08X_ciByRGBA.png",_ident.c_str(),(uint32)(r_crc64 & 0xffffffff), (n64fmt >> 8),(n64fmt & 0xf),(uint32)(r_crc64 >> 32)).c_str());
|
|
|
|
} else {
|
|
|
|
tmpbuf.SetNameExtension(stdstr_f("%ls#%08X#%01X#%01X_all.png",_ident.c_str(),(uint32)(r_crc64 & 0xffffffff),(n64fmt >> 8),(n64fmt & 0xf)).c_str());
|
|
|
|
}
|
2015-10-13 23:01:07 +00:00
|
|
|
#ifdef _WIN32
|
2013-04-17 10:30:38 +00:00
|
|
|
if ((fp = fopen(tmpbuf, "wb")) != NULL) {
|
|
|
|
#else
|
|
|
|
char cbuf[MAX_PATH];
|
|
|
|
wcstombs(cbuf, tmpbuf.c_str(), MAX_PATH);
|
|
|
|
if ((fp = fopen(cbuf, "wb")) != NULL) {
|
|
|
|
#endif
|
|
|
|
_txImage->writePNG(src, fp, width, height, (rowStridePixel << 2), 0x0003, 0);
|
|
|
|
fclose(fp);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean
|
|
|
|
TxFilter::reloadhirestex()
|
|
|
|
{
|
|
|
|
DBG_INFO(80, L"Reload hires textures from texture pack.\n");
|
|
|
|
|
|
|
|
if (_txHiResCache->load(0)) {
|
|
|
|
if (_txHiResCache->empty()) _options &= ~HIRESTEXTURES_MASK;
|
|
|
|
else _options |= HIRESTEXTURES_MASK;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|