libvlc.c 77.7 KB
Newer Older
1 2 3
/*****************************************************************************
 * libvlc.c: main libvlc source
 *****************************************************************************
4
 * Copyright (C) 1998-2006 the VideoLAN team
5
 * $Id$
6 7 8
 *
 * Authors: Vincent Seguin <seguin@via.ecp.fr>
 *          Samuel Hocevar <sam@zoy.org>
9
 *          Gildas Bazin <gbazin@videolan.org>
10
 *          Derk-Jan Hartman <hartman at videolan dot org>
11
 *          Rémi Denis-Courmont <rem # videolan : org>
12 13 14 15 16 17 18 19 20 21 22 23 24
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
Antoine Cellerier's avatar
Antoine Cellerier committed
25
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
26 27 28 29 30 31
 *****************************************************************************/

/*****************************************************************************
 * Pretend we are a builtin module
 *****************************************************************************/
#define MODULE_NAME main
32
#define MODULE_PATH main
33 34 35 36 37
#define __BUILTIN__

/*****************************************************************************
 * Preamble
 *****************************************************************************/
38
#include <vlc/vlc.h>
39
#include <vlc/input.h>
40

41
#include <errno.h>                                                 /* ENOMEM */
42 43 44 45 46 47 48 49 50 51
#include <stdio.h>                                              /* sprintf() */
#include <string.h>                                            /* strerror() */
#include <stdlib.h>                                                /* free() */

#ifndef WIN32
#   include <netinet/in.h>                            /* BSD: struct in_addr */
#endif

#ifdef HAVE_UNISTD_H
#   include <unistd.h>
Gildas Bazin's avatar
 
Gildas Bazin committed
52
#elif defined( WIN32 ) && !defined( UNDER_CE )
53 54 55
#   include <io.h>
#endif

56
#ifdef WIN32                       /* optind, getopt(), included in unistd.h */
57
#   include "extras/getopt.h"
58 59
#endif

60 61 62 63
#ifdef HAVE_LOCALE_H
#   include <locale.h>
#endif

64 65 66 67
#ifdef HAVE_HAL
#   include <hal/libhal.h>
#endif

68
#include "vlc_cpu.h"                                        /* CPU detection */
69
#include "os_specific.h"
70

71
#include "vlc_error.h"
72

73
#include "vlc_playlist.h"
74
#include "vlc_interface.h"
75 76 77

#include "audio_output.h"

78
#include "vlc_video.h"
79 80
#include "video_output.h"

81
#include "stream_output.h"
82
#include "charset.h"
83

84 85 86
#include "libvlc.h"

/*****************************************************************************
87
 * The evil global variable. We handle it with care, don't worry.
88
 *****************************************************************************/
89 90 91
static libvlc_t   libvlc;
static libvlc_t * p_libvlc;
static vlc_t *    p_static_vlc;
92 93 94 95

/*****************************************************************************
 * Local prototypes
 *****************************************************************************/
96 97 98 99
static int AddIntfInternal( int i_object, char const *psz_module,
                             vlc_bool_t b_block, vlc_bool_t b_play,
                             int i_options, char **ppsz_options );

100 101
void LocaleInit( vlc_object_t * );
void LocaleDeinit( void );
Sam Hocevar's avatar
Sam Hocevar committed
102
static void SetLanguage   ( char const * );
103
static int  GetFilenames  ( vlc_t *, int, char *[] );
104
static void Help          ( vlc_t *, char const *psz_help_name );
105
static void Usage         ( vlc_t *, char const *psz_module_name );
106
static void ListModules   ( vlc_t * );
107 108 109
static void Version       ( void );

#ifdef WIN32
110
static void ShowConsole   ( vlc_bool_t );
111
static void PauseConsole  ( void );
112
#endif
113
static int  ConsoleWidth  ( void );
114

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
115 116
static int  VerboseCallback( vlc_object_t *, char const *,
                             vlc_value_t, vlc_value_t, void * );
117

118 119
static void InitDeviceValues( vlc_t * );

120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
/*****************************************************************************
 * vlc_current_object: return the current object.
 *****************************************************************************
 * If i_object is non-zero, return the corresponding object. Otherwise,
 * return the statically allocated p_vlc object.
 *****************************************************************************/
vlc_t * vlc_current_object( int i_object )
{
    if( i_object )
    {
         return vlc_object_get( p_libvlc, i_object );
    }

    return p_static_vlc;
}

136
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
137
 * VLC_Version: return the libvlc version.
138
 *****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
139
 * This function returns full version string (numeric version and codename).
140
 *****************************************************************************/
141
char const * VLC_Version( void )
142
{
Sam Hocevar's avatar
Sam Hocevar committed
143
    return VERSION_MESSAGE;
144 145
}

146
/*****************************************************************************
147
 * VLC_CompileBy, VLC_CompileHost, VLC_CompileDomain,
148 149 150 151 152
 * VLC_Compiler, VLC_Changeset
 *****************************************************************************/
#define DECLARE_VLC_VERSION( func, var )                                    \
char const * VLC_##func ( void )                                            \
{                                                                           \
153
    return VLC_##var ;                                                      \
154 155
}

156 157
DECLARE_VLC_VERSION( CompileBy, COMPILE_BY );
DECLARE_VLC_VERSION( CompileHost, COMPILE_HOST );
Christophe Mutricy's avatar
Typo  
Christophe Mutricy committed
158
DECLARE_VLC_VERSION( CompileDomain, COMPILE_DOMAIN );
159 160 161 162 163 164 165
DECLARE_VLC_VERSION( Compiler, COMPILER );

