]> git.sesse.net Git - vlc/blobdiff - src/interface/main.c
* IPv6 network module, courtesy of Alexis Guillard <alexis.guillard@bt.com>,
[vlc] / src / interface / main.c
index 7e383c6501448a45bc7b0cceb40120d5fc21bb04..f8e41ab55069ebf8fc1e8d0330f9a7266e8e666f 100644 (file)
@@ -3,11 +3,12 @@
  * Includes the main() function for vlc. Parses command line, start interface
  * and spawn threads.
  *****************************************************************************
- * Copyright (C) 1998, 1999, 2000 VideoLAN
- * $Id: main.c,v 1.114 2001/09/25 11:46:14 massiot Exp $
+ * Copyright (C) 1998-2001 VideoLAN
+ * $Id: main.c,v 1.160 2002/03/04 23:56:38 massiot Exp $
  *
  * Authors: Vincent Seguin <seguin@via.ecp.fr>
  *          Samuel Hocevar <sam@zoy.org>
+ *          Gildas Bazin <gbazin@netcourrier.com>
  *
  * 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
 /*****************************************************************************
  * Preamble
  *****************************************************************************/
-#include "defs.h"
-
 #include <signal.h>                               /* SIGHUP, SIGINT, SIGKILL */
 #include <stdio.h>                                              /* sprintf() */
 #include <setjmp.h>                                       /* longjmp, setjmp */
 
+#include <videolan/vlc.h>
+
 #ifdef HAVE_GETOPT_LONG
 #   ifdef HAVE_GETOPT_H
 #       include <getopt.h>                                       /* getopt() */
 #   include <io.h>
 #endif
 
+#ifdef HAVE_LOCALE_H
+#    include <locale.h>
+#endif
+
 #include <errno.h>                                                 /* ENOMEM */
 #include <stdlib.h>                                  /* getenv(), strtol(),  */
 #include <string.h>                                            /* strerror() */
 #include <fcntl.h>                                       /* open(), O_WRONLY */
 #include <sys/stat.h>                                             /* S_IREAD */
 
-#include "config.h"
-#include "common.h"
-#include "debug.h"
-#include "threads.h"
-#include "mtime.h"
-#include "tests.h"                                              /* TestCPU() */
-#include "modules.h"
+#include "netutils.h"                                 /* network_ChannelJoin */
 
 #include "stream_control.h"
 #include "input_ext-intf.h"
 
-#include "intf_msg.h"
 #include "intf_playlist.h"
 #include "interface.h"
 
 #include "video.h"
 #include "video_output.h"
 
-#ifdef SYS_BEOS
-#   include "beos_specific.h"
-#endif
-
-#ifdef SYS_DARWIN
-#   include "darwin_specific.h"
-#endif
-
-#include "netutils.h"                                 /* network_ChannelJoin */
-
-#include "main.h"
+#include "debug.h"
 
 /*****************************************************************************
- * Command line options constants. If something is changed here, be sure that
- * GetConfiguration and Usage are also changed.
+ * Configuration options for the main program. Each plugin will also separatly
+ * define its own configuration options.
+ * Look into configuration.h if you need to know more about the following
+ * macros.
+ * 
+ *****************************************************************************/
+#define BUILTIN
+#define MODULE_NAME main
+#include "modules_inner.h"                        /* for configuration stuff */
+
+/* Quick usage guide
+MODULE_CONFIG_START
+MODULE_CONFIG_STOP
+ADD_CATEGORY_HINT( text, longtext )
+ADD_SUBCATEGORY_HINT( text, longtext )
+ADD_STRING( option_name, value, p_callback, text, longtext )
+ADD_FILE( option_name, psz_value, p_callback, text, longtext )
+ADD_PLUGIN( option_name, psz_value, i_capability, p_callback, text, longtext )
+ADD_INTEGER( option_name, i_value, p_callback, text, longtext )
+ADD_BOOL( option_name, p_callback, text, longtext )
+*/
+
+MODULE_CONFIG_START
+
+/* Help options */
+ADD_CATEGORY_HINT( "Help Options", NULL )
+ADD_BOOL    ( "help", NULL,"print help and exit (or use -h)", NULL )
+ADD_BOOL    ( "longhelp", NULL, "print long help version and exit (or use -H)",
+              NULL )
+ADD_BOOL    ( "list", NULL, "list available plugins (or use -l)", NULL )
+ADD_STRING  ( "pluginhelp", NULL, NULL,"print help on a plugin and exit",NULL )
+ADD_BOOL    ( "version", NULL, "output version information and exit", NULL )
+
+/* Interface options */
+ADD_CATEGORY_HINT( "Interface Options", NULL)
+ADD_PLUGIN  ( INTF_METHOD_VAR, MODULE_CAPABILITY_INTF, NULL, NULL,
+              "interface method", NULL )
+ADD_INTEGER ( INTF_WARNING_VAR, 0, NULL, "warning level (or use -v)", NULL )
+ADD_BOOL    ( INTF_STATS_VAR, NULL, "output statistics", NULL )
+ADD_STRING  ( INTF_PATH_VAR, NULL, NULL, "interface default search path", NULL)
+
+/* Audio Options */
+ADD_CATEGORY_HINT( "Audio Options", NULL)
+ADD_BOOL    ( AOUT_NOAUDIO_VAR, NULL, "disable audio", NULL )
+ADD_PLUGIN  ( AOUT_METHOD_VAR, MODULE_CAPABILITY_AOUT, NULL, NULL,
+              "audio output method", NULL )
+ADD_BOOL    ( AOUT_MONO_VAR, NULL, "mono audio", NULL )
+ADD_INTEGER ( AOUT_VOLUME_VAR, VOLUME_DEFAULT, NULL, "VLC output volume", NULL)
+ADD_INTEGER ( AOUT_RATE_VAR, 44100, NULL, "VLC output frequency", NULL )
+ADD_INTEGER ( AOUT_DESYNC_VAR, 0, NULL, "Compensate desynchronization of the "
+                                        "audio (in ms)", NULL )
+
+/* Video options */
+ADD_CATEGORY_HINT( "Video Options", NULL )
+ADD_BOOL    ( VOUT_NOVIDEO_VAR, NULL, "disable video", NULL )
+ADD_PLUGIN  ( VOUT_METHOD_VAR, MODULE_CAPABILITY_VOUT, NULL, NULL,
+              "video output method", NULL )
+ADD_STRING  ( VOUT_DISPLAY_VAR, NULL, NULL, "display string", NULL )
+ADD_INTEGER ( VOUT_WIDTH_VAR, 720, NULL, "display width", NULL )
+ADD_INTEGER ( VOUT_HEIGHT_VAR, 576, NULL, "display height", NULL )
+ADD_BOOL    ( VOUT_GRAYSCALE_VAR, NULL, "grayscale output", NULL )
+ADD_BOOL    ( VOUT_FULLSCREEN_VAR, NULL, "fullscreen output", NULL )
+ADD_BOOL    ( VOUT_NOOVERLAY_VAR, NULL, "disable accelerated display", NULL )
+ADD_PLUGIN  ( VOUT_FILTER_VAR, MODULE_CAPABILITY_VOUT, NULL, NULL,
+              "video filter module", NULL )
+ADD_INTEGER ( VOUT_SPUMARGIN_VAR, -1, NULL, "force SPU position", NULL )
+
+/* Input options */
+ADD_CATEGORY_HINT( "Input Options", NULL )
+ADD_STRING  ( INPUT_METHOD_VAR, NULL, NULL, "input method", NULL )
+ADD_INTEGER ( INPUT_PORT_VAR, 1234, NULL, "server port", NULL )
+ADD_BOOL    ( INPUT_NETWORK_CHANNEL_VAR, NULL, "enable network channel mode",
+              NULL )
+ADD_STRING  ( INPUT_CHANNEL_SERVER_VAR, "localhost", NULL,
+              "channel server address", NULL )
+ADD_INTEGER ( INPUT_CHANNEL_PORT_VAR, 6010, NULL, "channel server port", NULL )
+ADD_STRING  ( INPUT_IFACE_VAR, "eth0", NULL, "network interface", NULL )
+
+ADD_INTEGER ( INPUT_AUDIO_VAR, -1, NULL, "choose audio", NULL )
+ADD_INTEGER ( INPUT_CHANNEL_VAR, -1, NULL, "choose channel", NULL )
+ADD_INTEGER ( INPUT_SUBTITLE_VAR, -1, NULL, "choose subtitles", NULL )
+
+ADD_STRING  ( INPUT_DVD_DEVICE_VAR, "/dev/dvd", NULL, "DVD device", NULL )
+ADD_STRING  ( INPUT_VCD_DEVICE_VAR, "/dev/cdrom", NULL, "VCD device", NULL )
+ADD_BOOL    ( INPUT_IPV6_VAR, NULL, "force IPv6", NULL )
+ADD_BOOL    ( INPUT_IPV4_VAR, NULL, "force IPv4", NULL )
+
+/* Decoder options */
+ADD_CATEGORY_HINT( "Decoders Options", NULL )
+ADD_PLUGIN  ( ADEC_MPEG_VAR, MODULE_CAPABILITY_DECODER, NULL, NULL,
+              "choose MPEG audio decoder", NULL )
+ADD_PLUGIN  ( ADEC_AC3_VAR, MODULE_CAPABILITY_DECODER, NULL, NULL,
+              "choose AC3 audio decoder", NULL )
+ADD_INTEGER ( VDEC_SMP_VAR, 0, NULL, "use additional processors", NULL )
+ADD_STRING  ( VPAR_SYNCHRO_VAR, NULL, NULL, "force synchro algorithm "
+                                            "{I|I+|IP|IP+|IPB}", NULL )
+
+/* CPU options */
+ADD_CATEGORY_HINT( "CPU Options Options", NULL )
+ADD_BOOL    ( NOMMX_VAR, NULL, "disable CPU's MMX support", NULL )
+ADD_BOOL    ( NO3DN_VAR, NULL, "disable CPU's 3D Now! support", NULL )
+ADD_BOOL    ( NOMMXEXT_VAR, NULL, "disable CPU's MMX EXT support", NULL )
+ADD_BOOL    ( NOSSE_VAR, NULL, "disable CPU's SSE support", NULL )
+ADD_BOOL    ( NOALTIVEC_VAR, NULL, "disable CPU's AltiVec support", NULL )
+
+/* Playlist options */
+ADD_BOOL    ( PLAYLIST_STARTUP_VAR, NULL, "launch playlist on startup", NULL )
+ADD_BOOL    ( PLAYLIST_ENQUEUE_VAR, NULL, "enqueue playlist as default", NULL )
+ADD_BOOL    ( PLAYLIST_LOOP_VAR, NULL, "loop on playlist end", NULL )
+
+/* Misc options */
+ADD_CATEGORY_HINT( "Miscellaneous Options", NULL )
+ADD_PLUGIN  ( MEMCPY_METHOD_VAR, MODULE_CAPABILITY_MEMCPY, NULL, NULL,
+              "memory copy method", NULL )
+
+MODULE_CONFIG_STOP
+
+MODULE_INIT_START
+    SET_DESCRIPTION( "Main program" )
+    ADD_CAPABILITY( MAIN, 100/*whatever*/ )
+MODULE_INIT_STOP
+
+MODULE_ACTIVATE_START
+MODULE_ACTIVATE_STOP
+
+MODULE_DEACTIVATE_START
+MODULE_DEACTIVATE_STOP
+/*****************************************************************************
+ * End configuration.
  *****************************************************************************/
-
-/* Long options return values - note that values corresponding to short options
- * chars, and in general any regular char, should be avoided */
-#define OPT_NOAUDIO             150
-#define OPT_STEREO              151
-#define OPT_MONO                152
-#define OPT_SPDIF               153
-
-#define OPT_NOVIDEO             160
-#define OPT_DISPLAY             161
-#define OPT_WIDTH               162
-#define OPT_HEIGHT              163
-#define OPT_COLOR               164
-#define OPT_FULLSCREEN          165
-#define OPT_OVERLAY             166
-#define OPT_XVADAPTOR           167
-#define OPT_SMP                 168
-
-#define OPT_CHANNELS            170
-#define OPT_SERVER              171
-#define OPT_PORT                172
-#define OPT_BROADCAST           173
-
-#define OPT_INPUT               180
-#define OPT_MOTION              181
-#define OPT_IDCT                182
-#define OPT_YUV                 183
-#define OPT_DOWNMIX             184
-#define OPT_IMDCT               185
-
-#define OPT_SYNCHRO             190
-#define OPT_WARNING             191
-#define OPT_VERSION             192
-#define OPT_STDOUT              193
-
-/* Usage fashion */
-#define USAGE                     0
-#define SHORT_HELP                1
-#define LONG_HELP                 2
-
-/* Needed for x86 CPU capabilities detection */
-#define cpuid( a )                 \
-    asm volatile ( "cpuid"         \
-                 : "=a" ( i_eax ), \
-                   "=b" ( i_ebx ), \
-                   "=c" ( i_ecx ), \
-                   "=d" ( i_edx )  \
-                 : "a"  ( a )      \
-                 : "cc" );
-
-/* Long options */
-static const struct option longopts[] =
-{
-    /*  name,               has_arg,    flag,   val */
-
-    /* General/common options */
-    {   "help",             0,          0,      'h' },
-    {   "longhelp",         0,          0,      'H' },
-    {   "version",          0,          0,      OPT_VERSION },
-
-    /* Interface options */
-    {   "intf",             1,          0,      'I' },
-    {   "warning",          1,          0,      OPT_WARNING },
-    {   "stdout",           1,          0,      OPT_STDOUT },
-
-    /* Audio options */
-    {   "noaudio",          0,          0,      OPT_NOAUDIO },
-    {   "aout",             1,          0,      'A' },
-    {   "stereo",           0,          0,      OPT_STEREO },
-    {   "mono",             0,          0,      OPT_MONO },
-    {   "spdif",            0,          0,      OPT_SPDIF },
-    {   "downmix",          1,          0,      OPT_DOWNMIX },
-    {   "imdct",            1,          0,      OPT_IMDCT },
-
-    /* Video options */
-    {   "novideo",          0,          0,      OPT_NOVIDEO },
-    {   "vout",             1,          0,      'V' },
-    {   "display",          1,          0,      OPT_DISPLAY },
-    {   "width",            1,          0,      OPT_WIDTH },
-    {   "height",           1,          0,      OPT_HEIGHT },
-    {   "grayscale",        0,          0,      'g' },
-    {   "color",            0,          0,      OPT_COLOR },
-    {   "motion",           1,          0,      OPT_MOTION },
-    {   "idct",             1,          0,      OPT_IDCT },
-    {   "yuv",              1,          0,      OPT_YUV },
-    {   "fullscreen",       0,          0,      OPT_FULLSCREEN },
-    {   "overlay",          0,          0,      OPT_OVERLAY },
-    {   "xvadaptor",        1,          0,      OPT_XVADAPTOR },
-    {   "smp",              1,          0,      OPT_SMP },
-
-    /* DVD options */
-    {   "dvdtitle",         1,          0,      't' },
-    {   "dvdchapter",       1,          0,      'T' },
-    {   "dvdangle",         1,          0,      'u' },
-    {   "dvdaudio",         1,          0,      'a' },
-    {   "dvdchannel",       1,          0,      'c' },
-    {   "dvdsubtitle",      1,          0,      's' },
-    
-    /* Input options */
-    {   "input",            1,          0,      OPT_INPUT },
-    {   "server",           1,          0,      OPT_SERVER },
-    {   "port",             1,          0,      OPT_PORT },
-    {   "broadcast",        1,          0,      OPT_BROADCAST },
-    {   "channels",         0,          0,      OPT_CHANNELS },
-
-    /* Synchro options */
-    {   "synchro",          1,          0,      OPT_SYNCHRO },
-    {   0,                  0,          0,      0 }
-};
-
-/* Short options */
-static const char *psz_shortopts = "hHvgt:T:u:a:s:c:I:A:V:";
 
 /*****************************************************************************
- * Global variable program_data - these are the only ones, see main.h and
- * modules.h
+ * Global variables - these are the only ones, see main.h and modules.h
  *****************************************************************************/
 main_t        *p_main;
 module_bank_t *p_module_bank;
+input_bank_t  *p_input_bank;
 aout_bank_t   *p_aout_bank;
 vout_bank_t   *p_vout_bank;
 
 /*****************************************************************************
  * Local prototypes
  *****************************************************************************/
-static int  GetConfiguration        ( int *pi_argc, char *ppsz_argv[],
-                                      char *ppsz_env[] );
-static int  GetFilenames            ( int i_argc, char *ppsz_argv[] );
-static void Usage                   ( int i_fashion );
-static void Version                 ( void );
-
-static void InitSignalHandler       ( void );
-static void SimpleSignalHandler     ( int i_signal );
-static void FatalSignalHandler      ( int i_signal );
-static void InstructionSignalHandler( int i_signal );
-static int  CPUCapabilities         ( void );
-
-static int  RedirectSTDOUT          ( void );
-static void ShowConsole             ( void );
+static int  GetConfigurationFromFile    ( void ){return 0;};
+static int  GetConfigurationFromCmdLine ( int *pi_argc, char *ppsz_argv[],
+                                          boolean_t b_ignore_errors );
+static int  GetFilenames                ( int i_argc, char *ppsz_argv[] );
+static void Usage                       ( const char *psz_module_name );
+static void ListModules                 ( void );
+static void Version                     ( void );
+
+static void InitSignalHandler           ( void );
+static void SimpleSignalHandler         ( int i_signal );
+static void FatalSignalHandler          ( int i_signal );
+static void IllegalSignalHandler        ( int i_signal );
+static u32  CPUCapabilities             ( void );
+
+#ifdef WIN32
+static void ShowConsole                 ( void );
+#endif
 
 static jmp_buf env;
