forked from silicontrip/SkyReader
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathchecksum.cpp
289 lines (252 loc) · 8.56 KB
/
checksum.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
#include "checksum.h"
/*
data checksums
The checksums are a mess. There are four "types" of checksums:
Type 0: this is a CRC16 checksum of the first 0x1E unsigned chars of sector 0. The checksum itself is stored in block 0x01, offset 0x0E.
Type 1: this is a CRC16 checksum of the data area header. As there are two data areas, there are two of these checksums.
One is at block 0x08, offset 0x0E, and the other is at block 0x24, offset 0x0E.
Type 2: this is a CRC16 checksum of the data area. As there are two data areas, there are two of these checksums.
One is at block 0x08, offset 0x0C, and the other is at block 0x24, offset 0x0C.
Type 3: this is another CRC16 checksum of the data area, except padded with zeroes. As there are two data areas,
there are two of these checksums. One is at block 0x08, offset 0x0A, and the other is at block 0x24, offset 0x0A.
As type 0 is a checksum of a *supposedly* read-only sector, it's not all that important. It's also very straightforward to understand.
The type 1 checksum is a checksum of just one block, the data area header (0x08 and 0x24). As it's also stored WITHIN the
data area header, a default value must be supplied for the checksum before actually calculating it. That value is 0x0005.
The type 2 checksum is actually only a checksum of the first 4 blocks (EXCLUDING the data area header, and the access control blocks).
The type 3 checksum is a checksum of the next 4 blocks (EXCLUDING the data area header, and the access control blocks),
and then 0x0E blocks of zeroes.
Just to re-iterate, the encryption is applied AFTER all this checksum mess is done.
*/
// CCITT CRC Code
// Update the CRC for transmitted and received data using
// the CCITT 16bit algorithm (X^16 + X^12 + X^5 + 1).
unsigned short Checksum::UpdateCcittCrc16(unsigned short crc16, unsigned char data)
{
unsigned short num2 = (unsigned short) (data << 8);
for (unsigned int i = 0; i < 8; i++)
{
int num3;
if ((crc16 ^ num2) > 0x7fff)
{
num3 = 1;
}
else
{
num3 = 0;
}
crc16 = (unsigned short) ((crc16 << 1) ^ (num3 * 0x1021));
num2 = (unsigned short) (num2 << 1);
}
return crc16;
}
unsigned short Checksum::ComputeCcittCrc16(void const* data, unsigned int bytes)
{
unsigned short crc = 0xffff;
unsigned char const* numPtr = (unsigned char const*)data;
for (unsigned int i = 0; i < bytes; i++)
{
crc = UpdateCcittCrc16(crc, *((unsigned char const*) (numPtr + i)));
}
return crc;
}
bool Checksum::getChecksumParameters(int checksumType, unsigned int* checksumOffset, unsigned int* dataOffset, unsigned int* dataLength)
{
switch (checksumType)
{
case 0:
// Type 0 checksum.
*checksumOffset = 0x1E; // The checksum itself is stored in block 0x01, offset 0x0E.
*dataOffset = 0;
*dataLength = 0x1E; // checksum of the first 0x1E unsigned chars of sector 0.
break;
case 1:
// Type 1 checksum.
// Type 1: this is a CRC16 checksum of the data area header. As there are two data areas,
// there are two of these checksums. One is at block 0x08, offset 0x0E, and the other is at block 0x24, offset 0x0E.
*checksumOffset = 0x0E; // Checksum is stored within the data header block.
*dataOffset = 0;
*dataLength = 0x10; // The type 1 checksum is a checksum of just one block, the data area header (blocks 0x08 and 0x24).
break;
case 2:
// Type 2 checksum.
// Type 2: this is a CRC16 checksum of the data area. As there are two data areas, there are two of these checksums.
// One is at block 0x08, offset 0x0C, and the other is at block 0x24, offset 0x0C.
*checksumOffset = 0x0C;
*dataOffset = 0x10;
*dataLength = 0x40; // Checksum of the first 4 blocks (EXCLUDING the data area header, and the access control blocks).
break;
case 3:
// Type 3 checksum.
// Type 3: this is another CRC16 checksum of the data area, except padded with zeroes. As there are two data areas,
// there are two of these checksums. One is at block 0x08, offset 0x0A, and the other is at block 0x24, offset 0x0A.
// The type 3 checksum is a checksum of the next 4 blocks after the type 2 checksum
// (EXCLUDING the data area header, and the access control blocks), and then 0x0E blocks of zeroes.
*checksumOffset = 0x0A;
*dataOffset = 0x50;
*dataLength = 0x40;
break;
default:
return false;
}
return true;
}
bool Checksum::computeChecksum(int type, void const* memoryIn, unsigned short* checksum)
{
unsigned int startBlock;
unsigned int cntBlock;
unsigned int block;
unsigned char const* numPtr = (unsigned char const*)memoryIn;
Crypt crypt;
if ((type == 0) || (type == 1))
{
unsigned int dataLength;
unsigned int dataOffset;
unsigned int checksumOffset;
if (!getChecksumParameters(type, &checksumOffset, &dataOffset, &dataLength))
{
return false;
}
if (type == 1)
{
unsigned char header[0x10];
memcpy(header, (void const*) (numPtr + dataOffset), 0x10);
*(header + 14) = 5;
*(header + 15) = 0;
*checksum = ComputeCcittCrc16((void const*) &header, dataLength);
}
else
{
*checksum = ComputeCcittCrc16((void const*) (numPtr + dataOffset), dataLength);
}
return true;
}
if (type == 2)
{
startBlock = 1;
cntBlock = 4;
}
else if (type == 3)
{
startBlock = 5;
cntBlock = 4;
}
else
{
return false;
}
numPtr += (startBlock * 0x10);
*checksum = 0xffff;
block = startBlock;
while(true)
{
if (block >= (startBlock + cntBlock))
{
if (type != 3)
{
return true;
}
block = startBlock + cntBlock;
break;
}
if (!crypt.IsAccessControlBlock(block))
{
for (unsigned int i = 0; i < 0x10; i++)
{
*checksum = UpdateCcittCrc16(*checksum, *((unsigned char*) (numPtr + i)));
}
}
numPtr += 0x10;
block++;
}
// Pad Type 3 checksum with 0x0E blocks of zeroes
while (block < 0x1c)
{
if (!crypt.IsAccessControlBlock(block))
{
for (unsigned int j = 0; j < 0x10; j++)
{
*checksum = UpdateCcittCrc16(*checksum, 0);
}
}
block++;
}
return true;
}
// validateChecksum
// buffer: pointer to entire decrypted character as single chunk of memory
// type: checksum type
// dataArea: dataArea to validate.
// A value of 0 indicates the first data area starting at 0x08
// A value of 1 indicates the second data area starting at 0x24
// overwrite: if true, replace checksum in buffer with newly computed checksum
//
// returns true if old checksum in buffer matches computed checksum.
bool Checksum::validateChecksum(unsigned char *buffer, int type, int dataArea, bool overwrite)
{
unsigned int checksumOffset;
unsigned int areaSequenceOffset;
unsigned short computedChecksum;
unsigned int dataLength;
unsigned int dataOffset;
unsigned int offset = 0;
unsigned char *ptr;
bool match;
if (!getChecksumParameters(type, &checksumOffset, &dataOffset, &dataLength))
{
return false;
}
if (type != 0)
{
int dataAreaBlock;
if (dataArea == 0)
{
dataAreaBlock = 0x08;
}
else
{
dataAreaBlock = 0x24;
}
offset += (unsigned int) (dataAreaBlock * 0x10);
}
ptr = buffer + offset;
if(overwrite && type == 1) {
// Before computing checksum 1 (and after computing checksum 2 and 3)
// update sequence number.
areaSequenceOffset = 0x09;
ptr[areaSequenceOffset]++; // increment sequence
}
if (!computeChecksum(type, ptr, &computedChecksum))
{
return false;
}
unsigned short oldChecksum = (unsigned short) ((ptr[checksumOffset] & 0xff) | ((ptr[(int) (checksumOffset + 1)] & 0xff) << 8));
match = (oldChecksum == computedChecksum);
if(overwrite) {
// overwrite old value with newly computed checksum
ptr[checksumOffset] = computedChecksum & 0xff;
ptr[checksumOffset+1] = (computedChecksum >> 8) & 0xff;
}
return match;
}
bool Checksum::ValidateAllChecksums(unsigned char *buffer, bool overwrite)
{
bool OK = true;
bool res;
int dataArea;
int type;
// When computing checksums for overwrite, they have to be done in the following order.
// Compute checksum 3 and 2, then increment the area sequence number by 1,
// then compute checksum 1.
//
// In the logic below, the area sequence number is set just prior to computing checksum 1.
for(dataArea = 0; dataArea <= 1; dataArea++) {
for(type = 3; type >=0; type--) {
res = validateChecksum(buffer, type, dataArea, overwrite);
if(!res && !overwrite) {
fprintf(stderr, "Checksum failure for checksum type %d, data area %d", type, dataArea);
}
OK = OK && res;
}
}
return OK;
}