mirror of
https://github.com/PabloMK7/citra.git
synced 2024-11-28 10:20:17 +00:00
common: Add more robust ZSTD handling. (#7071)
This commit is contained in:
parent
07839fb3ce
commit
4c59443ed2
3 changed files with 35 additions and 8 deletions
|
@ -11,20 +11,25 @@ namespace Common::Compression {
|
||||||
|
|
||||||
std::vector<u8> CompressDataZSTD(std::span<const u8> source, s32 compression_level) {
|
std::vector<u8> CompressDataZSTD(std::span<const u8> source, s32 compression_level) {
|
||||||
compression_level = std::clamp(compression_level, ZSTD_minCLevel(), ZSTD_maxCLevel());
|
compression_level = std::clamp(compression_level, ZSTD_minCLevel(), ZSTD_maxCLevel());
|
||||||
|
|
||||||
const std::size_t max_compressed_size = ZSTD_compressBound(source.size());
|
const std::size_t max_compressed_size = ZSTD_compressBound(source.size());
|
||||||
std::vector<u8> compressed(max_compressed_size);
|
|
||||||
|
|
||||||
|
if (ZSTD_isError(max_compressed_size)) {
|
||||||
|
LOG_ERROR(Common, "Error determining ZSTD maximum compressed size: {} ({})",
|
||||||
|
ZSTD_getErrorName(max_compressed_size), max_compressed_size);
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<u8> compressed(max_compressed_size);
|
||||||
const std::size_t compressed_size = ZSTD_compress(
|
const std::size_t compressed_size = ZSTD_compress(
|
||||||
compressed.data(), compressed.size(), source.data(), source.size(), compression_level);
|
compressed.data(), compressed.size(), source.data(), source.size(), compression_level);
|
||||||
|
|
||||||
if (ZSTD_isError(compressed_size)) {
|
if (ZSTD_isError(compressed_size)) {
|
||||||
// Compression failed
|
LOG_ERROR(Common, "Error compressing ZSTD data: {} ({})",
|
||||||
|
ZSTD_getErrorName(compressed_size), compressed_size);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
compressed.resize(compressed_size);
|
compressed.resize(compressed_size);
|
||||||
|
|
||||||
return compressed;
|
return compressed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,15 +40,32 @@ std::vector<u8> CompressDataZSTDDefault(std::span<const u8> source) {
|
||||||
std::vector<u8> DecompressDataZSTD(std::span<const u8> compressed) {
|
std::vector<u8> DecompressDataZSTD(std::span<const u8> compressed) {
|
||||||
const std::size_t decompressed_size =
|
const std::size_t decompressed_size =
|
||||||
ZSTD_getFrameContentSize(compressed.data(), compressed.size());
|
ZSTD_getFrameContentSize(compressed.data(), compressed.size());
|
||||||
std::vector<u8> decompressed(decompressed_size);
|
|
||||||
|
|
||||||
|
if (decompressed_size == ZSTD_CONTENTSIZE_UNKNOWN) {
|
||||||
|
LOG_ERROR(Common, "ZSTD decompressed size could not be determined.");
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
if (decompressed_size == ZSTD_CONTENTSIZE_ERROR || ZSTD_isError(decompressed_size)) {
|
||||||
|
LOG_ERROR(Common, "Error determining ZSTD decompressed size: {} ({})",
|
||||||
|
ZSTD_getErrorName(decompressed_size), decompressed_size);
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<u8> decompressed(decompressed_size);
|
||||||
const std::size_t uncompressed_result_size = ZSTD_decompress(
|
const std::size_t uncompressed_result_size = ZSTD_decompress(
|
||||||
decompressed.data(), decompressed.size(), compressed.data(), compressed.size());
|
decompressed.data(), decompressed.size(), compressed.data(), compressed.size());
|
||||||
|
|
||||||
if (decompressed_size != uncompressed_result_size || ZSTD_isError(uncompressed_result_size)) {
|
if (decompressed_size != uncompressed_result_size) {
|
||||||
// Decompression failed
|
LOG_ERROR(Common, "ZSTD decompression expected {} bytes, got {}", decompressed_size,
|
||||||
|
uncompressed_result_size);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
if (ZSTD_isError(uncompressed_result_size)) {
|
||||||
|
LOG_ERROR(Common, "Error decompressing ZSTD data: {} ({})",
|
||||||
|
ZSTD_getErrorName(uncompressed_result_size), uncompressed_result_size);
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
return decompressed;
|
return decompressed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -206,6 +206,10 @@ ShaderDiskCache::LoadPrecompiledFile(FileUtil::IOFile& file, bool compressed) {
|
||||||
if (compressed) {
|
if (compressed) {
|
||||||
const std::vector<u8> decompressed =
|
const std::vector<u8> decompressed =
|
||||||
Common::Compression::DecompressDataZSTD(precompiled_file);
|
Common::Compression::DecompressDataZSTD(precompiled_file);
|
||||||
|
if (decompressed.empty()) {
|
||||||
|
LOG_ERROR(Render_OpenGL, "Could not decompress precompiled shader cache.");
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
SaveArrayToPrecompiled(decompressed.data(), decompressed.size());
|
SaveArrayToPrecompiled(decompressed.data(), decompressed.size());
|
||||||
} else {
|
} else {
|
||||||
SaveArrayToPrecompiled(precompiled_file.data(), precompiled_file.size());
|
SaveArrayToPrecompiled(precompiled_file.data(), precompiled_file.size());
|
||||||
|
|
|
@ -10,10 +10,11 @@
|
||||||
|
|
||||||
namespace Pica::Shader::Generator {
|
namespace Pica::Shader::Generator {
|
||||||
|
|
||||||
|
// NOTE: Changing the order impacts shader transferable and precompiled cache loading.
|
||||||
enum ProgramType : u32 {
|
enum ProgramType : u32 {
|
||||||
VS = 0,
|
VS = 0,
|
||||||
GS = 2,
|
|
||||||
FS = 1,
|
FS = 1,
|
||||||
|
GS = 2,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum Attributes {
|
enum Attributes {
|
||||||
|
|
Loading…
Reference in a new issue