extern const char psz_vlc_changeset[];
char const * VLC_Changeset( void )
{
    return psz_vlc_changeset;
}
166

167 168 169 170 171 172 173 174 175 176
/*****************************************************************************
 * VLC_Error: strerror() equivalent
 *****************************************************************************
 * This function returns full version string (numeric version and codename).
 *****************************************************************************/
char const * VLC_Error( int i_err )
{
    return vlc_error( i_err );
}

Sam Hocevar's avatar
Sam Hocevar committed
177 178 179 180 181 182 183
/*****************************************************************************
 * VLC_Create: allocate a vlc_t structure, and initialize libvlc if needed.
 *****************************************************************************
 * This function allocates a vlc_t structure and returns a negative value
 * in case of failure. Also, the thread system is initialized.
 *****************************************************************************/
int VLC_Create( void )
184
{
185
    int i_ret;
186
    vlc_t * p_vlc = NULL;
187
    vlc_value_t lockval;
188

189 190 191
    /* &libvlc never changes, so we can safely call this multiple times. */
    p_libvlc = &libvlc;

192 193
    /* vlc_threads_init *must* be the first internal call! No other call is
     * allowed before the thread system has been initialized. */
194
    i_ret = vlc_threads_init( p_libvlc );
Sam Hocevar's avatar
Sam Hocevar committed
195
    if( i_ret < 0 )
196
    {
Sam Hocevar's avatar
Sam Hocevar committed
197
        return i_ret;
198 199 200
    }

    /* Now that the thread system is initialized, we don't have much, but
201
     * at least we have var_Create */
202 203
    var_Create( p_libvlc, "libvlc", VLC_VAR_MUTEX );
    var_Get( p_libvlc, "libvlc", &lockval );
204
    vlc_mutex_lock( lockval.p_address );
205 206
    if( !libvlc.b_ready )
    {
207 208
        char *psz_env;

209 210 211
        /* Guess what CPU we have */
        libvlc.i_cpu = CPUCapabilities();

212 213
        /* Find verbosity from VLC_VERBOSE environment variable */
        psz_env = getenv( "VLC_VERBOSE" );
214
        libvlc.i_verbose = psz_env ? atoi( psz_env ) : -1;
215

Gildas Bazin's avatar
 
Gildas Bazin committed
216
#if defined( HAVE_ISATTY ) && !defined( WIN32 )
217 218 219 220 221
        libvlc.b_color = isatty( 2 ); /* 2 is for stderr */
#else
        libvlc.b_color = VLC_FALSE;
#endif

222
        /* Initialize message queue */
223
        msg_Create( p_libvlc );
224 225

        /* Announce who we are */
226 227
        msg_Dbg( p_libvlc, COPYRIGHT_MESSAGE );
        msg_Dbg( p_libvlc, "libvlc was configured with %s", CONFIGURE_LINE );
228

229 230
        /* The module bank will be initialized later */
        libvlc.p_module_bank = NULL;
231 232

        libvlc.b_ready = VLC_TRUE;
233
    }
234
    vlc_mutex_unlock( lockval.p_address );
235
    var_Destroy( p_libvlc, "libvlc" );
236 237

    /* Allocate a vlc object */
238
    p_vlc = vlc_object_create( p_libvlc, VLC_OBJECT_VLC );
239 240
    if( p_vlc == NULL )
    {
Sam Hocevar's avatar
Sam Hocevar committed
241
        return VLC_EGENERIC;
242
    }
243
    p_vlc->thread_id = 0;
244 245 246 247

    p_vlc->psz_object_name = "root";

    /* Initialize mutexes */
248
    vlc_mutex_init( p_vlc, &p_vlc->config_lock );
249
#ifdef __APPLE__
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
250
    vlc_mutex_init( p_vlc, &p_vlc->quicktime_lock );
251
    vlc_thread_set_priority( p_vlc, VLC_THREAD_PRIORITY_LOW );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
252
#endif
253 254

    /* Store our newly allocated structure in the global list */
255
    vlc_object_attach( p_vlc, p_libvlc );
256

Sam Hocevar's avatar
Sam Hocevar committed
257 258 259 260
    /* Store data for the non-reentrant API */
    p_static_vlc = p_vlc;

    return p_vlc->i_object_id;
261 262 263
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
264
 * VLC_Init: initialize a vlc_t structure.
265 266 267 268 269 270 271
 *****************************************************************************
 * This function initializes a previously allocated vlc_t structure:
 *  - CPU detection
 *  - gettext initialization
 *  - message queue, module bank and playlist initialization
 *  - configuration and commandline parsing
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
272
int VLC_Init( int i_object, int i_argc, char *ppsz_argv[] )
273
{
274 275
    char         p_capabilities[200];
    char *       p_tmp;
276 277
    char *       psz_modules;
    char *       psz_parser;
278
    char *       psz_control;
279
    vlc_bool_t   b_exit = VLC_FALSE;
280
    int          i_ret = VLC_EEXIT;
281
    vlc_t *      p_vlc = vlc_current_object( i_object );
282 283
    module_t    *p_help_module;
    playlist_t  *p_playlist;
284
    vlc_value_t  val;
285 286
#if defined( ENABLE_NLS ) \
     && ( defined( HAVE_GETTEXT ) || defined( HAVE_INCLUDED_GETTEXT ) )
287
    char *       psz_language;
288
#endif
289

290
    if( !p_vlc )
291
    {
292
        return VLC_ENOOBJ;
293 294 295
    }

    /*
296
     * System specific initialization code
297
     */
298
    system_Init( p_vlc, &i_argc, ppsz_argv );
299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314

    /* Get the executable name (similar to the basename command) */
    if( i_argc > 0 )
    {
        p_vlc->psz_object_name = p_tmp = ppsz_argv[ 0 ];
        while( *p_tmp )
        {
            if( *p_tmp == '/' ) p_vlc->psz_object_name = ++p_tmp;
            else ++p_tmp;
        }
    }
    else
    {
        p_vlc->psz_object_name = "vlc";
    }

315 316 317 318
    /*
     * Support for gettext
     */
    SetLanguage( "" );
Sam Hocevar's avatar
Sam Hocevar committed
319

320 321 322 323
    /*
     * Global iconv, must be done after setlocale()
     * so that vlc_current_charset() works.
     */
324
    LocaleInit( (vlc_object_t *)p_vlc );
325

Sam Hocevar's avatar
Sam Hocevar committed
326
    /* Translate "C" to the language code: "fr", "en_GB", "nl", "ru"... */
327 328
    msg_Dbg( p_vlc, "translation test: code is \"%s\"", _("C") );

329 330 331 332
    /* Initialize the module bank 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) */
333
    module_InitBank( p_vlc );
334

335 336 337 338
    /* Hack: insert the help module here */
    p_help_module = vlc_object_create( p_vlc, VLC_OBJECT_MODULE );
    if( p_help_module == NULL )
    {
339
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
340
        if( i_object ) vlc_object_release( p_vlc );
341 342 343
        return VLC_EGENERIC;
    }
    p_help_module->psz_object_name = "help";
Gildas Bazin's avatar
 
Gildas Bazin committed
344
    p_help_module->psz_longname = N_("Help options");
345
    config_Duplicate( p_help_module, p_help_config );
346
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
347 348
    /* End hack */

Sam Hocevar's avatar
Sam Hocevar committed
349
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE ) )
350
    {
351
        vlc_object_detach( p_help_module );
352 353
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
354
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
355
        if( i_object ) vlc_object_release( p_vlc );
356 357 358 359 360 361
        return VLC_EGENERIC;
    }

    /* Check for short help option */
    if( config_GetInt( p_vlc, "help" ) )
    {
362
        Help( p_vlc, "help" );
363
        b_exit = VLC_TRUE;
364
        i_ret = VLC_EEXITSUCCESS;
365 366
    }
    /* Check for version option */
