From 92548fcaa1219e1673f8dd33f71fd796fabe5a43 Mon Sep 17 00:00:00 2001
From: Jindrich Makovicka <jindrich.makovicka@nangu.tv>
Date: Thu, 5 May 2011 14:33:32 +0200
Subject: [PATCH] set Random Access indicator on keyframe start packets

needed for some DVB recorders

Signed-off-by: Jindrich Makovicka <makovick@gmail.com>
---
 libavformat/mpegtsenc.c |   36 ++++++++++++++++++++----------------
 1 files changed, 20 insertions(+), 16 deletions(-)

diff --git a/libavformat/mpegtsenc.c b/libavformat/mpegtsenc.c
index bf66aa0..db2920d 100644
--- a/libavformat/mpegtsenc.c
+++ b/libavformat/mpegtsenc.c
@@ -693,7 +693,7 @@ static void write_pts(uint8_t *q, int fourbits, int64_t pts)
  */
 static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
                              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;
     MpegTSWrite *ts = s->priv_data;
@@ -738,18 +738,21 @@ static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
         *q++ = val;
         *q++ = ts_st->pid;
         ts_st->cc = (ts_st->cc + 1) & 0xf;
-        *q++ = 0x10 | ts_st->cc | (write_pcr ? 0x20 : 0);
-        if (write_pcr) {
-            // add 11, pcr references the last byte of program clock reference base
-            if (ts->mux_rate > 1)
-                pcr = get_pcr(ts, s->pb);
-            else
-                pcr = (dts - delay)*300;
-            if (dts != AV_NOPTS_VALUE && dts < pcr / 300)
-                av_log(s, AV_LOG_WARNING, "dts < pcr, TS is invalid\n");
-            *q++ = 7; /* AFC length */
-            *q++ = 0x10; /* flags: PCR present */
-            q = write_pcr_bits(q, pcr);
+        *q++ = 0x10 | ts_st->cc | ((write_pcr || (key && is_start)) ? 0x20 : 0);
+        if (write_pcr || (key && is_start)) {
+            *q++ = 1 + (write_pcr ? 6 : 0); /* AFC length */
+            /* flags: PCR present, Random Access on keyframe */
+            *q++ = (write_pcr ? 0x10 : 0) | ((key && is_start) ? 0x40 : 0);
+            if (write_pcr) {
+                // add 11, pcr references the last byte of program clock reference base
+                if (ts->mux_rate > 1)
+                    pcr = get_pcr(ts, s->pb);
+                else
+                    pcr = (dts - delay)*300;
+                if (dts != AV_NOPTS_VALUE && dts < pcr / 300)
+                    av_log(s, AV_LOG_WARNING, "dts < pcr, TS is invalid\n");
+                q = write_pcr_bits(q, pcr);
+            }
         }
         if (is_start) {
             int pes_extension = 0;
@@ -947,14 +950,15 @@ static int mpegts_write_packet(AVFormatContext *s, AVPacket *pkt)
 
     if (st->codec->codec_type != AVMEDIA_TYPE_AUDIO) {
         // 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);
         return 0;
     }
 
     if (ts_st->payload_index + size > DEFAULT_PES_PAYLOAD_SIZE) {
         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,
+                         pkt->flags & AV_PKT_FLAG_KEY);
         ts_st->payload_index = 0;
     }
 
@@ -985,7 +989,7 @@ static int mpegts_write_end(AVFormatContext *s)
         ts_st = st->priv_data;
         if (ts_st->payload_index > 0) {
             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, 0);
         }
         av_freep(&ts_st->adts);
     }
-- 
1.7.4.4

