yaze 0.2.2
Link to the Past ROM Editor
 
Loading...
Searching...
No Matches
compression.h
Go to the documentation of this file.
1#ifndef YAZE_APP_GFX_COMPRESSION_H
2#define YAZE_APP_GFX_COMPRESSION_H
3
4#include <cstdint>
5#include <memory>
6#include <string>
7#include <vector>
8
9#include "absl/status/status.h"
10#include "absl/status/statusor.h"
11#include "util/macro.h"
12
13#define BUILD_HEADER(command, length) (command << 5) + (length - 1)
14
15namespace yaze {
16namespace gfx {
17
18std::vector<uint8_t> HyruleMagicCompress(uint8_t const* const src,
19 int const oldsize, int* const size,
20 int const flag);
21
22std::vector<uint8_t> HyruleMagicDecompress(uint8_t const* src, int* const size,
23 int const p_big_endian);
24
29namespace lc_lz2 {
30
31const int D_NINTENDO_C_MODE1 = 0;
32const int D_NINTENDO_C_MODE2 = 1;
33
34const int D_CMD_COPY = 0;
35const int D_CMD_BYTE_REPEAT = 1;
36const int D_CMD_WORD_REPEAT = 2;
37const int D_CMD_BYTE_INC = 3;
38const int D_CMD_COPY_EXISTING = 4;
39
40const int D_MAX_NORMAL_LENGTH = 32;
41const int D_MAX_LENGTH = 1024;
42
43const int INITIAL_ALLOC_SIZE = 1024;
44
45constexpr int kCommandDirectCopy = 0;
46constexpr int kCommandByteFill = 1;
47constexpr int kCommandWordFill = 2;
48constexpr int kCommandIncreasingFill = 3;
49constexpr int kCommandRepeatingBytes = 4;
50constexpr int kCommandLongLength = 7;
51constexpr int kMaxLengthNormalHeader = 32;
52constexpr int kMaxLengthCompression = 1024;
53constexpr int kNintendoMode1 = 0;
54constexpr int kNintendoMode2 = 1;
55constexpr int kSnesByteMax = 0xFF;
56constexpr int kCommandMod = 0x07;
57constexpr int kExpandedMod = 0xE0;
58constexpr int kExpandedLengthMod = 0x3FF;
59constexpr int kNormalLengthMod = 0x1F;
60constexpr int kCompressionStringMod = 7 << 5;
61
62// Represents a command in the compression algorithm.
64 // The command arguments for each possible command.
65 std::array<std::array<char, 2>, 5> arguments;
66
67 // The size of each possible command.
68 std::array<uint, 5> cmd_size;
69
70 // The size of the data processed by each possible command.
71 std::array<uint, 5> data_size;
72};
73
74using CommandArgumentArray = std::array<std::array<char, 2>, 5>;
75using CommandSizeArray = std::array<uint, 5>;
76using DataSizeArray = std::array<uint, 5>;
77
78// Represents a piece of compressed data.
80 char command;
81 int length;
83 std::string argument;
84 std::shared_ptr<CompressionPiece> next = nullptr;
85 CompressionPiece() = default;
86 CompressionPiece(int cmd, int len, std::string& args, int arg_len)
87 : command(cmd), length(len), argument_length(arg_len), argument(args) {}
88};
90using CompressionPiecePointer = std::shared_ptr<CompressionPiece>;
91
93
95
96// Compression V1
97
98void CheckByteRepeat(const uint8_t* rom_data, DataSizeArray& data_size_taken,
99 CommandArgumentArray& cmd_args, uint& src_data_pos,
100 const unsigned int last_pos);
101
102void CheckWordRepeat(const uint8_t* rom_data, DataSizeArray& data_size_taken,
103 CommandArgumentArray& cmd_args, uint& src_data_pos,
104 const unsigned int last_pos);
105
106void CheckIncByte(const uint8_t* rom_data, DataSizeArray& data_size_taken,
107 CommandArgumentArray& cmd_args, uint& src_data_pos,
108 const unsigned int last_pos);
109
110void CheckIntraCopy(const uint8_t* rom_data, DataSizeArray& data_size_taken,
111 CommandArgumentArray& cmd_args, uint& src_data_pos,
112 const unsigned int last_pos, unsigned int start);
113
114void ValidateForByteGain(const DataSizeArray& data_size_taken,
115 const CommandSizeArray& cmd_size, uint& max_win,
116 uint& cmd_with_max);
117
118void CompressionCommandAlternative(const uint8_t* rom_data,
119 CompressionPiecePointer& compressed_chain,
120 const CommandSizeArray& cmd_size,
121 const CommandArgumentArray& cmd_args,
122 uint& src_data_pos, uint& comp_accumulator,
123 uint& cmd_with_max, uint& max_win);
124
125// Compression V2
126
127void CheckByteRepeatV2(const uint8_t* data, uint& src_pos, const unsigned int last_pos,
128 CompressionCommand& cmd);
129
130void CheckWordRepeatV2(const uint8_t* data, uint& src_pos, const unsigned int last_pos,
131 CompressionCommand& cmd);
132
133void CheckIncByteV2(const uint8_t* data, uint& src_pos, const unsigned int last_pos,
134 CompressionCommand& cmd);
135
136void CheckIntraCopyV2(const uint8_t* data, uint& src_pos, const unsigned int last_pos,
137 unsigned int start, CompressionCommand& cmd);
138
139void ValidateForByteGainV2(const CompressionCommand& cmd, uint& max_win,
140 uint& cmd_with_max);
141
142void CompressionCommandAlternativeV2(const uint8_t* data,
143 const CompressionCommand& cmd,
144 CompressionPiecePointer& compressed_chain,
145 uint& src_pos, uint& comp_accumulator,
146 uint& cmd_with_max, uint& max_win);
147
152absl::StatusOr<std::vector<uint8_t>> CompressV2(const uint8_t* data,
153 const int start,
154 const int length, int mode = 1,
155 bool check = false);
156
157absl::StatusOr<std::vector<uint8_t>> CompressGraphics(const uint8_t* data,
158 const int pos,
159 const int length);
160absl::StatusOr<std::vector<uint8_t>> CompressOverworld(const uint8_t* data,
161 const int pos,
162 const int length);
163absl::StatusOr<std::vector<uint8_t>> CompressOverworld(
164 const std::vector<uint8_t> data, const int pos, const int length);
165
166absl::StatusOr<CompressionPiecePointer> SplitCompressionPiece(
167 CompressionPiecePointer& piece, int mode);
168
169std::vector<uint8_t> CreateCompressionString(CompressionPiecePointer& start,
170 int mode);
171
173 int mode, int start, int src_data_pos);
174
176
178 std::vector<uint8_t> data;
179 std::vector<uint8_t> compressed_data;
180 std::vector<CompressionPiece> compression_pieces;
181 std::vector<uint8_t> compression_string;
182 unsigned int src_pos;
183 unsigned int last_pos;
184 unsigned int start;
185 unsigned int comp_accumulator = 0;
187 unsigned int max_win = 0;
189 int mode;
190
191 // Constructor to initialize the context
192 CompressionContext(const std::vector<uint8_t>& data_, const int start,
193 const int length)
194 : data(data_), src_pos(start), last_pos(start + length - 1), mode(0) {}
195
196 // Constructor to initialize the context
197 CompressionContext(const std::vector<uint8_t>& data_, const int start,
198 const int length, int mode_)
199 : data(data_),
200 src_pos(start),
201 last_pos(start + length - 1),
202 mode(mode_) {}
203};
204
205void CheckByteRepeatV3(CompressionContext& context);
206void CheckWordRepeatV3(CompressionContext& context);
207void CheckIncByteV3(CompressionContext& context);
208void CheckIntraCopyV3(CompressionContext& context);
209
210void InitializeCompression(CompressionContext& context);
211void CheckAvailableCompressionCommands(CompressionContext& context);
212void DetermineBestCompression(CompressionContext& context);
213void HandleDirectCopy(CompressionContext& context);
214void AddCompressionToChain(CompressionContext& context);
215absl::Status ValidateCompressionResultV3(const CompressionContext& context);
216
217absl::StatusOr<CompressionPiece> SplitCompressionPieceV3(
218 CompressionPiece& piece, int mode);
219void FinalizeCompression(CompressionContext& context);
220
225absl::StatusOr<std::vector<uint8_t>> CompressV3(
226 const std::vector<uint8_t>& data, const int start, const int length,
227 int mode = 1, bool check = false);
228
229std::string SetBuffer(const std::vector<uint8_t>& data, int src_pos,
230 int comp_accumulator);
231std::string SetBuffer(const uint8_t* data, int src_pos, int comp_accumulator);
232void memfill(const uint8_t* data, std::vector<uint8_t>& buffer, int buffer_pos,
233 int offset, int length);
234
240absl::StatusOr<std::vector<uint8_t>> DecompressV2(const uint8_t* data,
241 int offset, int size = 0x800,
242 int mode = 1);
243absl::StatusOr<std::vector<uint8_t>> DecompressGraphics(const uint8_t* data,
244 int pos, int size);
245absl::StatusOr<std::vector<uint8_t>> DecompressOverworld(const uint8_t* data,
246 int pos, int size);
247absl::StatusOr<std::vector<uint8_t>> DecompressOverworld(
248 const std::vector<uint8_t> data, int pos, int size);
249
250} // namespace lc_lz2
251} // namespace gfx
252} // namespace yaze
253
254#endif // YAZE_APP_GFX_COMPRESSION_H
const int D_CMD_COPY_EXISTING
Definition compression.h:38
void CheckIntraCopyV3(CompressionContext &context)
absl::StatusOr< std::vector< uint8_t > > CompressGraphics(const uint8_t *data, const int pos, const int length)
absl::StatusOr< std::vector< uint8_t > > CompressV3(const std::vector< uint8_t > &data, const int start, const int length, int mode, bool check)
Compresses a buffer of data using the LC_LZ2 algorithm.
constexpr int kExpandedLengthMod
Definition compression.h:58
const int D_CMD_COPY
Definition compression.h:34
constexpr int kCommandDirectCopy
Definition compression.h:45
void memfill(const uint8_t *data, std::vector< uint8_t > &buffer, int buffer_pos, int offset, int length)
void FinalizeCompression(CompressionContext &context)
void CheckWordRepeatV2(const uint8_t *data, uint &src_pos, const unsigned int last_pos, CompressionCommand &cmd)
void InitializeCompression(CompressionContext &context)
struct CompressionPiece CompressionPiece
Definition compression.h:89
absl::StatusOr< std::vector< uint8_t > > CompressOverworld(const uint8_t *data, const int pos, const int length)
constexpr int kSnesByteMax
Definition compression.h:55
const int D_NINTENDO_C_MODE2
Definition compression.h:32
constexpr int kNintendoMode1
Definition compression.h:53
std::shared_ptr< CompressionPiece > CompressionPiecePointer
Definition compression.h:90
void CheckIncByteV3(CompressionContext &context)
absl::Status ValidateCompressionResultV3(const CompressionContext &context)
constexpr int kNintendoMode2
Definition compression.h:54
std::string SetBuffer(const uint8_t *data, int src_pos, int comp_accumulator)
constexpr int kCommandRepeatingBytes
Definition compression.h:49
std::array< std::array< char, 2 >, 5 > CommandArgumentArray
Definition compression.h:74
const int D_MAX_LENGTH
Definition compression.h:41
void CheckByteRepeat(const uint8_t *rom_data, DataSizeArray &data_size_taken, CommandArgumentArray &cmd_args, uint &src_data_pos, const unsigned int last_pos)
const int INITIAL_ALLOC_SIZE
Definition compression.h:43
void CheckByteRepeatV2(const uint8_t *data, uint &src_pos, const unsigned int last_pos, CompressionCommand &cmd)
void HandleDirectCopy(CompressionContext &context)
void CheckIncByteV2(const uint8_t *rom_data, uint &src_data_pos, const unsigned int last_pos, CompressionCommand &cmd)
absl::StatusOr< std::vector< uint8_t > > DecompressGraphics(const uint8_t *data, int pos, int size)
const int D_CMD_BYTE_INC
Definition compression.h:37
constexpr int kCommandMod
Definition compression.h:56
void PrintCompressionPiece(const CompressionPiecePointer &piece)
constexpr int kCommandWordFill
Definition compression.h:47
constexpr int kMaxLengthNormalHeader
Definition compression.h:51
void CheckAvailableCompressionCommands(CompressionContext &context)
constexpr int kMaxLengthCompression
Definition compression.h:52
const int D_MAX_NORMAL_LENGTH
Definition compression.h:40
void CompressionCommandAlternativeV2(const uint8_t *rom_data, const CompressionCommand &cmd, CompressionPiecePointer &compressed_chain, uint &src_data_pos, uint &comp_accumulator, uint &cmd_with_max, uint &max_win)
void CheckIntraCopyV2(const uint8_t *rom_data, uint &src_data_pos, const unsigned int last_pos, unsigned int start, CompressionCommand &cmd)
std::vector< uint8_t > CreateCompressionString(CompressionPiecePointer &start, int mode)
absl::StatusOr< std::vector< uint8_t > > DecompressOverworld(const uint8_t *data, int pos, int size)
std::array< uint, 5 > DataSizeArray
Definition compression.h:76
constexpr int kCommandIncreasingFill
Definition compression.h:48
void ValidateForByteGainV2(const CompressionCommand &cmd, uint &max_win, uint &cmd_with_max)
void ValidateForByteGain(const DataSizeArray &data_size_taken, const CommandSizeArray &cmd_size, uint &max_win, uint &cmd_with_max)
constexpr int kCommandByteFill
Definition compression.h:46
void PrintCompressionChain(const CompressionPiecePointer &chain_head)
void CheckIncByte(const uint8_t *rom_data, DataSizeArray &data_size_taken, CommandArgumentArray &cmd_args, uint &src_data_pos, const unsigned int last_pos)
constexpr int kExpandedMod
Definition compression.h:57
absl::StatusOr< std::vector< uint8_t > > DecompressV2(const uint8_t *data, int offset, int size, int mode)
Decompresses a buffer of data using the LC_LZ2 algorithm.
void AddCompressionToChain(CompressionContext &context)
absl::StatusOr< CompressionPiecePointer > SplitCompressionPiece(CompressionPiecePointer &piece, int mode)
void CompressionCommandAlternative(const uint8_t *rom_data, CompressionPiecePointer &compressed_chain, const CommandSizeArray &cmd_size, const CommandArgumentArray &cmd_args, uint &src_data_pos, uint &comp_accumulator, uint &cmd_with_max, uint &max_win)
constexpr int kNormalLengthMod
Definition compression.h:59
constexpr int kCompressionStringMod
Definition compression.h:60
void CheckWordRepeat(const uint8_t *rom_data, DataSizeArray &data_size_taken, CommandArgumentArray &cmd_args, uint &src_data_pos, const unsigned int last_pos)
constexpr int kCommandLongLength
Definition compression.h:50
void CheckByteRepeatV3(CompressionContext &context)
const int D_CMD_WORD_REPEAT
Definition compression.h:36
CompressionPiecePointer MergeCopy(CompressionPiecePointer &start)
std::array< uint, 5 > CommandSizeArray
Definition compression.h:75
void DetermineBestCompression(CompressionContext &context)
absl::StatusOr< CompressionPiece > SplitCompressionPieceV3(CompressionPiece &piece, int mode)
absl::StatusOr< std::vector< uint8_t > > CompressV2(const uint8_t *data, const int start, const int length, int mode, bool check)
Compresses a buffer of data using the LC_LZ2 algorithm.
void CheckWordRepeatV3(CompressionContext &context)
const int D_CMD_BYTE_REPEAT
Definition compression.h:35
const int D_NINTENDO_C_MODE1
Definition compression.h:31
void CheckIntraCopy(const uint8_t *rom_data, DataSizeArray &data_size_taken, CommandArgumentArray &cmd_args, uint &src_data_pos, const unsigned int last_pos, unsigned int start)
absl::Status ValidateCompressionResult(CompressionPiecePointer &chain_head, int mode, int start, int src_data_pos)
std::vector< uint8_t > HyruleMagicDecompress(uint8_t const *src, int *const size, int const p_big_endian)
std::vector< uint8_t > HyruleMagicCompress(uint8_t const *const src, int const oldsize, int *const size, int const flag)
Main namespace for the application.
Definition controller.cc:18
std::array< std::array< char, 2 >, 5 > arguments
Definition compression.h:65
std::vector< CompressionPiece > compression_pieces
std::vector< uint8_t > compression_string
std::vector< uint8_t > compressed_data
CompressionContext(const std::vector< uint8_t > &data_, const int start, const int length, int mode_)
CompressionContext(const std::vector< uint8_t > &data_, const int start, const int length)
std::shared_ptr< CompressionPiece > next
Definition compression.h:84
CompressionPiece(int cmd, int len, std::string &args, int arg_len)
Definition compression.h:86