367
    else if( config_GetInt( p_vlc, "version" ) )
368 369
    {
        Version();
370
        b_exit = VLC_TRUE;
371
        i_ret = VLC_EEXITSUCCESS;
372 373
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
374 375
    /* Set the config file stuff */
    p_vlc->psz_homedir = config_GetHomeDir();
376 377 378
    p_vlc->psz_userdir = config_GetUserDir();
    if( p_vlc->psz_userdir == NULL )
        p_vlc->psz_userdir = strdup(p_vlc->psz_homedir);
Gildas Bazin's avatar
 
Gildas Bazin committed
379
    p_vlc->psz_configfile = config_GetPsz( p_vlc, "config" );
380 381 382
    if( p_vlc->psz_configfile != NULL && p_vlc->psz_configfile[0] == '~'
         && p_vlc->psz_configfile[1] == '/' )
    {
383
        char *psz = malloc( strlen(p_vlc->psz_userdir)
384 385
                             + strlen(p_vlc->psz_configfile) );
        /* This is incomplete : we should also support the ~cmassiot/ syntax. */
386
        sprintf( psz, "%s/%s", p_vlc->psz_userdir,
387 388 389 390
                               p_vlc->psz_configfile + 2 );
        free( p_vlc->psz_configfile );
        p_vlc->psz_configfile = psz;
    }
Gildas Bazin's avatar
 
Gildas Bazin committed
391

392 393 394 395 396 397
    /* Check for plugins cache options */
    if( config_GetInt( p_vlc, "reset-plugins-cache" ) )
    {
        libvlc.p_module_bank->b_cache_delete = VLC_TRUE;
    }

398 399 400 401
    /* Hack: remove the help module here */
    vlc_object_detach( p_help_module );
    /* End hack */

Gildas Bazin's avatar
Gildas Bazin committed
402 403 404 405 406 407 408
    /* Will be re-done properly later on */
    p_vlc->p_libvlc->i_verbose = config_GetInt( p_vlc, "verbose" );

    /* Check for daemon mode */
#ifndef WIN32
    if( config_GetInt( p_vlc, "daemon" ) )
    {
409
#if HAVE_DAEMON
410
        if( daemon( 1, 0) != 0 )
411 412 413 414 415 416 417 418 419
        {
            msg_Err( p_vlc, "Unable to fork vlc to daemon mode" );
            b_exit = VLC_TRUE;
        }

        p_vlc->p_libvlc->b_daemon = VLC_TRUE;

#else
        pid_t i_pid;
Gildas Bazin's avatar
Gildas Bazin committed
420 421 422 423 424 425 426 427 428 429 430

        if( ( i_pid = fork() ) < 0 )
        {
            msg_Err( p_vlc, "Unable to fork vlc to daemon mode" );
            b_exit = VLC_TRUE;
        }
        else if( i_pid )
        {
            /* This is the parent, exit right now */
            msg_Dbg( p_vlc, "closing parent process" );
            b_exit = VLC_TRUE;
431
            i_ret = VLC_EEXITSUCCESS;
Gildas Bazin's avatar
Gildas Bazin committed
432 433 434 435 436
        }
        else
        {
            /* We are the child */
            msg_Dbg( p_vlc, "daemon spawned" );
437 438 439
            close( STDIN_FILENO );
            close( STDOUT_FILENO );
            close( STDERR_FILENO );
Gildas Bazin's avatar
Gildas Bazin committed
440 441 442

            p_vlc->p_libvlc->b_daemon = VLC_TRUE;
        }
443
#endif
Gildas Bazin's avatar
Gildas Bazin committed
444 445 446
    }
#endif

447 448
    if( b_exit )
    {
449 450
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
451
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
452
        if( i_object ) vlc_object_release( p_vlc );
453
        return i_ret;
454 455
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
456
    /* Check for translation config option */
Gildas Bazin's avatar
 
Gildas Bazin committed
457 458 459 460 461 462 463 464
#if defined( ENABLE_NLS ) \
     && ( defined( HAVE_GETTEXT ) || defined( HAVE_INCLUDED_GETTEXT ) )

    /* This ain't really nice to have to reload the config here but it seems
     * the only way to do it. */
    config_LoadConfigFile( p_vlc, "main" );
    config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE );

465 466 467
    /* Check if the user specified a custom language */
    psz_language = config_GetPsz( p_vlc, "language" );
    if( psz_language && *psz_language && strcmp( psz_language, "auto" ) )
Gildas Bazin's avatar
 
Gildas Bazin committed
468
    {
469 470
        vlc_bool_t b_cache_delete = libvlc.p_module_bank->b_cache_delete;

Gildas Bazin's avatar
 
Gildas Bazin committed
471
        /* Reset the default domain */
472
        SetLanguage( psz_language );
473 474 475 476 477 478 479

        /* Should not be needed (otherwise, fixes should rather be
         * attempted on vlc_current_charset().
         * Also, if the locale charset is overriden, anything that has been
         * translated until now would have to be retranslated. */
        /*LocaleDeinit();
        LocaleInit( (vlc_object_t *)p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
480

Gildas Bazin's avatar
 
Gildas Bazin committed
481 482 483
        /* Translate "C" to the language code: "fr", "en_GB", "nl", "ru"... */
        msg_Dbg( p_vlc, "translation test: code is \"%s\"", _("C") );

Gildas Bazin's avatar
 
Gildas Bazin committed
484
        module_EndBank( p_vlc );
485
        module_InitBank( p_vlc );
486
        config_LoadConfigFile( p_vlc, "main" );
Gildas Bazin's avatar
 
Gildas Bazin committed
487
        config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE );
488
        libvlc.p_module_bank->b_cache_delete = b_cache_delete;
Gildas Bazin's avatar
 
Gildas Bazin committed
489
    }
