2016-04-30 13:44:31 -07:00
|
|
|
// Copyright (c) 2016- PPSSPP Project.
|
|
|
|
|
|
|
|
// This program 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, version 2.0 or later versions.
|
|
|
|
|
|
|
|
// This program 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 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
2016-04-30 15:03:39 -07:00
|
|
|
#ifndef USING_QT_UI
|
2016-04-30 15:07:56 -07:00
|
|
|
#include <libpng17/png.h>
|
2016-04-30 15:03:39 -07:00
|
|
|
#endif
|
|
|
|
|
2016-04-30 14:19:23 -07:00
|
|
|
#include "ext/xxhash.h"
|
2016-04-30 15:03:39 -07:00
|
|
|
#include "Common/ColorConv.h"
|
2016-04-30 14:05:03 -07:00
|
|
|
#include "Common/FileUtil.h"
|
|
|
|
#include "Core/Config.h"
|
|
|
|
#include "Core/System.h"
|
2016-04-30 15:07:56 -07:00
|
|
|
#include "Core/TextureReplacer.h"
|
2016-04-30 14:05:03 -07:00
|
|
|
#include "Core/ELF/ParamSFO.h"
|
2016-04-30 14:19:23 -07:00
|
|
|
#include "GPU/Common/TextureDecoder.h"
|
2016-04-30 13:44:31 -07:00
|
|
|
|
|
|
|
TextureReplacer::TextureReplacer() : enabled_(false) {
|
|
|
|
}
|
|
|
|
|
|
|
|
TextureReplacer::~TextureReplacer() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureReplacer::Init() {
|
2016-04-30 14:05:03 -07:00
|
|
|
NotifyConfigChanged();
|
2016-04-30 13:44:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureReplacer::NotifyConfigChanged() {
|
2016-04-30 14:05:03 -07:00
|
|
|
gameID_ = g_paramSFO.GetValueString("DISC_ID");
|
|
|
|
|
|
|
|
enabled_ = !gameID_.empty() && (g_Config.bReplaceTextures || g_Config.bSaveNewTextures);
|
|
|
|
if (enabled_) {
|
|
|
|
basePath_ = GetSysDirectory(DIRECTORY_TEXTURES) + gameID_ + "/";
|
|
|
|
|
|
|
|
// If we're saving, auto-create the directory.
|
|
|
|
if (g_Config.bSaveNewTextures && !File::Exists(basePath_)) {
|
|
|
|
File::CreateFullPath(basePath_);
|
|
|
|
}
|
|
|
|
|
|
|
|
enabled_ = File::Exists(basePath_) && File::IsDirectory(basePath_);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Load ini file.
|
2016-04-30 13:44:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 TextureReplacer::ComputeHash(u32 addr, int bufw, int w, int h, GETextureFormat fmt, u16 maxSeenV) {
|
2016-04-30 14:05:03 -07:00
|
|
|
_dbg_assert_msg_(G3D, enabled_, "Replacement not enabled");
|
2016-04-30 14:19:23 -07:00
|
|
|
|
|
|
|
if (!LookupHashRange(addr, w, h)) {
|
|
|
|
// There wasn't any hash range, let's fall back to maxSeenV logic.
|
|
|
|
if (h == 512 && maxSeenV < 512 && maxSeenV != 0) {
|
|
|
|
h = (int)maxSeenV;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: In order to have the most stable hash possible, skip space between w/bufw?
|
|
|
|
// TODO: Use hash based on ini file, or always crc32c, or etc.
|
|
|
|
const u32 sizeInRAM = (textureBitsPerPixel[fmt] * bufw * h) / 8;
|
|
|
|
const u32 *checkp = (const u32 *)Memory::GetPointer(addr);
|
|
|
|
return DoQuickTexHash(checkp, sizeInRAM);
|
2016-04-30 13:44:31 -07:00
|
|
|
}
|
|
|
|
|
2016-04-30 15:18:45 -07:00
|
|
|
ReplacedTexture TextureReplacer::FindReplacement(u64 cachekey, u32 hash) {
|
2016-04-30 14:05:03 -07:00
|
|
|
_assert_msg_(G3D, enabled_, "Replacement not enabled");
|
|
|
|
|
2016-04-30 13:44:31 -07:00
|
|
|
ReplacedTexture result;
|
|
|
|
result.alphaStatus_ = ReplacedTextureAlpha::UNKNOWN;
|
2016-04-30 14:05:03 -07:00
|
|
|
|
|
|
|
// Only actually replace if we're replacing. We might just be saving.
|
|
|
|
if (g_Config.bReplaceTextures) {
|
2016-04-30 15:41:12 -07:00
|
|
|
std::string hashfile = LookupHashFile(cachekey, hash, 0);
|
|
|
|
const std::string filename = basePath_ + hashfile;
|
|
|
|
|
|
|
|
if (!hashfile.empty() && File::Exists(filename)) {
|
|
|
|
// TODO: Count levels that exist, etc.
|
|
|
|
}
|
2016-04-30 14:05:03 -07:00
|
|
|
}
|
2016-04-30 13:44:31 -07:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-04-30 15:03:39 -07:00
|
|
|
#ifndef USING_QT_UI
|
|
|
|
static bool WriteTextureToPNG(png_imagep image, const std::string &filename, int convert_to_8bit, const void *buffer, png_int_32 row_stride, const void *colormap) {
|
|
|
|
FILE *fp = File::OpenCFile(filename, "wb");
|
|
|
|
if (!fp) {
|
|
|
|
ERROR_LOG(COMMON, "Unable to open texture file for writing.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (png_image_write_to_stdio(image, fp, convert_to_8bit, buffer, row_stride, colormap)) {
|
|
|
|
if (fclose(fp) != 0) {
|
|
|
|
ERROR_LOG(COMMON, "Texture file write failed.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
ERROR_LOG(COMMON, "Texture PNG encode failed.");
|
|
|
|
fclose(fp);
|
|
|
|
remove(filename.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-04-30 15:21:48 -07:00
|
|
|
void TextureReplacer::NotifyTextureDecoded(u64 cachekey, u32 hash, u32 addr, const void *data, int pitch, int level, int w, int h, ReplacedTextureFormat fmt) {
|
2016-04-30 14:05:03 -07:00
|
|
|
_assert_msg_(G3D, enabled_, "Replacement not enabled");
|
|
|
|
if (!g_Config.bSaveNewTextures) {
|
|
|
|
// Ignore.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-04-30 15:41:12 -07:00
|
|
|
std::string hashfile = LookupHashFile(cachekey, hash, level);
|
|
|
|
const std::string filename = basePath_ + hashfile;
|
2016-04-30 15:03:39 -07:00
|
|
|
|
2016-04-30 15:41:12 -07:00
|
|
|
// If it's empty, it's an ignored hash, we intentionally don't save.
|
|
|
|
if (hashfile.empty() || File::Exists(filename)) {
|
|
|
|
// If it exists, must've been decoded and saved as a new texture already.
|
2016-04-30 15:03:39 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USING_QT_UI
|
|
|
|
ERROR_LOG(G3D, "Replacement texture saving not implemented for Qt");
|
|
|
|
#else
|
|
|
|
if (fmt != ReplacedTextureFormat::F_8888) {
|
|
|
|
saveBuf.resize((pitch * h) / sizeof(u16));
|
|
|
|
switch (fmt) {
|
|
|
|
case ReplacedTextureFormat::F_5650:
|
|
|
|
ConvertRGBA565ToRGBA8888(saveBuf.data(), (const u16 *)data, (pitch * h) / sizeof(u16));
|
|
|
|
break;
|
|
|
|
case ReplacedTextureFormat::F_5551:
|
|
|
|
ConvertRGBA5551ToRGBA8888(saveBuf.data(), (const u16 *)data, (pitch * h) / sizeof(u16));
|
|
|
|
break;
|
|
|
|
case ReplacedTextureFormat::F_4444:
|
|
|
|
ConvertRGBA4444ToRGBA8888(saveBuf.data(), (const u16 *)data, (pitch * h) / sizeof(u16));
|
|
|
|
break;
|
|
|
|
case ReplacedTextureFormat::F_8888_BGRA:
|
|
|
|
ConvertBGRA8888ToRGBA8888(saveBuf.data(), (const u32 *)data, (pitch * h) / sizeof(u32));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = saveBuf.data();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only save the hashed portion of the PNG.
|
|
|
|
LookupHashRange(addr, w, h);
|
|
|
|
|
|
|
|
png_image png;
|
|
|
|
memset(&png, 0, sizeof(png));
|
|
|
|
png.version = PNG_IMAGE_VERSION;
|
|
|
|
png.format = PNG_FORMAT_RGBA;
|
|
|
|
png.width = w;
|
|
|
|
png.height = h;
|
|
|
|
bool success = WriteTextureToPNG(&png, filename, 0, data, pitch, nullptr);
|
|
|
|
png_image_free(&png);
|
|
|
|
|
|
|
|
if (png.warning_or_error >= 2) {
|
|
|
|
ERROR_LOG(COMMON, "Saving screenshot to PNG produced errors.");
|
|
|
|
} else if (success) {
|
|
|
|
NOTICE_LOG(G3D, "Saving texture for replacement: %08x / %dx%d", hash, w, h);
|
|
|
|
}
|
|
|
|
#endif
|
2016-04-30 13:44:31 -07:00
|
|
|
}
|
|
|
|
|
2016-04-30 15:41:12 -07:00
|
|
|
std::string TextureReplacer::LookupHashFile(u64 cachekey, u32 hash, int level) {
|
|
|
|
// TODO: Look up via ini for alias / ignored.
|
|
|
|
return HashName(cachekey, hash, level) + ".png";
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string TextureReplacer::HashName(u64 cachekey, u32 hash, int level) {
|
|
|
|
char hashname[16 + 8 + 1 + 11 + 1] = {};
|
|
|
|
if (level > 0) {
|
|
|
|
snprintf(hashname, sizeof(hashname), "%016llx%08x_%d.png", cachekey, hash, level);
|
|
|
|
} else {
|
|
|
|
snprintf(hashname, sizeof(hashname), "%016llx%08x.png", cachekey, hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hashname;
|
|
|
|
}
|
|
|
|
|
2016-04-30 14:19:23 -07:00
|
|
|
bool TextureReplacer::LookupHashRange(u32 addr, int &w, int &h) {
|
|
|
|
// TODO: Pull from table loaded via ini.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-04-30 13:44:31 -07:00
|
|
|
void ReplacedTexture::Load(int level, void *out, int rowPitch) {
|
2016-04-30 14:05:03 -07:00
|
|
|
_assert_msg_(G3D, (size_t)level < levels_.size(), "Invalid miplevel");
|
|
|
|
_assert_msg_(G3D, out != nullptr && rowPitch > 0, "Invalid out/pitch");
|
|
|
|
|
|
|
|
// TODO
|
2016-04-30 13:44:31 -07:00
|
|
|
}
|