[vlc-devel] [RFC PATCH 3/7] h264_nal: add h264_get_spspps
    Thomas Guillem 
    thomas at gllm.fr
       
    Thu Apr 23 19:30:51 CEST 2015
    
    
  
It fetches the sps and pps pointers from an AnnexB buffer.
---
 modules/packetizer/h264_nal.c | 74 +++++++++++++++++++++++++++++++++++++++++++
 modules/packetizer/h264_nal.h |  6 ++++
 2 files changed, 80 insertions(+)
diff --git a/modules/packetizer/h264_nal.c b/modules/packetizer/h264_nal.c
index 2a375c2..de3ce6f 100644
--- a/modules/packetizer/h264_nal.c
+++ b/modules/packetizer/h264_nal.c
@@ -22,6 +22,8 @@
 
 #include <limits.h>
 
+static const uint8_t annexb_startcode[] = { 0x00, 0x00, 0x00, 0x01 };
+
 int convert_sps_pps( decoder_t *p_dec, const uint8_t *p_buf,
                      uint32_t i_buf_size, uint8_t *p_out_buf,
                      uint32_t i_out_buf_size, uint32_t *p_sps_pps_size,
@@ -138,6 +140,78 @@ void convert_h264_to_annexb( uint8_t *p_buf, uint32_t i_len,
     }
 }
 
+/* Nal unit order: all units are optional but in that order (cf. 7.4.1.2.3) */
+static int get_nal_order( int i_nal_type )
+{
+    static const int nal_orders[] = {
+        NAL_UNKNOWN, NAL_AU_DELIMITER, NAL_SPS, NAL_PPS
+    };
+
+    for( int i = 0; i < 4; ++i )
+        if( i_nal_type == nal_orders[i] )
+            return i;
+    return -1;
+}
+
+int h264_get_spspps( uint8_t *p_buf, size_t i_buf,
+                     uint8_t **pp_sps, size_t *p_sps_size,
+                     uint8_t **pp_pps, size_t *p_pps_size )
+{
+    int i_last_nal_order = 0;
+    uint8_t *p_sps = NULL, *p_pps = NULL;
+    bool b_is_annexb = false;
+
+    while( i_buf > 5 )
+    {
+        if( memcmp( p_buf, annexb_startcode, 4 ) == 0 )
+        {
+            int i_nal_type = p_buf[4] & 0x1F;
+
+            b_is_annexb = true;
+
+            if( i_nal_type == NAL_AU_DELIMITER ||
+                i_nal_type == NAL_SPS ||
+                i_nal_type == NAL_PPS )
+            {
+                int i_nal_order = get_nal_order( i_nal_type );
+
+                /* nal must be in order */
+                if( i_nal_order < i_last_nal_order )
+                    break;
+                /* pointer to the beginning of the sps/pps  */
+                if( i_nal_type == NAL_SPS && !p_sps )
+                    p_sps = p_buf;
+                if( i_nal_type == NAL_PPS && !p_pps )
+                    p_pps = p_buf;
+                i_last_nal_order = i_nal_order;
+            }
+            else
+            {
+                /* A non AU, SPS, PPS nal unit: stop parsing the annexB.
+                 * Return the pointer to the sps/pps if we found some */
+                if( !p_sps && !p_pps )
+                    break;
+                if( p_sps )
+                    *p_sps_size = (p_pps ? p_pps : p_buf) - p_sps;
+                if( p_pps )
+                    *p_pps_size = p_buf - p_pps;
+                *pp_sps = p_sps;
+                *pp_pps = p_pps;
+                return 0;
+            }
+            /* size of startcode + nal_type */
+            i_buf -= 5;
+            p_buf += 5;
+        } else {
+            i_buf--;
+            p_buf++;
+            if( !b_is_annexb )
+                break;
+        }
+    }
+    return -1;
+}
+
 bool h264_get_profile_level(const es_format_t *p_fmt, size_t *p_profile,
                             size_t *p_level, size_t *p_nal_size)
 {
diff --git a/modules/packetizer/h264_nal.h b/modules/packetizer/h264_nal.h
index 6358fe6..c991ddc 100644
--- a/modules/packetizer/h264_nal.h
+++ b/modules/packetizer/h264_nal.h
@@ -82,6 +82,12 @@ void convert_h264_to_annexb( uint8_t *p_buf, uint32_t i_len,
                              size_t i_nal_size,
                              struct H264ConvertState *state );
 
+/* Get the sps/pps pointers from an AnnexB buffer
+ * Returns 0 if a sps and/or a pps is found */
+int h264_get_spspps( uint8_t *p_buf, size_t i_buf,
+                     uint8_t **pp_sps, size_t *p_sps_size,
+                     uint8_t **pp_pps, size_t *p_pps_size );
+
 /* Get level and Profile */
 bool h264_get_profile_level(const es_format_t *p_fmt, size_t *p_profile,
                             size_t *p_level, size_t *p_nal_size);
-- 
2.1.3
    
    
More information about the vlc-devel
mailing list