📄 vorbis.c
字号:
/***************************************************************************** * vorbis.c: vorbis decoder/encoder/packetizer module making use of libvorbis. ***************************************************************************** * Copyright (C) 2001-2003 VideoLAN * $Id: vorbis.c 10666 2005-04-12 22:47:36Z fkuehne $ * * 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 <vlc/sout.h>#include <ogg/ogg.h>#ifdef MODULE_NAME_IS_tremor#include <tremor/ivorbiscodec.h>#else#include <vorbis/codec.h>/* vorbis header */#ifdef HAVE_VORBIS_VORBISENC_H# include <vorbis/vorbisenc.h># ifndef OV_ECTL_RATEMANAGE_AVG# define OV_ECTL_RATEMANAGE_AVG 0x0# endif#endif#endif/***************************************************************************** * decoder_sys_t : vorbis decoder descriptor *****************************************************************************/struct decoder_sys_t{ /* Module mode */ vlc_bool_t b_packetizer; /* * Input properties */ int i_headers; /* * Vorbis properties */ vorbis_info vi; /* struct that stores all the static vorbis bitstream settings */ vorbis_comment vc; /* struct that stores all the bitstream user * comments */ vorbis_dsp_state vd; /* central working state for the packet->PCM * decoder */ vorbis_block vb; /* local working space for packet->PCM decode */ /* * Common properties */ audio_date_t end_date; int i_last_block_size;};static int pi_channels_maps[7] ={ 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, AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_CENTER | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT | AOUT_CHAN_LFE};/**************************************************************************** * 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 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 ParseVorbisComments( decoder_t * );#ifdef MODULE_NAME_IS_tremorstatic void Interleave ( int32_t *, const int32_t **, int, int );#elsestatic void Interleave ( float *, const float **, int, int );#endif#ifndef MODULE_NAME_IS_tremorstatic int OpenEncoder ( vlc_object_t * );static void CloseEncoder ( vlc_object_t * );static block_t *Encode ( encoder_t *, aout_buffer_t * );#endif/***************************************************************************** * Module descriptor *****************************************************************************/#define ENC_QUALITY_TEXT N_("Encoding quality")#define ENC_QUALITY_LONGTEXT N_( \ "Allows you to specify a quality between 1 (low) and 10 (high), instead " \ "of specifying a particular bitrate. This will produce a VBR stream." )#define ENC_MAXBR_TEXT N_("Maximum encoding bitrate")#define ENC_MAXBR_LONGTEXT N_( \ "Allows you to specify a maximum bitrate in kbps. " \ "Useful for streaming applications." )#define ENC_MINBR_TEXT N_("Minimum encoding bitrate")#define ENC_MINBR_LONGTEXT N_( \ "Allows you to specify a minimum bitrate in kbps. " \ "Useful for encoding for a fixed-size channel." )#define ENC_CBR_TEXT N_("CBR encoding")#define ENC_CBR_LONGTEXT N_( \ "Allows you to force a constant bitrate encoding (CBR)." )vlc_module_begin(); set_shortname( "Vorbis" ); set_description( _("Vorbis audio decoder") );#ifdef MODULE_NAME_IS_tremor set_capability( "decoder", 90 );#else set_capability( "decoder", 100 );#endif set_category( CAT_INPUT ); set_subcategory( SUBCAT_INPUT_ACODEC ); set_callbacks( OpenDecoder, CloseDecoder ); add_submodule(); set_description( _("Vorbis audio packetizer") ); set_capability( "packetizer", 100 ); set_callbacks( OpenPacketizer, CloseDecoder );#ifndef MODULE_NAME_IS_tremor# define ENC_CFG_PREFIX "sout-vorbis-" add_submodule(); set_description( _("Vorbis audio encoder") ); set_capability( "encoder", 100 );#if defined(HAVE_VORBIS_VORBISENC_H) set_callbacks( OpenEncoder, CloseEncoder );#endif add_integer( ENC_CFG_PREFIX "quality", 3, NULL, ENC_QUALITY_TEXT, ENC_QUALITY_LONGTEXT, VLC_FALSE ); add_integer( ENC_CFG_PREFIX "max-bitrate", 0, NULL, ENC_MAXBR_TEXT, ENC_MAXBR_LONGTEXT, VLC_FALSE ); add_integer( ENC_CFG_PREFIX "min-bitrate", 0, NULL, ENC_MINBR_TEXT, ENC_MINBR_LONGTEXT, VLC_FALSE ); add_bool( ENC_CFG_PREFIX "cbr", 0, NULL, ENC_CBR_TEXT, ENC_CBR_LONGTEXT, VLC_FALSE );#endifvlc_module_end();#ifndef MODULE_NAME_IS_tremorstatic const char *ppsz_enc_options[] = { "quality", "max-bitrate", "min-bitrate", "cbr", NULL};#endif/***************************************************************************** * 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; if( p_dec->fmt_in.i_codec != VLC_FOURCC('v','o','r','b') ) { 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; } /* Misc init */ aout_DateSet( &p_sys->end_date, 0 ); p_sys->i_last_block_size = 0; p_sys->b_packetizer = VLC_FALSE; p_sys->i_headers = 0; /* Take care of vorbis init */ vorbis_info_init( &p_sys->vi ); vorbis_comment_init( &p_sys->vc ); /* Set output properties */ p_dec->fmt_out.i_cat = AUDIO_ES;#ifdef MODULE_NAME_IS_tremor p_dec->fmt_out.i_codec = VLC_FOURCC('f','i','3','2');#else p_dec->fmt_out.i_codec = VLC_FOURCC('f','l','3','2');#endif /* 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; 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('v','o','r','b'); } 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 = 3; } else if( oggpacket.bytes && p_sys->i_headers < 3 ) { /* 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 = (uint8_t *)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 == 3 ) { 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 Vorbis 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; } if( vorbis_synthesis_headerin( &p_sys->vi, &p_sys->vc, &oggpacket ) < 0 ) { msg_Err( p_dec, "this bitstream does not contain Vorbis audio data"); return VLC_EGENERIC; } /* Setup the format */ p_dec->fmt_out.audio.i_rate = p_sys->vi.rate; p_dec->fmt_out.audio.i_channels = p_sys->vi.channels; p_dec->fmt_out.audio.i_physical_channels = p_dec->fmt_out.audio.i_original_channels = pi_channels_maps[p_sys->vi.channels]; p_dec->fmt_out.i_bitrate = p_sys->vi.bitrate_nominal; aout_DateInit( &p_sys->end_date, p_sys->vi.rate ); aout_DateSet( &p_sys->end_date, 0 ); msg_Dbg( p_dec, "channels:%d samplerate:%ld bitrate:%ld", p_sys->vi.channels, p_sys->vi.rate, p_sys->vi.bitrate_nominal ); /* 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; } if( vorbis_synthesis_headerin( &p_sys->vi, &p_sys->vc, &oggpacket ) < 0 ) { msg_Err( p_dec, "2nd Vorbis header is corrupted" ); return VLC_EGENERIC; } ParseVorbisComments( p_dec ); /* The next packet in order is the codebooks header * We need to watch out that this packet is not missing as a * missing or corrupted header is fatal. */ oggpacket.bytes = *(p_extra++) << 8; oggpacket.bytes |= (*(p_extra++) & 0xFF); oggpacket.packet = p_extra; i_extra -= (oggpacket.bytes + 2); if( i_extra < 0 ) { msg_Err( p_dec, "header data corrupted"); return VLC_EGENERIC; } if( vorbis_synthesis_headerin( &p_sys->vi, &p_sys->vc, &oggpacket ) < 0 ) { msg_Err( p_dec, "3rd Vorbis header is corrupted" ); return VLC_EGENERIC; } if( !p_sys->b_packetizer ) { /* Initialize the Vorbis packet->PCM decoder */ vorbis_synthesis_init( &p_sys->vd, &p_sys->vi ); vorbis_block_init( &p_sys->vd, &p_sys->vb ); } else { 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;}/***************************************************************************** * ProcessPacket: processes a Vorbis 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 ) ) { aout_DateSet( &p_sys->end_date, p_block->i_pts ); }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -