📄 vorbis.c
字号:
/***************************************************************************** * vorbis.c: vorbis decoder/encoder/packetizer module using of libvorbis. ***************************************************************************** * Copyright (C) 2001-2003 the VideoLAN team * Copyright (C) 2007 Société des arts technologiques * Copyright (C) 2007 Savoir-faire Linux * * $Id$ * * 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., 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_codec.h>#include <vlc_aout.h>#include <vlc_input.h>#include <vlc_playlist.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 */ bool 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; int i_input_rate; /* ** Channel reordering */ int pi_chan_table[AOUT_CHAN_MAX];};static const int pi_channels_maps[9] ={ 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, AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_CENTER | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT | AOUT_CHAN_MIDDLELEFT | AOUT_CHAN_MIDDLERIGHT, AOUT_CHAN_LEFT | AOUT_CHAN_RIGHT | AOUT_CHAN_CENTER | AOUT_CHAN_REARLEFT | AOUT_CHAN_REARRIGHT | AOUT_CHAN_MIDDLELEFT | AOUT_CHAN_MIDDLERIGHT | AOUT_CHAN_LFE};/*** channel order as defined in http://www.ogghelp.com/ogg/glossary.cfm#Audio_Channels*//* recommended vorbis channel order for 6 channels */static const uint32_t pi_6channels_in[] ={ AOUT_CHAN_LEFT, AOUT_CHAN_CENTER, AOUT_CHAN_RIGHT, AOUT_CHAN_REARLEFT, AOUT_CHAN_REARRIGHT,AOUT_CHAN_LFE,0 };/* recommended vorbis channel order for 4 channels */static const uint32_t pi_4channels_in[] ={ AOUT_CHAN_LEFT, AOUT_CHAN_RIGHT, AOUT_CHAN_CENTER, AOUT_CHAN_LFE, 0 };/* recommended vorbis channel order for 3 channels */static const uint32_t pi_3channels_in[] ={ AOUT_CHAN_LEFT, AOUT_CHAN_CENTER, AOUT_CHAN_RIGHT, 0 };/**************************************************************************** * 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 * );static void ConfigureChannelOrder(int *, int, uint32_t, bool );#ifdef MODULE_NAME_IS_tremorstatic void Interleave ( int32_t *, const int32_t **, int, int, int * );#elsestatic void Interleave ( float *, const float **, int, 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_( \ "Enforce 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_( \ "Maximum bitrate in kbps. This is useful for streaming applications." )#define ENC_MINBR_TEXT N_("Minimum encoding bitrate")#define ENC_MINBR_LONGTEXT N_( \ "Minimum bitrate in kbps. This is useful for encoding for a fixed-size channel." )#define ENC_CBR_TEXT N_("CBR encoding")#define ENC_CBR_LONGTEXT N_( \ "Force a constant bitrate encoding (CBR)." )vlc_module_begin(); set_shortname( "Vorbis" ); set_description( N_("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( N_("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( N_("Vorbis audio encoder") ); set_capability( "encoder", 100 );#if defined(HAVE_VORBIS_VORBISENC_H) set_callbacks( OpenEncoder, CloseEncoder );#endif add_integer( ENC_CFG_PREFIX "quality", 0, NULL, ENC_QUALITY_TEXT, ENC_QUALITY_LONGTEXT, false ); add_integer( ENC_CFG_PREFIX "max-bitrate", 0, NULL, ENC_MAXBR_TEXT, ENC_MAXBR_LONGTEXT, false ); add_integer( ENC_CFG_PREFIX "min-bitrate", 0, NULL, ENC_MINBR_TEXT, ENC_MINBR_LONGTEXT, false ); add_bool( ENC_CFG_PREFIX "cbr", 0, NULL, ENC_CBR_TEXT, ENC_CBR_LONGTEXT, false );#endifvlc_module_end();#ifndef MODULE_NAME_IS_tremorstatic const char *const 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 ) return VLC_ENOMEM; /* Misc init */ aout_DateSet( &p_sys->end_date, 0 ); p_sys->i_last_block_size = 0; p_sys->b_packetizer = false; p_sys->i_headers = 0; p_sys->i_input_rate = INPUT_RATE_DEFAULT; /* 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 = 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; if( (*pp_block)->i_rate > 0 ) p_sys->i_input_rate = (*pp_block)->i_rate; } 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 */ msg_Dbg( p_dec, "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++;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -