]> git.sesse.net Git - vlc/commitdiff
* modules/codec/realaudio.c: real audio (cook) decoder using real's libraries.
authorGildas Bazin <gbazin@videolan.org>
Sun, 18 Sep 2005 17:23:13 +0000 (17:23 +0000)
committerGildas Bazin <gbazin@videolan.org>
Sun, 18 Sep 2005 17:23:13 +0000 (17:23 +0000)
configure.ac
modules/codec/Modules.am
modules/codec/realaudio.c [new file with mode: 0644]

index d5ce09069b634b5e6d7e46eadf362453a0af6a6e..a1a7a262ca2d8d3665355028d75a7d5d5581f3a1 100644 (file)
@@ -2393,6 +2393,15 @@ then
   fi
 fi
 
+dnl
+dnl  Real plugin
+dnl
+AC_ARG_ENABLE(real,
+  [  --enable-real           Real audio module (default disabled)])
+if test "${enable_real}" = "yes"; then
+  VLC_ADD_PLUGINS([realaudio])
+fi
+
 dnl
 dnl MP4 module
 dnl
@@ -4308,6 +4317,8 @@ AS_IF([test "${enable_loader}" = "yes"],
     VLC_ADD_LDFLAGS([dmo],[../../../loader/libloader.a])
     VLC_ADD_CPPFLAGS([quicktime],[-I../../@top_srcdir@/loader])
     VLC_ADD_LDFLAGS([quicktime],[../../loader/libloader.a])
+    VLC_ADD_CPPFLAGS([realaudio],[-I../../@top_srcdir@/loader])
+    VLC_ADD_LDFLAGS([realaudio],[../../loader/libloader.a])
   ])
 
 dnl
index 85ba3688867d49bff8817a3d28626bf13b7f3e5b..f71b31f20e1d1043c4037bf95de92d475b2192d8 100644 (file)
@@ -26,4 +26,5 @@ SOURCES_png = png.c
 SOURCES_svcdsub = svcdsub.c
 SOURCES_cvdsub = cvdsub.c
 SOURCES_fake = fake.c
+SOURCES_realaudio = realaudio.c
 SOURCES_sdl_image = sdl_image.c
