FFMPEG中關於ts流的時長估計的實現

ts流中的時間估計 咱們知道ts流中是沒有時間信息的,我門來看看ffmpeg是怎麼估計其duration方法1.經過pts來估計 static void estimate_timings_from_pts(AVFormatContext *ic, int64_t old_offset) { AVPacket pkt1, *pkt = &pkt1; AVStream *st; int read_size, i, ret; int64_t end_time; int64_t filesize, offset, duration; int retry=0; /* flush packet queue */ flush_packet_queue(ic); for (i=0; i<ic->nb_streams; i++) { st = ic->streams[i]; if (st->start_time == AV_NOPTS_VALUE && st->first_dts == AV_NOPTS_VALUE) av_log(st->codec, AV_LOG_WARNING, "start time is not set in estimate_timings_from_pts\n"); if (st->parser) { av_parser_close(st->parser); st->parser= NULL; } } /* estimate the end time (duration) */ /* XXX: may need to support wrapping */ filesize = ic->pb ? avio_size(ic->pb) : 0;//獲得文件大小 end_time = AV_NOPTS_VALUE; do{ offset = filesize - (DURATION_MAX_READ_SIZE<<retry); if (offset < 0) offset = 0; avio_seek(ic->pb, offset, SEEK_SET);//儘可能日後查找,pts越靠近文件末尾,利用pts估計時長越準確 read_size = 0; for(;;) { if (read_size >= DURATION_MAX_READ_SIZE<<(FFMAX(retry-1,0))) break; do { ret = ff_read_packet(ic, pkt);//從接近文件末尾的地方讀取數據,直到最後一個合法的數據包 } while(ret == AVERROR(EAGAIN)); if (ret != 0) break; read_size += pkt->size; st = ic->streams[pkt->stream_index]; if (pkt->pts != AV_NOPTS_VALUE && (st->start_time != AV_NOPTS_VALUE || st->first_dts != AV_NOPTS_VALUE)) { duration = end_time = pkt->pts;//利用該包的pts數據獲得比較接近的時長 if (st->start_time != AV_NOPTS_VALUE) duration -= st->start_time;//減去初始時間 else duration -= st->first_dts; if (duration > 0) { if (st->duration == AV_NOPTS_VALUE || st->duration < duration) st->duration = duration; } } av_free_packet(pkt); } }while( end_time==AV_NOPTS_VALUE && filesize > (DURATION_MAX_READ_SIZE<<retry) && ++retry <= DURATION_MAX_RETRY);//嘗試 DURATION_MAX_RETRY這麼屢次 } 方法2經過文件大小和碼流來估計 static void estimate_timings_from_bit_rate(AVFormatContext *ic) { int64_t filesize, duration; int bit_rate, i; AVStream *st; /* if bit_rate is already set, we believe it */ if (ic->bit_rate <= 0) { bit_rate = 0; for(i=0;i<ic->nb_streams;i++) {//經過累積各個子流的平均碼率獲得文件的平均碼率 st = ic->streams[i]; if (st->codec->bit_rate > 0) bit_rate += st->codec->bit_rate; } ic->bit_rate = bit_rate; } /* if duration is already set, we believe it */ if (ic->duration == AV_NOPTS_VALUE && ic->bit_rate != 0) { filesize = ic->pb ? avio_size(ic->pb) : 0; if (filesize > 0) { for(i = 0; i < ic->nb_streams; i++) { st = ic->streams[i]; duration= av_rescale(8*filesize, st->time_base.den, ic->bit_rate*(int64_t)st->time_base.num);//經過文件大小除以文件平均碼率獲得文件時長,之因此還有time_base信息,是由於最後要把秒轉換爲以time_base爲單位的值 if (st->duration == AV_NOPTS_VALUE) st->duration = duration; } } } } 對應的,也能夠在android stagefright中加入相似的實現: uint64_t MPEG2TSExtractor::estimateDuration() { Mutex::Autolock autoLock(mLock); int64_t filesize; int64_t end_time; int64_t offset, duration; int retry=1; status_t re=mDataSource->getSize(&filesize);//android中有相似的函數去獲得文件的大小 if (re != OK) { ALOGE("Failed to get file size"); return ERROR_MALFORMED; } uint8_t packet[kTSPacketSize]; unsigned payload_unit_start_indicator = 0; unsigned PID = 0; unsigned adaptation_field_control = 0; //實現思想:從文件末尾開始讀取188個字節,直到找到第一個pes包的邊界,而且跳過adp filed的包,這裏認爲adp field不含有合法的pts while (1) { offset = filesize - kTSPacketSize*retry; ssize_t n = mDataSource->readAt(offset, packet, kTSPacketSize); if (n < (ssize_t)kTSPacketSize) { return (n < 0) ? (status_t)n : ERROR_END_OF_STREAM; } ABitReader* br= new ABitReader((const uint8_t *)packet, kTSPacketSize);//主要此類實現了一個bit讀取器,對於碼流的解析很是方便,相似於ffmpeg中的get_bits.h中實現的功能 unsigned sync_byte = br->getBits(8); CHECK_EQ(sync_byte, 0x47u); br->skipBits(1); unsigned payload_unit_start_indicator = br->getBits(1); br->skipBits(1); PID = br->getBits(13); br->skipBits(2); adaptation_field_control = br->getBits(2); if ((payload_unit_start_indicator == 1) && (adaptation_field_control == 1) && (PID != 0x00u) && (PID != 0x01u) && (PID != 0x02u) ) { break; } retry++; delete br; } ; ABitReader* br= new ABitReader((const uint8_t *)packet, kTSPacketSize); br->skipBits(8 + 3 + 13); br->skipBits(2); adaptation_field_control = br->getBits(2); ALOGV("adaptation_field_control = %u", adaptation_field_control); br->skipBits(4); if (adaptation_field_control == 2 || adaptation_field_control == 3) { unsigned adaptation_field_length = br->getBits(8); if (adaptation_field_length > 0) { br->skipBits(adaptation_field_length * 8); } ALOGV("adaptation_field_length = %u", adaptation_field_length); } if (adaptation_field_control == 1 || adaptation_field_control == 3) { unsigned packet_startcode_prefix = br->getBits(24); ALOGV("packet_startcode_prefix = 0x%08x", packet_startcode_prefix); CHECK_EQ(packet_startcode_prefix, 0x000001u); unsigned stream_id = br->getBits(8); ALOGV("stream_id = 0x%02x", stream_id); br->skipBits(16);  //如下能夠參考標準,標準上解釋的很詳細,應該不難理解 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) { // H.222.1 type E CHECK_EQ(br->getBits(2), 2u); br->skipBits(6); unsigned PTS_DTS_flags = br->getBits(2); ALOGV("PTS_DTS_flags = %u", PTS_DTS_flags); br->skipBits(6); unsigned PES_header_data_length = br->getBits(8); unsigned optional_bytes_remaining = PES_header_data_length; uint64_t PTS = 0, DTS = 0; if (PTS_DTS_flags == 2 || PTS_DTS_flags == 3) { CHECK_GE(optional_bytes_remaining, 5u); CHECK_EQ(br->getBits(4), PTS_DTS_flags); PTS = ((uint64_t)br->getBits(3)) << 30; CHECK_EQ(br->getBits(1), 1u); PTS |= ((uint64_t)br->getBits(15)) << 15; CHECK_EQ(br->getBits(1), 1u); PTS |= br->getBits(15); CHECK_EQ(br->getBits(1), 1u); ALOGV("PTS = %llu", PTS); ALOGV("PTS = %.2f secs", PTS / 90000.0f); PTS = (PTS * 1000 * 1000ll) / 90000; return PTS; } // ES data follows. } } delete br; return 0; }
相關文章
相關標籤/搜索