-static int  i_illegal;
+static int     i_illegal;
+static char   *psz_capability;
 
 /*****************************************************************************
  * main: parse command line, start interface and spawn threads
@@ -257,69 +264,200 @@ int main( int i_argc, char *ppsz_argv[], char *ppsz_env[] )
 {
     main_t        main_data;                /* root of all data - see main.h */
     module_bank_t module_bank;
+    input_bank_t  input_bank;
     aout_bank_t   aout_bank;
     vout_bank_t   vout_bank;
+    char *p_tmp;
 
     p_main        = &main_data;               /* set up the global variables */
     p_module_bank = &module_bank;
+    p_input_bank  = &input_bank;
     p_aout_bank   = &aout_bank;
     p_vout_bank   = &vout_bank;
 
+    p_main->i_warning_level = 0;
+
+#if defined( ENABLE_NLS ) && defined ( HAVE_GETTEXT )
+    /*
+     * Support for getext
+     */
+#if defined( HAVE_LOCALE_H ) && defined( HAVE_LC_MESSAGES )
+    if( !setlocale( LC_MESSAGES, "" ) )
+    {
+        fprintf( stderr, "warning: unsupported locale.\n" );
+    }
+#endif
+
+    if( !bindtextdomain( PACKAGE, LOCALEDIR ) )
+    {
+        fprintf( stderr, "warning: no domain %s in directory %s\n",
+                 PACKAGE, LOCALEDIR );
+    }
+
+    textdomain( PACKAGE );
+#endif
+
     /*
      * Initialize threads system
      */
     vlc_threads_init( );
 
     /*
-     * Test if our code is likely to run on this CPU 
+     * Test if our code is likely to run on this CPU
      */
     p_main->i_cpu_capabilities = CPUCapabilities();
-    
-#if defined( __pentium__ ) || defined( __pentiumpro__ )
-    if( ! TestCPU( CPU_CAPABILITY_586 ) )
-    {
-        fprintf( stderr, "error: this program needs a Pentium CPU,\n"
-                         "please try a version without Pentium support\n" );
-        return( 1 );
-    }
-#endif
 
     /*
      * System specific initialization code
      */
-#if defined( SYS_BEOS ) || defined( SYS_DARWIN )
+#if defined( SYS_BEOS ) || defined( SYS_DARWIN ) || defined( WIN32 )
     system_Init( &i_argc, ppsz_argv, ppsz_env );
-#elif defined( WIN32 )
-    _fmode = _O_BINARY;   /* sets the default file-translation mode on Win32 */
+
+#elif defined( SYS_LINUX )
+#   ifdef DEBUG
+    /* Activate malloc checking routines to detect heap corruptions. */
+    putenv( "MALLOC_CHECK_=2" );
+#   endif
 #endif
 
     /*
      * Initialize messages interface
      */
-    p_main->p_msg = intf_MsgCreate();
-    if( !p_main->p_msg )                         /* start messages interface */
+    intf_MsgCreate();
+
+    intf_Msg( COPYRIGHT_MESSAGE "\n" );
+
+
+    /* Get the executable name (similar to the basename command) */
+    p_main->psz_arg0 = p_tmp = ppsz_argv[ 0 ];
+    while( *p_tmp )
+    {
+        if( *p_tmp == '/' ) p_main->psz_arg0 = ++p_tmp;
+        else ++p_tmp;
+    }
+
+    /*
+     * Initialize the module bank and and load the configuration of the main
+     * module. We need to do this at this stage to be able to display a short
+     * help if required by the user. (short help == main module options)
+     */
+    module_InitBank();
+    module_LoadMain();
+
+    if( GetConfigurationFromCmdLine( &i_argc, ppsz_argv, 1 ) )
     {
-        fprintf( stderr, "error: can't initialize messages interface (%s)\n",
-                 strerror(errno) );
+        intf_MsgDestroy();
         return( errno );
     }
 
-    intf_MsgImm( COPYRIGHT_MESSAGE );
+    /* Check for short help option */
+    if( config_GetIntVariable( "help" ) )
+    {
+        Usage( "main" );
+        return( -1 );
+    }
+
+    /* Check for version option */
+    if( config_GetIntVariable( "version" ) )
+    {
+        Version();
+        return( -1 );
+    }
+
+    /*
+     * Load the builtins and plugins into the module_bank.
+     * We have to do it before GetConfiguration() because this also gets the
+     * list of configuration options exported by each plugin and loads their
+     * default values.
+     */
+    module_LoadBuiltins();
+    module_LoadPlugins();
+    intf_WarnMsg( 2, "module: module bank initialized, found %i modules",
+                  p_module_bank->i_count );
+
+    /* Check for help on plugins */
+    if( (p_tmp = config_GetPszVariable( "pluginhelp" )) )
+    {
+        Usage( p_tmp );
+        free( p_tmp );
+        return( -1 );
+    }
+
+    /* Check for long help option */
+    if( config_GetIntVariable( "longhelp" ) )
+    {
+        Usage( NULL );
+        return( -1 );
+    }
+
+    /* Check for plugin list option */
+    if( config_GetIntVariable( "list" ) )
+    {
+        ListModules();
+        return( -1 );
+    }
 
     /*
-     * Read configuration
+     * Override default configuration with config file settings
      */
-    if( GetConfiguration( &i_argc, ppsz_argv, ppsz_env ) ) /* parse cmd line */
+    if( GetConfigurationFromFile() )
     {
         intf_MsgDestroy();
         return( errno );
     }
 
     /*
-     * Redirect the standard output if required by the user, and on Win32 we
-     * also open a console to display the debug messages.
+     * Override configuration with command line settings
      */
-    RedirectSTDOUT();
+    if( GetConfigurationFromCmdLine( &i_argc, ppsz_argv, 0 ) )
+    {
+        intf_MsgDestroy();
+        return( errno );
+    }
+
+    /* p_main inititalization. FIXME ? */
+    p_main->i_desync = (mtime_t)config_GetIntVariable( AOUT_DESYNC_VAR )
+      * (mtime_t)1000;
+    p_main->b_stats = config_GetIntVariable( INTF_STATS_VAR );
+    p_main->b_audio = !config_GetIntVariable( AOUT_NOAUDIO_VAR );
+    p_main->b_stereo= !config_GetIntVariable( AOUT_MONO_VAR );
+    p_main->b_video = !config_GetIntVariable( VOUT_NOVIDEO_VAR );
+    if( config_GetIntVariable( NOMMX_VAR ) )
+        p_main->i_cpu_capabilities &= ~CPU_CAPABILITY_MMX;
+    if( config_GetIntVariable( NO3DN_VAR ) )
+        p_main->i_cpu_capabilities &= ~CPU_CAPABILITY_3DNOW;
+    if( config_GetIntVariable( NOMMXEXT_VAR ) )
+        p_main->i_cpu_capabilities &= ~CPU_CAPABILITY_MMXEXT;
+    if( config_GetIntVariable( NOSSE_VAR ) )
+        p_main->i_cpu_capabilities &= ~CPU_CAPABILITY_SSE;
+    if( config_GetIntVariable( NOALTIVEC_VAR ) )
+        p_main->i_cpu_capabilities &= ~CPU_CAPABILITY_ALTIVEC;
+
+
+    if( p_main->b_stats )
+    {
+        char          p_capabilities[200];
+        p_capabilities[0] = '\0';
+
+#define PRINT_CAPABILITY( capability, string )                              \
+        if( p_main->i_cpu_capabilities & capability )                       \
+        {                                                                   \
+            strncat( p_capabilities, string " ",                            \
+                     sizeof(p_capabilities) - strlen(p_capabilities) );     \
+            p_capabilities[sizeof(p_capabilities) - 1] = '\0';              \
+        }
+
+        PRINT_CAPABILITY( CPU_CAPABILITY_486, "486" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_586, "586" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_PPRO, "Pentium Pro" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_MMX, "MMX" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_3DNOW, "3DNow!" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_MMXEXT, "MMXEXT" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_SSE, "SSE" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_ALTIVEC, "Altivec" );
+        PRINT_CAPABILITY( CPU_CAPABILITY_FPU, "FPU" );
+        intf_StatMsg( "info: CPU has capabilities : %s", p_capabilities );
+    }
 
     /*
      * Initialize playlist and get commandline files
@@ -339,23 +477,39 @@ int main( int i_argc, char *ppsz_argv[], char *ppsz_env[] )
     GetFilenames( i_argc, ppsz_argv );
 
     /*
-     * Initialize module, aout and vout banks
+     * Initialize input, aout and vout banks
      */
-    module_InitBank();
+    input_InitBank();
     aout_InitBank();
     vout_InitBank();
 
+    /*
+     * Choose the best memcpy module
+     */
+    p_main->p_memcpy_module = module_Need( MODULE_CAPABILITY_MEMCPY, NULL,
+                                           NULL );
+    if( p_main->p_memcpy_module == NULL )
+    {
+        intf_ErrMsg( "intf error: no suitable memcpy module, "
+                     "using libc default" );
+        p_main->pf_memcpy = memcpy;
+    }
+    else
+    {
+        p_main->pf_memcpy = p_main->p_memcpy_module->p_functions
+                                  ->memcpy.functions.memcpy.pf_memcpy;
+    }
+
     /*
      * Initialize shared resources and libraries
      */
