libvlc.c 68.9 KB
Newer Older
1 2 3
/*****************************************************************************
 * libvlc.c: main libvlc source
 *****************************************************************************
4
 * Copyright (C) 1998-2004 VideoLAN
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 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
 *****************************************************************************/

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

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

40
#include <errno.h>                                                 /* ENOMEM */
41 42 43 44 45 46 47 48 49 50
#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
51
#elif defined( WIN32 ) && !defined( UNDER_CE )
52 53 54
#   include <io.h>
#endif

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

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

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

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

70
#include "vlc_error.h"
71

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

#include "audio_output.h"

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

80 81
#include "stream_output.h"

82 83 84
#include "libvlc.h"

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

/*****************************************************************************
 * Local prototypes
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
94
static void SetLanguage   ( char const * );
95
static int  GetFilenames  ( vlc_t *, int, char *[] );
96
static void Help          ( vlc_t *, char const *psz_help_name );
97
static void Usage         ( vlc_t *, char const *psz_module_name );
98
static void ListModules   ( vlc_t * );
99 100 101 102
static void Version       ( void );

#ifdef WIN32
static void ShowConsole   ( void );
103
static void PauseConsole  ( void );
104
#endif
105
static int  ConsoleWidth  ( void );
106

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
107 108
static int  VerboseCallback( vlc_object_t *, char const *,
                             vlc_value_t, vlc_value_t, void * );
109

110 111
static void InitDeviceValues( vlc_t * );

112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
/*****************************************************************************
 * 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;
}

128
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
129
 * VLC_Version: return the libvlc version.
130
 *****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
131
 * This function returns full version string (numeric version and codename).
132
 *****************************************************************************/
133
char const * VLC_Version( void )
134
{
Sam Hocevar's avatar
Sam Hocevar committed
135
    return VERSION_MESSAGE;
136 137
}

138 139 140 141 142 143 144 145 146 147
/*****************************************************************************
 * 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
148 149 150 151 152 153 154
/*****************************************************************************
 * 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 )
155
{
156
    int i_ret;
157
    vlc_t * p_vlc = NULL;
158
    vlc_value_t lockval;
159

160 161 162
    /* &libvlc never changes, so we can safely call this multiple times. */
    p_libvlc = &libvlc;

163 164
    /* vlc_threads_init *must* be the first internal call! No other call is
     * allowed before the thread system has been initialized. */
165
    i_ret = vlc_threads_init( p_libvlc );
Sam Hocevar's avatar
Sam Hocevar committed
166
    if( i_ret < 0 )
167
    {
Sam Hocevar's avatar
Sam Hocevar committed
168
        return i_ret;
169 170 171
    }

    /* Now that the thread system is initialized, we don't have much, but
172
     * at least we have var_Create */
173 174
    var_Create( p_libvlc, "libvlc", VLC_VAR_MUTEX );
    var_Get( p_libvlc, "libvlc", &lockval );
175
    vlc_mutex_lock( lockval.p_address );
176 177
    if( !libvlc.b_ready )
    {
178 179
        char *psz_env;

180 181 182
        /* Guess what CPU we have */
        libvlc.i_cpu = CPUCapabilities();

183 184
        /* Find verbosity from VLC_VERBOSE environment variable */
        psz_env = getenv( "VLC_VERBOSE" );
185
        libvlc.i_verbose = psz_env ? atoi( psz_env ) : -1;
186

Gildas Bazin's avatar
 
Gildas Bazin committed
187
#if defined( HAVE_ISATTY ) && !defined( WIN32 )
188 189 190 191 192
        libvlc.b_color = isatty( 2 ); /* 2 is for stderr */
#else
        libvlc.b_color = VLC_FALSE;
#endif

193
        /* Initialize message queue */
194
        msg_Create( p_libvlc );
195 196

        /* Announce who we are */
197 198
        msg_Dbg( p_libvlc, COPYRIGHT_MESSAGE );
        msg_Dbg( p_libvlc, "libvlc was configured with %s", CONFIGURE_LINE );
199

200 201
        /* The module bank will be initialized later */
        libvlc.p_module_bank = NULL;
202 203

        libvlc.b_ready = VLC_TRUE;
204
    }
205
    vlc_mutex_unlock( lockval.p_address );
