7 #include "packager/media/formats/mp2t/ts_segmenter.h" 11 #include "packager/media/base/audio_stream_info.h" 12 #include "packager/media/base/muxer_util.h" 13 #include "packager/media/base/video_stream_info.h" 14 #include "packager/media/event/muxer_listener.h" 15 #include "packager/media/formats/mp2t/pes_packet.h" 16 #include "packager/media/formats/mp2t/program_map_table_writer.h" 17 #include "packager/status.h" 24 const double kTsTimescale = 90000;
26 bool IsAudioCodec(Codec codec) {
27 return codec >= kCodecAudio && codec < kCodecAudioMaxPlusOne;
30 bool IsVideoCodec(Codec codec) {
31 return codec >= kCodecVideo && codec < kCodecVideoMaxPlusOne;
37 : muxer_options_(options),
39 transport_stream_timestamp_offset_(
40 options.transport_stream_timestamp_offset_ms * kTsTimescale / 1000),
41 pes_packet_generator_(
44 TsSegmenter::~TsSegmenter() {}
48 return Status(error::MUXER_FAILURE,
"Segment template not specified.");
49 if (!pes_packet_generator_->Initialize(stream_info)) {
50 return Status(error::MUXER_FAILURE,
51 "Failed to initialize PesPacketGenerator.");
54 const StreamType stream_type = stream_info.stream_type();
55 if (stream_type != StreamType::kStreamVideo &&
56 stream_type != StreamType::kStreamAudio) {
57 LOG(ERROR) <<
"TsWriter cannot handle stream type " << stream_type
59 return Status(error::MUXER_FAILURE,
"Unsupported stream type.");
62 codec_ = stream_info.codec();
63 if (stream_type == StreamType::kStreamAudio)
64 audio_codec_config_ = stream_info.codec_config();
66 timescale_scale_ = kTsTimescale / stream_info.time_scale();
76 std::unique_ptr<ProgramMapTableWriter> pmt_writer;
77 if (codec_ == kCodecAC3) {
84 const size_t kSetupDataSize = 10u;
85 if (sample.data_size() < kSetupDataSize) {
86 LOG(ERROR) <<
"Sample is too small for AC3: " << sample.data_size();
87 return Status(error::MUXER_FAILURE,
"Sample is too small for AC3.");
89 const std::vector<uint8_t> setup_data(sample.data(),
90 sample.data() + kSetupDataSize);
92 }
else if (IsAudioCodec(codec_)) {
96 DCHECK(IsVideoCodec(codec_));
99 ts_writer_.reset(
new TsWriter(std::move(pmt_writer)));
102 if (sample.is_encrypted())
103 ts_writer_->SignalEncrypted();
105 if (!ts_writer_file_opened_ && !sample.is_key_frame())
106 LOG(WARNING) <<
"A segment will start with a non key frame.";
108 if (!pes_packet_generator_->PushSample(sample)) {
109 return Status(error::MUXER_FAILURE,
110 "Failed to add sample to PesPacketGenerator.");
112 return WritePesPacketsToFile();
116 ts_writer_ = std::move(writer);
120 std::unique_ptr<PesPacketGenerator> generator) {
121 pes_packet_generator_ = std::move(generator);
125 ts_writer_file_opened_ = value;
128 Status TsSegmenter::OpenNewSegmentIfClosed(uint32_t next_pts) {
129 if (ts_writer_file_opened_)
131 const std::string segment_name =
133 segment_number_++, muxer_options_.
bandwidth);
134 if (!ts_writer_->NewSegment(segment_name))
135 return Status(error::MUXER_FAILURE,
"Failed to initilize TsPacketWriter.");
136 current_segment_path_ = segment_name;
137 ts_writer_file_opened_ =
true;
141 Status TsSegmenter::WritePesPacketsToFile() {
142 while (pes_packet_generator_->NumberOfReadyPesPackets() > 0u) {
143 std::unique_ptr<PesPacket> pes_packet =
144 pes_packet_generator_->GetNextPesPacket();
146 Status status = OpenNewSegmentIfClosed(pes_packet->pts());
150 if (listener_ && IsVideoCodec(codec_) && pes_packet->is_key_frame()) {
151 base::Optional<uint64_t> start_pos = ts_writer_->GetFilePosition();
153 const int64_t timestamp = pes_packet->pts();
154 if (!ts_writer_->AddPesPacket(std::move(pes_packet)))
155 return Status(error::MUXER_FAILURE,
"Failed to add PES packet.");
157 base::Optional<uint64_t> end_pos = ts_writer_->GetFilePosition();
158 if (!start_pos || !end_pos) {
159 return Status(error::MUXER_FAILURE,
160 "Failed to get file position in WritePesPacketsToFile.");
162 listener_->
OnKeyFrame(timestamp, *start_pos, *end_pos - *start_pos);
164 if (!ts_writer_->AddPesPacket(std::move(pes_packet)))
165 return Status(error::MUXER_FAILURE,
"Failed to add PES packet.");
173 if (!pes_packet_generator_->Flush()) {
174 return Status(error::MUXER_FAILURE,
175 "Failed to flush PesPacketGenerator.");
177 Status status = WritePesPacketsToFile();
183 if (ts_writer_file_opened_) {
184 if (!ts_writer_->FinalizeSegment()) {
185 return Status(error::MUXER_FAILURE,
"Failed to finalize TsWriter.");
188 const int64_t file_size =
191 start_timestamp * timescale_scale_ +
192 transport_stream_timestamp_offset_,
193 duration * timescale_scale_, file_size);
195 ts_writer_file_opened_ =
false;
197 current_segment_path_.clear();
All the methods that are virtual are virtual for mocking.
static int64_t GetFileSize(const char *file_name)