-    if( main_GetIntVariable( INPUT_NETWORK_CHANNEL_VAR,
-                             INPUT_NETWORK_CHANNEL_DEFAULT ) &&
+    if( config_GetIntVariable( INPUT_NETWORK_CHANNEL_VAR ) &&
         network_ChannelCreate() )
     {
         /* On error during Channels initialization, switch off channels */
-        intf_Msg( "Channels initialization failed : "
-                  "Channel management is deactivated" );
-        main_PutIntVariable( INPUT_NETWORK_CHANNEL_VAR, 0 );
+        intf_ErrMsg( "intf error: channels initialization failed, " 
+                                 "deactivating channels" );
+        config_PutIntVariable( INPUT_NETWORK_CHANNEL_VAR, 0 );
     }
 
     /*
@@ -386,29 +540,41 @@ int main( int i_argc, char *ppsz_argv[], char *ppsz_env[] )
         /*
          * Go back into channel 0 which is the network
          */
-        if( main_GetIntVariable( INPUT_NETWORK_CHANNEL_VAR,
-                                 INPUT_NETWORK_CHANNEL_DEFAULT ) )
+        if( config_GetIntVariable( INPUT_NETWORK_CHANNEL_VAR ) )
         {
             network_ChannelJoin( COMMON_CHANNEL );
         }
     }
 
     /*
-     * Free module, aout and vout banks
+     * Free input, aout and vout banks
      */
+    input_EndBank();
     vout_EndBank();
     aout_EndBank();
-    module_EndBank();
 
     /*
      * Free playlist
      */
     intf_PlaylistDestroy( p_main->p_playlist );
 
+    /*
+     * Free memcpy module if it was allocated
+     */
+    if( p_main->p_memcpy_module != NULL )
+    {
+        module_Unneed( p_main->p_memcpy_module );
+    }
+
+    /*
+     * Free module bank
+     */
+    module_EndBank();
+
     /*
      * System specific cleaning code
      */
-#if defined( SYS_BEOS ) || defined( SYS_DARWIN )
+#if defined( SYS_BEOS ) || defined( SYS_DARWIN ) || defined( WIN32 )
     system_End();
 #endif
 
@@ -416,7 +582,7 @@ int main( int i_argc, char *ppsz_argv[], char *ppsz_env[] )
     /*
      * Terminate messages interface and program
      */
-    intf_Msg( "intf: program terminated" );
+    intf_WarnMsg( 1, "intf: program terminated" );
     intf_MsgDestroy();
 
     /*
@@ -427,130 +593,37 @@ int main( int i_argc, char *ppsz_argv[], char *ppsz_env[] )
     return 0;
 }
 
-/*****************************************************************************
- * main_GetIntVariable: get the int value of an environment variable
- *****************************************************************************
- * This function is used to read some default parameters in modules.
- *****************************************************************************/
-int main_GetIntVariable( char *psz_name, int i_default )
-{
-    char *      psz_env;                                /* environment value */
-    char *      psz_end;                             /* end of parsing index */
-    long int    i_value;                                            /* value */
-
-    psz_env = getenv( psz_name );
-    if( psz_env )
-    {
-        i_value = strtol( psz_env, &psz_end, 0 );
-        if( (*psz_env != '\0') && (*psz_end == '\0') )
-        {
-            return( i_value );
-        }
-    }
-    return( i_default );
-}
-
-/*****************************************************************************
- * main_GetPszVariable: get the string value of an environment variable
- *****************************************************************************
- * This function is used to read some default parameters in modules.
- *****************************************************************************/
-char * main_GetPszVariable( char *psz_name, char *psz_default )
-{
-    char *psz_env;
-
-    psz_env = getenv( psz_name );
-    if( psz_env )
-    {
-        return( psz_env );
-    }
-    return( psz_default );
-}
-
-/*****************************************************************************
- * main_PutPszVariable: set the string value of an environment variable
- *****************************************************************************
- * This function is used to set some default parameters in modules. The use of
- * this function will cause some memory leak: since some systems use the pointer
- * passed to putenv to store the environment string, it can't be freed.
- *****************************************************************************/
-void main_PutPszVariable( char *psz_name, char *psz_value )
-{
-    char *psz_env;
 
-    psz_env = malloc( strlen(psz_name) + strlen(psz_value) + 2 );
-    if( psz_env == NULL )
-    {
-        intf_ErrMsg( "intf error: cannot create psz_env (%s)",
-                     strerror(ENOMEM) );
-    }
-    else
-    {
-        sprintf( psz_env, "%s=%s", psz_name, psz_value );
-        if( putenv( psz_env ) )
-        {
-            intf_ErrMsg( "intf error: cannot putenv (%s)", strerror(errno) );
-        }
-    }
-}
+/* following functions are local */
 
 /*****************************************************************************
- * main_PutIntVariable: set the integer value of an environment variable
+ * GetConfigurationFromCmdLine: parse command line
  *****************************************************************************
- * This function is used to set some default parameters in modules. The use of
- * this function will cause some memory leak: since some systems use the pointer
- * passed to putenv to store the environment string, it can't be freed.
+ * Parse command line for configuration. If the inline help is requested, the
+ * function Usage() is called and the function returns -1 (causing main() to
+ * exit).
+ * Now that the module_bank has been initialized, we can dynamically
+ * generate the longopts structure used by getops. We have to do it this way
+ * because we don't know (and don't want to know) in advance the configuration
+ * options used (ie. exported) by each module.
  *****************************************************************************/
-void main_PutIntVariable( char *psz_name, int i_value )
+static int GetConfigurationFromCmdLine( int *pi_argc, char *ppsz_argv[],
+                                        boolean_t b_ignore_errors )
 {
-    char psz_value[ 256 ];                               /* buffer for value */
+    int i_cmd, i, i_index, i_longopts_size;
+    module_t *p_module;
+    struct option *p_longopts;
 
-    sprintf( psz_value, "%d", i_value );
-    main_PutPszVariable( psz_name, psz_value );
-}
-
-/* following functions are local */
+    /* Short options */
+    const char *psz_shortopts = "hHvl";
 
