libvlc.c 51.4 KB
Newer Older
1 2 3
/*****************************************************************************
 * libvlc.c: main libvlc source
 *****************************************************************************
4
 * Copyright (C) 1998-2004 VideoLAN
5
 * $Id: libvlc.c,v 1.114 2004/01/29 14:39:08 sigmunau Exp $
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
 *
 * 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
 * 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
30
#define MODULE_PATH main
31 32 33 34 35
#define __BUILTIN__

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

#ifdef HAVE_ERRNO_H
#   include <errno.h>                                              /* ENOMEM */
#endif
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 63
#ifdef HAVE_LOCALE_H
#   include <locale.h>
#endif

#include "vlc_cpu.h"                                        /* CPU detection */
64
#include "os_specific.h"
65

66
#include "vlc_error.h"
67 68 69 70

#include "stream_control.h"
#include "input_ext-intf.h"

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

#include "audio_output.h"

76
#include "vlc_video.h"
77 78 79 80 81
#include "video_output.h"

#include "libvlc.h"

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

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

#ifdef WIN32
static void ShowConsole   ( void );
#endif
100
static int  ConsoleWidth  ( void );
101

Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
102 103
static int  VerboseCallback( vlc_object_t *, char const *,
                             vlc_value_t, vlc_value_t, void * );
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120

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

121
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
122
 * VLC_Version: return the libvlc version.
123
 *****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
124
 * This function returns full version string (numeric version and codename).
125
 *****************************************************************************/
126
char const * VLC_Version( void )
127
{
Sam Hocevar's avatar
Sam Hocevar committed
128
    return VERSION_MESSAGE;
129 130
}

131 132 133 134 135 136 137 138 139 140
/*****************************************************************************
 * 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
141 142 143 144 145 146 147
/*****************************************************************************
 * 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 )
148
{
149
    int i_ret;
150
    vlc_t * p_vlc = NULL;
151
    vlc_value_t lockval;
152

153 154 155
    /* &libvlc never changes, so we can safely call this multiple times. */
    p_libvlc = &libvlc;

156 157
    /* vlc_threads_init *must* be the first internal call! No other call is
     * allowed before the thread system has been initialized. */
158
    i_ret = vlc_threads_init( p_libvlc );
Sam Hocevar's avatar
Sam Hocevar committed
159
    if( i_ret < 0 )
160
    {
Sam Hocevar's avatar
Sam Hocevar committed
161
        return i_ret;
162 163 164
    }

    /* Now that the thread system is initialized, we don't have much, but
165
     * at least we have var_Create */
166 167
    var_Create( p_libvlc, "libvlc", VLC_VAR_MUTEX );
    var_Get( p_libvlc, "libvlc", &lockval );
168
    vlc_mutex_lock( lockval.p_address );
169 170
    if( !libvlc.b_ready )
    {
171 172
        char *psz_env;

173 174 175
        /* Guess what CPU we have */
        libvlc.i_cpu = CPUCapabilities();

176 177
        /* Find verbosity from VLC_VERBOSE environment variable */
        psz_env = getenv( "VLC_VERBOSE" );
178
        libvlc.i_verbose = psz_env ? atoi( psz_env ) : -1;
179

Gildas Bazin's avatar
 
Gildas Bazin committed
180
#if defined( HAVE_ISATTY ) && !defined( WIN32 )
181 182 183 184 185
        libvlc.b_color = isatty( 2 ); /* 2 is for stderr */
#else
        libvlc.b_color = VLC_FALSE;
#endif

186
        /* Initialize message queue */
187
        msg_Create( p_libvlc );
188 189

        /* Announce who we are */
190 191
        msg_Dbg( p_libvlc, COPYRIGHT_MESSAGE );
        msg_Dbg( p_libvlc, "libvlc was configured with %s", CONFIGURE_LINE );
192

193 194
        /* The module bank will be initialized later */
        libvlc.p_module_bank = NULL;
195 196

        libvlc.b_ready = VLC_TRUE;
197
    }
198
    vlc_mutex_unlock( lockval.p_address );
199
    var_Destroy( p_libvlc, "libvlc" );
200 201

    /* Allocate a vlc object */