206
    var_Destroy( p_libvlc, "libvlc" );
207 208

    /* Allocate a vlc object */
209
    p_vlc = vlc_object_create( p_libvlc, VLC_OBJECT_VLC );
210 211
    if( p_vlc == NULL )
    {
Sam Hocevar's avatar
Sam Hocevar committed
212
        return VLC_EGENERIC;
213
    }
214
    p_vlc->thread_id = 0;
215 216 217 218

    p_vlc->psz_object_name = "root";

    /* Initialize mutexes */
219
    vlc_mutex_init( p_vlc, &p_vlc->config_lock );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
220 221
#ifdef SYS_DARWIN
    vlc_mutex_init( p_vlc, &p_vlc->quicktime_lock );
222
    vlc_thread_set_priority( p_vlc, VLC_THREAD_PRIORITY_LOW );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
223
#endif
224 225

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

Sam Hocevar's avatar
Sam Hocevar committed
228 229 230 231
    /* Store data for the non-reentrant API */
    p_static_vlc = p_vlc;

    return p_vlc->i_object_id;
232 233 234
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
235
 * VLC_Init: initialize a vlc_t structure.
236 237 238 239 240 241 242
 *****************************************************************************
 * 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
243
int VLC_Init( int i_object, int i_argc, char *ppsz_argv[] )
244
{
245 246
    char         p_capabilities[200];
    char *       p_tmp;
247 248
    char *       psz_modules;
    char *       psz_parser;
249
    char *       psz_control;
250
    char *       psz_language;
251
    vlc_bool_t   b_exit = VLC_FALSE;
252
    vlc_t *      p_vlc = vlc_current_object( i_object );
253 254
    module_t    *p_help_module;
    playlist_t  *p_playlist;
255

256
    if( !p_vlc )
257
    {
258
        return VLC_ENOOBJ;
259 260 261
    }

    /*
262
     * System specific initialization code
263
     */
264
    system_Init( p_vlc, &i_argc, ppsz_argv );
265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280

    /* 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";
    }

281 282 283 284
    /*
     * Support for gettext
     */
    SetLanguage( "" );
Sam Hocevar's avatar
Sam Hocevar committed
285 286

    /* Translate "C" to the language code: "fr", "en_GB", "nl", "ru"... */
287 288
    msg_Dbg( p_vlc, "translation test: code is \"%s\"", _("C") );

289 290 291 292
    /* 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) */
293
    module_InitBank( p_vlc );
294

295 296 297 298
    /* Hack: insert the help module here */
    p_help_module = vlc_object_create( p_vlc, VLC_OBJECT_MODULE );
    if( p_help_module == NULL )
    {
299
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
300
        if( i_object ) vlc_object_release( p_vlc );
301 302 303
        return VLC_EGENERIC;
    }
    p_help_module->psz_object_name = "help";
Gildas Bazin's avatar
 
Gildas Bazin committed
304
    p_help_module->psz_longname = N_("Help options");
305
    config_Duplicate( p_help_module, p_help_config );
306
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
307 308
    /* End hack */

Sam Hocevar's avatar
Sam Hocevar committed
309
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE ) )
310
    {
311
        vlc_object_detach( p_help_module );
312 313
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
314
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
315
        if( i_object ) vlc_object_release( p_vlc );
316 317 318 319 320 321
        return VLC_EGENERIC;
    }

    /* Check for short help option */
    if( config_GetInt( p_vlc, "help" ) )
    {
322
        Help( p_vlc, "help" );
323
        b_exit = VLC_TRUE;
324 325
    }
    /* Check for version option */
326
    else if( config_GetInt( p_vlc, "version" ) )