diff --git a/modules/codec/realaudio.c b/modules/codec/realaudio.c
new file mode 100644 (file)
index 0000000..90e6ebd
--- /dev/null
@@ -0,0 +1,699 @@
+/*****************************************************************************
+ * realaudio.c: a realaudio decoder that uses the realaudio library/dll
+ *****************************************************************************
+ * Copyright (C) 2005 the VideoLAN team
+ * $Id: quicktime.c 11664 2005-07-09 06:17:09Z courmisch $
+ *
+ * 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/aout.h>
+#include <vlc/vout.h>
+#include <vlc/decoder.h>
+
+#ifdef LOADER
+/* Need the w32dll loader from mplayer */
+#   include <wine/winerror.h>
+#   include <ldt_keeper.h>
+#   include <wine/windef.h>
+
+void *WINAPI LoadLibraryA( char *name );
+void *WINAPI GetProcAddress( void *handle, char *func );
+int WINAPI FreeLibrary( void *handle );
+#endif
+
+#ifndef WINAPI
+#   define WINAPI
+#endif
+
+#if defined(HAVE_DL_DLOPEN)
+#   if defined(HAVE_DLFCN_H) /* Linux, BSD, Hurd */
+#       include <dlfcn.h>
+#   endif
+#   if defined(HAVE_SYS_DL_H)
+#       include <sys/dl.h>
+#   endif
+#endif
+
+/*****************************************************************************
+ * Module descriptor
+ *****************************************************************************/
+static int  Open ( vlc_object_t * );
+static void Close( vlc_object_t * );
+
+vlc_module_begin();
+    set_description( _("RealAudio library decoder") );
+    set_capability( "decoder", 10 );
+    set_category( CAT_INPUT );
+    set_subcategory( SUBCAT_INPUT_VCODEC );
+    set_callbacks( Open, Close );
+vlc_module_end();
+
+/*****************************************************************************
+ * Local prototypes
+ *****************************************************************************/
+static int  OpenDll( decoder_t * );
+static int  OpenNativeDll( decoder_t *, char *, char * );
+static int  OpenWin32Dll( decoder_t *, char *, char * );
+static void CloseDll( decoder_t * );
+
+static aout_buffer_t *Decode( decoder_t *, block_t ** );
+
+struct decoder_sys_t
+{
+    audio_date_t end_date;
+
+    /* Frame buffer for data reordering */
+    int i_subpacket;
+    int i_frame_size;
+    char *p_frame;
+    int i_frame;
+
+    /* Output buffer */
+    char *p_out;
+
+    /* Codec params */
+    void *context;
+    int i_subpacket_size;
+    int i_subpacket_h;
+    int i_codec_flavor;
+    int i_coded_frame_size;
+    int i_extra;
+    uint8_t *p_extra;
+
+    void *dll;
+    unsigned long (*raCloseCodec)(void*);
+    unsigned long (*raDecode)(void*, char*, unsigned long, char*,
+                              unsigned int*, long);
+    unsigned long (*raFlush)(unsigned long, unsigned long, unsigned long);
+    unsigned long (*raFreeDecoder)(void*);
+    void*         (*raGetFlavorProperty)(void*, unsigned long,
+                                         unsigned long, int*);
+    unsigned long (*raInitDecoder)(void*, void*);
+    unsigned long (*raOpenCodec)(void*);
+    unsigned long (*raOpenCodec2)(void*, void*);
+    unsigned long (*raSetFlavor)(void*, unsigned long);
+    void          (*raSetDLLAccessPath)(char*);
+    void          (*raSetPwd)(char*, char*);
+
+#if defined(LOADER)
+    ldt_fs_t *ldt_fs;
+#endif
+
+    void *win32_dll;
+    unsigned long WINAPI (*wraCloseCodec)(void*);
+    unsigned long WINAPI (*wraDecode)(void*, char*, unsigned long, char*,
+                                      unsigned int*, long);
+    unsigned long WINAPI (*wraFlush)(unsigned long, unsigned long,
+                                     unsigned long);
+    unsigned long WINAPI (*wraFreeDecoder)(void*);
+    void*         WINAPI (*wraGetFlavorProperty)(void*, unsigned long,
+                                                 unsigned long, int*);
+    unsigned long WINAPI (*wraInitDecoder)(void*, void*);
+    unsigned long WINAPI (*wraOpenCodec)(void*);
+    unsigned long WINAPI (*wraOpenCodec2)(void*, void*);
+    unsigned long WINAPI (*wraSetFlavor)(void*, unsigned long);
+    void          WINAPI (*wraSetDLLAccessPath)(char*);
+    void          WINAPI (*wraSetPwd)(char*, char*);
+};
+
+/* linux dlls doesn't need packing */
+typedef struct /*__attribute__((__packed__))*/ {
+    int samplerate;
+    short bits;
+    short channels;
+    short quality;
+    /* 2bytes padding here, by gcc */
+    int bits_per_frame;
+    int packetsize;
+    int extradata_len;
+    void* extradata;
+} ra_init_t;
+
+/* windows dlls need packed structs (no padding) */
+typedef struct __attribute__((__packed__)) {
+    int samplerate;
+    short bits;
+    short channels;
+    short quality;
+    int bits_per_frame;
+    int packetsize;
+    int extradata_len;
+    void* extradata;
+} wra_init_t;
+
+void *__builtin_new(unsigned long size) {return malloc(size);}
+void __builtin_delete(void *p) {free(p);}
+
+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
+};
+
+/*****************************************************************************
+ * Open: probe the decoder and return score
+ *****************************************************************************
+ * Tries to launch a decoder and return score so that the interface is able
+ * to choose.
+ *****************************************************************************/
+static int Open( vlc_object_t *p_this )
+{
+    decoder_t *p_dec = (decoder_t*)p_this;
+    decoder_sys_t *p_sys = p_dec->p_sys;
+
+    switch( p_dec->fmt_in.i_codec )
+    {
+    case VLC_FOURCC('c','o','o','k'):
+        break;
+
+    default:
+        return VLC_EGENERIC;
+    }
+
+    p_dec->p_sys = p_sys = malloc( sizeof( decoder_sys_t ) );
+    memset( p_sys, 0, sizeof(decoder_sys_t) );
+
+    if( p_dec->fmt_in.i_extra >= 10 )
+    {
+        p_sys->i_subpacket_size = ((short *)(p_dec->fmt_in.p_extra))[0];
+        p_sys->i_subpacket_h = ((short *)(p_dec->fmt_in.p_extra))[1];
+        p_sys->i_codec_flavor = ((short *)(p_dec->fmt_in.p_extra))[2];
+        p_sys->i_coded_frame_size = ((short *)(p_dec->fmt_in.p_extra))[3];
+        p_sys->i_extra = ((short*)(p_dec->fmt_in.p_extra))[4];
+        p_sys->p_extra = p_dec->fmt_in.p_extra + 10;
+    }
+
+    if( OpenDll( p_dec ) != VLC_SUCCESS )
+    {
+        free( p_sys );
+        return VLC_EGENERIC;
+    }
+
+#ifdef LOADER
+    if( p_sys->win32_dll ) Close( p_this );
+#endif
+
+    es_format_Init( &p_dec->fmt_out, AUDIO_ES, AOUT_FMT_S16_NE );
+    p_dec->fmt_out.audio.i_rate = p_dec->fmt_in.audio.i_rate;
+    p_dec->fmt_out.audio.i_channels = p_dec->fmt_in.audio.i_channels;
+    p_dec->fmt_out.audio.i_bitspersample =
+        p_dec->fmt_in.audio.i_bitspersample;
+    p_dec->fmt_out.audio.i_physical_channels =
+    p_dec->fmt_out.audio.i_original_channels =
+        pi_channels_maps[p_dec->fmt_out.audio.i_channels];
+
+    aout_DateInit( &p_sys->end_date, p_dec->fmt_out.audio.i_rate );
+    aout_DateSet( &p_sys->end_date, 0 );
+
+    p_dec->pf_decode_audio = Decode;
+
+    p_sys->i_frame_size =
+        p_dec->fmt_in.audio.i_blockalign * p_sys->i_subpacket_h;
+    p_sys->p_frame = malloc( p_sys->i_frame_size );
+    p_sys->i_subpacket = 0;
+    p_sys->i_frame = 0;
+
+    p_sys->p_out = malloc( 4096 * 10 );
+
+    return VLC_SUCCESS;
+}
+
+/*****************************************************************************
+ * Close:
+ *****************************************************************************/
+static void Close( vlc_object_t *p_this )
+{
+    decoder_t *p_dec = (decoder_t*)p_this;
+
+    CloseDll( p_dec );
+    if( p_dec->p_sys->p_frame ) free( p_dec->p_sys->p_frame );
+    if( p_dec->p_sys->p_out ) free( p_dec->p_sys->p_out );
+    free( p_dec->p_sys );
+}
+
+/*****************************************************************************
+ * OpenDll:
+ *****************************************************************************/
+static int OpenDll( decoder_t *p_dec )
+{
+    char *psz_codec, *psz_dll;
+    int i, i_result;
+
+    char *ppsz_path[] =
+    {
+      ".",
+      "z:\\home\\videolan\\vlc",
+      "/home/videolan/vlc",
+#ifndef WIN32
+      "/usr/local/RealPlayer8/Codecs",
+      "/usr/RealPlayer8/Codecs",
+      "/usr/lib/RealPlayer8/Codecs",
+      "/opt/RealPlayer8/Codecs",
+      "/usr/lib/RealPlayer9/users/Real/Codecs",
+      "/usr/lib64/RealPlayer8/Codecs",
+      "/usr/lib64/RealPlayer9/users/Real/Codecs",
+      "/usr/lib/win32",
+#endif
+      NULL
+    };
+
+    switch( p_dec->fmt_in.i_codec )
+    {
+    case VLC_FOURCC('c','o','o','k'):
+        psz_codec = "cook.so.6.0";
+        break;
+    case VLC_FOURCC('2','8','_','8'):
+        psz_codec = "28_8.so.6.0";
+        break;
+    case VLC_FOURCC('1','4','_','4'):
+        psz_codec = "14_4.so.6.0";
+        break;
+    default:
+        return VLC_EGENERIC;
+    }
+
+    for( i = 0; ppsz_path[i]; i++ )
+    {
+        asprintf( &psz_dll, "%s/%s", ppsz_path[i], psz_codec );
+        i_result = OpenNativeDll( p_dec, ppsz_path[i], psz_dll );
+        free( psz_dll );
+        if( i_result == VLC_SUCCESS ) return VLC_SUCCESS;
+    }
+
+    switch( p_dec->fmt_in.i_codec )
+    {
+    case VLC_FOURCC('c','o','o','k'):
+        psz_codec = "cook3260.dll";
+        break;
+    case VLC_FOURCC('2','8','_','8'):
+        psz_codec = "28_83260.dll";
+        break;
+    case VLC_FOURCC('1','4','_','4'):
+        psz_codec = "14_43260.dll";
+        break;
+    default:
+        return VLC_EGENERIC;
+    }
+
+    for( i = 0; ppsz_path[i]; i++ )
+    {
+        asprintf( &psz_dll, "%s\\%s", ppsz_path[i], psz_codec );
+        i_result = OpenWin32Dll( p_dec, ppsz_path[i], psz_dll );
+        free( psz_dll );
+        if( i_result == VLC_SUCCESS ) return VLC_SUCCESS;
+    }
+
+    return VLC_EGENERIC;
+}
+
+static int OpenNativeDll( decoder_t *p_dec, char *psz_path, char *psz_dll )
+{
+#if defined(HAVE_DL_DLOPEN)
+    decoder_sys_t *p_sys = p_dec->p_sys;
+    void *handle = 0, *context = 0;
+    unsigned int i_result;
+    void *p_prop;
+    int i_prop;
+
+    ra_init_t init_data =
+    {
+        p_dec->fmt_in.audio.i_rate,
+        p_dec->fmt_in.audio.i_bitspersample,
+        p_dec->fmt_in.audio.i_channels,
+        100, /* quality */
+        p_sys->i_subpacket_size,
+        p_sys->i_coded_frame_size,
+        p_sys->i_extra, p_sys->p_extra
+    };
+
+    msg_Dbg( p_dec, "opening library '%s'", psz_dll );
+    if( !(handle = dlopen( psz_dll, RTLD_LAZY )) )
+    {
+        msg_Dbg( p_dec, "couldn't load library '%s' (%s)",
+                 psz_dll, dlerror() );
+        return VLC_EGENERIC;
+    }
+
+    p_sys->raCloseCodec = dlsym( handle, "RACloseCodec" );
+    p_sys->raDecode = dlsym( handle, "RADecode" );
+    p_sys->raFlush = dlsym( handle, "RAFlush" );
+    p_sys->raFreeDecoder = dlsym( handle, "RAFreeDecoder" );
+    p_sys->raGetFlavorProperty = dlsym( handle, "RAGetFlavorProperty" );
+    p_sys->raOpenCodec = dlsym( handle, "RAOpenCodec" );
+    p_sys->raOpenCodec2 = dlsym( handle, "RAOpenCodec2" );
+    p_sys->raInitDecoder = dlsym( handle, "RAInitDecoder" );
+    p_sys->raSetFlavor = dlsym( handle, "RASetFlavor" );
+    p_sys->raSetDLLAccessPath = dlsym( handle, "SetDLLAccessPath" );
+    p_sys->raSetPwd = dlsym( handle, "RASetPwd" ); // optional, used by SIPR
+
+    if( !(p_sys->raOpenCodec || p_sys->raOpenCodec2) ||
+        !p_sys->raCloseCodec || !p_sys->raInitDecoder ||
+        !p_sys->raDecode || !p_sys->raFreeDecoder ||
+        !p_sys->raGetFlavorProperty || !p_sys->raSetFlavor
+        /* || !p_sys->raFlush || !p_sys->raSetDLLAccessPath */ )
+    {
+        goto error_native;
+    }
+
+    if( p_sys->raOpenCodec2 )
+        i_result = p_sys->raOpenCodec2( &context, psz_path );
+    else
+        i_result = p_sys->raOpenCodec( &context );
+
+    if( i_result )
+    {
+        msg_Err( p_dec, "decoder open failed, error code: 0x%x", i_result );
+        goto error_native;
+    }
+
+    i_result = p_sys->raInitDecoder( context, &init_data );
+    if( i_result )
+    {
+        msg_Err( p_dec, "decoder init failed, error code: 0x%x", i_result );
+        goto error_native;
+    }
+
+    i_result = p_sys->raSetFlavor( context, p_sys->i_codec_flavor );
+    if( i_result )
+    {
+        msg_Err( p_dec, "decoder flavor setup failed, error code: 0x%x",
+                 i_result );
+        goto error_native;
+    }
+
+    p_prop = p_sys->raGetFlavorProperty( context, p_sys->i_codec_flavor,
+                                         0, &i_prop );
+    msg_Dbg( p_dec, "audio codec: [%d] %s",
+             p_sys->i_codec_flavor, (char *)p_prop );
+
+    p_prop = p_sys->raGetFlavorProperty( context, p_sys->i_codec_flavor,
+                                         1, &i_prop );
+    if( p_prop )
+    {
+        int i_bps = ((*((int*)p_prop))+4)/8;
+        msg_Dbg( p_dec, "audio bitrate: %5.3f kbit/s (%d bps)",
+                 (*((int*)p_prop))*0.001f, i_bps );
+    }
+
+    p_sys->context = context;
+    p_sys->dll = handle;
+    return VLC_SUCCESS;
+
+ error_native:
+    if( context ) p_sys->raFreeDecoder( context );
+    if( context ) p_sys->raCloseCodec( context );
+    dlclose( handle );
+#endif
+
+    return VLC_EGENERIC;
+}
+
+static int OpenWin32Dll( decoder_t *p_dec, char *psz_path, char *psz_dll )
+{
+#if defined(LOADER) || defined(WIN32)
+    decoder_sys_t *p_sys = p_dec->p_sys;
+    void *handle = 0, *context = 0;
+    unsigned int i_result;
+    void *p_prop;
+    int i_prop;
+
+    wra_init_t init_data =
+    {
+        p_dec->fmt_in.audio.i_rate,
+        p_dec->fmt_in.audio.i_bitspersample,
+        p_dec->fmt_in.audio.i_channels,
+        100, /* quality */
+        p_sys->i_subpacket_size,
+        p_sys->i_coded_frame_size,
+        p_sys->i_extra, p_sys->p_extra
+    };
+
+    msg_Err( p_dec, "opening win32 dll '%s'", psz_dll );
+
+#ifdef LOADER
+    Setup_LDT_Keeper();
+#endif
+
+    if( !(handle = LoadLibraryA( psz_dll )) )
+    {
+        msg_Dbg( p_dec, "couldn't load dll '%s'", psz_dll );
+        return VLC_EGENERIC;
+    }
+
+    p_sys->wraCloseCodec = GetProcAddress( handle, "RACloseCodec" );
+    p_sys->wraDecode = GetProcAddress( handle, "RADecode" );
+    p_sys->wraFlush = GetProcAddress( handle, "RAFlush" );
+    p_sys->wraFreeDecoder = GetProcAddress( handle, "RAFreeDecoder" );
+    p_sys->wraGetFlavorProperty =
+        GetProcAddress( handle, "RAGetFlavorProperty" );
+    p_sys->wraOpenCodec = GetProcAddress( handle, "RAOpenCodec" );
+    p_sys->wraOpenCodec2 = GetProcAddress( handle, "RAOpenCodec2" );
+    p_sys->wraInitDecoder = GetProcAddress( handle, "RAInitDecoder" );
+    p_sys->wraSetFlavor = GetProcAddress( handle, "RASetFlavor" );
+    p_sys->wraSetDLLAccessPath = GetProcAddress( handle, "SetDLLAccessPath" );
+    p_sys->wraSetPwd =
+        GetProcAddress( handle, "RASetPwd" ); // optional, used by SIPR
+
+    if( !(p_sys->wraOpenCodec || p_sys->wraOpenCodec2) ||
+        !p_sys->wraCloseCodec || !p_sys->wraInitDecoder ||
+        !p_sys->wraDecode || !p_sys->wraFreeDecoder ||
+        !p_sys->wraGetFlavorProperty || !p_sys->wraSetFlavor
+        /* || !p_sys->wraFlush || !p_sys->wraSetDLLAccessPath */ )
+    {
+        FreeLibrary( handle );
+        return VLC_EGENERIC;
+    }
+
+    if( p_sys->wraOpenCodec2 )
+        i_result = p_sys->wraOpenCodec2( &context, psz_path );
+    else
+        i_result = p_sys->wraOpenCodec( &context );
+
+    if( i_result )
+    {
+        msg_Err( p_dec, "decoder open failed, error code: 0x%x", i_result );
+        goto error_win32;
+    }
+
+    i_result = p_sys->wraInitDecoder( context, &init_data );
+    if( i_result )
+    {
+        msg_Err( p_dec, "decoder init failed, error code: 0x%x", i_result );
+        goto error_win32;
+    }
+
+    i_result = p_sys->wraSetFlavor( context, p_sys->i_codec_flavor );
+    if( i_result )
+    {
+        msg_Err( p_dec, "decoder flavor setup failed, error code: 0x%x",
+                 i_result );
+        goto error_win32;
+    }
+
+    p_prop = p_sys->wraGetFlavorProperty( context, p_sys->i_codec_flavor,
+                                          0, &i_prop );
+    msg_Dbg( p_dec, "audio codec: [%d] %s",
+             p_sys->i_codec_flavor, (char *)p_prop );
+
+    p_prop = p_sys->wraGetFlavorProperty( context, p_sys->i_codec_flavor,
+                                          1, &i_prop );
+    if( p_prop )
+    {
+        int i_bps = ((*((int*)p_prop))+4)/8;
+        msg_Dbg( p_dec, "audio bitrate: %5.3f kbit/s (%d bps)",
+                 (*((int*)p_prop))*0.001f, i_bps );
+    }
+
+    p_sys->context = context;
+    p_sys->win32_dll = handle;
+    return VLC_SUCCESS;
+
+ error_win32:
+    if( context ) p_sys->wraFreeDecoder( context );
+    if( context ) p_sys->wraCloseCodec( context );
+    FreeLibrary( handle );
+#endif
+
+    return VLC_EGENERIC;
+}
+
+/*****************************************************************************
+ * CloseDll:
+ *****************************************************************************/
+static void CloseDll( decoder_t *p_dec )
+{
+    decoder_sys_t *p_sys = p_dec->p_sys;
+
+    if( p_sys->context && p_sys->dll )
+    {
+        p_sys->raFreeDecoder( p_sys->context );
+        p_sys->raCloseCodec( p_sys->context );
+    }
+
+    if( p_sys->context && p_sys->win32_dll )
+    {
+        p_sys->wraFreeDecoder( p_sys->context );
+        p_sys->wraCloseCodec( p_sys->context );
+    }
+
+#if defined(HAVE_DL_DLOPEN)
+    if( p_sys->dll ) dlclose( p_sys->dll );
+#endif
+
+#if defined(LOADER) || defined(WIN32)
+    if( p_sys->win32_dll ) FreeLibrary( p_sys->win32_dll );
+
+#if 0 //def LOADER /* Segfaults */
+    Restore_LDT_Keeper( p_sys->ldt_fs );
+    msg_Dbg( p_dec, "Restore_LDT_Keeper" );
+#endif
+#endif
+
+    p_sys->dll = 0;
+    p_sys->win32_dll = 0;
+    p_sys->context = 0;
+}
+
+/*****************************************************************************
+ * DecodeAudio:
+ *****************************************************************************/
+static aout_buffer_t *Decode( decoder_t *p_dec, block_t **pp_block )
+{
+    decoder_sys_t *p_sys = p_dec->p_sys;
+    aout_buffer_t *p_aout_buffer = 0;
+    block_t *p_block;
+
+#ifdef LOADER
+    if( !p_sys->win32_dll && !p_sys->dll )
+    {
+        /* We must do open and close in the same thread (unless we do
+         * Setup_LDT_Keeper in the main thread before all others */
+        if( OpenDll( p_dec ) != VLC_SUCCESS )
+        {
+            /* Fatal */
+            p_dec->b_error = VLC_TRUE;
+            return NULL;
+        }
+    }
+#endif
+
+    /* If we have a full frame ready then we can start decoding */
+    if( p_sys->i_frame )
+    {
+        char *p_out = p_sys->p_out;
+        unsigned int i_out, i_result;
+        int i_samples;
+
+        if( p_sys->dll )
+            i_result = p_sys->raDecode( p_sys->context, p_sys->p_frame +
+                                        p_sys->i_frame_size - p_sys->i_frame,
+                                        p_dec->fmt_in.audio.i_blockalign,
+                                        p_out, &i_out, -1 );
+        else
+            i_result = p_sys->wraDecode( p_sys->context, p_sys->p_frame +
+                                         p_sys->i_frame_size - p_sys->i_frame,
+                                         p_dec->fmt_in.audio.i_blockalign,
+                                         p_out, &i_out, -1 );
+
+        i_samples = i_out * 8 / p_dec->fmt_out.audio.i_bitspersample /
+            p_dec->fmt_out.audio.i_channels;
+
+#if 0
+        msg_Err( p_dec, "decoded: %i samples (%i)",
+                i_samples, i_result );
+#endif
+
+        p_sys->i_frame -= p_dec->fmt_in.audio.i_blockalign;
+
+        p_aout_buffer = p_dec->pf_aout_buffer_new( p_dec, i_samples );
+        if( p_aout_buffer == NULL ) return NULL;
+        memcpy( p_aout_buffer->p_buffer, p_out, i_out );
+
+        /* Date management */
+        p_aout_buffer->start_date = aout_DateGet( &p_sys->end_date );
+        p_aout_buffer->end_date =
+            aout_DateIncrement( &p_sys->end_date, i_samples );
+
+        return p_aout_buffer;
+    }
+
+
+    if( pp_block == NULL || *pp_block == NULL ) return NULL;
+    p_block = *pp_block;
+
+    //msg_Err( p_dec, "Decode: "I64Fd", %i", p_block->i_pts, p_block->i_buffer );
+
+    /* Date management */
+    if( /* !p_sys->i_subpacket && */ 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 );
+    }
+
+#if 0
+    if( !aout_DateGet( &p_sys->end_date ) )
+    {
+        /* We've just started the stream, wait for the first PTS. */
+        if( p_block ) block_Release( p_block );
+        return NULL;
+    }
+#endif
+
+    if( p_sys->i_subpacket_size )
+    {
+        /* 'cook' way */
+        int sps = p_sys->i_subpacket_size;
+        int w = p_dec->fmt_in.audio.i_blockalign;
+        int h = p_sys->i_subpacket_h;
+        /* int cfs = p_sys->i_coded_frame_size; */
+        int x, y;
+
+        w /= sps;
+        y = p_sys->i_subpacket;
+
+        for( x = 0; x < w; x++ )
+        {
+            memcpy( p_sys->p_frame + sps * (h*x+((h+1)/2)*(y&1)+(y>>1)),
+                    p_block->p_buffer, sps );
+            p_block->p_buffer += sps;
+        }
+
+        p_sys->i_subpacket = (p_sys->i_subpacket + 1) % p_sys->i_subpacket_h;
+
+        if( !p_sys->i_subpacket )
+        {
+            /* We have a complete frame */
+            p_sys->i_frame = p_sys->i_frame_size;
+            p_aout_buffer = Decode( p_dec, pp_block );
+        }
+    }
+
+    block_Release( p_block );
+    *pp_block = 0;
+    return p_aout_buffer;
+}