202
    p_vlc = vlc_object_create( p_libvlc, VLC_OBJECT_VLC );
203 204
    if( p_vlc == NULL )
    {
Sam Hocevar's avatar
Sam Hocevar committed
205
        return VLC_EGENERIC;
206
    }
207
    vlc_thread_set_priority( p_vlc, VLC_THREAD_PRIORITY_LOW );
208 209 210 211

    p_vlc->psz_object_name = "root";

    /* Initialize mutexes */
212
    vlc_mutex_init( p_vlc, &p_vlc->config_lock );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
213 214 215
#ifdef SYS_DARWIN
    vlc_mutex_init( p_vlc, &p_vlc->quicktime_lock );
#endif
216 217

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

Sam Hocevar's avatar
Sam Hocevar committed
220 221 222 223
    /* Store data for the non-reentrant API */
    p_static_vlc = p_vlc;

    return p_vlc->i_object_id;
224 225 226
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
227
 * VLC_Init: initialize a vlc_t structure.
228 229 230 231 232 233 234
 *****************************************************************************
 * 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
235
int VLC_Init( int i_object, int i_argc, char *ppsz_argv[] )
236
{
237 238
    char         p_capabilities[200];
    char *       p_tmp;
239 240
    char *       psz_modules;
    char *       psz_parser;
241
    char *       psz_language;
242
    vlc_bool_t   b_exit = VLC_FALSE;
243
    vlc_t *      p_vlc = vlc_current_object( i_object );
244 245
    module_t    *p_help_module;
    playlist_t  *p_playlist;
246
    vlc_value_t  lockval;
247

248
    if( !p_vlc )
249
    {
250
        return VLC_ENOOBJ;
251 252 253
    }

    /*
254
     * System specific initialization code
255
     */
256
    system_Init( p_vlc, &i_argc, ppsz_argv );
257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272

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

273 274 275 276
    /*
     * Support for gettext
     */
    SetLanguage( "" );
Sam Hocevar's avatar
Sam Hocevar committed
277 278

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

281 282 283 284
    /* 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) */
285 286
    var_Create( p_libvlc, "libvlc", VLC_VAR_MUTEX );
    var_Get( p_libvlc, "libvlc", &lockval );
287 288 289
    vlc_mutex_lock( lockval.p_address );
    if( libvlc.p_module_bank == NULL )
    {
290 291
        module_InitBank( p_vlc );
        module_LoadMain( p_vlc );
292 293
    }
    vlc_mutex_unlock( lockval.p_address );
294
    var_Destroy( p_libvlc, "libvlc" );
295

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

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

    /* Check for short help option */
    if( config_GetInt( p_vlc, "help" ) )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
323
        fprintf( stdout, _("Usage: %s [options] [items]...\n"),
324
                         p_vlc->psz_object_name );
325
        Usage( p_vlc, "main" );
326 327
        Usage( p_vlc, "help" );
        b_exit = VLC_TRUE;
328 329
    }
    /* Check for version option */
330
    else if( config_GetInt( p_vlc, "version" ) )
331 332
    {
        Version();
333 334 335
        b_exit = VLC_TRUE;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
336 337 338 339
    /* Set the config file stuff */
    p_vlc->psz_homedir = config_GetHomeDir();
    p_vlc->psz_configfile = config_GetPsz( p_vlc, "config" );

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

    if( b_exit )
    {
346 347
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
348
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
349
        if( i_object ) vlc_object_release( p_vlc );
350 351 352
        return VLC_EEXIT;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
353
    /* Check for translation config option */
Gildas Bazin's avatar
 
Gildas Bazin committed
354 355 356 357 358 359 360 361
#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 );

362 363 364
    /* 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
365 366
    {
        /* Reset the default domain */
367
        SetLanguage( psz_language );
Gildas Bazin's avatar
 
Gildas Bazin committed
368

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

372
        textdomain( PACKAGE );
Gildas Bazin's avatar
 
Gildas Bazin committed
373

Gildas Bazin's avatar
 
Gildas Bazin committed
374 375
#if defined( SYS_BEOS ) || defined ( SYS_DARWIN ) || \
    ( defined( WIN32 ) && !defined( HAVE_INCLUDED_GETTEXT ) )
376 377 378 379 380
        /* BeOS only support UTF8 strings */
        /* Mac OS X prefers UTF8 */
        bind_textdomain_codeset( PACKAGE, "UTF-8" );
#endif

Gildas Bazin's avatar
 
Gildas Bazin committed
381
        module_EndBank( p_vlc );
382 383
        module_InitBank( p_vlc );
        module_LoadMain( p_vlc );
Gildas Bazin's avatar
 
Gildas Bazin committed
384
        config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE );
Gildas Bazin's avatar
 
Gildas Bazin committed
385
    }
