forked from silicontrip/SkyReader
-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathskylander.cpp
452 lines (365 loc) · 15.1 KB
/
skylander.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
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
#include "skylander.h"
/*
character data contents
Even though there are two "data areas" (headers at blocks 0x08 and 0x24, data starts at blocks 0x09 and 0x25), some data is stored outside of the area, so here's a breakdown of the whole 1KB:
Block Block Offset Size Description
Area 0 Area 1 (bytes)
0x00 N/A 0x00 0x02 Unique serial number for the toy.
0x00 N/A 0x04 0x0E Unknown.
0x01 N/A 0x00 0x02 Identifier for the character/toy type. In the dump above, you can see it's 0E 00 (Little Endian), or 0x000E (Gill Grunt).
0x01 N/A 0x04 0x08 Trading card ID.
0x01 N/A 0x0C 0x02 Unknown. Zeroes for me.
0x01 N/A 0x0E 0x02 Type 0 CRC16 checksum.
0x08 0x24 0x00 0x03 24-bit experience/level value. Maximum unknown. Set this really high to max out the level.
0x08 0x24 0x03 0x02 16-bit money value. Maximum 65000. Set it higher and the game rounds down to 65000.
0x08 0x24 0x05 0x02 Unknown.
0x08 0x24 0x07 0x02 Unknown. Zeroes for me.
0x08 0x24 0x09 0x01 8-bit sequence value for this data area. I'm not totally sure how it works yet, but I think the area with the higher value is the "primary" one at the moment.
0x08 0x24 0x0A 0x02 Type 3 CRC16 checksum.
0x08 0x24 0x0C 0x02 Type 2 CRC16 checksum.
0x08 0x24 0x0E 0x02 Type 1 CRC16 checksum.
0x09 0x25 0x00 0x04 Skills given by Fairy. Bit 7 = path chosen. FD0F = Left, FF0F = Right
0x09 0x25 0x01 0x02 Unknown. Zeroes for me.
0x09 0x25 0x03 0x01 8-bit value, bitmap of platforms the character has touched. Bit 0 is the Wii and bit 1 is the Xbox 360, evidently.
0x09 0x25 0x04 0x02 ID of hat the character is currently wearing.
0x09 0x25 0x06 0x02 Unknown. Zeroes for me.
0x09 0x25 0x08 0x08 Unknown. I've seen FF BF 1B 7F FF 2F B9 7E and FF 83 EE 7E FF 19 30 7F.
0x0A 0x26 0x00 0x10 First half of Unicode name of character, zero-terminated, maximum 14 characters.
0x0C 0x28 0x00 0x10 Second half of Unicode name of character, zero-terminated, maximum 14 characters.
0x0D 0x29 0x00 0x0A Unknown.
0x0D 0x29 0x0A 0x02 16-bit hero points value. Maximum 100.
0x0D 0x29 0x0C 0x03 Unknown. Zeroes for me.
0x0D 0x29 0x0E 0x01 Unknown. 01 for me.
0x10 0x2C 0x00 0x0C Unknown. Zeroes for me.
0x10 0x2C 0x0C 0x04 32 bit flag value indicating heroic challenges completed.
*/
void Skylander::fprinthex(FILE *f, unsigned char *c, unsigned int n) {
unsigned int h,i;
unsigned char j;
for (h=0; h<n; h+=16) {
fprintf (f,"%04x: ",h);
for (i=0; i<16; i++) {
if (i+h < n)
fprintf (f,"%02x ",*(c+i+h) & 0xff);
else
fprintf (f," ");
}
for (i=0; i<16; i++) {
if (i+h < n) {
j = *(c+i+h);
if (j<32) j='.';
if (j>=127) j='.';
fprintf (f,"%c",j);
} else
fprintf(f," ");
}
fprintf(f,"\n");
}
}
void Skylander::dump(void)
{
fprinthex(stdout,data,SKYLANDER_SIZE);
}
Skylander::Skylander(unsigned char *in)
{
initSkylander(in);
}
void Skylander::initSkylander(unsigned char *in)
{
memcpy (data,in,SKYLANDER_SIZE);
setAreaFromSequence();
readName();
}
unsigned char *Skylander::getData() { return data; }
unsigned char Skylander::getByte (int block, int offset) { return data[block * 16 + offset]; }
unsigned short Skylander::getShort (int block, int offset) { return data[block * 16 + offset] + data[block * 16 + offset + 1] * 0x100; }
void Skylander::setByte(int block, int offset, unsigned char b) { data[block * 16 + offset] = b; }
void Skylander::setShort(int block, int offset, unsigned short b)
{
data[block * 16 + offset] = b & 0xff;
data[block * 16 + offset + 1] = (b & 0xff00) / 0x100;
}
void Skylander::readName()
{
int block = getBlockNumberForArea() + 2;
for (int i=0; i<15; i++)
{
int offset = (i * 2) & 0xf;
if (i == 8) { block += 2; }
unsigned short utf = getShort(block,offset);
name[i] = utf & 0xff;
}
}
void Skylander::setAreaFromSequence()
{
area = 0;
if (getArea0Sequence() < getArea1Sequence()) { area = 1;}
}
int Skylander::getBlockNumberForArea() { return area * 28 + 8; }
// should validate for 0 or 1
void Skylander::setArea(int a) { if (a == 1 || a == 0) {area = a;} }
int Skylander::getArea() { return area; }
unsigned short Skylander::getSerial() { return getShort(0,0); }
const char * Skylander::toyName(int toy) {
switch (toy) {
case kTfbSpyroTag_ToyType_Character_Bash: return "Bash";
case kTfbSpyroTag_ToyType_Character_Boomer: return "Boomer";
case kTfbSpyroTag_ToyType_Character_Camo: return "Camo";
case kTfbSpyroTag_ToyType_Character_ChopChop: return "ChopChop";
case kTfbSpyroTag_ToyType_Character_Cynder: return "Cynder";
case kTfbSpyroTag_ToyType_Character_DarkSpyro: return "DarkSpyro";
case kTfbSpyroTag_ToyType_Character_DinoRang: return "DinoRang";
case kTfbSpyroTag_ToyType_Character_DoubleTrouble: return "DoubleTrouble";
case kTfbSpyroTag_ToyType_Character_DrillSergeant: return "DrillSergeant";
case kTfbSpyroTag_ToyType_Character_Drobot: return "Drobot";
case kTfbSpyroTag_ToyType_Character_Eruptor: return "Eruptor";
case kTfbSpyroTag_ToyType_Character_Flameslinger: return "Flameslinger";
case kTfbSpyroTag_ToyType_Character_GhostRoaster: return "GhostRoaster";
case kTfbSpyroTag_ToyType_Character_GillGrunt: return "GillGrunt";
case kTfbSpyroTag_ToyType_Character_Hex: return "Hex";
case kTfbSpyroTag_ToyType_Character_Ignitor: return "Ignitor";
case kTfbSpyroTag_ToyType_Character_LightningRod: return "LightningRod";
case kTfbSpyroTag_ToyType_Character_PrismBreak: return "PrismBreak";
case kTfbSpyroTag_ToyType_Character_SlamBam: return "SlamBam";
case kTfbSpyroTag_ToyType_Character_SonicBoom: return "SonicBoom";
case kTfbSpyroTag_ToyType_Character_Spyro: return "Spyro";
case kTfbSpyroTag_ToyType_Character_StealthElf: return "StealthElf";
case kTfbSpyroTag_ToyType_Character_StumpSmash: return "StumpSmash";
case kTfbSpyroTag_ToyType_Character_Sunburn: return "Sunburn";
case kTfbSpyroTag_ToyType_Character_Terrafin: return "Terrafin";
case kTfbSpyroTag_ToyType_Character_TriggerHappy: return "TriggerHappy";
case kTfbSpyroTag_ToyType_Character_Voodood: return "Voodood";
case kTfbSpyroTag_ToyType_Character_Warnado: return "Warnado";
case kTfbSpyroTag_ToyType_Character_WhamShell: return "WhamShell";
case kTfbSpyroTag_ToyType_Character_Whirlwind: return "Whirlwind";
case kTfbSpyroTag_ToyType_Character_WreckingBall: return "WreckingBall";
case kTfbSpyroTag_ToyType_Character_Zap: return "Zap";
case kTfbSpyroTag_ToyType_Character_Zook: return "Zook";
case kTfbSpyroTag_ToyType_Expansion_Dragon: return "Dragon's Peak";
case kTfbSpyroTag_ToyType_Expansion_Ice: return "Empire of Ice";
case kTfbSpyroTag_ToyType_Expansion_Pirate: return "Pirate Seas";
case kTfbSpyroTag_ToyType_Expansion_PVPUnlock: return "PVPUnlock";
case kTfbSpyroTag_ToyType_Expansion_Undead: return "Undead";
case kTfbSpyroTag_ToyType_Item_Anvil: return "Anvil Rain";
case kTfbSpyroTag_ToyType_Item_CrossedSwords: return "CrossedSwords";
case kTfbSpyroTag_ToyType_Item_Hourglass: return "Hourglass";
case kTfbSpyroTag_ToyType_Item_Regeneration: return "Regeneration";
case kTfbSpyroTag_ToyType_Item_SecretStash: return "SecretStash";
case kTfbSpyroTag_ToyType_Item_Shield: return "Shield";
case kTfbSpyroTag_ToyType_Item_Sparx: return "Sparx";
case kTfbSpyroTag_ToyType_Item_SpeedBoots: return "SpeedBoots";
case kTfbSpyroTag_ToyType_LEGENDARY: return "LEGENDARY";
case kTfbSpyroTag_ToyType_Legendary_Bash: return "Bash";
case kTfbSpyroTag_ToyType_Legendary_ChopChop: return "ChopChop";
case kTfbSpyroTag_ToyType_Legendary_Spyro: return "Spyro";
case kTfbSpyroTag_ToyType_Legendary_TriggerHappy: return "TriggerHappy";
case kTfbSpyroTag_ToyType_PET: return "PET";
case kTfbSpyroTag_ToyType_Pet_GillGrunt: return "GillGrunt";
case kTfbSpyroTag_ToyType_Pet_StealthElf: return "StealthElf";
case kTfbSpyroTag_ToyType_Pet_Terrafin: return "Terrafin";
case kTfbSpyroTag_ToyType_Pet_TriggerHappy: return "TriggerHappy";
// giants
case kTfbGiantsTag_ToyType_Giant_TreeRex: return "TreeRex";
case kTfbGiantsTag_ToyType_Giant_GnarlyTreeRex: return "GnarlyTreeRex";
case kTfbGiantsTag_ToyType_Giant_Bouncer: return "Bouncer";
case kTfbGiantsTag_ToyType_Giant_LegendaryBouncer: return "LegendaryBouncer";
case kTfbGiantsTag_ToyType_Giant_Crusher: return "Crusher";
case kTfbGiantsTag_ToyType_Giant_GraniteCrusher: return "GraniteCrusher";
case kTfbGiantsTag_ToyType_Giant_Swarm: return "Swarm";
case kTfbGiantsTag_ToyType_Giant_HotHead: return "HotHead";
case kTfbGiantsTag_ToyType_Giant_EyeBrawl: return "EyeBrawl";
case kTfbGiantsTag_ToyType_Giant_Ninjini: return "Ninjini/ScarletNinjini";
case kTfbGiantsTag_ToyType_Giant_Thumpback: return "Thumpback";
default: return "UNKNOWN";
}
}
unsigned short Skylander::getToyType() { return getShort(1,0); }
const char * Skylander::getToyTypeName() { return toyName(getToyType()); }
unsigned char * Skylander::getTradingID()
{
return data + 20;
}
unsigned int Skylander::getXP()
{
int block = getBlockNumberForArea();
return getByte(block,0) + getByte(block,1) * 0x100 + getByte(block,2) * 0x10000;
}
void Skylander::setXP(unsigned int xp)
{
if (xp < 0x1000000 )
{
int block = getBlockNumberForArea();
setByte(block,0, xp & 0xff);
setByte(block,1, (xp & 0xff00) / 0x100);
setByte(block,2, (xp & 0xff0000) / 0x10000);
}
}
unsigned short Skylander::getMoney()
{
int block = getBlockNumberForArea();
return getShort(block, 3);
}
void Skylander::setMoney(unsigned short money)
{
int block = getBlockNumberForArea();
setShort(block,3,money);
}
// apparently the largest here gives the area.
unsigned char Skylander::getArea0Sequence() { return getByte(8, 9); }
unsigned char Skylander::getArea1Sequence() { return getByte(24, 9); }
unsigned short Skylander::getSkill() {
int block = getBlockNumberForArea();
return getShort(block+1, 0);
}
void Skylander::setSkillLeft(unsigned short skill)
{
skill = skill | 0x2;
setSkill(skill);
}
void Skylander::setSkillRight(unsigned short skill)
{
skill = skill & 0xfffd;
setSkill(skill);
}
void Skylander::setSkill(unsigned short skill)
{
int block = getBlockNumberForArea();
setShort(block+1, 0, skill);
}
unsigned char Skylander::getPlatform()
{
int block = getBlockNumberForArea();
return getByte(block+1, 3);
}
const char * Skylander::getPlatformName()
{
char platform = getPlatform();
if ( (platform & 1) == 1) return "Wii";
if ( (platform & 2) == 2) return "Xbox 360";
if ( (platform & 4) == 4) return "PS3";
return "UNKNOWN";
}
unsigned short Skylander::getHat()
{
int block = getBlockNumberForArea();
return getShort(block+1, 4);
}
void Skylander::setHat(unsigned short hat)
{
int block = getBlockNumberForArea();
return setShort(block+1, 4,hat);
}
char * Skylander::getName()
{
return name;
}
unsigned short Skylander::getHeroPoints()
{
int block = getBlockNumberForArea();
return getShort(block+5, 0xA);
}
void Skylander::setHeroPoints(unsigned short hp)
{
int block = getBlockNumberForArea() + 5;
setShort(block,0xA,hp);
}
unsigned int Skylander::getHeroicChallenges()
{
int block = getBlockNumberForArea() + 8;
return getByte(block , 0xc) + getByte(block , 0xd) * 0x100 + getByte(block , 0xe) * 0x10000 + getByte(block , 0xf) * 0x1000000;
}
void Skylander::setHeroicChallenges(unsigned int hc)
{
int block = getBlockNumberForArea() + 8 ;
setByte(block , 0xc, hc & 0xff);
setByte(block , 0xd, (hc & 0xff00) / 0x100);
setByte(block , 0xe, (hc & 0xff0000) / 0x10000);
setByte(block , 0xf, (hc & 0xff000000) / 0x1000000);
}
void Skylander::UpdateBuf( int block, int offset, int size, unsigned char val)
{
unsigned char *ptr = data;
ptr += block * 0x10 + offset;
for(int i=0; i<size; i++) {
ptr[i] = val;
}
}
bool Skylander::validateChecksum()
{
// want to put the checksum files into a class.
return crc.ValidateAllChecksums(data,false);
}
void Skylander::computeChecksum()
{
// want to put the checksum files into a class.
crc.ValidateAllChecksums(data,true);
}
// Write bytes to buffer
static void UpdateBuf(unsigned char *buffer, int block, int offset, int size, unsigned char val)
{
unsigned char *ptr = buffer;
ptr += block * 0x10 + offset;
for(int i=0; i<size; i++) {
ptr[i] = val;
}
}
void Skylander::MaxXP()
{
UpdateBuf( 0x08, 0x00, 0x03, 0xFF); // Max XP, data area 0
UpdateBuf( 0x24, 0x00, 0x03, 0xFF); // Max XP, data area 1
}
void Skylander::MaxMoney()
{
UpdateBuf( 0x08, 0x03, 0x02, 0xFF); // Max money, data area 0
UpdateBuf( 0x24, 0x03, 0x02, 0xFF); // Max money, data area 1
}
void Skylander::MaxHeroPoints()
{
UpdateBuf( 0x0D, 0x0A, 0x01, 0x64); // Max hero points, data area 0
UpdateBuf( 0x29, 0x0A, 0x01, 0x64); // Max hero points, data area 1
}
void Skylander::MaxHeroicChallenges()
{
UpdateBuf( 0x0D, 0x06, 0x04, 0xFF); // Mark all heroic challenges as complete, data area 0
UpdateBuf( 0x29, 0x06, 0x04, 0xFF); // Mark all heroic challenges as complete, data area 1
}
void Skylander::MaxSkills(unsigned char path)
{
if (path == 0xFF || path == 0xFD)
{
// Upgrade all skills, data area 0
UpdateBuf( 0x09, 0x00, 0x01, path);
UpdateBuf( 0x09, 0x01, 0x01, 0x0F);
// Upgrade all skills, data area 1
UpdateBuf( 0x25, 0x00, 0x01, path);
UpdateBuf( 0x25, 0x01, 0x01, 0x0F);
}
}
// Update Skylander character with common upgrades
void MaxStats(unsigned char *buffer, char skill_path) {
unsigned char path;
// UpdateBuf(buffer, block, offset, size, val)
UpdateBuf(buffer, 0x08, 0x00, 0x03, 0xFF); // Max XP, data area 0
UpdateBuf(buffer, 0x24, 0x00, 0x03, 0xFF); // Max XP, data area 1
UpdateBuf(buffer, 0x08, 0x03, 0x02, 0xFF); // Max money, data area 0
UpdateBuf(buffer, 0x24, 0x03, 0x02, 0xFF); // Max money, data area 1
UpdateBuf(buffer, 0x0D, 0x0A, 0x01, 0x64); // Max hero points, data area 0
UpdateBuf(buffer, 0x29, 0x0A, 0x01, 0x64); // Max hero points, data area 1
UpdateBuf(buffer, 0x0D, 0x06, 0x04, 0xFF); // Mark all heroic challenges as complete, data area 0
UpdateBuf(buffer, 0x29, 0x06, 0x04, 0xFF); // Mark all heroic challenges as complete, data area 1
if(skill_path != 0) {
if(skill_path == '-') {
// Choose left skills upgrade path
path = 0xFD;
} else {
// Choose right skills upgrade path
path = 0xFF;
}
// Upgrade all skills, data area 0
UpdateBuf(buffer, 0x09, 0x00, 0x01, path);
UpdateBuf(buffer, 0x09, 0x01, 0x01, 0x0F);
// Upgrade all skills, data area 1
UpdateBuf(buffer, 0x25, 0x00, 0x01, path);
UpdateBuf(buffer, 0x25, 0x01, 0x01, 0x0F);
}
}