327 328
    {
        Version();
329 330 331
        b_exit = VLC_TRUE;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
332 333 334 335
    /* Set the config file stuff */
    p_vlc->psz_homedir = config_GetHomeDir();
    p_vlc->psz_configfile = config_GetPsz( p_vlc, "config" );

336 337 338 339 340 341
    /* Check for plugins cache options */
    if( config_GetInt( p_vlc, "reset-plugins-cache" ) )
    {
        libvlc.p_module_bank->b_cache_delete = VLC_TRUE;
    }

342 343 344 345
    /* Hack: remove the help module here */
    vlc_object_detach( p_help_module );
    /* End hack */

Gildas Bazin's avatar
Gildas Bazin committed
346 347 348 349 350 351 352
    /* 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" ) )
    {
353 354 355 356 357 358 359 360 361 362 363
#if HAVE_DAEMON
        if( daemon( 0, 0) != 0 )
        {
            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
364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379

        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;
        }
        else
        {
            /* We are the child */
            msg_Dbg( p_vlc, "daemon spawned" );
380 381 382
            close( STDIN_FILENO );
            close( STDOUT_FILENO );
            close( STDERR_FILENO );
Gildas Bazin's avatar
Gildas Bazin committed
383 384 385

            p_vlc->p_libvlc->b_daemon = VLC_TRUE;
        }
386
#endif
Gildas Bazin's avatar
Gildas Bazin committed
387 388 389
    }
#endif

390 391
    if( b_exit )
    {
392 393
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
394
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
395
        if( i_object ) vlc_object_release( p_vlc );
396 397 398
        return VLC_EEXIT;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
399
    /* Check for translation config option */
Gildas Bazin's avatar
 
Gildas Bazin committed
400 401 402 403 404 405 406 407
#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 );

408 409 410
    /* 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
411
    {
412 413
        vlc_bool_t b_cache_delete = libvlc.p_module_bank->b_cache_delete;

Gildas Bazin's avatar
 
Gildas Bazin committed
414
        /* Reset the default domain */
415
        SetLanguage( psz_language );
Gildas Bazin's avatar
 
Gildas Bazin committed
416

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

Sam Hocevar's avatar
Sam Hocevar committed
420
        textdomain( PACKAGE_NAME );
Gildas Bazin's avatar
 
Gildas Bazin committed
421

Gildas Bazin's avatar
 
Gildas Bazin committed
422
#if defined( ENABLE_UTF8 )
Sam Hocevar's avatar
Sam Hocevar committed
423
        bind_textdomain_codeset( PACKAGE_NAME, "UTF-8" );
424 425
#endif

Gildas Bazin's avatar
 
Gildas Bazin committed
426
        module_EndBank( p_vlc );
427
        module_InitBank( p_vlc );
428
        config_LoadConfigFile( p_vlc, "main" );
Gildas Bazin's avatar
 
Gildas Bazin committed
429
        config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE );
430
        libvlc.p_module_bank->b_cache_delete = b_cache_delete;
Gildas Bazin's avatar
 
Gildas Bazin committed
431
    }
432
    if( psz_language ) free( psz_language );
433
#endif
Gildas Bazin's avatar
 
Gildas Bazin committed
434

435 436 437 438 439 440
    /*
     * 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.
     */
441 442 443 444 445 446 447
    module_LoadBuiltins( p_vlc );
    module_LoadPlugins( p_vlc );
    if( p_vlc->b_die )
    {
        b_exit = VLC_TRUE;
    }

448
    msg_Dbg( p_vlc, "module bank initialized, found %i modules",
449
                    libvlc.p_module_bank->i_children );
450 451

    /* Hack: insert the help module here */
452
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
453 454 455 456 457
    /* End hack */

    /* Check for help on modules */
    if( (p_tmp = config_GetPsz( p_vlc, "module" )) )
    {
458
        Help( p_vlc, p_tmp );
459
        free( p_tmp );
460
        b_exit = VLC_TRUE;
461 462
    }
    /* Check for long help option */
463
    else if( config_GetInt( p_vlc, "longhelp" ) )
464
    {
465
        Help( p_vlc, "longhelp" );
466
        b_exit = VLC_TRUE;
467 468
    }
    /* Check for module list option */
469
    else if( config_GetInt( p_vlc, "list" ) )
470
    {
471
        ListModules( p_vlc );
472
        b_exit = VLC_TRUE;
473 474
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492
    /* 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 );
    }

493
    /* Hack: remove the help module here */
494
    vlc_object_detach( p_help_module );
495 496
    /* End hack */

497 498
    if( b_exit )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
499 500
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
501
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
502
        if( i_object ) vlc_object_release( p_vlc );
503 504 505
        return VLC_EEXIT;
    }

506 507 508 509 510
    /*
     * Init device values
     */
    InitDeviceValues( p_vlc );

511 512 513
    /*
     * Override default configuration with config file settings
     */