386
    if( psz_language ) free( psz_language );
387
#endif
Gildas Bazin's avatar
 
Gildas Bazin committed
388

389 390 391 392 393 394
    /*
     * 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.
     */
395 396 397 398 399 400 401
    module_LoadBuiltins( p_vlc );
    module_LoadPlugins( p_vlc );
    if( p_vlc->b_die )
    {
        b_exit = VLC_TRUE;
    }

402
    msg_Dbg( p_vlc, "module bank initialized, found %i modules",
403
                    libvlc.p_module_bank->i_children );
404 405

    /* Hack: insert the help module here */
406
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
407 408 409 410 411
    /* End hack */

    /* Check for help on modules */
    if( (p_tmp = config_GetPsz( p_vlc, "module" )) )
    {
412
        Usage( p_vlc, p_tmp );
413
        free( p_tmp );
414
        b_exit = VLC_TRUE;
415 416
    }
    /* Check for long help option */
417
    else if( config_GetInt( p_vlc, "longhelp" ) )
418
    {
419
        Usage( p_vlc, NULL );
420
        b_exit = VLC_TRUE;
421 422
    }
    /* Check for module list option */
423
    else if( config_GetInt( p_vlc, "list" ) )
424
    {
425
        ListModules( p_vlc );
426
        b_exit = VLC_TRUE;
427 428
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446
    /* 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 );
    }

447
    /* Hack: remove the help module here */
448
    vlc_object_detach( p_help_module );
449 450
    /* End hack */

451 452
    if( b_exit )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
453 454
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
455
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
456
        if( i_object ) vlc_object_release( p_vlc );
457 458 459
        return VLC_EEXIT;
    }

460 461 462
    /*
     * Override default configuration with config file settings
     */
463
    config_LoadConfigFile( p_vlc, NULL );
464

Gildas Bazin's avatar
 
Gildas Bazin committed
465 466 467 468
    /* Hack: insert the help module here */
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
    /* End hack */

469 470 471
    /*
     * Override configuration with command line settings
     */
