Home Reference Source

src/demux/base-audio-demuxer.ts

  1. import * as ID3 from '../demux/id3';
  2. import {
  3. DemuxerResult,
  4. Demuxer,
  5. DemuxedAudioTrack,
  6. AudioFrame,
  7. DemuxedMetadataTrack,
  8. DemuxedVideoTrack,
  9. DemuxedUserdataTrack,
  10. KeyData,
  11. MetadataSchema,
  12. } from '../types/demuxer';
  13. import { dummyTrack } from './dummy-demuxed-track';
  14. import { appendUint8Array } from '../utils/mp4-tools';
  15. import { sliceUint8 } from '../utils/typed-array';
  16.  
  17. class BaseAudioDemuxer implements Demuxer {
  18. protected _audioTrack!: DemuxedAudioTrack;
  19. protected _id3Track!: DemuxedMetadataTrack;
  20. protected frameIndex: number = 0;
  21. protected cachedData: Uint8Array | null = null;
  22. protected basePTS: number | null = null;
  23. protected initPTS: number | null = null;
  24. protected lastPTS: number | null = null;
  25.  
  26. resetInitSegment(
  27. initSegment: Uint8Array | undefined,
  28. audioCodec: string | undefined,
  29. videoCodec: string | undefined,
  30. trackDuration: number
  31. ) {
  32. this._id3Track = {
  33. type: 'id3',
  34. id: 3,
  35. pid: -1,
  36. inputTimeScale: 90000,
  37. sequenceNumber: 0,
  38. samples: [],
  39. dropped: 0,
  40. };
  41. }
  42.  
  43. resetTimeStamp(deaultTimestamp) {
  44. this.initPTS = deaultTimestamp;
  45. this.resetContiguity();
  46. }
  47.  
  48. resetContiguity(): void {
  49. this.basePTS = null;
  50. this.lastPTS = null;
  51. this.frameIndex = 0;
  52. }
  53.  
  54. canParse(data: Uint8Array, offset: number): boolean {
  55. return false;
  56. }
  57.  
  58. appendFrame(
  59. track: DemuxedAudioTrack,
  60. data: Uint8Array,
  61. offset: number
  62. ): AudioFrame | void {}
  63.  
  64. // feed incoming data to the front of the parsing pipeline
  65. demux(data: Uint8Array, timeOffset: number): DemuxerResult {
  66. if (this.cachedData) {
  67. data = appendUint8Array(this.cachedData, data);
  68. this.cachedData = null;
  69. }
  70.  
  71. let id3Data: Uint8Array | undefined = ID3.getID3Data(data, 0);
  72. let offset = id3Data ? id3Data.length : 0;
  73. let lastDataIndex;
  74. const track = this._audioTrack;
  75. const id3Track = this._id3Track;
  76. const timestamp = id3Data ? ID3.getTimeStamp(id3Data) : undefined;
  77. const length = data.length;
  78.  
  79. if (
  80. this.basePTS === null ||
  81. (this.frameIndex === 0 && Number.isFinite(timestamp))
  82. ) {
  83. this.basePTS = initPTSFn(timestamp, timeOffset, this.initPTS);
  84. this.lastPTS = this.basePTS;
  85. }
  86.  
  87. if (this.lastPTS === null) {
  88. this.lastPTS = this.basePTS;
  89. }
  90.  
  91. // more expressive than alternative: id3Data?.length
  92. if (id3Data && id3Data.length > 0) {
  93. id3Track.samples.push({
  94. pts: this.lastPTS,
  95. dts: this.lastPTS,
  96. data: id3Data,
  97. type: MetadataSchema.audioId3,
  98. duration: Number.POSITIVE_INFINITY,
  99. });
  100. }
  101.  
  102. while (offset < length) {
  103. if (this.canParse(data, offset)) {
  104. const frame = this.appendFrame(track, data, offset);
  105. if (frame) {
  106. this.frameIndex++;
  107. this.lastPTS = frame.sample.pts;
  108. offset += frame.length;
  109. lastDataIndex = offset;
  110. } else {
  111. offset = length;
  112. }
  113. } else if (ID3.canParse(data, offset)) {
  114. // after a ID3.canParse, a call to ID3.getID3Data *should* always returns some data
  115. id3Data = ID3.getID3Data(data, offset)!;
  116. id3Track.samples.push({
  117. pts: this.lastPTS,
  118. dts: this.lastPTS,
  119. data: id3Data,
  120. type: MetadataSchema.audioId3,
  121. duration: Number.POSITIVE_INFINITY,
  122. });
  123. offset += id3Data.length;
  124. lastDataIndex = offset;
  125. } else {
  126. offset++;
  127. }
  128. if (offset === length && lastDataIndex !== length) {
  129. const partialData = sliceUint8(data, lastDataIndex);
  130. if (this.cachedData) {
  131. this.cachedData = appendUint8Array(this.cachedData, partialData);
  132. } else {
  133. this.cachedData = partialData;
  134. }
  135. }
  136. }
  137.  
  138. return {
  139. audioTrack: track,
  140. videoTrack: dummyTrack() as DemuxedVideoTrack,
  141. id3Track,
  142. textTrack: dummyTrack() as DemuxedUserdataTrack,
  143. };
  144. }
  145.  
  146. demuxSampleAes(
  147. data: Uint8Array,
  148. keyData: KeyData,
  149. timeOffset: number
  150. ): Promise<DemuxerResult> {
  151. return Promise.reject(
  152. new Error(`[${this}] This demuxer does not support Sample-AES decryption`)
  153. );
  154. }
  155.  
  156. flush(timeOffset: number): DemuxerResult {
  157. // Parse cache in case of remaining frames.
  158. const cachedData = this.cachedData;
  159. if (cachedData) {
  160. this.cachedData = null;
  161. this.demux(cachedData, 0);
  162. }
  163.  
  164. return {
  165. audioTrack: this._audioTrack,
  166. videoTrack: dummyTrack() as DemuxedVideoTrack,
  167. id3Track: this._id3Track,
  168. textTrack: dummyTrack() as DemuxedUserdataTrack,
  169. };
  170. }
  171.  
  172. destroy() {}
  173. }
  174.  
  175. /**
  176. * Initialize PTS
  177. * <p>
  178. * use timestamp unless it is undefined, NaN or Infinity
  179. * </p>
  180. */
  181. export const initPTSFn = (
  182. timestamp: number | undefined,
  183. timeOffset: number,
  184. initPTS: number | null
  185. ): number => {
  186. if (Number.isFinite(timestamp as number)) {
  187. return timestamp! * 90;
  188. }
  189. return timeOffset * 90000 + (initPTS || 0);
  190. };
  191. export default BaseAudioDemuxer;