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

📄 speex.c

📁 video linux conference
💻 C
📖 第 1 页 / 共 2 页
字号:
/***************************************************************************** * speex.c: speex decoder/packetizer/encoder module making use of libspeex. ***************************************************************************** * Copyright (C) 2003 VideoLAN * $Id: speex.c 10101 2005-03-02 16:47:31Z robux4 $ * * Authors: Gildas Bazin <gbazin@videolan.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., 59 Temple Place - Suite 330, Boston, MA  02111, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#include <vlc/vlc.h>#include <vlc/decoder.h>#include <vlc/input.h>#include <ogg/ogg.h>#include <speex/speex.h>#include <speex/speex_header.h>#include <speex/speex_stereo.h>#include <speex/speex_callbacks.h>/***************************************************************************** * decoder_sys_t : speex decoder descriptor *****************************************************************************/struct decoder_sys_t{    /* Module mode */    vlc_bool_t b_packetizer;    /*     * Input properties     */    int i_headers;    int i_frame_in_packet;    /*     * Speex properties     */    SpeexBits bits;    SpeexHeader *p_header;    SpeexStereoState stereo;    void *p_state;    /*     * Common properties     */    audio_date_t end_date;};static int pi_channels_maps[6] ={    0,    AOUT_CHAN_CENTER,   AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT,    AOUT_CHAN_CENTER | AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT,    AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_REARLEFT     | AOUT_CHAN_REARRIGHT,    AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_CENTER     | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT};/**************************************************************************** * Local prototypes ****************************************************************************/static int  OpenDecoder   ( vlc_object_t * );static int  OpenPacketizer( vlc_object_t * );static void CloseDecoder  ( vlc_object_t * );static void *DecodeBlock  ( decoder_t *, block_t ** );static int  ProcessHeaders( decoder_t * );static int  ProcessInitialHeader ( decoder_t *, ogg_packet * );static void *ProcessPacket( decoder_t *, ogg_packet *, block_t ** );static aout_buffer_t *DecodePacket( decoder_t *, ogg_packet * );static block_t *SendPacket( decoder_t *, ogg_packet *, block_t * );static void ParseSpeexComments( decoder_t *, ogg_packet * );static int OpenEncoder   ( vlc_object_t * );static void CloseEncoder ( vlc_object_t * );static block_t *Encode   ( encoder_t *, aout_buffer_t * );/***************************************************************************** * Module descriptor *****************************************************************************/vlc_module_begin();    set_category( CAT_INPUT );    set_subcategory( SUBCAT_INPUT_ACODEC );    set_description( _("Speex audio decoder") );    set_capability( "decoder", 100 );    set_callbacks( OpenDecoder, CloseDecoder );    add_submodule();    set_description( _("Speex audio packetizer") );    set_capability( "packetizer", 100 );    set_callbacks( OpenPacketizer, CloseDecoder );    add_submodule();    set_description( _("Speex audio encoder") );    set_capability( "encoder", 100 );    set_callbacks( OpenEncoder, CloseEncoder );vlc_module_end();/***************************************************************************** * OpenDecoder: probe the decoder and return score *****************************************************************************/static int OpenDecoder( vlc_object_t *p_this ){    decoder_t *p_dec = (decoder_t*)p_this;    decoder_sys_t *p_sys = p_dec->p_sys;    if( p_dec->fmt_in.i_codec != VLC_FOURCC('s','p','x',' ') )    {        return VLC_EGENERIC;    }    /* Allocate the memory needed to store the decoder's structure */    if( ( p_dec->p_sys = p_sys =          (decoder_sys_t *)malloc(sizeof(decoder_sys_t)) ) == NULL )    {        msg_Err( p_dec, "out of memory" );        return VLC_EGENERIC;    }    p_dec->p_sys->b_packetizer = VLC_FALSE;    aout_DateSet( &p_sys->end_date, 0 );    /* Set output properties */    p_dec->fmt_out.i_cat = AUDIO_ES;    p_dec->fmt_out.i_codec = AOUT_FMT_S16_NE;    /* Set callbacks */    p_dec->pf_decode_audio = (aout_buffer_t *(*)(decoder_t *, block_t **))        DecodeBlock;    p_dec->pf_packetize    = (block_t *(*)(decoder_t *, block_t **))        DecodeBlock;    p_sys->i_headers = 0;    p_sys->p_state = NULL;    p_sys->p_header = NULL;    p_sys->i_frame_in_packet = 0;    return VLC_SUCCESS;}static int OpenPacketizer( vlc_object_t *p_this ){    decoder_t *p_dec = (decoder_t*)p_this;    int i_ret = OpenDecoder( p_this );    if( i_ret == VLC_SUCCESS )    {        p_dec->p_sys->b_packetizer = VLC_TRUE;        p_dec->fmt_out.i_codec = VLC_FOURCC('s','p','x',' ');    }    return i_ret;}/**************************************************************************** * DecodeBlock: the whole thing **************************************************************************** * This function must be fed with ogg packets. ****************************************************************************/static void *DecodeBlock( decoder_t *p_dec, block_t **pp_block ){    decoder_sys_t *p_sys = p_dec->p_sys;    ogg_packet oggpacket;    if( !pp_block ) return NULL;    if( *pp_block )    {        /* Block to Ogg packet */        oggpacket.packet = (*pp_block)->p_buffer;        oggpacket.bytes = (*pp_block)->i_buffer;    }    else    {        if( p_sys->b_packetizer ) return NULL;        /* Block to Ogg packet */        oggpacket.packet = NULL;        oggpacket.bytes = 0;    }    oggpacket.granulepos = -1;    oggpacket.b_o_s = 0;    oggpacket.e_o_s = 0;    oggpacket.packetno = 0;    /* Check for headers */    if( p_sys->i_headers == 0 && p_dec->fmt_in.i_extra )    {        /* Headers already available as extra data */        p_sys->i_headers = 2;    }    else if( oggpacket.bytes && p_sys->i_headers < 2 )    {        /* Backup headers as extra data */        uint8_t *p_extra;        p_dec->fmt_in.p_extra =            realloc( p_dec->fmt_in.p_extra, p_dec->fmt_in.i_extra +                     oggpacket.bytes + 2 );        p_extra = p_dec->fmt_in.p_extra + p_dec->fmt_in.i_extra;        *(p_extra++) = oggpacket.bytes >> 8;        *(p_extra++) = oggpacket.bytes & 0xFF;        memcpy( p_extra, oggpacket.packet, oggpacket.bytes );        p_dec->fmt_in.i_extra += oggpacket.bytes + 2;        block_Release( *pp_block );        p_sys->i_headers++;        return NULL;    }    if( p_sys->i_headers == 2 )    {        if( ProcessHeaders( p_dec ) != VLC_SUCCESS )        {            p_sys->i_headers = 0;            p_dec->fmt_in.i_extra = 0;            block_Release( *pp_block );            return NULL;        }        else p_sys->i_headers++;    }    return ProcessPacket( p_dec, &oggpacket, pp_block );}/***************************************************************************** * ProcessHeaders: process Speex headers. *****************************************************************************/static int ProcessHeaders( decoder_t *p_dec ){    decoder_sys_t *p_sys = p_dec->p_sys;    ogg_packet oggpacket;    uint8_t *p_extra;    int i_extra;    if( !p_dec->fmt_in.i_extra ) return VLC_EGENERIC;    oggpacket.granulepos = -1;    oggpacket.b_o_s = 1; /* yes this actually is a b_o_s packet :) */    oggpacket.e_o_s = 0;    oggpacket.packetno = 0;    p_extra = p_dec->fmt_in.p_extra;    i_extra = p_dec->fmt_in.i_extra;    /* Take care of the initial Vorbis header */    oggpacket.bytes = *(p_extra++) << 8;    oggpacket.bytes |= (*(p_extra++) & 0xFF);    oggpacket.packet = p_extra;    p_extra += oggpacket.bytes;    i_extra -= (oggpacket.bytes + 2);    if( i_extra < 0 )    {        msg_Err( p_dec, "header data corrupted");        return VLC_EGENERIC;    }    /* Take care of the initial Speex header */    if( ProcessInitialHeader( p_dec, &oggpacket ) != VLC_SUCCESS )    {        msg_Err( p_dec, "initial Speex header is corrupted" );        return VLC_EGENERIC;    }    /* The next packet in order is the comments header */    oggpacket.b_o_s = 0;    oggpacket.bytes = *(p_extra++) << 8;    oggpacket.bytes |= (*(p_extra++) & 0xFF);    oggpacket.packet = p_extra;    p_extra += oggpacket.bytes;    i_extra -= (oggpacket.bytes + 2);    if( i_extra < 0 )    {        msg_Err( p_dec, "header data corrupted");        return VLC_EGENERIC;    }    ParseSpeexComments( p_dec, &oggpacket );    if( p_sys->b_packetizer )    {        p_dec->fmt_out.i_extra = p_dec->fmt_in.i_extra;        p_dec->fmt_out.p_extra =            realloc( p_dec->fmt_out.p_extra, p_dec->fmt_out.i_extra );        memcpy( p_dec->fmt_out.p_extra,                p_dec->fmt_in.p_extra, p_dec->fmt_out.i_extra );    }    return VLC_SUCCESS;}/***************************************************************************** * ProcessInitialHeader: processes the inital Speex header packet. *****************************************************************************/static int ProcessInitialHeader( decoder_t *p_dec, ogg_packet *p_oggpacket ){    decoder_sys_t *p_sys = p_dec->p_sys;    void *p_state;    SpeexHeader *p_header;    const SpeexMode *p_mode;    SpeexCallback callback;    p_sys->p_header = p_header =        speex_packet_to_header( p_oggpacket->packet, p_oggpacket->bytes );    if( !p_header )    {        msg_Err( p_dec, "cannot read Speex header" );        return VLC_EGENERIC;    }    if( p_header->mode >= SPEEX_NB_MODES )    {        msg_Err( p_dec, "mode number %d does not (yet/any longer) exist in "                 "this version of libspeex.", p_header->mode );        return VLC_EGENERIC;    }    p_mode = speex_mode_list[p_header->mode];    if( p_header->speex_version_id > 1 )    {        msg_Err( p_dec, "this file was encoded with Speex bit-stream "                 "version %d, which I don't know how to decode.",                 p_header->speex_version_id );        return VLC_EGENERIC;    }    if( p_mode->bitstream_version < p_header->mode_bitstream_version )    {        msg_Err( p_dec, "file encoded with a newer version of Speex." );        return VLC_EGENERIC;    }    if( p_mode->bitstream_version > p_header->mode_bitstream_version )    {        msg_Err( p_dec, "file encoded with an older version of Speex." );        return VLC_EGENERIC;    }    msg_Dbg( p_dec, "Speex %d Hz audio using %s mode %s%s",             p_header->rate, p_mode->modeName,             ( p_header->nb_channels == 1 ) ? " (mono" : " (stereo",             p_header->vbr ? ", VBR)" : ")" );    /* Take care of speex decoder init */    speex_bits_init( &p_sys->bits );    p_sys->p_state = p_state = speex_decoder_init( p_mode );    if( !p_state )    {        msg_Err( p_dec, "decoder initialization failed" );        return VLC_EGENERIC;    }    if( p_header->nb_channels == 2 )    {        SpeexStereoState stereo = SPEEX_STEREO_STATE_INIT;        p_sys->stereo = stereo;        callback.callback_id = SPEEX_INBAND_STEREO;        callback.func = speex_std_stereo_request_handler;        callback.data = &p_sys->stereo;        speex_decoder_ctl( p_state, SPEEX_SET_HANDLER, &callback );    }    /* Setup the format */    p_dec->fmt_out.audio.i_physical_channels =        p_dec->fmt_out.audio.i_original_channels =            pi_channels_maps[p_header->nb_channels];    p_dec->fmt_out.audio.i_channels = p_header->nb_channels;    p_dec->fmt_out.audio.i_rate = p_header->rate;    aout_DateInit( &p_sys->end_date, p_header->rate );    return VLC_SUCCESS;}/***************************************************************************** * ProcessPacket: processes a Speex packet. *****************************************************************************/static void *ProcessPacket( decoder_t *p_dec, ogg_packet *p_oggpacket,                            block_t **pp_block ){    decoder_sys_t *p_sys = p_dec->p_sys;    block_t *p_block = *pp_block;    /* Date management */    if( p_block && p_block->i_pts > 0 &&        p_block->i_pts != aout_DateGet( &p_sys->end_date ) )

⌨️ 快捷键说明

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