@kenzuya/mediabunny 1.26.0 → 1.28.6
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 +1 -1
- package/dist/bundles/{mediabunny.mjs → mediabunny.js} +21963 -21390
- package/dist/bundles/mediabunny.min.js +490 -0
- package/dist/modules/shared/mp3-misc.d.ts.map +1 -1
- package/dist/modules/src/adts/adts-demuxer.d.ts +6 -6
- package/dist/modules/src/adts/adts-demuxer.d.ts.map +1 -1
- package/dist/modules/src/adts/adts-muxer.d.ts +4 -4
- package/dist/modules/src/adts/adts-muxer.d.ts.map +1 -1
- package/dist/modules/src/adts/adts-reader.d.ts +1 -1
- package/dist/modules/src/adts/adts-reader.d.ts.map +1 -1
- package/dist/modules/src/avi/avi-demuxer.d.ts +44 -0
- package/dist/modules/src/avi/avi-demuxer.d.ts.map +1 -0
- package/dist/modules/src/avi/avi-misc.d.ts +88 -0
- package/dist/modules/src/avi/avi-misc.d.ts.map +1 -0
- package/dist/modules/src/avi/avi-muxer.d.ts +45 -0
- package/dist/modules/src/avi/avi-muxer.d.ts.map +1 -0
- package/dist/modules/src/avi/riff-writer.d.ts +26 -0
- package/dist/modules/src/avi/riff-writer.d.ts.map +1 -0
- package/dist/modules/src/codec-data.d.ts +8 -3
- package/dist/modules/src/codec-data.d.ts.map +1 -1
- package/dist/modules/src/codec.d.ts +10 -10
- package/dist/modules/src/codec.d.ts.map +1 -1
- package/dist/modules/src/conversion.d.ts +33 -16
- package/dist/modules/src/conversion.d.ts.map +1 -1
- package/dist/modules/src/custom-coder.d.ts +8 -8
- package/dist/modules/src/custom-coder.d.ts.map +1 -1
- package/dist/modules/src/demuxer.d.ts +3 -3
- package/dist/modules/src/demuxer.d.ts.map +1 -1
- package/dist/modules/src/encode.d.ts +8 -8
- package/dist/modules/src/encode.d.ts.map +1 -1
- package/dist/modules/src/flac/flac-demuxer.d.ts +7 -7
- package/dist/modules/src/flac/flac-demuxer.d.ts.map +1 -1
- package/dist/modules/src/flac/flac-misc.d.ts +3 -3
- package/dist/modules/src/flac/flac-misc.d.ts.map +1 -1
- package/dist/modules/src/flac/flac-muxer.d.ts +5 -5
- package/dist/modules/src/flac/flac-muxer.d.ts.map +1 -1
- package/dist/modules/src/id3.d.ts +3 -3
- package/dist/modules/src/id3.d.ts.map +1 -1
- package/dist/modules/src/index.d.ts +20 -20
- package/dist/modules/src/index.d.ts.map +1 -1
- package/dist/modules/src/input-format.d.ts +22 -0
- package/dist/modules/src/input-format.d.ts.map +1 -1
- package/dist/modules/src/input-track.d.ts +8 -8
- package/dist/modules/src/input-track.d.ts.map +1 -1
- package/dist/modules/src/input.d.ts +12 -12
- package/dist/modules/src/isobmff/isobmff-boxes.d.ts +2 -2
- package/dist/modules/src/isobmff/isobmff-boxes.d.ts.map +1 -1
- package/dist/modules/src/isobmff/isobmff-demuxer.d.ts +12 -12
- package/dist/modules/src/isobmff/isobmff-demuxer.d.ts.map +1 -1
- package/dist/modules/src/isobmff/isobmff-misc.d.ts.map +1 -1
- package/dist/modules/src/isobmff/isobmff-muxer.d.ts +11 -11
- package/dist/modules/src/isobmff/isobmff-muxer.d.ts.map +1 -1
- package/dist/modules/src/isobmff/isobmff-reader.d.ts +2 -2
- package/dist/modules/src/isobmff/isobmff-reader.d.ts.map +1 -1
- package/dist/modules/src/matroska/ebml.d.ts +3 -3
- package/dist/modules/src/matroska/ebml.d.ts.map +1 -1
- package/dist/modules/src/matroska/matroska-demuxer.d.ts +13 -13
- package/dist/modules/src/matroska/matroska-demuxer.d.ts.map +1 -1
- package/dist/modules/src/matroska/matroska-input.d.ts +33 -0
- package/dist/modules/src/matroska/matroska-input.d.ts.map +1 -0
- package/dist/modules/src/matroska/matroska-misc.d.ts.map +1 -1
- package/dist/modules/src/matroska/matroska-muxer.d.ts +5 -5
- package/dist/modules/src/matroska/matroska-muxer.d.ts.map +1 -1
- package/dist/modules/src/media-sink.d.ts +5 -5
- package/dist/modules/src/media-sink.d.ts.map +1 -1
- package/dist/modules/src/media-source.d.ts +22 -4
- package/dist/modules/src/media-source.d.ts.map +1 -1
- package/dist/modules/src/metadata.d.ts +2 -2
- package/dist/modules/src/metadata.d.ts.map +1 -1
- package/dist/modules/src/misc.d.ts +5 -4
- package/dist/modules/src/misc.d.ts.map +1 -1
- package/dist/modules/src/mp3/mp3-demuxer.d.ts +7 -7
- package/dist/modules/src/mp3/mp3-demuxer.d.ts.map +1 -1
- package/dist/modules/src/mp3/mp3-muxer.d.ts +4 -4
- package/dist/modules/src/mp3/mp3-muxer.d.ts.map +1 -1
- package/dist/modules/src/mp3/mp3-reader.d.ts +2 -2
- package/dist/modules/src/mp3/mp3-reader.d.ts.map +1 -1
- package/dist/modules/src/mp3/mp3-writer.d.ts +1 -1
- package/dist/modules/src/mp3/mp3-writer.d.ts.map +1 -1
- package/dist/modules/src/muxer.d.ts +4 -4
- package/dist/modules/src/muxer.d.ts.map +1 -1
- package/dist/modules/src/ogg/ogg-demuxer.d.ts +7 -7
- package/dist/modules/src/ogg/ogg-demuxer.d.ts.map +1 -1
- package/dist/modules/src/ogg/ogg-misc.d.ts +1 -1
- package/dist/modules/src/ogg/ogg-misc.d.ts.map +1 -1
- package/dist/modules/src/ogg/ogg-muxer.d.ts +5 -5
- package/dist/modules/src/ogg/ogg-muxer.d.ts.map +1 -1
- package/dist/modules/src/ogg/ogg-reader.d.ts +1 -1
- package/dist/modules/src/ogg/ogg-reader.d.ts.map +1 -1
- package/dist/modules/src/output-format.d.ts +51 -6
- package/dist/modules/src/output-format.d.ts.map +1 -1
- package/dist/modules/src/output.d.ts +13 -13
- package/dist/modules/src/output.d.ts.map +1 -1
- package/dist/modules/src/packet.d.ts +1 -1
- package/dist/modules/src/packet.d.ts.map +1 -1
- package/dist/modules/src/pcm.d.ts.map +1 -1
- package/dist/modules/src/reader.d.ts +2 -2
- package/dist/modules/src/reader.d.ts.map +1 -1
- package/dist/modules/src/sample.d.ts +57 -15
- package/dist/modules/src/sample.d.ts.map +1 -1
- package/dist/modules/src/source.d.ts +3 -3
- package/dist/modules/src/source.d.ts.map +1 -1
- package/dist/modules/src/subtitles.d.ts +1 -1
- package/dist/modules/src/subtitles.d.ts.map +1 -1
- package/dist/modules/src/target.d.ts +2 -2
- package/dist/modules/src/target.d.ts.map +1 -1
- package/dist/modules/src/tsconfig.tsbuildinfo +1 -1
- package/dist/modules/src/wave/riff-writer.d.ts +1 -1
- package/dist/modules/src/wave/riff-writer.d.ts.map +1 -1
- package/dist/modules/src/wave/wave-demuxer.d.ts +6 -6
- package/dist/modules/src/wave/wave-demuxer.d.ts.map +1 -1
- package/dist/modules/src/wave/wave-muxer.d.ts +4 -4
- package/dist/modules/src/wave/wave-muxer.d.ts.map +1 -1
- package/dist/modules/src/writer.d.ts +1 -1
- package/dist/modules/src/writer.d.ts.map +1 -1
- package/dist/packages/eac3/eac3.wasm +0 -0
- package/dist/packages/eac3/mediabunny-eac3.js +1058 -0
- package/dist/packages/eac3/mediabunny-eac3.min.js +44 -0
- package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.js +694 -0
- package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.min.js +58 -0
- package/dist/packages/mpeg4/mediabunny-mpeg4.js +1198 -0
- package/dist/packages/mpeg4/mediabunny-mpeg4.min.js +44 -0
- package/dist/packages/mpeg4/xvid.wasm +0 -0
- package/package.json +18 -57
- package/dist/bundles/mediabunny.cjs +0 -26140
- package/dist/bundles/mediabunny.min.cjs +0 -147
- package/dist/bundles/mediabunny.min.mjs +0 -146
- package/dist/mediabunny.d.ts +0 -3319
- package/dist/modules/shared/mp3-misc.js +0 -147
- package/dist/modules/src/adts/adts-demuxer.js +0 -239
- package/dist/modules/src/adts/adts-muxer.js +0 -80
- package/dist/modules/src/adts/adts-reader.js +0 -63
- package/dist/modules/src/codec-data.js +0 -1730
- package/dist/modules/src/codec.js +0 -869
- package/dist/modules/src/conversion.js +0 -1459
- package/dist/modules/src/custom-coder.js +0 -117
- package/dist/modules/src/demuxer.js +0 -12
- package/dist/modules/src/encode.js +0 -442
- package/dist/modules/src/flac/flac-demuxer.js +0 -504
- package/dist/modules/src/flac/flac-misc.js +0 -135
- package/dist/modules/src/flac/flac-muxer.js +0 -222
- package/dist/modules/src/id3.js +0 -848
- package/dist/modules/src/index.js +0 -28
- package/dist/modules/src/input-format.js +0 -480
- package/dist/modules/src/input-track.js +0 -372
- package/dist/modules/src/input.js +0 -188
- package/dist/modules/src/isobmff/isobmff-boxes.js +0 -1480
- package/dist/modules/src/isobmff/isobmff-demuxer.js +0 -2618
- package/dist/modules/src/isobmff/isobmff-misc.js +0 -20
- package/dist/modules/src/isobmff/isobmff-muxer.js +0 -966
- package/dist/modules/src/isobmff/isobmff-reader.js +0 -72
- package/dist/modules/src/matroska/ebml.js +0 -653
- package/dist/modules/src/matroska/matroska-demuxer.js +0 -2133
- package/dist/modules/src/matroska/matroska-misc.js +0 -20
- package/dist/modules/src/matroska/matroska-muxer.js +0 -1017
- package/dist/modules/src/media-sink.js +0 -1736
- package/dist/modules/src/media-source.js +0 -1825
- package/dist/modules/src/metadata.js +0 -193
- package/dist/modules/src/misc.js +0 -623
- package/dist/modules/src/mp3/mp3-demuxer.js +0 -285
- package/dist/modules/src/mp3/mp3-muxer.js +0 -123
- package/dist/modules/src/mp3/mp3-reader.js +0 -26
- package/dist/modules/src/mp3/mp3-writer.js +0 -78
- package/dist/modules/src/muxer.js +0 -50
- package/dist/modules/src/node.d.ts +0 -9
- package/dist/modules/src/node.d.ts.map +0 -1
- package/dist/modules/src/node.js +0 -9
- package/dist/modules/src/ogg/ogg-demuxer.js +0 -763
- package/dist/modules/src/ogg/ogg-misc.js +0 -78
- package/dist/modules/src/ogg/ogg-muxer.js +0 -353
- package/dist/modules/src/ogg/ogg-reader.js +0 -65
- package/dist/modules/src/output-format.js +0 -527
- package/dist/modules/src/output.js +0 -300
- package/dist/modules/src/packet.js +0 -182
- package/dist/modules/src/pcm.js +0 -85
- package/dist/modules/src/reader.js +0 -236
- package/dist/modules/src/sample.js +0 -1056
- package/dist/modules/src/source.js +0 -1182
- package/dist/modules/src/subtitles.js +0 -575
- package/dist/modules/src/target.js +0 -140
- package/dist/modules/src/wave/riff-writer.js +0 -30
- package/dist/modules/src/wave/wave-demuxer.js +0 -447
- package/dist/modules/src/wave/wave-muxer.js +0 -318
- package/dist/modules/src/writer.js +0 -370
- package/src/adts/adts-demuxer.ts +0 -331
- package/src/adts/adts-muxer.ts +0 -111
- package/src/adts/adts-reader.ts +0 -85
- package/src/codec-data.ts +0 -2078
- package/src/codec.ts +0 -1092
- package/src/conversion.ts +0 -2112
- package/src/custom-coder.ts +0 -197
- package/src/demuxer.ts +0 -24
- package/src/encode.ts +0 -739
- package/src/flac/flac-demuxer.ts +0 -730
- package/src/flac/flac-misc.ts +0 -164
- package/src/flac/flac-muxer.ts +0 -320
- package/src/id3.ts +0 -925
- package/src/index.ts +0 -221
- package/src/input-format.ts +0 -541
- package/src/input-track.ts +0 -529
- package/src/input.ts +0 -235
- package/src/isobmff/isobmff-boxes.ts +0 -1719
- package/src/isobmff/isobmff-demuxer.ts +0 -3190
- package/src/isobmff/isobmff-misc.ts +0 -29
- package/src/isobmff/isobmff-muxer.ts +0 -1348
- package/src/isobmff/isobmff-reader.ts +0 -91
- package/src/matroska/ebml.ts +0 -730
- package/src/matroska/matroska-demuxer.ts +0 -2481
- package/src/matroska/matroska-misc.ts +0 -29
- package/src/matroska/matroska-muxer.ts +0 -1276
- package/src/media-sink.ts +0 -2179
- package/src/media-source.ts +0 -2243
- package/src/metadata.ts +0 -320
- package/src/misc.ts +0 -798
- package/src/mp3/mp3-demuxer.ts +0 -383
- package/src/mp3/mp3-muxer.ts +0 -166
- package/src/mp3/mp3-reader.ts +0 -34
- package/src/mp3/mp3-writer.ts +0 -120
- package/src/muxer.ts +0 -88
- package/src/node.ts +0 -11
- package/src/ogg/ogg-demuxer.ts +0 -1053
- package/src/ogg/ogg-misc.ts +0 -116
- package/src/ogg/ogg-muxer.ts +0 -497
- package/src/ogg/ogg-reader.ts +0 -93
- package/src/output-format.ts +0 -945
- package/src/output.ts +0 -488
- package/src/packet.ts +0 -263
- package/src/pcm.ts +0 -112
- package/src/reader.ts +0 -323
- package/src/sample.ts +0 -1461
- package/src/source.ts +0 -1688
- package/src/subtitles.ts +0 -711
- package/src/target.ts +0 -204
- package/src/tsconfig.json +0 -16
- package/src/wave/riff-writer.ts +0 -36
- package/src/wave/wave-demuxer.ts +0 -529
- package/src/wave/wave-muxer.ts +0 -371
- package/src/writer.ts +0 -490
package/src/ogg/ogg-demuxer.ts
DELETED
|
@@ -1,1053 +0,0 @@
|
|
|
1
|
-
/*!
|
|
2
|
-
* Copyright (c) 2025-present, Vanilagy and contributors
|
|
3
|
-
*
|
|
4
|
-
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
5
|
-
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
6
|
-
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import { OPUS_SAMPLE_RATE } from '../codec';
|
|
10
|
-
import { parseModesFromVorbisSetupPacket, parseOpusIdentificationHeader, readVorbisComments } from '../codec-data';
|
|
11
|
-
import { Demuxer } from '../demuxer';
|
|
12
|
-
import { Input } from '../input';
|
|
13
|
-
import { InputAudioTrack, InputAudioTrackBacking } from '../input-track';
|
|
14
|
-
import { PacketRetrievalOptions } from '../media-sink';
|
|
15
|
-
import { DEFAULT_TRACK_DISPOSITION, MetadataTags } from '../metadata';
|
|
16
|
-
import {
|
|
17
|
-
assert,
|
|
18
|
-
AsyncMutex,
|
|
19
|
-
binarySearchLessOrEqual,
|
|
20
|
-
findLast,
|
|
21
|
-
last,
|
|
22
|
-
roundIfAlmostInteger,
|
|
23
|
-
toDataView,
|
|
24
|
-
UNDETERMINED_LANGUAGE,
|
|
25
|
-
} from '../misc';
|
|
26
|
-
import { EncodedPacket, PLACEHOLDER_DATA } from '../packet';
|
|
27
|
-
import { readBytes, Reader } from '../reader';
|
|
28
|
-
import { buildOggMimeType, computeOggPageCrc, extractSampleMetadata, OggCodecInfo } from './ogg-misc';
|
|
29
|
-
import {
|
|
30
|
-
findNextPageHeader,
|
|
31
|
-
MAX_PAGE_HEADER_SIZE,
|
|
32
|
-
MAX_PAGE_SIZE,
|
|
33
|
-
MIN_PAGE_HEADER_SIZE,
|
|
34
|
-
Page,
|
|
35
|
-
readPageHeader,
|
|
36
|
-
} from './ogg-reader';
|
|
37
|
-
|
|
38
|
-
type LogicalBitstream = {
|
|
39
|
-
serialNumber: number;
|
|
40
|
-
bosPage: Page;
|
|
41
|
-
description: Uint8Array | null;
|
|
42
|
-
numberOfChannels: number;
|
|
43
|
-
sampleRate: number;
|
|
44
|
-
codecInfo: OggCodecInfo;
|
|
45
|
-
|
|
46
|
-
lastMetadataPacket: Packet | null;
|
|
47
|
-
};
|
|
48
|
-
|
|
49
|
-
type Packet = {
|
|
50
|
-
data: Uint8Array;
|
|
51
|
-
endPage: Page;
|
|
52
|
-
endSegmentIndex: number;
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
export class OggDemuxer extends Demuxer {
|
|
56
|
-
reader: Reader;
|
|
57
|
-
|
|
58
|
-
metadataPromise: Promise<void> | null = null;
|
|
59
|
-
bitstreams: LogicalBitstream[] = [];
|
|
60
|
-
tracks: InputAudioTrack[] = [];
|
|
61
|
-
metadataTags: MetadataTags = {};
|
|
62
|
-
|
|
63
|
-
constructor(input: Input) {
|
|
64
|
-
super(input);
|
|
65
|
-
|
|
66
|
-
this.reader = input._reader;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
async readMetadata() {
|
|
70
|
-
return this.metadataPromise ??= (async () => {
|
|
71
|
-
let currentPos = 0;
|
|
72
|
-
|
|
73
|
-
while (true) {
|
|
74
|
-
let slice = this.reader.requestSliceRange(currentPos, MIN_PAGE_HEADER_SIZE, MAX_PAGE_HEADER_SIZE);
|
|
75
|
-
if (slice instanceof Promise) slice = await slice;
|
|
76
|
-
if (!slice) break;
|
|
77
|
-
|
|
78
|
-
const page = readPageHeader(slice);
|
|
79
|
-
if (!page) {
|
|
80
|
-
break;
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
const isBos = !!(page.headerType & 0x02);
|
|
84
|
-
if (!isBos) {
|
|
85
|
-
// All bos pages for all bitstreams are required to be at the start, so if the page is not bos then
|
|
86
|
-
// we know we've seen all bitstreams (minus chaining)
|
|
87
|
-
break;
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
this.bitstreams.push({
|
|
91
|
-
serialNumber: page.serialNumber,
|
|
92
|
-
bosPage: page,
|
|
93
|
-
description: null,
|
|
94
|
-
numberOfChannels: -1,
|
|
95
|
-
sampleRate: -1,
|
|
96
|
-
codecInfo: {
|
|
97
|
-
codec: null,
|
|
98
|
-
vorbisInfo: null,
|
|
99
|
-
opusInfo: null,
|
|
100
|
-
},
|
|
101
|
-
lastMetadataPacket: null,
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
currentPos = page.headerStartPos + page.totalSize;
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
for (const bitstream of this.bitstreams) {
|
|
108
|
-
const firstPacket = await this.readPacket(bitstream.bosPage, 0);
|
|
109
|
-
if (!firstPacket) {
|
|
110
|
-
continue;
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
if (
|
|
114
|
-
// Check for Vorbis
|
|
115
|
-
firstPacket.data.byteLength >= 7
|
|
116
|
-
&& firstPacket.data[0] === 0x01 // Packet type 1 = identification header
|
|
117
|
-
&& firstPacket.data[1] === 0x76 // 'v'
|
|
118
|
-
&& firstPacket.data[2] === 0x6f // 'o'
|
|
119
|
-
&& firstPacket.data[3] === 0x72 // 'r'
|
|
120
|
-
&& firstPacket.data[4] === 0x62 // 'b'
|
|
121
|
-
&& firstPacket.data[5] === 0x69 // 'i'
|
|
122
|
-
&& firstPacket.data[6] === 0x73 // 's'
|
|
123
|
-
) {
|
|
124
|
-
await this.readVorbisMetadata(firstPacket, bitstream);
|
|
125
|
-
} else if (
|
|
126
|
-
// Check for Opus
|
|
127
|
-
firstPacket.data.byteLength >= 8
|
|
128
|
-
&& firstPacket.data[0] === 0x4f // 'O'
|
|
129
|
-
&& firstPacket.data[1] === 0x70 // 'p'
|
|
130
|
-
&& firstPacket.data[2] === 0x75 // 'u'
|
|
131
|
-
&& firstPacket.data[3] === 0x73 // 's'
|
|
132
|
-
&& firstPacket.data[4] === 0x48 // 'H'
|
|
133
|
-
&& firstPacket.data[5] === 0x65 // 'e'
|
|
134
|
-
&& firstPacket.data[6] === 0x61 // 'a'
|
|
135
|
-
&& firstPacket.data[7] === 0x64 // 'd'
|
|
136
|
-
) {
|
|
137
|
-
await this.readOpusMetadata(firstPacket, bitstream);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
if (bitstream.codecInfo.codec !== null) {
|
|
141
|
-
this.tracks.push(new InputAudioTrack(this.input, new OggAudioTrackBacking(bitstream, this)));
|
|
142
|
-
}
|
|
143
|
-
}
|
|
144
|
-
})();
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
async readVorbisMetadata(firstPacket: Packet, bitstream: LogicalBitstream) {
|
|
148
|
-
let nextPacketPosition = await this.findNextPacketStart(firstPacket);
|
|
149
|
-
if (!nextPacketPosition) {
|
|
150
|
-
return;
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
const secondPacket = await this.readPacket(nextPacketPosition.startPage, nextPacketPosition.startSegmentIndex);
|
|
154
|
-
if (!secondPacket) {
|
|
155
|
-
return;
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
nextPacketPosition = await this.findNextPacketStart(secondPacket);
|
|
159
|
-
if (!nextPacketPosition) {
|
|
160
|
-
return;
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
const thirdPacket = await this.readPacket(nextPacketPosition.startPage, nextPacketPosition.startSegmentIndex);
|
|
164
|
-
if (!thirdPacket) {
|
|
165
|
-
return;
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
if (secondPacket.data[0] !== 0x03 || thirdPacket.data[0] !== 0x05) {
|
|
169
|
-
return;
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
const lacingValues: number[] = [];
|
|
173
|
-
const addBytesToSegmentTable = (bytes: number) => {
|
|
174
|
-
while (true) {
|
|
175
|
-
lacingValues.push(Math.min(255, bytes));
|
|
176
|
-
|
|
177
|
-
if (bytes < 255) {
|
|
178
|
-
break;
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
bytes -= 255;
|
|
182
|
-
}
|
|
183
|
-
};
|
|
184
|
-
|
|
185
|
-
addBytesToSegmentTable(firstPacket.data.length);
|
|
186
|
-
addBytesToSegmentTable(secondPacket.data.length);
|
|
187
|
-
// We don't add the last packet to the segment table, as it is assumed to be whatever bytes remain
|
|
188
|
-
|
|
189
|
-
const description = new Uint8Array(
|
|
190
|
-
1 + lacingValues.length
|
|
191
|
-
+ firstPacket.data.length + secondPacket.data.length + thirdPacket.data.length,
|
|
192
|
-
);
|
|
193
|
-
description[0] = 2; // Num entries in the segment table
|
|
194
|
-
description.set(
|
|
195
|
-
lacingValues, 1,
|
|
196
|
-
);
|
|
197
|
-
description.set(
|
|
198
|
-
firstPacket.data, 1 + lacingValues.length,
|
|
199
|
-
);
|
|
200
|
-
description.set(
|
|
201
|
-
secondPacket.data, 1 + lacingValues.length + firstPacket.data.length,
|
|
202
|
-
);
|
|
203
|
-
description.set(
|
|
204
|
-
thirdPacket.data, 1 + lacingValues.length + firstPacket.data.length + secondPacket.data.length,
|
|
205
|
-
);
|
|
206
|
-
|
|
207
|
-
bitstream.codecInfo.codec = 'vorbis';
|
|
208
|
-
bitstream.description = description;
|
|
209
|
-
bitstream.lastMetadataPacket = thirdPacket;
|
|
210
|
-
|
|
211
|
-
const view = toDataView(firstPacket.data);
|
|
212
|
-
bitstream.numberOfChannels = view.getUint8(11);
|
|
213
|
-
bitstream.sampleRate = view.getUint32(12, true);
|
|
214
|
-
|
|
215
|
-
const blockSizeByte = view.getUint8(28);
|
|
216
|
-
bitstream.codecInfo.vorbisInfo = {
|
|
217
|
-
blocksizes: [
|
|
218
|
-
1 << (blockSizeByte & 0xf),
|
|
219
|
-
1 << (blockSizeByte >> 4),
|
|
220
|
-
],
|
|
221
|
-
modeBlockflags: parseModesFromVorbisSetupPacket(thirdPacket.data).modeBlockflags,
|
|
222
|
-
};
|
|
223
|
-
|
|
224
|
-
readVorbisComments(secondPacket.data.subarray(7), this.metadataTags); // Skip header type and 'vorbis'
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
async readOpusMetadata(firstPacket: Packet, bitstream: LogicalBitstream) {
|
|
228
|
-
// From https://datatracker.ietf.org/doc/html/rfc7845#section-5:
|
|
229
|
-
// "An Ogg Opus logical stream contains exactly two mandatory header packets: an identification header and a
|
|
230
|
-
// comment header."
|
|
231
|
-
const nextPacketPosition = await this.findNextPacketStart(firstPacket);
|
|
232
|
-
if (!nextPacketPosition) {
|
|
233
|
-
return;
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
const secondPacket = await this.readPacket(
|
|
237
|
-
nextPacketPosition.startPage,
|
|
238
|
-
nextPacketPosition.startSegmentIndex,
|
|
239
|
-
);
|
|
240
|
-
if (!secondPacket) {
|
|
241
|
-
return;
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
bitstream.codecInfo.codec = 'opus';
|
|
245
|
-
bitstream.description = firstPacket.data;
|
|
246
|
-
bitstream.lastMetadataPacket = secondPacket;
|
|
247
|
-
|
|
248
|
-
const header = parseOpusIdentificationHeader(firstPacket.data);
|
|
249
|
-
bitstream.numberOfChannels = header.outputChannelCount;
|
|
250
|
-
bitstream.sampleRate = OPUS_SAMPLE_RATE; // Always the same
|
|
251
|
-
|
|
252
|
-
bitstream.codecInfo.opusInfo = {
|
|
253
|
-
preSkip: header.preSkip,
|
|
254
|
-
};
|
|
255
|
-
|
|
256
|
-
readVorbisComments(secondPacket.data.subarray(8), this.metadataTags); // Skip 'OpusTags'
|
|
257
|
-
}
|
|
258
|
-
|
|
259
|
-
async readPacket(startPage: Page, startSegmentIndex: number): Promise<Packet | null> {
|
|
260
|
-
assert(startSegmentIndex < startPage.lacingValues.length);
|
|
261
|
-
|
|
262
|
-
let startDataOffset = 0;
|
|
263
|
-
for (let i = 0; i < startSegmentIndex; i++) {
|
|
264
|
-
startDataOffset += startPage.lacingValues[i]!;
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
let currentPage: Page = startPage;
|
|
268
|
-
let currentDataOffset = startDataOffset;
|
|
269
|
-
let currentSegmentIndex = startSegmentIndex;
|
|
270
|
-
|
|
271
|
-
const chunks: Uint8Array[] = [];
|
|
272
|
-
|
|
273
|
-
outer:
|
|
274
|
-
while (true) {
|
|
275
|
-
// Load the entire page data
|
|
276
|
-
let pageSlice = this.reader.requestSlice(currentPage.dataStartPos, currentPage.dataSize);
|
|
277
|
-
if (pageSlice instanceof Promise) pageSlice = await pageSlice;
|
|
278
|
-
assert(pageSlice);
|
|
279
|
-
const pageData = readBytes(pageSlice, currentPage.dataSize);
|
|
280
|
-
|
|
281
|
-
while (true) {
|
|
282
|
-
if (currentSegmentIndex === currentPage.lacingValues.length) {
|
|
283
|
-
chunks.push(pageData.subarray(startDataOffset, currentDataOffset));
|
|
284
|
-
break;
|
|
285
|
-
}
|
|
286
|
-
|
|
287
|
-
const lacingValue = currentPage.lacingValues[currentSegmentIndex]!;
|
|
288
|
-
currentDataOffset += lacingValue;
|
|
289
|
-
|
|
290
|
-
if (lacingValue < 255) {
|
|
291
|
-
chunks.push(pageData.subarray(startDataOffset, currentDataOffset));
|
|
292
|
-
break outer;
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
currentSegmentIndex++;
|
|
296
|
-
}
|
|
297
|
-
|
|
298
|
-
// The packet extends to the next page; let's find it
|
|
299
|
-
let currentPos = currentPage.headerStartPos + currentPage.totalSize;
|
|
300
|
-
while (true) {
|
|
301
|
-
let headerSlice = this.reader.requestSliceRange(currentPos, MIN_PAGE_HEADER_SIZE, MAX_PAGE_HEADER_SIZE);
|
|
302
|
-
if (headerSlice instanceof Promise) headerSlice = await headerSlice;
|
|
303
|
-
if (!headerSlice) {
|
|
304
|
-
return null;
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
const nextPage = readPageHeader(headerSlice);
|
|
308
|
-
if (!nextPage) {
|
|
309
|
-
return null;
|
|
310
|
-
}
|
|
311
|
-
|
|
312
|
-
currentPage = nextPage;
|
|
313
|
-
if (currentPage.serialNumber === startPage.serialNumber) {
|
|
314
|
-
break;
|
|
315
|
-
}
|
|
316
|
-
currentPos = currentPage.headerStartPos + currentPage.totalSize;
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
startDataOffset = 0;
|
|
320
|
-
currentDataOffset = 0;
|
|
321
|
-
currentSegmentIndex = 0;
|
|
322
|
-
}
|
|
323
|
-
|
|
324
|
-
const totalPacketSize = chunks.reduce((sum, chunk) => sum + chunk.length, 0);
|
|
325
|
-
const packetData = new Uint8Array(totalPacketSize);
|
|
326
|
-
|
|
327
|
-
let offset = 0;
|
|
328
|
-
for (let i = 0; i < chunks.length; i++) {
|
|
329
|
-
const chunk = chunks[i]!;
|
|
330
|
-
packetData.set(chunk, offset);
|
|
331
|
-
offset += chunk.length;
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
return {
|
|
335
|
-
data: packetData,
|
|
336
|
-
endPage: currentPage,
|
|
337
|
-
endSegmentIndex: currentSegmentIndex,
|
|
338
|
-
};
|
|
339
|
-
}
|
|
340
|
-
|
|
341
|
-
async findNextPacketStart(lastPacket: Packet) {
|
|
342
|
-
// If there's another segment in the same page, return it
|
|
343
|
-
if (lastPacket.endSegmentIndex < lastPacket.endPage.lacingValues.length - 1) {
|
|
344
|
-
return { startPage: lastPacket.endPage, startSegmentIndex: lastPacket.endSegmentIndex + 1 };
|
|
345
|
-
}
|
|
346
|
-
|
|
347
|
-
const isEos = !!(lastPacket.endPage.headerType & 0x04);
|
|
348
|
-
if (isEos) {
|
|
349
|
-
// The page is marked as the last page of the logical bitstream, so we won't find anything beyond it
|
|
350
|
-
return null;
|
|
351
|
-
}
|
|
352
|
-
|
|
353
|
-
// Otherwise, search for the next page belonging to the same bitstream
|
|
354
|
-
let currentPos = lastPacket.endPage.headerStartPos + lastPacket.endPage.totalSize;
|
|
355
|
-
while (true) {
|
|
356
|
-
let slice = this.reader.requestSliceRange(currentPos, MIN_PAGE_HEADER_SIZE, MAX_PAGE_HEADER_SIZE);
|
|
357
|
-
if (slice instanceof Promise) slice = await slice;
|
|
358
|
-
if (!slice) {
|
|
359
|
-
return null;
|
|
360
|
-
}
|
|
361
|
-
|
|
362
|
-
const nextPage = readPageHeader(slice);
|
|
363
|
-
if (!nextPage) {
|
|
364
|
-
return null;
|
|
365
|
-
}
|
|
366
|
-
|
|
367
|
-
if (nextPage.serialNumber === lastPacket.endPage.serialNumber) {
|
|
368
|
-
return { startPage: nextPage, startSegmentIndex: 0 };
|
|
369
|
-
}
|
|
370
|
-
|
|
371
|
-
currentPos = nextPage.headerStartPos + nextPage.totalSize;
|
|
372
|
-
}
|
|
373
|
-
}
|
|
374
|
-
|
|
375
|
-
async getMimeType() {
|
|
376
|
-
await this.readMetadata();
|
|
377
|
-
|
|
378
|
-
const codecStrings = await Promise.all(this.tracks.map(x => x.getCodecParameterString()));
|
|
379
|
-
|
|
380
|
-
return buildOggMimeType({
|
|
381
|
-
codecStrings: codecStrings.filter(Boolean) as string[],
|
|
382
|
-
});
|
|
383
|
-
}
|
|
384
|
-
|
|
385
|
-
async getTracks() {
|
|
386
|
-
await this.readMetadata();
|
|
387
|
-
return this.tracks;
|
|
388
|
-
}
|
|
389
|
-
|
|
390
|
-
async computeDuration() {
|
|
391
|
-
const tracks = await this.getTracks();
|
|
392
|
-
const trackDurations = await Promise.all(tracks.map(x => x.computeDuration()));
|
|
393
|
-
return Math.max(0, ...trackDurations);
|
|
394
|
-
}
|
|
395
|
-
|
|
396
|
-
async getMetadataTags() {
|
|
397
|
-
await this.readMetadata();
|
|
398
|
-
return this.metadataTags;
|
|
399
|
-
}
|
|
400
|
-
}
|
|
401
|
-
|
|
402
|
-
type EncodedPacketMetadata = {
|
|
403
|
-
packet: Packet;
|
|
404
|
-
timestampInSamples: number;
|
|
405
|
-
durationInSamples: number;
|
|
406
|
-
vorbisLastBlockSize: number | null;
|
|
407
|
-
vorbisBlockSize: number | null;
|
|
408
|
-
};
|
|
409
|
-
|
|
410
|
-
class OggAudioTrackBacking implements InputAudioTrackBacking {
|
|
411
|
-
internalSampleRate: number;
|
|
412
|
-
encodedPacketToMetadata = new WeakMap<EncodedPacket, EncodedPacketMetadata>();
|
|
413
|
-
sequentialScanCache: EncodedPacketMetadata[] = [];
|
|
414
|
-
sequentialScanMutex = new AsyncMutex();
|
|
415
|
-
|
|
416
|
-
constructor(public bitstream: LogicalBitstream, public demuxer: OggDemuxer) {
|
|
417
|
-
// Opus always uses a fixed sample rate for its internal calculations, even if the actual rate is different
|
|
418
|
-
this.internalSampleRate = bitstream.codecInfo.codec === 'opus'
|
|
419
|
-
? OPUS_SAMPLE_RATE
|
|
420
|
-
: bitstream.sampleRate;
|
|
421
|
-
}
|
|
422
|
-
|
|
423
|
-
getId() {
|
|
424
|
-
return this.bitstream.serialNumber;
|
|
425
|
-
}
|
|
426
|
-
|
|
427
|
-
getNumberOfChannels() {
|
|
428
|
-
return this.bitstream.numberOfChannels;
|
|
429
|
-
}
|
|
430
|
-
|
|
431
|
-
getSampleRate() {
|
|
432
|
-
return this.bitstream.sampleRate;
|
|
433
|
-
}
|
|
434
|
-
|
|
435
|
-
getTimeResolution() {
|
|
436
|
-
return this.bitstream.sampleRate;
|
|
437
|
-
}
|
|
438
|
-
|
|
439
|
-
getCodec() {
|
|
440
|
-
return this.bitstream.codecInfo.codec;
|
|
441
|
-
}
|
|
442
|
-
|
|
443
|
-
getInternalCodecId() {
|
|
444
|
-
return null;
|
|
445
|
-
}
|
|
446
|
-
|
|
447
|
-
async getDecoderConfig(): Promise<AudioDecoderConfig | null> {
|
|
448
|
-
assert(this.bitstream.codecInfo.codec);
|
|
449
|
-
|
|
450
|
-
return {
|
|
451
|
-
codec: this.bitstream.codecInfo.codec,
|
|
452
|
-
numberOfChannels: this.bitstream.numberOfChannels,
|
|
453
|
-
sampleRate: this.bitstream.sampleRate,
|
|
454
|
-
description: this.bitstream.description ?? undefined,
|
|
455
|
-
};
|
|
456
|
-
}
|
|
457
|
-
|
|
458
|
-
getName() {
|
|
459
|
-
return null;
|
|
460
|
-
}
|
|
461
|
-
|
|
462
|
-
getLanguageCode() {
|
|
463
|
-
return UNDETERMINED_LANGUAGE;
|
|
464
|
-
}
|
|
465
|
-
|
|
466
|
-
getDisposition() {
|
|
467
|
-
return {
|
|
468
|
-
...DEFAULT_TRACK_DISPOSITION,
|
|
469
|
-
};
|
|
470
|
-
}
|
|
471
|
-
|
|
472
|
-
async getFirstTimestamp() {
|
|
473
|
-
return 0;
|
|
474
|
-
}
|
|
475
|
-
|
|
476
|
-
async computeDuration() {
|
|
477
|
-
const lastPacket = await this.getPacket(Infinity, { metadataOnly: true });
|
|
478
|
-
return (lastPacket?.timestamp ?? 0) + (lastPacket?.duration ?? 0);
|
|
479
|
-
}
|
|
480
|
-
|
|
481
|
-
granulePositionToTimestampInSamples(granulePosition: number) {
|
|
482
|
-
if (this.bitstream.codecInfo.codec === 'opus') {
|
|
483
|
-
assert(this.bitstream.codecInfo.opusInfo);
|
|
484
|
-
return granulePosition - this.bitstream.codecInfo.opusInfo.preSkip;
|
|
485
|
-
}
|
|
486
|
-
|
|
487
|
-
return granulePosition;
|
|
488
|
-
}
|
|
489
|
-
|
|
490
|
-
createEncodedPacketFromOggPacket(
|
|
491
|
-
packet: Packet | null,
|
|
492
|
-
additional: {
|
|
493
|
-
timestampInSamples: number;
|
|
494
|
-
vorbisLastBlocksize: number | null;
|
|
495
|
-
},
|
|
496
|
-
options: PacketRetrievalOptions,
|
|
497
|
-
) {
|
|
498
|
-
if (!packet) {
|
|
499
|
-
return null;
|
|
500
|
-
}
|
|
501
|
-
|
|
502
|
-
const { durationInSamples, vorbisBlockSize } = extractSampleMetadata(
|
|
503
|
-
packet.data,
|
|
504
|
-
this.bitstream.codecInfo,
|
|
505
|
-
additional.vorbisLastBlocksize,
|
|
506
|
-
);
|
|
507
|
-
|
|
508
|
-
const encodedPacket = new EncodedPacket(
|
|
509
|
-
options.metadataOnly ? PLACEHOLDER_DATA : packet.data,
|
|
510
|
-
'key',
|
|
511
|
-
Math.max(0, additional.timestampInSamples) / this.internalSampleRate,
|
|
512
|
-
durationInSamples / this.internalSampleRate,
|
|
513
|
-
packet.endPage.headerStartPos + packet.endSegmentIndex,
|
|
514
|
-
packet.data.byteLength,
|
|
515
|
-
);
|
|
516
|
-
|
|
517
|
-
this.encodedPacketToMetadata.set(encodedPacket, {
|
|
518
|
-
packet,
|
|
519
|
-
timestampInSamples: additional.timestampInSamples,
|
|
520
|
-
durationInSamples,
|
|
521
|
-
vorbisLastBlockSize: additional.vorbisLastBlocksize,
|
|
522
|
-
vorbisBlockSize,
|
|
523
|
-
});
|
|
524
|
-
return encodedPacket;
|
|
525
|
-
}
|
|
526
|
-
|
|
527
|
-
async getFirstPacket(options: PacketRetrievalOptions) {
|
|
528
|
-
assert(this.bitstream.lastMetadataPacket);
|
|
529
|
-
const packetPosition = await this.demuxer.findNextPacketStart(this.bitstream.lastMetadataPacket);
|
|
530
|
-
if (!packetPosition) {
|
|
531
|
-
return null;
|
|
532
|
-
}
|
|
533
|
-
|
|
534
|
-
let timestampInSamples = 0;
|
|
535
|
-
if (this.bitstream.codecInfo.codec === 'opus') {
|
|
536
|
-
assert(this.bitstream.codecInfo.opusInfo);
|
|
537
|
-
timestampInSamples -= this.bitstream.codecInfo.opusInfo.preSkip;
|
|
538
|
-
}
|
|
539
|
-
|
|
540
|
-
const packet = await this.demuxer.readPacket(packetPosition.startPage, packetPosition.startSegmentIndex);
|
|
541
|
-
|
|
542
|
-
return this.createEncodedPacketFromOggPacket(
|
|
543
|
-
packet,
|
|
544
|
-
{
|
|
545
|
-
timestampInSamples,
|
|
546
|
-
vorbisLastBlocksize: null,
|
|
547
|
-
},
|
|
548
|
-
options,
|
|
549
|
-
);
|
|
550
|
-
}
|
|
551
|
-
|
|
552
|
-
async getNextPacket(prevPacket: EncodedPacket, options: PacketRetrievalOptions) {
|
|
553
|
-
const prevMetadata = this.encodedPacketToMetadata.get(prevPacket);
|
|
554
|
-
if (!prevMetadata) {
|
|
555
|
-
throw new Error('Packet was not created from this track.');
|
|
556
|
-
}
|
|
557
|
-
|
|
558
|
-
const packetPosition = await this.demuxer.findNextPacketStart(prevMetadata.packet);
|
|
559
|
-
if (!packetPosition) {
|
|
560
|
-
return null;
|
|
561
|
-
}
|
|
562
|
-
|
|
563
|
-
const timestampInSamples = prevMetadata.timestampInSamples + prevMetadata.durationInSamples;
|
|
564
|
-
|
|
565
|
-
const packet = await this.demuxer.readPacket(
|
|
566
|
-
packetPosition.startPage,
|
|
567
|
-
packetPosition.startSegmentIndex,
|
|
568
|
-
);
|
|
569
|
-
|
|
570
|
-
return this.createEncodedPacketFromOggPacket(
|
|
571
|
-
packet,
|
|
572
|
-
{
|
|
573
|
-
timestampInSamples,
|
|
574
|
-
vorbisLastBlocksize: prevMetadata.vorbisBlockSize,
|
|
575
|
-
},
|
|
576
|
-
options,
|
|
577
|
-
);
|
|
578
|
-
}
|
|
579
|
-
|
|
580
|
-
async getPacket(timestamp: number, options: PacketRetrievalOptions) {
|
|
581
|
-
if (this.demuxer.reader.fileSize === null) {
|
|
582
|
-
// No file size known, can't do binary search, but fall back to sequential algo instead
|
|
583
|
-
return this.getPacketSequential(timestamp, options);
|
|
584
|
-
}
|
|
585
|
-
|
|
586
|
-
const timestampInSamples = roundIfAlmostInteger(timestamp * this.internalSampleRate);
|
|
587
|
-
if (timestampInSamples === 0) {
|
|
588
|
-
// Fast path for timestamp 0 - avoids binary search when playing back from the start
|
|
589
|
-
return this.getFirstPacket(options);
|
|
590
|
-
}
|
|
591
|
-
if (timestampInSamples < 0) {
|
|
592
|
-
// There's nothing here
|
|
593
|
-
return null;
|
|
594
|
-
}
|
|
595
|
-
|
|
596
|
-
assert(this.bitstream.lastMetadataPacket);
|
|
597
|
-
const startPosition = await this.demuxer.findNextPacketStart(this.bitstream.lastMetadataPacket);
|
|
598
|
-
if (!startPosition) {
|
|
599
|
-
return null;
|
|
600
|
-
}
|
|
601
|
-
|
|
602
|
-
let lowPage = startPosition.startPage;
|
|
603
|
-
let high = this.demuxer.reader.fileSize;
|
|
604
|
-
|
|
605
|
-
const lowPages: Page[] = [lowPage];
|
|
606
|
-
|
|
607
|
-
// First, let's perform a binary serach (bisection search) on the file to find the approximate page where
|
|
608
|
-
// we'll find the packet. We want to find a page whose end packet position is less than or equal to the
|
|
609
|
-
// packet position we're searching for.
|
|
610
|
-
|
|
611
|
-
// Outer loop: Does the binary serach
|
|
612
|
-
outer:
|
|
613
|
-
while (lowPage.headerStartPos + lowPage.totalSize < high) {
|
|
614
|
-
const low = lowPage.headerStartPos;
|
|
615
|
-
const mid = Math.floor((low + high) / 2);
|
|
616
|
-
|
|
617
|
-
let searchStartPos = mid;
|
|
618
|
-
|
|
619
|
-
// Inner loop: Does a linear forward scan if the page cannot be found immediately
|
|
620
|
-
while (true) {
|
|
621
|
-
const until = Math.min(
|
|
622
|
-
searchStartPos + MAX_PAGE_SIZE,
|
|
623
|
-
high - MIN_PAGE_HEADER_SIZE,
|
|
624
|
-
);
|
|
625
|
-
|
|
626
|
-
let searchSlice = this.demuxer.reader.requestSlice(searchStartPos, until - searchStartPos);
|
|
627
|
-
if (searchSlice instanceof Promise) searchSlice = await searchSlice;
|
|
628
|
-
assert(searchSlice);
|
|
629
|
-
|
|
630
|
-
const found = findNextPageHeader(searchSlice, until);
|
|
631
|
-
if (!found) {
|
|
632
|
-
high = mid + MIN_PAGE_HEADER_SIZE;
|
|
633
|
-
continue outer;
|
|
634
|
-
}
|
|
635
|
-
|
|
636
|
-
let headerSlice = this.demuxer.reader.requestSliceRange(
|
|
637
|
-
searchSlice.filePos,
|
|
638
|
-
MIN_PAGE_HEADER_SIZE,
|
|
639
|
-
MAX_PAGE_HEADER_SIZE,
|
|
640
|
-
);
|
|
641
|
-
if (headerSlice instanceof Promise) headerSlice = await headerSlice;
|
|
642
|
-
assert(headerSlice);
|
|
643
|
-
|
|
644
|
-
const page = readPageHeader(headerSlice);
|
|
645
|
-
assert(page);
|
|
646
|
-
|
|
647
|
-
let pageValid = false;
|
|
648
|
-
if (page.serialNumber === this.bitstream.serialNumber) {
|
|
649
|
-
// Serial numbers are basically random numbers, and the chance of finding a fake page with
|
|
650
|
-
// matching serial number is astronomically low, so we can be pretty sure this page is legit.
|
|
651
|
-
pageValid = true;
|
|
652
|
-
} else {
|
|
653
|
-
let pageSlice = this.demuxer.reader.requestSlice(page.headerStartPos, page.totalSize);
|
|
654
|
-
if (pageSlice instanceof Promise) pageSlice = await pageSlice;
|
|
655
|
-
assert(pageSlice);
|
|
656
|
-
|
|
657
|
-
// Validate the page by checking checksum
|
|
658
|
-
const bytes = readBytes(pageSlice, page.totalSize);
|
|
659
|
-
const crc = computeOggPageCrc(bytes);
|
|
660
|
-
|
|
661
|
-
pageValid = crc === page.checksum;
|
|
662
|
-
}
|
|
663
|
-
|
|
664
|
-
if (!pageValid) {
|
|
665
|
-
// Keep searching for a valid page
|
|
666
|
-
searchStartPos = page.headerStartPos + 4; // 'OggS' is 4 bytes
|
|
667
|
-
continue;
|
|
668
|
-
}
|
|
669
|
-
|
|
670
|
-
if (pageValid && page.serialNumber !== this.bitstream.serialNumber) {
|
|
671
|
-
// Page is valid but from a different bitstream, so keep searching forward until we find one
|
|
672
|
-
// belonging to the our bitstream
|
|
673
|
-
searchStartPos = page.headerStartPos + page.totalSize;
|
|
674
|
-
continue;
|
|
675
|
-
}
|
|
676
|
-
|
|
677
|
-
const isContinuationPage = page.granulePosition === -1;
|
|
678
|
-
if (isContinuationPage) {
|
|
679
|
-
// No packet ends on this page - keep looking
|
|
680
|
-
searchStartPos = page.headerStartPos + page.totalSize;
|
|
681
|
-
continue;
|
|
682
|
-
}
|
|
683
|
-
|
|
684
|
-
// The page is valid and belongs to our bitstream; let's check its granule position to see where we
|
|
685
|
-
// need to take the bisection search.
|
|
686
|
-
if (this.granulePositionToTimestampInSamples(page.granulePosition) > timestampInSamples) {
|
|
687
|
-
high = page.headerStartPos;
|
|
688
|
-
} else {
|
|
689
|
-
lowPage = page;
|
|
690
|
-
lowPages.push(page);
|
|
691
|
-
}
|
|
692
|
-
|
|
693
|
-
continue outer;
|
|
694
|
-
}
|
|
695
|
-
}
|
|
696
|
-
|
|
697
|
-
// Now we have the last page with a packet position <= the packet position we're looking for, but there
|
|
698
|
-
// might be multiple pages with the packet position, in which case we actually need to find the first of
|
|
699
|
-
// such pages. We'll do this in two steps: First, let's find the latest page we know with an earlier packet
|
|
700
|
-
// position, and then linear scan ourselves forward until we find the correct page.
|
|
701
|
-
|
|
702
|
-
let lowerPage = startPosition.startPage;
|
|
703
|
-
for (const otherLowPage of lowPages) {
|
|
704
|
-
if (otherLowPage.granulePosition === lowPage.granulePosition) {
|
|
705
|
-
break;
|
|
706
|
-
}
|
|
707
|
-
|
|
708
|
-
if (!lowerPage || otherLowPage.headerStartPos > lowerPage.headerStartPos) {
|
|
709
|
-
lowerPage = otherLowPage;
|
|
710
|
-
}
|
|
711
|
-
}
|
|
712
|
-
|
|
713
|
-
let currentPage = lowerPage;
|
|
714
|
-
// Keep track of the pages we traversed, we need these later for backwards seeking
|
|
715
|
-
const previousPages: Page[] = [currentPage];
|
|
716
|
-
|
|
717
|
-
while (true) {
|
|
718
|
-
// This loop must terminate as we'll eventually reach lowPage
|
|
719
|
-
if (
|
|
720
|
-
currentPage.serialNumber === this.bitstream.serialNumber
|
|
721
|
-
&& currentPage.granulePosition === lowPage.granulePosition
|
|
722
|
-
) {
|
|
723
|
-
break;
|
|
724
|
-
}
|
|
725
|
-
|
|
726
|
-
const nextPos = currentPage.headerStartPos + currentPage.totalSize;
|
|
727
|
-
let slice = this.demuxer.reader.requestSliceRange(nextPos, MIN_PAGE_HEADER_SIZE, MAX_PAGE_HEADER_SIZE);
|
|
728
|
-
if (slice instanceof Promise) slice = await slice;
|
|
729
|
-
assert(slice);
|
|
730
|
-
|
|
731
|
-
const nextPage = readPageHeader(slice);
|
|
732
|
-
assert(nextPage);
|
|
733
|
-
|
|
734
|
-
currentPage = nextPage;
|
|
735
|
-
|
|
736
|
-
if (currentPage.serialNumber === this.bitstream.serialNumber) {
|
|
737
|
-
previousPages.push(currentPage);
|
|
738
|
-
}
|
|
739
|
-
}
|
|
740
|
-
|
|
741
|
-
assert(currentPage.granulePosition !== -1);
|
|
742
|
-
|
|
743
|
-
let currentSegmentIndex: number | null = null;
|
|
744
|
-
let currentTimestampInSamples: number;
|
|
745
|
-
let currentTimestampIsCorrect: boolean;
|
|
746
|
-
|
|
747
|
-
// These indicate the end position of the packet that the granule position belongs to
|
|
748
|
-
let endPage = currentPage;
|
|
749
|
-
let endSegmentIndex = 0;
|
|
750
|
-
|
|
751
|
-
if (currentPage.headerStartPos === startPosition.startPage.headerStartPos) {
|
|
752
|
-
currentTimestampInSamples = this.granulePositionToTimestampInSamples(0);
|
|
753
|
-
currentTimestampIsCorrect = true;
|
|
754
|
-
currentSegmentIndex = 0;
|
|
755
|
-
} else {
|
|
756
|
-
currentTimestampInSamples = 0; // Placeholder value! We'll refine it once we can
|
|
757
|
-
currentTimestampIsCorrect = false;
|
|
758
|
-
|
|
759
|
-
// Find the segment index of the next packet
|
|
760
|
-
for (let i = currentPage.lacingValues.length - 1; i >= 0; i--) {
|
|
761
|
-
const value = currentPage.lacingValues[i]!;
|
|
762
|
-
if (value < 255) {
|
|
763
|
-
// We know the last packet ended at i, so the next one starts at i + 1
|
|
764
|
-
currentSegmentIndex = i + 1;
|
|
765
|
-
break;
|
|
766
|
-
}
|
|
767
|
-
}
|
|
768
|
-
|
|
769
|
-
// This must hold: Since this page has a granule position set, that means there must be a packet that
|
|
770
|
-
// ends in this page.
|
|
771
|
-
if (currentSegmentIndex === null) {
|
|
772
|
-
throw new Error('Invalid page with granule position: no packets end on this page.');
|
|
773
|
-
}
|
|
774
|
-
|
|
775
|
-
endSegmentIndex = currentSegmentIndex - 1;
|
|
776
|
-
const pseudopacket: Packet = {
|
|
777
|
-
data: PLACEHOLDER_DATA,
|
|
778
|
-
endPage,
|
|
779
|
-
endSegmentIndex,
|
|
780
|
-
};
|
|
781
|
-
const nextPosition = await this.demuxer.findNextPacketStart(pseudopacket);
|
|
782
|
-
|
|
783
|
-
if (nextPosition) {
|
|
784
|
-
// Let's rewind a single step (packet) - this previous packet ensures that we'll correctly compute
|
|
785
|
-
// the duration for the packet we're looking for.
|
|
786
|
-
const endPosition = findPreviousPacketEndPosition(previousPages, currentPage, currentSegmentIndex);
|
|
787
|
-
assert(endPosition);
|
|
788
|
-
|
|
789
|
-
const startPosition = findPacketStartPosition(
|
|
790
|
-
previousPages, endPosition.page, endPosition.segmentIndex,
|
|
791
|
-
);
|
|
792
|
-
if (startPosition) {
|
|
793
|
-
currentPage = startPosition.page;
|
|
794
|
-
currentSegmentIndex = startPosition.segmentIndex;
|
|
795
|
-
}
|
|
796
|
-
} else {
|
|
797
|
-
// There is no next position, which means we're looking for the last packet in the bitstream. The
|
|
798
|
-
// granule position on the last page tends to be fucky, so let's instead start the search on the
|
|
799
|
-
// page before that. So let's loop until we find a packet that ends in a previous page.
|
|
800
|
-
while (true) {
|
|
801
|
-
const endPosition = findPreviousPacketEndPosition(
|
|
802
|
-
previousPages, currentPage, currentSegmentIndex,
|
|
803
|
-
);
|
|
804
|
-
if (!endPosition) {
|
|
805
|
-
break;
|
|
806
|
-
}
|
|
807
|
-
|
|
808
|
-
const startPosition = findPacketStartPosition(
|
|
809
|
-
previousPages, endPosition.page, endPosition.segmentIndex,
|
|
810
|
-
);
|
|
811
|
-
if (!startPosition) {
|
|
812
|
-
break;
|
|
813
|
-
}
|
|
814
|
-
|
|
815
|
-
currentPage = startPosition.page;
|
|
816
|
-
currentSegmentIndex = startPosition.segmentIndex;
|
|
817
|
-
|
|
818
|
-
if (endPosition.page.headerStartPos !== endPage.headerStartPos) {
|
|
819
|
-
endPage = endPosition.page;
|
|
820
|
-
endSegmentIndex = endPosition.segmentIndex;
|
|
821
|
-
break;
|
|
822
|
-
}
|
|
823
|
-
}
|
|
824
|
-
}
|
|
825
|
-
}
|
|
826
|
-
|
|
827
|
-
let lastEncodedPacket: EncodedPacket | null = null;
|
|
828
|
-
let lastEncodedPacketMetadata: EncodedPacketMetadata | null = null;
|
|
829
|
-
|
|
830
|
-
// Alright, now it's time for the final, granular seek: We keep iterating over packets until we've found the
|
|
831
|
-
// one with the correct timestamp - i.e., the last one with a timestamp <= the timestamp we're looking for.
|
|
832
|
-
while (currentPage !== null) {
|
|
833
|
-
assert(currentSegmentIndex !== null);
|
|
834
|
-
|
|
835
|
-
const packet = await this.demuxer.readPacket(currentPage, currentSegmentIndex);
|
|
836
|
-
if (!packet) {
|
|
837
|
-
break;
|
|
838
|
-
}
|
|
839
|
-
|
|
840
|
-
// We might need to skip the packet if it's a metadata one
|
|
841
|
-
const skipPacket = currentPage.headerStartPos === startPosition.startPage.headerStartPos
|
|
842
|
-
&& currentSegmentIndex < startPosition.startSegmentIndex;
|
|
843
|
-
|
|
844
|
-
if (!skipPacket) {
|
|
845
|
-
let encodedPacket = this.createEncodedPacketFromOggPacket(
|
|
846
|
-
packet,
|
|
847
|
-
{
|
|
848
|
-
timestampInSamples: currentTimestampInSamples,
|
|
849
|
-
vorbisLastBlocksize: lastEncodedPacketMetadata?.vorbisBlockSize ?? null,
|
|
850
|
-
},
|
|
851
|
-
options,
|
|
852
|
-
);
|
|
853
|
-
assert(encodedPacket);
|
|
854
|
-
|
|
855
|
-
let encodedPacketMetadata = this.encodedPacketToMetadata.get(encodedPacket);
|
|
856
|
-
assert(encodedPacketMetadata);
|
|
857
|
-
|
|
858
|
-
if (
|
|
859
|
-
!currentTimestampIsCorrect
|
|
860
|
-
&& packet.endPage.headerStartPos === endPage.headerStartPos
|
|
861
|
-
&& packet.endSegmentIndex === endSegmentIndex
|
|
862
|
-
) {
|
|
863
|
-
// We know this packet end timestamp can be derived from the page's granule position
|
|
864
|
-
currentTimestampInSamples = this.granulePositionToTimestampInSamples(
|
|
865
|
-
currentPage.granulePosition,
|
|
866
|
-
);
|
|
867
|
-
currentTimestampIsCorrect = true;
|
|
868
|
-
|
|
869
|
-
// Let's backpatch the packet we just created with the correct timestamp
|
|
870
|
-
encodedPacket = this.createEncodedPacketFromOggPacket(
|
|
871
|
-
packet,
|
|
872
|
-
{
|
|
873
|
-
timestampInSamples: currentTimestampInSamples - encodedPacketMetadata.durationInSamples,
|
|
874
|
-
vorbisLastBlocksize: lastEncodedPacketMetadata?.vorbisBlockSize ?? null,
|
|
875
|
-
},
|
|
876
|
-
options,
|
|
877
|
-
);
|
|
878
|
-
assert(encodedPacket);
|
|
879
|
-
|
|
880
|
-
encodedPacketMetadata = this.encodedPacketToMetadata.get(encodedPacket);
|
|
881
|
-
assert(encodedPacketMetadata);
|
|
882
|
-
} else {
|
|
883
|
-
currentTimestampInSamples += encodedPacketMetadata.durationInSamples;
|
|
884
|
-
}
|
|
885
|
-
|
|
886
|
-
lastEncodedPacket = encodedPacket;
|
|
887
|
-
lastEncodedPacketMetadata = encodedPacketMetadata;
|
|
888
|
-
|
|
889
|
-
if (
|
|
890
|
-
currentTimestampIsCorrect
|
|
891
|
-
&& (
|
|
892
|
-
// Next timestamp will be too late
|
|
893
|
-
Math.max(currentTimestampInSamples, 0) > timestampInSamples
|
|
894
|
-
// This timestamp already matches
|
|
895
|
-
|| Math.max(encodedPacketMetadata.timestampInSamples, 0) === timestampInSamples
|
|
896
|
-
)
|
|
897
|
-
) {
|
|
898
|
-
break;
|
|
899
|
-
}
|
|
900
|
-
}
|
|
901
|
-
|
|
902
|
-
const nextPosition = await this.demuxer.findNextPacketStart(packet);
|
|
903
|
-
if (!nextPosition) {
|
|
904
|
-
break;
|
|
905
|
-
}
|
|
906
|
-
|
|
907
|
-
currentPage = nextPosition.startPage;
|
|
908
|
-
currentSegmentIndex = nextPosition.startSegmentIndex;
|
|
909
|
-
}
|
|
910
|
-
|
|
911
|
-
return lastEncodedPacket;
|
|
912
|
-
}
|
|
913
|
-
|
|
914
|
-
// A slower but simpler and sequential algorithm for finding a packet in a file
|
|
915
|
-
async getPacketSequential(timestamp: number, options: PacketRetrievalOptions) {
|
|
916
|
-
const release = await this.sequentialScanMutex.acquire(); // Requires exclusivity because we write to a cache
|
|
917
|
-
|
|
918
|
-
try {
|
|
919
|
-
const timestampInSamples = roundIfAlmostInteger(timestamp * this.internalSampleRate);
|
|
920
|
-
timestamp = timestampInSamples / this.internalSampleRate;
|
|
921
|
-
|
|
922
|
-
const index = binarySearchLessOrEqual(
|
|
923
|
-
this.sequentialScanCache,
|
|
924
|
-
timestampInSamples,
|
|
925
|
-
x => x.timestampInSamples,
|
|
926
|
-
);
|
|
927
|
-
|
|
928
|
-
let currentPacket: EncodedPacket | null;
|
|
929
|
-
if (index !== -1) {
|
|
930
|
-
// We don't need to start from the beginning, we can start at a previous scan point
|
|
931
|
-
const cacheEntry = this.sequentialScanCache[index]!;
|
|
932
|
-
currentPacket = this.createEncodedPacketFromOggPacket(
|
|
933
|
-
cacheEntry.packet,
|
|
934
|
-
{
|
|
935
|
-
timestampInSamples: cacheEntry.timestampInSamples,
|
|
936
|
-
vorbisLastBlocksize: cacheEntry.vorbisLastBlockSize,
|
|
937
|
-
},
|
|
938
|
-
options,
|
|
939
|
-
);
|
|
940
|
-
} else {
|
|
941
|
-
currentPacket = await this.getFirstPacket(options);
|
|
942
|
-
}
|
|
943
|
-
|
|
944
|
-
let i = 0;
|
|
945
|
-
|
|
946
|
-
while (currentPacket && currentPacket.timestamp < timestamp) {
|
|
947
|
-
const nextPacket = await this.getNextPacket(currentPacket, options);
|
|
948
|
-
if (!nextPacket || nextPacket.timestamp > timestamp) {
|
|
949
|
-
break;
|
|
950
|
-
}
|
|
951
|
-
|
|
952
|
-
currentPacket = nextPacket;
|
|
953
|
-
i++;
|
|
954
|
-
|
|
955
|
-
if (i === 100) {
|
|
956
|
-
// Add "checkpoints" every once in a while to speed up subsequent random accesses
|
|
957
|
-
i = 0;
|
|
958
|
-
const metadata = this.encodedPacketToMetadata.get(currentPacket);
|
|
959
|
-
assert(metadata);
|
|
960
|
-
|
|
961
|
-
if (this.sequentialScanCache.length > 0) {
|
|
962
|
-
// If we reach this case, we must be at the end of the cache
|
|
963
|
-
assert(last(this.sequentialScanCache)!.timestampInSamples <= metadata.timestampInSamples);
|
|
964
|
-
}
|
|
965
|
-
|
|
966
|
-
this.sequentialScanCache.push(metadata);
|
|
967
|
-
}
|
|
968
|
-
}
|
|
969
|
-
|
|
970
|
-
return currentPacket;
|
|
971
|
-
} finally {
|
|
972
|
-
release();
|
|
973
|
-
}
|
|
974
|
-
}
|
|
975
|
-
|
|
976
|
-
getKeyPacket(timestamp: number, options: PacketRetrievalOptions) {
|
|
977
|
-
return this.getPacket(timestamp, options);
|
|
978
|
-
}
|
|
979
|
-
|
|
980
|
-
getNextKeyPacket(packet: EncodedPacket, options: PacketRetrievalOptions) {
|
|
981
|
-
return this.getNextPacket(packet, options);
|
|
982
|
-
}
|
|
983
|
-
}
|
|
984
|
-
|
|
985
|
-
/** Finds the start position of a packet given its end position. */
|
|
986
|
-
const findPacketStartPosition = (pageList: Page[], endPage: Page, endSegmentIndex: number) => {
|
|
987
|
-
let page = endPage;
|
|
988
|
-
let segmentIndex = endSegmentIndex;
|
|
989
|
-
|
|
990
|
-
outer:
|
|
991
|
-
while (true) {
|
|
992
|
-
segmentIndex--;
|
|
993
|
-
|
|
994
|
-
for (segmentIndex; segmentIndex >= 0; segmentIndex--) {
|
|
995
|
-
const lacingValue = page.lacingValues[segmentIndex]!;
|
|
996
|
-
if (lacingValue < 255) {
|
|
997
|
-
segmentIndex++; // We know the last packet starts here
|
|
998
|
-
break outer;
|
|
999
|
-
}
|
|
1000
|
-
}
|
|
1001
|
-
|
|
1002
|
-
assert(segmentIndex === -1);
|
|
1003
|
-
|
|
1004
|
-
const pageStartsWithFreshPacket = !(page.headerType & 0x01);
|
|
1005
|
-
if (pageStartsWithFreshPacket) {
|
|
1006
|
-
// Fast exit: We know we don't need to look in the previous page
|
|
1007
|
-
segmentIndex = 0;
|
|
1008
|
-
break;
|
|
1009
|
-
}
|
|
1010
|
-
|
|
1011
|
-
const previousPage = findLast(
|
|
1012
|
-
pageList,
|
|
1013
|
-
x => x.headerStartPos < page.headerStartPos,
|
|
1014
|
-
);
|
|
1015
|
-
if (!previousPage) {
|
|
1016
|
-
return null;
|
|
1017
|
-
}
|
|
1018
|
-
|
|
1019
|
-
page = previousPage;
|
|
1020
|
-
segmentIndex = page.lacingValues.length;
|
|
1021
|
-
}
|
|
1022
|
-
|
|
1023
|
-
assert(segmentIndex !== -1);
|
|
1024
|
-
|
|
1025
|
-
if (segmentIndex === page.lacingValues.length) {
|
|
1026
|
-
// Wrap back around to the first segment of the next page
|
|
1027
|
-
const nextPage = pageList[pageList.indexOf(page) + 1];
|
|
1028
|
-
assert(nextPage);
|
|
1029
|
-
|
|
1030
|
-
page = nextPage;
|
|
1031
|
-
segmentIndex = 0;
|
|
1032
|
-
}
|
|
1033
|
-
|
|
1034
|
-
return { page, segmentIndex };
|
|
1035
|
-
};
|
|
1036
|
-
|
|
1037
|
-
/** Finds the end position of a packet given the start position of the following packet. */
|
|
1038
|
-
const findPreviousPacketEndPosition = (pageList: Page[], startPage: Page, startSegmentIndex: number) => {
|
|
1039
|
-
if (startSegmentIndex > 0) {
|
|
1040
|
-
// Easy
|
|
1041
|
-
return { page: startPage, segmentIndex: startSegmentIndex - 1 };
|
|
1042
|
-
}
|
|
1043
|
-
|
|
1044
|
-
const previousPage = findLast(
|
|
1045
|
-
pageList,
|
|
1046
|
-
x => x.headerStartPos < startPage.headerStartPos,
|
|
1047
|
-
);
|
|
1048
|
-
if (!previousPage) {
|
|
1049
|
-
return null;
|
|
1050
|
-
}
|
|
1051
|
-
|
|
1052
|
-
return { page: previousPage, segmentIndex: previousPage.lacingValues.length - 1 };
|
|
1053
|
-
};
|