⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 h264.c

📁 VLC Player Source Code
💻 C
📖 第 1 页 / 共 3 页
字号:
/***************************************************************************** * h264.c: h264/avc video packetizer ***************************************************************************** * Copyright (C) 2001, 2002, 2006 the VideoLAN team * $Id$ * * Authors: Laurent Aimar <fenrir@via.ecp.fr> *          Eric Petit <titer@videolan.org> *          Gildas Bazin <gbazin@videolan.org> *          Derk-Jan Hartman <hartman at videolan dot org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#ifdef HAVE_CONFIG_H# include "config.h"#endif#include <vlc_common.h>#include <vlc_plugin.h>#include <vlc_sout.h>#include <vlc_codec.h>#include <vlc_block.h>#include "vlc_block_helper.h"#include "vlc_bits.h"/***************************************************************************** * Module descriptor *****************************************************************************/static int  Open ( vlc_object_t * );static void Close( vlc_object_t * );vlc_module_begin();    set_category( CAT_SOUT );    set_subcategory( SUBCAT_SOUT_PACKETIZER );    set_description( N_("H.264 video packetizer") );    set_capability( "packetizer", 50 );    set_callbacks( Open, Close );vlc_module_end();/**************************************************************************** * Local prototypes ****************************************************************************/static block_t *Packetize( decoder_t *, block_t ** );static block_t *PacketizeAVC1( decoder_t *, block_t ** );typedef struct{    int i_nal_type;    int i_nal_ref_idc;    int i_frame_type;    int i_pic_parameter_set_id;    int i_frame_num;    int i_field_pic_flag;    int i_bottom_field_flag;    int i_idr_pic_id;    int i_pic_order_cnt_lsb;    int i_delta_pic_order_cnt_bottom;    int i_delta_pic_order_cnt0;    int i_delta_pic_order_cnt1;} slice_t;#define SPS_MAX (32)#define PPS_MAX (256)struct decoder_sys_t{    block_bytestream_t bytestream;    int     i_state;    size_t  i_offset;    uint8_t startcode[4];    bool    b_slice;    block_t *p_frame;    bool   b_header;    bool   b_sps;    bool   b_pps;    block_t *pp_sps[SPS_MAX];    block_t *pp_pps[PPS_MAX];    /* avcC data */    int i_avcC_length_size;    /* Useful values of the Sequence Parameter Set */    int i_log2_max_frame_num;    int b_frame_mbs_only;    int i_pic_order_cnt_type;    int i_delta_pic_order_always_zero_flag;    int i_log2_max_pic_order_cnt_lsb;    /* Value from Picture Parameter Set */    int i_pic_order_present_flag;    /* Useful values of the Slice Header */    slice_t slice;    /* */    mtime_t i_frame_pts;    mtime_t i_frame_dts;};enum{    STATE_NOSYNC,    STATE_NEXT_SYNC,};enum nal_unit_type_e{    NAL_UNKNOWN = 0,    NAL_SLICE   = 1,    NAL_SLICE_DPA   = 2,    NAL_SLICE_DPB   = 3,    NAL_SLICE_DPC   = 4,    NAL_SLICE_IDR   = 5,    /* ref_idc != 0 */    NAL_SEI         = 6,    /* ref_idc == 0 */    NAL_SPS         = 7,    NAL_PPS         = 8,    NAL_AU_DELIMITER= 9    /* ref_idc == 0 for 6,9,10,11,12 */};enum nal_priority_e{    NAL_PRIORITY_DISPOSABLE = 0,    NAL_PRIORITY_LOW        = 1,    NAL_PRIORITY_HIGH       = 2,    NAL_PRIORITY_HIGHEST    = 3,};static block_t *ParseNALBlock( decoder_t *, bool *pb_used_ts, block_t * );static block_t *CreateAnnexbNAL( decoder_t *, const uint8_t *p, int );static block_t *OutputPicture( decoder_t *p_dec );static void PutSPS( decoder_t *p_dec, block_t *p_frag );static void PutPPS( decoder_t *p_dec, block_t *p_frag );static void ParseSlice( decoder_t *p_dec, bool *pb_new_picture, slice_t *p_slice,                        int i_nal_ref_idc, int i_nal_type, const block_t *p_frag );/***************************************************************************** * Open: probe the packetizer and return score * When opening after demux, the packetizer is only loaded AFTER the decoder * That means that what you set in fmt_out is ignored by the decoder in this special case *****************************************************************************/static int Open( vlc_object_t *p_this ){    decoder_t     *p_dec = (decoder_t*)p_this;    decoder_sys_t *p_sys;    int i;    if( p_dec->fmt_in.i_codec != VLC_FOURCC( 'h', '2', '6', '4') &&        p_dec->fmt_in.i_codec != VLC_FOURCC( 'H', '2', '6', '4') &&        p_dec->fmt_in.i_codec != VLC_FOURCC( 'V', 'S', 'S', 'H') &&        p_dec->fmt_in.i_codec != VLC_FOURCC( 'v', 's', 's', 'h') &&        p_dec->fmt_in.i_codec != VLC_FOURCC( 'D', 'A', 'V', 'C') &&        ( p_dec->fmt_in.i_codec != VLC_FOURCC( 'a', 'v', 'c', '1') ||          p_dec->fmt_in.i_extra < 7 ) )    {        return VLC_EGENERIC;    }    /* Allocate the memory needed to store the decoder's structure */    if( ( p_dec->p_sys = p_sys = malloc( sizeof(decoder_sys_t) ) ) == NULL )    {        return VLC_ENOMEM;    }    p_sys->i_state = STATE_NOSYNC;    p_sys->i_offset = 0;    p_sys->startcode[0] = 0;    p_sys->startcode[1] = 0;    p_sys->startcode[2] = 0;    p_sys->startcode[3] = 1;    p_sys->bytestream = block_BytestreamInit();    p_sys->b_slice = false;    p_sys->p_frame = NULL;    p_sys->b_header= false;    p_sys->b_sps   = false;    p_sys->b_pps   = false;    for( i = 0; i < SPS_MAX; i++ )        p_sys->pp_sps[i] = NULL;    for( i = 0; i < PPS_MAX; i++ )        p_sys->pp_pps[i] = NULL;    p_sys->slice.i_nal_type = -1;    p_sys->slice.i_nal_ref_idc = -1;    p_sys->slice.i_idr_pic_id = -1;    p_sys->slice.i_frame_num = -1;    p_sys->slice.i_frame_type = 0;    p_sys->slice.i_pic_parameter_set_id = -1;    p_sys->slice.i_field_pic_flag = 0;    p_sys->slice.i_bottom_field_flag = -1;    p_sys->slice.i_pic_order_cnt_lsb = -1;    p_sys->slice.i_delta_pic_order_cnt_bottom = -1;    p_sys->i_frame_dts = -1;    p_sys->i_frame_pts = -1;    /* Setup properties */    es_format_Copy( &p_dec->fmt_out, &p_dec->fmt_in );    p_dec->fmt_out.i_codec = VLC_FOURCC( 'h', '2', '6', '4' );    if( p_dec->fmt_in.i_codec == VLC_FOURCC( 'a', 'v', 'c', '1' ) )    {        /* This type of stream is produced by mp4 and matroska         * when we want to store it in another streamformat, you need to convert         * The fmt_in.p_extra should ALWAYS contain the avcC         * The fmt_out.p_extra should contain all the SPS and PPS with 4 byte startcodes */        uint8_t *p = &((uint8_t*)p_dec->fmt_in.p_extra)[4];        int i_sps, i_pps;        bool b_dummy;        int i;        /* Parse avcC */        p_sys->i_avcC_length_size = 1 + ((*p++)&0x03);        /* Read SPS */        i_sps = (*p++)&0x1f;        for( i = 0; i < i_sps; i++ )        {            uint16_t i_length = GetWBE( p ); p += 2;            if( i_length >                (uint8_t*)p_dec->fmt_in.p_extra + p_dec->fmt_in.i_extra - p )            {                return VLC_EGENERIC;            }            block_t *p_sps = CreateAnnexbNAL( p_dec, p, i_length );            if( !p_sps )                return VLC_EGENERIC;            ParseNALBlock( p_dec, &b_dummy, p_sps );            p += i_length;        }        /* Read PPS */        i_pps = *p++;        for( i = 0; i < i_pps; i++ )        {            uint16_t i_length = GetWBE( p ); p += 2;            if( i_length >                (uint8_t*)p_dec->fmt_in.p_extra + p_dec->fmt_in.i_extra - p )            {                return VLC_EGENERIC;            }            block_t *p_pps = CreateAnnexbNAL( p_dec, p, i_length );            if( !p_pps )                return VLC_EGENERIC;            ParseNALBlock( p_dec, &b_dummy, p_pps );            p += i_length;        }        msg_Dbg( p_dec, "avcC length size=%d, sps=%d, pps=%d",                 p_sys->i_avcC_length_size, i_sps, i_pps );        if( !p_sys->b_sps || !p_sys->b_pps )            return VLC_EGENERIC;        /* FIXME: FFMPEG isn't happy at all if you leave this */        if( p_dec->fmt_out.i_extra > 0 )            free( p_dec->fmt_out.p_extra );        p_dec->fmt_out.i_extra = 0;        p_dec->fmt_out.p_extra = NULL;        /* Set the new extradata */        for( i = 0; i < SPS_MAX; i++ )        {            if( p_sys->pp_sps[i] )                p_dec->fmt_out.i_extra += p_sys->pp_sps[i]->i_buffer;        }        for( i = 0; i < PPS_MAX; i++ )        {            if( p_sys->pp_pps[i] )                p_dec->fmt_out.i_extra += p_sys->pp_pps[i]->i_buffer;        }        p_dec->fmt_out.p_extra = malloc( p_dec->fmt_out.i_extra );        if( p_dec->fmt_out.p_extra )        {            uint8_t *p_dst = p_dec->fmt_out.p_extra;            for( i = 0; i < SPS_MAX; i++ )            {                if( p_sys->pp_sps[i] )                {                    memcpy( p_dst, p_sys->pp_sps[i]->p_buffer, p_sys->pp_sps[i]->i_buffer );                    p_dst += p_sys->pp_sps[i]->i_buffer;                }            }            for( i = 0; i < PPS_MAX; i++ )            {                if( p_sys->pp_pps[i] )                {                    memcpy( p_dst, p_sys->pp_pps[i]->p_buffer, p_sys->pp_pps[i]->i_buffer );                    p_dst += p_sys->pp_pps[i]->i_buffer;                }            }            p_sys->b_header = true;        }        else        {            p_dec->fmt_out.i_extra = 0;        }        /* Set callback */        p_dec->pf_packetize = PacketizeAVC1;    }    else    {        /* This type of stream contains data with 3 of 4 byte startcodes         * The fmt_in.p_extra MAY contain SPS/PPS with 4 byte startcodes         * The fmt_out.p_extra should be the same */         /* Set callback */        p_dec->pf_packetize = Packetize;        /* */        if( p_dec->fmt_in.i_extra > 0 )        {            block_t *p_init = block_New( p_dec, p_dec->fmt_in.i_extra );            block_t *p_pic;            memcpy( p_init->p_buffer, p_dec->fmt_in.p_extra,                    p_dec->fmt_in.i_extra );            while( ( p_pic = Packetize( p_dec, &p_init ) ) )            {                /* Should not occur because we should only receive SPS/PPS */                block_Release( p_pic );            }        }

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -