libvlc.c 51 KB
Newer Older
1 2 3
/*****************************************************************************
 * libvlc.c: main libvlc source
 *****************************************************************************
4
 * Copyright (C) 1998-2004 VideoLAN
5
 * $Id: libvlc.c,v 1.117 2004/02/22 15:41:27 massiot 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
    p_vlc->thread_id = 0;
208
    vlc_thread_set_priority( p_vlc, VLC_THREAD_PRIORITY_LOW );
209 210 211 212

    p_vlc->psz_object_name = "root";

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

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

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

    return p_vlc->i_object_id;
225 226 227
}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

Gildas Bazin's avatar
 
Gildas Bazin committed
375
#if defined( ENABLE_UTF8 )
376 377 378
        bind_textdomain_codeset( PACKAGE, "UTF-8" );
#endif

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

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

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

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

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

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

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

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

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

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

467 468 469
    /*
     * Override configuration with command line settings
     */
Sam Hocevar's avatar
Sam Hocevar committed
470
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_FALSE ) )
471 472 473 474 475 476 477 478
    {
#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
479 480 481
        vlc_object_detach( p_help_module );
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
482
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
483
        if( i_object ) vlc_object_release( p_vlc );
484 485 486
        return VLC_EGENERIC;
    }

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

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

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

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

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

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

519
    /* p_vlc initialization. FIXME ? */
520

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

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

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

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

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

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

598 599 600 601 602 603 604
    /*
     * 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
605
        char *psz_module, *psz_temp;
606 607 608 609 610 611 612
        psz_module = psz_parser;
        psz_parser = strchr( psz_module, ',' );
        if ( psz_parser )
        {
            *psz_parser = '\0';
            psz_parser++;
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
613 614 615 616 617 618 619
        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 );
        }
620 621 622 623 624
    }
    if ( psz_modules )
    {
        free( psz_modules );
    }
625

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

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

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

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

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

670
    if( !p_vlc )
671
    {
672
        return VLC_ENOOBJ;
673 674 675
    }

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

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

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

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

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

710
    if( !p_vlc )
711
    {
712
        return VLC_ENOOBJ;
713 714
    }

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

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

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

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

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

747 748 749 750
    /*
     * System specific cleaning code
     */
    system_End( p_vlc );
751

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

758 759
    vlc_object_detach( p_vlc );

760 761 762
    /* Release object before destroying it */
    if( i_object ) vlc_object_release( p_vlc );

763 764
    vlc_object_destroy( p_vlc );

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

768 769 770
    return VLC_SUCCESS;
}

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

    if( !p_vlc )
    {
783
        return VLC_ENOOBJ;
784 785 786 787
    }

    p_vlc->b_die = VLC_TRUE;

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

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
793
 * VLC_AddTarget: adds a target for playing.
794 795 796 797
 *****************************************************************************
 * 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
798 799 800
int VLC_AddTarget( int i_object, char const *psz_target,
                   char const **ppsz_options, int i_options,
                   int i_mode, int i_pos )
801
{
Sam Hocevar's avatar
Sam Hocevar committed
802
    int i_err;
Sam Hocevar's avatar
Sam Hocevar committed
803
    playlist_t *p_playlist;
804
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
805

806
    if( !p_vlc )
807
    {
808
        return VLC_ENOOBJ;
809 810
    }

Sam Hocevar's avatar
Sam Hocevar committed
811 812 813 814 815 816 817 818 819
    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
820
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
821 822 823 824 825 826
            return VLC_EGENERIC;
        }

        vlc_object_yield( p_playlist );
    }

827 828
    i_err = playlist_AddExt( p_playlist, psz_target, psz_target,
                             i_mode, i_pos, -1, ppsz_options, i_options);
829

Sam Hocevar's avatar
Sam Hocevar committed
830
    vlc_object_release( p_playlist );
831

Gildas Bazin's avatar
 
Gildas Bazin committed
832
    if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
833
    return i_err;
834 835
}

836
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
837
 * VLC_Set: set a vlc variable
838 839 840
 *****************************************************************************
 *
 *****************************************************************************/
