@jxstjh/jhvideo
Version:
HTML5 jhvideo base on MPEG2-TS Stream Player
1,006 lines (847 loc) • 41 kB
text/typescript
/*
* Copyright (C) 2021 magicxqq. All Rights Reserved.
*
* @author magicxqq <xqq@xqq.im>
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import Log from '../utils/logger';
import DemuxErrors from './demux-errors';
import MediaInfo from '../core/media-info';
import {IllegalStateException} from '../utils/exception';
import BaseDemuxer from './base-demuxer';
import { PAT, PESData, PESSliceQueue, PIDToPESSliceQueues, PMT, ProgramToPMTMap, StreamType } from './pat-pmt-pes';
import { AVCDecoderConfigurationRecord, H264AnnexBParser, H264NaluAVC1, H264NaluPayload, H264NaluType } from './h264';
import SPSParser from './sps-parser';
import { AACADTSParser, AACFrame, AudioSpecificConfig } from './aac';
import { MPEG4AudioObjectTypes, MPEG4SamplingFrequencyIndex } from './mpeg4-audio';
import { PESPrivateData, PESPrivateDataDescriptor } from './pes-private-data';
class TSDemuxer extends BaseDemuxer {
private readonly TAG: string = 'TSDemuxer';
private config_: any;
private ts_packet_size_: number;
private sync_offset_: number;
private first_parse_: boolean = true;
private media_info_ = new MediaInfo();
private timescale_ = 90;
private duration_ = 0;
private pat_: PAT;
private current_program_: number;
private current_pmt_pid_: number = -1;
private pmt_: PMT;
private program_pmt_map_: ProgramToPMTMap = {};
private pes_slice_queues_: PIDToPESSliceQueues = {};
private video_metadata_: {
sps: H264NaluAVC1 | undefined,
pps: H264NaluAVC1 | undefined,
sps_details: any
} = {
sps: undefined,
pps: undefined,
sps_details: undefined
};
private audio_metadata_: {
audio_object_type: MPEG4AudioObjectTypes;
sampling_freq_index: MPEG4SamplingFrequencyIndex;
sampling_frequency: number;
channel_config: number;
} = {
audio_object_type: undefined,
sampling_freq_index: undefined,
sampling_frequency: undefined,
channel_config: undefined
};
private aac_last_sample_pts_: number = undefined;
private aac_last_incomplete_data_: Uint8Array = null;
private has_video_ = false;
private has_audio_ = false;
private video_init_segment_dispatched_ = false;
private audio_init_segment_dispatched_ = false;
private video_metadata_changed_ = false;
private audio_metadata_changed_ = false;
private video_track_ = {type: 'video', id: 1, sequenceNumber: 0, samples: [], length: 0};
private audio_track_ = {type: 'audio', id: 2, sequenceNumber: 0, samples: [], length: 0};
public constructor(probe_data: any, config: any) {
super();
this.ts_packet_size_ = probe_data.ts_packet_size;
this.sync_offset_ = probe_data.sync_offset;
this.config_ = config;
}
public destroy() {
this.media_info_ = null;
this.pes_slice_queues_ = null;
this.video_metadata_ = null;
this.audio_metadata_ = null;
this.aac_last_incomplete_data_ = null;
this.video_track_ = null;
this.audio_track_ = null;
super.destroy();
}
public static probe(buffer: ArrayBuffer) {
let data = new Uint8Array(buffer);
let sync_offset = -1;
let ts_packet_size = 188;
if (data.byteLength <= 3 * ts_packet_size) {
Log.e('TSDemuxer', `Probe data ${data.byteLength} bytes is too few for judging MPEG-TS stream format!`);
return {match: false};
}
while (sync_offset === -1) {
let scan_window = Math.min(1000, data.byteLength - 3 * ts_packet_size);
for (let i = 0; i < scan_window; ) {
// sync_byte should all be 0x47(71)
if (data[i] === 0x47
&& data[i + ts_packet_size] === 0x47
&& data[i + 2 * ts_packet_size] === 0x47) {
sync_offset = i;
break;
} else {
i++;
}
}
// find sync_offset failed in previous ts_packet_size
if (sync_offset === -1) {
if (ts_packet_size === 188) {
// try 192 packet size (BDAV, etc.)
ts_packet_size = 192;
} else if (ts_packet_size === 192) {
// try 204 packet size (European DVB, etc.)
ts_packet_size = 204;
} else {
// 192, 204 also failed, exit
break;
}
}
}
if (sync_offset === -1) {
// both 188, 192, 204 failed, Non MPEG-TS
return {match: false};
}
if (ts_packet_size === 192 && sync_offset >= 4) {
Log.v('TSDemuxer', `ts_packet_size = 192, m2ts mode`);
sync_offset -= 4;
} else if (ts_packet_size === 204) {
Log.v('TSDemuxer', `ts_packet_size = 204, RS encoded MPEG2-TS stream`);
}
return {
match: true,
consumed: 0,
ts_packet_size,
sync_offset
};
}
public bindDataSource(loader) {
loader.onDataArrival = this.parseChunks.bind(this);
return this;
}
public resetMediaInfo() {
this.media_info_ = new MediaInfo();
}
public parseChunks(chunk: ArrayBuffer, byte_start: number): number {
if (!this.onError
|| !this.onMediaInfo
|| !this.onTrackMetadata
|| !this.onDataAvailable) {
throw new IllegalStateException('onError & onMediaInfo & onTrackMetadata & onDataAvailable callback must be specified');
}
let offset = 0;
if (this.first_parse_) {
this.first_parse_ = false;
offset = this.sync_offset_;
}
while (offset + this.ts_packet_size_ <= chunk.byteLength) {
let file_position = byte_start + offset;
if (this.ts_packet_size_ === 192) {
// skip ATS field (2-bits copy-control + 30-bits timestamp) for m2ts
offset += 4;
}
let data = new Uint8Array(chunk, offset, 188);
let sync_byte = data[0];
if (sync_byte !== 0x47) {
Log.e(this.TAG, `sync_byte = ${sync_byte}, not 0x47`);
break;
}
let payload_unit_start_indicator = (data[1] & 0x40) >>> 6;
let transport_priority = (data[1] & 0x20) >>> 5;
let pid = ((data[1] & 0x1F) << 8) | data[2];
let adaptation_field_control = (data[3] & 0x30) >>> 4;
let continuity_conunter = (data[3] & 0x0F);
let adaptation_field_info: {
discontinuity_indicator?: number,
random_access_indicator?: number,
elementary_stream_priority_indicator?: number
} = {};
let ts_payload_start_index = 4;
if (adaptation_field_control == 0x02 || adaptation_field_control == 0x03) {
let adaptation_field_length = data[4];
if (5 + adaptation_field_length === 188) {
// TS packet only has adaption field, jump to next
offset += 188;
if (this.ts_packet_size_ === 204) {
// skip parity word (16 bytes) for RS encoded TS
offset += 16;
}
continue;
} else {
// parse leading adaptation_field if has payload
if (adaptation_field_length > 0) {
adaptation_field_info = this.parseAdaptationField(chunk,
offset + 4,
1 + adaptation_field_length);
}
ts_payload_start_index = 4 + 1 + adaptation_field_length;
}
}
if (adaptation_field_control == 0x01 || adaptation_field_control == 0x03) {
if (pid === 0 || pid === this.current_pmt_pid_) { // PAT(pid === 0) or PMT
if (payload_unit_start_indicator) {
let pointer_field = data[ts_payload_start_index];
// skip pointer_field and strange data
ts_payload_start_index += 1 + pointer_field;
}
let ts_payload_length = 188 - ts_payload_start_index;
if (pid === 0) {
this.parsePAT(chunk,
offset + ts_payload_start_index,
ts_payload_length,
{payload_unit_start_indicator, continuity_conunter});
} else {
this.parsePMT(chunk,
offset + ts_payload_start_index,
ts_payload_length,
{payload_unit_start_indicator, continuity_conunter});
}
} else if (this.pmt_ != undefined && this.pmt_.pid_stream_type[pid] != undefined) {
// PES
let ts_payload_length = 188 - ts_payload_start_index;
let stream_type = this.pmt_.pid_stream_type[pid];
// process PES only for known common_pids
if (pid === this.pmt_.common_pids.h264
|| pid === this.pmt_.common_pids.adts_aac
|| this.pmt_.pes_private_data_pids[pid] === true) {
this.handlePESSlice(chunk,
offset + ts_payload_start_index,
ts_payload_length,
{
pid,
stream_type,
file_position,
payload_unit_start_indicator,
continuity_conunter,
random_access_indicator: adaptation_field_info.random_access_indicator
});
}
}
}
offset += 188;
if (this.ts_packet_size_ === 204) {
// skip parity word (16 bytes) for RS encoded TS
offset += 16;
}
}
// dispatch parsed frames to the remuxer (consumer)
this.dispatchAudioVideoMediaSegment();
return offset; // consumed bytes
}
private parseAdaptationField(buffer: ArrayBuffer, offset: number, length: number): {
discontinuity_indicator?: number,
random_access_indicator?: number,
elementary_stream_priority_indicator?: number
} {
let data = new Uint8Array(buffer, offset, length);
let adaptation_field_length = data[0];
if (adaptation_field_length > 0) {
if (adaptation_field_length > 183) {
Log.w(this.TAG, `Illegal adaptation_field_length: ${adaptation_field_length}`);
return {};
}
let discontinuity_indicator: number = (data[1] & 0x80) >>> 7;
let random_access_indicator: number = (data[1] & 0x40) >>> 6;
let elementary_stream_priority_indicator: number = (data[1] & 0x20) >>> 5;
return {
discontinuity_indicator,
random_access_indicator,
elementary_stream_priority_indicator
};
}
return {};
}
private parsePAT(buffer: ArrayBuffer, offset: number, length: number, misc: any): void {
let data = new Uint8Array(buffer, offset, length);
let table_id = data[0];
if (table_id !== 0x00) {
Log.e(this.TAG, `parsePAT: table_id ${table_id} is not corresponded to PAT!`);
return;
}
let section_length = ((data[1] & 0x0F) << 8) | data[2];
let transport_stream_id = (data[3] << 8) | data[4];
let version_number = (data[5] & 0x3E) >>> 1;
let current_next_indicator = data[5] & 0x01;
let section_number = data[6];
let last_section_number = data[7];
let pat: PAT = null;
if (current_next_indicator === 1 && section_number === 0) {
pat = new PAT();
pat.version_number = version_number;
} else {
pat = this.pat_;
if (pat == undefined) {
return;
}
}
let program_start_index = 8;
let program_bytes = section_length - 5 - 4; // section_length - (headers + crc)
let first_program_number = -1;
let first_pmt_pid = -1;
for (let i = program_start_index; i < program_start_index + program_bytes; i += 4) {
let program_number = (data[i] << 8) | data[i + 1];
let pid = ((data[i + 2] & 0x1F) << 8) | data[i + 3];
if (program_number === 0) {
// network_PID
pat.network_pid = pid;
} else {
// program_map_PID
pat.program_pmt_pid[program_number] = pid;
if (first_program_number === -1) {
first_program_number = program_number;
}
if (first_pmt_pid === -1) {
first_pmt_pid = pid;
}
}
}
// Currently we only deal with first appeared PMT pid
if (current_next_indicator === 1 && section_number === 0) {
if (this.pat_ == undefined) {
Log.v(this.TAG, `Parsed first PAT: ${JSON.stringify(pat)}`);
}
this.pat_ = pat;
this.current_program_ = first_program_number;
this.current_pmt_pid_ = first_pmt_pid;
}
}
private parsePMT(buffer: ArrayBuffer, offset: number, length: number, misc: any): void {
let data = new Uint8Array(buffer, offset, length);
let table_id = data[0];
if (table_id !== 0x02) {
Log.e(this.TAG, `parsePMT: table_id ${table_id} is not corresponded to PMT!`);
return;
}
let section_length = ((data[1] & 0x0F) << 8) | data[2];
let program_number = (data[3] << 8) | data[4];
let version_number = (data[5] & 0x3E) >>> 1;
let current_next_indicator = data[5] & 0x01;
let section_number = data[6];
let last_section_number = data[7];
let pmt: PMT = null;
if (current_next_indicator === 1 && section_number === 0) {
pmt = new PMT();
pmt.program_number = program_number;
pmt.version_number = version_number;
this.program_pmt_map_[program_number] = pmt;
} else {
pmt = this.program_pmt_map_[program_number];
if (pmt == undefined) {
return;
}
}
let PCR_PID = ((data[8] & 0x1F) << 8) | data[9];
let program_info_length = ((data[10] & 0x0F) << 8) | data[11];
let info_start_index = 12 + program_info_length;
let info_bytes = section_length - 9 - program_info_length - 4;
for (let i = info_start_index; i < info_start_index + info_bytes; ) {
let stream_type = data[i] as StreamType;
let elementary_PID = ((data[i + 1] & 0x1F) << 8) | data[i + 2];
let ES_info_length = ((data[i + 3] & 0x0F) << 8) | data[i + 4];
pmt.pid_stream_type[elementary_PID] = stream_type;
if (stream_type === StreamType.kH264 && !pmt.common_pids.h264) {
pmt.common_pids.h264 = elementary_PID;
} else if (stream_type === StreamType.kADTSAAC && !pmt.common_pids.adts_aac) {
pmt.common_pids.adts_aac = elementary_PID;
} else if (stream_type === StreamType.kPESPrivateData) {
pmt.pes_private_data_pids[elementary_PID] = true;
if (ES_info_length > 0) {
// provide descriptor for PES private data via callback
let descriptor = data.subarray(i + 5, i + 5 + ES_info_length);
this.dispatchPESPrivateDataDescriptor(elementary_PID, stream_type, descriptor);
}
}
i += 5 + ES_info_length;
}
if (program_number === this.current_program_) {
if (this.pmt_ == undefined) {
Log.v(this.TAG, `Parsed first PMT: ${JSON.stringify(pmt)}`);
}
this.pmt_ = pmt;
if (pmt.common_pids.h264) {
this.has_video_ = true;
}
if (pmt.common_pids.adts_aac) {
this.has_audio_ = true;
}
}
}
private handlePESSlice(buffer: ArrayBuffer, offset: number, length: number, misc: any): void {
let data = new Uint8Array(buffer, offset, length);
let packet_start_code_prefix = (data[0] << 16) | (data[1] << 8) | (data[2]);
let stream_id = data[3];
let PES_packet_length = (data[4] << 8) | data[5];
if (misc.payload_unit_start_indicator) {
if (packet_start_code_prefix !== 1) {
Log.e(this.TAG, `handlePESSlice: packet_start_code_prefix should be 1 but with value ${packet_start_code_prefix}`);
return;
}
// handle queued PES slices:
// Merge into a big Uint8Array then call parsePES()
let slice_queue = this.pes_slice_queues_[misc.pid];
if (slice_queue) {
let data = new Uint8Array(slice_queue.total_length);
for (let i = 0, offset = 0; i < slice_queue.slices.length; i++) {
let slice = slice_queue.slices[i];
data.set(slice, offset);
offset += slice.byteLength;
}
slice_queue.slices = [];
slice_queue.total_length = 0;
let pes_data = new PESData();
pes_data.pid = misc.pid;
pes_data.data = data;
pes_data.stream_type = misc.stream_type;
pes_data.file_position = slice_queue.file_position;
pes_data.random_access_indicator = slice_queue.random_access_indicator;
this.parsePES(pes_data);
}
// Make a new PES queue for new PES slices
this.pes_slice_queues_[misc.pid] = new PESSliceQueue();
this.pes_slice_queues_[misc.pid].file_position = misc.file_position;
this.pes_slice_queues_[misc.pid].random_access_indicator = misc.random_access_indicator;
}
if (this.pes_slice_queues_[misc.pid] == undefined) {
// ignore PES slices without [PES slice that has payload_unit_start_indicator]
return;
}
// push subsequent PES slices into pes_queue
let slice_queue = this.pes_slice_queues_[misc.pid];
slice_queue.slices.push(data);
slice_queue.total_length += data.byteLength;
}
private parsePES(pes_data: PESData): void {
let data = pes_data.data;
let packet_start_code_prefix = (data[0] << 16) | (data[1] << 8) | (data[2]);
let stream_id = data[3];
let PES_packet_length = (data[4] << 8) | data[5];
if (packet_start_code_prefix !== 1) {
Log.e(this.TAG, `parsePES: packet_start_code_prefix should be 1 but with value ${packet_start_code_prefix}`);
return;
}
if (stream_id !== 0xBC // program_stream_map
&& stream_id !== 0xBE // padding_stream
&& stream_id !== 0xBF // private_stream_2
&& stream_id !== 0xF0 // ECM
&& stream_id !== 0xF1 // EMM
&& stream_id !== 0xFF // program_stream_directory
&& stream_id !== 0xF2 // DSMCC
&& stream_id !== 0xF8) {
let PES_scrambling_control = (data[6] & 0x30) >>> 4;
let PTS_DTS_flags = (data[7] & 0xC0) >>> 6;
let PES_header_data_length = data[8];
let pts: number | undefined;
let dts: number | undefined;
if (PTS_DTS_flags === 0x02 || PTS_DTS_flags === 0x03) {
pts = (data[9] & 0x0E) * 536870912 + // 1 << 29
(data[10] & 0xFF) * 4194304 + // 1 << 22
(data[11] & 0xFE) * 16384 + // 1 << 14
(data[12] & 0xFF) * 128 + // 1 << 7
(data[13] & 0xFE) / 2;
if (PTS_DTS_flags === 0x03) {
dts = (data[14] & 0x0E) * 536870912 + // 1 << 29
(data[15] & 0xFF) * 4194304 + // 1 << 22
(data[16] & 0xFE) * 16384 + // 1 << 14
(data[17] & 0xFF) * 128 + // 1 << 7
(data[18] & 0xFE) / 2;
} else {
dts = pts;
}
}
let payload_start_index = 6 + 3 + PES_header_data_length;
let payload_length: number;
if (PES_packet_length !== 0) {
if (PES_packet_length < 3 + PES_header_data_length) {
Log.v(this.TAG, `Malformed PES: PES_packet_length < 3 + PES_header_data_length`);
return;
}
payload_length = PES_packet_length - 3 - PES_header_data_length;
} else { // PES_packet_length === 0
payload_length = data.byteLength - payload_start_index;
}
let payload = data.subarray(payload_start_index, payload_start_index + payload_length);
switch (pes_data.stream_type) {
case StreamType.kMPEG1Audio:
case StreamType.kMPEG2Audio:
break;
case StreamType.kPESPrivateData:
this.parsePESPrivateDataPayload(payload, pts, dts, pes_data.pid, stream_id);
break;
case StreamType.kADTSAAC:
this.parseAACPayload(payload, pts);
break;
case StreamType.kID3:
break;
case StreamType.kH264:
this.parseH264Payload(payload, pts, dts, pes_data.file_position, pes_data.random_access_indicator);
break;
case StreamType.kH265:
default:
break;
}
} else if (stream_id === 0xBC // program_stream_map
|| stream_id === 0xBF // private_stream_2
|| stream_id === 0xF0 // ECM
|| stream_id === 0xF1 // EMM
|| stream_id === 0xFF // program_stream_directory
|| stream_id === 0xF2 // DSMCC_stream
|| stream_id === 0xF8) { // ITU-T Rec. H.222.1 type E stream
if (pes_data.stream_type === StreamType.kPESPrivateData) {
let payload_start_index = 6;
let payload_length: number;
if (PES_packet_length !== 0) {
payload_length = PES_packet_length;
} else { // PES_packet_length === 0
payload_length = data.byteLength - payload_start_index;
}
let payload = data.subarray(payload_start_index, payload_start_index + payload_length);
this.parsePESPrivateDataPayload(payload, undefined, undefined, pes_data.pid, stream_id);
}
}
}
private parseH264Payload(data: Uint8Array, pts: number, dts: number, file_position: number, random_access_indicator: number) {
let annexb_parser = new H264AnnexBParser(data);
let nalu_payload: H264NaluPayload = null;
let units: {type: H264NaluType, data: Uint8Array}[] = [];
let length = 0;
let keyframe = false;
while ((nalu_payload = annexb_parser.readNextNaluPayload()) != null) {
let nalu_avc1 = new H264NaluAVC1(nalu_payload);
if (nalu_avc1.type === H264NaluType.kSliceSPS) {
// Notice: parseSPS requires Nalu without startcode or length-header
let sps_details = SPSParser.parseSPS(nalu_payload.data);
if (!this.video_init_segment_dispatched_) {
this.video_metadata_.sps = nalu_avc1;
this.video_metadata_.sps_details = sps_details;
} else if (this.detectVideoMetadataChange(nalu_avc1, sps_details) === true) {
Log.v(this.TAG, `H264: Critical h264 metadata has been changed, attempt to re-generate InitSegment`);
this.video_metadata_changed_ = true;
this.video_metadata_ = {sps: nalu_avc1, pps: undefined, sps_details: sps_details};
}
} else if (nalu_avc1.type === H264NaluType.kSlicePPS) {
if (!this.video_init_segment_dispatched_ || this.video_metadata_changed_) {
this.video_metadata_.pps = nalu_avc1;
if (this.video_metadata_.sps && this.video_metadata_.pps) {
if (this.video_metadata_changed_) {
// flush stashed frames before changing codec metadata
this.dispatchVideoMediaSegment();
}
// notify new codec metadata (maybe changed)
this.dispatchVideoInitSegment();
}
}
} else if (nalu_avc1.type === H264NaluType.kSliceIDR) {
keyframe = true;
} else if (nalu_avc1.type === H264NaluType.kSliceNonIDR && random_access_indicator === 1) {
// For open-gop stream, use random_access_indicator to identify keyframe
keyframe = true;
}
// Push samples to remuxer only if initialization metadata has been dispatched
if (this.video_init_segment_dispatched_) {
units.push(nalu_avc1);
length += nalu_avc1.data.byteLength;
}
}
let pts_ms = Math.floor(pts / this.timescale_);
let dts_ms = Math.floor(dts / this.timescale_);
if (units.length) {
let track = this.video_track_;
let avc_sample = {
units,
length,
isKeyframe: keyframe,
dts: dts_ms,
pts: pts_ms,
cts: pts_ms - dts_ms,
file_position
};
track.samples.push(avc_sample);
track.length += length;
}
}
private detectVideoMetadataChange(new_sps: H264NaluAVC1, new_sps_details: any): boolean {
if (new_sps.data.byteLength !== this.video_metadata_.sps.data.byteLength) {
return true;
}
if (new_sps_details.codec_mimetype !== this.video_metadata_.sps_details.codec_mimetype) {
Log.v(this.TAG, `H264: Codec mimeType changed from ` +
`${this.video_metadata_.sps_details.codec_mimetype} to ${new_sps_details.codec_mimetype}`);
return true;
}
if (new_sps_details.codec_size.width !== this.video_metadata_.sps_details.codec_size.width
|| new_sps_details.codec_size.height !== this.video_metadata_.sps_details.codec_size.height) {
let old_size = this.video_metadata_.sps_details.codec_size;
let new_size = new_sps_details.codec_size;
Log.v(this.TAG, `H264: Coded Resolution changed from ` +
`${old_size.width}x${old_size.height} to ${new_size.width}x${new_size.height}`);
return true;
}
if (new_sps_details.present_size.width !== this.video_metadata_.sps_details.present_size.width) {
Log.v(this.TAG, `H264: Present resolution width changed from ` +
`${this.video_metadata_.sps_details.present_size.width} to ${new_sps_details.present_size.width}`);
return true;
}
return false;
}
private isInitSegmentDispatched(): boolean {
if (this.has_video_ && this.has_audio_) { // both video & audio
return this.video_init_segment_dispatched_ && this.audio_init_segment_dispatched_;
}
if (this.has_video_ && !this.has_audio_) { // video only
return this.video_init_segment_dispatched_;
}
if (!this.has_video_ && this.has_audio_) { // audio only
return this.audio_init_segment_dispatched_;
}
return false;
}
private dispatchVideoInitSegment() {
let sps_details = this.video_metadata_.sps_details;
let meta: any = {};
meta.type = 'video';
meta.id = this.video_track_.id;
meta.timescale = 1000;
meta.duration = this.duration_;
meta.codecWidth = sps_details.codec_size.width;
meta.codecHeight = sps_details.codec_size.height;
meta.presentWidth = sps_details.present_size.width;
meta.presentHeight = sps_details.present_size.height;
meta.profile = sps_details.profile_string;
meta.level = sps_details.level_string;
meta.bitDepth = sps_details.bit_depth;
meta.chromaFormat = sps_details.chroma_format;
meta.sarRatio = sps_details.sar_ratio;
meta.frameRate = sps_details.frame_rate;
let fps_den = meta.frameRate.fps_den;
let fps_num = meta.frameRate.fps_num;
meta.refSampleDuration = 1000 * (fps_den / fps_num);
meta.codec = sps_details.codec_mimetype;
let sps_without_header = this.video_metadata_.sps.data.subarray(4);
let pps_without_header = this.video_metadata_.pps.data.subarray(4);
let avcc = new AVCDecoderConfigurationRecord(sps_without_header, pps_without_header, sps_details);
meta.avcc = avcc.getData();
if (this.video_init_segment_dispatched_ == false) {
Log.v(this.TAG, `Generated first AVCDecoderConfigurationRecord for mimeType: ${meta.codec}`);
}
this.onTrackMetadata('video', meta);
this.video_init_segment_dispatched_ = true;
this.video_metadata_changed_ = false;
// notify new MediaInfo
let mi = this.media_info_;
mi.hasVideo = true;
mi.width = meta.codecWidth;
mi.height = meta.codecHeight;
mi.fps = meta.frameRate.fps;
mi.profile = meta.profile;
mi.level = meta.level;
mi.refFrames = sps_details.ref_frames;
mi.chromaFormat = sps_details.chroma_format_string;
mi.sarNum = meta.sarRatio.width;
mi.sarDen = meta.sarRatio.height;
mi.videoCodec = meta.codec;
if (mi.hasAudio && mi.audioCodec) {
mi.mimeType = `video/mp2t; codecs="${mi.videoCodec},${mi.audioCodec}"`;
} else {
mi.mimeType = `video/mp2t; codecs="${mi.videoCodec}"`;
}
if (mi.isComplete()) {
this.onMediaInfo(mi);
}
}
private dispatchVideoMediaSegment() {
if (this.isInitSegmentDispatched()) {
if (this.video_track_.length) {
this.onDataAvailable(null, this.video_track_);
}
}
}
private dispatchAudioMediaSegment() {
if (this.isInitSegmentDispatched()) {
if (this.audio_track_.length) {
this.onDataAvailable(this.audio_track_, null);
}
}
}
private dispatchAudioVideoMediaSegment() {
if (this.isInitSegmentDispatched()) {
if (this.audio_track_.length || this.video_track_.length) {
this.onDataAvailable(this.audio_track_, this.video_track_);
}
}
}
private parseAACPayload(data: Uint8Array, pts: number) {
if (this.has_video_ && !this.video_init_segment_dispatched_) {
// If first video IDR frame hasn't been detected,
// Wait for first IDR frame and video init segment being dispatched
return;
}
if (this.aac_last_incomplete_data_) {
let buf = new Uint8Array(data.byteLength + this.aac_last_incomplete_data_.byteLength);
buf.set(this.aac_last_incomplete_data_, 0);
buf.set(data, this.aac_last_incomplete_data_.byteLength);
data = buf;
}
let ref_sample_duration: number;
let base_pts_ms: number;
if (pts != undefined) {
base_pts_ms = pts / this.timescale_;
} else if (this.aac_last_sample_pts_ != undefined) {
ref_sample_duration = 1024 / this.audio_metadata_.sampling_frequency * 1000;
base_pts_ms = this.aac_last_sample_pts_ + ref_sample_duration;
} else {
Log.w(this.TAG, `AAC: Unknown pts`);
return;
}
if (this.aac_last_incomplete_data_ && this.aac_last_sample_pts_) {
ref_sample_duration = 1024 / this.audio_metadata_.sampling_frequency * 1000;
let new_pts_ms = this.aac_last_sample_pts_ + ref_sample_duration;
if (Math.abs(new_pts_ms - base_pts_ms) > 1) {
Log.w(this.TAG, `AAC: Detected pts overlapped, ` +
`expected: ${new_pts_ms}ms, PES pts: ${base_pts_ms}ms`);
base_pts_ms = new_pts_ms;
}
}
let adts_parser = new AACADTSParser(data);
let aac_frame: AACFrame = null;
let sample_pts_ms = base_pts_ms;
let last_sample_pts_ms: number;
while ((aac_frame = adts_parser.readNextAACFrame()) != null) {
ref_sample_duration = 1024 / aac_frame.sampling_frequency * 1000;
if (this.audio_init_segment_dispatched_ == false) {
this.audio_metadata_.audio_object_type = aac_frame.audio_object_type;
this.audio_metadata_.sampling_freq_index = aac_frame.sampling_freq_index;
this.audio_metadata_.sampling_frequency = aac_frame.sampling_frequency;
this.audio_metadata_.channel_config = aac_frame.channel_config;
this.dispatchAudioInitSegment(aac_frame);
} else if (this.detectAudioMetadataChange(aac_frame)) {
// flush stashed frames before notify new AudioSpecificConfig
this.dispatchAudioMediaSegment();
// notify new AAC AudioSpecificConfig
this.dispatchAudioInitSegment(aac_frame);
}
last_sample_pts_ms = sample_pts_ms;
let sample_pts_ms_int = Math.floor(sample_pts_ms);
let aac_sample = {
unit: aac_frame.data,
length: aac_frame.data.byteLength,
pts: sample_pts_ms_int,
dts: sample_pts_ms_int
};
this.audio_track_.samples.push(aac_sample);
this.audio_track_.length += aac_frame.data.byteLength;
sample_pts_ms += ref_sample_duration;
}
if (adts_parser.hasIncompleteData()) {
this.aac_last_incomplete_data_ = adts_parser.getIncompleteData();
}
if (last_sample_pts_ms) {
this.aac_last_sample_pts_ = last_sample_pts_ms;
}
}
private detectAudioMetadataChange(frame: AACFrame): boolean {
if (frame.audio_object_type !== this.audio_metadata_.audio_object_type) {
Log.v(this.TAG, `AAC: AudioObjectType changed from ` +
`${this.audio_metadata_.audio_object_type} to ${frame.audio_object_type}`);
return true;
}
if (frame.sampling_freq_index !== this.audio_metadata_.sampling_freq_index) {
Log.v(this.TAG, `AAC: SamplingFrequencyIndex changed from ` +
`${this.audio_metadata_.sampling_freq_index} to ${frame.sampling_freq_index}`);
return true;
}
if (frame.channel_config !== this.audio_metadata_.channel_config) {
Log.v(this.TAG, `AAC: Channel configuration changed from ` +
`${this.audio_metadata_.channel_config} to ${frame.channel_config}`);
return true;
}
return false;
}
private dispatchAudioInitSegment(aac_frame: AACFrame) {
let audio_specific_config = new AudioSpecificConfig(aac_frame);
let meta: any = {};
meta.type = 'audio';
meta.id = this.audio_track_.id;
meta.timescale = 1000;
meta.duration = this.duration_;
meta.audioSampleRate = audio_specific_config.sampling_rate;
meta.channelCount = audio_specific_config.channel_count;
meta.codec = audio_specific_config.codec_mimetype;
meta.originalCodec = audio_specific_config.original_codec_mimetype;
meta.config = audio_specific_config.config;
meta.refSampleDuration = 1024 / meta.audioSampleRate * meta.timescale;
if (this.audio_init_segment_dispatched_ == false) {
Log.v(this.TAG, `Generated first AudioSpecificConfig for mimeType: ${meta.codec}`);
}
this.onTrackMetadata('audio', meta);
this.audio_init_segment_dispatched_ = true;
this.video_metadata_changed_ = false;
// notify new MediaInfo
let mi = this.media_info_;
mi.hasAudio = true;
mi.audioCodec = meta.originalCodec;
mi.audioSampleRate = meta.audioSampleRate;
mi.audioChannelCount = meta.channelCount;
if (mi.hasVideo && mi.videoCodec) {
mi.mimeType = `video/mp2t; codecs="${mi.videoCodec},${mi.audioCodec}"`;
} else {
mi.mimeType = `video/mp2t; codecs="${mi.audioCodec}"`;
}
if (mi.isComplete()) {
this.onMediaInfo(mi);
}
}
private dispatchPESPrivateDataDescriptor(pid: number, stream_type: number, descriptor: Uint8Array) {
let desc = new PESPrivateDataDescriptor();
desc.pid = pid;
desc.stream_type = stream_type;
desc.descriptor = descriptor;
if (this.onPESPrivateDataDescriptor) {
this.onPESPrivateDataDescriptor(desc);
}
}
private parsePESPrivateDataPayload(data: Uint8Array, pts: number, dts: number, pid: number, stream_id: number) {
let private_data = new PESPrivateData();
private_data.pid = pid;
private_data.stream_id = stream_id;
private_data.len = data.byteLength;
private_data.data = data;
if (pts != undefined) {
let pts_ms = Math.floor(pts / this.timescale_);
private_data.pts = pts_ms;
} else {
private_data.nearest_pts = this.aac_last_sample_pts_;
}
if (dts != undefined) {
let dts_ms = Math.floor(dts / this.timescale_);
private_data.dts = dts_ms;
}
if (this.onPESPrivateData) {
this.onPESPrivateData(private_data);
}
}
}
export default TSDemuxer;