Shaka Packager SDK
Loading...
Searching...
No Matches
media_playlist.cc
1// Copyright 2016 Google LLC. All rights reserved.
2//
3// Use of this source code is governed by a BSD-style
4// license that can be found in the LICENSE file or at
5// https://developers.google.com/open-source/licenses/bsd
6
7#include <packager/hls/base/media_playlist.h>
8
9#include <algorithm>
10#include <cinttypes>
11#include <cmath>
12#include <memory>
13#include <optional>
14
15#include <absl/log/check.h>
16#include <absl/log/log.h>
17#include <absl/strings/numbers.h>
18#include <absl/strings/str_format.h>
19
20#include <packager/file.h>
21#include <packager/hls/base/tag.h>
22#include <packager/macros/logging.h>
23#include <packager/media/base/language_utils.h>
24#include <packager/media/base/muxer_util.h>
25#include <packager/version/version.h>
26
27namespace shaka {
28namespace hls {
29
30namespace {
31int32_t GetTimeScale(const MediaInfo& media_info) {
32 if (media_info.has_reference_time_scale())
33 return media_info.reference_time_scale();
34
35 if (media_info.has_video_info())
36 return media_info.video_info().time_scale();
37
38 if (media_info.has_audio_info())
39 return media_info.audio_info().time_scale();
40 return 0;
41}
42
43std::string AdjustVideoCodec(const std::string& codec) {
44 // Apple does not like video formats with the parameter sets stored in the
45 // samples. It also fails mediastreamvalidator checks and some Apple devices /
46 // platforms refused to play.
47 // See https://apple.co/30n90DC 1.10 and
48 // https://github.com/shaka-project/shaka-packager/issues/587#issuecomment-489182182.
49 // Replaced with the corresponding formats with the parameter sets stored in
50 // the sample descriptions instead.
51 std::string adjusted_codec = codec;
52 std::string fourcc = codec.substr(0, 4);
53 if (fourcc == "avc3")
54 adjusted_codec = "avc1" + codec.substr(4);
55 else if (fourcc == "hev1")
56 adjusted_codec = "hvc1" + codec.substr(4);
57 else if (fourcc == "dvhe")
58 adjusted_codec = "dvh1" + codec.substr(4);
59 if (adjusted_codec != codec) {
60 VLOG(1) << "Adusting video codec string from " << codec << " to "
61 << adjusted_codec;
62 }
63 return adjusted_codec;
64}
65
66// Duplicated from MpdUtils because:
67// 1. MpdUtils header depends on libxml header, which is not in the deps here
68// 2. GetLanguage depends on MediaInfo from packager/mpd/
69// 3. Moving GetLanguage to LanguageUtils would create a a media => mpd dep.
70// TODO(https://github.com/shaka-project/shaka-packager/issues/373): Fix this
71// dependency situation and factor this out to a common location.
72std::string GetLanguage(const MediaInfo& media_info) {
73 std::string lang;
74 if (media_info.has_audio_info()) {
75 lang = media_info.audio_info().language();
76 } else if (media_info.has_text_info()) {
77 lang = media_info.text_info().language();
78 }
79 return LanguageToShortestForm(lang);
80}
81
82void AppendExtXMap(const MediaInfo& media_info, std::string* out) {
83 if (media_info.has_init_segment_url()) {
84 Tag tag("#EXT-X-MAP", out);
85 tag.AddQuotedString("URI", media_info.init_segment_url().data());
86 out->append("\n");
87 } else if (media_info.has_media_file_url() && media_info.has_init_range()) {
88 // It only makes sense for single segment media to have EXT-X-MAP if
89 // there is init_range.
90 Tag tag("#EXT-X-MAP", out);
91 tag.AddQuotedString("URI", media_info.media_file_url().data());
92
93 if (media_info.has_init_range()) {
94 const uint64_t begin = media_info.init_range().begin();
95 const uint64_t end = media_info.init_range().end();
96 const uint64_t length = end - begin + 1;
97
98 tag.AddQuotedNumberPair("BYTERANGE", length, '@', begin);
99 }
100
101 out->append("\n");
102 } else {
103 // This media info does not need an ext-x-map tag.
104 }
105}
106
107std::string CreatePlaylistHeader(
108 const MediaInfo& media_info,
109 int32_t target_duration,
110 HlsPlaylistType type,
111 MediaPlaylist::MediaPlaylistStreamType stream_type,
112 uint32_t media_sequence_number,
113 int discontinuity_sequence_number,
114 std::optional<double> start_time_offset) {
115 const std::string version = GetPackagerVersion();
116 std::string version_line;
117 if (!version.empty()) {
118 version_line =
119 absl::StrFormat("## Generated with %s version %s\n",
120 GetPackagerProjectUrl().c_str(), version.c_str());
121 }
122
123 // 6 is required for EXT-X-MAP without EXT-X-I-FRAMES-ONLY.
124 std::string header = absl::StrFormat(
125 "#EXTM3U\n"
126 "#EXT-X-VERSION:6\n"
127 "%s"
128 "#EXT-X-TARGETDURATION:%d\n",
129 version_line.c_str(), target_duration);
130
131 switch (type) {
132 case HlsPlaylistType::kVod:
133 header += "#EXT-X-PLAYLIST-TYPE:VOD\n";
134 break;
135 case HlsPlaylistType::kEvent:
136 header += "#EXT-X-PLAYLIST-TYPE:EVENT\n";
137 break;
138 case HlsPlaylistType::kLive:
139 if (media_sequence_number > 0) {
140 absl::StrAppendFormat(&header, "#EXT-X-MEDIA-SEQUENCE:%d\n",
141 media_sequence_number);
142 }
143 if (discontinuity_sequence_number > 0) {
144 absl::StrAppendFormat(&header, "#EXT-X-DISCONTINUITY-SEQUENCE:%d\n",
145 discontinuity_sequence_number);
146 }
147 break;
148 default:
149 NOTIMPLEMENTED() << "Unexpected MediaPlaylistType "
150 << static_cast<int>(type);
151 }
152 if (stream_type ==
153 MediaPlaylist::MediaPlaylistStreamType::kVideoIFramesOnly) {
154 absl::StrAppendFormat(&header, "#EXT-X-I-FRAMES-ONLY\n");
155 }
156 if (start_time_offset.has_value()) {
157 absl::StrAppendFormat(&header, "#EXT-X-START:TIME-OFFSET=%f\n",
158 start_time_offset.value());
159 }
160
161 // Put EXT-X-MAP at the end since the rest of the playlist is about the
162 // segment and key info.
163 AppendExtXMap(media_info, &header);
164
165 return header;
166}
167
168
169} // namespace
170
171HlsEntry::HlsEntry(HlsEntry::EntryType type) : type_(type) {}
172HlsEntry::~HlsEntry() {}
173
174class SegmentInfoEntry : public HlsEntry {
175 public:
176 // If |use_byte_range| true then this will append EXT-X-BYTERANGE
177 // after EXTINF.
178 // It uses |previous_segment_end_offset| to determine if it has to also
179 // specify the start byte offset in the tag.
180 // |start_time| is in timescale.
181 // |duration_seconds| is duration in seconds.
182 SegmentInfoEntry(const std::string& file_name,
183 int64_t start_time,
184 double duration_seconds,
185 bool use_byte_range,
186 uint64_t start_byte_offset,
187 uint64_t segment_file_size,
188 uint64_t previous_segment_end_offset);
189
190 std::string ToString() override;
191 int64_t start_time() const { return start_time_; }
192 double duration_seconds() const { return duration_seconds_; }
193 void set_duration_seconds(double duration_seconds) {
194 duration_seconds_ = duration_seconds;
195 }
196
197 private:
198 SegmentInfoEntry(const SegmentInfoEntry&) = delete;
199 SegmentInfoEntry& operator=(const SegmentInfoEntry&) = delete;
200
201 const std::string file_name_;
202 const int64_t start_time_;
203 double duration_seconds_;
204 const bool use_byte_range_;
205 const uint64_t start_byte_offset_;
206 const uint64_t segment_file_size_;
207 const uint64_t previous_segment_end_offset_;
208};
209
210SegmentInfoEntry::SegmentInfoEntry(const std::string& file_name,
211 int64_t start_time,
212 double duration_seconds,
213 bool use_byte_range,
214 uint64_t start_byte_offset,
215 uint64_t segment_file_size,
216 uint64_t previous_segment_end_offset)
217 : HlsEntry(HlsEntry::EntryType::kExtInf),
218 file_name_(file_name),
219 start_time_(start_time),
220 duration_seconds_(duration_seconds),
221 use_byte_range_(use_byte_range),
222 start_byte_offset_(start_byte_offset),
223 segment_file_size_(segment_file_size),
224 previous_segment_end_offset_(previous_segment_end_offset) {}
225
226std::string SegmentInfoEntry::ToString() {
227 std::string result = absl::StrFormat("#EXTINF:%.3f,", duration_seconds_);
228
229 if (use_byte_range_) {
230 absl::StrAppendFormat(&result, "\n#EXT-X-BYTERANGE:%" PRIu64,
231 segment_file_size_);
232 if (previous_segment_end_offset_ + 1 != start_byte_offset_) {
233 absl::StrAppendFormat(&result, "@%" PRIu64, start_byte_offset_);
234 }
235 }
236
237 absl::StrAppendFormat(&result, "\n%s", file_name_.c_str());
238
239 return result;
240}
241
242
243class DiscontinuityEntry : public HlsEntry {
244 public:
245 DiscontinuityEntry();
246
247 std::string ToString() override;
248
249 private:
250 DiscontinuityEntry(const DiscontinuityEntry&) = delete;
251 DiscontinuityEntry& operator=(const DiscontinuityEntry&) = delete;
252};
253
254DiscontinuityEntry::DiscontinuityEntry()
255 : HlsEntry(HlsEntry::EntryType::kExtDiscontinuity) {}
256
257std::string DiscontinuityEntry::ToString() {
258 return "#EXT-X-DISCONTINUITY";
259}
260
261ProgramDateTimeEntry::ProgramDateTimeEntry(const absl::Time& program_time)
262 : HlsEntry(HlsEntry::EntryType::kProgramDateTime),
263 program_time_(program_time) {}
264
265std::string ProgramDateTimeEntry::ToString() {
266 absl::CivilSecond cs =
267 absl::ToCivilSecond(program_time_, absl::UTCTimeZone());
268
269 int64_t total_ms = absl::ToUnixMillis(program_time_);
270 int ms = static_cast<int>(total_ms % 1000);
271 if (ms < 0)
272 ms += 1000; // correction for possible negative times
273
274 return absl::StrFormat(
275 "#EXT-X-PROGRAM-DATE-TIME:%04d-%02d-%02dT%02d:%02d:%02d.%03dZ", cs.year(),
276 cs.month(), cs.day(), cs.hour(), cs.minute(), cs.second(), ms);
277}
278
279class PlacementOpportunityEntry : public HlsEntry {
280 public:
281 PlacementOpportunityEntry();
282
283 std::string ToString() override;
284
285 private:
286 PlacementOpportunityEntry(const PlacementOpportunityEntry&) = delete;
287 PlacementOpportunityEntry& operator=(const PlacementOpportunityEntry&) =
288 delete;
289};
290
291PlacementOpportunityEntry::PlacementOpportunityEntry()
292 : HlsEntry(HlsEntry::EntryType::kExtPlacementOpportunity) {}
293
294std::string PlacementOpportunityEntry::ToString() {
295 return "#EXT-X-PLACEMENT-OPPORTUNITY";
296}
297
298EncryptionInfoEntry::EncryptionInfoEntry(MediaPlaylist::EncryptionMethod method,
299 const std::string& url,
300 const std::string& key_id,
301 const std::string& iv,
302 const std::string& key_format,
303 const std::string& key_format_versions)
304 : HlsEntry(HlsEntry::EntryType::kExtKey),
305 method_(method),
306 url_(url),
307 key_id_(key_id),
308 iv_(iv),
309 key_format_(key_format),
310 key_format_versions_(key_format_versions) {}
311
312std::string EncryptionInfoEntry::ToString() {
313 return ToString("");
314}
315
316std::string EncryptionInfoEntry::ToString(std::string tag_name) {
317 std::string tag_string;
318 if (tag_name.empty())
319 tag_name = "#EXT-X-KEY";
320 Tag tag(tag_name, &tag_string);
321
322 if (method_ == MediaPlaylist::EncryptionMethod::kSampleAes) {
323 tag.AddString("METHOD", "SAMPLE-AES");
324 } else if (method_ == MediaPlaylist::EncryptionMethod::kAes128) {
325 tag.AddString("METHOD", "AES-128");
326 } else if (method_ == MediaPlaylist::EncryptionMethod::kSampleAesCenc) {
327 tag.AddString("METHOD", "SAMPLE-AES-CTR");
328 } else {
329 DCHECK(method_ == MediaPlaylist::EncryptionMethod::kNone);
330 tag.AddString("METHOD", "NONE");
331 }
332
333 tag.AddQuotedString("URI", url_);
334
335 if (!key_id_.empty()) {
336 tag.AddString("KEYID", key_id_);
337 }
338 if (!iv_.empty()) {
339 tag.AddString("IV", iv_);
340 }
341 if (!key_format_versions_.empty()) {
342 tag.AddQuotedString("KEYFORMATVERSIONS", key_format_versions_);
343 }
344 if (!key_format_.empty()) {
345 tag.AddQuotedString("KEYFORMAT", key_format_);
346 }
347
348 return tag_string;
349}
350
351MediaPlaylist::MediaPlaylist(const HlsParams& hls_params,
352 const std::string& file_name,
353 const std::string& name,
354 const std::string& group_id)
355 : hls_params_(hls_params),
356 file_name_(file_name),
357 name_(name),
358 group_id_(group_id),
359 media_sequence_number_(hls_params_.media_sequence_number),
360 reference_time_(absl::InfinitePast()) {
361 // When there's a forced media_sequence_number, start with discontinuity
362 if (media_sequence_number_ > 0)
363 entries_.emplace_back(new DiscontinuityEntry());
364}
365
366MediaPlaylist::~MediaPlaylist() {}
367
369 MediaPlaylistStreamType stream_type) {
370 stream_type_ = stream_type;
371}
372
373void MediaPlaylist::SetCodecForTesting(const std::string& codec) {
374 codec_ = codec;
375}
376
377void MediaPlaylist::SetLanguageForTesting(const std::string& language) {
378 language_ = language;
379}
380
382 const std::vector<std::string>& characteristics) {
383 characteristics_ = characteristics;
384}
385
386void MediaPlaylist::SetForcedSubtitleForTesting(const bool forced_subtitle) {
387 forced_subtitle_ = forced_subtitle;
388}
389
391 MediaPlaylist::EncryptionMethod method,
392 const std::string& url,
393 const std::string& key_id,
394 const std::string& iv,
395 const std::string& key_format,
396 const std::string& key_format_versions) {
397 entries_.emplace_back(new EncryptionInfoEntry(
398 method, url, key_id, iv, key_format, key_format_versions));
399}
400
401bool MediaPlaylist::SetMediaInfo(const MediaInfo& media_info) {
402 const int32_t time_scale = GetTimeScale(media_info);
403 if (time_scale == 0) {
404 LOG(ERROR) << "MediaInfo does not contain a valid timescale.";
405 return false;
406 }
407
408 if (media_info.has_video_info()) {
409 stream_type_ = MediaPlaylistStreamType::kVideo;
410 codec_ = AdjustVideoCodec(media_info.video_info().codec());
411 if (media_info.video_info().has_supplemental_codec() &&
412 media_info.video_info().has_compatible_brand()) {
413 supplemental_codec_ =
414 AdjustVideoCodec(media_info.video_info().supplemental_codec());
415 compatible_brand_ = static_cast<media::FourCC>(
416 media_info.video_info().compatible_brand());
417 }
418 } else if (media_info.has_audio_info()) {
419 stream_type_ = MediaPlaylistStreamType::kAudio;
420 codec_ = media_info.audio_info().codec();
421 } else {
422 stream_type_ = MediaPlaylistStreamType::kSubtitle;
423 codec_ = media_info.text_info().codec();
424 }
425
426 time_scale_ = time_scale;
427 media_info_ = media_info;
428 language_ = GetLanguage(media_info);
429 use_byte_range_ = !media_info_.has_segment_template_url() &&
430 media_info_.container_type() != MediaInfo::CONTAINER_TEXT;
431 characteristics_ =
432 std::vector<std::string>(media_info_.hls_characteristics().begin(),
433 media_info_.hls_characteristics().end());
434
435 forced_subtitle_ = media_info_.forced_subtitle();
436
437 return true;
438}
439
440void MediaPlaylist::SetSampleDuration(int32_t sample_duration) {
441 if (media_info_.has_video_info())
442 media_info_.mutable_video_info()->set_frame_duration(sample_duration);
443}
444
445void MediaPlaylist::AddSegment(const std::string& file_name,
446 int64_t start_time,
447 int64_t duration,
448 uint64_t start_byte_offset,
449 uint64_t size) {
450 if (stream_type_ == MediaPlaylistStreamType::kVideoIFramesOnly) {
451 if (key_frames_.empty())
452 return;
453
454 AdjustLastSegmentInfoEntryDuration(key_frames_.front().timestamp);
455
456 for (auto iter = key_frames_.begin(); iter != key_frames_.end(); ++iter) {
457 // Last entry duration may be adjusted later when the next iframe becomes
458 // available.
459 const int64_t next_timestamp = std::next(iter) == key_frames_.end()
460 ? (start_time + duration)
461 : std::next(iter)->timestamp;
462 AddSegmentInfoEntry(file_name, iter->timestamp,
463 next_timestamp - iter->timestamp,
464 iter->start_byte_offset, iter->size);
465 }
466 key_frames_.clear();
467 return;
468 }
469 return AddSegmentInfoEntry(file_name, start_time, duration, start_byte_offset,
470 size);
471}
472
473void MediaPlaylist::SetReferenceTime(const absl::Time& reference_time) {
474 reference_time_ = reference_time;
475}
476
477void MediaPlaylist::AddKeyFrame(int64_t timestamp,
478 uint64_t start_byte_offset,
479 uint64_t size) {
480 if (stream_type_ != MediaPlaylistStreamType::kVideoIFramesOnly) {
481 if (stream_type_ != MediaPlaylistStreamType::kVideo) {
482 LOG(WARNING)
483 << "I-Frames Only playlist applies to video renditions only.";
484 return;
485 }
486 stream_type_ = MediaPlaylistStreamType::kVideoIFramesOnly;
487 use_byte_range_ = true;
488 }
489 key_frames_.push_back({timestamp, start_byte_offset, size, std::string("")});
490}
491
492void MediaPlaylist::AddEncryptionInfo(MediaPlaylist::EncryptionMethod method,
493 const std::string& url,
494 const std::string& key_id,
495 const std::string& iv,
496 const std::string& key_format,
497 const std::string& key_format_versions) {
498 if (!inserted_discontinuity_tag_) {
499 // Insert discontinuity tag only for the first EXT-X-KEY, only if there
500 // are non-encrypted media segments.
501 if (!entries_.empty())
502 entries_.emplace_back(new DiscontinuityEntry());
503 inserted_discontinuity_tag_ = true;
504 }
505 entries_.emplace_back(new EncryptionInfoEntry(
506 method, url, key_id, iv, key_format, key_format_versions));
507}
508
510 entries_.emplace_back(new PlacementOpportunityEntry());
511}
512
513bool MediaPlaylist::WriteToFile(const std::filesystem::path& file_path) {
514 if (!target_duration_set_) {
516 }
517
518 std::string content = CreatePlaylistHeader(
519 media_info_, target_duration_, hls_params_.playlist_type, stream_type_,
520 media_sequence_number_, discontinuity_sequence_number_,
521 hls_params_.start_time_offset);
522
523 for (const auto& entry : entries_)
524 absl::StrAppendFormat(&content, "%s\n", entry->ToString().c_str());
525
526 if (hls_params_.playlist_type == HlsPlaylistType::kVod) {
527 content += "#EXT-X-ENDLIST\n";
528 }
529
530 if (!File::WriteFileAtomically(file_path.string().c_str(), content)) {
531 LOG(ERROR) << "Failed to write playlist to: " << file_path.string();
532 return false;
533 }
534 return true;
535}
536
538 if (media_info_.has_bandwidth())
539 return media_info_.bandwidth();
540 return bandwidth_estimator_.Max();
541}
542
544 return bandwidth_estimator_.Estimate();
545}
546
548 return longest_segment_duration_seconds_;
549}
550
551void MediaPlaylist::SetTargetDuration(int32_t target_duration) {
552 if (target_duration_set_) {
553 if (target_duration_ == target_duration)
554 return;
555 VLOG(1) << "Updating target duration from " << target_duration_ << " to "
556 << target_duration;
557 }
558 target_duration_ = target_duration;
559 target_duration_set_ = true;
560}
561
563 return media_info_.audio_info().num_channels();
564}
565
567 return media_info_.audio_info().codec_specific_data().ec3_joc_complexity();
568}
569
571 return media_info_.audio_info().codec_specific_data().ac4_ims_flag();
572}
573
575 return media_info_.audio_info().codec_specific_data().ac4_cbi_flag();
576}
577
579 uint32_t* height) const {
580 DCHECK(width);
581 DCHECK(height);
582 if (media_info_.has_video_info()) {
583 const double pixel_aspect_ratio =
584 media_info_.video_info().pixel_height() > 0
585 ? static_cast<double>(media_info_.video_info().pixel_width()) /
586 media_info_.video_info().pixel_height()
587 : 1.0;
588 *width = static_cast<uint32_t>(media_info_.video_info().width() *
589 pixel_aspect_ratio);
590 *height = media_info_.video_info().height();
591 return true;
592 }
593 return false;
594}
595
596std::string MediaPlaylist::GetVideoRange() const {
597 // Dolby Vision (dvh1 or dvhe) is always HDR.
598 if (codec_.find("dvh") == 0)
599 return "PQ";
600
601 // HLS specification:
602 // https://tools.ietf.org/html/draft-pantos-hls-rfc8216bis-02#section-4.4.4.2
603 switch (media_info_.video_info().transfer_characteristics()) {
604 case 1:
605 case 6:
606 case 13:
607 case 14:
608 // Dolby Vision profile 8.4 may have a transfer_characteristics 14, the
609 // actual value refers to preferred_transfer_characteristic value in SEI
610 // message, using compatible brand as a workaround
611 if (!supplemental_codec_.empty() &&
612 compatible_brand_ == media::FOURCC_db4g)
613 return "HLG";
614 else
615 return "SDR";
616 case 15:
617 return "SDR";
618 case 16:
619 return "PQ";
620 case 18:
621 return "HLG";
622 default:
623 // Leave it empty if we do not have the transfer characteristics
624 // information.
625 return "";
626 }
627}
628
630 if (media_info_.video_info().frame_duration() == 0)
631 return 0;
632 return static_cast<double>(time_scale_) /
633 media_info_.video_info().frame_duration();
634}
635
636void MediaPlaylist::AddSegmentInfoEntry(const std::string& segment_file_name,
637 int64_t start_time,
638 int64_t duration,
639 uint64_t start_byte_offset,
640 uint64_t size) {
641 if (time_scale_ == 0) {
642 LOG(WARNING) << "Timescale is not set and the duration for " << duration
643 << " cannot be calculated. The output will be wrong.";
644
645 entries_.emplace_back(new SegmentInfoEntry(
646 segment_file_name, 0.0, 0.0, use_byte_range_, start_byte_offset, size,
647 previous_segment_end_offset_));
648 return;
649 }
650
651 // In order for the oldest segment to be accessible for at least
652 // |time_shift_buffer_depth| seconds, the latest segment should not be in the
653 // sliding window since the player could be playing any part of the latest
654 // segment. So the current segment duration is added to the sum of segment
655 // durations (in the manifest/playlist) after sliding the window.
656 SlideWindow();
657
658 const double segment_duration_seconds =
659 static_cast<double>(duration) / time_scale_;
660 longest_segment_duration_seconds_ =
661 std::max(longest_segment_duration_seconds_, segment_duration_seconds);
662 bandwidth_estimator_.AddBlock(size, segment_duration_seconds);
663 current_buffer_depth_ += segment_duration_seconds;
664
665 if (!entries_.empty() &&
666 entries_.back()->type() == HlsEntry::EntryType::kExtInf) {
667 const SegmentInfoEntry* segment_info =
668 static_cast<SegmentInfoEntry*>(entries_.back().get());
669 if (segment_info->start_time() > start_time) {
670 LOG(WARNING)
671 << "Insert a discontinuity tag after the segment with start time "
672 << segment_info->start_time() << " as the next segment starts at "
673 << start_time << ".";
674 entries_.emplace_back(new DiscontinuityEntry());
675 }
676 }
677
678 if (hls_params_.add_program_date_time &&
679 reference_time_ != absl::InfinitePast()) {
680 // See if we need to add a program date time tag. It is added before the
681 // first segment, and after every discontinuity.
682 bool is_first_segment = true;
683 bool is_discontinuity = false;
684 if (!entries_.empty()) {
685 for (auto it = entries_.rbegin(); it != entries_.rend(); ++it) {
686 if ((*it)->type() == HlsEntry::EntryType::kExtInf) {
687 is_first_segment = false;
688 break;
689 }
690 }
691
692 const auto& last = *entries_.back();
693 if (last.type() == HlsEntry::EntryType::kExtDiscontinuity) {
694 is_discontinuity = true;
695 } else if (entries_.size() >= 2) {
696 const auto& second_last = **std::prev(entries_.cend(), 2);
697 if (last.type() == HlsEntry::EntryType::kExtKey &&
698 second_last.type() == HlsEntry::EntryType::kExtDiscontinuity) {
699 is_discontinuity = true;
700 }
701 }
702 }
703
704 if (is_first_segment || is_discontinuity) {
705 const absl::Time program_time =
706 reference_time_ +
707 absl::Seconds(static_cast<double>(start_time) / time_scale_);
708 entries_.emplace_back(new ProgramDateTimeEntry(program_time));
709 }
710 }
711
712 entries_.emplace_back(new SegmentInfoEntry(
713 segment_file_name, start_time, segment_duration_seconds, use_byte_range_,
714 start_byte_offset, size, previous_segment_end_offset_));
715 previous_segment_end_offset_ = start_byte_offset + size - 1;
716}
717
718void MediaPlaylist::AdjustLastSegmentInfoEntryDuration(int64_t next_timestamp) {
719 if (time_scale_ == 0)
720 return;
721
722 const double next_timestamp_seconds =
723 static_cast<double>(next_timestamp) / time_scale_;
724
725 for (auto iter = entries_.rbegin(); iter != entries_.rend(); ++iter) {
726 if (iter->get()->type() == HlsEntry::EntryType::kExtInf) {
727 SegmentInfoEntry* segment_info =
728 reinterpret_cast<SegmentInfoEntry*>(iter->get());
729
730 const double segment_duration_seconds =
731 next_timestamp_seconds -
732 static_cast<double>(segment_info->start_time()) / time_scale_;
733 // It could be negative if timestamp messed up.
734 if (segment_duration_seconds > 0)
735 segment_info->set_duration_seconds(segment_duration_seconds);
736 longest_segment_duration_seconds_ =
737 std::max(longest_segment_duration_seconds_, segment_duration_seconds);
738 break;
739 }
740 }
741}
742
743// TODO(kqyang): Right now this class manages the segments including the
744// deletion of segments when it is no longer needed. However, this class does
745// not have access to the segment file paths, which is already translated to
746// segment URLs by HlsNotifier. We have to re-generate segment file paths from
747// segment template here in order to delete the old segments.
748// To make the pipeline cleaner, we should move all file manipulations including
749// segment management to an intermediate layer between HlsNotifier and
750// MediaPlaylist.
751void MediaPlaylist::SlideWindow() {
752 if (hls_params_.time_shift_buffer_depth <= 0.0 ||
753 hls_params_.playlist_type != HlsPlaylistType::kLive) {
754 return;
755 }
756 DCHECK_GT(time_scale_, 0);
757
758 if (current_buffer_depth_ <= hls_params_.time_shift_buffer_depth)
759 return;
760
761 // Temporary list to hold the EXT-X-KEYs. For example, this allows us to
762 // remove <3> without removing <1> and <2> below (<1> and <2> are moved to the
763 // temporary list and added back later).
764 // #EXT-X-KEY <1>
765 // #EXT-X-KEY <2>
766 // #EXTINF <3>
767 // #EXTINF <4>
768 std::list<std::unique_ptr<HlsEntry>> ext_x_keys;
769 // Consecutive key entries are either fully removed or not removed at all.
770 // Keep track of entry types so we know if it is consecutive key entries.
771 HlsEntry::EntryType prev_entry_type = HlsEntry::EntryType::kExtInf;
772
773 std::list<std::unique_ptr<HlsEntry>>::iterator last = entries_.begin();
774 for (; last != entries_.end(); ++last) {
775 HlsEntry::EntryType entry_type = last->get()->type();
776 if (entry_type == HlsEntry::EntryType::kExtKey) {
777 if (prev_entry_type != HlsEntry::EntryType::kExtKey)
778 ext_x_keys.clear();
779 ext_x_keys.push_back(std::move(*last));
780 } else if (entry_type == HlsEntry::EntryType::kExtDiscontinuity) {
781 ++discontinuity_sequence_number_;
782 } else {
783 DCHECK_EQ(static_cast<int>(entry_type),
784 static_cast<int>(HlsEntry::EntryType::kExtInf));
785
786 const SegmentInfoEntry& segment_info =
787 *reinterpret_cast<SegmentInfoEntry*>(last->get());
788 // Remove the current segment only if it falls completely out of time
789 // shift buffer range.
790 const bool segment_within_time_shift_buffer =
791 current_buffer_depth_ - segment_info.duration_seconds() <
792 hls_params_.time_shift_buffer_depth;
793 if (segment_within_time_shift_buffer)
794 break;
795 current_buffer_depth_ -= segment_info.duration_seconds();
796 RemoveOldSegment(segment_info.start_time());
797 media_sequence_number_++;
798 }
799 prev_entry_type = entry_type;
800 }
801 entries_.erase(entries_.begin(), last);
802 // Add key entries back.
803 entries_.insert(entries_.begin(), std::make_move_iterator(ext_x_keys.begin()),
804 std::make_move_iterator(ext_x_keys.end()));
805}
806
807void MediaPlaylist::RemoveOldSegment(int64_t start_time) {
808 if (hls_params_.preserved_segments_outside_live_window == 0)
809 return;
810 if (stream_type_ == MediaPlaylistStreamType::kVideoIFramesOnly)
811 return;
812
813 segments_to_be_removed_.push_back(media::GetSegmentName(
814 media_info_.segment_template(), start_time, media_sequence_number_ + 1,
815 media_info_.bandwidth()));
816 while (segments_to_be_removed_.size() >
817 hls_params_.preserved_segments_outside_live_window) {
818 VLOG(2) << "Deleting " << segments_to_be_removed_.front();
819 if (!File::Delete(segments_to_be_removed_.front().c_str())) {
820 LOG(WARNING) << "Failed to delete " << segments_to_be_removed_.front()
821 << "; Will retry later.";
822 break;
823 }
824 segments_to_be_removed_.pop_front();
825 }
826}
827
828} // namespace hls
829} // namespace shaka
void AddBlock(uint64_t size_in_bytes, double duration)
virtual bool GetAC4ImsFlag() const
void SetStreamTypeForTesting(MediaPlaylistStreamType stream_type)
For testing only.
virtual void AddEncryptionInfo(EncryptionMethod method, const std::string &url, const std::string &key_id, const std::string &iv, const std::string &key_format, const std::string &key_format_versions)
void SetCharacteristicsForTesting(const std::vector< std::string > &characteristics)
For testing only.
virtual void AddKeyFrame(int64_t timestamp, uint64_t start_byte_offset, uint64_t size)
virtual double GetFrameRate() const
virtual uint64_t AvgBitrate() const
virtual bool GetDisplayResolution(uint32_t *width, uint32_t *height) const
virtual double GetLongestSegmentDuration() const
virtual int GetEC3JocComplexity() const
virtual void SetReferenceTime(const absl::Time &reference_time)
virtual uint64_t MaxBitrate() const
void SetLanguageForTesting(const std::string &language)
For testing only.
virtual int GetNumChannels() const
virtual bool WriteToFile(const std::filesystem::path &file_path)
virtual std::string GetVideoRange() const
void SetCodecForTesting(const std::string &codec)
For testing only.
virtual void AddPlacementOpportunity()
const std::string & language() const
void AddEncryptionInfoForTesting(MediaPlaylist::EncryptionMethod method, const std::string &url, const std::string &key_id, const std::string &iv, const std::string &key_format, const std::string &key_format_versions)
For testing only.
virtual bool SetMediaInfo(const MediaInfo &media_info)
virtual void SetTargetDuration(int32_t target_duration)
virtual bool GetAC4CbiFlag() const
void SetForcedSubtitleForTesting(const bool forced_subtitle)
For testing only.
virtual void SetSampleDuration(int32_t sample_duration)
virtual void AddSegment(const std::string &file_name, int64_t start_time, int64_t duration, uint64_t start_byte_offset, uint64_t size)
All the methods that are virtual are virtual for mocking.
std::string LanguageToShortestForm(const std::string &language)