514
    config_LoadConfigFile( p_vlc, NULL );
515

Gildas Bazin's avatar
 
Gildas Bazin committed
516 517 518 519
    /* Hack: insert the help module here */
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
    /* End hack */

520 521 522
    /*
     * Override configuration with command line settings
     */
Sam Hocevar's avatar
Sam Hocevar committed
523
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_FALSE ) )
524 525 526 527 528
    {
#ifdef WIN32
        ShowConsole();
        /* Pause the console because it's destroyed when we exit */
        fprintf( stderr, "The command line options couldn't be loaded, check "
529 530
                 "that they are valid.\n" );
        PauseConsole();
531
#endif
Gildas Bazin's avatar
 
Gildas Bazin committed
532 533 534
        vlc_object_detach( p_help_module );
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
535
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
536
        if( i_object ) vlc_object_release( p_vlc );
537 538 539
        return VLC_EGENERIC;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
540 541 542 543 544 545
    /* 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 */

546 547 548
    /*
     * System specific configuration
     */
Gildas Bazin's avatar
 
Gildas Bazin committed
549
    system_Configure( p_vlc, &i_argc, ppsz_argv );
550

551 552 553
    /*
     * Message queue options
     */
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
554 555

    var_Create( p_vlc, "verbose", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
556 557
    if( config_GetInt( p_vlc, "quiet" ) )
    {
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
558 559 560
        vlc_value_t val;
        val.i_int = -1;
        var_Set( p_vlc, "verbose", val );
561
    }
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
562 563
    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
564

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

567 568 569 570 571
    /*
     * Output messages that may still be in the queue
     */
    msg_Flush( p_vlc );

572
    /* p_vlc initialization. FIXME ? */
573

574
#if defined( __i386__ )
575
    if( !config_GetInt( p_vlc, "mmx" ) )
576
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMX;
577
    if( !config_GetInt( p_vlc, "3dn" ) )
578
        libvlc.i_cpu &= ~CPU_CAPABILITY_3DNOW;
579
    if( !config_GetInt( p_vlc, "mmxext" ) )
580
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMXEXT;
581
    if( !config_GetInt( p_vlc, "sse" ) )
582
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE;
583 584
    if( !config_GetInt( p_vlc, "sse2" ) )
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE2;
585 586
#endif
#if defined( __powerpc__ ) || defined( SYS_DARWIN )
587
    if( !config_GetInt( p_vlc, "altivec" ) )
588
        libvlc.i_cpu &= ~CPU_CAPABILITY_ALTIVEC;
589
#endif
590 591

#define PRINT_CAPABILITY( capability, string )                              \
592
    if( libvlc.i_cpu & capability )                                         \
593 594 595 596 597 598 599 600 601 602 603 604 605 606
    {                                                                       \
        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" );
607
    PRINT_CAPABILITY( CPU_CAPABILITY_SSE2, "SSE2" );
608 609 610 611 612 613 614
    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
615
    p_vlc->p_memcpy_module = module_Need( p_vlc, "memcpy", "$memcpy", 0 );
616

617
    if( p_vlc->pf_memcpy == NULL )
618 619 620
    {
        p_vlc->pf_memcpy = memcpy;
    }
621 622 623 624 625

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

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
627 628 629 630
    /*
     * Initialize hotkey handling
     */
    var_Create( p_vlc, "key-pressed", VLC_VAR_INTEGER );
Gildas Bazin's avatar
 
Gildas Bazin committed
631 632
    p_vlc->p_hotkeys = malloc( sizeof(p_hotkeys) );
    /* Do a copy (we don't need to modify the strings) */
633
    memcpy( p_vlc->p_hotkeys, p_hotkeys, sizeof(p_hotkeys) );
Gildas Bazin's avatar
 
Gildas Bazin committed
634

635 636 637
    /*
     * Initialize playlist and get commandline files
     */
638
    p_playlist = playlist_Create( p_vlc );
639 640 641
    if( !p_playlist )
    {
        msg_Err( p_vlc, "playlist initialization failed" );
642 643
        if( p_vlc->p_memcpy_module != NULL )
        {
644
            module_Unneed( p_vlc, p_vlc->p_memcpy_module );
645
        }
646
        module_EndBank( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
647
        if( i_object ) vlc_object_release( p_vlc );
648 649 650
        return VLC_EGENERIC;
    }

651 652 653 654 655 656 657
    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 );
658

659 660 661 662
    /*
     * Load background interfaces
     */
    psz_modules = config_GetPsz( p_vlc, "extraintf" );
663 664 665 666 667 668 669 670 671 672 673 674 675 676
    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 );
        sprintf( psz_modules, "%s,%s", psz_modules, psz_control );
    }
    else if( psz_control && *psz_control )
    {
        if( psz_modules ) free( psz_modules );
        psz_modules = strdup( psz_control );
    }

677 678 679
    psz_parser = psz_modules;
    while ( psz_parser && *psz_parser )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
680
        char *psz_module, *psz_temp;
681 682 683 684 685 686 687
        psz_module = psz_parser;
        psz_parser = strchr( psz_module, ',' );
        if ( psz_parser )
        {
            *psz_parser = '\0';
            psz_parser++;
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
688 689 690 691
        psz_temp = (char *)malloc( strlen(psz_module) + sizeof(",none") );
        if( psz_temp )
        {
            sprintf( psz_temp, "%s,none", psz_module );
692
            VLC_AddIntf( 0, psz_temp, VLC_FALSE, VLC_FALSE );
Gildas Bazin's avatar
 
Gildas Bazin committed
693 694
            free( psz_temp );
        }
695 696 697 698 699
    }
    if ( psz_modules )
    {
        free( psz_modules );
    }
700

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
701 702 703
    /*
     * Allways load the hotkeys interface if it exists
     */
704
    VLC_AddIntf( 0, "hotkeys,none", VLC_FALSE, VLC_FALSE );
Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
705

706 707 708 709
    /*
     * FIXME: kludge to use a p_vlc-local variable for the Mozilla plugin
     */
    var_Create( p_vlc, "drawable", VLC_VAR_INTEGER );
710 711 712 713 714 715 716 717 718 719 720
    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 );
