From a573036f6b7c4301373a7bad4193fe5626f773fa Mon Sep 17 00:00:00 2001 From: wheremyfoodat Date: Tue, 21 Mar 2023 19:49:01 +0200 Subject: [PATCH] [LZ77 decompressor] Move to own source file --- CMakeLists.txt | 2 +- include/loader/lz77.hpp | 82 +++------------------------------------- src/core/loader/lz77.cpp | 80 +++++++++++++++++++++++++++++++++++++++ 3 files changed, 86 insertions(+), 78 deletions(-) create mode 100644 src/core/loader/lz77.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index d5f662b1..1235b7ab 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -68,7 +68,7 @@ set(PICA_SOURCE_FILES src/core/PICA/gpu.cpp src/core/PICA/regs.cpp src/core/PICA ) set(RENDERER_GL_SOURCE_FILES src/core/renderer_gl/renderer_gl.cpp src/core/renderer_gl/textures.cpp src/core/renderer_gl/etc1.cpp) -set(LOADER_SOURCE_FILES src/core/loader/elf.cpp src/core/loader/ncsd.cpp src/core/loader/ncch.cpp) +set(LOADER_SOURCE_FILES src/core/loader/elf.cpp src/core/loader/ncsd.cpp src/core/loader/ncch.cpp src/core/loader/lz77.cpp) set(FS_SOURCE_FILES src/core/fs/archive_self_ncch.cpp src/core/fs/archive_save_data.cpp src/core/fs/archive_sdmc.cpp src/core/fs/archive_ext_save_data.cpp src/core/fs/archive_ncch.cpp ) diff --git a/include/loader/lz77.hpp b/include/loader/lz77.hpp index 0b0d94f2..8adbe22d 100644 --- a/include/loader/lz77.hpp +++ b/include/loader/lz77.hpp @@ -1,89 +1,17 @@ #pragma once -#include -#include #include #include "helpers.hpp" // For parsing the LZ77 format used for compressing the .code file in the ExeFS namespace CartLZ77 { - // The difference in size between the compressed and decompressed file is stored - // As a footer in the compressed file. To get the decompressed size, we extract the diff - // And add it to the compressed size - static u32 decompressedSize(const u8* buffer, u32 compressedSize) { - u32 sizeDiff; - std::memcpy(&sizeDiff, buffer + compressedSize - 4, sizeof(u32)); - return sizeDiff + compressedSize; - } + // Retrieves the uncompressed size of the compressed LZ77 data stored in buffer with a specific compressed size + u32 decompressedSize(const u8* buffer, u32 compressedSize); template static u32 decompressedSize(const std::vector& buffer) { return decompressedSize((u8*)buffer.data(), u32(buffer.size() * sizeof(T))); } - static bool decompress(std::vector& output, const std::vector& input) { - u32 sizeCompressed = input.size() * sizeof(u8); - u32 sizeDecompressed = decompressedSize(input); - output.resize(sizeDecompressed); - - const u8* compressed = (u8*)input.data(); - const u8* footer = compressed + sizeCompressed - 8; - - u32 bufferTopAndBottom; - std::memcpy(&bufferTopAndBottom, footer, sizeof(u32)); - - u32 out = sizeDecompressed; // TODO: Is this meant to be u32 or s32? - u32 index = sizeCompressed - ((bufferTopAndBottom >> 24) & 0xff); - u32 stopIndex = sizeCompressed - (bufferTopAndBottom & 0xffffff); - - // Set all of the decompressed buffer to 0 and copy the compressed buffer to the start of it - std::fill(output.begin(), output.end(), 0); - std::copy(input.begin(), input.end(), output.begin()); - - while (index > stopIndex) { - u8 control = compressed[--index]; - - for (uint i = 0; i < 8; i++) { - if (index <= stopIndex) - break; - if (index <= 0) - break; - if (out <= 0) - break; - - if (control & 0x80) { - // Check if compression is out of bounds - if (index < 2) - return false; - index -= 2; - - u32 segmentOffset = compressed[index] | (compressed[index + 1] << 8); - u32 segment_size = ((segmentOffset >> 12) & 15) + 3; - segmentOffset &= 0x0FFF; - segmentOffset += 2; - - // Check if compression is out of bounds - if (out < segment_size) - return false; - - for (uint j = 0; j < segment_size; j++) { - // Check if compression is out of bounds - if (out + segmentOffset >= sizeDecompressed) - return false; - - u8 data = output[out + segmentOffset]; - output[--out] = data; - } - } - else { - // Check if compression is out of bounds - if (out < 1) - return false; - output[--out] = compressed[--index]; - } - control <<= 1; - } - } - - return true; - } -} + // Decompresses an LZ77-compressed buffer stored in input to output + bool decompress(std::vector& output, const std::vector& input); +} // End namespace CartLZ77 diff --git a/src/core/loader/lz77.cpp b/src/core/loader/lz77.cpp new file mode 100644 index 00000000..85bb96d2 --- /dev/null +++ b/src/core/loader/lz77.cpp @@ -0,0 +1,80 @@ +#pragma once +#include +#include +#include "loader/lz77.hpp" + +// The difference in size between the compressed and decompressed file is stored +// As a footer in the compressed file. To get the decompressed size, we extract the diff +// And add it to the compressed size +u32 CartLZ77::decompressedSize(const u8* buffer, u32 compressedSize) { + u32 sizeDiff; + std::memcpy(&sizeDiff, buffer + compressedSize - 4, sizeof(u32)); + return sizeDiff + compressedSize; +} + +bool CartLZ77::decompress(std::vector& output, const std::vector& input) { + u32 sizeCompressed = input.size() * sizeof(u8); + u32 sizeDecompressed = decompressedSize(input); + output.resize(sizeDecompressed); + + const u8* compressed = (u8*)input.data(); + const u8* footer = compressed + sizeCompressed - 8; + + u32 bufferTopAndBottom; + std::memcpy(&bufferTopAndBottom, footer, sizeof(u32)); + + u32 out = sizeDecompressed; // TODO: Is this meant to be u32 or s32? + u32 index = sizeCompressed - ((bufferTopAndBottom >> 24) & 0xff); + u32 stopIndex = sizeCompressed - (bufferTopAndBottom & 0xffffff); + + // Set all of the decompressed buffer to 0 and copy the compressed buffer to the start of it + std::fill(output.begin(), output.end(), 0); + std::copy(input.begin(), input.end(), output.begin()); + + while (index > stopIndex) { + u8 control = compressed[--index]; + + for (uint i = 0; i < 8; i++) { + if (index <= stopIndex) + break; + if (index <= 0) + break; + if (out <= 0) + break; + + if (control & 0x80) { + // Check if compression is out of bounds + if (index < 2) + return false; + index -= 2; + + u32 segmentOffset = compressed[index] | (compressed[index + 1] << 8); + u32 segment_size = ((segmentOffset >> 12) & 15) + 3; + segmentOffset &= 0x0FFF; + segmentOffset += 2; + + // Check if compression is out of bounds + if (out < segment_size) + return false; + + for (uint j = 0; j < segment_size; j++) { + // Check if compression is out of bounds + if (out + segmentOffset >= sizeDecompressed) + return false; + + u8 data = output[out + segmentOffset]; + output[--out] = data; + } + } + else { + // Check if compression is out of bounds + if (out < 1) + return false; + output[--out] = compressed[--index]; + } + control <<= 1; + } + } + + return true; +} \ No newline at end of file