5#include <packager/media/formats/webm/webm_cluster_parser.h>
10#include <absl/base/internal/endian.h>
11#include <absl/log/check.h>
12#include <absl/log/log.h>
14#include <packager/macros/logging.h>
15#include <packager/media/base/timestamp.h>
16#include <packager/media/codecs/vp8_parser.h>
17#include <packager/media/codecs/vp9_parser.h>
18#include <packager/media/codecs/webvtt_util.h>
19#include <packager/media/formats/webm/webm_constants.h>
20#include <packager/media/formats/webm/webm_crypto_helpers.h>
21#include <packager/media/formats/webm/webm_webvtt_parser.h>
27const int64_t kMicrosecondsPerMillisecond = 1000;
32 int64_t timecode_scale,
33 std::shared_ptr<AudioStreamInfo> audio_stream_info,
34 std::shared_ptr<VideoStreamInfo> video_stream_info,
36 int64_t audio_default_duration,
37 int64_t video_default_duration,
38 const WebMTracksParser::TextTracks& text_tracks,
39 const std::set<int64_t>& ignored_tracks,
40 const std::string& audio_encryption_key_id,
41 const std::string& video_encryption_key_id,
45 : timecode_multiplier_(timecode_scale /
46 static_cast<double>(kMicrosecondsPerMillisecond)),
47 audio_stream_info_(audio_stream_info),
48 video_stream_info_(video_stream_info),
49 vp_config_(vp_config),
50 ignored_tracks_(ignored_tracks),
51 audio_encryption_key_id_(audio_encryption_key_id),
52 video_encryption_key_id_(video_encryption_key_id),
53 parser_(kWebMIdCluster, this),
56 cluster_start_time_(kNoTimestamp),
57 audio_(audio_stream_info ? audio_stream_info->track_id() : -1,
59 audio_default_duration,
61 video_(video_stream_info ? video_stream_info->track_id() : -1,
63 video_default_duration,
65 if (decryption_key_source) {
67 if (audio_stream_info_)
68 audio_stream_info_->set_is_encrypted(
false);
69 if (video_stream_info_)
70 video_stream_info_->set_is_encrypted(
false);
72 for (WebMTracksParser::TextTracks::const_iterator it = text_tracks.begin();
73 it != text_tracks.end(); ++it) {
74 text_track_map_.insert(std::make_pair(
75 it->first, Track(it->first,
false, kNoTimestamp, new_sample_cb)));
79WebMClusterParser::~WebMClusterParser() {}
82 last_block_timecode_ = -1;
83 cluster_timecode_ = -1;
84 cluster_start_time_ = kNoTimestamp;
85 cluster_ended_ =
false;
94 bool audio_result = audio_.ApplyDurationEstimateIfNeeded();
95 bool video_result = video_.ApplyDurationEstimateIfNeeded();
97 return audio_result && video_result;
101 int result = parser_.
Parse(buf, size);
104 cluster_ended_ =
false;
109 if (cluster_ended_) {
112 if (cluster_start_time_ == kNoTimestamp) {
115 if (cluster_timecode_ < 0)
118 cluster_start_time_ = cluster_timecode_ * timecode_multiplier_;
126 last_block_timecode_ = -1;
127 cluster_timecode_ = -1;
134 if (
id == kWebMIdCluster) {
135 cluster_timecode_ = -1;
136 cluster_start_time_ = kNoTimestamp;
137 }
else if (
id == kWebMIdBlockGroup) {
139 block_data_size_ = -1;
140 block_duration_ = -1;
141 discard_padding_ = -1;
142 discard_padding_set_ =
false;
143 reference_block_set_ =
false;
144 }
else if (
id == kWebMIdBlockAdditions) {
146 block_additional_data_.reset();
147 block_additional_data_size_ = 0;
153bool WebMClusterParser::OnListEnd(
int id) {
154 if (
id != kWebMIdBlockGroup)
158 if (block_data_size_ == -1) {
159 LOG(ERROR) <<
"Block missing from BlockGroup.";
163 bool result = ParseBlock(
164 false, block_data_.get(), block_data_size_, block_additional_data_.get(),
165 block_additional_data_size_, block_duration_,
166 discard_padding_set_ ? discard_padding_ : 0, reference_block_set_);
168 block_data_size_ = -1;
169 block_duration_ = -1;
171 block_additional_data_.reset();
172 block_additional_data_size_ = 0;
173 discard_padding_ = -1;
174 discard_padding_set_ =
false;
175 reference_block_set_ =
false;
179bool WebMClusterParser::OnUInt(
int id, int64_t val) {
182 case kWebMIdTimecode:
183 dst = &cluster_timecode_;
185 case kWebMIdBlockDuration:
186 dst = &block_duration_;
188 case kWebMIdBlockAddID:
189 dst = &block_add_id_;
200bool WebMClusterParser::ParseBlock(
bool is_simple_block,
203 const uint8_t* additional,
206 int64_t discard_padding,
207 bool reference_block_set) {
213 if (!(buf[0] & 0x80)) {
214 LOG(ERROR) <<
"TrackNumber over 127 not supported";
218 int track_num = buf[0] & 0x7f;
219 int timecode = buf[1] << 8 | buf[2];
220 int flags = buf[3] & 0xff;
221 int lacing = (flags >> 1) & 0x3;
224 LOG(ERROR) <<
"Lacing " << lacing <<
" is not supported yet.";
229 if (timecode & 0x8000)
237 is_simple_block ? (flags & 0x80) != 0 : !reference_block_set;
239 const uint8_t* frame_data = buf + 4;
240 int frame_size = size - (frame_data - buf);
241 return OnBlock(is_simple_block, track_num, timecode, duration, frame_data,
242 frame_size, additional, additional_size, discard_padding,
246bool WebMClusterParser::OnBinary(
int id,
const uint8_t* data,
int size) {
248 case kWebMIdSimpleBlock:
249 return ParseBlock(
true, data, size, NULL, 0, -1, 0,
false);
253 LOG(ERROR) <<
"More than 1 Block in a BlockGroup is not "
257 block_data_.reset(
new uint8_t[size]);
258 memcpy(block_data_.get(), data, size);
259 block_data_size_ = size;
262 case kWebMIdBlockAdditional: {
263 uint64_t block_add_id = absl::big_endian::FromHost64(block_add_id_);
264 if (block_additional_data_) {
269 LOG(ERROR) <<
"More than 1 BlockAdditional in a "
270 "BlockGroup is not supported.";
276 block_additional_data_size_ = size +
sizeof(block_add_id);
277 block_additional_data_.reset(
new uint8_t[block_additional_data_size_]);
278 memcpy(block_additional_data_.get(), &block_add_id,
sizeof(block_add_id));
279 memcpy(block_additional_data_.get() + 8, data, size);
282 case kWebMIdDiscardPadding: {
283 if (discard_padding_set_ || size <= 0 || size > 8)
285 discard_padding_set_ =
true;
288 discard_padding_ =
static_cast<int8_t
>(data[0]);
289 for (
int i = 1; i < size; ++i)
290 discard_padding_ = (discard_padding_ << 8) | data[i];
294 case kWebMIdReferenceBlock:
298 reference_block_set_ =
true;
305bool WebMClusterParser::OnBlock(
bool is_simple_block,
311 const uint8_t* additional,
316 if (cluster_timecode_ == -1) {
317 LOG(ERROR) <<
"Got a block before cluster timecode.";
324 LOG(ERROR) <<
"Got a block with negative timecode offset " << timecode;
328 if (last_block_timecode_ != -1 && timecode < last_block_timecode_) {
329 LOG(ERROR) <<
"Got a block with a timecode before the previous block.";
334 StreamType stream_type = kStreamUnknown;
335 std::string encryption_key_id;
336 if (track_num == audio_.track_num()) {
338 encryption_key_id = audio_encryption_key_id_;
339 stream_type = kStreamAudio;
340 }
else if (track_num == video_.track_num()) {
342 encryption_key_id = video_encryption_key_id_;
343 stream_type = kStreamVideo;
344 }
else if (ignored_tracks_.find(track_num) != ignored_tracks_.end()) {
346 }
else if (Track*
const text_track = FindTextTrack(track_num)) {
349 if (block_duration < 0)
352 stream_type = kStreamText;
354 LOG(ERROR) <<
"Unexpected track number " << track_num;
357 DCHECK_NE(stream_type, kStreamUnknown);
359 last_block_timecode_ = timecode;
361 int64_t timestamp = (cluster_timecode_ + timecode) * timecode_multiplier_;
363 std::shared_ptr<MediaSample> buffer;
364 if (stream_type != kStreamText) {
368 std::unique_ptr<DecryptConfig> decrypt_config;
370 if (!encryption_key_id.empty() &&
371 !WebMCreateDecryptConfig(
373 reinterpret_cast<const uint8_t*
>(encryption_key_id.data()),
374 encryption_key_id.size(), &decrypt_config, &data_offset)) {
378 const uint8_t* media_data = data + data_offset;
379 const size_t media_data_size = size - data_offset;
382 const size_t kDummyDataSize = 0;
384 additional_size, is_key_frame);
386 if (decrypt_config) {
387 if (!decryptor_source_) {
388 buffer->SetData(media_data, media_data_size);
391 buffer->set_decrypt_config(std::move(decrypt_config));
392 buffer->set_is_encrypted(
true);
394 std::shared_ptr<uint8_t> decrypted_media_data(
395 new uint8_t[media_data_size], std::default_delete<uint8_t[]>());
396 if (!decryptor_source_->DecryptSampleBuffer(
397 decrypt_config.get(), media_data, media_data_size,
398 decrypted_media_data.get())) {
399 LOG(ERROR) <<
"Cannot decrypt samples";
402 buffer->TransferData(std::move(decrypted_media_data), media_data_size);
405 buffer->SetData(media_data, media_data_size);
408 std::string id, settings, content;
411 std::vector<uint8_t> side_data;
412 MakeSideData(
id.begin(),
id.end(), settings.begin(), settings.end(),
416 reinterpret_cast<const uint8_t*
>(content.data()), content.length(),
417 &side_data[0], side_data.size(),
true);
420 buffer->set_dts(timestamp);
421 buffer->set_pts(timestamp);
422 if (cluster_start_time_ == kNoTimestamp)
423 cluster_start_time_ = timestamp;
424 buffer->set_duration(block_duration > 0
425 ? (block_duration * timecode_multiplier_)
428 if (init_cb_ && !initialized_) {
429 std::vector<std::shared_ptr<StreamInfo>> streams;
430 if (audio_stream_info_)
431 streams.push_back(audio_stream_info_);
432 if (video_stream_info_) {
433 if (stream_type == kStreamVideo) {
437 if (video_stream_info_->codec() != kCodecAV1) {
438 std::unique_ptr<VPxParser> vpx_parser;
439 switch (video_stream_info_->codec()) {
441 vpx_parser.reset(
new VP8Parser);
444 vpx_parser.reset(
new VP9Parser);
448 <<
"Unsupported codec " << video_stream_info_->codec();
451 std::vector<VPxFrameInfo> vpx_frames;
452 if (!vpx_parser->Parse(buffer->data(), buffer->data_size(),
454 LOG(ERROR) <<
"Failed to parse vpx frame.";
457 if (vpx_frames.size() != 1u || !vpx_frames[0].is_keyframe) {
458 LOG(ERROR) <<
"The first frame should be a key frame.";
462 vp_config_.
MergeFrom(vpx_parser->codec_config());
463 video_stream_info_->set_codec_string(
465 std::vector<uint8_t> config_serialized;
466 vp_config_.
WriteMP4(&config_serialized);
467 video_stream_info_->set_codec_config(config_serialized);
470 streams.push_back(video_stream_info_);
480 return track->EmitBuffer(buffer);
483WebMClusterParser::Track::Track(
486 int64_t default_duration,
488 : track_num_(track_num),
490 default_duration_(default_duration),
491 estimated_next_frame_duration_(kNoTimestamp),
492 new_sample_cb_(new_sample_cb) {
493 DCHECK(default_duration_ == kNoTimestamp || default_duration_ > 0);
496WebMClusterParser::Track::~Track() {}
498bool WebMClusterParser::Track::EmitBuffer(
499 const std::shared_ptr<MediaSample>& buffer) {
500 DVLOG(2) <<
"EmitBuffer() : " << track_num_ <<
" ts " << buffer->pts()
501 <<
" dur " << buffer->duration() <<
" kf " << buffer->is_key_frame()
502 <<
" size " << buffer->data_size();
504 if (last_added_buffer_missing_duration_.get()) {
505 int64_t derived_duration =
506 buffer->pts() - last_added_buffer_missing_duration_->pts();
507 last_added_buffer_missing_duration_->set_duration(derived_duration);
509 DVLOG(2) <<
"EmitBuffer() : applied derived duration to held-back buffer : "
510 <<
" ts " << last_added_buffer_missing_duration_->pts() <<
" dur "
511 << last_added_buffer_missing_duration_->duration() <<
" kf "
512 << last_added_buffer_missing_duration_->is_key_frame() <<
" size "
513 << last_added_buffer_missing_duration_->data_size();
514 std::shared_ptr<MediaSample> updated_buffer =
515 last_added_buffer_missing_duration_;
516 last_added_buffer_missing_duration_ = NULL;
517 if (!EmitBufferHelp(updated_buffer))
521 if (buffer->duration() == kNoTimestamp) {
522 last_added_buffer_missing_duration_ = buffer;
523 DVLOG(2) <<
"EmitBuffer() : holding back buffer that is missing duration";
527 return EmitBufferHelp(buffer);
530bool WebMClusterParser::Track::ApplyDurationEstimateIfNeeded() {
531 if (!last_added_buffer_missing_duration_.get())
534 int64_t estimated_duration = GetDurationEstimate();
535 last_added_buffer_missing_duration_->set_duration(estimated_duration);
537 VLOG(1) <<
"Track " << track_num_ <<
": Estimating WebM block duration to be "
538 << estimated_duration / 1000
539 <<
"ms for the last (Simple)Block in the Cluster for this Track. Use "
540 "BlockGroups with BlockDurations at the end of each Track in a "
541 "Cluster to avoid estimation.";
543 DVLOG(2) <<
" new dur : ts " << last_added_buffer_missing_duration_->pts()
544 <<
" dur " << last_added_buffer_missing_duration_->duration()
545 <<
" kf " << last_added_buffer_missing_duration_->is_key_frame()
546 <<
" size " << last_added_buffer_missing_duration_->data_size();
550 if (!new_sample_cb_(track_num_, last_added_buffer_missing_duration_))
552 last_added_buffer_missing_duration_ = NULL;
556void WebMClusterParser::Track::Reset() {
557 last_added_buffer_missing_duration_ = NULL;
560bool WebMClusterParser::Track::EmitBufferHelp(
561 const std::shared_ptr<MediaSample>& buffer) {
562 DCHECK(!last_added_buffer_missing_duration_.get());
564 int64_t duration = buffer->duration();
565 if (duration < 0 || duration == kNoTimestamp) {
566 LOG(ERROR) <<
"Invalid buffer duration: " << duration;
573 int64_t orig_duration_estimate = estimated_next_frame_duration_;
574 if (estimated_next_frame_duration_ == kNoTimestamp) {
575 estimated_next_frame_duration_ = duration;
577 estimated_next_frame_duration_ =
578 std::max(duration, estimated_next_frame_duration_);
581 if (orig_duration_estimate != estimated_next_frame_duration_) {
582 DVLOG(3) <<
"Updated duration estimate:" << orig_duration_estimate
583 <<
" -> " << estimated_next_frame_duration_
584 <<
" at timestamp: " << buffer->dts();
588 return new_sample_cb_(track_num_, buffer);
591int64_t WebMClusterParser::Track::GetDurationEstimate() {
592 int64_t duration = kNoTimestamp;
593 if (default_duration_ != kNoTimestamp) {
594 duration = default_duration_;
595 DVLOG(3) << __FUNCTION__ <<
" : using track default duration " << duration;
596 }
else if (estimated_next_frame_duration_ != kNoTimestamp) {
597 duration = estimated_next_frame_duration_;
598 DVLOG(3) << __FUNCTION__ <<
" : using estimated duration " << duration;
601 duration = kDefaultVideoBufferDurationInMs * kMicrosecondsPerMillisecond;
603 duration = kDefaultAudioBufferDurationInMs * kMicrosecondsPerMillisecond;
605 DVLOG(3) << __FUNCTION__ <<
" : using hardcoded default duration "
609 DCHECK_GT(duration, 0);
610 DCHECK_NE(duration, kNoTimestamp);
614void WebMClusterParser::ResetTextTracks() {
615 for (TextTrackMap::iterator it = text_track_map_.begin();
616 it != text_track_map_.end(); ++it) {
621WebMClusterParser::Track* WebMClusterParser::FindTextTrack(
int track_num) {
622 const TextTrackMap::iterator it = text_track_map_.find(track_num);
624 if (it == text_track_map_.end())
All the methods that are virtual are virtual for mocking.