libvlc.c 47.9 KB
Newer Older
1 2 3
/*****************************************************************************
 * libvlc.c: main libvlc source
 *****************************************************************************
4
 * Copyright (C) 1998-2002 VideoLAN
Gildas Bazin's avatar
 
Gildas Bazin committed
5
 * $Id: libvlc.c,v 1.100 2003/10/27 21:54:10 gbazin 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

102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118

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

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

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

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

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

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

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

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

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

184
        /* Initialize message queue */
185
        msg_Create( p_libvlc );
186 187

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

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

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

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

    p_vlc->psz_object_name = "root";

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

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

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

    return p_vlc->i_object_id;
222 223 224
}

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

246
    if( !p_vlc )
247
    {
248
        return VLC_ENOOBJ;
249 250 251
    }

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

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

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

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

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

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

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

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

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

337 338 339 340 341 342
    /* Hack: remove the help module here */
    vlc_object_detach( p_help_module );
    /* End hack */

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

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

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

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

369
        textdomain( PACKAGE );
Gildas Bazin's avatar
 
Gildas Bazin committed
370

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

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

386 387 388 389 390 391
    /*
     * 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.
     */
392 393
    module_LoadBuiltins( p_libvlc );
    module_LoadPlugins( p_libvlc );
394
    msg_Dbg( p_vlc, "module bank initialized, found %i modules",
395
                    libvlc.p_module_bank->i_children );
396 397

    /* Hack: insert the help module here */
398
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
399 400 401 402 403
    /* End hack */

    /* Check for help on modules */
    if( (p_tmp = config_GetPsz( p_vlc, "module" )) )
    {
404
        Usage( p_vlc, p_tmp );
405
        free( p_tmp );
406
        b_exit = VLC_TRUE;
407 408
    }
    /* Check for long help option */
409
    else if( config_GetInt( p_vlc, "longhelp" ) )
410
    {
411
        Usage( p_vlc, NULL );
412
        b_exit = VLC_TRUE;
413 414
    }
    /* Check for module list option */
415
    else if( config_GetInt( p_vlc, "list" ) )
416
    {
417
        ListModules( p_vlc );
418
        b_exit = VLC_TRUE;
419 420
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438
    /* 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 );
    }

439
    /* Hack: remove the help module here */
440
    vlc_object_detach( p_help_module );
441 442
    /* End hack */

443 444
    if( b_exit )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
445 446
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
447
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
448
        if( i_object ) vlc_object_release( p_vlc );
449 450 451
        return VLC_EEXIT;
    }

452 453 454
    /*
     * Override default configuration with config file settings
     */
455
    config_LoadConfigFile( p_vlc, NULL );
456

Gildas Bazin's avatar
 
Gildas Bazin committed
457 458 459 460
    /* Hack: insert the help module here */
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
    /* End hack */

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

Gildas Bazin's avatar
 
Gildas Bazin committed
481 482 483 484 485 486
    /* 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 */

487 488 489
    /*
     * System specific configuration
     */
Gildas Bazin's avatar
 
Gildas Bazin committed
490
    system_Configure( p_vlc, &i_argc, ppsz_argv );
491

492 493 494 495 496
    /*
     * Message queue options
     */
    if( config_GetInt( p_vlc, "quiet" ) )
    {
497
        libvlc.i_verbose = -1;
498 499 500 501
    }
    else
    {
        int i_tmp = config_GetInt( p_vlc, "verbose" );
502
        if( i_tmp >= 0 )
503
        {
504
            libvlc.i_verbose = __MIN( i_tmp, 2 );
505 506
        }
    }
Gildas Bazin's avatar
 
Gildas Bazin committed
507
    libvlc.b_color = libvlc.b_color && config_GetInt( p_vlc, "color" );
508

509 510 511 512 513
    /*
     * Output messages that may still be in the queue
     */
    msg_Flush( p_vlc );

514
    /* p_vlc initialization. FIXME ? */
515

516
#if defined( __i386__ )
517
    if( !config_GetInt( p_vlc, "mmx" ) )
518
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMX;
519
    if( !config_GetInt( p_vlc, "3dn" ) )
520
        libvlc.i_cpu &= ~CPU_CAPABILITY_3DNOW;