-/*****************************************************************************
- * GetConfiguration: parse command line
- *****************************************************************************
- * Parse command line and configuration file for configuration. If the inline
- * help is requested, the function Usage() is called and the function returns
- * -1 (causing main() to exit). The messages interface is initialized at this
- * stage, but most structures are not allocated, so only environment should
- * be used.
- *****************************************************************************/
-static int GetConfiguration( int *pi_argc, char *ppsz_argv[], char *ppsz_env[] )
-{
-    int   i_cmd;
-    char *p_tmp;
 
     /* Set default configuration and copy arguments */
     p_main->i_argc    = *pi_argc;
     p_main->ppsz_argv = ppsz_argv;
-    p_main->ppsz_env  = ppsz_env;
-
-    p_main->b_audio     = 1;
-    p_main->b_video     = 1;
-
-    p_main->i_warning_level = 0;
 
     p_main->p_channel = NULL;
 
-    /* Get the executable name (similar to the basename command) */
-    p_main->psz_arg0 = p_tmp = ppsz_argv[ 0 ];
-    while( *p_tmp )
-    {
-        if( *p_tmp == '/' )
-        {
-            p_main->psz_arg0 = ++p_tmp;
-        }
-        else
-        {
-            ++p_tmp;
-        }
-    }
-
 #ifdef SYS_DARWIN
     /* When vlc.app is run by double clicking in Mac OS X, the 2nd arg
      * is the PSN - process serial number (a unique PID-ish thingie)
@@ -571,213 +644,132 @@ static int GetConfiguration( int *pi_argc, char *ppsz_argv[], char *ppsz_env[] )
     }
 #endif
 
-    /* Parse command line options */
+
+    /*
+     * Generate the longopts structure used by getopt_long
+     */
+    i_longopts_size = 0;
+    for( p_module = p_module_bank->first ;
+         p_module != NULL ;
+         p_module = p_module->next )
+    {
+        /* count the number of exported configuration options (to allocate
+         * longopts). The i_config_options we use is an approximation of the
+         * real number of options (it also includes markers like: category ...)
+         * but it is enough for our purpose */
+        i_longopts_size += p_module->i_config_options -1;
+    }
+
+    p_longopts = (struct option *)malloc( sizeof(struct option)
+                                          * (i_longopts_size + 1) );
+    if( p_longopts == NULL )
+    {
+        intf_ErrMsg( "GetConfigurationFromCmdLine error: "
+                     "can't allocate p_longopts" );
+        return( -1 );
+    }
+
+    /* Fill the longopts structure */
+    i_index = 0;
+    for( p_module = p_module_bank->first ;
+         p_module != NULL ;
+         p_module = p_module->next )
+    {
+        for( i = 1; i < (p_module->i_config_options -1); i++ )
+        {
+            if( (p_module->p_config[i].i_type == MODULE_CONFIG_ITEM_CATEGORY)||
+                (p_module->p_config[i].i_type ==
+                     MODULE_CONFIG_ITEM_SUBCATEGORY)||
+                (p_module->p_config[i].i_type ==
+                     MODULE_CONFIG_ITEM_SUBCATEGORY_END) )
+                 continue;
+            p_longopts[i_index].name = p_module->p_config[i].psz_name;
+            p_longopts[i_index].has_arg =
+                (p_module->p_config[i].i_type == MODULE_CONFIG_ITEM_BOOL)?
+                                               no_argument : required_argument;
+            p_longopts[i_index].flag = 0;
+            p_longopts[i_index].val = 0;
+            i_index++;
+        }
+    }
+    /* Close the longopts structure */
+    memset( &p_longopts[i_index], 0, sizeof(struct option) );
+
+
+    /*
+     * Parse the command line options
+     */
     opterr = 0;
-    while( ( i_cmd = getopt_long( *pi_argc, ppsz_argv,
-                                   psz_shortopts, longopts, 0 ) ) != EOF )
+    optind = 1;
+    while( ( i_cmd = getopt_long( *pi_argc, ppsz_argv, psz_shortopts,
+                                  p_longopts, &i_index ) ) != EOF )
     {
-        switch( i_cmd )
+
+        if( i_cmd == 0 )
         {
-        /* General/common options */
-        case 'h':                                              /* -h, --help */
-            ShowConsole();
-            RedirectSTDOUT();
-            Usage( SHORT_HELP );
-#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
-            if( strcmp( "", main_GetPszVariable( INTF_STDOUT_VAR,
-                                                 INTF_STDOUT_DEFAULT ) ) == 0 )
-            {
-                /* No stdout redirection has been asked for */
-                intf_MsgImm( "\nPress the RETURN key to continue..." );
-                getchar();
-            }
-#endif
-            return( -1 );
-            break;
-        case 'H':                                          /* -H, --longhelp */
-            ShowConsole();
-            RedirectSTDOUT();
-            Usage( LONG_HELP );
-#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
-            if( strcmp( "", main_GetPszVariable( INTF_STDOUT_VAR,
-                                                 INTF_STDOUT_DEFAULT ) ) == 0 )
-            {
-                /* No stdout redirection has been asked for */
-                intf_MsgImm( "\nPress the RETURN key to continue..." );
-                getchar();
-            }
-#endif
-            return( -1 );
-            break;
-        case OPT_VERSION:                                       /* --version */
-            ShowConsole();
-            RedirectSTDOUT();
-            Version();
-#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
-            if( strcmp( "", main_GetPszVariable( INTF_STDOUT_VAR,
-                                                 INTF_STDOUT_DEFAULT ) ) == 0 )
-            {
-                /* No stdout redirection has been asked for */
-                intf_MsgImm( "\nPress the RETURN key to continue..." );
-                getchar();
-            }
-#endif
-            return( -1 );
-            break;
-        case 'v':                                           /* -v, --verbose */
-            p_main->i_warning_level++;
-            break;
+            /* A long option has been recognized */
 
-        /* Interface warning messages level */
-        case 'I':                                              /* -I, --intf */
-            main_PutPszVariable( INTF_METHOD_VAR, optarg );
-            break;
-        case OPT_WARNING:                                       /* --warning */
-            intf_ErrMsg( "intf error: `--warning' is deprecated, use `-v'" );
-            p_main->i_warning_level = atoi(optarg);
-            break;
+            module_config_t *p_conf;
 
-        case OPT_STDOUT:                                         /* --stdout */
-            main_PutPszVariable( INTF_STDOUT_VAR, optarg );
-            break;
+            /* Store the configuration option */
+            p_conf = config_FindConfig( p_longopts[i_index].name );
 
-        /* Audio options */
-        case OPT_NOAUDIO:                                       /* --noaudio */
-            p_main->b_audio = 0;
-            break;
-        case 'A':                                              /* -A, --aout */
-            main_PutPszVariable( AOUT_METHOD_VAR, optarg );
-            break;
-        case OPT_STEREO:                                         /* --stereo */
-            main_PutIntVariable( AOUT_STEREO_VAR, 1 );
-            break;
-        case OPT_MONO:                                             /* --mono */
-            main_PutIntVariable( AOUT_STEREO_VAR, 0 );
-            break;
-        case OPT_SPDIF:                                           /* --spdif */
-            main_PutIntVariable( AOUT_SPDIF_VAR, 1 );
-            break;
-        case OPT_DOWNMIX:                                       /* --downmix */
-            main_PutPszVariable( DOWNMIX_METHOD_VAR, optarg );
-            break;
-        case OPT_IMDCT:                                           /* --imdct */
-            main_PutPszVariable( IMDCT_METHOD_VAR, optarg );
-            break;
+            switch( p_conf->i_type )
+            {
+            case MODULE_CONFIG_ITEM_STRING:
+            case MODULE_CONFIG_ITEM_FILE:
+            case MODULE_CONFIG_ITEM_PLUGIN:
+                config_PutPszVariable( p_longopts[i_index].name, optarg );
+                break;
+            case MODULE_CONFIG_ITEM_INTEGER:
+                config_PutIntVariable( p_longopts[i_index].name, atoi(optarg));
+                break;
+            case MODULE_CONFIG_ITEM_BOOL:
+                config_PutIntVariable( p_longopts[i_index].name, 1 );
+                break;
+            }
 
-        /* Video options */
-        case OPT_NOVIDEO:                                       /* --novideo */
-            p_main->b_video = 0;
-            break;
-        case 'V':                                              /* -V, --vout */
-            main_PutPszVariable( VOUT_METHOD_VAR, optarg );
-            break;
-        case OPT_DISPLAY:                                       /* --display */
-            main_PutPszVariable( VOUT_DISPLAY_VAR, optarg );
-            break;
-        case OPT_WIDTH:                                           /* --width */
-            main_PutPszVariable( VOUT_WIDTH_VAR, optarg );
-            break;
-        case OPT_HEIGHT:                                         /* --height */
-            main_PutPszVariable( VOUT_HEIGHT_VAR, optarg );
-            break;
-        case 'g':                                         /* -g, --grayscale */
-            main_PutIntVariable( VOUT_GRAYSCALE_VAR, 1 );
-            break;
-        case OPT_COLOR:                                           /* --color */
-            main_PutIntVariable( VOUT_GRAYSCALE_VAR, 0 );
-            break;
-        case OPT_FULLSCREEN:                                 /* --fullscreen */
-            main_PutIntVariable( VOUT_FULLSCREEN_VAR, 1 );
-            break;
-        case OPT_OVERLAY:                                       /* --overlay */
-            main_PutIntVariable( VOUT_OVERLAY_VAR, 1 );
-            break;
-        case OPT_XVADAPTOR:                                   /* --xvadaptor */
-            main_PutIntVariable( VOUT_XVADAPTOR_VAR, atoi(optarg) );
-            break;
-        case OPT_MOTION:                                         /* --motion */
-            main_PutPszVariable( MOTION_METHOD_VAR, optarg );
-            break;
-        case OPT_IDCT:                                             /* --idct */
-            main_PutPszVariable( IDCT_METHOD_VAR, optarg );
-            break;
-        case OPT_YUV:                                               /* --yuv */
-            main_PutPszVariable( YUV_METHOD_VAR, optarg );
-            break;
-        case OPT_SMP:                                               /* --smp */
-            main_PutIntVariable( VDEC_SMP_VAR, atoi(optarg) );
-            break;
+            continue;
+        }
 
-        /* DVD options */
-        case 't':
-            main_PutIntVariable( INPUT_TITLE_VAR, atoi(optarg) );
-            break;
-        case 'T':
-            main_PutIntVariable( INPUT_CHAPTER_VAR, atoi(optarg) );
-            break;
-        case 'u':
-            main_PutIntVariable( INPUT_ANGLE_VAR, atoi(optarg) );
-            break;
-        case 'a':
-            if ( ! strcmp(optarg, "ac3") )
-                main_PutIntVariable( INPUT_AUDIO_VAR, REQUESTED_AC3 );
-            else if ( ! strcmp(optarg, "lpcm") )
-                main_PutIntVariable( INPUT_AUDIO_VAR, REQUESTED_LPCM );
-            else if ( ! strcmp(optarg, "mpeg") )
-                main_PutIntVariable( INPUT_AUDIO_VAR, REQUESTED_MPEG );
-            else
-                main_PutIntVariable( INPUT_AUDIO_VAR, REQUESTED_NOAUDIO );
-            break;
-        case 'c':
-            main_PutIntVariable( INPUT_CHANNEL_VAR, atoi(optarg) );
-            break;
-        case 's':
-            main_PutIntVariable( INPUT_SUBTITLE_VAR, atoi(optarg) );
-            break;
+        /* short options handled here for now */
+        switch( i_cmd )
+        {
 
-        /* Input options */
-        case OPT_INPUT:                                           /* --input */
-            main_PutPszVariable( INPUT_METHOD_VAR, optarg );
-            break;
-        case OPT_CHANNELS:                                     /* --channels */
-            main_PutIntVariable( INPUT_NETWORK_CHANNEL_VAR, 1 );
+        /* General/common options */
+        case 'h':                                              /* -h, --help */
+            config_PutIntVariable( "help", 1 );
             break;
-        case OPT_SERVER:                                         /* --server */
-            main_PutPszVariable( INPUT_SERVER_VAR, optarg );
+        case 'H':                                          /* -H, --longhelp */
+            config_PutIntVariable( "longhelp", 1 );
             break;
-        case OPT_PORT:                                             /* --port */
-            main_PutPszVariable( INPUT_PORT_VAR, optarg );
+        case 'l':                                              /* -l, --list */
+            config_PutIntVariable( "list", 1 );
             break;
-        case OPT_BROADCAST:                                   /* --broadcast */
-            main_PutIntVariable( INPUT_BROADCAST_VAR, 1 );
-            main_PutPszVariable( INPUT_BCAST_ADDR_VAR, optarg );
+        case 'v':                                           /* -v, --verbose */
+            p_main->i_warning_level++;
             break;
 
-        /* Synchro options */
-        case OPT_SYNCHRO:                                      
-            main_PutPszVariable( VPAR_SYNCHRO_VAR, optarg );
-            break;
-            
         /* Internal error: unknown option */
         case '?':
         default:
-            ShowConsole();
-            RedirectSTDOUT();
-            intf_ErrMsg( "intf error: unknown option `%s'",
-                         ppsz_argv[optind - 1] );
-            Usage( USAGE );
-#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
-            if( strcmp( "", main_GetPszVariable( INTF_STDOUT_VAR,
-                                                 INTF_STDOUT_DEFAULT ) ) == 0 )
+            if( !b_ignore_errors )
             {
-                /* No stdout redirection has been asked for */
-                intf_MsgImm( "\nPress the RETURN key to continue..." );
+                intf_ErrMsg( "intf error: unknown option `%s'",
+                             ppsz_argv[optind] );
+                intf_Msg( "Try `%s --help' for more information.\n",
+                          p_main->psz_arg0 );
+
+#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
+                intf_Msg( "\nPress the RETURN key to continue..." );
                 getchar();
-            }
 #endif
-            return( EINVAL );
-            break;
+                free( p_longopts );
+                return( EINVAL );
+                break;
+            }
         }
+
     }
 
     if( p_main->i_warning_level < 0 )
@@ -785,6 +777,7 @@ static int GetConfiguration( int *pi_argc, char *ppsz_argv[], char *ppsz_env[] )
         p_main->i_warning_level = 0;
     }
 
+    free( p_longopts );
     return( 0 );
 }
 
@@ -812,120 +805,159 @@ static int GetFilenames( int i_argc, char *ppsz_argv[] )
  *****************************************************************************
  * Print a short inline help. Message interface is initialized at this stage.
  *****************************************************************************/
-static void Usage( int i_fashion )
+static void Usage( const char *psz_module_name )
 {
+    int i;
+    module_t *p_module;
+    char psz_spaces[30];
+
+    memset( psz_spaces, 32, 30 );
+
+#ifdef WIN32
+    ShowConsole();
+#endif
+
     /* Usage */
-    intf_MsgImm( "Usage: %s [options] [parameters] [file]...",
-                 p_main->psz_arg0 );
+    intf_Msg( "Usage: %s [options] [parameters] [file]...\n",
+              p_main->psz_arg0 );
 
-    if( i_fashion == USAGE )
+    /* Enumerate the config of each module */
+    for( p_module = p_module_bank->first ;
+         p_module != NULL ;
+         p_module = p_module->next )
     {
-        intf_MsgImm( "Try `%s --help' for more information.",
-                     p_main->psz_arg0 );
-        return;
+
+        if( psz_module_name && strcmp( psz_module_name, p_module->psz_name ) )
+            continue;
+
+        /* print module name */
+        intf_Msg( "%s configuration:\n", p_module->psz_name );
+
+        for( i = 0; i < (p_module->i_config_options -1); i++ )
+        {
+            int j;
+
+            switch( p_module->p_config[i].i_type )
+            {
+            case MODULE_CONFIG_ITEM_CATEGORY:
+                intf_Msg( " %s", p_module->p_config[i].psz_text );
+                break;
+
+            case MODULE_CONFIG_ITEM_STRING:
+            case MODULE_CONFIG_ITEM_FILE:
+            case MODULE_CONFIG_ITEM_PLUGIN:
+                /* Nasty hack, but right now I'm too tired to think about
+                 * a nice solution */
+                j = 25 - strlen( p_module->p_config[i].psz_name )
+                    - strlen(" <string>") - 1;
+                if( j < 0 ) j = 0; psz_spaces[j] = 0;
+
+                intf_Msg( "  --%s <string>%s %s",
+                          p_module->p_config[i].psz_name, psz_spaces,
+                          p_module->p_config[i].psz_text );
+                psz_spaces[j] = 32;
+                break;
+            case MODULE_CONFIG_ITEM_INTEGER:
+                /* Nasty hack, but right now I'm too tired to think about
+                 * a nice solution */
+                j = 25 - strlen( p_module->p_config[i].psz_name )
+                    - strlen(" <integer>") - 1;
+                if( j < 0 ) j = 0; psz_spaces[j] = 0;
+
+                intf_Msg( "  --%s <integer>%s %s",
+                          p_module->p_config[i].psz_name, psz_spaces,
+                          p_module->p_config[i].psz_text );
+                psz_spaces[j] = 32;
+                break;
+            default:
+                /* Nasty hack, but right now I'm too tired to think about
+                 * a nice solution */
+                j = 25 - strlen( p_module->p_config[i].psz_name ) - 1;
+                if( j < 0 ) j = 0; psz_spaces[j] = 0;
+
+                intf_Msg( "  --%s%s %s",
+                          p_module->p_config[i].psz_name, psz_spaces,
+                          p_module->p_config[i].psz_text );
+                psz_spaces[j] = 32;
+                break;
+            }
+        }
+
+        /* Yet another nasty hack.
+         * Maybe we could use MODULE_CONFIG_ITEM_END to display tail messages
+         * for each module?? */
+        if( !strcmp( "main", p_module->psz_name ) )
+            intf_Msg( "\nPlaylist items:"
+                "\n  *.mpg, *.vob                   \tPlain MPEG-1/2 files"
+                "\n  [dvd:][device][@raw_device][@[title][,[chapter][,angle]]]"
+                "\n                                 \tDVD device"
+                "\n  vcd:<device>                   \tVCD device"
+                "\n  udpstream:[<server>[:<server port>]][@[<bind address>]"
+                      "[:<bind port>]]"
+                "\n                                 \tUDP stream sent by VLS"
+                "\n  vlc:loop                       \tLoop execution of the "
+                      "playlist"
+                "\n  vlc:pause                      \tPause execution of "
+                      "playlist items"
+                "\n  vlc:quit                       \tQuit VLC" );
+
+        intf_Msg( "" );
+
     }
 
-    /* Options */
-    intf_MsgImm( "\nOptions:"
-          "\n  -I, --intf <module>            \tinterface method"
-          "\n  -v, --verbose                  \tverbose mode (cumulative)"
-          "\n      --stdout <filename>        \tredirect console stdout"
-          "\n"
-          "\n      --noaudio                  \tdisable audio"
-          "\n  -A, --aout <module>            \taudio output method"
-          "\n      --stereo, --mono           \tstereo/mono audio"
-          "\n      --spdif                    \tAC3 pass-through mode"
-          "\n      --downmix <module>         \tAC3 downmix method"
-          "\n      --imdct <module>           \tAC3 IMDCT method"
-          "\n"
-          "\n      --novideo                  \tdisable video"
-          "\n  -V, --vout <module>            \tvideo output method"
-          "\n      --display <display>        \tdisplay string"
-          "\n      --width <w>, --height <h>  \tdisplay dimensions"
-          "\n  -g, --grayscale                \tgrayscale output"
-          "\n      --fullscreen               \tfullscreen output"
-          "\n      --overlay                  \taccelerated display"
-          "\n      --xvadaptor <adaptor>      \tXVideo adaptor"
-          "\n      --color                    \tcolor output"
-          "\n      --motion <module>          \tmotion compensation method"
-          "\n      --idct <module>            \tIDCT method"
-          "\n      --yuv <module>             \tYUV method"
-          "\n      --synchro <type>           \tforce synchro algorithm"
-          "\n      --smp <number of threads>  \tuse several processors"
-          "\n"
-          "\n  -t, --dvdtitle <num>           \tchoose DVD title"
-          "\n  -T, --dvdchapter <num>         \tchoose DVD chapter"
-          "\n  -u, --dvdangle <num>           \tchoose DVD angle"
-          "\n  -a, --dvdaudio <type>          \tchoose DVD audio type"
-          "\n  -c, --dvdchannel <channel>     \tchoose DVD audio channel"
-          "\n  -s, --dvdsubtitle <channel>    \tchoose DVD subtitle channel"
-          "\n"
-          "\n      --input                    \tinput method"
-          "\n      --channels                 \tenable channels"
-          "\n      --server <host>            \tvideo server address"
-          "\n      --port <port>              \tvideo server port"
-          "\n      --broadcast                \tlisten to a broadcast"
-          "\n"
-          "\n  -h, --help                     \tprint help and exit"
-          "\n  -H, --longhelp                 \tprint long help and exit"
-          "\n      --version                  \toutput version information and exit" );
-
-    if( i_fashion == SHORT_HELP )
-        return;
-
-    /* Interface parameters */
-    intf_MsgImm( "\nInterface parameters:"
-        "\n  " INTF_METHOD_VAR "=<method name>        \tinterface method"
-        "\n  " INTF_INIT_SCRIPT_VAR "=<filename>              \tinitialization script"
-        "\n  " INTF_CHANNELS_VAR "=<filename>         \tchannels list"
-        "\n  " INTF_STDOUT_VAR "=<filename>           \tredirect console stdout" );
-
-    /* Audio parameters */
-    intf_MsgImm( "\nAudio parameters:"
-        "\n  " AOUT_METHOD_VAR "=<method name>        \taudio method"
-        "\n  " AOUT_DSP_VAR "=<filename>              \tdsp device path"
-        "\n  " AOUT_STEREO_VAR "={1|0}                \tstereo or mono output"
-        "\n  " AOUT_SPDIF_VAR "={1|0}                 \tAC3 pass-through mode"
-        "\n  " DOWNMIX_METHOD_VAR "=<method name>     \tAC3 downmix method"
-        "\n  " IMDCT_METHOD_VAR "=<method name>       \tAC3 IMDCT method"
-        "\n  " AOUT_RATE_VAR "=<rate>             \toutput rate" );
-
-    /* Video parameters */
-    intf_MsgImm( "\nVideo parameters:"
-        "\n  " VOUT_METHOD_VAR "=<method name>        \tdisplay method"
-        "\n  " VOUT_DISPLAY_VAR "=<display name>      \tdisplay used"
-        "\n  " VOUT_WIDTH_VAR "=<width>               \tdisplay width"
-        "\n  " VOUT_HEIGHT_VAR "=<height>             \tdislay height"
-        "\n  " VOUT_FB_DEV_VAR "=<filename>           \tframebuffer device path"
-        "\n  " VOUT_GRAYSCALE_VAR "={1|0}             \tgrayscale or color output"
-        "\n  " VOUT_FULLSCREEN_VAR "={1|0}            \tfullscreen"
-        "\n  " VOUT_OVERLAY_VAR "={1|0}               \toverlay"
-        "\n  " VOUT_XVADAPTOR_VAR "=<adaptor>         \tXVideo adaptor"
-        "\n  " MOTION_METHOD_VAR "=<method name>      \tmotion compensation method"
-        "\n  " IDCT_METHOD_VAR "=<method name>        \tIDCT method"
-        "\n  " YUV_METHOD_VAR "=<method name>         \tYUV method"
-        "\n  " VPAR_SYNCHRO_VAR "={I|I+|IP|IP+|IPB}   \tsynchro algorithm"
-        "\n  " VDEC_SMP_VAR "=<number of threads>     \tuse several processors" );
-
-    /* DVD parameters */
-    intf_MsgImm( "\nDVD parameters:"
-        "\n  " INPUT_DVD_DEVICE_VAR "=<device>           \tDVD device"
-        "\n  " INPUT_TITLE_VAR "=<title>             \ttitle number"
-        "\n  " INPUT_CHAPTER_VAR "=<chapter>         \tchapter number"
-        "\n  " INPUT_ANGLE_VAR "=<angle>             \tangle number"
-        "\n  " INPUT_AUDIO_VAR "={ac3|lpcm|mpeg|off} \taudio type"
-        "\n  " INPUT_CHANNEL_VAR "=[0-15]            \taudio channel"
-        "\n  " INPUT_SUBTITLE_VAR "=[0-31]           \tsubtitle channel" );
-
-    /* Input parameters */
-    intf_MsgImm( "\nInput parameters:"
-        "\n  " INPUT_SERVER_VAR "=<hostname>         \tvideo server"
-        "\n  " INPUT_PORT_VAR "=<port>               \tvideo server port"
-        "\n  " INPUT_IFACE_VAR "=<interface>         \tnetwork interface"
-        "\n  " INPUT_BCAST_ADDR_VAR "=<addr>         \tbroadcast mode"
-        "\n  " INPUT_CHANNEL_SERVER_VAR "=<hostname> \tchannel server"
-        "\n  " INPUT_CHANNEL_PORT_VAR "=<port>       \tchannel server port" );
+#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
+        intf_Msg( "\nPress the RETURN key to continue..." );
+        getchar();
+#endif
+}
+
+/*****************************************************************************
+ * ListModules: list the available modules with their description
+ *****************************************************************************
+ * Print a list of all available modules (builtins and plugins) and a short
+ * description for each one.
+ *****************************************************************************/
+static void ListModules( void )
+{
+    module_t *p_module;
+    char psz_spaces[20];
+
+    memset( psz_spaces, 32, 20 );
+
+#ifdef WIN32
+    ShowConsole();
+#endif
+
+    /* Usage */
+    intf_Msg( "Usage: %s [options] [parameters] [file]...\n",
+              p_main->psz_arg0 );
 
+    intf_Msg( "[plugin]              [description]" );
+
+    /* Enumerate each module */
+    for( p_module = p_module_bank->first ;
+         p_module != NULL ;
+         p_module = p_module->next )
+    {
+        int i;
+
+        /* Nasty hack, but right now I'm too tired to think about a nice
+         * solution */
+        i = 20 - strlen( p_module->psz_name ) - 1;
+        if( i < 0 ) i = 0;
+        psz_spaces[i] = 0;
+
+        intf_Msg( "  %s%s %s", p_module->psz_name, psz_spaces,
+                  p_module->psz_longname );
+
+        psz_spaces[i] = 32;
+
+    }
+
+#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
+        intf_Msg( "\nPress the RETURN key to continue..." );
+        getchar();
+#endif
 }
 
 /*****************************************************************************
@@ -935,11 +967,18 @@ static void Usage( int i_fashion )
  *****************************************************************************/
 static void Version( void )
 {
-    intf_MsgImm( VERSION_MESSAGE
+#ifdef WIN32
+    ShowConsole();
+#endif
+    intf_Msg( VERSION_MESSAGE
         "This program comes with NO WARRANTY, to the extent permitted by law.\n"
         "You may redistribute it under the terms of the GNU General Public License;\n"
         "see the file named COPYING for details.\n"
         "Written by the VideoLAN team at Ecole Centrale, Paris." );
+#ifdef WIN32        /* Pause the console because it's destroyed when we exit */
+        intf_Msg( "\nPress the RETURN key to continue..." );
+        getchar();
+#endif
 }
 
 /*****************************************************************************
@@ -963,7 +1002,6 @@ static void InitSignalHandler( void )
 #endif
 }
 
-
 /*****************************************************************************
  * SimpleSignalHandler: system signal handler
  *****************************************************************************
@@ -975,7 +1013,6 @@ static void SimpleSignalHandler( int i_signal )
     intf_WarnMsg( 0, "intf: ignoring signal %d", i_signal );
 }
 
-
 /*****************************************************************************
  * FatalSignalHandler: system signal handler
  *****************************************************************************
@@ -994,7 +1031,7 @@ static void FatalSignalHandler( int i_signal )
 #endif
 
     /* Acknowledge the signal received */
-    intf_ErrMsgImm( "intf error: signal %d received, exiting", i_signal );
+    intf_ErrMsg( "intf error: signal %d received, exiting", i_signal );
 
     /* Try to terminate everything - this is done by requesting the end of the
      * interface thread */
@@ -1002,30 +1039,28 @@ static void FatalSignalHandler( int i_signal )
 }
 
 /*****************************************************************************
- * InstructionSignalHandler: system signal handler
+ * IllegalSignalHandler: system signal handler
  *****************************************************************************
- * This function is called when a illegal instruction signal is received by
- * the program.
- * We use this function to test OS and CPU_Capabilities
+ * This function is called when an illegal instruction signal is received by
+ * the program. We use this function to test OS and CPU capabilities
  *****************************************************************************/
-static void InstructionSignalHandler( int i_signal )
+static void IllegalSignalHandler( int i_signal )
 {
-    /* Once a signal has been trapped, the termination sequence will be
-     * armed and following signals will be ignored to avoid sending messages
-     * to an interface having been destroyed */
-
     /* Acknowledge the signal received */
-    fprintf( stderr, "warning: extended instructions unsupported, "
-                     "some optimizations will be disabled\n" );
-#ifdef SYS_LINUX
-    fprintf( stderr, "upgrade to kernel 2.4.x to get rid of this warning\n" );
-#endif
-
     i_illegal = 1;
-    
+
 #ifdef HAVE_SIGRELSE
     sigrelse( i_signal );
 #endif
+
+    fprintf( stderr, "warning: your CPU has %s instructions, but not your "
+                     "operating system.\n", psz_capability );
+    fprintf( stderr, "         some optimizations will be disabled unless "
+                     "you upgrade your OS\n" );
+#ifdef SYS_LINUX
+    fprintf( stderr, "         (for instance Linux kernel 2.4.x or later)" );
+#endif
+
     longjmp( env, 1 );
 }
 
@@ -1034,18 +1069,11 @@ static void InstructionSignalHandler( int i_signal )
  *****************************************************************************
  * This function is called to list extensions the CPU may have.
  *****************************************************************************/
