7 #include "packager/media/formats/webm/webm_muxer.h"
9 #include "packager/media/base/fourccs.h"
10 #include "packager/media/base/media_sample.h"
11 #include "packager/media/base/stream_info.h"
12 #include "packager/media/formats/webm/mkv_writer.h"
13 #include "packager/media/formats/webm/multi_segment_segmenter.h"
14 #include "packager/media/formats/webm/single_segment_segmenter.h"
15 #include "packager/media/formats/webm/two_pass_single_segment_segmenter.h"
22 WebMMuxer::~WebMMuxer() {}
24 Status WebMMuxer::InitializeMuxer() {
25 CHECK_EQ(streams().size(), 1U);
27 if (streams()[0]->is_encrypted() &&
28 streams()[0]->encryption_config().protection_scheme != FOURCC_cenc) {
29 LOG(ERROR) <<
"WebM does not support protection scheme other than 'cenc'.";
30 return Status(error::INVALID_ARGUMENT,
31 "WebM does not support protection scheme other than 'cenc'.");
34 if (!options().segment_template.empty()) {
35 segmenter_.reset(
new MultiSegmentSegmenter(options()));
37 segmenter_.reset(
new TwoPassSingleSegmentSegmenter(options()));
40 Status initialized = segmenter_->Initialize(
41 *streams()[0], progress_listener(), muxer_listener());
42 if (!initialized.ok())
45 FireOnMediaStartEvent();
49 Status WebMMuxer::Finalize() {
52 Status segmenter_finalized = segmenter_->Finalize();
54 if (!segmenter_finalized.ok())
55 return segmenter_finalized;
57 FireOnMediaEndEvent();
62 Status WebMMuxer::AddMediaSample(
size_t stream_id,
const MediaSample& sample) {
64 DCHECK_EQ(stream_id, 0u);
65 if (sample.pts() < 0) {
66 LOG(ERROR) <<
"Seeing negative timestamp " << sample.pts();
67 return Status(error::MUXER_FAILURE,
"Unsupported negative timestamp.");
69 return segmenter_->AddSample(sample);
72 Status WebMMuxer::FinalizeSegment(
size_t stream_id,
73 const SegmentInfo& segment_info) {
75 DCHECK_EQ(stream_id, 0u);
77 if (segment_info.key_rotation_encryption_config) {
78 NOTIMPLEMENTED() <<
"Key rotation is not implemented for WebM.";
79 return Status(error::UNIMPLEMENTED,
80 "Key rotation is not implemented for WebM");
82 return segmenter_->FinalizeSegment(segment_info.start_timestamp,
83 segment_info.duration,
84 segment_info.is_subsegment);
87 void WebMMuxer::FireOnMediaStartEvent() {
88 if (!muxer_listener())
91 DCHECK(!streams().empty()) <<
"Media started without a stream.";
93 const uint32_t timescale = streams().front()->time_scale();
94 muxer_listener()->
OnMediaStart(options(), *streams().front(), timescale,
95 MuxerListener::kContainerWebM);
98 void WebMMuxer::FireOnMediaEndEvent() {
99 if (!muxer_listener())
102 MuxerListener::MediaRanges media_range;
104 uint64_t init_range_start = 0;
105 uint64_t init_range_end = 0;
106 const bool has_init_range =
107 segmenter_->GetInitRangeStartAndEnd(&init_range_start, &init_range_end);
108 if (has_init_range) {
110 r.start = init_range_start;
111 r.end = init_range_end;
112 media_range.init_range = r;
115 uint64_t index_range_start = 0;
116 uint64_t index_range_end = 0;
117 const bool has_index_range = segmenter_->GetIndexRangeStartAndEnd(
118 &index_range_start, &index_range_end);
119 if (has_index_range) {
121 r.start = index_range_start;
122 r.end = index_range_end;
123 media_range.index_range = r;
126 media_range.subsegment_ranges = segmenter_->GetSegmentRanges();
128 const float duration_seconds = segmenter_->GetDurationInSeconds();
129 muxer_listener()->
OnMediaEnd(media_range, duration_seconds);