841
int VLC_Set( int i_object, char const *psz_var, vlc_value_t value )
842
{
843
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
844
    int i_ret;
845 846 847

    if( !p_vlc )
    {
848
        return VLC_ENOOBJ;
849 850
    }

Sam Hocevar's avatar
Sam Hocevar committed
851 852 853
    /* 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 ) )
854
    {
Sam Hocevar's avatar
Sam Hocevar committed
855
        module_config_t *p_item;
856
        char const *psz_newvar = psz_var + 6;
857

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

Sam Hocevar's avatar
Sam Hocevar committed
860
        if( p_item )
861
        {
Sam Hocevar's avatar
Sam Hocevar committed
862
            switch( p_item->i_type )
863
            {
Sam Hocevar's avatar
Sam Hocevar committed
864 865 866 867 868 869 870 871 872 873 874 875
                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;
876
            }
Gildas Bazin's avatar
 
Gildas Bazin committed
877
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
878
            return VLC_SUCCESS;
879 880 881
        }
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
882 883 884 885
    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
886 887 888 889 890 891 892
}

/*****************************************************************************
 * VLC_Get: get a vlc variable
 *****************************************************************************
 *
 *****************************************************************************/
893
int VLC_Get( int i_object, char const *psz_var, vlc_value_t *p_value )
Sam Hocevar's avatar
Sam Hocevar committed
894
{
895
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
896
    int i_ret;
Sam Hocevar's avatar
Sam Hocevar committed
897 898

    if( !p_vlc )
899
    {
900
        return VLC_ENOOBJ;
901 902
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
903 904 905 906
    i_ret = var_Get( p_vlc, psz_var, p_value );

    if( i_object ) vlc_object_release( p_vlc );
    return i_ret;
907 908
}

Sam Hocevar's avatar
Sam Hocevar committed
909
/* FIXME: temporary hacks */
910

911 912 913 914 915 916 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 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

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


982
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
983
 * VLC_Play: play
984
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
985
int VLC_Play( int i_object )
986 987
{
    playlist_t * p_playlist;
988
    vlc_t *p_vlc = vlc_current_object( i_object );
989 990

    /* Check that the handle is valid */
991
    if( !p_vlc )
992
    {
993
        return VLC_ENOOBJ;
994
    }
995

996 997 998 999
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );

    if( !p_playlist )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1000
        if( i_object ) vlc_object_release( p_vlc );
1001
        return VLC_ENOOBJ;
1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016
    }

    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
1017
    if( i_object ) vlc_object_release( p_vlc );
1018 1019 1020 1021
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1022
 * VLC_Stop: stop
1023
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1024
int VLC_Stop( int i_object )
1025 1026 1027 1028 1029
{
    intf_thread_t *   p_intf;
    playlist_t    *   p_playlist;
    vout_thread_t *   p_vout;
    aout_instance_t * p_aout;
1030
    vlc_t *p_vlc = vlc_current_object( i_object );
1031 1032

    /* Check that the handle is valid */
1033
    if( !p_vlc )
1034
    {
1035
        return VLC_ENOOBJ;
1036 1037 1038 1039 1040 1041
    }

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

1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053
    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
1054

1055 1056 1057 1058 1059 1060 1061 1062
    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
1063

1064 1065 1066 1067 1068 1069 1070 1071
    /*
     * 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 );
1072
        vout_Destroy( p_vout );
1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085
    }

    /*
     * 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
1086
    if( i_object ) vlc_object_release( p_vlc );
1087 1088 1089 1090
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1091
 * VLC_Pause: toggle pause
1092
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1093
int VLC_Pause( int i_object )
1094 1095
{
    input_thread_t *p_input;
1096
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
1097 1098 1099

    if( !p_vlc )
    {
1100
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
1101
    }
1102 1103 1104 1105 1106

    p_input = vlc_object_find( p_vlc, VLC_OBJECT_INPUT, FIND_CHILD );

    if( !p_input )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1107
        if( i_object ) vlc_object_release( p_vlc );
1108
        return VLC_ENOOBJ;
1109 1110 1111 1112 1113
    }

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

Gildas Bazin's avatar
 
Gildas Bazin committed
1114
    if( i_object ) vlc_object_release( p_vlc );
1115 1116 1117 1118
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1119
 * VLC_FullScreen: toggle fullscreen mode
1120
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1121
int VLC_FullScreen( int i_object )
1122 1123
{
    vout_thread_t *p_vout;
1124
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
1125 1126 1127

    if( !p_vlc )
    {
1128
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
1129
    }