721

722 723 724 725 726
    /*
     * Get input filenames given as commandline arguments
     */
    GetFilenames( p_vlc, i_argc, ppsz_argv );

Gildas Bazin's avatar
 
Gildas Bazin committed
727
    if( i_object ) vlc_object_release( p_vlc );
728 729 730 731
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
732
 * VLC_AddIntf: add an interface
733 734
 *****************************************************************************
 * This function opens an interface plugin and runs it. If b_block is set
Sam Hocevar's avatar
Sam Hocevar committed
735 736
 * 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
737 738
 * user requests to quit. If b_play is set to 1, VLC_AddIntf will start playing
 * the playlist when it is completely initialised.
739
 *****************************************************************************/
740 741
int VLC_AddIntf( int i_object, char const *psz_module,
                 vlc_bool_t b_block, vlc_bool_t b_play )
742
{
Sam Hocevar's avatar
Sam Hocevar committed
743
    int i_err;
744
    intf_thread_t *p_intf;
745
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
746

747
    if( !p_vlc )
748
    {
749
        return VLC_ENOOBJ;
750 751
    }

Gildas Bazin's avatar
Gildas Bazin committed
752 753 754 755 756 757 758 759 760 761 762
#ifndef WIN32
    if( p_vlc->p_libvlc->b_daemon && b_block && !psz_module )
    {
        /* Daemon mode hack.
         * We prefer the dummy interface if none is specified. */
        char *psz_interface = config_GetPsz( p_vlc, "intf" );
        if( !psz_interface || !*psz_interface ) psz_module = "dummy";
        if( psz_interface ) free( psz_interface );
    }
#endif

763
    /* Try to create the interface */
764
    p_intf = intf_Create( p_vlc, psz_module ? psz_module : "$intf" );
765 766 767

    if( p_intf == NULL )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
768
        msg_Err( p_vlc, "interface \"%s\" initialization failed", psz_module );
Gildas Bazin's avatar
 
Gildas Bazin committed
769
        if( i_object ) vlc_object_release( p_vlc );
770 771 772
        return VLC_EGENERIC;
    }

773 774 775
    /* Interface doesn't handle play on start so do it ourselves */
    if( !p_intf->b_play && b_play ) VLC_Play( i_object );

776
    /* Try to run the interface */
777
    p_intf->b_play = b_play;
778
    p_intf->b_block = b_block;
