zstdify 1.1.2 → 1.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +30 -1
- package/dist/bitstream/bitReaderReverse.js +27 -1
- package/dist/bitstream/bitReaderReverse.js.map +1 -1
- package/dist/compress.js +1 -2
- package/dist/compress.js.map +1 -1
- package/dist/decode/decompressFrame.d.ts +2 -1
- package/dist/decode/decompressFrame.js +73 -57
- package/dist/decode/decompressFrame.js.map +1 -1
- package/dist/decode/literals.js +55 -52
- package/dist/decode/literals.js.map +1 -1
- package/dist/decode/reconstruct.d.ts +21 -1
- package/dist/decode/reconstruct.js +231 -21
- package/dist/decode/reconstruct.js.map +1 -1
- package/dist/decode/sequences.d.ts +1 -1
- package/dist/decode/sequences.js +118 -105
- package/dist/decode/sequences.js.map +1 -1
- package/dist/decompress.d.ts +7 -0
- package/dist/decompress.js +5 -3
- package/dist/decompress.js.map +1 -1
- package/dist/encode/blockWriter.js +8 -2
- package/dist/encode/blockWriter.js.map +1 -1
- package/dist/encode/compressedBlock.js +161 -106
- package/dist/encode/compressedBlock.js.map +1 -1
- package/dist/encode/greedySequences.js +63 -40
- package/dist/encode/greedySequences.js.map +1 -1
- package/dist/entropy/fse.js +1 -5
- package/dist/entropy/fse.js.map +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -25,6 +25,9 @@ Pure JavaScript/TypeScript zstd compression/decompression library. No native dep
|
|
|
25
25
|
- **Dictionary generation**:
|
|
26
26
|
- Pure TypeScript dictionary training from sample payloads.
|
|
27
27
|
- Zstd-inspired training options (`fastcover`/`cover`/`legacy` style knobs).
|
|
28
|
+
- **Tree-shaken bundle size (Rollup + Terser, compressed)**:
|
|
29
|
+
- `zstdify/compress`: ~4.91 KiB gzip / ~4.43 KiB brotli.
|
|
30
|
+
- `zstdify/decompress`: ~7.46 KiB gzip / ~6.62 KiB brotli.
|
|
28
31
|
- **Interop-focused**: `zstdify` output is decoded by the official `zstd` CLI and by the [zstddec](https://www.npmjs.com/package/zstddec) npm package; `zstd` CLI output is decoded by `zstdify`.
|
|
29
32
|
- **Extensively tested**:
|
|
30
33
|
- Round-trip and property-based tests.
|
|
@@ -62,7 +65,7 @@ import { generateDictionary } from 'zstdify/dictionary';
|
|
|
62
65
|
## API
|
|
63
66
|
|
|
64
67
|
- `compress(input: Uint8Array, options?: { level?: number; checksum?: boolean; dictionary?: Uint8Array | { bytes: Uint8Array; id?: number }; noDictId?: boolean }): Uint8Array`
|
|
65
|
-
- `decompress(input: Uint8Array, options?: { maxSize?: number; dictionary?: Uint8Array | { bytes: Uint8Array; id?: number } }): Uint8Array`
|
|
68
|
+
- `decompress(input: Uint8Array, options?: { maxSize?: number; dictionary?: Uint8Array | { bytes: Uint8Array; id?: number }; validateChecksum?: boolean }): Uint8Array`
|
|
66
69
|
- `generateDictionary(samples: Uint8Array[], options?: { maxDictSize?: number; dictId?: number; algorithm?: "fastcover" | "cover" | "legacy"; k?: number; d?: number; steps?: number; split?: number; f?: number; accel?: number; selectivity?: number; shrink?: boolean | number }): Uint8Array`
|
|
67
70
|
|
|
68
71
|
Dictionary generation outputs a raw-content dictionary. If you want a specific `dictID` written into compressed frames, pass it to `compress()` via `dictionary: { bytes, id }`.
|
|
@@ -122,6 +125,32 @@ All of the following run as part of the test suite (`pnpm test` / `pnpm vitest`)
|
|
|
122
125
|
- **Compression regression**: Compressed sizes for fixed payloads are checked against golden values (ratio stability).
|
|
123
126
|
- **Decompress robustness**: Each corpus fixture is decompressed in its own test (one test per file), so the suite tracks decompress behavior per input. See [upstream zstd TESTING.md](https://github.com/facebook/zstd/blob/dev/TESTING.md) for comparison.
|
|
124
127
|
|
|
128
|
+
## Benchmark: zstdify vs Node built-in zstd
|
|
129
|
+
|
|
130
|
+
Throughput and compression ratio are compared against Node’s built-in `node:zlib` zstd on synthetic payloads (text, binary, repetitive) at levels 3, 5, and 9.
|
|
131
|
+
|
|
132
|
+

|
|
133
|
+
|
|
134
|
+
To regenerate the chart and tables (run benchmarks and render SVG):
|
|
135
|
+
|
|
136
|
+
```bash
|
|
137
|
+
pnpm --filter zstdify-tests run bench:update
|
|
138
|
+
```
|
|
139
|
+
|
|
140
|
+
### Bundle size benchmark (Rollup)
|
|
141
|
+
|
|
142
|
+
| Target | Raw | Gzip | Brotli |
|
|
143
|
+
|---|---:|---:|---:|
|
|
144
|
+
| zstdify/compress | 13.90 KiB | 5.27 KiB | 4.75 KiB |
|
|
145
|
+
| zstdify/decompress | 27.70 KiB | 8.63 KiB | 7.67 KiB |
|
|
146
|
+
| zstddec decoder + wasm | 127.37 KiB | 49.69 KiB | 40.66 KiB |
|
|
147
|
+
|
|
148
|
+
To regenerate this snapshot:
|
|
149
|
+
|
|
150
|
+
```bash
|
|
151
|
+
pnpm --filter zstdify-tests run bench:bundle-size
|
|
152
|
+
```
|
|
153
|
+
|
|
125
154
|
## Publishing
|
|
126
155
|
|
|
127
156
|
Publish the npm packages (library first, then CLI so it gets the correct `zstdify` version):
|
|
@@ -5,6 +5,11 @@
|
|
|
5
5
|
* - consume bits by decrementing that offset,
|
|
6
6
|
* - extract bits in little-endian bit order.
|
|
7
7
|
*/
|
|
8
|
+
const BIT_MASKS = Array.from({ length: 33 }, (_, i) => {
|
|
9
|
+
if (i === 32)
|
|
10
|
+
return 0xffffffff;
|
|
11
|
+
return ((1 << i) - 1) >>> 0;
|
|
12
|
+
});
|
|
8
13
|
export class BitReaderReverse {
|
|
9
14
|
data;
|
|
10
15
|
startBit;
|
|
@@ -25,7 +30,28 @@ export class BitReaderReverse {
|
|
|
25
30
|
throw new RangeError(`BitReaderReverse.readBits: n must be 1-32, got ${n}`);
|
|
26
31
|
}
|
|
27
32
|
const requestedStart = this.bitOffset - n;
|
|
28
|
-
this.
|
|
33
|
+
const clampedStart = requestedStart < this.startBit ? this.startBit : requestedStart;
|
|
34
|
+
this.bitOffset = clampedStart;
|
|
35
|
+
if (requestedStart >= this.startBit) {
|
|
36
|
+
const byteIndex = requestedStart >>> 3;
|
|
37
|
+
const bitInByte = requestedStart & 7;
|
|
38
|
+
const readWord = (idx) => ((this.data[idx] ?? 0) |
|
|
39
|
+
((this.data[idx + 1] ?? 0) << 8) |
|
|
40
|
+
((this.data[idx + 2] ?? 0) << 16) |
|
|
41
|
+
((this.data[idx + 3] ?? 0) << 24)) >>>
|
|
42
|
+
0;
|
|
43
|
+
const word0 = readWord(byteIndex);
|
|
44
|
+
if (bitInByte + n <= 32) {
|
|
45
|
+
const value = word0 >>> bitInByte;
|
|
46
|
+
return n === 32 ? value >>> 0 : (value & BIT_MASKS[n]) >>> 0;
|
|
47
|
+
}
|
|
48
|
+
const low = word0 >>> bitInByte;
|
|
49
|
+
const highBits = n - (32 - bitInByte);
|
|
50
|
+
const word1 = readWord(byteIndex + 4);
|
|
51
|
+
const high = ((word1 & BIT_MASKS[highBits]) << (32 - bitInByte)) >>> 0;
|
|
52
|
+
const merged = (low | high) >>> 0;
|
|
53
|
+
return n === 32 ? merged : (merged & BIT_MASKS[n]) >>> 0;
|
|
54
|
+
}
|
|
29
55
|
let value = 0;
|
|
30
56
|
for (let i = 0; i < n; i++) {
|
|
31
57
|
const absoluteBit = requestedStart + i;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"bitReaderReverse.js","sourceRoot":"","sources":["../../src/bitstream/bitReaderReverse.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,MAAM,OAAO,gBAAgB;IACV,IAAI,CAAa;IACjB,QAAQ,CAAS;IACjB,MAAM,CAAS;IACxB,SAAS,CAAS;IAE1B,YAAY,IAAgB,EAAE,eAAuB,EAAE,WAAmB,EAAE,eAAe,GAAG,CAAC,EAAE;QAC/F,IAAI,WAAW,GAAG,CAAC,EAAE,CAAC;YACpB,MAAM,IAAI,UAAU,CAAC,qCAAqC,WAAW,EAAE,CAAC,CAAC;QAC3E,CAAC;QACD,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,QAAQ,GAAG,eAAe,GAAG,CAAC,GAAG,eAAe,CAAC;QACtD,IAAI,CAAC,MAAM,GAAG,CAAC,eAAe,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC;QAClD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC;IAAA,CAC9B;IAED,6EAA6E;IAC7E,QAAQ,CAAC,CAAS,EAAU;QAC1B,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;YACpB,MAAM,IAAI,UAAU,CAAC,kDAAkD,CAAC,EAAE,CAAC,CAAC;QAC9E,CAAC;QAED,MAAM,cAAc,GAAG,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC;QAC1C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,EAAE,
|
|
1
|
+
{"version":3,"file":"bitReaderReverse.js","sourceRoot":"","sources":["../../src/bitstream/bitReaderReverse.ts"],"names":[],"mappings":"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"}
|
package/dist/compress.js
CHANGED
|
@@ -40,8 +40,7 @@ export function compress(input, options) {
|
|
|
40
40
|
const payload = buildCompressedBlockPayload(plan.literals, plan.sequences);
|
|
41
41
|
if (payload) {
|
|
42
42
|
const compressed = writeCompressedBlock(payload, last);
|
|
43
|
-
|
|
44
|
-
if (compressed.length < raw.length) {
|
|
43
|
+
if (compressed.length < 3 + size) {
|
|
45
44
|
chunks.push(compressed);
|
|
46
45
|
offset += size;
|
|
47
46
|
blockIndex++;
|
package/dist/compress.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"compress.js","sourceRoot":"","sources":["../src/compress.ts"],"names":[],"mappings":"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
|
|
1
|
+
{"version":3,"file":"compress.js","sourceRoot":"","sources":["../src/compress.ts"],"names":[],"mappings":"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"}
|
|
@@ -3,7 +3,8 @@
|
|
|
3
3
|
*/
|
|
4
4
|
import type { DecoderDictionaryContext } from '../dictionary/decoderDictionary.js';
|
|
5
5
|
import type { FrameHeader } from '../frame/frameHeader.js';
|
|
6
|
-
|
|
6
|
+
import { type DecoderReuseBag } from './reconstruct.js';
|
|
7
|
+
export declare function decompressFrame(data: Uint8Array, offset: number, header: FrameHeader, dictionary?: DecoderDictionaryContext | null, maxSize?: number, validateChecksum?: boolean, reuseContext?: DecoderReuseBag): {
|
|
7
8
|
output: Uint8Array;
|
|
8
9
|
bytesConsumed: number;
|
|
9
10
|
};
|
|
@@ -6,21 +6,42 @@ import { ZstdError } from '../errors.js';
|
|
|
6
6
|
import { validateContentChecksum } from '../frame/checksum.js';
|
|
7
7
|
import { parseBlockHeader } from './block.js';
|
|
8
8
|
import { decodeCompressedLiterals, decodeRawLiterals, decodeRLELiterals, decodeTreelessLiterals, parseLiteralsSectionHeader, } from './literals.js';
|
|
9
|
-
import {
|
|
9
|
+
import { appendRangeToHistoryWindow, appendRLEToHistoryWindow, appendToHistoryWindow, executeSequencesInto, getOrCreateHistoryWindow, } from './reconstruct.js';
|
|
10
10
|
import { decodeSequences } from './sequences.js';
|
|
11
|
-
export function decompressFrame(data, offset, header, dictionary, maxSize) {
|
|
11
|
+
export function decompressFrame(data, offset, header, dictionary, maxSize, validateChecksum = true, reuseContext) {
|
|
12
12
|
let pos = offset + 4 + header.headerSize;
|
|
13
|
-
const
|
|
13
|
+
const knownOutputSize = header.contentSize ?? null;
|
|
14
|
+
let outputBuffer = knownOutputSize !== null ? new Uint8Array(knownOutputSize) : new Uint8Array(0);
|
|
14
15
|
let totalSize = 0;
|
|
15
16
|
const repOffsets = dictionary?.repOffsets
|
|
16
17
|
? [dictionary.repOffsets[0], dictionary.repOffsets[1], dictionary.repOffsets[2]]
|
|
17
18
|
: [1, 4, 8];
|
|
18
|
-
|
|
19
|
-
if (dictionary?.historyPrefix && dictionary.historyPrefix.length > 0) {
|
|
20
|
-
history = dictionary.historyPrefix.slice();
|
|
21
|
-
}
|
|
19
|
+
const history = getOrCreateHistoryWindow(header.windowSize, dictionary?.historyPrefix, reuseContext);
|
|
22
20
|
let prevHuffmanTable = dictionary?.huffmanTable ?? null;
|
|
23
21
|
let prevSeqTables = dictionary?.sequenceTables ?? null;
|
|
22
|
+
const ensureOutputCapacity = (additional) => {
|
|
23
|
+
const needed = totalSize + additional;
|
|
24
|
+
if (needed <= outputBuffer.length) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
let nextCapacity = outputBuffer.length === 0 ? 64 * 1024 : outputBuffer.length;
|
|
28
|
+
while (nextCapacity < needed) {
|
|
29
|
+
nextCapacity *= 2;
|
|
30
|
+
}
|
|
31
|
+
const grown = new Uint8Array(nextCapacity);
|
|
32
|
+
if (totalSize > 0) {
|
|
33
|
+
grown.set(outputBuffer.subarray(0, totalSize), 0);
|
|
34
|
+
}
|
|
35
|
+
outputBuffer = grown;
|
|
36
|
+
};
|
|
37
|
+
const appendOutput = (chunk) => {
|
|
38
|
+
if (chunk.length === 0) {
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
ensureOutputCapacity(chunk.length);
|
|
42
|
+
outputBuffer.set(chunk, totalSize);
|
|
43
|
+
totalSize += chunk.length;
|
|
44
|
+
};
|
|
24
45
|
while (true) {
|
|
25
46
|
if (pos + 3 > data.length) {
|
|
26
47
|
throw new ZstdError('Block header truncated', 'corruption_detected');
|
|
@@ -28,17 +49,28 @@ export function decompressFrame(data, offset, header, dictionary, maxSize) {
|
|
|
28
49
|
const block = parseBlockHeader(data, pos);
|
|
29
50
|
pos += 3;
|
|
30
51
|
if (block.blockType === 0) {
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
52
|
+
if (pos + block.blockSize > data.length) {
|
|
53
|
+
throw new ZstdError('Raw literals truncated', 'corruption_detected');
|
|
54
|
+
}
|
|
55
|
+
ensureOutputCapacity(block.blockSize);
|
|
56
|
+
outputBuffer.set(data.subarray(pos, pos + block.blockSize), totalSize);
|
|
57
|
+
if (!block.lastBlock) {
|
|
58
|
+
appendRangeToHistoryWindow(history, data, pos, block.blockSize);
|
|
59
|
+
}
|
|
60
|
+
totalSize += block.blockSize;
|
|
35
61
|
pos += block.blockSize;
|
|
36
62
|
}
|
|
37
63
|
else if (block.blockType === 1) {
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
64
|
+
if (pos >= data.length) {
|
|
65
|
+
throw new ZstdError('RLE literals truncated', 'corruption_detected');
|
|
66
|
+
}
|
|
67
|
+
const byte = data[pos];
|
|
68
|
+
ensureOutputCapacity(block.blockSize);
|
|
69
|
+
outputBuffer.fill(byte, totalSize, totalSize + block.blockSize);
|
|
70
|
+
if (!block.lastBlock) {
|
|
71
|
+
appendRLEToHistoryWindow(history, byte, block.blockSize);
|
|
72
|
+
}
|
|
73
|
+
totalSize += block.blockSize;
|
|
42
74
|
pos += 1;
|
|
43
75
|
}
|
|
44
76
|
else if (block.blockType === 2) {
|
|
@@ -69,23 +101,35 @@ export function decompressFrame(data, offset, header, dictionary, maxSize) {
|
|
|
69
101
|
litBytesConsumed = litHeader.headerSize + comp.bytesRead;
|
|
70
102
|
}
|
|
71
103
|
const seqSectionSize = block.blockSize - litBytesConsumed;
|
|
72
|
-
let output;
|
|
73
104
|
if (seqSectionSize <= 0) {
|
|
74
|
-
|
|
105
|
+
appendOutput(literals);
|
|
106
|
+
if (!block.lastBlock) {
|
|
107
|
+
appendToHistoryWindow(history, literals);
|
|
108
|
+
}
|
|
75
109
|
}
|
|
76
110
|
else {
|
|
77
|
-
const seqResult = decodeSequences(blockContent, litBytesConsumed, seqSectionSize, prevSeqTables);
|
|
111
|
+
const seqResult = decodeSequences(blockContent, litBytesConsumed, seqSectionSize, prevSeqTables, reuseContext?._sequences);
|
|
112
|
+
if (reuseContext) {
|
|
113
|
+
reuseContext._sequences = seqResult.sequences;
|
|
114
|
+
}
|
|
78
115
|
prevSeqTables = seqResult.tables;
|
|
79
116
|
if (seqResult.sequences.length === 0) {
|
|
80
|
-
|
|
117
|
+
appendOutput(literals);
|
|
118
|
+
if (!block.lastBlock) {
|
|
119
|
+
appendToHistoryWindow(history, literals);
|
|
120
|
+
}
|
|
81
121
|
}
|
|
82
122
|
else {
|
|
83
|
-
|
|
123
|
+
let decodedSize = literals.length;
|
|
124
|
+
for (const seq of seqResult.sequences) {
|
|
125
|
+
decodedSize += seq.matchLength;
|
|
126
|
+
}
|
|
127
|
+
ensureOutputCapacity(decodedSize);
|
|
128
|
+
const start = totalSize;
|
|
129
|
+
const written = executeSequencesInto(literals, seqResult.sequences, header.windowSize, outputBuffer, start, repOffsets, history, !block.lastBlock);
|
|
130
|
+
totalSize += written;
|
|
84
131
|
}
|
|
85
132
|
}
|
|
86
|
-
chunks.push(output);
|
|
87
|
-
totalSize += output.length;
|
|
88
|
-
history = appendToHistory(history, output, header.windowSize);
|
|
89
133
|
pos += block.blockSize;
|
|
90
134
|
}
|
|
91
135
|
else {
|
|
@@ -97,7 +141,7 @@ export function decompressFrame(data, offset, header, dictionary, maxSize) {
|
|
|
97
141
|
if (block.lastBlock)
|
|
98
142
|
break;
|
|
99
143
|
}
|
|
100
|
-
const output =
|
|
144
|
+
const output = outputBuffer.subarray(0, totalSize);
|
|
101
145
|
if (header.contentSize !== null && output.length !== header.contentSize) {
|
|
102
146
|
throw new ZstdError('Frame content size mismatch', 'corruption_detected');
|
|
103
147
|
}
|
|
@@ -105,43 +149,15 @@ export function decompressFrame(data, offset, header, dictionary, maxSize) {
|
|
|
105
149
|
if (pos + 4 > data.length) {
|
|
106
150
|
throw new ZstdError('Content checksum truncated', 'corruption_detected');
|
|
107
151
|
}
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
152
|
+
if (validateChecksum) {
|
|
153
|
+
const storedChecksum = readU32LE(data, pos);
|
|
154
|
+
if (!validateContentChecksum(output, storedChecksum)) {
|
|
155
|
+
throw new ZstdError('Content checksum mismatch', 'corruption_detected');
|
|
156
|
+
}
|
|
111
157
|
}
|
|
112
158
|
pos += 4;
|
|
113
159
|
return { output, bytesConsumed: pos - offset };
|
|
114
160
|
}
|
|
115
161
|
return { output, bytesConsumed: pos - offset };
|
|
116
162
|
}
|
|
117
|
-
function appendToHistory(history, chunk, windowSize) {
|
|
118
|
-
if (windowSize <= 0 || chunk.length === 0) {
|
|
119
|
-
return history;
|
|
120
|
-
}
|
|
121
|
-
const maxHistory = Math.max(1, windowSize);
|
|
122
|
-
if (chunk.length >= maxHistory) {
|
|
123
|
-
return new Uint8Array(chunk.subarray(chunk.length - maxHistory));
|
|
124
|
-
}
|
|
125
|
-
const keepFromHistory = Math.min(history.length, maxHistory - chunk.length);
|
|
126
|
-
const next = new Uint8Array(keepFromHistory + chunk.length);
|
|
127
|
-
if (keepFromHistory > 0) {
|
|
128
|
-
next.set(history.subarray(history.length - keepFromHistory), 0);
|
|
129
|
-
}
|
|
130
|
-
next.set(chunk, keepFromHistory);
|
|
131
|
-
return next;
|
|
132
|
-
}
|
|
133
|
-
function concatenateChunks(chunks) {
|
|
134
|
-
if (chunks.length === 0)
|
|
135
|
-
return new Uint8Array(0);
|
|
136
|
-
if (chunks.length === 1)
|
|
137
|
-
return chunks[0];
|
|
138
|
-
const total = chunks.reduce((s, c) => s + c.length, 0);
|
|
139
|
-
const result = new Uint8Array(total);
|
|
140
|
-
let pos = 0;
|
|
141
|
-
for (const chunk of chunks) {
|
|
142
|
-
result.set(chunk, pos);
|
|
143
|
-
pos += chunk.length;
|
|
144
|
-
}
|
|
145
|
-
return result;
|
|
146
|
-
}
|
|
147
163
|
//# sourceMappingURL=decompressFrame.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"decompressFrame.js","sourceRoot":"","sources":["../../src/decode/decompressFrame.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,SAAS,EAAE,MAAM,8BAA8B,CAAC;AAEzD,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AACzC,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAE/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAC9C,OAAO,EACL,wBAAwB,EACxB,iBAAiB,EACjB,iBAAiB,EACjB,sBAAsB,EACtB,0BAA0B,GAC3B,MAAM,eAAe,CAAC;AACvB,OAAO,
|
|
1
|
+
{"version":3,"file":"decompressFrame.js","sourceRoot":"","sources":["../../src/decode/decompressFrame.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,SAAS,EAAE,MAAM,8BAA8B,CAAC;AAEzD,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AACzC,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAE/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAC9C,OAAO,EACL,wBAAwB,EACxB,iBAAiB,EACjB,iBAAiB,EACjB,sBAAsB,EACtB,0BAA0B,GAC3B,MAAM,eAAe,CAAC;AACvB,OAAO,EACL,0BAA0B,EAC1B,wBAAwB,EACxB,qBAAqB,EACrB,oBAAoB,EACpB,wBAAwB,GAEzB,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAE,eAAe,EAAuB,MAAM,gBAAgB,CAAC;AAEtE,MAAM,UAAU,eAAe,CAC7B,IAAgB,EAChB,MAAc,EACd,MAAmB,EACnB,UAA4C,EAC5C,OAAgB,EAChB,gBAAgB,GAAG,IAAI,EACvB,YAA8B,EACiB;IAC/C,IAAI,GAAG,GAAG,MAAM,GAAG,CAAC,GAAG,MAAM,CAAC,UAAU,CAAC;IACzC,MAAM,eAAe,GAAG,MAAM,CAAC,WAAW,IAAI,IAAI,CAAC;IACnD,IAAI,YAAY,GAAG,eAAe,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,UAAU,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,IAAI,UAAU,CAAC,CAAC,CAAC,CAAC;IAClG,IAAI,SAAS,GAAG,CAAC,CAAC;IAClB,MAAM,UAAU,GAA6B,UAAU,EAAE,UAAU;QACjE,CAAC,CAAC,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAChF,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;IACd,MAAM,OAAO,GAAG,wBAAwB,CACtC,MAAM,CAAC,UAAU,EACjB,UAAU,EAAE,aAAa,EACzB,YAAY,CACb,CAAC;IACF,IAAI,gBAAgB,GAGT,UAAU,EAAE,YAAY,IAAI,IAAI,CAAC;IAC5C,IAAI,aAAa,GAA0B,UAAU,EAAE,cAAc,IAAI,IAAI,CAAC;IAE9E,MAAM,oBAAoB,GAAG,CAAC,UAAkB,EAAQ,EAAE,CAAC;QACzD,MAAM,MAAM,GAAG,SAAS,GAAG,UAAU,CAAC;QACtC,IAAI,MAAM,IAAI,YAAY,CAAC,MAAM,EAAE,CAAC;YAClC,OAAO;QACT,CAAC;QACD,IAAI,YAAY,GAAG,YAAY,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC;QAC/E,OAAO,YAAY,GAAG,MAAM,EAAE,CAAC;YAC7B,YAAY,IAAI,CAAC,CAAC;QACpB,CAAC;QACD,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,YAAY,CAAC,CAAC;QAC3C,IAAI,SAAS,GAAG,CAAC,EAAE,CAAC;YAClB,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC;QACpD,CAAC;QACD,YAAY,GAAG,KAAK,CAAC;IAAA,CACtB,CAAC;IAEF,MAAM,YAAY,GAAG,CAAC,KAAiB,EAAQ,EAAE,CAAC;QAChD,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACvB,OAAO;QACT,CAAC;QACD,oBAAoB,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACnC,YAAY,CAAC,GAAG,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;QACnC,SAAS,IAAI,KAAK,CAAC,MAAM,CAAC;IAAA,CAC3B,CAAC;IAEF,OAAO,IAAI,EAAE,CAAC;QACZ,IAAI,GAAG,GAAG,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;YAC1B,MAAM,IAAI,SAAS,CAAC,wBAAwB,EAAE,qBAAqB,CAAC,CAAC;QACvE,CAAC;QACD,MAAM,KAAK,GAAG,gBAAgB,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;QAC1C,GAAG,IAAI,CAAC,CAAC;QAET,IAAI,KAAK,CAAC,SAAS,KAAK,CAAC,EAAE,CAAC;YAC1B,IAAI,GAAG,GAAG,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;gBACxC,MAAM,IAAI,SAAS,CAAC,wBAAwB,EAAE,qBAAqB,CAAC,CAAC;YACvE,CAAC;YACD,oBAAoB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YACtC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,EAAE,SAAS,CAAC,CAAC;YACvE,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;gBACrB,0BAA0B,CAAC,OAAO,EAAE,IAAI,EAAE,GAAG,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;YAClE,CAAC;YACD,SAAS,IAAI,KAAK,CAAC,SAAS,CAAC;YAC7B,GAAG,IAAI,KAAK,CAAC,SAAS,CAAC;QACzB,CAAC;aAAM,IAAI,KAAK,CAAC,SAAS,KAAK,CAAC,EAAE,CAAC;YACjC,IAAI,GAAG,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;gBACvB,MAAM,IAAI,SAAS,CAAC,wBAAwB,EAAE,qBAAqB,CAAC,CAAC;YACvE,CAAC;YACD,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,CAAE,CAAC;YACxB,oBAAoB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YACtC,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,SAAS,EAAE,SAAS,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC;YAChE,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;gBACrB,wBAAwB,CAAC,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;YAC3D,CAAC;YACD,SAAS,IAAI,KAAK,CAAC,SAAS,CAAC;YAC7B,GAAG,IAAI,CAAC,CAAC;QACX,CAAC;aAAM,IAAI,KAAK,CAAC,SAAS,KAAK,CAAC,EAAE,CAAC;YACjC,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC;YAC/D,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,UAAU,EAAE,aAAa,EAAE,GAAG,0BAA0B,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC;YAErG,IAAI,QAAoB,CAAC;YACzB,IAAI,gBAAwB,CAAC;YAE7B,IAAI,SAAS,CAAC,SAAS,KAAK,CAAC,EAAE,CAAC;gBAC9B,QAAQ,GAAG,iBAAiB,CAAC,YAAY,EAAE,aAAa,EAAE,SAAS,CAAC,eAAe,CAAC,CAAC;gBACrF,gBAAgB,GAAG,SAAS,CAAC,UAAU,GAAG,SAAS,CAAC,eAAe,CAAC;YACtE,CAAC;iBAAM,IAAI,SAAS,CAAC,SAAS,KAAK,CAAC,EAAE,CAAC;gBACrC,QAAQ,GAAG,iBAAiB,CAAC,YAAY,EAAE,aAAa,EAAE,SAAS,CAAC,eAAe,CAAC,CAAC;gBACrF,gBAAgB,GAAG,SAAS,CAAC,UAAU,GAAG,CAAC,CAAC;YAC9C,CAAC;iBAAM,IAAI,SAAS,CAAC,SAAS,KAAK,CAAC,EAAE,CAAC;gBACrC,MAAM,IAAI,GAAG,wBAAwB,CACnC,YAAY,EACZ,aAAa,EACb,SAAS,CAAC,cAAe,EACzB,SAAS,CAAC,eAAe,EACzB,SAAS,CAAC,UAAU,CACrB,CAAC;gBACF,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;gBACzB,gBAAgB,GAAG,IAAI,CAAC,YAAY,CAAC;gBACrC,gBAAgB,GAAG,SAAS,CAAC,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC;YAC3D,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,gBAAgB,EAAE,CAAC;oBACtB,MAAM,IAAI,SAAS,CAAC,kDAAkD,EAAE,qBAAqB,CAAC,CAAC;gBACjG,CAAC;gBACD,MAAM,IAAI,GAAG,sBAAsB,CACjC,YAAY,EACZ,aAAa,EACb,SAAS,CAAC,cAAe,EACzB,SAAS,CAAC,eAAe,EACzB,SAAS,CAAC,UAAU,EACpB,gBAAgB,CACjB,CAAC;gBACF,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;gBACzB,gBAAgB,GAAG,SAAS,CAAC,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC;YAC3D,CAAC;YAED,MAAM,cAAc,GAAG,KAAK,CAAC,SAAS,GAAG,gBAAgB,CAAC;YAC1D,IAAI,cAAc,IAAI,CAAC,EAAE,CAAC;gBACxB,YAAY,CAAC,QAAQ,CAAC,CAAC;gBACvB,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;oBACrB,qBAAqB,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;gBAC3C,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,MAAM,SAAS,GAAG,eAAe,CAC/B,YAAY,EACZ,gBAAgB,EAChB,cAAc,EACd,aAAa,EACb,YAAY,EAAE,UAAU,CACzB,CAAC;gBACF,IAAI,YAAY,EAAE,CAAC;oBACjB,YAAY,CAAC,UAAU,GAAG,SAAS,CAAC,SAAS,CAAC;gBAChD,CAAC;gBACD,aAAa,GAAG,SAAS,CAAC,MAAM,CAAC;gBACjC,IAAI,SAAS,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBACrC,YAAY,CAAC,QAAQ,CAAC,CAAC;oBACvB,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;wBACrB,qBAAqB,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;oBAC3C,CAAC;gBACH,CAAC;qBAAM,CAAC;oBACN,IAAI,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC;oBAClC,KAAK,MAAM,GAAG,IAAI,SAAS,CAAC,SAAS,EAAE,CAAC;wBACtC,WAAW,IAAI,GAAG,CAAC,WAAW,CAAC;oBACjC,CAAC;oBACD,oBAAoB,CAAC,WAAW,CAAC,CAAC;oBAClC,MAAM,KAAK,GAAG,SAAS,CAAC;oBACxB,MAAM,OAAO,GAAG,oBAAoB,CAClC,QAAQ,EACR,SAAS,CAAC,SAAS,EACnB,MAAM,CAAC,UAAU,EACjB,YAAY,EACZ,KAAK,EACL,UAAU,EACV,OAAO,EACP,CAAC,KAAK,CAAC,SAAS,CACjB,CAAC;oBACF,SAAS,IAAI,OAAO,CAAC;gBACvB,CAAC;YACH,CAAC;YACD,GAAG,IAAI,KAAK,CAAC,SAAS,CAAC;QACzB,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,SAAS,CAAC,wBAAwB,EAAE,qBAAqB,CAAC,CAAC;QACvE,CAAC;QAED,IAAI,OAAO,KAAK,SAAS,IAAI,SAAS,GAAG,OAAO,EAAE,CAAC;YACjD,MAAM,IAAI,SAAS,CAAC,mCAAmC,EAAE,uBAAuB,CAAC,CAAC;QACpF,CAAC;QAED,IAAI,KAAK,CAAC,SAAS;YAAE,MAAM;IAC7B,CAAC;IAED,MAAM,MAAM,GAAG,YAAY,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;IACnD,IAAI,MAAM,CAAC,WAAW,KAAK,IAAI,IAAI,MAAM,CAAC,MAAM,KAAK,MAAM,CAAC,WAAW,EAAE,CAAC;QACxE,MAAM,IAAI,SAAS,CAAC,6BAA6B,EAAE,qBAAqB,CAAC,CAAC;IAC5E,CAAC;IAED,IAAI,MAAM,CAAC,kBAAkB,EAAE,CAAC;QAC9B,IAAI,GAAG,GAAG,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;YAC1B,MAAM,IAAI,SAAS,CAAC,4BAA4B,EAAE,qBAAqB,CAAC,CAAC;QAC3E,CAAC;QACD,IAAI,gBAAgB,EAAE,CAAC;YACrB,MAAM,cAAc,GAAG,SAAS,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;YAC5C,IAAI,CAAC,uBAAuB,CAAC,MAAM,EAAE,cAAc,CAAC,EAAE,CAAC;gBACrD,MAAM,IAAI,SAAS,CAAC,2BAA2B,EAAE,qBAAqB,CAAC,CAAC;YAC1E,CAAC;QACH,CAAC;QACD,GAAG,IAAI,CAAC,CAAC;QACT,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,GAAG,GAAG,MAAM,EAAE,CAAC;IACjD,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,aAAa,EAAE,GAAG,GAAG,MAAM,EAAE,CAAC;AAAA,CAChD"}
|
package/dist/decode/literals.js
CHANGED
|
@@ -14,7 +14,7 @@ export function parseLiteralsSectionHeader(data, offset) {
|
|
|
14
14
|
if (offset >= data.length) {
|
|
15
15
|
throw new ZstdError('Literals section header truncated', 'corruption_detected');
|
|
16
16
|
}
|
|
17
|
-
const b0 = data[offset]
|
|
17
|
+
const b0 = data[offset];
|
|
18
18
|
const blockType = (b0 & 3);
|
|
19
19
|
const sizeFormat = (b0 >> 2) & 3;
|
|
20
20
|
if (blockType === 0 || blockType === 1) {
|
|
@@ -29,7 +29,7 @@ export function parseLiteralsSectionHeader(data, offset) {
|
|
|
29
29
|
if (offset + 2 > data.length) {
|
|
30
30
|
throw new ZstdError('Literals section header truncated', 'corruption_detected');
|
|
31
31
|
}
|
|
32
|
-
const b1 = data[offset + 1]
|
|
32
|
+
const b1 = data[offset + 1];
|
|
33
33
|
const regeneratedSize = (b0 >> 4) + (b1 << 4);
|
|
34
34
|
return {
|
|
35
35
|
header: { blockType, regeneratedSize, headerSize: 2, numStreams: 1 },
|
|
@@ -40,8 +40,8 @@ export function parseLiteralsSectionHeader(data, offset) {
|
|
|
40
40
|
if (offset + 3 > data.length) {
|
|
41
41
|
throw new ZstdError('Literals section header truncated', 'corruption_detected');
|
|
42
42
|
}
|
|
43
|
-
const b1 = data[offset + 1]
|
|
44
|
-
const b2 = data[offset + 2]
|
|
43
|
+
const b1 = data[offset + 1];
|
|
44
|
+
const b2 = data[offset + 2];
|
|
45
45
|
const regeneratedSize = (b0 >> 4) + (b1 << 4) + (b2 << 12);
|
|
46
46
|
return {
|
|
47
47
|
header: { blockType, regeneratedSize, headerSize: 3, numStreams: 1 },
|
|
@@ -79,7 +79,8 @@ export function decodeRawLiterals(data, offset, size) {
|
|
|
79
79
|
if (offset + size > data.length) {
|
|
80
80
|
throw new ZstdError('Raw literals truncated', 'corruption_detected');
|
|
81
81
|
}
|
|
82
|
-
|
|
82
|
+
// Return a view to avoid extra copy; callers copy into destination as needed.
|
|
83
|
+
return data.subarray(offset, offset + size);
|
|
83
84
|
}
|
|
84
85
|
/**
|
|
85
86
|
* Decode RLE literals block - single byte repeated.
|
|
@@ -88,7 +89,7 @@ export function decodeRLELiterals(data, offset, size) {
|
|
|
88
89
|
if (offset >= data.length) {
|
|
89
90
|
throw new ZstdError('RLE literals truncated', 'corruption_detected');
|
|
90
91
|
}
|
|
91
|
-
const byte = data[offset]
|
|
92
|
+
const byte = data[offset];
|
|
92
93
|
const result = new Uint8Array(size);
|
|
93
94
|
result.fill(byte);
|
|
94
95
|
return result;
|
|
@@ -110,20 +111,41 @@ function weightsToHuffmanTable(weights) {
|
|
|
110
111
|
throw new ZstdError('Invalid Huffman weights: cannot complete to power of 2', 'corruption_detected');
|
|
111
112
|
}
|
|
112
113
|
const lastWeight = 32 - Math.clz32(remainder);
|
|
113
|
-
const fullWeights =
|
|
114
|
-
|
|
115
|
-
fullWeights
|
|
114
|
+
const fullWeights = new Array(256).fill(0);
|
|
115
|
+
for (let i = 0; i < weights.length; i++) {
|
|
116
|
+
fullWeights[i] = weights[i] ?? 0;
|
|
116
117
|
}
|
|
118
|
+
fullWeights[weights.length] = lastWeight;
|
|
117
119
|
const numBits = weightsToNumBits(fullWeights, maxNumBits);
|
|
118
120
|
const table = buildHuffmanDecodeTable(numBits, maxNumBits);
|
|
119
121
|
return { table, maxNumBits };
|
|
120
122
|
}
|
|
121
|
-
function
|
|
123
|
+
function decodeHuffmanStreamByCountInto(data, streamOffset, streamLength, table, maxNumBits, out, outOffset, numSymbols) {
|
|
124
|
+
if (numSymbols === 0)
|
|
125
|
+
return 0;
|
|
126
|
+
if (streamLength <= 0) {
|
|
127
|
+
throw new ZstdError('Huffman stream truncated', 'corruption_detected');
|
|
128
|
+
}
|
|
129
|
+
const reader = new BitReaderReverse(data, streamOffset, streamLength);
|
|
130
|
+
reader.skipPadding();
|
|
131
|
+
let written = 0;
|
|
132
|
+
for (let i = 0; i < numSymbols; i++) {
|
|
133
|
+
const peek = reader.readBits(maxNumBits);
|
|
134
|
+
const row = table[peek];
|
|
135
|
+
if (!row) {
|
|
136
|
+
throw new ZstdError('Huffman invalid code', 'corruption_detected');
|
|
137
|
+
}
|
|
138
|
+
out[outOffset + written] = row.symbol;
|
|
139
|
+
written++;
|
|
140
|
+
}
|
|
141
|
+
return written;
|
|
142
|
+
}
|
|
143
|
+
function decodeHuffmanStreamToEndInto(data, streamOffset, streamLength, table, maxNumBits, out, outOffset) {
|
|
122
144
|
if (streamLength <= 0) {
|
|
123
145
|
throw new ZstdError('Huffman stream truncated', 'corruption_detected');
|
|
124
146
|
}
|
|
125
147
|
const stream = data.subarray(streamOffset, streamOffset + streamLength);
|
|
126
|
-
const lastByte = stream[stream.length - 1]
|
|
148
|
+
const lastByte = stream[stream.length - 1];
|
|
127
149
|
if (lastByte === 0) {
|
|
128
150
|
throw new ZstdError('Huffman invalid end marker', 'corruption_detected');
|
|
129
151
|
}
|
|
@@ -137,24 +159,28 @@ function decodeHuffmanStreamToEnd(data, streamOffset, streamLength, table, maxNu
|
|
|
137
159
|
let value = 0;
|
|
138
160
|
for (let i = 0; i < numBits; i++) {
|
|
139
161
|
const abs = bitOffset + i;
|
|
140
|
-
if (abs < 0)
|
|
162
|
+
if (abs < 0 || abs >= streamLength * 8)
|
|
141
163
|
continue;
|
|
142
164
|
const byteIndex = abs >>> 3;
|
|
143
165
|
const bitInByte = abs & 7;
|
|
144
|
-
const bit = (
|
|
166
|
+
const bit = (stream[byteIndex] >>> bitInByte) & 1;
|
|
145
167
|
value |= bit << i;
|
|
146
168
|
}
|
|
147
169
|
return value;
|
|
148
170
|
};
|
|
149
|
-
const out = [];
|
|
150
171
|
const mask = (1 << maxNumBits) - 1;
|
|
151
172
|
let state = readBitsZeroExtended(maxNumBits);
|
|
173
|
+
let written = 0;
|
|
152
174
|
while (bitOffset > -maxNumBits) {
|
|
153
175
|
const row = table[state];
|
|
154
176
|
if (!row) {
|
|
155
177
|
throw new ZstdError('Huffman invalid code', 'corruption_detected');
|
|
156
178
|
}
|
|
157
|
-
out.
|
|
179
|
+
if (outOffset + written >= out.length) {
|
|
180
|
+
throw new ZstdError('Huffman literals size mismatch', 'corruption_detected');
|
|
181
|
+
}
|
|
182
|
+
out[outOffset + written] = row.symbol;
|
|
183
|
+
written++;
|
|
158
184
|
const nb = row.numBits;
|
|
159
185
|
const rest = nb > 0 ? readBitsZeroExtended(nb) : 0;
|
|
160
186
|
state = ((state << nb) & mask) + rest;
|
|
@@ -162,26 +188,7 @@ function decodeHuffmanStreamToEnd(data, streamOffset, streamLength, table, maxNu
|
|
|
162
188
|
if (bitOffset !== -maxNumBits) {
|
|
163
189
|
throw new ZstdError('Huffman stream did not end cleanly', 'corruption_detected');
|
|
164
190
|
}
|
|
165
|
-
return
|
|
166
|
-
}
|
|
167
|
-
function decodeHuffmanStreamByCount(data, streamOffset, streamLength, table, maxNumBits, numSymbols) {
|
|
168
|
-
if (numSymbols === 0)
|
|
169
|
-
return new Uint8Array(0);
|
|
170
|
-
if (streamLength <= 0) {
|
|
171
|
-
throw new ZstdError('Huffman stream truncated', 'corruption_detected');
|
|
172
|
-
}
|
|
173
|
-
const reader = new BitReaderReverse(data, streamOffset, streamLength);
|
|
174
|
-
reader.skipPadding();
|
|
175
|
-
const out = new Uint8Array(numSymbols);
|
|
176
|
-
for (let i = 0; i < numSymbols; i++) {
|
|
177
|
-
const peek = reader.readBits(maxNumBits);
|
|
178
|
-
const row = table[peek];
|
|
179
|
-
if (!row) {
|
|
180
|
-
throw new ZstdError('Huffman invalid code', 'corruption_detected');
|
|
181
|
-
}
|
|
182
|
-
out[i] = row.symbol;
|
|
183
|
-
}
|
|
184
|
-
return out;
|
|
191
|
+
return written;
|
|
185
192
|
}
|
|
186
193
|
/**
|
|
187
194
|
* Decode compressed literals (Huffman). Requires Huffman table from tree description.
|
|
@@ -192,7 +199,7 @@ export function decodeCompressedLiterals(data, offset, compressedSize, regenerat
|
|
|
192
199
|
if (pos >= data.length) {
|
|
193
200
|
throw new ZstdError('Huffman tree description truncated', 'corruption_detected');
|
|
194
201
|
}
|
|
195
|
-
const headerByte = data[pos]
|
|
202
|
+
const headerByte = data[pos];
|
|
196
203
|
pos++;
|
|
197
204
|
let weights;
|
|
198
205
|
let treeBytes;
|
|
@@ -217,16 +224,15 @@ export function decodeCompressedLiterals(data, offset, compressedSize, regenerat
|
|
|
217
224
|
const result = new Uint8Array(regeneratedSize);
|
|
218
225
|
let outPos = 0;
|
|
219
226
|
if (numStreams === 1) {
|
|
220
|
-
|
|
221
|
-
result.set(lit);
|
|
227
|
+
decodeHuffmanStreamByCountInto(data, pos, totalStreamsSize, huffmanTable.table, huffmanTable.maxNumBits, result, 0, regeneratedSize);
|
|
222
228
|
}
|
|
223
229
|
else {
|
|
224
230
|
if (totalStreamsSize < 10) {
|
|
225
231
|
throw new ZstdError('4-stream mode requires at least 10 bytes', 'corruption_detected');
|
|
226
232
|
}
|
|
227
|
-
const s1 =
|
|
228
|
-
const s2 =
|
|
229
|
-
const s3 =
|
|
233
|
+
const s1 = data[pos] | (data[pos + 1] << 8);
|
|
234
|
+
const s2 = data[pos + 2] | (data[pos + 3] << 8);
|
|
235
|
+
const s3 = data[pos + 4] | (data[pos + 5] << 8);
|
|
230
236
|
const stream1Size = s1;
|
|
231
237
|
const stream2Size = s2;
|
|
232
238
|
const stream3Size = s3;
|
|
@@ -236,9 +242,8 @@ export function decodeCompressedLiterals(data, offset, compressedSize, regenerat
|
|
|
236
242
|
}
|
|
237
243
|
let streamOffset = pos + 6;
|
|
238
244
|
const decodeStream = (size) => {
|
|
239
|
-
const
|
|
240
|
-
|
|
241
|
-
outPos += lit.length;
|
|
245
|
+
const written = decodeHuffmanStreamToEndInto(data, streamOffset, size, huffmanTable.table, huffmanTable.maxNumBits, result, outPos);
|
|
246
|
+
outPos += written;
|
|
242
247
|
streamOffset += size;
|
|
243
248
|
};
|
|
244
249
|
decodeStream(stream1Size);
|
|
@@ -263,16 +268,15 @@ export function decodeTreelessLiterals(data, offset, compressedSize, regenerated
|
|
|
263
268
|
let outPos = 0;
|
|
264
269
|
let pos = offset;
|
|
265
270
|
if (numStreams === 1) {
|
|
266
|
-
|
|
267
|
-
result.set(lit);
|
|
271
|
+
decodeHuffmanStreamByCountInto(data, pos, compressedSize, huffmanTable.table, huffmanTable.maxNumBits, result, 0, regeneratedSize);
|
|
268
272
|
}
|
|
269
273
|
else {
|
|
270
274
|
if (compressedSize < 10) {
|
|
271
275
|
throw new ZstdError('4-stream mode requires at least 10 bytes', 'corruption_detected');
|
|
272
276
|
}
|
|
273
|
-
const s1 =
|
|
274
|
-
const s2 =
|
|
275
|
-
const s3 =
|
|
277
|
+
const s1 = data[pos] | (data[pos + 1] << 8);
|
|
278
|
+
const s2 = data[pos + 2] | (data[pos + 3] << 8);
|
|
279
|
+
const s3 = data[pos + 4] | (data[pos + 5] << 8);
|
|
276
280
|
const stream1Size = s1;
|
|
277
281
|
const stream2Size = s2;
|
|
278
282
|
const stream3Size = s3;
|
|
@@ -282,9 +286,8 @@ export function decodeTreelessLiterals(data, offset, compressedSize, regenerated
|
|
|
282
286
|
}
|
|
283
287
|
pos += 6;
|
|
284
288
|
const decodeStream = (size) => {
|
|
285
|
-
const
|
|
286
|
-
|
|
287
|
-
outPos += lit.length;
|
|
289
|
+
const written = decodeHuffmanStreamToEndInto(data, pos, size, huffmanTable.table, huffmanTable.maxNumBits, result, outPos);
|
|
290
|
+
outPos += written;
|
|
288
291
|
pos += size;
|
|
289
292
|
};
|
|
290
293
|
decodeStream(stream1Size);
|