mpegtsenc: set Random Access indicator on keyframe start packets
Signed-off-by: Jindrich Makovicka <jindrich.makovicka@nangu.tv> Signed-off-by: Anton Khirnov <anton@khirnov.net>
This commit is contained in:
parent
a58858d60d
commit
575c38d76c
@ -204,6 +204,7 @@ typedef struct MpegTSWriteStream {
|
|||||||
int first_pts_check; ///< first pts check needed
|
int first_pts_check; ///< first pts check needed
|
||||||
int64_t payload_pts;
|
int64_t payload_pts;
|
||||||
int64_t payload_dts;
|
int64_t payload_dts;
|
||||||
|
int payload_flags;
|
||||||
uint8_t payload[DEFAULT_PES_PAYLOAD_SIZE];
|
uint8_t payload[DEFAULT_PES_PAYLOAD_SIZE];
|
||||||
ADTSContext *adts;
|
ADTSContext *adts;
|
||||||
} MpegTSWriteStream;
|
} MpegTSWriteStream;
|
||||||
@ -621,7 +622,7 @@ static int64_t get_pcr(const MpegTSWrite *ts, AVIOContext *pb)
|
|||||||
ts->first_pcr;
|
ts->first_pcr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint8_t* write_pcr_bits(uint8_t *buf, int64_t pcr)
|
static int write_pcr_bits(uint8_t *buf, int64_t pcr)
|
||||||
{
|
{
|
||||||
int64_t pcr_low = pcr % 300, pcr_high = pcr / 300;
|
int64_t pcr_low = pcr % 300, pcr_high = pcr / 300;
|
||||||
|
|
||||||
@ -632,7 +633,7 @@ static uint8_t* write_pcr_bits(uint8_t *buf, int64_t pcr)
|
|||||||
*buf++ = pcr_high << 7 | pcr_low >> 8 | 0x7e;
|
*buf++ = pcr_high << 7 | pcr_low >> 8 | 0x7e;
|
||||||
*buf++ = pcr_low;
|
*buf++ = pcr_low;
|
||||||
|
|
||||||
return buf;
|
return 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Write a single null transport stream packet */
|
/* Write a single null transport stream packet */
|
||||||
@ -668,7 +669,7 @@ static void mpegts_insert_pcr_only(AVFormatContext *s, AVStream *st)
|
|||||||
*q++ = 0x10; /* Adaptation flags: PCR present */
|
*q++ = 0x10; /* Adaptation flags: PCR present */
|
||||||
|
|
||||||
/* PCR coded into 6 bytes */
|
/* PCR coded into 6 bytes */
|
||||||
q = write_pcr_bits(q, get_pcr(ts, s->pb));
|
q += write_pcr_bits(q, get_pcr(ts, s->pb));
|
||||||
|
|
||||||
/* stuffing bytes */
|
/* stuffing bytes */
|
||||||
memset(q, 0xFF, TS_PACKET_SIZE - (q - buf));
|
memset(q, 0xFF, TS_PACKET_SIZE - (q - buf));
|
||||||
@ -689,6 +690,39 @@ static void write_pts(uint8_t *q, int fourbits, int64_t pts)
|
|||||||
*q++ = val;
|
*q++ = val;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Set an adaptation field flag in an MPEG-TS packet*/
|
||||||
|
static void set_af_flag(uint8_t *pkt, int flag)
|
||||||
|
{
|
||||||
|
// expect at least one flag to set
|
||||||
|
assert(flag);
|
||||||
|
|
||||||
|
if ((pkt[3] & 0x20) == 0) {
|
||||||
|
// no AF yet, set adaptation field flag
|
||||||
|
pkt[3] |= 0x20;
|
||||||
|
// 1 byte length, no flags
|
||||||
|
pkt[4] = 1;
|
||||||
|
pkt[5] = 0;
|
||||||
|
}
|
||||||
|
pkt[5] |= flag;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Extend the adaptation field by size bytes */
|
||||||
|
static void extend_af(uint8_t *pkt, int size)
|
||||||
|
{
|
||||||
|
// expect already existing adaptation field
|
||||||
|
assert(pkt[3] & 0x20);
|
||||||
|
pkt[4] += size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get a pointer to MPEG-TS payload (right after TS packet header) */
|
||||||
|
static uint8_t *get_ts_payload_start(uint8_t *pkt)
|
||||||
|
{
|
||||||
|
if (pkt[3] & 0x20)
|
||||||
|
return pkt + 5 + pkt[4];
|
||||||
|
else
|
||||||
|
return pkt + 4;
|
||||||
|
}
|
||||||
|
|
||||||
/* Add a pes header to the front of payload, and segment into an integer number of
|
/* Add a pes header to the front of payload, and segment into an integer number of
|
||||||
* ts packets. The final ts packet is padded using an over-sized adaptation header
|
* ts packets. The final ts packet is padded using an over-sized adaptation header
|
||||||
* to exactly fill the last ts packet.
|
* to exactly fill the last ts packet.
|
||||||
@ -696,7 +730,7 @@ static void write_pts(uint8_t *q, int fourbits, int64_t pts)
|
|||||||
*/
|
*/
|
||||||
static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
|
static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
|
||||||
const uint8_t *payload, int payload_size,
|
const uint8_t *payload, int payload_size,
|
||||||
int64_t pts, int64_t dts)
|
int64_t pts, int64_t dts, int key)
|
||||||
{
|
{
|
||||||
MpegTSWriteStream *ts_st = st->priv_data;
|
MpegTSWriteStream *ts_st = st->priv_data;
|
||||||
MpegTSWrite *ts = s->priv_data;
|
MpegTSWrite *ts = s->priv_data;
|
||||||
@ -741,8 +775,17 @@ static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
|
|||||||
*q++ = val;
|
*q++ = val;
|
||||||
*q++ = ts_st->pid;
|
*q++ = ts_st->pid;
|
||||||
ts_st->cc = (ts_st->cc + 1) & 0xf;
|
ts_st->cc = (ts_st->cc + 1) & 0xf;
|
||||||
*q++ = 0x10 | ts_st->cc | (write_pcr ? 0x20 : 0);
|
*q++ = 0x10 | ts_st->cc; // payload indicator + CC
|
||||||
|
if (key && is_start && pts != AV_NOPTS_VALUE) {
|
||||||
|
// set Random Access for key frames
|
||||||
|
if (ts_st->pid == ts_st->service->pcr_pid)
|
||||||
|
write_pcr = 1;
|
||||||
|
set_af_flag(buf, 0x40);
|
||||||
|
q = get_ts_payload_start(buf);
|
||||||
|
}
|
||||||
if (write_pcr) {
|
if (write_pcr) {
|
||||||
|
set_af_flag(buf, 0x10);
|
||||||
|
q = get_ts_payload_start(buf);
|
||||||
// add 11, pcr references the last byte of program clock reference base
|
// add 11, pcr references the last byte of program clock reference base
|
||||||
if (ts->mux_rate > 1)
|
if (ts->mux_rate > 1)
|
||||||
pcr = get_pcr(ts, s->pb);
|
pcr = get_pcr(ts, s->pb);
|
||||||
@ -750,9 +793,8 @@ static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
|
|||||||
pcr = (dts - delay)*300;
|
pcr = (dts - delay)*300;
|
||||||
if (dts != AV_NOPTS_VALUE && dts < pcr / 300)
|
if (dts != AV_NOPTS_VALUE && dts < pcr / 300)
|
||||||
av_log(s, AV_LOG_WARNING, "dts < pcr, TS is invalid\n");
|
av_log(s, AV_LOG_WARNING, "dts < pcr, TS is invalid\n");
|
||||||
*q++ = 7; /* AFC length */
|
extend_af(buf, write_pcr_bits(q, pcr));
|
||||||
*q++ = 0x10; /* flags: PCR present */
|
q = get_ts_payload_start(buf);
|
||||||
q = write_pcr_bits(q, pcr);
|
|
||||||
}
|
}
|
||||||
if (is_start) {
|
if (is_start) {
|
||||||
int pes_extension = 0;
|
int pes_extension = 0;
|
||||||
@ -950,20 +992,22 @@ static int mpegts_write_packet(AVFormatContext *s, AVPacket *pkt)
|
|||||||
|
|
||||||
if (st->codec->codec_type != AVMEDIA_TYPE_AUDIO) {
|
if (st->codec->codec_type != AVMEDIA_TYPE_AUDIO) {
|
||||||
// for video and subtitle, write a single pes packet
|
// for video and subtitle, write a single pes packet
|
||||||
mpegts_write_pes(s, st, buf, size, pts, dts);
|
mpegts_write_pes(s, st, buf, size, pts, dts, pkt->flags & AV_PKT_FLAG_KEY);
|
||||||
av_free(data);
|
av_free(data);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ts_st->payload_index + size > DEFAULT_PES_PAYLOAD_SIZE) {
|
if (ts_st->payload_index + size > DEFAULT_PES_PAYLOAD_SIZE) {
|
||||||
mpegts_write_pes(s, st, ts_st->payload, ts_st->payload_index,
|
mpegts_write_pes(s, st, ts_st->payload, ts_st->payload_index,
|
||||||
ts_st->payload_pts, ts_st->payload_dts);
|
ts_st->payload_pts, ts_st->payload_dts,
|
||||||
|
ts_st->payload_flags & AV_PKT_FLAG_KEY);
|
||||||
ts_st->payload_index = 0;
|
ts_st->payload_index = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ts_st->payload_index) {
|
if (!ts_st->payload_index) {
|
||||||
ts_st->payload_pts = pts;
|
ts_st->payload_pts = pts;
|
||||||
ts_st->payload_dts = dts;
|
ts_st->payload_dts = dts;
|
||||||
|
ts_st->payload_flags = pkt->flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(ts_st->payload + ts_st->payload_index, buf, size);
|
memcpy(ts_st->payload + ts_st->payload_index, buf, size);
|
||||||
@ -988,7 +1032,8 @@ static int mpegts_write_end(AVFormatContext *s)
|
|||||||
ts_st = st->priv_data;
|
ts_st = st->priv_data;
|
||||||
if (ts_st->payload_index > 0) {
|
if (ts_st->payload_index > 0) {
|
||||||
mpegts_write_pes(s, st, ts_st->payload, ts_st->payload_index,
|
mpegts_write_pes(s, st, ts_st->payload, ts_st->payload_index,
|
||||||
ts_st->payload_pts, ts_st->payload_dts);
|
ts_st->payload_pts, ts_st->payload_dts,
|
||||||
|
ts_st->payload_flags & AV_PKT_FLAG_KEY);
|
||||||
}
|
}
|
||||||
av_freep(&ts_st->adts);
|
av_freep(&ts_st->adts);
|
||||||
}
|
}
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
d260ac0534ff2e26b44b5192fd4fdc21 *./tests/data/lavf/lavf.ts
|
9708011dd80212a0041a96da878122c2 *./tests/data/lavf/lavf.ts
|
||||||
406644 ./tests/data/lavf/lavf.ts
|
406644 ./tests/data/lavf/lavf.ts
|
||||||
./tests/data/lavf/lavf.ts CRC=0x133216c1
|
./tests/data/lavf/lavf.ts CRC=0x133216c1
|
||||||
|
Loading…
x
Reference in New Issue
Block a user