Sam Hocevar's avatar
Sam Hocevar committed
779 780
    i_err = intf_RunThread( p_intf );
    if( i_err )
781
    {
782
        vlc_object_detach( p_intf );
783
        intf_Destroy( p_intf );
Gildas Bazin's avatar
 
Gildas Bazin committed
784
        if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
785
        return i_err;
786 787
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
788
    if( i_object ) vlc_object_release( p_vlc );
789 790 791 792
    return VLC_SUCCESS;
}

/*****************************************************************************
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 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 892 893 894
 * 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;
    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 );
    }

    /*
     * Free playlists
     */
    msg_Dbg( p_vlc, "removing all playlists" );
    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 );
    }

    /*
     * Free announce handler(s?)
     */
    msg_Dbg( p_vlc, "removing announce handler" );
    while( (p_announce = vlc_object_find( p_vlc, VLC_OBJECT_ANNOUNCE,
                                                 FIND_CHILD ) ) )
   {
        vlc_object_detach( p_announce );
        vlc_object_release( p_announce );
        announce_HandlerDestroy( p_announce );
   }

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

/*****************************************************************************
 * VLC_Destroy: Destroy everything.
895
 *****************************************************************************
896
 * This function requests the running threads to finish, waits for their
897 898
 * termination, and destroys their structure.
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
899
int VLC_Destroy( int i_object )
900
{
901
    vlc_t *p_vlc = vlc_current_object( i_object );
902

903
    if( !p_vlc )
904
    {
905
        return VLC_ENOOBJ;
906 907
    }

908 909 910 911 912 913 914
    /*
     * Free allocated memory
     */
    if( p_vlc->p_memcpy_module )
    {
        module_Unneed( p_vlc, p_vlc->p_memcpy_module );
        p_vlc->p_memcpy_module = NULL;
915 916
    }

917
    /*
918
     * Free module bank !
919 920 921
     */
    module_EndBank( p_vlc );

922 923 924 925 926
    if( p_vlc->psz_homedir )
    {
        free( p_vlc->psz_homedir );
        p_vlc->psz_homedir = NULL;
    }
927

Gildas Bazin's avatar
 
Gildas Bazin committed
928 929 930 931 932 933
    if( p_vlc->psz_configfile )
    {
        free( p_vlc->psz_configfile );
        p_vlc->psz_configfile = NULL;
    }

934 935 936 937 938 939
    if( p_vlc->p_hotkeys )
    {
        free( p_vlc->p_hotkeys );
        p_vlc->p_hotkeys = NULL;
    }

940 941 942 943
    /*
     * System specific cleaning code
     */
    system_End( p_vlc );
944

945 946 947
    /* Destroy mutexes */
    vlc_mutex_destroy( &p_vlc->config_lock );

948 949
    vlc_object_detach( p_vlc );

950 951 952
    /* Release object before destroying it */
    if( i_object ) vlc_object_release( p_vlc );

953 954
    vlc_object_destroy( p_vlc );

955
    /* Stop thread system: last one out please shut the door! */
956
    vlc_threads_end( p_libvlc );
957

958 959 960
    return VLC_SUCCESS;
}

961
/*****************************************************************************
962
 * VLC_VariableSet: set a vlc variable
963
 *****************************************************************************/