521
    if( !config_GetInt( p_vlc, "mmxext" ) )
522
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMXEXT;
523
    if( !config_GetInt( p_vlc, "sse" ) )
524
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE;
525 526
#endif
#if defined( __powerpc__ ) || defined( SYS_DARWIN )
527
    if( !config_GetInt( p_vlc, "altivec" ) )
528
        libvlc.i_cpu &= ~CPU_CAPABILITY_ALTIVEC;
529
#endif
530 531

#define PRINT_CAPABILITY( capability, string )                              \
532
    if( libvlc.i_cpu & capability )                                         \
533 534 535 536 537 538 539 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" );
    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
     */
554
    p_vlc->p_memcpy_module = module_Need( p_vlc, "memcpy", "$memcpy" );
555

556
    if( p_vlc->pf_memcpy == NULL )
557 558 559
    {
        p_vlc->pf_memcpy = memcpy;
    }
560 561 562 563 564

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

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
566 567 568 569
    /*
     * Initialize hotkey handling
     */
    var_Create( p_vlc, "key-pressed", VLC_VAR_INTEGER );
570 571 572
    /*
     * Initialize playlist and get commandline files
     */
573
    p_playlist = playlist_Create( p_vlc );
574 575 576
    if( !p_playlist )
    {
        msg_Err( p_vlc, "playlist initialization failed" );
577 578
        if( p_vlc->p_memcpy_module != NULL )
        {
579
            module_Unneed( p_vlc, p_vlc->p_memcpy_module );
580
        }
581
        /*module_EndBank( p_vlc );*/
Gildas Bazin's avatar
 
Gildas Bazin committed
582
        if( i_object ) vlc_object_release( p_vlc );
583 584 585
        return VLC_EGENERIC;
    }

586 587 588 589 590 591 592
    /*
     * 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
593
        char *psz_module, *psz_temp;
594 595 596 597 598 599 600
        psz_module = psz_parser;
        psz_parser = strchr( psz_module, ',' );
        if ( psz_parser )
        {
            *psz_parser = '\0';
            psz_parser++;
        }
Gildas Bazin's avatar
 
Gildas Bazin committed
601 602 603 604 605 606 607
        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 );
        }
608 609 610 611 612
    }
    if ( psz_modules )
    {
        free( psz_modules );
    }
613

Sigmund Augdal Helberg's avatar
Sigmund Augdal Helberg committed
614 615 616 617 618
    /*
     * Allways load the hotkeys interface if it exists
     */
    VLC_AddIntf( 0, "hotkeys,none", VLC_FALSE );

619 620 621 622
    /*
     * FIXME: kludge to use a p_vlc-local variable for the Mozilla plugin
     */
    var_Create( p_vlc, "drawable", VLC_VAR_INTEGER );
623 624 625 626 627 628 629 630 631 632 633
    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 );
634

635 636 637 638 639
    /*
     * Get input filenames given as commandline arguments
     */
    GetFilenames( p_vlc, i_argc, ppsz_argv );

Gildas Bazin's avatar
 
Gildas Bazin committed
640
    if( i_object ) vlc_object_release( p_vlc );
641 642 643 644
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
645
 * VLC_AddIntf: add an interface
646 647
 *****************************************************************************
 * This function opens an interface plugin and runs it. If b_block is set
Sam Hocevar's avatar
Sam Hocevar committed
648 649
 * 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
650 651
 * user requests to quit.
 *****************************************************************************/
652
int VLC_AddIntf( int i_object, char const *psz_module, vlc_bool_t b_block )
653
{
Sam Hocevar's avatar
Sam Hocevar committed
654
    int i_err;
655
    intf_thread_t *p_intf;
656
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
657

658
    if( !p_vlc )
659
    {
660
        return VLC_ENOOBJ;
661 662 663
    }

    /* Try to create the interface */
664
    p_intf = intf_Create( p_vlc, psz_module ? psz_module : "$intf" );
665 666 667

    if( p_intf == NULL )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
668
        msg_Err( p_vlc, "interface \"%s\" initialization failed", psz_module );
Gildas Bazin's avatar
 
Gildas Bazin committed
669
        if( i_object ) vlc_object_release( p_vlc );
670 671 672 673 674
        return VLC_EGENERIC;
    }

    /* Try to run the interface */
    p_intf->b_block = b_block;
