2014-02-14 23:21:05 +00:00
|
|
|
// Copyright 2014 Google Inc. All rights reserved.
|
|
|
|
//
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file or at
|
|
|
|
// https://developers.google.com/open-source/licenses/bsd
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2017-12-20 00:56:36 +00:00
|
|
|
#ifndef PACKAGER_MEDIA_FORMATS_MP4_FRAGMENTER_H_
|
|
|
|
#define PACKAGER_MEDIA_FORMATS_MP4_FRAGMENTER_H_
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2016-08-17 17:41:40 +00:00
|
|
|
#include <memory>
|
2013-11-12 20:37:58 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2015-12-22 19:19:22 +00:00
|
|
|
#include "packager/base/logging.h"
|
2017-06-29 22:23:53 +00:00
|
|
|
#include "packager/status.h"
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2016-05-20 21:19:33 +00:00
|
|
|
namespace shaka {
|
2013-11-12 20:37:58 +00:00
|
|
|
namespace media {
|
|
|
|
|
|
|
|
class BufferWriter;
|
|
|
|
class MediaSample;
|
2016-05-06 23:56:50 +00:00
|
|
|
class StreamInfo;
|
2013-11-12 20:37:58 +00:00
|
|
|
|
|
|
|
namespace mp4 {
|
|
|
|
|
2018-02-01 20:25:07 +00:00
|
|
|
struct KeyFrameInfo;
|
2014-03-21 17:26:49 +00:00
|
|
|
struct SegmentReference;
|
|
|
|
struct TrackFragment;
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2014-05-08 20:53:08 +00:00
|
|
|
/// Fragmenter is responsible for the generation of MP4 fragments, i.e. 'traf'
|
|
|
|
/// box and corresponding 'mdat' box.
|
2014-04-08 20:21:07 +00:00
|
|
|
class Fragmenter {
|
2013-11-12 20:37:58 +00:00
|
|
|
public:
|
2016-05-06 23:56:50 +00:00
|
|
|
/// @param info contains stream information.
|
2014-01-23 22:34:39 +00:00
|
|
|
/// @param traf points to a TrackFragment box.
|
2017-09-12 17:24:24 +00:00
|
|
|
Fragmenter(std::shared_ptr<const StreamInfo> info, TrackFragment* traf);
|
2014-04-18 18:49:49 +00:00
|
|
|
|
2017-03-11 02:49:55 +00:00
|
|
|
~Fragmenter();
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2014-01-23 22:34:39 +00:00
|
|
|
/// Add a sample to the fragmenter.
|
2014-05-08 20:53:08 +00:00
|
|
|
/// @param sample points to the sample to be added.
|
|
|
|
/// @return OK on success, an error status otherwise.
|
2017-09-12 17:24:24 +00:00
|
|
|
Status AddSample(const MediaSample& sample);
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2014-01-23 22:34:39 +00:00
|
|
|
/// Initialize the fragment with default data.
|
2014-06-30 16:32:06 +00:00
|
|
|
/// @param first_sample_dts specifies the decoding timestamp for the first
|
|
|
|
/// sample for this fragment.
|
2014-04-18 18:49:49 +00:00
|
|
|
/// @return OK on success, an error status otherwise.
|
2017-03-11 02:49:55 +00:00
|
|
|
Status InitializeFragment(int64_t first_sample_dts);
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2014-01-23 22:34:39 +00:00
|
|
|
/// Finalize and optimize the fragment.
|
2017-03-11 02:49:55 +00:00
|
|
|
Status FinalizeFragment();
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2014-01-23 22:34:39 +00:00
|
|
|
/// Fill @a reference with current fragment information.
|
2018-03-27 16:27:17 +00:00
|
|
|
void GenerateSegmentReference(SegmentReference* reference) const;
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2017-02-24 01:17:47 +00:00
|
|
|
void ClearFragmentFinalized() { fragment_finalized_ = false; }
|
|
|
|
|
2014-09-30 21:52:21 +00:00
|
|
|
uint64_t fragment_duration() const { return fragment_duration_; }
|
|
|
|
uint64_t first_sap_time() const { return first_sap_time_; }
|
|
|
|
uint64_t earliest_presentation_time() const {
|
2013-11-12 20:37:58 +00:00
|
|
|
return earliest_presentation_time_;
|
|
|
|
}
|
2014-06-30 16:32:06 +00:00
|
|
|
bool fragment_initialized() const { return fragment_initialized_; }
|
2013-11-12 20:37:58 +00:00
|
|
|
bool fragment_finalized() const { return fragment_finalized_; }
|
|
|
|
BufferWriter* data() { return data_.get(); }
|
2018-02-01 20:25:07 +00:00
|
|
|
const std::vector<KeyFrameInfo>& key_frame_infos() const {
|
|
|
|
return key_frame_infos_;
|
|
|
|
}
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2016-07-12 20:48:28 +00:00
|
|
|
/// Set the flag use_decoding_timestamp_in_timeline, which if set to true, use
|
|
|
|
/// decoding timestamp instead of presentation timestamp in media timeline,
|
|
|
|
/// which is needed to workaround a Chromium bug that decoding timestamp is
|
|
|
|
/// used in buffered range, https://crbug.com/398130.
|
|
|
|
void set_use_decoding_timestamp_in_timeline(
|
|
|
|
bool use_decoding_timestamp_in_timeline) {
|
|
|
|
use_decoding_timestamp_in_timeline_ = use_decoding_timestamp_in_timeline;
|
|
|
|
}
|
|
|
|
|
2018-05-30 22:02:33 +00:00
|
|
|
/// Set the flag allow_use_adjust_earliest_presentation_time, which if set to
|
|
|
|
/// true, earlist_presentation_time (EPT) may be adjusted not to be smaller
|
|
|
|
/// than the decoding timestamp (dts) for the first fragment.
|
|
|
|
void set_allow_adjust_earliest_presentation_time(
|
|
|
|
bool allow_adjust_earliest_presentation_time) {
|
|
|
|
allow_adjust_earliest_presentation_time_ =
|
|
|
|
allow_adjust_earliest_presentation_time;
|
|
|
|
}
|
|
|
|
|
2014-04-18 18:49:49 +00:00
|
|
|
protected:
|
|
|
|
TrackFragment* traf() { return traf_; }
|
|
|
|
|
2014-05-08 20:53:08 +00:00
|
|
|
/// Optimize sample entries table. If all values in @a entries are identical,
|
|
|
|
/// then @a entries is cleared and the value is assigned to @a default_value;
|
|
|
|
/// otherwise it is a NOP. Return true if the table is optimized.
|
|
|
|
template <typename T>
|
|
|
|
bool OptimizeSampleEntries(std::vector<T>* entries, T* default_value);
|
2014-04-18 18:49:49 +00:00
|
|
|
|
2013-11-12 20:37:58 +00:00
|
|
|
private:
|
2017-03-11 02:49:55 +00:00
|
|
|
Status FinalizeFragmentForEncryption();
|
2013-11-12 20:37:58 +00:00
|
|
|
// Check if the current fragment starts with SAP.
|
2018-03-27 16:27:17 +00:00
|
|
|
bool StartsWithSAP() const;
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2017-09-12 17:24:24 +00:00
|
|
|
std::shared_ptr<const StreamInfo> stream_info_;
|
2016-07-12 20:48:28 +00:00
|
|
|
bool use_decoding_timestamp_in_timeline_;
|
2014-04-18 18:49:49 +00:00
|
|
|
TrackFragment* traf_;
|
2016-05-06 23:56:50 +00:00
|
|
|
uint64_t seek_preroll_;
|
2014-06-30 16:32:06 +00:00
|
|
|
bool fragment_initialized_;
|
2013-11-12 20:37:58 +00:00
|
|
|
bool fragment_finalized_;
|
2014-09-30 21:52:21 +00:00
|
|
|
uint64_t fragment_duration_;
|
|
|
|
int64_t earliest_presentation_time_;
|
2018-03-27 16:27:17 +00:00
|
|
|
bool first_fragment_ = true;
|
2018-05-30 22:02:33 +00:00
|
|
|
bool allow_adjust_earliest_presentation_time_ = false;
|
2014-09-30 21:52:21 +00:00
|
|
|
int64_t first_sap_time_;
|
2016-08-17 17:41:40 +00:00
|
|
|
std::unique_ptr<BufferWriter> data_;
|
2018-02-01 20:25:07 +00:00
|
|
|
// Saves key frames information, for Video.
|
|
|
|
std::vector<KeyFrameInfo> key_frame_infos_;
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2014-04-08 20:21:07 +00:00
|
|
|
DISALLOW_COPY_AND_ASSIGN(Fragmenter);
|
2013-11-12 20:37:58 +00:00
|
|
|
};
|
|
|
|
|
2014-05-08 20:53:08 +00:00
|
|
|
template <typename T>
|
|
|
|
bool Fragmenter::OptimizeSampleEntries(std::vector<T>* entries,
|
|
|
|
T* default_value) {
|
|
|
|
DCHECK(entries);
|
|
|
|
DCHECK(default_value);
|
|
|
|
DCHECK(!entries->empty());
|
|
|
|
|
|
|
|
typename std::vector<T>::const_iterator it = entries->begin();
|
|
|
|
T value = *it;
|
|
|
|
for (; it < entries->end(); ++it)
|
|
|
|
if (value != *it)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Clear |entries| if it contains only one value.
|
|
|
|
entries->clear();
|
|
|
|
*default_value = value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-12 20:37:58 +00:00
|
|
|
} // namespace mp4
|
|
|
|
} // namespace media
|
2016-05-20 21:19:33 +00:00
|
|
|
} // namespace shaka
|
2013-11-12 20:37:58 +00:00
|
|
|
|
2017-12-20 00:56:36 +00:00
|
|
|
#endif // PACKAGER_MEDIA_FORMATS_MP4_FRAGMENTER_H_
|