964
int VLC_VariableSet( int i_object, char const *psz_var, vlc_value_t value )
965
{
966
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
967
    int i_ret;
968 969 970

    if( !p_vlc )
    {
971
        return VLC_ENOOBJ;
972 973
    }

Sam Hocevar's avatar
Sam Hocevar committed
974 975 976
    /* 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 ) )
977
    {
Sam Hocevar's avatar
Sam Hocevar committed
978
        module_config_t *p_item;
979
        char const *psz_newvar = psz_var + 6;
980

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

Sam Hocevar's avatar
Sam Hocevar committed
983
        if( p_item )
984
        {
Sam Hocevar's avatar
Sam Hocevar committed
985
            switch( p_item->i_type )
986
            {
Sam Hocevar's avatar
Sam Hocevar committed
987 988 989 990 991 992 993 994 995 996 997 998
                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;
999
            }
Gildas Bazin's avatar
 
Gildas Bazin committed
1000
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
1001
            return VLC_SUCCESS;
1002 1003 1004
        }
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
1005 1006 1007 1008
    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
1009 1010 1011
}

/*****************************************************************************
1012
 * VLC_VariableGet: get a vlc variable
Sam Hocevar's avatar
Sam Hocevar committed
1013
 *****************************************************************************/
1014
int VLC_VariableGet( int i_object, char const *psz_var, vlc_value_t *p_value )
Sam Hocevar's avatar
Sam Hocevar committed
1015
{
1016
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
1017
    int i_ret;
Sam Hocevar's avatar
Sam Hocevar committed
1018 1019

    if( !p_vlc )
1020
    {
1021
        return VLC_ENOOBJ;
1022 1023
    }

1024
    i_ret = var_Get( p_vlc , psz_var, p_value );
Gildas Bazin's avatar
 
Gildas Bazin committed
1025 1026 1027

    if( i_object ) vlc_object_release( p_vlc );
    return i_ret;
1028 1029
}

1030
/*****************************************************************************
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1031 1032 1033 1034
 * VLC_AddTarget: adds a target for playing.
 *****************************************************************************
 * This function adds psz_target to the current playlist. If a playlist does
 * not exist, it will create one.
1035
 *****************************************************************************/
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1036 1037 1038
int VLC_AddTarget( int i_object, char const *psz_target,
                   char const **ppsz_options, int i_options,
                   int i_mode, int i_pos )
1039
{
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1040 1041
    int i_err;
    playlist_t *p_playlist;
1042 1043 1044 1045 1046 1047 1048
    vlc_t *p_vlc = vlc_current_object( i_object );

    if( !p_vlc )
    {
        return VLC_ENOOBJ;
    }

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1049
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_ANYWHERE );
1050

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1051
    if( p_playlist == NULL )
1052
    {
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1053 1054 1055 1056 1057 1058 1059 1060 1061 1062
        msg_Dbg( p_vlc, "no playlist present, creating one" );
        p_playlist = playlist_Create( p_vlc );

        if( p_playlist == NULL )
        {
            if( i_object ) vlc_object_release( p_vlc );
            return VLC_EGENERIC;
        }

        vlc_object_yield( p_playlist );
1063 1064
    }

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1065 1066
    i_err = playlist_AddExt( p_playlist, psz_target, psz_target,
                             i_mode, i_pos, -1, ppsz_options, i_options);
1067 1068 1069 1070

    vlc_object_release( p_playlist );

    if( i_object ) vlc_object_release( p_vlc );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1071
    return i_err;
1072 1073 1074
}

/*****************************************************************************
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1075
 * VLC_Play: play
1076
 *****************************************************************************/
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1077
int VLC_Play( int i_object )
1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095
{
    playlist_t * p_playlist;
    vlc_t *p_vlc = vlc_current_object( i_object );

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

    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );

    if( !p_playlist )
    {
        if( i_object ) vlc_object_release( p_vlc );
        return VLC_ENOOBJ;
    }

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1096
    playlist_Play( p_playlist );
1097 1098 1099
    vlc_object_release( p_playlist );

    if( i_object ) vlc_object_release( p_vlc );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1100
    return VLC_SUCCESS;
1101 1102
}

1103
/*****************************************************************************
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1104
 * VLC_Pause: toggle pause
1105
 *****************************************************************************/
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1106
int VLC_Pause( int i_object )
1107 1108
{
    playlist_t * p_playlist;
1109
    vlc_t *p_vlc = vlc_current_object( i_object );
1110 1111

    /* Check that the handle is valid */
1112
    if( !p_vlc )
1113
    {
1114
        return VLC_ENOOBJ;
1115
    }
1116

1117 1118 1119 1120
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );

    if( !p_playlist )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1121
        if( i_object ) vlc_object_release( p_vlc );
1122
        return VLC_ENOOBJ;
1123 1124
    }

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1125
    playlist_Pause( p_playlist );
1126 1127
    vlc_object_release( p_playlist );

Gildas Bazin's avatar
 
Gildas Bazin committed
1128
    if( i_object ) vlc_object_release( p_vlc );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1129
    return VLC_SUCCESS;
1130 1131 1132
}

/*****************************************************************************
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1133
 * VLC_Pause: toggle pause
1134
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1135
int VLC_Stop( int i_object )
1136
{
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
1137