2016-03-21 19:16:58 +00:00
|
|
|
// Copyright 2016 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
|
|
|
|
|
|
|
|
#ifndef PACKAGER_MEDIA_FORMATS_MP2T_TS_SEGMENTER_H_
|
|
|
|
#define PACKAGER_MEDIA_FORMATS_MP2T_TS_SEGMENTER_H_
|
|
|
|
|
2016-08-17 17:41:40 +00:00
|
|
|
#include <memory>
|
2016-03-21 19:16:58 +00:00
|
|
|
#include "packager/media/base/muxer_options.h"
|
|
|
|
#include "packager/media/file/file.h"
|
|
|
|
#include "packager/media/formats/mp2t/pes_packet_generator.h"
|
|
|
|
#include "packager/media/formats/mp2t/ts_writer.h"
|
2017-06-29 22:23:53 +00:00
|
|
|
#include "packager/status.h"
|
2016-03-21 19:16:58 +00:00
|
|
|
|
2016-05-20 21:19:33 +00:00
|
|
|
namespace shaka {
|
2016-03-21 19:16:58 +00:00
|
|
|
namespace media {
|
2016-04-16 22:58:47 +00:00
|
|
|
|
2016-04-27 22:04:50 +00:00
|
|
|
class KeySource;
|
2016-04-16 22:58:47 +00:00
|
|
|
class MuxerListener;
|
|
|
|
|
2016-03-21 19:16:58 +00:00
|
|
|
namespace mp2t {
|
|
|
|
|
|
|
|
// TODO(rkuroiwa): For now, this implements multifile segmenter. Like other
|
|
|
|
// make this an abstract super class and implement multifile and single file
|
|
|
|
// segmenters.
|
|
|
|
class TsSegmenter {
|
|
|
|
public:
|
2016-04-27 22:04:50 +00:00
|
|
|
// TODO(rkuroiwa): Add progress listener?
|
2016-03-21 19:16:58 +00:00
|
|
|
/// @param options is the options for this muxer. This must stay valid
|
|
|
|
/// throughout the life time of the instance.
|
2016-04-16 22:58:47 +00:00
|
|
|
/// @param listener is the MuxerListener that should be used to notify events.
|
|
|
|
/// This may be null, in which case no events are sent.
|
|
|
|
TsSegmenter(const MuxerOptions& options, MuxerListener* listener);
|
2016-03-21 19:16:58 +00:00
|
|
|
~TsSegmenter();
|
|
|
|
|
|
|
|
/// Initialize the object.
|
|
|
|
/// @param stream_info is the stream info for the segmenter.
|
|
|
|
/// @return OK on success.
|
2017-03-11 02:49:55 +00:00
|
|
|
Status Initialize(const StreamInfo& stream_info);
|
2016-03-21 19:16:58 +00:00
|
|
|
|
|
|
|
/// Finalize the segmenter.
|
|
|
|
/// @return OK on success.
|
|
|
|
Status Finalize();
|
|
|
|
|
|
|
|
/// @param sample gets added to this object.
|
|
|
|
/// @return OK on success.
|
2017-01-24 00:55:02 +00:00
|
|
|
Status AddSample(std::shared_ptr<MediaSample> sample);
|
2016-03-21 19:16:58 +00:00
|
|
|
|
2017-02-24 01:17:47 +00:00
|
|
|
/// Flush all the samples that are (possibly) buffered and write them to the
|
|
|
|
/// current segment, this will close the file. If a file is not already opened
|
|
|
|
/// before calling this, this will open one and write them to file.
|
|
|
|
/// @param start_timestamp is the segment's start timestamp in the input
|
|
|
|
/// stream's time scale.
|
|
|
|
/// @param duration is the segment's duration in the input stream's time
|
|
|
|
/// scale.
|
|
|
|
// TODO(kqyang): Remove the usage of segment start timestamp and duration in
|
|
|
|
// xx_segmenter, which could cause confusions on which is the source of truth
|
|
|
|
// as the segment start timestamp and duration could be tracked locally.
|
|
|
|
Status FinalizeSegment(uint64_t start_timestamp, uint64_t duration);
|
|
|
|
|
2016-03-21 19:16:58 +00:00
|
|
|
/// Only for testing.
|
2016-08-17 17:41:40 +00:00
|
|
|
void InjectTsWriterForTesting(std::unique_ptr<TsWriter> writer);
|
2016-03-21 19:16:58 +00:00
|
|
|
|
|
|
|
/// Only for testing.
|
|
|
|
void InjectPesPacketGeneratorForTesting(
|
2016-08-17 17:41:40 +00:00
|
|
|
std::unique_ptr<PesPacketGenerator> generator);
|
2016-03-21 19:16:58 +00:00
|
|
|
|
|
|
|
/// Only for testing.
|
|
|
|
void SetTsWriterFileOpenedForTesting(bool value);
|
|
|
|
|
|
|
|
private:
|
|
|
|
Status OpenNewSegmentIfClosed(uint32_t next_pts);
|
|
|
|
|
2016-04-06 05:00:05 +00:00
|
|
|
// Writes PES packets (carried in TsPackets) to a file. If a file is not open,
|
|
|
|
// it will open one. This will not close the file.
|
|
|
|
Status WritePesPacketsToFile();
|
|
|
|
|
2016-03-21 19:16:58 +00:00
|
|
|
const MuxerOptions& muxer_options_;
|
2016-04-16 22:58:47 +00:00
|
|
|
MuxerListener* const listener_;
|
2016-03-21 19:16:58 +00:00
|
|
|
|
2016-04-19 01:00:06 +00:00
|
|
|
// Scale used to scale the input stream to TS's timesccale (which is 90000).
|
|
|
|
// Used for calculating the duration in seconds fo the current segment.
|
|
|
|
double timescale_scale_ = 1.0;
|
|
|
|
|
2016-03-21 19:16:58 +00:00
|
|
|
// Used for segment template.
|
|
|
|
uint64_t segment_number_ = 0;
|
|
|
|
|
2016-08-17 17:41:40 +00:00
|
|
|
std::unique_ptr<TsWriter> ts_writer_;
|
2016-03-21 19:16:58 +00:00
|
|
|
// Set to true if TsWriter::NewFile() succeeds, set to false after
|
|
|
|
// TsWriter::FinalizeFile() succeeds.
|
|
|
|
bool ts_writer_file_opened_ = false;
|
2016-08-17 17:41:40 +00:00
|
|
|
std::unique_ptr<PesPacketGenerator> pes_packet_generator_;
|
2016-03-21 19:16:58 +00:00
|
|
|
|
2016-04-16 22:58:47 +00:00
|
|
|
// For OnNewSegment().
|
|
|
|
// Path of the current segment so that File::GetFileSize() can be used after
|
|
|
|
// the segment has been finalized.
|
|
|
|
std::string current_segment_path_;
|
|
|
|
|
2016-03-21 19:16:58 +00:00
|
|
|
DISALLOW_COPY_AND_ASSIGN(TsSegmenter);
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace mp2t
|
|
|
|
} // namespace media
|
2016-05-20 21:19:33 +00:00
|
|
|
} // namespace shaka
|
2016-03-21 19:16:58 +00:00
|
|
|
#endif // PACKAGER_MEDIA_FORMATS_MP2T_TS_SEGMENTER_H_
|