-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathCompress.cpp
113 lines (104 loc) · 3.8 KB
/
Compress.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
#include "Compress.h"
namespace ROMUtils
{
/// <summary>
/// Helper function to populate the dynamic programming jump table for fast RLE compression.
/// </summary>
/// <param name="jumpLimit">
/// The maximum jump size for the table (different for 8-bit/16-bit).
/// </param>
void RLEMetadata::InitializeJumpTableHelper(unsigned short jumpLimit)
{
// Define variables used in the creation of the jump table
unsigned char *data = (unsigned char *) this->data;
unsigned short *R = new unsigned short[data_len * 2];
memset(R, 0, data_len << 2);
unsigned short *C = R + data_len;
int cons = 0, minrun = GetMinimumRunSize();
// Seed the dynamic programming jump table
R[data_len - 1] = 1;
// Populate R backwards in the jump table
for (unsigned int i = data_len - 1; i >= 1; --i)
{
R[i - 1] = (R[i] == jumpLimit || data[i] != data[i - 1]) ? 1 : R[i] + 1;
}
// Populate C forwards in the jump table
for (unsigned int i = 0; i < data_len; ++i)
{
if (R[i] < minrun)
{
++cons;
}
if (R[i] >= minrun) {
C[i - cons] = cons;
cons = 0;
i += R[i] - 1;
} else if (cons == jumpLimit) {
C[i - cons + 1] = cons;
cons = 0;
}
}
if (cons)
{
C[data_len - cons] = cons;
}
JumpTable = R; // set the jump table since it has been populated
}
/// <summary>
/// Helper function to get the length of compressed data using the RLE metadata's jump table.
/// </summary>
/// <param name="opcodeSize">
/// Number of bytes to use for compression opcodes (different for 8-bit/16-bit).
/// </param>
unsigned int RLEMetadata::GetCompressedLengthHelper(unsigned int opcodeSize)
{
// Define the variables used to traverse the jump table
unsigned int size = 0;
unsigned int i = 0, minrun = GetMinimumRunSize();
unsigned short *R = JumpTable, *C = JumpTable + data_len;
// Calculate the size
while (i < data_len)
{
bool runmode = R[i] >= minrun;
unsigned short len = runmode ? R[i] : C[i];
size += opcodeSize + (runmode ? 1 : len);
i += len;
}
return 1 + size + opcodeSize; // type identifier, data, termination value
}
/// <summary>
/// Helper function to get the compressed data using the RLE metadata's jump table.
/// </summary>
/// <returns>
/// The compressed data in a dynamically allocated array.
/// </returns>
void *RLEMetadata::GetCompressedData()
{
// Define variables used by compression
QVector<unsigned char> compressedData;
unsigned int i = 0, minrun = GetMinimumRunSize();
unsigned short *R = JumpTable, *C = JumpTable + data_len;
unsigned char *data = (unsigned char *) this->data;
// Populate the compressed data
compressedData.append(GetTypeIdentifier());
while (i < data_len)
{
bool runmode = R[i] >= minrun;
unsigned short len = runmode ? R[i] : C[i];
AddOpcode(compressedData, len, runmode);
for (int j = 0; j < (runmode ? 1 : len); ++j)
{
compressedData.append(data[i + j]);
}
i += len;
}
AddOpcode(compressedData, 0, false);
// Create the dynamically allocated char array
unsigned char *compressed = new unsigned char[compressedData.size()];
for (int i = 0; i < compressedData.size(); ++i)
{
compressed[i] = compressedData[i];
}
return compressed;
}
} // namespace ROMUtils