490
    if( psz_language ) free( psz_language );
491
#endif
Gildas Bazin's avatar
 
Gildas Bazin committed
492

493 494 495 496 497 498
    /*
     * Load the builtins and plugins into the module_bank.
     * We have to do it before config_Load*() because this also gets the
     * list of configuration options exported by each module and loads their
     * default values.
     */
499 500 501 502 503 504 505
    module_LoadBuiltins( p_vlc );
    module_LoadPlugins( p_vlc );
    if( p_vlc->b_die )
    {
        b_exit = VLC_TRUE;
    }

506
    msg_Dbg( p_vlc, "module bank initialized, found %i modules",
507
                    libvlc.p_module_bank->i_children );
508 509

    /* Hack: insert the help module here */
510
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
511 512 513 514 515
    /* End hack */

    /* Check for help on modules */
    if( (p_tmp = config_GetPsz( p_vlc, "module" )) )
    {
516
        Help( p_vlc, p_tmp );
517
        free( p_tmp );
518
        b_exit = VLC_TRUE;
519
        i_ret = VLC_EEXITSUCCESS;
520 521
    }
    /* Check for long help option */
522
    else if( config_GetInt( p_vlc, "longhelp" ) )
523
    {
524
        Help( p_vlc, "longhelp" );
525
        b_exit = VLC_TRUE;
526
        i_ret = VLC_EEXITSUCCESS;
527 528
    }
    /* Check for module list option */
529
    else if( config_GetInt( p_vlc, "list" ) )
530
    {
531
        ListModules( p_vlc );
532
        b_exit = VLC_TRUE;
533
        i_ret = VLC_EEXITSUCCESS;
534 535
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553
    /* Check for config file options */
    if( config_GetInt( p_vlc, "reset-config" ) )
    {
        vlc_object_detach( p_help_module );
        config_ResetAll( p_vlc );
        config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE );
        config_SaveConfigFile( p_vlc, NULL );
        vlc_object_attach( p_help_module, libvlc.p_module_bank );
    }
    if( config_GetInt( p_vlc, "save-config" ) )
    {
        vlc_object_detach( p_help_module );
        config_LoadConfigFile( p_vlc, NULL );
        config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE );
        config_SaveConfigFile( p_vlc, NULL );
        vlc_object_attach( p_help_module, libvlc.p_module_bank );
    }