Sam Hocevar's avatar
Sam Hocevar committed
472
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_FALSE ) )
473 474 475 476 477 478 479 480
    {
#ifdef WIN32
        ShowConsole();
        /* Pause the console because it's destroyed when we exit */
        fprintf( stderr, "The command line options couldn't be loaded, check "
                 "that they are valid.\nPress the RETURN key to continue..." );
        getchar();
#endif
Gildas Bazin's avatar
 
Gildas Bazin committed
481 482 483
        vlc_object_detach( p_help_module );
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
484
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
485
        if( i_object ) vlc_object_release( p_vlc );
486 487 488
        return VLC_EGENERIC;
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
489 490 491 492 493 494
    /* 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 */

495 496 497
    /*
     * System specific configuration
     */
Gildas Bazin's avatar
 
Gildas Bazin committed
498
    system_Configure( p_vlc, &i_argc, ppsz_argv );
499

500 501 502
    /*
     * Message queue options
     */
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
503 504

    var_Create( p_vlc, "verbose", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
505 506
    if( config_GetInt( p_vlc, "quiet" ) )
    {
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
507 508 509
        vlc_value_t val;
        val.i_int = -1;
        var_Set( p_vlc, "verbose", val );
510
    }
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
511 512
    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
513

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

516 517 518 519 520
    /*
     * Output messages that may still be in the queue
     */
    msg_Flush( p_vlc );

521
    /* p_vlc initialization. FIXME ? */
522

523
#if defined( __i386__ )
524
    if( !config_GetInt( p_vlc, "mmx" ) )
525
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMX;
526
    if( !config_GetInt( p_vlc, "3dn" ) )
527
        libvlc.i_cpu &= ~CPU_CAPABILITY_3DNOW;
528
    if( !config_GetInt( p_vlc, "mmxext" ) )
529
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMXEXT;
530
    if( !config_GetInt( p_vlc, "sse" ) )
531
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE;
532 533
    if( !config_GetInt( p_vlc, "sse2" ) )
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE2;
534 535
#endif
#if defined( __powerpc__ ) || defined( SYS_DARWIN )
536
    if( !config_GetInt( p_vlc, "altivec" ) )
537
        libvlc.i_cpu &= ~CPU_CAPABILITY_ALTIVEC;
538
#endif
539 540

#define PRINT_CAPABILITY( capability, string )                              \
541
    if( libvlc.i_cpu & capability )                                         \
542 543 544 545 546 547 548 549 550 551 552 553 554 555
    {                                                                       \
        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" );
556
    PRINT_CAPABILITY( CPU_CAPABILITY_SSE2, "SSE2" );
557 558 559 560 561 562 563
    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
     */
564
    p_vlc->p_memcpy_module = module_Need( p_vlc, "memcpy", "$memcpy" );
565

566
    if( p_vlc->pf_memcpy == NULL )
567 568 569
    {
        p_vlc->pf_memcpy = memcpy;
    }
570 571 572 573 574

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

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
576 577 578 579
    /*
     * Initialize hotkey handling
     */
    var_Create( p_vlc, "key-pressed", VLC_VAR_INTEGER );
Gildas Bazin's avatar
 
Gildas Bazin committed
580 581
    p_vlc->p_hotkeys = malloc( sizeof(p_hotkeys) );
    /* Do a copy (we don't need to modify the strings) */
582
    memcpy( p_vlc->p_hotkeys, p_hotkeys, sizeof(p_hotkeys) );
Gildas Bazin's avatar
 
Gildas Bazin committed
583

584 585 586
    /*
     * Initialize playlist and get commandline files
     */
587
    p_playlist = playlist_Create( p_vlc );
588 589 590
    if( !p_playlist )
    {
        msg_Err( p_vlc, "playlist initialization failed" );
591 592
        if( p_vlc->p_memcpy_module != NULL )
        {
593
            module_Unneed( p_vlc, p_vlc->p_memcpy_module );
594
        }
595
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
596
        if( i_object ) vlc_object_release( p_vlc );
597 598 599
        return VLC_EGENERIC;
    }

600 601 602 603 604 605 606
    /*
     * Load background interfaces
     */
    psz_modules = config_GetPsz( p_vlc, "extraintf" );
    psz_parser = psz_modules;
    while ( psz_parser && *psz_parser )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
607
        char *psz_module, *psz_temp;
608 609 610 611 612 613 614
        psz_module = psz_parser;
        psz_parser = strchr( psz_module, ',' );
        if ( psz_parser )
        {
            *psz_parser = '\0';
            psz_parser++;
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
615 616 617 618 619 620 621
        psz_temp = (char *)malloc( strlen(psz_module) + sizeof(",none") );
        if( psz_temp )
        {
            sprintf( psz_temp, "%s,none", psz_module );
            VLC_AddIntf( 0, psz_temp, VLC_FALSE );
            free( psz_temp );
        }
622 623 624 625 626
    }
    if ( psz_modules )
    {
        free( psz_modules );
    }
627

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
628 629 630 631 632
    /*
     * Allways load the hotkeys interface if it exists
     */
    VLC_AddIntf( 0, "hotkeys,none", VLC_FALSE );

633 634 635 636
    /*
     * FIXME: kludge to use a p_vlc-local variable for the Mozilla plugin
     */
    var_Create( p_vlc, "drawable", VLC_VAR_INTEGER );
637 638 639 640 641 642 643 644 645 646 647
    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 );
648

649 650 651 652 653
    /*
     * Get input filenames given as commandline arguments
     */
    GetFilenames( p_vlc, i_argc, ppsz_argv );

Gildas Bazin's avatar
 
Gildas Bazin committed
654
    if( i_object ) vlc_object_release( p_vlc );
655 656 657 658
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
659
 * VLC_AddIntf: add an interface
660 661
 *****************************************************************************
 * This function opens an interface plugin and runs it. If b_block is set
Sam Hocevar's avatar
Sam Hocevar committed
662 663
 * 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
664 665
 * user requests to quit.
 *****************************************************************************/
666
int VLC_AddIntf( int i_object, char const *psz_module, vlc_bool_t b_block )
667
{
Sam Hocevar's avatar
Sam Hocevar committed
668
    int i_err;
669
    intf_thread_t *p_intf;
670
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
671

672
    if( !p_vlc )
673
    {
674
        return VLC_ENOOBJ;
675 676 677
    }

    /* Try to create the interface */
678
    p_intf = intf_Create( p_vlc, psz_module ? psz_module : "$intf" );
679 680 681

    if( p_intf == NULL )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
682
        msg_Err( p_vlc, "interface \"%s\" initialization failed", psz_module );
Gildas Bazin's avatar
 
Gildas Bazin committed
683
        if( i_object ) vlc_object_release( p_vlc );
684 685 686 687 688
        return VLC_EGENERIC;
    }

    /* Try to run the interface */
    p_intf->b_block = b_block;
Sam Hocevar's avatar
Sam Hocevar committed
689 690
    i_err = intf_RunThread( p_intf );
    if( i_err )
691
    {
692
        vlc_object_detach( p_intf );
693
        intf_Destroy( p_intf );
Gildas Bazin's avatar
 
Gildas Bazin committed
694
        if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
695
        return i_err;
696 697
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
698
    if( i_object ) vlc_object_release( p_vlc );
699 700 701 702
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
703
 * VLC_Destroy: stop playing and destroy everything.
704
 *****************************************************************************
705
 * This function requests the running threads to finish, waits for their
706 707
 * termination, and destroys their structure.
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
708
int VLC_Destroy( int i_object )
709
{
710
    vlc_t *p_vlc = vlc_current_object( i_object );
711

712
    if( !p_vlc )
713
    {
714
        return VLC_ENOOBJ;
715 716
    }

717 718 719 720 721 722 723
    /*
     * Free allocated memory
     */
    if( p_vlc->p_memcpy_module )
    {
        module_Unneed( p_vlc, p_vlc->p_memcpy_module );
        p_vlc->p_memcpy_module = NULL;
724 725
    }

726 727 728 729 730
    if( p_vlc->psz_homedir )
    {
        free( p_vlc->psz_homedir );
        p_vlc->psz_homedir = NULL;
    }
731

Gildas Bazin's avatar
 
Gildas Bazin committed
732 733 734 735 736 737
    if( p_vlc->psz_configfile )
    {
        free( p_vlc->psz_configfile );
        p_vlc->psz_configfile = NULL;
    }

738 739 740 741 742 743
    if( p_vlc->p_hotkeys )
    {
        free( p_vlc->p_hotkeys );
        p_vlc->p_hotkeys = NULL;
    }

744 745 746
    /*
     * XXX: Free module bank !
     */
747
    /*module_EndBank( p_vlc );*/
748

749 750 751 752
    /*
     * System specific cleaning code
     */
    system_End( p_vlc );
753

754 755
    /* Destroy mutexes */
    vlc_mutex_destroy( &p_vlc->config_lock );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
756 757 758
#ifdef SYS_DARWIN
    vlc_mutex_destroy( &p_vlc->quicktime_lock );
#endif
759

760 761
    vlc_object_detach( p_vlc );

762 763 764
    /* Release object before destroying it */
    if( i_object ) vlc_object_release( p_vlc );

765 766
    vlc_object_destroy( p_vlc );

767
    /* Stop thread system: last one out please shut the door! */
768
    vlc_threads_end( p_libvlc );
769

770 771 772
    return VLC_SUCCESS;
}

773
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
774
 * VLC_Die: ask vlc to die.
775 776
 *****************************************************************************
 * This function sets p_vlc->b_die to VLC_TRUE, but does not do any other
777
 * task. It is your duty to call VLC_End and VLC_Destroy afterwards.
778
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
779
int VLC_Die( int i_object )
780
{
781
    vlc_t *p_vlc = vlc_current_object( i_object );
782 783 784

    if( !p_vlc )
    {
785
        return VLC_ENOOBJ;
786 787 788 789
    }

    p_vlc->b_die = VLC_TRUE;

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

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
795
 * VLC_AddTarget: adds a target for playing.
796 797 798 799
 *****************************************************************************
 * This function adds psz_target to the current playlist. If a playlist does
 * not exist, it will create one.
 *****************************************************************************/
Gildas Bazin's avatar
 
Gildas Bazin committed
800 801 802
int VLC_AddTarget( int i_object, char const *psz_target,
                   char const **ppsz_options, int i_options,
                   int i_mode, int i_pos )
803
{
804
    int i;
Sam Hocevar's avatar
Sam Hocevar committed
805
    int i_err;
Sam Hocevar's avatar
Sam Hocevar committed
806
    playlist_t *p_playlist;
807
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
808

809
    if( !p_vlc )
810
    {
811
        return VLC_ENOOBJ;
812 813
    }

Sam Hocevar's avatar
Sam Hocevar committed
814 815 816 817 818 819 820 821 822
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_ANYWHERE );

    if( p_playlist == NULL )
    {
        msg_Dbg( p_vlc, "no playlist present, creating one" );
        p_playlist = playlist_Create( p_vlc );

        if( p_playlist == NULL )
        {
Gildas Bazin's avatar
 
Gildas Bazin committed
823
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
824 825 826 827 828 829
            return VLC_EGENERIC;
        }

        vlc_object_yield( p_playlist );
    }

830
    i_err = playlist_Add( p_playlist, psz_target, psz_target,
Gildas Bazin's avatar
 
Gildas Bazin committed
831
                          i_mode, i_pos );
Sam Hocevar's avatar
Sam Hocevar committed
832

833 834 835 836 837
    for( i = 0 ; i< i_options ; i++ )
    {
        playlist_AddOption( p_playlist, i_err , ppsz_options[i] );
    }

Sam Hocevar's avatar
Sam Hocevar committed
838
    vlc_object_release( p_playlist );
839

Gildas Bazin's avatar
 
Gildas Bazin committed
840
    if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
841
    return i_err;
842 843
}

844
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
845
 * VLC_Set: set a vlc variable
846 847 848
 *****************************************************************************
 *
 *****************************************************************************/
849
int VLC_Set( int i_object, char const *psz_var, vlc_value_t value )
850
{
851
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
852
    int i_ret;
853 854 855

    if( !p_vlc )
    {
856
        return VLC_ENOOBJ;
857 858
    }

Sam Hocevar's avatar
Sam Hocevar committed
859 860 861
    /* 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 ) )
862
    {
Sam Hocevar's avatar
Sam Hocevar committed
863
        module_config_t *p_item;
864
        char const *psz_newvar = psz_var + 6;
865

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

Sam Hocevar's avatar
Sam Hocevar committed
868
        if( p_item )
869
        {
Sam Hocevar's avatar
Sam Hocevar committed
870
            switch( p_item->i_type )
871
            {
Sam Hocevar's avatar
Sam Hocevar committed
872 873 874 875 876 877 878 879 880 881 882 883
                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;
884
            }
Gildas Bazin's avatar
 
Gildas Bazin committed
885
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
886
            return VLC_SUCCESS;
887 888 889
        }
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
890 891 892 893
    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
894 895 896 897 898 899 900
}

/*****************************************************************************
 * VLC_Get: get a vlc variable
 *****************************************************************************
 *
 *****************************************************************************/
901
int VLC_Get( int i_object, char const *psz_var, vlc_value_t *p_value )
Sam Hocevar's avatar
Sam Hocevar committed
902
{
903
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
904
    int i_ret;
Sam Hocevar's avatar
Sam Hocevar committed
905 906

    if( !p_vlc )
907
    {
908
        return VLC_ENOOBJ;
909 910
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
911 912 913 914
    i_ret = var_Get( p_vlc, psz_var, p_value );

    if( i_object ) vlc_object_release( p_vlc );
    return i_ret;
915 916
}

Sam Hocevar's avatar
Sam Hocevar committed
917
/* FIXME: temporary hacks */
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 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989

/*****************************************************************************
 * VLC_IsPlaying: Query for Playlist Status
 *****************************************************************************/
vlc_bool_t VLC_IsPlaying( int i_object )
{

    playlist_t * p_playlist;
    vlc_bool_t   playing;

    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;
    }

    playing = playlist_IsPlaying( p_playlist );

    vlc_object_release( p_playlist );

    if( i_object ) vlc_object_release( p_vlc );

    return playing;

}


/*****************************************************************************
 * VLC_ClearPlaylist: Query for Playlist Status
 *
 * return: 0
 *****************************************************************************/
int VLC_ClearPlaylist( int i_object )
{

    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;
    }

    playlist_Clear(p_playlist);

    vlc_object_release( p_playlist );

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


990
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
991
 * VLC_Play: play
992
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
993
int VLC_Play( int i_object )
994 995
{
    playlist_t * p_playlist;
996
    vlc_t *p_vlc = vlc_current_object( i_object );
997 998

    /* Check that the handle is valid */
999
    if( !p_vlc )
1000
    {
1001
        return VLC_ENOOBJ;
1002
    }
1003

1004 1005 1006 1007
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );

    if( !p_playlist )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1008
        if( i_object ) vlc_object_release( p_vlc );
1009
        return VLC_ENOOBJ;
1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024
    }

    vlc_mutex_lock( &p_playlist->object_lock );
    if( p_playlist->i_size )
    {
        vlc_mutex_unlock( &p_playlist->object_lock );
        playlist_Play( p_playlist );
    }
    else
    {
        vlc_mutex_unlock( &p_playlist->object_lock );
    }

    vlc_object_release( p_playlist );

Gildas Bazin's avatar
 
Gildas Bazin committed
1025
    if( i_object ) vlc_object_release( p_vlc );
1026 1027 1028 1029
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1030
 * VLC_Stop: stop
1031
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1032
int VLC_Stop( int i_object )
1033 1034 1035 1036 1037
{
    intf_thread_t *   p_intf;
    playlist_t    *   p_playlist;
    vout_thread_t *   p_vout;
    aout_instance_t * p_aout;
1038
    vlc_t *p_vlc = vlc_current_object( i_object );
1039 1040

    /* Check that the handle is valid */
1041
    if( !p_vlc )
1042
    {
1043
        return VLC_ENOOBJ;
1044 1045 1046 1047 1048 1049
    }

    /*
     * Ask the interfaces to stop and destroy them
     */
    msg_Dbg( p_vlc, "removing all interfaces" );
1050

1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061
    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
     */
Clément Stenac's avatar
Clément Stenac committed
1062

1063 1064 1065 1066 1067 1068 1069 1070
    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 );
    }
Clément Stenac's avatar
Clément Stenac committed
1071

1072 1073 1074 1075 1076 1077 1078 1079
    /*
     * 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 );
1080
        vout_Destroy( p_vout );
1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093
    }

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

Gildas Bazin's avatar
 
Gildas Bazin committed
1094
    if( i_object ) vlc_object_release( p_vlc );
1095 1096 1097 1098
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1099
 * VLC_Pause: toggle pause
1100
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1101
int VLC_Pause( int i_object )
1102 1103
{
    input_thread_t *p_input;
1104
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
1105 1106 1107

    if( !p_vlc )
    {
1108
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
1109
    }
1110 1111 1112 1113 1114

    p_input = vlc_object_find( p_vlc, VLC_OBJECT_INPUT, FIND_CHILD );

    if( !p_input )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1115
        if( i_object ) vlc_object_release( p_vlc );
1116
        return VLC_ENOOBJ;
1117 1118 1119 1120 1121
    }

    input_SetStatus( p_input, INPUT_STATUS_PAUSE );
    vlc_object_release( p_input );

Gildas Bazin's avatar
 
Gildas Bazin committed
1122
    if( i_object ) vlc_object_release( p_vlc );
1123 1124 1125 1126
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1127
 * VLC_FullScreen: toggle fullscreen mode
1128
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1129
int VLC_FullScreen( int i_object )
1130 1131
{
    vout_thread_t *p_vout;
1132
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
1133 1134 1135

    if( !