/*****************************************************************************
* ac3_adec.c: ac3 decoder module main file
*****************************************************************************
- * Copyright (C) 1999, 2000 VideoLAN
- * $Id: ac3_adec.c,v 1.1 2001/11/13 12:09:17 henri Exp $
+ * Copyright (C) 1999-2001 VideoLAN
+ * $Id: ac3_adec.c,v 1.35 2002/07/31 20:56:50 sam Exp $
*
* Authors: Michel Lespinasse <walken@zoy.org>
*
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
*****************************************************************************/
-/*
- * TODO :
- *
- * - vérifier l'état de la fifo de sortie avant d'y stocker les samples
- * décodés ;
- * - vlc_cond_signal() / vlc_cond_wait()
- *
- */
-#define MODULE_NAME ac3_adec
-#include "modules_inner.h"
-
/*****************************************************************************
* Preamble
*****************************************************************************/
-#include "defs.h"
-
-#ifdef HAVE_UNISTD_H
-#include <unistd.h> /* getpid() */
-#endif
-
#include <stdlib.h> /* malloc(), free() */
#include <string.h> /* memset() */
-#include "config.h"
-#include "common.h"
-#include "threads.h"
-#include "mtime.h"
-#include "intf_msg.h" /* intf_DbgMsg(), intf_ErrMsg() */
-
-#include "audio_output.h"
-
-#include "modules.h"
-#include "modules_export.h"
+#include <vlc/vlc.h>
+#include <vlc/aout.h>
+#include <vlc/decoder.h>
-#include "stream_control.h"
-#include "input_ext-dec.h"
-#include "input_ext-intf.h" /* MPEG?_AUDIO_ES */
+#ifdef HAVE_UNISTD_H
+# include <unistd.h> /* getpid() */
+#endif
#include "ac3_imdct.h"
#include "ac3_downmix.h"
-#include "ac3_decoder.h"
#include "ac3_adec.h"
#define AC3DEC_FRAME_SIZE (2*1536)
/*****************************************************************************
* Local prototypes
*****************************************************************************/
-static int ac3_adec_Probe ( probedata_t * );
-static int ac3_adec_Run ( decoder_config_t * );
-static int ac3_adec_Init (ac3dec_thread_t * p_adec);
-static void ac3_adec_ErrorThread (ac3dec_thread_t * p_adec);
-static void ac3_adec_EndThread (ac3dec_thread_t * p_adec);
-static void BitstreamCallback ( bit_stream_t *p_bit_stream,
- boolean_t b_new_pes );
+static int OpenDecoder ( vlc_object_t * );
+static int RunDecoder ( decoder_fifo_t * );
+static int InitThread ( ac3dec_t * p_adec );
+static void EndThread ( ac3dec_t * p_adec );
+static void BitstreamCallback ( bit_stream_t *p_bit_stream,
+ vlc_bool_t b_new_pes );
/*****************************************************************************
- * Capabilities
+ * Module descriptor
*****************************************************************************/
-void _M( adec_getfunctions )( function_list_t * p_function_list )
-{
- p_function_list->pf_probe = ac3_adec_Probe;
- p_function_list->functions.dec.pf_RunThread = ac3_adec_Run;
-}
-
-/*****************************************************************************
- * Build configuration tree.
- *****************************************************************************/
-MODULE_CONFIG_START
-ADD_WINDOW( "Configuration for ac3 decoder module" )
- ADD_COMMENT( "Nothing to configure" )
-MODULE_CONFIG_STOP
-
-MODULE_INIT_START
- p_module->i_capabilities = MODULE_CAPABILITY_DEC;
- p_module->psz_longname = "Ac3 sofware decoder";
-MODULE_INIT_STOP
-
-MODULE_ACTIVATE_START
- _M( adec_getfunctions )( &p_module->p_functions->dec );
-MODULE_ACTIVATE_STOP
-
-MODULE_DEACTIVATE_START
-MODULE_DEACTIVATE_STOP
-
+vlc_module_begin();
+ add_category_hint( N_("Miscellaneous"), NULL );
+ add_module ( "ac3-downmix", "downmix", NULL, NULL,
+ N_("AC3 downmix module"), NULL );
+ add_module ( "ac3-imdct", "imdct", NULL, NULL,
+ N_("AC3 IMDCT module"), NULL );
+ set_description( _("software AC3 decoder") );
+ set_capability( "decoder", 50 );
+ set_callbacks( OpenDecoder, NULL );
+ add_shortcut( "ac3" );
+vlc_module_end();
/*****************************************************************************
- * ac3_adec_Probe: probe the decoder and return score
+ * OpenDecoder: probe the decoder and return score
*****************************************************************************
* Tries to launch a decoder and return score so that the interface is able
* to chose.
*****************************************************************************/
-static int ac3_adec_Probe( probedata_t *p_data )
+static int OpenDecoder( vlc_object_t *p_this )
{
- if( p_data->i_type == AC3_AUDIO_ES )
- return( 50 );
- else
- return( 0 );
+ decoder_fifo_t *p_fifo = (decoder_fifo_t*) p_this;
+
+ if( p_fifo->i_fourcc != VLC_FOURCC('a','5','2',' ') )
+ {
+ return VLC_EGENERIC;
+ }
+
+ p_fifo->pf_run = RunDecoder;
+ return VLC_SUCCESS;
}
/*****************************************************************************
- * ac3_adec_Run: this function is called just after the thread is created
+ * RunDecoder: this function is called just after the thread is created
*****************************************************************************/
-static int ac3_adec_Run ( decoder_config_t * p_config )
+static int RunDecoder( decoder_fifo_t *p_fifo )
{
- ac3dec_thread_t * p_ac3thread;
- int sync;
-
- intf_DbgMsg( "ac3_adec debug: ac3_adec thread launched, initializing" );
+ ac3dec_t * p_ac3dec;
+ void * p_orig; /* pointer before memalign */
+ vlc_bool_t b_sync = 0;
/* Allocate the memory needed to store the thread's structure */
- p_ac3thread = (ac3dec_thread_t *)memalign(16, sizeof(ac3dec_thread_t));
+ p_ac3dec = (ac3dec_t *)vlc_memalign( &p_orig, 16, sizeof(ac3dec_t) );
+ memset( p_ac3dec, 0, sizeof( ac3dec_t ) );
- if(p_ac3thread == NULL)
+ if( p_ac3dec == NULL )
{
- intf_ErrMsg ( "ac3dec error: not enough memory "
- "for ac3_adec_Run() to allocate p_ac3thread" );
+ msg_Err( p_fifo, "out of memory" );
+ DecoderError( p_fifo );
return( -1 );
}
-
+
/*
* Initialize the thread properties
*/
- p_ac3thread->p_config = p_config;
- if( ac3_adec_Init( p_ac3thread ) )
+ p_ac3dec->p_fifo = p_fifo;
+ if( InitThread( p_ac3dec ) )
{
- intf_ErrMsg( "ac3_adec error : could not initialize thread" );
+ msg_Err( p_fifo, "could not initialize thread" );
+ DecoderError( p_fifo );
+ free( p_orig );
return( -1 );
}
- sync = 0;
- p_ac3thread->sync_ptr = 0;
-
/* ac3 decoder thread's main loop */
/* FIXME : do we have enough room to store the decoded frames ?? */
- while ((!p_ac3thread->p_fifo->b_die) && (!p_ac3thread->p_fifo->b_error))
+ while ((!p_ac3dec->p_fifo->b_die) && (!p_ac3dec->p_fifo->b_error))
{
s16 * buffer;
ac3_sync_info_t sync_info;
- int ptr;
-
- if (!sync) {
- do {
- GetBits(&p_ac3thread->ac3_decoder->bit_stream,8);
- } while ((!p_ac3thread->sync_ptr) && (!p_ac3thread->p_fifo->b_die)
- && (!p_ac3thread->p_fifo->b_error));
-
- ptr = p_ac3thread->sync_ptr;
-
- while(ptr-- && (!p_ac3thread->p_fifo->b_die)
- && (!p_ac3thread->p_fifo->b_error))
- {
- p_ac3thread->ac3_decoder->bit_stream.p_byte++;
- }
-
- /* we are in sync now */
- sync = 1;
+
+ if( !b_sync )
+ {
+ int i_sync_ptr;
+#define p_bit_stream (&p_ac3dec->bit_stream)
+
+ /* Go to the next PES packet and jump to sync_ptr */
+ do {
+ BitstreamNextDataPacket( p_bit_stream );
+ } while( !p_bit_stream->p_decoder_fifo->b_die
+ && !p_bit_stream->p_decoder_fifo->b_error
+ && p_bit_stream->p_data !=
+ p_bit_stream->p_decoder_fifo->p_first->p_first );
+ i_sync_ptr = *(p_bit_stream->p_byte - 2) << 8
+ | *(p_bit_stream->p_byte - 1);
+ p_bit_stream->p_byte += i_sync_ptr;
+
+ /* Empty the bit FIFO and realign the bit stream */
+ p_bit_stream->fifo.buffer = 0;
+ p_bit_stream->fifo.i_available = 0;
+ AlignWord( p_bit_stream );
+ b_sync = 1;
+#undef p_bit_stream
}
- if (DECODER_FIFO_START(*p_ac3thread->p_fifo)->i_pts)
+ if (ac3_sync_frame (p_ac3dec, &sync_info))
{
- p_ac3thread->p_aout_fifo->date[
- p_ac3thread->p_aout_fifo->l_end_frame] =
- DECODER_FIFO_START(*p_ac3thread->p_fifo)->i_pts;
- DECODER_FIFO_START(*p_ac3thread->p_fifo)->i_pts = 0;
- } else {
- p_ac3thread->p_aout_fifo->date[
- p_ac3thread->p_aout_fifo->l_end_frame] =
- LAST_MDATE;
+ b_sync = 0;
+ continue;
}
-
- if (ac3_sync_frame (p_ac3thread->ac3_decoder, &sync_info))
+
+ if( ( p_ac3dec->p_aout_fifo != NULL ) &&
+ ( p_ac3dec->p_aout_fifo->i_rate != sync_info.sample_rate ) )
{
- sync = 0;
- goto bad_frame;
+ /* Make sure the output thread leaves the NextFrame() function */
+ vlc_mutex_lock (&(p_ac3dec->p_aout_fifo->data_lock));
+ aout_DestroyFifo (p_ac3dec->p_aout_fifo);
+ vlc_cond_signal (&(p_ac3dec->p_aout_fifo->data_wait));
+ vlc_mutex_unlock (&(p_ac3dec->p_aout_fifo->data_lock));
+
+ p_ac3dec->p_aout_fifo = NULL;
}
- p_ac3thread->p_aout_fifo->l_rate = sync_info.sample_rate;
+ /* Creating the audio output fifo if not created yet */
+ if (p_ac3dec->p_aout_fifo == NULL ) {
+ p_ac3dec->p_aout_fifo =
+ aout_CreateFifo( p_ac3dec->p_fifo, AOUT_FIFO_PCM, 2,
+ sync_info.sample_rate, AC3DEC_FRAME_SIZE, NULL );
+ if ( p_ac3dec->p_aout_fifo == NULL )
+ {
+ p_ac3dec->p_fifo->b_error = 1;
+ break;
+ }
+ }
- buffer = ((s16 *)p_ac3thread->p_aout_fifo->buffer) +
- (p_ac3thread->p_aout_fifo->l_end_frame * AC3DEC_FRAME_SIZE);
+ CurrentPTS( &p_ac3dec->bit_stream,
+ &p_ac3dec->p_aout_fifo->date[p_ac3dec->p_aout_fifo->i_end_frame],
+ NULL );
+ if( !p_ac3dec->p_aout_fifo->date[p_ac3dec->p_aout_fifo->i_end_frame] )
+ {
+ p_ac3dec->p_aout_fifo->date[
+ p_ac3dec->p_aout_fifo->i_end_frame] =
+ LAST_MDATE;
+ }
+
+ buffer = ((s16 *)p_ac3dec->p_aout_fifo->buffer) +
+ (p_ac3dec->p_aout_fifo->i_end_frame * AC3DEC_FRAME_SIZE);
- if (ac3_decode_frame (p_ac3thread->ac3_decoder, buffer))
+ if (ac3_decode_frame (p_ac3dec, buffer))
{
- sync = 0;
- goto bad_frame;
+ b_sync = 0;
+ continue;
}
- vlc_mutex_lock (&p_ac3thread->p_aout_fifo->data_lock);
- p_ac3thread->p_aout_fifo->l_end_frame =
- (p_ac3thread->p_aout_fifo->l_end_frame + 1) & AOUT_FIFO_SIZE;
- vlc_cond_signal (&p_ac3thread->p_aout_fifo->data_wait);
- vlc_mutex_unlock (&p_ac3thread->p_aout_fifo->data_lock);
-
- bad_frame:
- RealignBits(&p_ac3thread->ac3_decoder->bit_stream);
+ vlc_mutex_lock (&p_ac3dec->p_aout_fifo->data_lock);
+ p_ac3dec->p_aout_fifo->i_end_frame =
+ (p_ac3dec->p_aout_fifo->i_end_frame + 1) & AOUT_FIFO_SIZE;
+ vlc_cond_signal (&p_ac3dec->p_aout_fifo->data_wait);
+ vlc_mutex_unlock (&p_ac3dec->p_aout_fifo->data_lock);
+
+ RealignBits(&p_ac3dec->bit_stream);
}
/* If b_error is set, the ac3 decoder thread enters the error loop */
- if (p_ac3thread->p_fifo->b_error)
+ if (p_ac3dec->p_fifo->b_error)
{
- ac3_adec_ErrorThread (p_ac3thread);
+ DecoderError( p_ac3dec->p_fifo );
}
/* End of the ac3 decoder thread */
- ac3_adec_EndThread (p_ac3thread);
-
+ EndThread (p_ac3dec);
+
+ free( p_orig );
+
return( 0 );
}
-
/*****************************************************************************
- * ac3_adec_Init: initialize data before entering main loop
+ * InitThread: initialize data before entering main loop
*****************************************************************************/
-static int ac3_adec_Init( ac3dec_thread_t * p_ac3thread )
+static int InitThread( ac3dec_t * p_ac3dec )
{
- /*
- * Thread properties
- */
- p_ac3thread->p_fifo = p_ac3thread->p_config->p_decoder_fifo;
- p_ac3thread->ac3_decoder = memalign(16, sizeof(ac3dec_t));
+ char *psz_name;
/*
* Choose the best downmix module
*/
-#define DOWNMIX p_ac3thread->ac3_decoder->downmix
- DOWNMIX.p_module = module_Need( MODULE_CAPABILITY_DOWNMIX, NULL );
+ p_ac3dec->p_downmix = vlc_object_create( p_ac3dec->p_fifo,
+ sizeof( downmix_t ) );
+ p_ac3dec->p_downmix->psz_object_name = "downmix";
- if( DOWNMIX.p_module == NULL )
+ psz_name = config_GetPsz( p_ac3dec->p_downmix, "ac3-downmix" );
+ p_ac3dec->p_downmix->p_module =
+ module_Need( p_ac3dec->p_downmix, "downmix", psz_name );
+ if( psz_name ) free( psz_name );
+
+ if( p_ac3dec->p_downmix->p_module == NULL )
{
- intf_ErrMsg( "ac3dec error: no suitable downmix module" );
- free( p_ac3thread->ac3_decoder );
- free( p_ac3thread );
+ msg_Err( p_ac3dec->p_fifo, "no suitable downmix module" );
+ vlc_object_destroy( p_ac3dec->p_downmix );
return( -1 );
}
-#define F DOWNMIX.p_module->p_functions->downmix.functions.downmix
- DOWNMIX.pf_downmix_3f_2r_to_2ch = F.pf_downmix_3f_2r_to_2ch;
- DOWNMIX.pf_downmix_2f_2r_to_2ch = F.pf_downmix_2f_2r_to_2ch;
- DOWNMIX.pf_downmix_3f_1r_to_2ch = F.pf_downmix_3f_1r_to_2ch;
- DOWNMIX.pf_downmix_2f_1r_to_2ch = F.pf_downmix_2f_1r_to_2ch;
- DOWNMIX.pf_downmix_3f_0r_to_2ch = F.pf_downmix_3f_0r_to_2ch;
- DOWNMIX.pf_stream_sample_2ch_to_s16 = F.pf_stream_sample_2ch_to_s16;
- DOWNMIX.pf_stream_sample_1ch_to_s16 = F.pf_stream_sample_1ch_to_s16;
-#undef F
-#undef DOWNMIX
-
/*
* Choose the best IMDCT module
*/
- p_ac3thread->ac3_decoder->imdct = memalign(16, sizeof(imdct_t));
+ p_ac3dec->p_imdct = vlc_object_create( p_ac3dec->p_fifo,
+ sizeof( imdct_t ) );
-#define IMDCT p_ac3thread->ac3_decoder->imdct
- IMDCT->p_module = module_Need( MODULE_CAPABILITY_IMDCT, NULL );
+#define IMDCT p_ac3dec->p_imdct
+ psz_name = config_GetPsz( p_ac3dec->p_fifo, "ac3-imdct" );
+ p_ac3dec->p_imdct->p_module =
+ module_Need( p_ac3dec->p_imdct, "imdct", psz_name );
+ if( psz_name ) free( psz_name );
- if( IMDCT->p_module == NULL )
+ if( p_ac3dec->p_imdct->p_module == NULL )
{
- intf_ErrMsg( "ac3dec error: no suitable IMDCT module" );
- module_Unneed( p_ac3thread->ac3_decoder->downmix.p_module );
- free( p_ac3thread->ac3_decoder->imdct );
- free( p_ac3thread->ac3_decoder );
- free( p_ac3thread );
+ msg_Err( p_ac3dec->p_fifo, "no suitable IMDCT module" );
+ vlc_object_destroy( p_ac3dec->p_imdct );
+ module_Unneed( p_ac3dec->p_downmix, p_ac3dec->p_downmix->p_module );
+ vlc_object_destroy( p_ac3dec->p_downmix );
return( -1 );
}
-#define F IMDCT->p_module->p_functions->imdct.functions.imdct
- IMDCT->pf_imdct_init = F.pf_imdct_init;
- IMDCT->pf_imdct_256 = F.pf_imdct_256;
- IMDCT->pf_imdct_256_nol = F.pf_imdct_256_nol;
- IMDCT->pf_imdct_512 = F.pf_imdct_512;
- IMDCT->pf_imdct_512_nol = F.pf_imdct_512_nol;
-#undef F
+ /* Initialize the ac3 decoder structures */
+ p_ac3dec->samples = vlc_memalign( &p_ac3dec->samples_orig,
+ 16, 6 * 256 * sizeof(float) );
+
+ IMDCT->buf = vlc_memalign( &IMDCT->buf_orig,
+ 16, N/4 * sizeof(complex_t) );
+ IMDCT->delay = vlc_memalign( &IMDCT->delay_orig,
+ 16, 6 * 256 * sizeof(float) );
+ IMDCT->delay1 = vlc_memalign( &IMDCT->delay1_orig,
+ 16, 6 * 256 * sizeof(float) );
+ IMDCT->xcos1 = vlc_memalign( &IMDCT->xcos1_orig,
+ 16, N/4 * sizeof(float) );
+ IMDCT->xsin1 = vlc_memalign( &IMDCT->xsin1_orig,
+ 16, N/4 * sizeof(float) );
+ IMDCT->xcos2 = vlc_memalign( &IMDCT->xcos2_orig,
+ 16, N/8 * sizeof(float) );
+ IMDCT->xsin2 = vlc_memalign( &IMDCT->xsin2_orig,
+ 16, N/8 * sizeof(float) );
+ IMDCT->xcos_sin_sse = vlc_memalign( &IMDCT->xcos_sin_sse_orig,
+ 16, 128 * 4 * sizeof(float) );
+ IMDCT->w_1 = vlc_memalign( &IMDCT->w_1_orig,
+ 16, 1 * sizeof(complex_t) );
+ IMDCT->w_2 = vlc_memalign( &IMDCT->w_2_orig,
+ 16, 2 * sizeof(complex_t) );
+ IMDCT->w_4 = vlc_memalign( &IMDCT->w_4_orig,
+ 16, 4 * sizeof(complex_t) );
+ IMDCT->w_8 = vlc_memalign( &IMDCT->w_8_orig,
+ 16, 8 * sizeof(complex_t) );
+ IMDCT->w_16 = vlc_memalign( &IMDCT->w_16_orig,
+ 16, 16 * sizeof(complex_t) );
+ IMDCT->w_32 = vlc_memalign( &IMDCT->w_32_orig,
+ 16, 32 * sizeof(complex_t) );
+ IMDCT->w_64 = vlc_memalign( &IMDCT->w_64_orig,
+ 16, 64 * sizeof(complex_t) );
#undef IMDCT
- /* Initialize the ac3 decoder structures */
-#if defined( __MINGW32__ )
- p_ac3thread->ac3_decoder->samples_back = memalign(16, 6 * 256 *
- sizeof(float) + 15);
- p_ac3thread->ac3_decoder->samples = (float *) (((unsigned long)
- p_ac3thread->ac3_decoder->samples_back+15) & ~0xFUL);
-#else
- p_ac3thread->ac3_decoder->samples = memalign(16, 6 * 256 * sizeof(float));
-#endif
- p_ac3thread->ac3_decoder->imdct->buf = memalign(16, N/4 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->delay = memalign(16, 6 * 256 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->delay1 = memalign(16, 6 * 256 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->xcos1 = memalign(16, N/4 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->xsin1 = memalign(16, N/4 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->xcos2 = memalign(16, N/8 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->xsin2 = memalign(16, N/8 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->xcos_sin_sse = memalign(16, 128 * 4 * sizeof(float));
- p_ac3thread->ac3_decoder->imdct->w_2 = memalign(16, 2 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->w_4 = memalign(16, 4 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->w_8 = memalign(16, 8 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->w_16 = memalign(16, 16 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->w_32 = memalign(16, 32 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->w_64 = memalign(16, 64 * sizeof(complex_t));
- p_ac3thread->ac3_decoder->imdct->w_1 = memalign(16, sizeof(complex_t));
-
- ac3_init (p_ac3thread->ac3_decoder);
+ E_( ac3_init )( p_ac3dec );
/*
* Initialize the output properties
*/
- p_ac3thread->p_aout_fifo = NULL;
-
- intf_DbgMsg ( "ac3_adec debug: ac3_adec thread (%p) initialized",
- p_ac3thread );
+ p_ac3dec->p_aout_fifo = NULL;
/*
* Bit stream
*/
- p_ac3thread->p_config->pf_init_bit_stream(
- &p_ac3thread->ac3_decoder->bit_stream,
- p_ac3thread->p_config->p_decoder_fifo,
- BitstreamCallback, (void *) p_ac3thread );
-
- /* Creating the audio output fifo */
- p_ac3thread->p_aout_fifo = aout_CreateFifo( AOUT_ADEC_STEREO_FIFO, 2, 0, 0,
- AC3DEC_FRAME_SIZE, NULL );
- if ( p_ac3thread->p_aout_fifo == NULL )
- {
- return( -1 );
- }
-
- intf_DbgMsg("ac3dec debug: ac3 decoder thread %p initialized", p_ac3thread);
+ InitBitstream( &p_ac3dec->bit_stream, p_ac3dec->p_fifo,
+ BitstreamCallback, (void *) p_ac3dec );
return( 0 );
}
-
/*****************************************************************************
- * ac3_adec_ErrorThread : ac3 decoder's RunThread() error loop
+ * EndThread : ac3 decoder thread destruction
*****************************************************************************/
-static void ac3_adec_ErrorThread (ac3dec_thread_t * p_ac3thread)
+static void EndThread (ac3dec_t * p_ac3dec)
{
- /* We take the lock, because we are going to read/write the start/end
- * indexes of the decoder fifo */
- vlc_mutex_lock (&p_ac3thread->p_fifo->data_lock);
-
- /* Wait until a `die' order is sent */
- while (!p_ac3thread->p_fifo->b_die)
- {
- /* Trash all received PES packets */
- while (!DECODER_FIFO_ISEMPTY(*p_ac3thread->p_fifo))
- {
- p_ac3thread->p_fifo->pf_delete_pes(
- p_ac3thread->p_fifo->p_packets_mgt,
- DECODER_FIFO_START(*p_ac3thread->p_fifo));
- DECODER_FIFO_INCSTART (*p_ac3thread->p_fifo);
- }
-
- /* Waiting for the input thread to put new PES packets in the fifo */
- vlc_cond_wait (&p_ac3thread->p_fifo->data_wait,
- &p_ac3thread->p_fifo->data_lock);
- }
-
- /* We can release the lock before leaving */
- vlc_mutex_unlock (&p_ac3thread->p_fifo->data_lock);
-}
-
-/*****************************************************************************
- * ac3_adec_EndThread : ac3 decoder thread destruction
- *****************************************************************************/
-static void ac3_adec_EndThread (ac3dec_thread_t * p_ac3thread)
-{
- intf_DbgMsg ("ac3dec debug: destroying ac3 decoder thread %p", p_ac3thread);
-
/* If the audio output fifo was created, we destroy it */
- if (p_ac3thread->p_aout_fifo != NULL)
+ if (p_ac3dec->p_aout_fifo != NULL)
{
- aout_DestroyFifo (p_ac3thread->p_aout_fifo);
+ aout_DestroyFifo (p_ac3dec->p_aout_fifo);
/* Make sure the output thread leaves the NextFrame() function */
- vlc_mutex_lock (&(p_ac3thread->p_aout_fifo->data_lock));
- vlc_cond_signal (&(p_ac3thread->p_aout_fifo->data_wait));
- vlc_mutex_unlock (&(p_ac3thread->p_aout_fifo->data_lock));
+ vlc_mutex_lock (&(p_ac3dec->p_aout_fifo->data_lock));
+ vlc_cond_signal (&(p_ac3dec->p_aout_fifo->data_wait));
+ vlc_mutex_unlock (&(p_ac3dec->p_aout_fifo->data_lock));
}
+ /* Free allocated structures */
+#define IMDCT p_ac3dec->p_imdct
+ free( IMDCT->w_1_orig );
+ free( IMDCT->w_64_orig );
+ free( IMDCT->w_32_orig );
+ free( IMDCT->w_16_orig );
+ free( IMDCT->w_8_orig );
+ free( IMDCT->w_4_orig );
+ free( IMDCT->w_2_orig );
+ free( IMDCT->xcos_sin_sse_orig );
+ free( IMDCT->xsin2_orig );
+ free( IMDCT->xcos2_orig );
+ free( IMDCT->xsin1_orig );
+ free( IMDCT->xcos1_orig );
+ free( IMDCT->delay1_orig );
+ free( IMDCT->delay_orig );
+ free( IMDCT->buf_orig );
+#undef IMDCT
+
+ free( p_ac3dec->samples_orig );
+
/* Unlock the modules */
- module_Unneed( p_ac3thread->ac3_decoder->downmix.p_module );
- module_Unneed( p_ac3thread->ac3_decoder->imdct->p_module );
-
- /* Destroy descriptor */
- free( p_ac3thread->ac3_decoder->imdct->w_1 );
- free( p_ac3thread->ac3_decoder->imdct->w_64 );
- free( p_ac3thread->ac3_decoder->imdct->w_32 );
- free( p_ac3thread->ac3_decoder->imdct->w_16 );
- free( p_ac3thread->ac3_decoder->imdct->w_8 );
- free( p_ac3thread->ac3_decoder->imdct->w_4 );
- free( p_ac3thread->ac3_decoder->imdct->w_2 );
- free( p_ac3thread->ac3_decoder->imdct->xcos_sin_sse );
- free( p_ac3thread->ac3_decoder->imdct->xsin2 );
- free( p_ac3thread->ac3_decoder->imdct->xcos2 );
- free( p_ac3thread->ac3_decoder->imdct->xsin1 );
- free( p_ac3thread->ac3_decoder->imdct->xcos1 );
- free( p_ac3thread->ac3_decoder->imdct->delay1 );
- free( p_ac3thread->ac3_decoder->imdct->delay );
- free( p_ac3thread->ac3_decoder->imdct->buf );
-#if defined( __MINGW32__ )
- free( p_ac3thread->ac3_decoder->samples_back );
-#else
- free( p_ac3thread->ac3_decoder->samples );
-#endif
- free( p_ac3thread->ac3_decoder->imdct );
- free( p_ac3thread->ac3_decoder );
- free( p_ac3thread->p_config );
- free( p_ac3thread );
+ module_Unneed( p_ac3dec->p_downmix, p_ac3dec->p_downmix->p_module );
+ vlc_object_destroy( p_ac3dec->p_downmix );
+
+ module_Unneed( p_ac3dec->p_imdct, p_ac3dec->p_imdct->p_module );
+ vlc_object_destroy( p_ac3dec->p_imdct );
- intf_DbgMsg ("ac3dec debug: ac3 decoder thread %p destroyed", p_ac3thread);
+ /* Free what's left of the decoder */
+ free( p_ac3dec->imdct_orig );
}
/*****************************************************************************
-* BitstreamCallback: Import parameters from the new data/PES packet
-*****************************************************************************
-* This function is called by input's NextDataPacket.
-*****************************************************************************/
+ * BitstreamCallback: Import parameters from the new data/PES packet
+ *****************************************************************************
+ * This function is called by input's NextDataPacket.
+ *****************************************************************************/
static void BitstreamCallback ( bit_stream_t * p_bit_stream,
- boolean_t b_new_pes)
+ vlc_bool_t b_new_pes )
{
-
- ac3dec_thread_t *p_ac3thread=(ac3dec_thread_t *)p_bit_stream->p_callback_arg;
-
if( b_new_pes )
{
- int ptr;
-
- ptr = *(p_bit_stream->p_byte + 1);
- ptr <<= 8;
- ptr |= *(p_bit_stream->p_byte + 2);
- p_ac3thread->sync_ptr = ptr;
- p_bit_stream->p_byte += 3;
+ /* Drop special AC3 header */
+/* p_bit_stream->p_byte += 3; */
}
}