Sam Hocevar's avatar
Sam Hocevar committed
675 676
    i_err = intf_RunThread( p_intf );
    if( i_err )
677
    {
678
        vlc_object_detach( p_intf );
679
        intf_Destroy( p_intf );
Gildas Bazin's avatar
 
Gildas Bazin committed
680
        if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
681
        return i_err;
682 683
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
684
    if( i_object ) vlc_object_release( p_vlc );
685 686 687 688
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
689
 * VLC_Destroy: stop playing and destroy everything.
690
 *****************************************************************************
691
 * This function requests the running threads to finish, waits for their
692 693
 * termination, and destroys their structure.
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
694
int VLC_Destroy( int i_object )
695
{
696
    vlc_t *p_vlc = vlc_current_object( i_object );
697

698
    if( !p_vlc )
699
    {
700
        return VLC_ENOOBJ;
701 702
    }

703 704 705 706 707 708 709
    /*
     * Free allocated memory
     */
    if( p_vlc->p_memcpy_module )
    {
        module_Unneed( p_vlc, p_vlc->p_memcpy_module );
        p_vlc->p_memcpy_module = NULL;
710 711
    }

712 713 714 715 716
    if( p_vlc->psz_homedir )
    {
        free( p_vlc->psz_homedir );
        p_vlc->psz_homedir = NULL;
    }
717

Gildas Bazin's avatar
 
Gildas Bazin committed
718 719 720 721 722 723
    if( p_vlc->psz_configfile )
    {
        free( p_vlc->psz_configfile );
        p_vlc->psz_configfile = NULL;
    }

724 725 726
    /*
     * XXX: Free module bank !
     */
727
    /*module_EndBank( p_vlc );*/
728

729 730 731 732
    /*
     * System specific cleaning code
     */
    system_End( p_vlc );
733

734 735
    /* Destroy mutexes */
    vlc_mutex_destroy( &p_vlc->config_lock );
Derk-Jan Hartman's avatar
Derk-Jan Hartman committed
736 737 738
#ifdef SYS_DARWIN
    vlc_mutex_destroy( &p_vlc->quicktime_lock );
#endif
739

740 741
    vlc_object_detach( p_vlc );

742 743 744
    /* Release object before destroying it */
    if( i_object ) vlc_object_release( p_vlc );

745 746
    vlc_object_destroy( p_vlc );

747
    /* Stop thread system: last one out please shut the door! */
748
    vlc_threads_end( p_libvlc );
749

750 751 752
    return VLC_SUCCESS;
}

753
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
754
 * VLC_Die: ask vlc to die.
755 756
 *****************************************************************************
 * This function sets p_vlc->b_die to VLC_TRUE, but does not do any other
Sam Hocevar's avatar
Sam Hocevar committed
757
 * task. It is your duty to call vlc_end and VLC_Destroy afterwards.
758
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
759
int VLC_Die( int i_object )
760
{
761
    vlc_t *p_vlc = vlc_current_object( i_object );
762 763 764

    if( !p_vlc )
    {
765
        return VLC_ENOOBJ;
766 767 768 769
    }

    p_vlc->b_die = VLC_TRUE;

Gildas Bazin's avatar
 
Gildas Bazin committed
770
    if( i_object ) vlc_object_release( p_vlc );
771 772 773 774
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
775
 * VLC_AddTarget: adds a target for playing.
776 777 778 779
 *****************************************************************************
 * 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
780 781 782
int VLC_AddTarget( int i_object, char const *psz_target,
                   char const **ppsz_options, int i_options,
                   int i_mode, int i_pos )
783
{
Sam Hocevar's avatar
Sam Hocevar committed
784
    int i_err;
Sam Hocevar's avatar
Sam Hocevar committed
785
    playlist_t *p_playlist;
786
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
787

788
    if( !p_vlc )
789
    {
790
        return VLC_ENOOBJ;
791 792
    }

Sam Hocevar's avatar
Sam Hocevar committed
793 794 795 796 797 798 799 800 801
    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
802
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
803 804 805 806 807 808
            return VLC_EGENERIC;
        }

        vlc_object_yield( p_playlist );
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
809 810
    i_err = playlist_Add( p_playlist, psz_target, ppsz_options, i_options,
                          i_mode, i_pos );
Sam Hocevar's avatar
Sam Hocevar committed
811 812

    vlc_object_release( p_playlist );
813

Gildas Bazin's avatar
 
Gildas Bazin committed
814
    if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
815
    return i_err;
816 817
}

818
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
819
 * VLC_Set: set a vlc variable
820 821 822
 *****************************************************************************
 *
 *****************************************************************************/
823
int VLC_Set( int i_object, char const *psz_var, vlc_value_t value )
824
{
825
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
826
    int i_ret;
827 828 829

    if( !p_vlc )
    {
830
        return VLC_ENOOBJ;
831 832
    }

Sam Hocevar's avatar
Sam Hocevar committed
833 834 835
    /* 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 ) )
836
    {
Sam Hocevar's avatar
Sam Hocevar committed
837
        module_config_t *p_item;
838
        char const *psz_newvar = psz_var + 6;
839

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

Sam Hocevar's avatar
Sam Hocevar committed
842
        if( p_item )
843
        {
Sam Hocevar's avatar
Sam Hocevar committed
844
            switch( p_item->i_type )
845
            {
Sam Hocevar's avatar
Sam Hocevar committed
846 847 848 849 850 851 852 853 854 855 856 857
                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;
858
            }
Gildas Bazin's avatar
 
Gildas Bazin committed
859
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
860
            return VLC_SUCCESS;
861 862 863
        }
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
864 865 866 867
    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
868 869 870 871 872 873 874
}

/*****************************************************************************
 * VLC_Get: get a vlc variable
 *****************************************************************************
 *
 *****************************************************************************/
875
int VLC_Get( int i_object, char const *psz_var, vlc_value_t *p_value )
Sam Hocevar's avatar
Sam Hocevar committed
876
{
877
    vlc_t *p_vlc = vlc_current_object( i_object );
Gildas Bazin's avatar
 
Gildas Bazin committed
878
    int i_ret;
Sam Hocevar's avatar
Sam Hocevar committed
879 880

    if( !p_vlc )
881
    {
882
        return VLC_ENOOBJ;
883 884
    }

Gildas Bazin's avatar
 
Gildas Bazin committed
885 886 887 888
    i_ret = var_Get( p_vlc, psz_var, p_value );

    if( i_object ) vlc_object_release( p_vlc );
    return i_ret;
889 890
}

Sam Hocevar's avatar
Sam Hocevar committed
891
/* FIXME: temporary hacks */
892 893

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
894
 * VLC_Play: play
895
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
896
int VLC_Play( int i_object )
897 898
{
    playlist_t * p_playlist;
899
    vlc_t *p_vlc = vlc_current_object( i_object );
900 901

    /* Check that the handle is valid */
902
    if( !p_vlc )
903
    {
904
        return VLC_ENOOBJ;
905
    }
906

907 908 909 910
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );

    if( !p_playlist )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
911
        if( i_object ) vlc_object_release( p_vlc );
912
        return VLC_ENOOBJ;
913 914 915 916 917 918 919 920 921 922 923 924 925 926 927
    }

    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
928
    if( i_object ) vlc_object_release( p_vlc );
929 930 931 932
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
933
 * VLC_Stop: stop
934
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
935
int VLC_Stop( int i_object )
936 937 938 939 940
{
    intf_thread_t *   p_intf;
    playlist_t    *   p_playlist;
    vout_thread_t *   p_vout;
    aout_instance_t * p_aout;
941
    vlc_t *p_vlc = vlc_current_object( i_object );
942 943

    /* Check that the handle is valid */
944
    if( !p_vlc )
945
    {
946
        return VLC_ENOOBJ;
947 948 949 950 951 952
    }

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

954 955 956 957 958 959 960 961 962 963 964
    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
     */
965
    
966 967 968 969 970 971 972 973
    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 );
    }
