-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathTZX_Blocks.h
558 lines (509 loc) · 17.5 KB
/
TZX_Blocks.h
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
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
#ifndef __TZX_BLOCKS_H__
#define __TZX_BLOCKS_H__
#include "types.h"
#include "ByteBuffer.h"
using namespace std;
using namespace Utility;
// TZX general defines
#define Z80HZ ((uint32_t)3500000) // ZX Standard 3.5 Mhz
// TZX/TSX BLOCKS
#define B10_STD_BLOCK 0x10
#define B11_TURBO_BLOCK 0x11
#define B12_PURE_TONE 0x12
#define B13_PULSE_SEQ 0x13
#define B14_PURE_DATA 0x14
#define B15_DIRECT_REC 0x15
#define B18_CSW_REC 0x18
#define B19_GEN_DATA 0x19
#define B20_SILENCE_BLOCK 0x20
#define B21_GRP_START 0x21
#define B22_GRP_END 0x22
#define B23_JUMP_BLOCK 0x23
#define B24_LOOP_START 0x24
#define B25_LOOP_END 0x25
#define B26_CALL_SEQ 0x26
#define B27_RET_SEQ 0x27
#define B28_SEL_BLOCK 0x28
#define B2A_STOP48K 0x2A
#define B2B_SIGNAL_LEVEL 0x2B
#define B30_TEXT_DESCRIP 0x30
#define B31_MSG_BLOCK 0x31
#define B32_ARCHIVE_INFO 0x32
#define B33_HARDWARE_TYPE 0x33
#define B35_CUSTOM_INFO 0x35
#define B4B_MSX_KCS 0x4B
#define B5A_GLUE_BLOCK 0x5A
// DEPRECATED BLOCKS
#define B16_C64ROM 0x16
#define B17_C64TURBO 0x17
#define B34_EMUINFO 0x34
#define B40_SNAPSHOT 0x40
namespace TZX_Blocks
{
// ============================================================================================
// Class Block
class Block
{
public:
Block();
Block(const Block& other);
~Block();
size_t getSize();
BYTE getId();
BYTE getHeadSize();
const char* getBytes();
void hexDump();
void hexCharDump();
virtual string toString();
Block& operator=(const Block &);
protected:
void init(BYTE, BYTE, char*, size_t);
void putByte(BYTE);
void putWord(WORD);
void putWord24(WORD24);
void putDWord(DWORD);
void putString(string);
void put(char*, size_t);
void put(istream &, size_t);
void put(ByteBuffer*);
protected:
ByteBuffer *bytes;
private:
BYTE id;
BYTE headSize; //Header size before block data (if any)
};
// ============================================================================================
// Class BlockWithPause
class BlockWithPause : public Block
{
protected:
virtual const uint8_t getPausePos() = 0;
public:
WORD getPause();
void setPause(WORD);
void addPause(WORD);
};
// ============================================================================================
// Block #10 - Standard speed data block
// 0x00 - WORD Pause after this block (ms.) {1000}
// 0x02 N WORD Length of data that follow
// 0x04 - BYTE[N] Data as in .TAP files
class Block10 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 1; };
public:
Block10(WORD pause, char* data, size_t size);
Block10(istream &);
BYTE getCurrentChecksum();
BYTE getChecksum(WORD, WORD);
bool checksumOk();
string toString() override;
};
// ============================================================================================
// Block #11 - Turbo speed data block
// 0x00 - WORD Length of PILOT pulse {2168}
// 0x02 - WORD Length of SYNC first pulse {667}
// 0x04 - WORD Length of SYNC second pulse {735}
// 0x06 - WORD Length of ZERO bit pulse {855}
// 0x08 - WORD Length of ONE bit pulse {1710}
// 0x0A - WORD Length of PILOT tone (number of pulses) {8063 header (flag<128), 3223 data (flag>=128)}
// 0x0C - BYTE Used bits in the last byte (other bits should be 0) {8}
// (e.g. if this is 6, then the bits used (x) in the last byte are: xxxxxx00, where MSb is the leftmost bit, LSb is the rightmost bit)
// 0x0D - WORD Pause after this block (ms.) {1000}
// 0x0F N BYTE[3] Length of data that follow
// 0x12 - BYTE[N] Data as in .TAP files
class Block11 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 14; };
public:
Block11(WORD pilotlen, WORD synclen1, WORD synclen2, WORD bit0len, WORD bit1len, WORD pilotnum, BYTE rbits, WORD pause, char *data, size_t size);
Block11(istream &);
string toString() override;
};
// ============================================================================================
// Block #12 - Pure tone
// 0x00 - WORD Length of one pulse in T-states
// 0x02 - WORD Number of pulses
class Block12 : public Block
{
public:
Block12(WORD pulselen, WORD pulsenum);
Block12(istream &);
string toString() override;
};
// ============================================================================================
// Block #13 - Sequence of pulses of various lengths
// 0x00 N BYTE Number of pulses
// 0x01 - WORD[N] Pulses' lengths
class Block13 : public Block
{
public:
Block13(BYTE pulsenum, WORD *lengths);
Block13(istream &);
string toString() override;
};
// ============================================================================================
// Block #14 - Pure data block
// 0x00 - WORD Length of ZERO bit pulse
// 0x02 - WORD Length of ONE bit pulse
// 0x04 - BYTE Used bits in last byte (other bits should be 0)
// (e.g. if this is 6, then the bits used (x) in the last byte are: xxxxxx00, where MSb is the leftmost bit, LSb is the rightmost bit)
// 0x05 - WORD Pause after this block (ms.)
// 0x07 N BYTE[3] Length of data that follow
// 0x0A - BYTE[N] Data as in .TAP files
class Block14 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 6; };
public:
Block14(WORD bit0len, WORD bit1len, BYTE rbits, WORD pause, char *data, size_t size);
Block14(istream &);
string toString() override;
};
// ============================================================================================
// Block #15 - Direct recording block
// 0x00 - WORD Number of T-states per sample (bit of data)
// 0x02 - WORD Pause after this block in milliseconds (ms.)
// 0x04 - BYTE Used bits (samples) in last byte of data (1-8)
// (e.g. if this is 2, only first two samples of the last byte will be played)
// 0x05 N BYTE[3] Length of samples' data
// 0x08 - BYTE[N] Samples data. Each bit represents a state on the EAR port (i.e. one sample).
// MSb is played first.
class Block15 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 3; };
public:
Block15(WORD numstates, WORD pause, BYTE rbits, char *data, size_t size);
Block15(istream &);
string toString() override;
};
// ============================================================================================
// Block #20 - Pause (silence) or 'Stop the tape' command
// 0x00 - WORD Pause duration (ms.)
class Block20 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 1; };
public:
Block20(WORD pause);
Block20(istream &);
string toString() override;
};
// ============================================================================================
// Block #21 - Group start
// 0x00 L BYTE Length of the group name string
// 0x01 - CHAR[L] Group name in ASCII format (please keep it under 30 characters long)
class Block21 : public Block
{
public:
Block21(string nameGroup);
Block21(istream &);
string toString() override;
};
// ============================================================================================
// Block #22 - Group end
class Block22 : public Block
{
public:
Block22();
Block22(istream &);
string toString() override;
};
// ============================================================================================
// Block #23 - Jump to block
// 0x00 - WORD Relative jump value
class Block23 : public Block
{
public:
Block23(WORD jump);
Block23(istream &);
string toString() override;
};
// ============================================================================================
// Block #24 - Loop start
// 0x00 - WORD Number of repetitions (greater than 1)
class Block24 : public Block
{
public:
Block24(WORD numRepeats);
Block24(istream &);
string toString() override;
};
// ============================================================================================
// Block #25 - Loop end
class Block25 : public Block
{
public:
Block25();
Block25(istream &);
string toString() override;
};
// ============================================================================================
// Block #26 - Call sequence
// 0x00 N WORD Number of calls to be made
// 0x02 - WORD[N] Array of call block numbers (relative-signed offsets)
class Block26 : public Block
{
public:
Block26(WORD num, WORD *calls);
Block26(istream &);
string toString() override;
};
// ============================================================================================
//Block #27 - Return from sequence
class Block27 : public Block
{
public:
Block27();
Block27(istream &);
string toString() override;
};
// ============================================================================================
// Block #28 - Select block
// 0x00 - WORD Length of the whole block (without these two bytes)
// 0x02 N BYTE Number of selections
// 0x03 - SELECT[N] List of selections
//
// SELECT structure format
// 0x00 - WORD Relative Offset
// 0x02 L BYTE Length of description text
// 0x03 - CHAR[L] Description text (please use single line and max. 30 chars)
class Block28 : public Block
{
public:
Block28(BYTE num, WORD *offsets, string *texts);
Block28(istream &);
string toString() override;
};
// ============================================================================================
// Block #2A - Stop the tape if in 48K mode
// 0x00 0 DWORD Length of the block without these four bytes (0)
class Block2A : public Block
{
public:
Block2A();
Block2A(istream &);
string toString() override;
};
// ============================================================================================
// ID 2B - Set signal level
// 0x00 1 DWORD Block length (without these four bytes)
// 0x04 - BYTE Signal level (0=low, 1=high)
class Block2B : public Block
{
public:
Block2B();
Block2B(istream &);
string toString() override;
};
// ============================================================================================
// Block #30 - Text description
// 0x00 N BYTE Length of the text description
// 0x01 - CHAR[N] Text description in ASCII format
class Block30 : public Block
{
public:
Block30(string text);
Block30(istream &);
string toString() override;
};
// ============================================================================================
// Block #31 - Message block
// 0x00 - BYTE Time (in seconds) for which the message should be displayed
// 0x01 N BYTE Length of the text message
// 0x02 - CHAR[N] Message that should be displayed in ASCII format
class Block31 : public Block
{
public:
Block31(BYTE seconds, string text);
Block31(istream &);
string toString() override;
};
// ============================================================================================
// Block #32 - Archive info
// 0x00 - WORD Length of the whole block (without these two bytes)
// 0x02 N BYTE Number of text strings
// 0x03 - TEXT[N] List of text strings
//
// TEXT structure format
// 0x00 - BYTE Text identification byte:
// 00 - Full title
// 01 - Software house/publisher
// 02 - Author(s)
// 03 - Year of publication
// 04 - Language
// 05 - Game/utility type
// 06 - Price
// 07 - Protection scheme/loader
// 08 - Origin
// FF - Comment(s)
// 0x01 L BYTE Length of text string
// 0x02 - CHAR[L] Text string in ASCII format
class Block32 : public Block
{
public:
Block32(BYTE num, BYTE *ids, string *texts);
Block32(istream &);
string toString() override;
};
// ============================================================================================
// Block #33 - Hardware type
// 0x00 N BYTE Number of machines and hardware types for which info is supplied
// 0x01 - HWINFO[N] List of machines and hardware
//
// HWINFO structure format
// 0x00 - BYTE Hardware type
// 0x01 - BYTE Hardware ID
// 0x02 - BYTE Hardware information:
// 00 - The tape RUNS on this machine or with this hardware, but may or may not use the hardware or special features of the machine.
// 01 - The tape USES the hardware or special features of the machine, such as extra memory or a sound chip.
// 02 - The tape RUNS but it DOESN'T use the hardware or special features of the machine.
// 03 - The tape DOESN'T RUN on this machine or with this hardware.
class Block33 : public Block
{
public:
Block33(BYTE num, BYTE *hwtype, BYTE *hwid, BYTE *hwinfo);
Block33(istream &);
string toString() override;
};
// ============================================================================================
// ID 35 - Custom info block
// 0x00 - CHAR[10] Identification string (in ASCII)
// 0x10 L DWORD Length of the custom info
// 0x14 - BYTE[L] Custom info
class Block35 : public Block
{
public:
Block35(string label, string info);
Block35(istream &);
string toString() override;
};
// ============================================================================================
// Block #4B - Kansas City Standard
// 0x00 N+12 DWORD Block length without these four bytes (extension rule)
// 0x04 - WORD Pause after this block in milliseconds
// 0x06 - WORD Duration of a PILOT pulse in T-states {same as ONE pulse}
// 0x08 - WORD Number of pulses in the PILOT tone
// 0x0A - WORD Duration of a ZERO pulse in T-states
// 0x0C - WORD Duration of a ONE pulse in T-states
// 0x0E - BYTE Bits 7-4: Number of ZERO pulses in a ZERO bit (0-16) {2}
// (bitmapped) Bits 3-0: Number of ONE pulses in a ONE bit (0-16) {4}
// 0x0F - BYTE Bits 7-6: Numbers of leading bits {1}
// (bitmapped) Bit 5: Value of leading bits {0}
// Bits 4-3: Number of trailing bits {2}
// Bit 2: Value of trailing bits {1}
// Bit 1: Reserved
// Bit 0: Endianless (0 for LSb first, 1 for MSb first) {0}
// 0x10 - BYTE[N] Data stream
class Block4B : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 5; };
public:
Block4B(WORD pause, WORD pilot, WORD pulses, WORD bit0len, WORD bit1len, BYTE bitcfg, BYTE bytecfg, char *data, size_t size);
Block4B(istream &);
BYTE getFileType();
string getFileTypeLoad();
string getFileTypeDescription();
string toString() override;
string toString(bool);
};
// ============================================================================================
// Block #5A - "Glue" block (90 dec, ASCII Letter 'Z')
// 0x00 - BYTE[9] Value: { "XTape!",0x1A,MajR,MinR }
// Just skip these 9 bytes and you will end up on the next ID.
class Block5A : public Block
{
public:
Block5A();
Block5A(istream &);
string toString() override;
};
//=================================
//NOT IMPLEMENTED BLOCKS
//=================================
// ============================================================================================
// Block #18 - CSW recording block
// 0x00 10+N DWORD Block length (without these four bytes)
// 0x04 - WORD Pause after this block (in ms).
// 0x06 - BYTE[3] Sampling rate
// 0x09 - BYTE Compression type
// 0x01: RLE
// 0x02: Z-RLE
// 0x0A - DWORD Number of stored pulses (after decompression, for validation purposes)
// 0x0E - BYTE[N] CSW data, encoded according to the CSW file format specification.
class Block18 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 5; };
public:
Block18();
Block18(istream &);
string toString() override;
};
// ============================================================================================
// Block #19 - Generalized data block
// 0x00 - DWORD Block length (without these four bytes)
// 0x04 - WORD Pause after this block (ms)
// 0x06 TOTP DWORD Total number of symbols in pilot/sync block (can be 0)
// 0x0A NPP BYTE Maximum number of pulses per pilot/sync symbol
// 0x0B ASP BYTE Number of pilot/sync symbols in the alphabet table (0=256)
// 0x0C TOTD DWORD Total number of symbols in data stream (can be 0)
// 0x10 NPD BYTE Maximum number of pulses per data symbol
// 0x11 ASD BYTE Number of data symbols in the alphabet table (0=256)
// 0x12 - SYMDEF[ASP] Pilot and sync symbols definition table
// This field is present only if TOTP>0
// 0x12+ - PRLE[TOTP] Pilot and sync data stream
// (2*NPP+1)*ASP This field is present only if TOTP>0
// 0x12+ - SYMDEF[ASD] Data symbols definition table
// (TOTP>0)*((2*NPP+1)*ASP)+ This field is present only if TOTD>0
// TOTP*3
// 0x12+ - BYTE[DS] Data stream
// (TOTP>0)*((2*NPP+1)*ASP)+ This field is present only if TOTD>0
// TOTP*3+
// (2*NPD+1)*ASD
class Block19 : public BlockWithPause
{
protected:
const uint8_t getPausePos() override { return 5; };
public:
Block19();
Block19(istream &);
string toString() override;
};
//=================================
// DEPRECATED BLOCKS
//=================================
// Block #16 - C64 ROM type data block
class Block16 : public Block {
public:
Block16();
Block16(istream &);
string toString() override;
};
// Block #17 - C64 turbo tape data block
class Block17 : public Block {
public:
Block17();
Block17(istream &);
string toString() override;
};
// Block #34 - Emulation info
class Block34 : public Block {
public:
Block34();
Block34(istream &);
string toString() override;
};
// Block #40 - Snapshot block
class Block40 : public Block {
public:
Block40();
Block40(istream &);
string toString() override;
};
}
#endif //__TZX_BLOCKS_H__