7#include <packager/media/formats/mp2t/ts_writer.h>
11#include <absl/log/log.h>
13#include <packager/media/base/buffer_writer.h>
14#include <packager/media/base/media_sample.h>
15#include <packager/media/formats/mp2t/pes_packet.h>
16#include <packager/media/formats/mp2t/program_map_table_writer.h>
17#include <packager/media/formats/mp2t/ts_packet_writer_util.h>
28const uint8_t kProgramAssociationTableId = 0x00;
31const uint8_t kPat[] = {
34 kProgramAssociationTableId,
44 ProgramMapTableWriter::kPmtPid,
46 0xF9, 0x62, 0xF5, 0x8B,
50const bool kHasPcr =
true;
51const bool kPayloadUnitStartIndicator =
true;
55const int kTsPacketHeaderSize = 4;
56const int kTsPacketSize = 188;
57const int kTsPacketMaximumPayloadSize = kTsPacketSize - kTsPacketHeaderSize;
59const size_t kMaxPesPacketLengthValue = 0xFFFF;
61void WritePatToBuffer(
const uint8_t* pat,
63 ContinuityCounter* continuity_counter,
64 BufferWriter* writer) {
65 const int kPatPid = 0;
66 WritePayloadToBufferWriter(pat, pat_size, kPayloadUnitStartIndicator, kPatPid,
67 !kHasPcr, 0, continuity_counter, writer);
71void WritePtsOrDts(uint8_t leading_bits,
73 BufferWriter* writer) {
76 leading_bits << 4 | (((pts_or_dts >> 30) & 0x07) << 1) | 1;
78 uint8_t second_byte = (pts_or_dts >> 22) & 0xFF;
80 uint8_t third_byte = (((pts_or_dts >> 15) & 0x7F) << 1) | 1;
82 uint8_t fourth_byte = ((pts_or_dts >> 7) & 0xFF);
84 uint8_t fifth_byte = ((pts_or_dts & 0x7F) << 1) | 1;
85 writer->AppendInt(first_byte);
86 writer->AppendInt(second_byte);
87 writer->AppendInt(third_byte);
88 writer->AppendInt(fourth_byte);
89 writer->AppendInt(fifth_byte);
92bool WritePesToBuffer(
const PesPacket& pes,
93 ContinuityCounter* continuity_counter,
94 BufferWriter* current_buffer) {
96 const int kAdaptationFieldLengthSize = 1;
98 const int kAdaptationFieldHeaderSize = 1;
99 const int kPcrFieldSize = 6;
100 const int kTsPacketMaxPayloadWithPcr =
101 kTsPacketMaximumPayloadSize - kAdaptationFieldLengthSize -
102 kAdaptationFieldHeaderSize - kPcrFieldSize;
103 const uint64_t pcr_base = pes.has_dts() ? pes.dts() : pes.pts();
104 const int pid = ProgramMapTableWriter::kElementaryPid;
107 BufferWriter pes_header_writer;
110 pes_header_writer.AppendInt(
static_cast<uint8_t
>(0x80));
111 pes_header_writer.AppendInt(
112 static_cast<uint8_t
>(
static_cast<int>(pes.has_pts()) << 7 |
113 static_cast<int>(pes.has_dts()) << 6
116 uint8_t pes_header_data_length = 0;
118 pes_header_data_length += 5;
120 pes_header_data_length += 5;
121 pes_header_writer.AppendInt(pes_header_data_length);
123 if (pes.has_pts() && pes.has_dts()) {
124 WritePtsOrDts(0x03, pes.pts(), &pes_header_writer);
125 WritePtsOrDts(0x01, pes.dts(), &pes_header_writer);
126 }
else if (pes.has_pts()) {
127 WritePtsOrDts(0x02, pes.pts(), &pes_header_writer);
132 BufferWriter first_ts_packet_buffer(kTsPacketSize);
133 first_ts_packet_buffer.AppendNBytes(
static_cast<uint64_t
>(0x000001), 3);
134 first_ts_packet_buffer.AppendInt(pes.stream_id());
135 const size_t pes_packet_length = pes.data().size() + pes_header_writer.Size();
136 first_ts_packet_buffer.AppendInt(
static_cast<uint16_t
>(
137 pes_packet_length > kMaxPesPacketLengthValue ? 0 : pes_packet_length));
138 first_ts_packet_buffer.AppendBuffer(pes_header_writer);
140 const size_t available_payload =
141 kTsPacketMaxPayloadWithPcr - first_ts_packet_buffer.Size();
142 const size_t bytes_consumed = std::min(pes.data().size(), available_payload);
143 first_ts_packet_buffer.AppendArray(pes.data().data(), bytes_consumed);
145 BufferWriter output_writer;
146 WritePayloadToBufferWriter(first_ts_packet_buffer.Buffer(),
147 first_ts_packet_buffer.Size(),
148 kPayloadUnitStartIndicator, pid, kHasPcr, pcr_base,
149 continuity_counter, &output_writer);
151 const size_t remaining_pes_data_size = pes.data().size() - bytes_consumed;
152 if (remaining_pes_data_size > 0) {
153 WritePayloadToBufferWriter(pes.data().data() + bytes_consumed,
154 remaining_pes_data_size,
155 !kPayloadUnitStartIndicator, pid, !kHasPcr, 0,
156 continuity_counter, &output_writer);
159 current_buffer->AppendBuffer(output_writer);
165TsWriter::TsWriter(std::unique_ptr<ProgramMapTableWriter> pmt_writer)
166 : pmt_writer_(std::move(pmt_writer)) {}
168TsWriter::~TsWriter() {}
172 WritePatToBuffer(kPat, std::size(kPat), &pat_continuity_counter_, &psi);
174 if (!pmt_writer_->EncryptedSegmentPmt(&psi)) {
178 if (!pmt_writer_->ClearSegmentPmt(&psi)) {
182 buffer->AppendBuffer(psi);
187void TsWriter::SignalEncrypted() {
191bool TsWriter::AddPesPacket(std::unique_ptr<PesPacket> pes_packet,
193 if (!WritePesToBuffer(*pes_packet, &elementary_stream_continuity_counter_,
195 LOG(ERROR) <<
"Failed to write pes to buffer.";
All the methods that are virtual are virtual for mocking.