974
    
975 976 977 978 979 980 981 982
    /*
     * 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 );
983
        vout_Destroy( p_vout );
984 985 986 987 988 989 990 991 992 993 994 995 996
    }

    /*
     * 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
997
    if( i_object ) vlc_object_release( p_vlc );
998 999 1000 1001
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
1002
 * VLC_Pause: toggle pause
1003
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1004
int VLC_Pause( int i_object )
1005 1006
{
    input_thread_t *p_input;
1007
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
1008 1009 1010

    if( !p_vlc )
    {
1011
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
1012
    }
1013 1014 1015 1016 1017

    p_input = vlc_object_find( p_vlc, VLC_OBJECT_INPUT, FIND_CHILD );

    if( !p_input )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1018
        if( i_object ) vlc_object_release( p_vlc );
1019
        return VLC_ENOOBJ;
1020 1021 1022 1023 1024
    }

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

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_FullScreen: toggle fullscreen mode
1031
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
1032
int VLC_FullScreen( int i_object )
1033 1034
{
    vout_thread_t *p_vout;
1035
    vlc_t *p_vlc = vlc_current_object( i_object );
Sam Hocevar's avatar
Sam Hocevar committed
1036 1037 1038

    if( !p_vlc )
    {
1039
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
1040
    }
1041 1042 1043 1044 1045

    p_vout = vlc_object_find( p_vlc, VLC_OBJECT_VOUT, FIND_CHILD );

    if( !p_vout )
    {
Gildas Bazin's avatar
 
Gildas Bazin committed
1046
        if( i_object ) vlc_object_release( p_vlc );
1047
        return VLC_ENOOBJ;
1048 1049 1050 1051 1052
    }

    p_vout->i_changes |= VOUT_FULLSCREEN_CHANGE;
    vlc_object_release( p_vout );

Gildas Bazin's avatar
 
Gildas Bazin committed
1053
    if( i_object ) vlc_object_release( p_vlc );
1054 1055 1056
    return VLC_SUCCESS;
}

1057 1058
/* following functions are local */