554
    /* Hack: remove the help module here */
555
    vlc_object_detach( p_help_module );
556 557
    /* End hack */

558 559
    if( b_exit )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
560 561
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
562
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
563
        if( i_object ) vlc_object_release( p_vlc );
564
        return i_ret;
565 566
    }

567 568 569 570 571
    /*
     * Init device values
     */
    InitDeviceValues( p_vlc );

572 573 574
    /*
     * Override default configuration with config file settings
     */
575
    config_LoadConfigFile( p_vlc, NULL );
576

Gildas Bazin's avatar
 
Gildas Bazin committed
577 578 579 580
    /* Hack: insert the help module here */
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
    /* End hack */

581 582 583
    /*
     * Override configuration with command line settings
     */
Sam Hocevar's avatar
Sam Hocevar committed
584
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_FALSE ) )
585 586
    {
#ifdef WIN32
587
        ShowConsole( VLC_FALSE );
588 589
        /* Pause the console because it's destroyed when we exit */
        fprintf( stderr, "The command line options couldn't be loaded, check "
590 591
                 "that they are valid.\n" );
        PauseConsole();
592
#endif
Gildas Bazin's avatar
 
Gildas Bazin committed
593 594 595
        vlc_object_detach( p_help_module );
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
596
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
597
        if( i_object ) vlc_object_release( p_vlc );
598 599 600
        return VLC_EGENERIC;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
601 602 603 604 605 606
    /* Hack: remove the help module here */
    vlc_object_detach( p_help_module );
    config_Free( p_help_module );
    vlc_object_destroy( p_help_module );
    /* End hack */

607 608 609
    /*
     * System specific configuration
     */
Gildas Bazin's avatar
 
Gildas Bazin committed
610
    system_Configure( p_vlc, &i_argc, ppsz_argv );
611

612 613 614
    /*
     * Message queue options
     */
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
615 616

    var_Create( p_vlc, "verbose", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
617 618
    if( config_GetInt( p_vlc, "quiet" ) )
    {
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
619 620
        val.i_int = -1;
        var_Set( p_vlc, "verbose", val );
621
    }
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
622 623
    var_AddCallback( p_vlc, "verbose", VerboseCallback, NULL );
    var_Change( p_vlc, "verbose", VLC_VAR_TRIGGER_CALLBACKS, NULL, NULL );
Clément Stenac's avatar
Clément Stenac committed
624

Gildas Bazin's avatar
 
Gildas Bazin committed
625
    libvlc.b_color = libvlc.b_color && config_GetInt( p_vlc, "color" );
626

627 628 629 630 631
    /*
     * Output messages that may still be in the queue
     */
    msg_Flush( p_vlc );

632
    /* p_vlc initialization. FIXME ? */
633

634 635 636
    if( !config_GetInt( p_vlc, "fpu" ) )
        libvlc.i_cpu &= ~CPU_CAPABILITY_FPU;

Christophe Massiot's avatar
Christophe Massiot committed
637
#if defined( __i386__ ) || defined( __x86_64__ )
638
    if( !config_GetInt( p_vlc, "mmx" ) )
639
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMX;
640
    if( !config_GetInt( p_vlc, "3dn" ) )
641
        libvlc.i_cpu &= ~CPU_CAPABILITY_3DNOW;
642
    if( !config_GetInt( p_vlc, "mmxext" ) )
643
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMXEXT;
644
    if( !config_GetInt( p_vlc, "sse" ) )
645
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE;
646 647
    if( !config_GetInt( p_vlc, "sse2" ) )
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE2;
648
#endif
649
#if defined( __powerpc__ ) || defined( __ppc__ ) || defined( __ppc64__ )
650
    if( !config_GetInt( p_vlc, "altivec" ) )
651
        libvlc.i_cpu &= ~CPU_CAPABILITY_ALTIVEC;
652
#endif
653 654

#define PRINT_CAPABILITY( capability, string )                              \
655
    if( libvlc.i_cpu & capability )                                         \
656 657 658 659 660 661 662 663 664 665 666 667 668 669
    {                                                                       \
        strncat( p_capabilities, string " ",                                \
                 sizeof(p_capabilities) - strlen(p_capabilities) );         \
        p_capabilities[sizeof(p_capabilities) - 1] = '\0';                  \
    }

    p_capabilities[0] = '\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" );
670
    PRINT_CAPABILITY( CPU_CAPABILITY_SSE2, "SSE2" );
671 672 673 674 675 676 677
    PRINT_CAPABILITY( CPU_CAPABILITY_ALTIVEC, "AltiVec" );
    PRINT_CAPABILITY( CPU_CAPABILITY_FPU, "FPU" );
    msg_Dbg( p_vlc, "CPU has capabilities %s", p_capabilities );

    /*
     * Choose the best memcpy module
     */
Gildas Bazin's avatar
 
Gildas Bazin committed
678
    p_vlc->p_memcpy_module = module_Need( p_vlc, "memcpy", "$memcpy", 0 );
679

680
    if( p_vlc->pf_memcpy == NULL )
681 682 683
    {
        p_vlc->pf_memcpy = memcpy;
    }
684 685 686 687 688

    if( p_vlc->pf_memset == NULL )
    {
        p_vlc->pf_memset = memset;
    }
689

690
    libvlc.b_stats = config_GetInt( p_vlc, "stats" );
691
    libvlc.p_stats = NULL;
692

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
693 694 695 696
    /*
     * Initialize hotkey handling
     */
    var_Create( p_vlc, "key-pressed", VLC_VAR_INTEGER );
Gildas Bazin's avatar
 
Gildas Bazin committed
697 698
    p_vlc->p_hotkeys = malloc( sizeof(p_hotkeys) );
    /* Do a copy (we don't need to modify the strings) */
699
    memcpy( p_vlc->p_hotkeys, p_hotkeys, sizeof(p_hotkeys) );
Gildas Bazin's avatar
 
Gildas Bazin committed
700

701 702 703
    /*
     * Initialize playlist and get commandline files
     */
704
    p_playlist = playlist_Create( p_vlc );
705 706 707
    if( !p_playlist )
    {
        msg_Err( p_vlc, "playlist initialization failed" );
708 709
        if( p_vlc->p_memcpy_module != NULL )
        {
710
            module_Unneed( p_vlc, p_vlc->p_memcpy_module );
711
        }
712
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
713
        if( i_object ) vlc_object_release( p_vlc );
714 715 716
        return VLC_EGENERIC;
    }

717 718 719 720 721 722 723
    psz_modules = config_GetPsz( p_playlist, "services-discovery" );
    if( psz_modules && *psz_modules )
    {
        /* Add service discovery modules */
        playlist_AddSDModules( p_playlist, psz_modules );
    }
    if( psz_modules ) free( psz_modules );
724

725 726 727 728
    /*
     * Load background interfaces
     */
    psz_modules = config_GetPsz( p_vlc, "extraintf" );
729 730 731 732 733 734
    psz_control = config_GetPsz( p_vlc, "control" );

    if( psz_modules && *psz_modules && psz_control && *psz_control )
    {
        psz_modules = (char *)realloc( psz_modules, strlen( psz_modules ) +
                                                    strlen( psz_control ) + 1 );
735
        sprintf( psz_modules, "%s:%s", psz_modules, psz_control );
736 737 738 739 740 741 742
    }
    else if( psz_control && *psz_control )
    {
        if( psz_modules ) free( psz_modules );
        psz_modules = strdup( psz_control );
    }

743 744 745
    psz_parser = psz_modules;
    while ( psz_parser && *psz_parser )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
746
        char *psz_module, *psz_temp;
747
        psz_module = psz_parser;
748
        psz_parser = strchr( psz_module, ':' );
749 750 751 752 753
        if ( psz_parser )
        {
            *psz_parser = '\0';
            psz_parser++;
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
754 755 756 757
        psz_temp = (char *)malloc( strlen(psz_module) + sizeof(",none") );
        if( psz_temp )
        {
            sprintf( psz_temp, "%s,none", psz_module );
758
            VLC_AddIntf( 0, psz_temp, VLC_FALSE, VLC_FALSE );
Gildas Bazin's avatar
 
Gildas Bazin committed
759 760
            free( psz_temp );
        }
761 762 763 764 765
    }
    if ( psz_modules )
    {
        free( psz_modules );
    }
766

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
767
    /*
768
     * Always load the hotkeys interface if it exists
Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
769
     */
770
    VLC_AddIntf( 0, "hotkeys,none", VLC_FALSE, VLC_FALSE );
Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
771

772 773 774 775 776 777 778 779 780 781 782
    /*
     * If needed, load the Xscreensaver interface
     * Currently, only for X
     */
#ifdef HAVE_X11_XLIB_H
    if( config_GetInt( p_vlc, "disable-screensaver" ) == 1 )
    {
        VLC_AddIntf( 0, "screensaver", VLC_FALSE, VLC_FALSE );
    }
#endif

783 784 785 786
    if( config_GetInt( p_vlc, "file-logging" ) == 1 )
    {
        VLC_AddIntf( 0, "logger", VLC_FALSE, VLC_FALSE );
    }
787
#ifdef HAVE_SYSLOG_H
788 789 790 791 792
    if( config_GetInt( p_vlc, "syslog" ) == 1 )
    {
        char *psz_logmode = "logmode=syslog";
        AddIntfInternal( 0, "logger", VLC_FALSE, VLC_FALSE, 1, &psz_logmode );
    }
793
#endif
794

795 796 797 798 799 800 801 802 803 804
    if( config_GetInt( p_vlc, "show-intf" ) == 1 )
    {
        VLC_AddIntf( 0, "showintf", VLC_FALSE, VLC_FALSE );
    }

    if( config_GetInt( p_vlc, "network-synchronisation") == 1 )
    {
        VLC_AddIntf( 0, "netsync", VLC_FALSE, VLC_FALSE );
    }

805 806 807 808
    /*
     * FIXME: kludge to use a p_vlc-local variable for the Mozilla plugin
     */
    var_Create( p_vlc, "drawable", VLC_VAR_INTEGER );
809 810 811 812 813 814 815 816 817 818 819
    var_Create( p_vlc, "drawableredraw", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawablet", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawablel", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawableb", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawabler", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawablex", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawabley", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawablew", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawableh", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawableportx", VLC_VAR_INTEGER );
    var_Create( p_vlc, "drawableporty", VLC_VAR_INTEGER );
820

821 822 823
    /* Create volume callback system. */
    var_Create( p_vlc, "volume-change", VLC_VAR_BOOL );

824 825 826 827 828
    /*
     * Get input filenames given as commandline arguments
     */
    GetFilenames( p_vlc, i_argc, ppsz_argv );

829 830 831 832 833 834 835 836 837 838 839 840
    /*
     * Get --open argument
     */
    var_Create( p_vlc, "open", VLC_VAR_STRING | VLC_VAR_DOINHERIT );
    var_Get( p_vlc, "open", &val );
    if ( val.psz_string != NULL && *val.psz_string )
    {
        VLC_AddTarget( p_vlc->i_object_id, val.psz_string, NULL, 0,
                       PLAYLIST_INSERT, 0 );
    }
    if ( val.psz_string != NULL ) free( val.psz_string );

Gildas Bazin's avatar
 
Gildas Bazin committed
841
    if( i_object ) vlc_object_release( p_vlc );
842 843 844 845
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
846
 * VLC_AddIntf: add an interface
847 848
 *****************************************************************************
 * This function opens an interface plugin and runs it. If b_block is set
Sam Hocevar's avatar
Sam Hocevar committed
849 850
 * to 0, VLC_AddIntf will return immediately and let the interface run in a
 * separate thread. If b_block is set to 1, VLC_AddIntf will continue until
851 852
 * user requests to quit. If b_play is set to 1, VLC_AddIntf will start playing
 * the playlist when it is completely initialised.
853
 *****************************************************************************/
854 855
int VLC_AddIntf( int i_object, char const *psz_module,
                 vlc_bool_t b_block, vlc_bool_t b_play )
856
{
857
    return AddIntfInternal( i_object, psz_module, b_block, b_play, 0, NULL );
858 859
}

860

861
/*****************************************************************************
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891
 * VLC_Die: ask vlc to die.
 *****************************************************************************
 * This function sets p_vlc->b_die to VLC_TRUE, but does not do any other
 * task. It is your duty to call VLC_CleanUp and VLC_Destroy afterwards.
 *****************************************************************************/
int VLC_Die( int i_object )
{
    vlc_t *p_vlc = vlc_current_object( i_object );

    if( !p_vlc )
    {
        return VLC_ENOOBJ;
    }

    p_vlc->b_die = VLC_TRUE;

    if( i_object ) vlc_object_release( p_vlc );
    return VLC_SUCCESS;
}

/*****************************************************************************
 * VLC_CleanUp: CleanUp all the intf, playlist, vout, aout
 *****************************************************************************/
int VLC_CleanUp( int i_object )
{
    intf_thread_t      * p_intf;
    playlist_t         * p_playlist;
    vout_thread_t      * p_vout;
    aout_instance_t    * p_aout;
    announce_handler_t * p_announce;
892
    stats_handler_t    * p_stats;
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913
    vlc_t *p_vlc = vlc_current_object( i_object );

    /* Check that the handle is valid */
    if( !p_vlc )
    {
        return VLC_ENOOBJ;
    }

    /*
     * Ask the interfaces to stop and destroy them
     */
    msg_Dbg( p_vlc, "removing all interfaces" );
    while( (p_intf = vlc_object_find( p_vlc, VLC_OBJECT_INTF, FIND_CHILD )) )
    {
        intf_StopThread( p_intf );
        vlc_object_detach( p_intf );
        vlc_object_release( p_intf );
        intf_Destroy( p_intf );
    }

    /*
Clément Stenac's avatar
Clément Stenac committed
914
     * Free playlist
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
915
     */
Clément Stenac's avatar
Clément Stenac committed
916
    msg_Dbg( p_vlc, "removing playlist handler" );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946
    while( (p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST,
                                          FIND_CHILD )) )
    {
        vlc_object_detach( p_playlist );
        vlc_object_release( p_playlist );
        playlist_Destroy( p_playlist );
    }

    /*
     * Free video outputs
     */
    msg_Dbg( p_vlc, "removing all video outputs" );
    while( (p_vout = vlc_object_find( p_vlc, VLC_OBJECT_VOUT, FIND_CHILD )) )
    {
        vlc_object_detach( p_vout );
        vlc_object_release( p_vout );
        vout_Destroy( p_vout );
    }

    /*
     * Free audio outputs
     */
    msg_Dbg( p_vlc, "removing all audio outputs" );
    while( (p_aout = vlc_object_find( p_vlc, VLC_OBJECT_AOUT, FIND_CHILD )) )
    {
        vlc_object_detach( (vlc_object_t *)p_aout );
        vlc_object_release( (vlc_object_t *)p_aout );
        aout_Delete( p_aout );
    }

947 948
    while( ( p_stats = vlc_object_find( p_vlc, VLC_OBJECT_STATS, FIND_CHILD) ))
    {
949
        stats_TimersDumpAll( p_vlc );
Clément Stenac's avatar
Clément Stenac committed
950
        stats_HandlerDestroy( p_stats );
951 952 953 954 955
        vlc_object_detach( (vlc_object_t*) p_stats );
        vlc_object_release( (vlc_object_t *)p_stats );
        // TODO: Delete it
    }

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
956 957 958 959 960
    /*
     * Free announce handler(s?)
     */
    while( (p_announce = vlc_object_find( p_vlc, VLC_OBJECT_ANNOUNCE,
                                                 FIND_CHILD ) ) )
Clément Stenac's avatar
Clément Stenac committed
961 962
    {
        msg_Dbg( p_vlc, "removing announce handler" );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
963 964 965
        vlc_object_detach( p_announce );
        vlc_object_release( p_announce );
        announce_HandlerDestroy( p_announce );
Clément Stenac's avatar
Clément Stenac committed
966
    }
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
967 968 969 970 971 972 973

    if( i_object ) vlc_object_release( p_vlc );
    return VLC_SUCCESS;
}

/*****************************************************************************
 * VLC_Destroy: Destroy everything.
974
 *****************************************************************************
975
 * This function requests the running threads to finish, waits for their
976 977
 * termination, and destroys their structure.
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
978
int VLC_Destroy( int i_object )
979
{
980
    vlc_t *p_vlc = vlc_current_object( i_object );
981

982
    if( !p_vlc )
983
    {
984
        return VLC_ENOOBJ;
985 986
    }

987 988 989 990 991 992 993
    /*
     * Free allocated memory
     */
    if( p_vlc->p_memcpy_module )
    {
        module_Unneed( p_vlc, p_vlc->p_memcpy_module );
        p_vlc->p_memcpy_module = NULL;
994 995
    }

996
    /*
997
     * Free module bank !
998 999 1000
     */
    module_EndBank( p_vlc );

1001 1002 1003 1004 1005
    if( p_vlc->psz_homedir )
    {
        free( p_vlc->psz_homedir );
        p_vlc->psz_homedir = NULL;
    }
1006

1007 1008 1009 1010 1011 1012
    if( p_vlc->psz_userdir )
    {
        free( p_vlc->psz_userdir );
        p_vlc->psz_userdir = NULL;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
1013 1014 1015 1016 1017 1018
    if( p_vlc->psz_configfile )
    {
        free( p_vlc->psz_configfile );
        p_vlc->psz_configfile = NULL;
    }

1019 1020 1021 1022 1023 1024
    if( p_vlc->p_hotkeys )
    {
        free( p_vlc->p_hotkeys );
        p_vlc->p_hotkeys = NULL;
    }

1025 1026 1027 1028
    /*
     * System specific cleaning code
     */
    system_End( p_vlc );
1029

1030 1031 1032 1033 1034 1035 1036
    /*
     * Free message queue.
     * Nobody shall use msg_* afterward.
     */
    msg_Flush( p_vlc );
    msg_Destroy( p_libvlc );

1037
    /* Destroy global iconv */
1038
    LocaleDeinit();
1039

1040 1041 1042
    /* Destroy mutexes */
    vlc_mutex_destroy( &p_vlc->config_lock );

1043 1044
    vlc_object_detach( p_vlc );

1045 1046 1047
    /* Release object before destroying it */
    if( i_object ) vlc_object_release( p_vlc );

1048 1049
    vlc_object_destroy( p_vlc );

1050
    /* Stop thread system: last one out please shut the door! */
1051
    vlc_threads_end( p_libvlc );
1052

1053 1054 1055
    return VLC_SUCCESS;
}

1056
/*****************************************************************************
1057
 * VLC_VariableSet: set a vlc variable
1058
 *****************************************************************************/
1059
int VLC_VariableSet( int i_object, char const *psz_var, vlc_value_t value )
1060
{
1061
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
1062
    int i_ret;
1063 1064 1065

    if( !p_vlc )
    {
1066
        return VLC_ENOOBJ;
1067 1068
    }

Sam Hocevar's avatar
Sam Hocevar committed
1069 1070 1071
    /* FIXME: Temporary hack for Mozilla, if variable starts with conf:: then
     * we handle it as a configuration variable. Don't tell Gildas :) -- sam */
    if( !strncmp( psz_var, "conf::", 6 ) )
1072
    {
Sam Hocevar's avatar
Sam Hocevar committed
1073
        module_config_t *p_item;
1074
        char const *psz_newvar = psz_var + 6;
1075

Sam Hocevar's avatar
Sam Hocevar committed
1076
        p_item = config_FindConfig( VLC_OBJECT(p_vlc), psz_newvar );
1077

Sam Hocevar's avatar
Sam Hocevar committed
1078
        if( p_item )
1079
        {
Sam Hocevar's avatar
Sam Hocevar committed
1080
            switch( p_item->i_type )
1081
            {
Sam Hocevar's avatar
Sam Hocevar committed
1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093
                case CONFIG_ITEM_BOOL:
                    config_PutInt( p_vlc, psz_newvar, value.b_bool );
                    break;
                case CONFIG_ITEM_INTEGER:
                    config_PutInt( p_vlc, psz_newvar, value.i_int );
                    break;
                case CONFIG_ITEM_FLOAT:
                    config_PutFloat( p_vlc, psz_newvar, value.f_float );
                    break;
                default:
                    config_PutPsz( p_vlc, psz_newvar, value.psz_string );
                    break;
1094
            }
Gildas Bazin's avatar
 
Gildas Bazin committed
1095
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
1096
            return VLC_SUCCESS;
1097 1098 1099
        }
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
1100 1101 1102 1103
    i_ret = var_Set( p_vlc, psz_var, value );

    if( i_object ) vlc_object_release( p_vlc );
    return i_ret;
Sam Hocevar's avatar
Sam Hocevar committed
1104 1105 1106
}

/*****************************************************************************
1107
 * VLC_VariableGet: get a vlc variable
Sam Hocevar's avatar
Sam Hocevar committed
1108
 *****************************************************************************/
1109
int VLC_VariableGet( int i_object, char const *psz_var, vlc_value_t *p_value )
Sam Hocevar's avatar
Sam Hocevar committed
1110
{
1111
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar