* 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
{
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
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 );
}
/*
/*
* 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
/*
* Terminate messages interface and program
*/
- intf_Msg( "intf: program terminated" );
+ intf_WarnMsg( 1, "intf: program terminated" );
intf_MsgDestroy();
/*
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)
}
#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 )
p_main->i_warning_level = 0;
}
+ free( p_longopts );
return( 0 );
}
*****************************************************************************
* 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
}
/*****************************************************************************
*****************************************************************************/
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
}
/*****************************************************************************
#endif
}
-
/*****************************************************************************
* SimpleSignalHandler: system signal handler
*****************************************************************************
intf_WarnMsg( 0, "intf: ignoring signal %d", i_signal );
}
-
/*****************************************************************************
* FatalSignalHandler: system signal handler
*****************************************************************************
#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 */
}
/*****************************************************************************
- * 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 );
}
*****************************************************************************
* 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;
int i_size;
char *psz_name, *psz_subname;
+ i_capabilities |= CPU_CAPABILITY_FPU;
+
/* Should 'never' fail? */
host = mach_host_self();
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 );
}
if( !i_eax )
{
- signal( SIGILL, NULL );
+# if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+ signal( SIGILL, NULL );
+# endif
return( i_capabilities );
}
if( ! (i_edx & 0x00800000) )
{
- signal( SIGILL, NULL );
+# if defined( CAN_COMPILE_SSE ) || defined ( CAN_COMPILE_3DNOW )
+ signal( SIGILL, NULL );
+# endif
return( i_capabilities );
}
{
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 )
{
{
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 )
{
__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 )
i_capabilities |= CPU_CAPABILITY_ALTIVEC;
}
- signal( SIGILL, NULL );
+ signal( SIGILL, NULL );
+# endif
return( i_capabilities );
#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