Sam Hocevar's avatar
Sam Hocevar committed
1059 1060 1061 1062 1063 1064 1065 1066 1067
/*****************************************************************************
 * SetLanguage: set the interface language.
 *****************************************************************************
 * We set the LC_MESSAGES locale category for interface messages and buttons,
 * as well as the LC_CTYPE category for string sorting and possible wide
 * character support.
 *****************************************************************************/
static void SetLanguage ( char const *psz_lang )
{
1068 1069 1070
#if defined( ENABLE_NLS ) \
     && ( defined( HAVE_GETTEXT ) || defined( HAVE_INCLUDED_GETTEXT ) )

1071
    char *          psz_path;
Eric Petit's avatar
Eric Petit committed
1072
#if defined( SYS_DARWIN ) || defined ( WIN32 ) || defined( SYS_BEOS )
1073 1074 1075
    char            psz_tmp[1024];
#endif

Gildas Bazin's avatar
 
Gildas Bazin committed
1076 1077
    if( psz_lang && !*psz_lang )
    {
1078
#   if defined( HAVE_LC_MESSAGES )
Gildas Bazin's avatar
 
Gildas Bazin committed
1079
        setlocale( LC_MESSAGES, psz_lang );
Sam Hocevar's avatar
Sam Hocevar committed
1080
#   endif
Gildas Bazin's avatar
 
Gildas Bazin committed
1081 1082
        setlocale( LC_CTYPE, psz_lang );
    }
Gildas Bazin's avatar
 
Gildas Bazin committed
1083
    else if( psz_lang )
Gildas Bazin's avatar
 
Gildas Bazin committed
1084
    {
1085 1086 1087 1088
#ifdef SYS_DARWIN
        /* I need that under Darwin, please check it doesn't disturb
         * other platforms. --Meuuh */
        setenv( "LANG", psz_lang, 1 );
Gildas Bazin's avatar
 
Gildas Bazin committed
1089 1090 1091 1092 1093 1094 1095

#elif defined( SYS_BEOS ) || defined( WIN32 )
        /* We set LC_ALL manually because it is the only way to set
         * the language at runtime under eg. Windows. Beware that this
         * makes the environment unconsistent when libvlc is unloaded and
         * should probably be moved to a safer place like vlc.c. */
        static char psz_lcall[20];
1096 1097 1098
        snprintf( psz_lcall, 19, "LC_ALL=%s", psz_lang );