[FFmpeg-cvslog] rtpenc_h264: Aggregate multiple NAL units into one RTP packet, if possible

Martin Storsjö git at videolan.org
Mon Feb 23 00:30:01 CET 2015


ffmpeg | branch: master | Martin Storsjö <martin at martin.st> | Fri Feb 20 21:21:27 2015 +0200| [e2220e734f3d01145ef9aefbd7b6ff29a89ae159] | committer: Martin Storsjö

rtpenc_h264: Aggregate multiple NAL units into one RTP packet, if possible

Signed-off-by: Martin Storsjö <martin at martin.st>

> http://git.videolan.org/gitweb.cgi/ffmpeg.git/?a=commit;h=e2220e734f3d01145ef9aefbd7b6ff29a89ae159
---

 libavformat/rtpenc.h      |    1 +
 libavformat/rtpenc_h264.c |   44 +++++++++++++++++++++++++++++++++++++++++++-
 2 files changed, 44 insertions(+), 1 deletion(-)

diff --git a/libavformat/rtpenc.h b/libavformat/rtpenc.h
index f2ac0fa..f492594 100644
--- a/libavformat/rtpenc.h
+++ b/libavformat/rtpenc.h
@@ -56,6 +56,7 @@ struct RTPMuxContext {
      * (1, 2 or 4)
      */
     int nal_length_size;
+    int buffered_nals;
 
     int flags;
 
diff --git a/libavformat/rtpenc_h264.c b/libavformat/rtpenc_h264.c
index 395c3a4..baf8b8a 100644
--- a/libavformat/rtpenc_h264.c
+++ b/libavformat/rtpenc_h264.c
@@ -25,21 +25,61 @@
  * @author Luca Abeni <lucabe72 at email.it>
  */
 
+#include "libavutil/intreadwrite.h"
+
 #include "avformat.h"
 #include "avc.h"
 #include "rtpenc.h"
 
+static void flush_buffered(AVFormatContext *s1, int last)
+{
+    RTPMuxContext *s = s1->priv_data;
+    if (s->buf_ptr != s->buf) {
+        // If we're only sending one single NAL unit, send it as such, skip
+        // the STAP-A framing
+        if (s->buffered_nals == 1)
+            ff_rtp_send_data(s1, s->buf + 3, s->buf_ptr - s->buf - 3, last);
+        else
+            ff_rtp_send_data(s1, s->buf, s->buf_ptr - s->buf, last);
+    }
+    s->buf_ptr = s->buf;
+    s->buffered_nals = 0;
+}
+
 static void nal_send(AVFormatContext *s1, const uint8_t *buf, int size, int last)
 {
     RTPMuxContext *s = s1->priv_data;
 
     av_log(s1, AV_LOG_DEBUG, "Sending NAL %x of len %d M=%d\n", buf[0] & 0x1F, size, last);
     if (size <= s->max_payload_size) {
-        ff_rtp_send_data(s1, buf, size, last);
+        int buffered_size = s->buf_ptr - s->buf;
+        // Flush buffered NAL units if the current unit doesn't fit
+        if (buffered_size + 2 + size > s->max_payload_size) {
+            flush_buffered(s1, 0);
+            buffered_size = 0;
+        }
+        // If we aren't using mode 0, and the NAL unit fits including the
+        // framing (2 bytes length, plus 1 byte for the STAP-A marker),
+        // write the unit to the buffer as a STAP-A packet, otherwise flush
+        // and send as single NAL.
+        if (buffered_size + 3 + size <= s->max_payload_size &&
+            !(s->flags & FF_RTP_FLAG_H264_MODE0)) {
+            if (buffered_size == 0)
+                *s->buf_ptr++ = 24;
+            AV_WB16(s->buf_ptr, size);
+            s->buf_ptr += 2;
+            memcpy(s->buf_ptr, buf, size);
+            s->buf_ptr += size;
+            s->buffered_nals++;
+        } else {
+            flush_buffered(s1, 0);
+            ff_rtp_send_data(s1, buf, size, last);
+        }
     } else {
         uint8_t type = buf[0] & 0x1F;
         uint8_t nri = buf[0] & 0x60;
 
+        flush_buffered(s1, 0);
         if (s->flags & FF_RTP_FLAG_H264_MODE0) {
             av_log(s1, AV_LOG_ERROR,
                    "NAL size %d > %d, try -slice-max-size %d\n", size,
@@ -72,6 +112,7 @@ void ff_rtp_send_h264(AVFormatContext *s1, const uint8_t *buf1, int size)
     RTPMuxContext *s = s1->priv_data;
 
     s->timestamp = s->cur_timestamp;
+    s->buf_ptr   = s->buf;
     if (s->nal_length_size)
         r = ff_avc_mp4_find_startcode(buf1, end, s->nal_length_size) ? buf1 : end;
     else
@@ -91,4 +132,5 @@ void ff_rtp_send_h264(AVFormatContext *s1, const uint8_t *buf1, int size)
         nal_send(s1, r, r1 - r, r1 == end);
         r = r1;
     }
+    flush_buffered(s1, 1);
 }



More information about the ffmpeg-cvslog mailing list