@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.
Files changed (238) hide show
  1. package/README.md +1 -1
  2. package/dist/bundles/{mediabunny.mjs → mediabunny.js} +21963 -21390
  3. package/dist/bundles/mediabunny.min.js +490 -0
  4. package/dist/modules/shared/mp3-misc.d.ts.map +1 -1
  5. package/dist/modules/src/adts/adts-demuxer.d.ts +6 -6
  6. package/dist/modules/src/adts/adts-demuxer.d.ts.map +1 -1
  7. package/dist/modules/src/adts/adts-muxer.d.ts +4 -4
  8. package/dist/modules/src/adts/adts-muxer.d.ts.map +1 -1
  9. package/dist/modules/src/adts/adts-reader.d.ts +1 -1
  10. package/dist/modules/src/adts/adts-reader.d.ts.map +1 -1
  11. package/dist/modules/src/avi/avi-demuxer.d.ts +44 -0
  12. package/dist/modules/src/avi/avi-demuxer.d.ts.map +1 -0
  13. package/dist/modules/src/avi/avi-misc.d.ts +88 -0
  14. package/dist/modules/src/avi/avi-misc.d.ts.map +1 -0
  15. package/dist/modules/src/avi/avi-muxer.d.ts +45 -0
  16. package/dist/modules/src/avi/avi-muxer.d.ts.map +1 -0
  17. package/dist/modules/src/avi/riff-writer.d.ts +26 -0
  18. package/dist/modules/src/avi/riff-writer.d.ts.map +1 -0
  19. package/dist/modules/src/codec-data.d.ts +8 -3
  20. package/dist/modules/src/codec-data.d.ts.map +1 -1
  21. package/dist/modules/src/codec.d.ts +10 -10
  22. package/dist/modules/src/codec.d.ts.map +1 -1
  23. package/dist/modules/src/conversion.d.ts +33 -16
  24. package/dist/modules/src/conversion.d.ts.map +1 -1
  25. package/dist/modules/src/custom-coder.d.ts +8 -8
  26. package/dist/modules/src/custom-coder.d.ts.map +1 -1
  27. package/dist/modules/src/demuxer.d.ts +3 -3
  28. package/dist/modules/src/demuxer.d.ts.map +1 -1
  29. package/dist/modules/src/encode.d.ts +8 -8
  30. package/dist/modules/src/encode.d.ts.map +1 -1
  31. package/dist/modules/src/flac/flac-demuxer.d.ts +7 -7
  32. package/dist/modules/src/flac/flac-demuxer.d.ts.map +1 -1
  33. package/dist/modules/src/flac/flac-misc.d.ts +3 -3
  34. package/dist/modules/src/flac/flac-misc.d.ts.map +1 -1
  35. package/dist/modules/src/flac/flac-muxer.d.ts +5 -5
  36. package/dist/modules/src/flac/flac-muxer.d.ts.map +1 -1
  37. package/dist/modules/src/id3.d.ts +3 -3
  38. package/dist/modules/src/id3.d.ts.map +1 -1
  39. package/dist/modules/src/index.d.ts +20 -20
  40. package/dist/modules/src/index.d.ts.map +1 -1
  41. package/dist/modules/src/input-format.d.ts +22 -0
  42. package/dist/modules/src/input-format.d.ts.map +1 -1
  43. package/dist/modules/src/input-track.d.ts +8 -8
  44. package/dist/modules/src/input-track.d.ts.map +1 -1
  45. package/dist/modules/src/input.d.ts +12 -12
  46. package/dist/modules/src/isobmff/isobmff-boxes.d.ts +2 -2
  47. package/dist/modules/src/isobmff/isobmff-boxes.d.ts.map +1 -1
  48. package/dist/modules/src/isobmff/isobmff-demuxer.d.ts +12 -12
  49. package/dist/modules/src/isobmff/isobmff-demuxer.d.ts.map +1 -1
  50. package/dist/modules/src/isobmff/isobmff-misc.d.ts.map +1 -1
  51. package/dist/modules/src/isobmff/isobmff-muxer.d.ts +11 -11
  52. package/dist/modules/src/isobmff/isobmff-muxer.d.ts.map +1 -1
  53. package/dist/modules/src/isobmff/isobmff-reader.d.ts +2 -2
  54. package/dist/modules/src/isobmff/isobmff-reader.d.ts.map +1 -1
  55. package/dist/modules/src/matroska/ebml.d.ts +3 -3
  56. package/dist/modules/src/matroska/ebml.d.ts.map +1 -1
  57. package/dist/modules/src/matroska/matroska-demuxer.d.ts +13 -13
  58. package/dist/modules/src/matroska/matroska-demuxer.d.ts.map +1 -1
  59. package/dist/modules/src/matroska/matroska-input.d.ts +33 -0
  60. package/dist/modules/src/matroska/matroska-input.d.ts.map +1 -0
  61. package/dist/modules/src/matroska/matroska-misc.d.ts.map +1 -1
  62. package/dist/modules/src/matroska/matroska-muxer.d.ts +5 -5
  63. package/dist/modules/src/matroska/matroska-muxer.d.ts.map +1 -1
  64. package/dist/modules/src/media-sink.d.ts +5 -5
  65. package/dist/modules/src/media-sink.d.ts.map +1 -1
  66. package/dist/modules/src/media-source.d.ts +22 -4
  67. package/dist/modules/src/media-source.d.ts.map +1 -1
  68. package/dist/modules/src/metadata.d.ts +2 -2
  69. package/dist/modules/src/metadata.d.ts.map +1 -1
  70. package/dist/modules/src/misc.d.ts +5 -4
  71. package/dist/modules/src/misc.d.ts.map +1 -1
  72. package/dist/modules/src/mp3/mp3-demuxer.d.ts +7 -7
  73. package/dist/modules/src/mp3/mp3-demuxer.d.ts.map +1 -1
  74. package/dist/modules/src/mp3/mp3-muxer.d.ts +4 -4
  75. package/dist/modules/src/mp3/mp3-muxer.d.ts.map +1 -1
  76. package/dist/modules/src/mp3/mp3-reader.d.ts +2 -2
  77. package/dist/modules/src/mp3/mp3-reader.d.ts.map +1 -1
  78. package/dist/modules/src/mp3/mp3-writer.d.ts +1 -1
  79. package/dist/modules/src/mp3/mp3-writer.d.ts.map +1 -1
  80. package/dist/modules/src/muxer.d.ts +4 -4
  81. package/dist/modules/src/muxer.d.ts.map +1 -1
  82. package/dist/modules/src/ogg/ogg-demuxer.d.ts +7 -7
  83. package/dist/modules/src/ogg/ogg-demuxer.d.ts.map +1 -1
  84. package/dist/modules/src/ogg/ogg-misc.d.ts +1 -1
  85. package/dist/modules/src/ogg/ogg-misc.d.ts.map +1 -1
  86. package/dist/modules/src/ogg/ogg-muxer.d.ts +5 -5
  87. package/dist/modules/src/ogg/ogg-muxer.d.ts.map +1 -1
  88. package/dist/modules/src/ogg/ogg-reader.d.ts +1 -1
  89. package/dist/modules/src/ogg/ogg-reader.d.ts.map +1 -1
  90. package/dist/modules/src/output-format.d.ts +51 -6
  91. package/dist/modules/src/output-format.d.ts.map +1 -1
  92. package/dist/modules/src/output.d.ts +13 -13
  93. package/dist/modules/src/output.d.ts.map +1 -1
  94. package/dist/modules/src/packet.d.ts +1 -1
  95. package/dist/modules/src/packet.d.ts.map +1 -1
  96. package/dist/modules/src/pcm.d.ts.map +1 -1
  97. package/dist/modules/src/reader.d.ts +2 -2
  98. package/dist/modules/src/reader.d.ts.map +1 -1
  99. package/dist/modules/src/sample.d.ts +57 -15
  100. package/dist/modules/src/sample.d.ts.map +1 -1
  101. package/dist/modules/src/source.d.ts +3 -3
  102. package/dist/modules/src/source.d.ts.map +1 -1
  103. package/dist/modules/src/subtitles.d.ts +1 -1
  104. package/dist/modules/src/subtitles.d.ts.map +1 -1
  105. package/dist/modules/src/target.d.ts +2 -2
  106. package/dist/modules/src/target.d.ts.map +1 -1
  107. package/dist/modules/src/tsconfig.tsbuildinfo +1 -1
  108. package/dist/modules/src/wave/riff-writer.d.ts +1 -1
  109. package/dist/modules/src/wave/riff-writer.d.ts.map +1 -1
  110. package/dist/modules/src/wave/wave-demuxer.d.ts +6 -6
  111. package/dist/modules/src/wave/wave-demuxer.d.ts.map +1 -1
  112. package/dist/modules/src/wave/wave-muxer.d.ts +4 -4
  113. package/dist/modules/src/wave/wave-muxer.d.ts.map +1 -1
  114. package/dist/modules/src/writer.d.ts +1 -1
  115. package/dist/modules/src/writer.d.ts.map +1 -1
  116. package/dist/packages/eac3/eac3.wasm +0 -0
  117. package/dist/packages/eac3/mediabunny-eac3.js +1058 -0
  118. package/dist/packages/eac3/mediabunny-eac3.min.js +44 -0
  119. package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.js +694 -0
  120. package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.min.js +58 -0
  121. package/dist/packages/mpeg4/mediabunny-mpeg4.js +1198 -0
  122. package/dist/packages/mpeg4/mediabunny-mpeg4.min.js +44 -0
  123. package/dist/packages/mpeg4/xvid.wasm +0 -0
  124. package/package.json +18 -57
  125. package/dist/bundles/mediabunny.cjs +0 -26140
  126. package/dist/bundles/mediabunny.min.cjs +0 -147
  127. package/dist/bundles/mediabunny.min.mjs +0 -146
  128. package/dist/mediabunny.d.ts +0 -3319
  129. package/dist/modules/shared/mp3-misc.js +0 -147
  130. package/dist/modules/src/adts/adts-demuxer.js +0 -239
  131. package/dist/modules/src/adts/adts-muxer.js +0 -80
  132. package/dist/modules/src/adts/adts-reader.js +0 -63
  133. package/dist/modules/src/codec-data.js +0 -1730
  134. package/dist/modules/src/codec.js +0 -869
  135. package/dist/modules/src/conversion.js +0 -1459
  136. package/dist/modules/src/custom-coder.js +0 -117
  137. package/dist/modules/src/demuxer.js +0 -12
  138. package/dist/modules/src/encode.js +0 -442
  139. package/dist/modules/src/flac/flac-demuxer.js +0 -504
  140. package/dist/modules/src/flac/flac-misc.js +0 -135
  141. package/dist/modules/src/flac/flac-muxer.js +0 -222
  142. package/dist/modules/src/id3.js +0 -848
  143. package/dist/modules/src/index.js +0 -28
  144. package/dist/modules/src/input-format.js +0 -480
  145. package/dist/modules/src/input-track.js +0 -372
  146. package/dist/modules/src/input.js +0 -188
  147. package/dist/modules/src/isobmff/isobmff-boxes.js +0 -1480
  148. package/dist/modules/src/isobmff/isobmff-demuxer.js +0 -2618
  149. package/dist/modules/src/isobmff/isobmff-misc.js +0 -20
  150. package/dist/modules/src/isobmff/isobmff-muxer.js +0 -966
  151. package/dist/modules/src/isobmff/isobmff-reader.js +0 -72
  152. package/dist/modules/src/matroska/ebml.js +0 -653
  153. package/dist/modules/src/matroska/matroska-demuxer.js +0 -2133
  154. package/dist/modules/src/matroska/matroska-misc.js +0 -20
  155. package/dist/modules/src/matroska/matroska-muxer.js +0 -1017
  156. package/dist/modules/src/media-sink.js +0 -1736
  157. package/dist/modules/src/media-source.js +0 -1825
  158. package/dist/modules/src/metadata.js +0 -193
  159. package/dist/modules/src/misc.js +0 -623
  160. package/dist/modules/src/mp3/mp3-demuxer.js +0 -285
  161. package/dist/modules/src/mp3/mp3-muxer.js +0 -123
  162. package/dist/modules/src/mp3/mp3-reader.js +0 -26
  163. package/dist/modules/src/mp3/mp3-writer.js +0 -78
  164. package/dist/modules/src/muxer.js +0 -50
  165. package/dist/modules/src/node.d.ts +0 -9
  166. package/dist/modules/src/node.d.ts.map +0 -1
  167. package/dist/modules/src/node.js +0 -9
  168. package/dist/modules/src/ogg/ogg-demuxer.js +0 -763
  169. package/dist/modules/src/ogg/ogg-misc.js +0 -78
  170. package/dist/modules/src/ogg/ogg-muxer.js +0 -353
  171. package/dist/modules/src/ogg/ogg-reader.js +0 -65
  172. package/dist/modules/src/output-format.js +0 -527
  173. package/dist/modules/src/output.js +0 -300
  174. package/dist/modules/src/packet.js +0 -182
  175. package/dist/modules/src/pcm.js +0 -85
  176. package/dist/modules/src/reader.js +0 -236
  177. package/dist/modules/src/sample.js +0 -1056
  178. package/dist/modules/src/source.js +0 -1182
  179. package/dist/modules/src/subtitles.js +0 -575
  180. package/dist/modules/src/target.js +0 -140
  181. package/dist/modules/src/wave/riff-writer.js +0 -30
  182. package/dist/modules/src/wave/wave-demuxer.js +0 -447
  183. package/dist/modules/src/wave/wave-muxer.js +0 -318
  184. package/dist/modules/src/writer.js +0 -370
  185. package/src/adts/adts-demuxer.ts +0 -331
  186. package/src/adts/adts-muxer.ts +0 -111
  187. package/src/adts/adts-reader.ts +0 -85
  188. package/src/codec-data.ts +0 -2078
  189. package/src/codec.ts +0 -1092
  190. package/src/conversion.ts +0 -2112
  191. package/src/custom-coder.ts +0 -197
  192. package/src/demuxer.ts +0 -24
  193. package/src/encode.ts +0 -739
  194. package/src/flac/flac-demuxer.ts +0 -730
  195. package/src/flac/flac-misc.ts +0 -164
  196. package/src/flac/flac-muxer.ts +0 -320
  197. package/src/id3.ts +0 -925
  198. package/src/index.ts +0 -221
  199. package/src/input-format.ts +0 -541
  200. package/src/input-track.ts +0 -529
  201. package/src/input.ts +0 -235
  202. package/src/isobmff/isobmff-boxes.ts +0 -1719
  203. package/src/isobmff/isobmff-demuxer.ts +0 -3190
  204. package/src/isobmff/isobmff-misc.ts +0 -29
  205. package/src/isobmff/isobmff-muxer.ts +0 -1348
  206. package/src/isobmff/isobmff-reader.ts +0 -91
  207. package/src/matroska/ebml.ts +0 -730
  208. package/src/matroska/matroska-demuxer.ts +0 -2481
  209. package/src/matroska/matroska-misc.ts +0 -29
  210. package/src/matroska/matroska-muxer.ts +0 -1276
  211. package/src/media-sink.ts +0 -2179
  212. package/src/media-source.ts +0 -2243
  213. package/src/metadata.ts +0 -320
  214. package/src/misc.ts +0 -798
  215. package/src/mp3/mp3-demuxer.ts +0 -383
  216. package/src/mp3/mp3-muxer.ts +0 -166
  217. package/src/mp3/mp3-reader.ts +0 -34
  218. package/src/mp3/mp3-writer.ts +0 -120
  219. package/src/muxer.ts +0 -88
  220. package/src/node.ts +0 -11
  221. package/src/ogg/ogg-demuxer.ts +0 -1053
  222. package/src/ogg/ogg-misc.ts +0 -116
  223. package/src/ogg/ogg-muxer.ts +0 -497
  224. package/src/ogg/ogg-reader.ts +0 -93
  225. package/src/output-format.ts +0 -945
  226. package/src/output.ts +0 -488
  227. package/src/packet.ts +0 -263
  228. package/src/pcm.ts +0 -112
  229. package/src/reader.ts +0 -323
  230. package/src/sample.ts +0 -1461
  231. package/src/source.ts +0 -1688
  232. package/src/subtitles.ts +0 -711
  233. package/src/target.ts +0 -204
  234. package/src/tsconfig.json +0 -16
  235. package/src/wave/riff-writer.ts +0 -36
  236. package/src/wave/wave-demuxer.ts +0 -529
  237. package/src/wave/wave-muxer.ts +0 -371
  238. package/src/writer.ts +0 -490
@@ -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
- };