-static int CPUCapabilities( void )
+static u32 CPUCapabilities( void )
 {
-    volatile int i_capabilities = CPU_CAPABILITY_NONE;
-
-#if defined( SYS_BEOS )
-    i_capabilities |= CPU_CAPABILITY_486
-                      | CPU_CAPABILITY_586
-                      | CPU_CAPABILITY_MMX;
+    volatile u32 i_capabilities = CPU_CAPABILITY_NONE;
 
-    return( i_capabilities );
-
-#elif defined( SYS_DARWIN )
+#if defined( SYS_DARWIN )
     struct host_basic_info hi;
     kern_return_t          ret;
     host_name_port_t       host;
@@ -1053,6 +1081,8 @@ static int CPUCapabilities( void )
     int i_size;
     char *psz_name, *psz_subname;
 
+    i_capabilities |= CPU_CAPABILITY_FPU;
+
     /* Should 'never' fail? */
     host = mach_host_self();
 
@@ -1079,25 +1109,47 @@ static int CPUCapabilities( void )
     volatile unsigned int  i_eax, i_ebx, i_ecx, i_edx;
     volatile boolean_t     b_amd;
 
-    signal( SIGILL, InstructionSignalHandler );
-    
+    /* Needed for x86 CPU capabilities detection */
+#   define cpuid( a )                      \
+        asm volatile ( "pushl %%ebx\n\t"   \
+                       "cpuid\n\t"         \
+                       "movl %%ebx,%1\n\t" \
+                       "popl %%ebx\n\t"    \
+                     : "=a" ( i_eax ),     \
+                       "=r" ( i_ebx ),     \
+                       "=c" ( i_ecx ),     \
+                       "=d" ( i_edx )      \
+                     : "a"  ( a )          \
+                     : "cc" );
+
+    i_capabilities |= CPU_CAPABILITY_FPU;
+
+#   if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+    signal( SIGILL, IllegalSignalHandler );
+#   endif
+
     /* test for a 486 CPU */
-    asm volatile ( "pushfl\n\t"
+    asm volatile ( "pushl %%ebx\n\t"
+                   "pushfl\n\t"
                    "popl %%eax\n\t"
                    "movl %%eax, %%ebx\n\t"
                    "xorl $0x200000, %%eax\n\t"
                    "pushl %%eax\n\t"
                    "popfl\n\t"
                    "pushfl\n\t"
-                   "popl %%eax"
+                   "popl %%eax\n\t"
+                   "movl %%ebx,%1\n\t"
+                   "popl %%ebx\n\t"
                  : "=a" ( i_eax ),
-                   "=b" ( i_ebx )
+                   "=r" ( i_ebx )
                  :
                  : "cc" );
 
     if( i_eax == i_ebx )
     {
-        signal( SIGILL, NULL );     
+#   if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+        signal( SIGILL, NULL );
+#   endif
         return( i_capabilities );
     }
 
@@ -1108,7 +1160,9 @@ static int CPUCapabilities( void )
 
     if( !i_eax )
     {
-        signal( SIGILL, NULL );     
+#   if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+        signal( SIGILL, NULL );
+#   endif
         return( i_capabilities );
     }
 
@@ -1124,7 +1178,9 @@ static int CPUCapabilities( void )
 
     if( ! (i_edx & 0x00800000) )
     {
-        signal( SIGILL, NULL );     
+#   if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+        signal( SIGILL, NULL );
+#   endif
         return( i_capabilities );
     }
 
@@ -1134,7 +1190,9 @@ static int CPUCapabilities( void )
     {
         i_capabilities |= CPU_CAPABILITY_MMXEXT;
 
+#   ifdef CAN_COMPILE_SSE
         /* We test if OS support the SSE instructions */
+        psz_capability = "SSE";
         i_illegal = 0;
         if( setjmp( env ) == 0 )
         {
@@ -1146,22 +1204,27 @@ static int CPUCapabilities( void )
         {
             i_capabilities |= CPU_CAPABILITY_SSE;
         }
+#   endif
     }
-    
+
     /* test for additional capabilities */
     cpuid( 0x80000000 );
 
     if( i_eax < 0x80000001 )
     {
-        signal( SIGILL, NULL );     
+#   if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+        signal( SIGILL, NULL );
+#   endif
         return( i_capabilities );
     }
 
     /* list these additional capabilities */
     cpuid( 0x80000001 );
 
+#   ifdef CAN_COMPILE_3DNOW
     if( i_edx & 0x80000000 )
     {
+        psz_capability = "3D Now!";
         i_illegal = 0;
         if( setjmp( env ) == 0 )
         {
@@ -1169,28 +1232,38 @@ static int CPUCapabilities( void )
             __asm__ __volatile__ ( "pfadd %%mm0,%%mm0\n" "femms\n" : : );
         }
 
-        if( i_illegal == 0 ) 
+        if( i_illegal == 0 )
         {
             i_capabilities |= CPU_CAPABILITY_3DNOW;
         }
     }
+#   endif
 
     if( b_amd && ( i_edx & 0x00400000 ) )
     {
         i_capabilities |= CPU_CAPABILITY_MMXEXT;
     }
 
-    signal( SIGILL, NULL );     
+#   if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+    signal( SIGILL, NULL );
+#   endif
     return( i_capabilities );
 
 #elif defined( __powerpc__ )
-    /* Test for Altivec */
-    signal( SIGILL, InstructionSignalHandler );
 
+    i_capabilities |= CPU_CAPABILITY_FPU;
+
+#   ifdef CAN_COMPILE_ALTIVEC
+    signal( SIGILL, IllegalSignalHandler );
+
+    psz_capability = "AltiVec";
     i_illegal = 0;
     if( setjmp( env ) == 0 )
     {
-        asm volatile ("mtspr 256,%0" : : "r" (-1));
+        asm volatile ("mtspr 256, %0\n\t"
+                      "vand %%v0, %%v0, %%v0"
+                      :
+                      : "r" (-1));
     }
 
     if( i_illegal == 0 )
@@ -1198,7 +1271,8 @@ static int CPUCapabilities( void )
         i_capabilities |= CPU_CAPABILITY_ALTIVEC;
     }
 
-    signal( SIGILL, NULL );     
+    signal( SIGILL, NULL );
+#   endif
 
     return( i_capabilities );
 
@@ -1209,64 +1283,18 @@ static int CPUCapabilities( void )
 #endif
 }
 
-/*****************************************************************************
- * RedirectSTDOUT: redirect stdout and stderr to a file
- *****************************************************************************
- * This function will redirect stdout and stderr to a file if the user has
- * specified so.
- *****************************************************************************/
-static int RedirectSTDOUT( void )
-{
-    int  i_stdout_filedesc;
-    char *psz_stdout_filename;
-
-    psz_stdout_filename = main_GetPszVariable( INTF_STDOUT_VAR,
-                                               INTF_STDOUT_DEFAULT );
-    if( strcmp( "", psz_stdout_filename ) != 0 )
-    {
-        ShowConsole();
-        i_stdout_filedesc = open( psz_stdout_filename,
-                                  O_CREAT | O_TRUNC | O_RDWR,
-                                  S_IREAD | S_IWRITE );
-
-        if( dup2( i_stdout_filedesc, fileno(stdout) ) == -1 )
-        {
-            intf_ErrMsg( "warning: unable to redirect stdout" );
-        }
-
-        if( dup2( i_stdout_filedesc, fileno(stderr) ) == -1 )
-        {
-            intf_ErrMsg( "warning: unable to redirect stderr" );
-        }
-
-        close( i_stdout_filedesc );
-    }
-    else
-    {
-        /* No stdout redirection has been asked so open a console */
-        if( p_main->i_warning_level )
-        {
-            ShowConsole();
-        }
-
-    }
-
-    return 0;
-}
-
 /*****************************************************************************
  * ShowConsole: On Win32, create an output console for debug messages
  *****************************************************************************
  * This function is usefull only on Win32.
  *****************************************************************************/
+#ifdef WIN32 /*  */
 static void ShowConsole( void )
 {
-#ifdef WIN32 /*  */
     AllocConsole();
     freopen( "CONOUT$", "w", stdout );
     freopen( "CONOUT$", "w", stderr );
     freopen( "CONIN$", "r", stdin );
-#endif
     return;
 }
-
+#endif