7 #include "packager/media/formats/mp4/fragmenter.h"
11 #include "packager/media/base/buffer_writer.h"
12 #include "packager/media/base/audio_stream_info.h"
13 #include "packager/media/base/media_sample.h"
14 #include "packager/media/formats/mp4/box_definitions.h"
21 const int64_t kInvalidTime = std::numeric_limits<int64_t>::max();
23 uint64_t GetSeekPreroll(
const StreamInfo& stream_info) {
24 if (stream_info.stream_type() != kStreamAudio)
26 const AudioStreamInfo& audio_stream_info =
27 static_cast<const AudioStreamInfo&
>(stream_info);
28 return audio_stream_info.seek_preroll_ns();
33 : use_decoding_timestamp_in_timeline_(false),
35 seek_preroll_(GetSeekPreroll(*info)),
36 fragment_initialized_(false),
37 fragment_finalized_(false),
38 fragment_duration_(0),
39 earliest_presentation_time_(kInvalidTime),
40 first_sap_time_(kInvalidTime) {
44 Fragmenter::~Fragmenter() {}
48 if (sample->duration() == 0) {
49 LOG(WARNING) <<
"Unexpected sample with zero duration @ dts "
53 if (!fragment_initialized_) {
59 if (sample->side_data_size() > 0)
60 LOG(WARNING) <<
"MP4 samples do not support side data. Side data ignored.";
63 traf_->runs[0].sample_sizes.push_back(sample->data_size());
64 traf_->runs[0].sample_durations.push_back(sample->duration());
65 traf_->runs[0].sample_flags.push_back(
66 sample->is_key_frame() ? 0 : TrackFragmentHeader::kNonKeySampleMask);
68 data_->AppendArray(sample->data(), sample->data_size());
69 fragment_duration_ += sample->duration();
71 const int64_t pts = sample->pts();
72 const int64_t dts = sample->dts();
74 const int64_t timestamp = use_decoding_timestamp_in_timeline_ ? dts : pts;
77 if (earliest_presentation_time_ > timestamp)
78 earliest_presentation_time_ = timestamp;
80 traf_->runs[0].sample_composition_time_offsets.push_back(pts - dts);
82 traf_->runs[0].flags |= TrackFragmentRun::kSampleCompTimeOffsetsPresentMask;
84 if (sample->is_key_frame()) {
85 if (first_sap_time_ == kInvalidTime)
86 first_sap_time_ = pts;
92 fragment_initialized_ =
true;
93 fragment_finalized_ =
false;
94 traf_->decode_time.decode_time = first_sample_dts;
96 traf_->runs.resize(1);
97 traf_->runs[0].flags = TrackFragmentRun::kDataOffsetPresentMask;
98 traf_->sample_group_descriptions.clear();
99 traf_->sample_to_groups.clear();
100 traf_->header.sample_description_index = 1;
101 traf_->header.flags = TrackFragmentHeader::kDefaultBaseIsMoofMask |
102 TrackFragmentHeader::kSampleDescriptionIndexPresentMask;
103 fragment_duration_ = 0;
104 earliest_presentation_time_ = kInvalidTime;
105 first_sap_time_ = kInvalidTime;
112 traf_->runs[0].sample_count = traf_->runs[0].sample_sizes.size();
114 &traf_->header.default_sample_duration)) {
115 traf_->header.flags |=
116 TrackFragmentHeader::kDefaultSampleDurationPresentMask;
118 traf_->runs[0].flags |= TrackFragmentRun::kSampleDurationPresentMask;
121 &traf_->header.default_sample_size)) {
122 traf_->header.flags |= TrackFragmentHeader::kDefaultSampleSizePresentMask;
124 traf_->runs[0].flags |= TrackFragmentRun::kSampleSizePresentMask;
127 &traf_->header.default_sample_flags)) {
128 traf_->header.flags |= TrackFragmentHeader::kDefaultSampleFlagsPresentMask;
130 traf_->runs[0].flags |= TrackFragmentRun::kSampleFlagsPresentMask;
138 DCHECK_EQ(traf_->sample_to_groups.size(), 0u);
139 if (seek_preroll_ > 0) {
140 traf_->sample_to_groups.resize(traf_->sample_to_groups.size() + 1);
141 SampleToGroup& sample_to_group = traf_->sample_to_groups.back();
142 sample_to_group.grouping_type = FOURCC_roll;
144 sample_to_group.entries.resize(1);
146 sample_to_group_entry.sample_count = traf_->runs[0].sample_count;
147 sample_to_group_entry.group_description_index =
148 SampleToGroupEntry::kTrackGroupDescriptionIndexBase + 1;
150 for (
const auto& sample_group_description :
151 traf_->sample_group_descriptions) {
152 traf_->sample_to_groups.resize(traf_->sample_to_groups.size() + 1);
153 SampleToGroup& sample_to_group = traf_->sample_to_groups.back();
154 sample_to_group.grouping_type = sample_group_description.grouping_type;
156 sample_to_group.entries.resize(1);
158 sample_to_group_entry.sample_count = traf_->runs[0].sample_count;
159 sample_to_group_entry.group_description_index =
160 SampleToGroupEntry::kTrackFragmentGroupDescriptionIndexBase + 1;
163 fragment_finalized_ =
true;
164 fragment_initialized_ =
false;
169 reference->reference_type =
false;
170 reference->subsegment_duration = fragment_duration_;
171 reference->starts_with_sap = StartsWithSAP();
172 if (kInvalidTime == first_sap_time_) {
173 reference->sap_type = SegmentReference::TypeUnknown;
174 reference->sap_delta_time = 0;
176 reference->sap_type = SegmentReference::Type1;
177 reference->sap_delta_time = first_sap_time_ - earliest_presentation_time_;
179 reference->earliest_presentation_time = earliest_presentation_time_;
182 bool Fragmenter::StartsWithSAP() {
183 DCHECK(!traf_->runs.empty());
184 uint32_t start_sample_flag;
185 if (traf_->runs[0].flags & TrackFragmentRun::kSampleFlagsPresentMask) {
186 DCHECK(!traf_->runs[0].sample_flags.empty());
187 start_sample_flag = traf_->runs[0].sample_flags[0];
189 DCHECK(traf_->header.flags &
190 TrackFragmentHeader::kDefaultSampleFlagsPresentMask);
191 start_sample_flag = traf_->header.default_sample_flags;
193 return (start_sample_flag & TrackFragmentHeader::kNonKeySampleMask) == 0;