[go: up one dir, main page]

File: DMSDecompressor.cpp

package info (click to toggle)
ancient 2.2.0-1
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 1,168 kB
  • sloc: cpp: 15,188; makefile: 217; sh: 31
file content (675 lines) | stat: -rw-r--r-- 18,591 bytes parent folder | download
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
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
/* Copyright (C) Teemu Suutari */

#include <cstdint>
#include <cstring>

#include "DMSDecompressor.hpp"

#include "HuffmanDecoder.hpp"
#include "DynamicHuffmanDecoder.hpp"
#include "InputStream.hpp"
#include "OutputStream.hpp"
#include "VariableLengthCodeDecoder.hpp"
#include "common/MemoryBuffer.hpp"
#include "common/CRC16.hpp"
#include "common/OverflowCheck.hpp"
#include "common/Common.hpp"


namespace ancient::internal
{

bool DMSDecompressor::detectHeader(uint32_t hdr) noexcept
{
	return hdr==FourCC("DMS!");
}

std::shared_ptr<Decompressor> DMSDecompressor::create(const Buffer &packedData,bool exactSizeKnown,bool verify)
{
	return std::make_shared<DMSDecompressor>(packedData,verify);
}

DMSDecompressor::DMSDecompressor(const Buffer &packedData,bool verify) :
	_packedData{packedData}
{
	uint32_t hdr{packedData.readBE32(0)};
	if (!detectHeader(hdr) || packedData.size()<56)
		throw InvalidFormatError();

	if (verify && CRC16(packedData,4,50,0)!=packedData.readBE16(54))
		throw VerificationError();

	uint16_t info{packedData.readBE16(10)};
	_isObsfuscated=info&2;				// using 16 bit key is not encryption, it is obsfuscation
	_isHD=info&16;
	if (info&32)
		throw InvalidFormatError();		// MS-DOS disk

	// packed data in header is useless, as is rawsize and track numbers
	// they are not always filled

	if (packedData.readBE16(50)>6)
		throw InvalidFormatError();		// either FMS or unknown

	const std::array<uint32_t,7> contextSizes{0,0,256,16384,16384,4096,8192};

	// now calculate the real packed size, including headers
	uint32_t offset{56};
	uint32_t accountedSize{0};
	uint32_t lastTrackSize{0};
	uint32_t numTracks{0};
	uint32_t minTrack{80};
	uint32_t prevTrack{0};
	while (offset+20<packedData.size())
	{
		if (_packedData.readBE16(offset)!=MultiChar2("TR"))
		{
			// secondary exit criteria, should not be like this, if the header would be trustworthy
			if (!accountedSize)
				throw InvalidFormatError();
			break;
		}
		uint32_t trackNo{_packedData.readBE16(offset+2)};
		// lets not go backwards on tracks!
		if (trackNo<prevTrack) break;

		// header check
		if (verify && CRC16(packedData,offset,18,0)!=packedData.readBE16(offset+18))
			throw VerificationError();

		uint8_t mode{_packedData.read8(offset+13)};
		if (mode>6)
			throw InvalidFormatError();

		_contextBufferSize=std::max(_contextBufferSize,contextSizes[mode]);

		uint8_t flags{_packedData.read8(offset+12)};
		if ((mode>=2 && mode<=4) || (mode>=5 && (flags&4)))
			_tmpBufferSize=std::max(_tmpBufferSize,uint32_t(_packedData.readBE16(offset+8)));
		uint32_t packedChunkLength{packedData.readBE16(offset+6)};
		if (OverflowCheck::sum(offset,20U,packedChunkLength)>packedData.size())
			throw InvalidFormatError();
		if (verify && CRC16(packedData,offset+20,packedChunkLength,0)!=packedData.readBE16(offset+16))
			throw VerificationError();

		if (trackNo<80)
		{
			if (trackNo>=numTracks) lastTrackSize=_packedData.readBE16(offset+10); 
			minTrack=std::min(minTrack,trackNo);
			numTracks=std::max(numTracks,trackNo);
			prevTrack=trackNo;
		}

		offset+=packedChunkLength+20;
		accountedSize+=packedChunkLength;
		// this is the real exit critea, unfortunately
		if (trackNo>=79 && trackNo<0x8000U) break;
	}
	uint32_t trackSize{(_isHD)?22528U:11264U};
	_rawOffset=minTrack*trackSize;
	if (minTrack>=numTracks)
		throw InvalidFormatError();
	_minTrack=minTrack;
	_rawSize=(numTracks-minTrack)*trackSize+lastTrackSize;
	_imageSize=trackSize*80;

	_packedSize=offset;
	if (_packedSize>getMaxPackedSize())
		throw InvalidFormatError();
}

const std::string &DMSDecompressor::getName() const noexcept
{
	static std::string name{"DMS: Disk Masher System"};
	return name;
}

size_t DMSDecompressor::getPackedSize() const noexcept
{
	return _packedSize;
}

size_t DMSDecompressor::getRawSize() const noexcept
{
	return _rawSize;
}

size_t DMSDecompressor::getImageSize() const noexcept
{
	return _imageSize;
}

size_t DMSDecompressor::getImageOffset() const noexcept
{
	return _rawOffset;
}

void DMSDecompressor::decompressImpl(Buffer &rawData,bool verify)
{
	uint32_t restartPosition{0};
	if (!_isObsfuscated)
	{
		decompressImpl(rawData,verify,restartPosition);
	} else {
		while (restartPosition<0x20000U)
		{
			// more than single run here is really rare. It means that first track CRC succeeds
			// but later something else fails
			try
			{
				decompressImpl(rawData,verify,restartPosition);
				return;
			} catch (const Buffer::Error &) {
				// just continue
			} catch (const Decompressor::Error &) {
				// just continue
			}
			restartPosition++;
		}
		throw DecompressionError();
	}
}

// TODO: Too much state for a single method. too convoluted
// needs to be split
void DMSDecompressor::decompressImpl(Buffer &rawData,bool verify,uint32_t &restartPosition)
{
	if (rawData.size()<_rawSize)
		throw DecompressionError();
	MemoryBuffer contextBuffer{_contextBufferSize};
	MemoryBuffer tmpBuffer{_tmpBufferSize};
	uint32_t limitedDecompress{~0U};

	class ObsfuscatedInputStream
	{
	public:
		ObsfuscatedInputStream(const Buffer &data,uint32_t start,uint32_t length,bool obsfuscate) :
			_inputStream{data,start,OverflowCheck::sum(start,length)},
			_obsfuscate{obsfuscate}
		{
			// nothing needed
		}
		~ObsfuscatedInputStream() noexcept=default;

		void reset(size_t startOffset,size_t endOffset)
		{
			_inputStream.reset(startOffset,endOffset);
		}

		uint8_t readByte()
		{
			uint8_t ch{_inputStream.readByte()};
			if (!_obsfuscate)
			{
				return ch;
			} else {
				uint8_t ret{uint8_t(ch^_passAccumulator)};
				_passAccumulator=(_passAccumulator>>1)+uint16_t(ch);
				return ret;
			}
		}

		void setCode(uint16_t passAccumulator)
		{
			_passAccumulator=passAccumulator;
		}

		bool eof() const { return _inputStream.getOffset()==_inputStream.getEndOffset(); }

	private:
		ForwardInputStream	_inputStream;
		bool			_obsfuscate;
		uint16_t		_passAccumulator{0};
	};

	ObsfuscatedInputStream inputStream{_packedData,0,0,_isObsfuscated};
	MSBBitReader<ObsfuscatedInputStream> bitReader{inputStream};
	auto initInputStream=[&](uint32_t start,uint32_t length)
	{
		inputStream.reset(start,OverflowCheck::sum(start,length));
		bitReader.reset();
	};
	auto finishStream=[&]()
	{
		if (_isObsfuscated && limitedDecompress==~0U)
			while (!inputStream.eof())
				inputStream.readByte();
	};
	auto readBits=[&](uint32_t count)->uint32_t
	{
		return bitReader.readBits8(count);
	};
	auto readBit=[&]()->uint32_t
	{
		return bitReader.readBits8(1);
	};

	ForwardOutputStream outputStream{rawData,0,0};
	auto initOutputStream=[&](uint32_t start,uint32_t length)
	{
		outputStream.reset(start,OverflowCheck::sum(start,length));
	};

	// fill unused tracks with zeros
	std::memset(rawData.data(),0,_rawSize);

	bool doInitContext{true};
	// quick: context used is 256 bytes
	// medium & deep: context used is 16384 bytes
	// heavy: context used is 4096/8192 bytes
	uint32_t contextLocation;
	std::unique_ptr<DynamicHuffmanDecoder<314>> deepDecoder;
	auto initContext=[&]()
	{
		if (doInitContext)
		{
			if (_contextBufferSize) std::memset(contextBuffer.data(),0,_contextBufferSize);
			contextLocation=0;
			deepDecoder.reset();
			doInitContext=false;
		}
	};

	auto checksum=[](const Buffer &src,uint32_t offset,uint32_t srcLength)->uint16_t
	{
		uint16_t ret{0};
		for (uint32_t i=0;i<srcLength;i++) ret+=uint16_t(src[i+offset]);
		return ret;
	};

	auto unpackNone=[&](auto &output)
	{
		for (uint32_t i=0;i<limitedDecompress&&!output.eof();i++)
			output.writeByte(inputStream.readByte());
	};

	// same as simple
	auto unRLE=[&](auto &output,auto &input)
	{
		while (!output.eof())
		{
			if (output.getOffset()>=limitedDecompress) return;
			uint8_t ch{input.readByte()};
			uint32_t count{1};
			if (ch==0x90U)
			{
				count=input.readByte();
				if (!count) count=1;
					else ch=input.readByte();
				if (count==0xffU)
				{
					count=uint32_t(input.readByte())<<8;
					count|=input.readByte();
				}
			}
			for (uint32_t i=0;i<count;i++) output.writeByte(ch);
		}
	};

	auto unpackQuick=[&](auto &output)
	{
		initContext();

		while (!output.eof())
		{
			if (output.getOffset()>=limitedDecompress) return;
			if (readBits(1))
			{
				output.writeByte(contextBuffer[contextLocation++]=readBits(8));
				contextLocation&=0xffU;
			} else {
				uint32_t count{readBits(2)+2};
				uint8_t offset{uint8_t(contextLocation-readBits(8)-1)};
				for (uint32_t i=0;i<count;i++)
				{
					output.writeByte(contextBuffer[contextLocation++]=contextBuffer[(i+offset)&0xffU]);
					contextLocation&=0xffU;
				}
			}
		}
		contextLocation+=5;
		contextLocation&=0xffU;
	};


	VariableLengthCodeDecoder lengthDecoder{7,7,8,8,8,9,9,9,9,10,10,10,11,11,11,12};

	auto unpackMedium=[&](auto &output)
	{
		initContext();

		while (!output.eof())
		{
			if (output.getOffset()>=limitedDecompress) return;
			if (readBits(1))
			{
				output.writeByte(contextBuffer[contextLocation++]=readBits(8));
				contextLocation&=0x3fffU;
			} else {
				uint32_t count{lengthDecoder.decode(readBits,readBits(4U))};
				uint32_t rawDist{lengthDecoder.decode([&](uint32_t c)
				{
					return (uint32_t(count&0xfU)<<(c-4U))|readBits(c-4U);
				},((count>>4U)&0xfU))};
				count=(count>>8U)+3U;

				uint32_t offset{contextLocation-rawDist-1U};
				for (uint32_t i=0;i<count;i++)
				{
					output.writeByte(contextBuffer[contextLocation++]=contextBuffer[(i+offset)&0x3fffU]);
					contextLocation&=0x3fffU;
				}
			}
		}
		contextLocation+=66;
		contextLocation&=0x3fffU;
	};

	auto unpackDeep=[&](auto &output)
	{
		initContext();
		if (!deepDecoder) deepDecoder=std::make_unique<DynamicHuffmanDecoder<314>>();

		while (!output.eof())
		{
			if (output.getOffset()>=limitedDecompress) return;
			uint32_t symbol=deepDecoder->decode(readBit);
			if (deepDecoder->getMaxFrequency()==0x8000U) deepDecoder->halve();
			deepDecoder->update(symbol);
			if (symbol<256)
			{
				output.writeByte(contextBuffer[contextLocation++]=symbol);
				contextLocation&=0x3fffU;
			} else {
				uint32_t count{symbol-253};	// minimum repeat is 3
				uint32_t offset{contextLocation-lengthDecoder.decode(readBits,readBits(4U))-1U};

				for (uint32_t i=0;i<count;i++)
				{
					output.writeByte(contextBuffer[contextLocation++]=contextBuffer[(i+offset)&0x3fffU]);
					contextLocation&=0x3fffU;
				}
			}
		}
		contextLocation+=60;
		contextLocation&=0x3fffU;
	};

	// these are not part of the initContext like other methods
	std::unique_ptr<OptionalHuffmanDecoder<uint32_t>> symbolDecoder,offsetDecoder;
	bool heavyLastInitialized{false};		// this is part of initContext on some implementations. screwy!!!
	uint32_t heavyLastOffset;
	auto unpackHeavy=[&](auto &output,bool initTables,bool use8kDict)
	{
		initContext();
		// well, this works. Why this works? dunno
		if (!heavyLastInitialized)
		{
			heavyLastOffset=use8kDict?0U:~0U;
			heavyLastInitialized=true;
		}

		auto readTable=[&](std::unique_ptr<OptionalHuffmanDecoder<uint32_t>> &decoder,uint32_t countBits,uint32_t valueBits)
		{
			decoder=std::make_unique<OptionalHuffmanDecoder<uint32_t>>();
			uint32_t count{readBits(countBits)};
			if (count)
			{
				std::array<uint8_t,512> lengthBuffer;
				// in order to speed up the deObsfuscation, do not send the hopeless
				// data into slow CreateOrderlyHuffmanTable
				uint64_t sum{0};
				for (uint32_t i=0;i<count;i++)
				{
					uint32_t bits{readBits(valueBits)};
					if (bits)
					{
						sum+=uint64_t(1U)<<(32-bits);
						if (sum>(uint64_t(1U)<<32))
							throw DecompressionError();
					}
					lengthBuffer[i]=bits;
				}
				decoder->createOrderlyHuffmanTable(lengthBuffer,count);
			} else {
				uint32_t index{readBits(countBits)};
				decoder->setEmpty(index);
			}
		};

		if (initTables)
		{
			readTable(symbolDecoder,9,5);
			readTable(offsetDecoder,5,4);
		}

		uint32_t mask{use8kDict?0x1fffU:0xfffU};
		uint32_t bitLength{use8kDict?14U:13U};

		while (!output.eof())
		{
			if (output.getOffset()>=limitedDecompress) return;
			uint32_t symbol=symbolDecoder->decode(readBit);
			if (symbol<256)
			{
				output.writeByte(contextBuffer[contextLocation++]=symbol);
				contextLocation&=mask;
			} else {
				uint32_t count{symbol-253};	// minimum repeat is 3
				uint32_t offsetLength{offsetDecoder->decode(readBit)};
				uint32_t rawOffset{heavyLastOffset};
				if (offsetLength!=bitLength)
				{
					if (offsetLength) rawOffset=(1<<(offsetLength-1))|readBits(offsetLength-1);
						else rawOffset=0;
					heavyLastOffset=rawOffset;
				}
				uint32_t offset{contextLocation-rawOffset-1};
				for (uint32_t i=0;i<count;i++)
				{
					output.writeByte(contextBuffer[contextLocation++]=contextBuffer[(i+offset)&mask]);
					contextLocation&=mask;
				}
			}
		}
	};

	bool _codeFixed{false};
	uint32_t trackLength=(_isHD)?22528:11264;
	for (uint32_t packedOffset=56,packedChunkLength=0;packedOffset!=_packedSize;packedOffset=OverflowCheck::sum(packedOffset,20U,packedChunkLength))
	{
		// There are some info tracks, at -1 or 80. ignore those (if still present)
		uint16_t trackNo{_packedData.readBE16(packedOffset+2)};
		packedChunkLength=_packedData.readBE16(packedOffset+6);
		if (trackNo==80) break;							// should not happen, this is already excluded
		// even though only -1 should be used I've seen -2 as well. ignore all negatives
		uint32_t tmpChunkLength{_packedData.readBE16(packedOffset+8)};		// after the first unpack (if twostage)
		uint32_t rawChunkLength{_packedData.readBE16(packedOffset+10)};		// after final unpack
		uint8_t flags{_packedData.read8(packedOffset+12)};
		uint8_t mode{_packedData.read8(packedOffset+13)};

		// could affect context, but in practice they are separate, even though there is no explicit reset
		// deal with decompression though...
		if (trackNo>=0x8000U)
		{
			initInputStream(packedOffset+20,packedChunkLength);
			finishStream();
			continue;
		}
		if (rawChunkLength>trackLength)
			throw DecompressionError();
		if (trackNo>80)
			throw DecompressionError();					// should not happen, already excluded

		uint32_t dataOffset{trackNo*trackLength};
		if (_rawOffset>dataOffset)
			throw DecompressionError();

		auto handleTrackSize=[&]()
		{
			if (limitedDecompress!=~0U) return;
			if (!outputStream.eof() && outputStream.getOffset()&0x3ffU)
				throw DecompressionError();
		};

		// this is screwy, but it is, what it is
		auto processBlock=[&](bool doRLE,auto func,auto&&... params)
		{
			auto applyFix=[&]()
			{
				if (mode>=5U && !_isObsfuscated)
				{
					uint32_t missingNo{uint32_t(outputStream.getEndOffset()-outputStream.getOffset())};
					uint16_t protoSum{checksum(rawData,dataOffset-_rawOffset,rawChunkLength-missingNo)};
					uint16_t fileSum{_packedData.readBE16(packedOffset+14)};

					// too many ways to interpret the data
					if (missingNo>1)
						throw DecompressionError();
					for (uint32_t i=0;i<missingNo;i++) outputStream.writeByte(0);

					if (protoSum!=fileSum)
					{
						// last byte can be trashed by the compressor, but fortunately it can be fixed
						protoSum-=*outputStream.history(1);
						uint16_t fixByte{uint16_t(fileSum-protoSum)};
						if (fixByte>=0x100U)
							throw DecompressionError();
						*outputStream.history(1)=uint8_t(fixByte);
					}
				} else handleTrackSize();
			};

			initInputStream(packedOffset+20,packedChunkLength);
			if (doRLE)
			{
				ForwardOutputStream tmpOutputStream{tmpBuffer,0,tmpChunkLength};
				try
				{
					func(tmpOutputStream,params...);
				} catch (const DecompressionError &) {
					// nothing needed
				}
				finishStream();
				uint32_t missingNo{static_cast<uint32_t>(tmpOutputStream.getEndOffset()-tmpOutputStream.getOffset())};
				ForwardInputStream tmpInputStream{tmpBuffer,0,tmpChunkLength-missingNo};
				initOutputStream(dataOffset-_rawOffset,rawChunkLength);
				try
				{
					unRLE(outputStream,tmpInputStream);
				} catch (const DecompressionError &) {
					// nothing needed
				}
				applyFix();
			} else {
				initOutputStream(dataOffset-_rawOffset,rawChunkLength);
				try
				{
					func(outputStream,params...);
				} catch (const DecompressionError &) {
					// nothing needed
				}
				applyFix();
			}
			finishStream();
		};

		auto processBlockCode=[&](bool doRLE,auto func,auto&&... params)
		{
			if (!_isObsfuscated || trackNo!=_minTrack || _codeFixed) return processBlock(doRLE,func,params...);

			// fast try
			if (!trackNo && restartPosition<0x10000U) for (;restartPosition<0x10000U;restartPosition++)
			{
				try
				{
					doInitContext=true;
					inputStream.setCode(restartPosition);
					limitedDecompress=8;
					processBlock(doRLE,func,params...);
					if ((rawData.readBE32(0)&0xffff'ff00U)!=FourCC("DOS\0")) continue;

					// now see if the candidate is any good
					doInitContext=true;
					inputStream.setCode(restartPosition);
					limitedDecompress=~0U;
					processBlock(doRLE,func,params...);
					if (checksum(rawData,dataOffset-_rawOffset,rawChunkLength)!=_packedData.readBE16(packedOffset+14)) continue;
					_codeFixed=true;
					return;
				} catch (const Buffer::Error &) {
					// just continue
				} catch (const Decompressor::Error &) {
					// just continue
				}
			}

			// slow round
			limitedDecompress=~0U;
			for (;restartPosition<0x20000U;restartPosition++)
			{
				try
				{
					doInitContext=true;
					inputStream.setCode(restartPosition);
					processBlock(doRLE,func,params...);
					if (checksum(rawData,dataOffset-_rawOffset,rawChunkLength)!=_packedData.readBE16(packedOffset+14)) continue;
					_codeFixed=true;
					return;
				} catch (const Buffer::Error &) {
					// just continue
				} catch (const Decompressor::Error &) {
					// just continue
				}
			}
			throw DecompressionError();
		};

		switch (mode)
		{
			case 0:
			processBlockCode(false,unpackNone);
			rawChunkLength=packedChunkLength;
			break;

			case 1:
			processBlockCode(false,[&](auto &output)
			{
				unRLE(output,inputStream);
				handleTrackSize();
			});
			break;

			case 2:
			processBlockCode(true,unpackQuick);
			break;

			case 3:
			processBlockCode(true,unpackMedium);
			break;

			case 4:
			processBlockCode(true,unpackDeep);
			break;

			// heavy flags:
			// 2: (re-)initialize/read tables
			// 4: do RLE
			// heavy1 uses 4k dictionary (mode 5), whereas heavy2 uses 8k dictionary
			case 5:
			[[fallthrough]];
			case 6:
			processBlockCode(flags&4,unpackHeavy,flags&2,mode==6);
			break;

			default:
			throw DecompressionError();
		}
		if (!(flags&1)) doInitContext=true;

		if (verify && checksum(rawData,dataOffset-_rawOffset,rawChunkLength)!=_packedData.readBE16(packedOffset+14))
			throw VerificationError();
	}
}

}