libvlc.c 40.8 KB
Newer Older
1
2
3
/*****************************************************************************
 * libvlc.c: main libvlc source
 *****************************************************************************
4
 * Copyright (C) 1998-2002 VideoLAN
5
 * $Id: libvlc.c,v 1.58 2003/01/22 22:19:29 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>
51
#elif defined( _MSC_VER ) && 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 "error.h"
67
68
69
70
71
#include "netutils.h"                                 /* network_ChannelJoin */

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

72
#include "vlc_playlist.h"
73
74
75
76
77
78
79
80
81
82
#include "interface.h"

#include "audio_output.h"

#include "video.h"
#include "video_output.h"

#include "libvlc.h"

/*****************************************************************************
83
 * The evil global variable. We handle it with care, don't worry.
84
 *****************************************************************************/
85
static libvlc_t libvlc;
Sam Hocevar's avatar
Sam Hocevar committed
86
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
100
101
static void Version       ( void );

#ifdef WIN32
static void ShowConsole   ( void );
#endif

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
102
 * VLC_Version: return the libvlc version.
103
 *****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
104
 * This function returns full version string (numeric version and codename).
105
 *****************************************************************************/
106
char const * VLC_Version( void )
107
{
Sam Hocevar's avatar
Sam Hocevar committed
108
    return VERSION_MESSAGE;
109
110
}

111
112
113
114
115
116
117
118
119
120
/*****************************************************************************
 * 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
121
122
123
124
125
126
127
/*****************************************************************************
 * 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 )
128
{
129
    int i_ret;
130
    vlc_t * p_vlc = NULL;
131
    vlc_value_t lockval;
132

133
134
135
    /* vlc_threads_init *must* be the first internal call! No other call is
     * allowed before the thread system has been initialized. */
    i_ret = vlc_threads_init( &libvlc );
Sam Hocevar's avatar
Sam Hocevar committed
136
    if( i_ret < 0 )
137
    {
Sam Hocevar's avatar
Sam Hocevar committed
138
        return i_ret;
139
140
141
    }

    /* Now that the thread system is initialized, we don't have much, but
142
143
144
145
     * at least we have var_Create */
    var_Create( &libvlc, "libvlc", VLC_VAR_MUTEX );
    var_Get( &libvlc, "libvlc", &lockval );
    vlc_mutex_lock( lockval.p_address );
146
147
    if( !libvlc.b_ready )
    {
148
149
        char *psz_env;

150
151
152
        /* Guess what CPU we have */
        libvlc.i_cpu = CPUCapabilities();

153
154
        /* Find verbosity from VLC_VERBOSE environment variable */
        psz_env = getenv( "VLC_VERBOSE" );
155
        libvlc.i_verbose = psz_env ? atoi( psz_env ) : -1;
156
157
158
159
160
161
162

#ifdef HAVE_ISATTY
        libvlc.b_color = isatty( 2 ); /* 2 is for stderr */
#else
        libvlc.b_color = VLC_FALSE;
#endif

163
164
165
166
167
168
169
        /* Initialize message queue */
        msg_Create( &libvlc );

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

170
        /* Initialize the module bank and load the configuration of the
171
172
173
174
175
         * 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) */
        module_InitBank( &libvlc );
        module_LoadMain( &libvlc );
176
177

        libvlc.b_ready = VLC_TRUE;
178
    }
179
180
    vlc_mutex_unlock( lockval.p_address );
    var_Destroy( &libvlc, "libvlc" );
181
182
183

    /* Allocate a vlc object */
    p_vlc = vlc_object_create( &libvlc, VLC_OBJECT_VLC );
184
185
    if( p_vlc == NULL )
    {
Sam Hocevar's avatar
Sam Hocevar committed
186
        return VLC_EGENERIC;
187
188
189
190
191
    }

    p_vlc->psz_object_name = "root";

    /* Initialize mutexes */
192
    vlc_mutex_init( p_vlc, &p_vlc->config_lock );
193
194

    /* Store our newly allocated structure in the global list */
195
    vlc_object_attach( p_vlc, &libvlc );
196

Sam Hocevar's avatar
Sam Hocevar committed
197
198
199
200
    /* Store data for the non-reentrant API */
    p_static_vlc = p_vlc;

    return p_vlc->i_object_id;
201
202
203
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
204
 * VLC_Init: initialize a vlc_t structure.
205
206
207
208
209
210
211
 *****************************************************************************
 * 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
212
int VLC_Init( int i_object, int i_argc, char *ppsz_argv[] )
213
{
214
215
    char         p_capabilities[200];
    char *       p_tmp;
216
217
    char *       psz_modules;
    char *       psz_parser;
218
    vlc_bool_t   b_exit;
Sam Hocevar's avatar
Sam Hocevar committed
219
    vlc_t *      p_vlc;
220
221
    module_t    *p_help_module;
    playlist_t  *p_playlist;
222

Sam Hocevar's avatar
Sam Hocevar committed
223
224
    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;

225
    if( !p_vlc )
226
    {
227
        return VLC_ENOOBJ;
228
229
230
    }

    /*
231
     * System specific initialization code
232
     */
233
    system_Init( p_vlc, &i_argc, ppsz_argv );
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249

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

250
251
252
253
    /*
     * Support for gettext
     */
    SetLanguage( "" );
Sam Hocevar's avatar
Sam Hocevar committed
254
255

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

258
259
260
261
    /* Hack: insert the help module here */
    p_help_module = vlc_object_create( p_vlc, VLC_OBJECT_MODULE );
    if( p_help_module == NULL )
    {
262
        //module_EndBank( p_vlc );
gbazin's avatar
   
gbazin committed
263
        if( i_object ) vlc_object_release( p_vlc );
264
265
266
267
        return VLC_EGENERIC;
    }
    p_help_module->psz_object_name = "help";
    config_Duplicate( p_help_module, p_help_config );
268
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
269
270
    /* End hack */

Sam Hocevar's avatar
Sam Hocevar committed
271
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_TRUE ) )
272
    {
273
        vlc_object_detach( p_help_module );
274
275
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
276
        //module_EndBank( p_vlc );
gbazin's avatar
   
gbazin committed
277
        if( i_object ) vlc_object_release( p_vlc );
278
279
280
        return VLC_EGENERIC;
    }

281
282
    b_exit = VLC_FALSE;

283
284
285
    /* Check for short help option */
    if( config_GetInt( p_vlc, "help" ) )
    {
286
        fprintf( stdout, _("Usage: %s [options] [items]...\n\n"),
287
                         p_vlc->psz_object_name );
288
        Usage( p_vlc, "main" );
289
290
        Usage( p_vlc, "help" );
        b_exit = VLC_TRUE;
291
292
    }
    /* Check for version option */
293
    else if( config_GetInt( p_vlc, "version" ) )
294
295
    {
        Version();
296
297
298
299
300
301
302
303
304
        b_exit = VLC_TRUE;
    }

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

    if( b_exit )
    {
305
306
        config_Free( p_help_module );
        vlc_object_destroy( p_help_module );
307
        //module_EndBank( p_vlc );
gbazin's avatar
   
gbazin committed
308
        if( i_object ) vlc_object_release( p_vlc );
309
310
311
312
313
314
315
316
317
        return VLC_EEXIT;
    }

    /*
     * 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.
     */
318
319
    module_LoadBuiltins( &libvlc );
    module_LoadPlugins( &libvlc );
320
    msg_Dbg( p_vlc, "module bank initialized, found %i modules",
321
                    libvlc.p_module_bank->i_children );
322
323

    /* Hack: insert the help module here */
324
    vlc_object_attach( p_help_module, libvlc.p_module_bank );
325
326
327
328
329
    /* End hack */

    /* Check for help on modules */
    if( (p_tmp = config_GetPsz( p_vlc, "module" )) )
    {
330
        Usage( p_vlc, p_tmp );
331
        free( p_tmp );
332
        b_exit = VLC_TRUE;
333
334
    }
    /* Check for long help option */
335
    else if( config_GetInt( p_vlc, "longhelp" ) )
336
    {
337
        Usage( p_vlc, NULL );
338
        b_exit = VLC_TRUE;
339
340
    }
    /* Check for module list option */
341
    else if( config_GetInt( p_vlc, "list" ) )
342
    {
343
        ListModules( p_vlc );
344
        b_exit = VLC_TRUE;
345
346
347
    }

    /* Hack: remove the help module here */
348
    vlc_object_detach( p_help_module );
349
350
351
352
    config_Free( p_help_module );
    vlc_object_destroy( p_help_module );
    /* End hack */

353
354
    if( b_exit )
    {
355
        //module_EndBank( p_vlc );
gbazin's avatar
   
gbazin committed
356
        if( i_object ) vlc_object_release( p_vlc );
357
358
359
        return VLC_EEXIT;
    }

360
361
362
363
    /*
     * Override default configuration with config file settings
     */
    p_vlc->psz_homedir = config_GetHomeDir();
364
    config_LoadConfigFile( p_vlc, NULL );
365
366
367
368

    /*
     * Override configuration with command line settings
     */
Sam Hocevar's avatar
Sam Hocevar committed
369
    if( config_LoadCmdLine( p_vlc, &i_argc, ppsz_argv, VLC_FALSE ) )
370
371
372
373
374
375
376
377
    {
#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
378
        //module_EndBank( p_vlc );
gbazin's avatar
   
gbazin committed
379
        if( i_object ) vlc_object_release( p_vlc );
380
381
382
383
384
385
        return VLC_EGENERIC;
    }

    /*
     * System specific configuration
     */
386
    system_Configure( p_vlc );
387

388
389
390
391
392
    /*
     * Message queue options
     */
    if( config_GetInt( p_vlc, "quiet" ) )
    {
393
        libvlc.i_verbose = -1;
394
395
396
397
    }
    else
    {
        int i_tmp = config_GetInt( p_vlc, "verbose" );
398
        if( i_tmp >= 0 )
399
        {
400
            libvlc.i_verbose = __MIN( i_tmp, 2 );
401
402
403
404
        }
    }
    libvlc.b_color = libvlc.b_color || config_GetInt( p_vlc, "color" );

405
406
407
408
409
410
411
    /*
     * Output messages that may still be in the queue
     */
    msg_Flush( p_vlc );

    /* p_vlc inititalization. FIXME ? */
    p_vlc->i_desync = config_GetInt( p_vlc, "desync" ) * (mtime_t)1000;
412

413
#if defined( __i386__ )
414
    if( !config_GetInt( p_vlc, "mmx" ) )
415
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMX;
416
    if( !config_GetInt( p_vlc, "3dn" ) )
417
        libvlc.i_cpu &= ~CPU_CAPABILITY_3DNOW;
418
    if( !config_GetInt( p_vlc, "mmxext" ) )
419
        libvlc.i_cpu &= ~CPU_CAPABILITY_MMXEXT;
420
    if( !config_GetInt( p_vlc, "sse" ) )
421
        libvlc.i_cpu &= ~CPU_CAPABILITY_SSE;
422
423
#endif
#if defined( __powerpc__ ) || defined( SYS_DARWIN )
424
    if( !config_GetInt( p_vlc, "altivec" ) )
425
        libvlc.i_cpu &= ~CPU_CAPABILITY_ALTIVEC;
426
#endif
427
428

#define PRINT_CAPABILITY( capability, string )                              \
429
    if( libvlc.i_cpu & capability )                                         \
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
    {                                                                       \
        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
     */
451
    p_vlc->p_memcpy_module = module_Need( p_vlc, "memcpy", "$memcpy" );
452

453
    if( p_vlc->pf_memcpy == NULL )
454
455
456
    {
        p_vlc->pf_memcpy = memcpy;
    }
457
458
459
460
461

    if( p_vlc->pf_memset == NULL )
    {
        p_vlc->pf_memset = memset;
    }
462
463
464
465
466

    /*
     * Initialize shared resources and libraries
     */
    if( config_GetInt( p_vlc, "network-channel" )
467
         && network_ChannelCreate( p_vlc ) )
468
469
    {
        /* On error during Channels initialization, switch off channels */
470
471
        msg_Warn( p_vlc,
                  "channels initialization failed, deactivating channels" );
Sam Hocevar's avatar
Sam Hocevar committed
472
        config_PutInt( p_vlc, "network-channel", VLC_FALSE );
473
474
475
476
477
    }

    /*
     * Initialize playlist and get commandline files
     */
478
    p_playlist = playlist_Create( p_vlc );
479
480
481
    if( !p_playlist )
    {
        msg_Err( p_vlc, "playlist initialization failed" );
482
483
        if( p_vlc->p_memcpy_module != NULL )
        {
484
            module_Unneed( p_vlc, p_vlc->p_memcpy_module );
485
        }
486
        //module_EndBank( p_vlc );
gbazin's avatar
   
gbazin committed
487
        if( i_object ) vlc_object_release( p_vlc );
488
489
490
        return VLC_EGENERIC;
    }

491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
    /*
     * Load background interfaces
     */
    psz_modules = config_GetPsz( p_vlc, "extraintf" );
    psz_parser = psz_modules;
    while ( psz_parser && *psz_parser )
    {
        char *psz_module;
        psz_module = psz_parser;
        psz_parser = strchr( psz_module, ',' );
        if ( psz_parser )
        {
            *psz_parser = '\0';
            psz_parser++;
        }
        VLC_AddIntf( 0, psz_module, VLC_FALSE );
    }
    if ( psz_modules )
    {
        free( psz_modules );
    }
        
513
514
515
516
517
    /*
     * Get input filenames given as commandline arguments
     */
    GetFilenames( p_vlc, i_argc, ppsz_argv );

gbazin's avatar
   
gbazin committed
518
    if( i_object ) vlc_object_release( p_vlc );
519
520
521
522
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
523
 * VLC_AddIntf: add an interface
524
525
 *****************************************************************************
 * This function opens an interface plugin and runs it. If b_block is set
Sam Hocevar's avatar
Sam Hocevar committed
526
527
 * 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
528
529
 * user requests to quit.
 *****************************************************************************/
530
int VLC_AddIntf( int i_object, char const *psz_module, vlc_bool_t b_block )
531
{
Sam Hocevar's avatar
Sam Hocevar committed
532
    int i_err;
533
    intf_thread_t *p_intf;
Sam Hocevar's avatar
Sam Hocevar committed
534
    vlc_t *p_vlc;
535
536
    char *psz_oldmodule = NULL;

Sam Hocevar's avatar
Sam Hocevar committed
537
538
    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;

539
    if( !p_vlc )
540
    {
541
        return VLC_ENOOBJ;
542
543
544
545
546
547
548
549
550
    }

    if( psz_module )
    {
        psz_oldmodule = config_GetPsz( p_vlc, "intf" );
        config_PutPsz( p_vlc, "intf", psz_module );
    }

    /* Try to create the interface */
551
    p_intf = intf_Create( p_vlc );
552
553
554
555
556
557
558
559
560
561
562
563
564

    if( psz_module )
    {
        config_PutPsz( p_vlc, "intf", psz_oldmodule );
        if( psz_oldmodule )
        {
            free( psz_oldmodule );
        }
    }

    if( p_intf == NULL )
    {
        msg_Err( p_vlc, "interface initialization failed" );
gbazin's avatar
   
gbazin committed
565
        if( i_object ) vlc_object_release( p_vlc );
566
567
568
569
570
        return VLC_EGENERIC;
    }

    /* Try to run the interface */
    p_intf->b_block = b_block;
Sam Hocevar's avatar
Sam Hocevar committed
571
572
    i_err = intf_RunThread( p_intf );
    if( i_err )
573
    {
574
        vlc_object_detach( p_intf );
575
        intf_Destroy( p_intf );
gbazin's avatar
   
gbazin committed
576
        if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
577
        return i_err;
578
579
    }

gbazin's avatar
   
gbazin committed
580
    if( i_object ) vlc_object_release( p_vlc );
581
582
583
584
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
585
 * VLC_Destroy: stop playing and destroy everything.
586
 *****************************************************************************
587
 * This function requests the running threads to finish, waits for their
588
589
 * termination, and destroys their structure.
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
590
int VLC_Destroy( int i_object )
591
{
Sam Hocevar's avatar
Sam Hocevar committed
592
593
594
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;
595

596
    if( !p_vlc )
597
    {
598
        return VLC_ENOOBJ;
599
600
    }

601
602
603
604
    /*
     * Go back into channel 0 which is the network
     */
    if( config_GetInt( p_vlc, "network-channel" ) && p_vlc->p_channel )
605
    {
606
607
        network_ChannelJoin( p_vlc, COMMON_CHANNEL );
    }
608
    
609
610
611
612
613
614
615
    /*
     * Free allocated memory
     */
    if( p_vlc->p_memcpy_module )
    {
        module_Unneed( p_vlc, p_vlc->p_memcpy_module );
        p_vlc->p_memcpy_module = NULL;
616
617
    }

618
619
620
621
622
    if( p_vlc->psz_homedir )
    {
        free( p_vlc->psz_homedir );
        p_vlc->psz_homedir = NULL;
    }
623

624
625
626
627
628
629
630
631
632
633
    /*
     * XXX: Free module bank !
     */
    //module_EndBank( p_vlc );
    
    /*
     * System specific cleaning code
     */
    system_End( p_vlc );
    
634
635
636
    /* Destroy mutexes */
    vlc_mutex_destroy( &p_vlc->config_lock );

637
638
    vlc_object_detach( p_vlc );

639
640
    vlc_object_destroy( p_vlc );

641
642
643
    /* Stop thread system: last one out please shut the door! */
    vlc_threads_end( &libvlc );

gbazin's avatar
   
gbazin committed
644
    if( i_object ) vlc_object_release( p_vlc );
645
646
647
    return VLC_SUCCESS;
}

648
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
649
 * VLC_Die: ask vlc to die.
650
651
 *****************************************************************************
 * This function sets p_vlc->b_die to VLC_TRUE, but does not do any other
Sam Hocevar's avatar
Sam Hocevar committed
652
 * task. It is your duty to call vlc_end and VLC_Destroy afterwards.
653
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
654
int VLC_Die( int i_object )
655
{
Sam Hocevar's avatar
Sam Hocevar committed
656
657
658
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;
659
660
661

    if( !p_vlc )
    {
662
        return VLC_ENOOBJ;
663
664
665
666
    }

    p_vlc->b_die = VLC_TRUE;

gbazin's avatar
   
gbazin committed
667
    if( i_object ) vlc_object_release( p_vlc );
668
669
670
671
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
672
 * VLC_AddTarget: adds a target for playing.
673
674
675
676
 *****************************************************************************
 * This function adds psz_target to the current playlist. If a playlist does
 * not exist, it will create one.
 *****************************************************************************/
677
int VLC_AddTarget( int i_object, char const *psz_target, int i_mode, int i_pos )
678
{
Sam Hocevar's avatar
Sam Hocevar committed
679
    int i_err;
Sam Hocevar's avatar
Sam Hocevar committed
680
    playlist_t *p_playlist;
Sam Hocevar's avatar
Sam Hocevar committed
681
682
683
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;
Sam Hocevar's avatar
Sam Hocevar committed
684

685
    if( !p_vlc )
686
    {
687
        return VLC_ENOOBJ;
688
689
    }

Sam Hocevar's avatar
Sam Hocevar committed
690
691
692
693
694
695
696
697
698
    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 )
        {
gbazin's avatar
   
gbazin committed
699
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
700
701
702
703
704
705
            return VLC_EGENERIC;
        }

        vlc_object_yield( p_playlist );
    }

Sam Hocevar's avatar
Sam Hocevar committed
706
    i_err = playlist_Add( p_playlist, psz_target, i_mode, i_pos );
Sam Hocevar's avatar
Sam Hocevar committed
707
708

    vlc_object_release( p_playlist );
709

gbazin's avatar
   
gbazin committed
710
    if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
711
    return i_err;
712
713
}

714
/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
715
 * VLC_Set: set a vlc variable
716
717
718
 *****************************************************************************
 *
 *****************************************************************************/
719
int VLC_Set( int i_object, char const *psz_var, vlc_value_t value )
720
{
Sam Hocevar's avatar
Sam Hocevar committed
721
    vlc_t *p_vlc;
gbazin's avatar
   
gbazin committed
722
    int i_ret;
723

Sam Hocevar's avatar
Sam Hocevar committed
724
    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;
725
726
727

    if( !p_vlc )
    {
728
        return VLC_ENOOBJ;
729
730
    }

Sam Hocevar's avatar
Sam Hocevar committed
731
732
733
    /* 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 ) )
734
    {
Sam Hocevar's avatar
Sam Hocevar committed
735
        module_config_t *p_item;
736
        char const *psz_newvar = psz_var + 6;
737

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

Sam Hocevar's avatar
Sam Hocevar committed
740
        if( p_item )
741
        {
Sam Hocevar's avatar
Sam Hocevar committed
742
            switch( p_item->i_type )
743
            {
Sam Hocevar's avatar
Sam Hocevar committed
744
745
746
747
748
749
750
751
752
753
754
755
                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;
756
            }
gbazin's avatar
   
gbazin committed
757
            if( i_object ) vlc_object_release( p_vlc );
Sam Hocevar's avatar
Sam Hocevar committed
758
            return VLC_SUCCESS;
759
760
761
        }
    }

gbazin's avatar
   
gbazin committed
762
763
764
765
    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
766
767
768
769
770
771
772
}

/*****************************************************************************
 * VLC_Get: get a vlc variable
 *****************************************************************************
 *
 *****************************************************************************/
773
int VLC_Get( int i_object, char const *psz_var, vlc_value_t *p_value )
Sam Hocevar's avatar
Sam Hocevar committed
774
775
{
    vlc_t *p_vlc;
gbazin's avatar
   
gbazin committed
776
    int i_ret;
Sam Hocevar's avatar
Sam Hocevar committed
777
778
779
780

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;

    if( !p_vlc )
781
    {
782
        return VLC_ENOOBJ;
783
784
    }

gbazin's avatar
   
gbazin committed
785
786
787
788
    i_ret = var_Get( p_vlc, psz_var, p_value );

    if( i_object ) vlc_object_release( p_vlc );
    return i_ret;
789
790
}

Sam Hocevar's avatar
Sam Hocevar committed
791
/* FIXME: temporary hacks */
792
793

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
794
 * VLC_Play: play
795
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
796
int VLC_Play( int i_object )
797
798
{
    playlist_t * p_playlist;
Sam Hocevar's avatar
Sam Hocevar committed
799
800
801
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;
802
803

    /* Check that the handle is valid */
804
    if( !p_vlc )
805
    {
806
        return VLC_ENOOBJ;
807
    }
808
    
809
810
    vlc_thread_set_priority( p_vlc, VLC_THREAD_PRIORITY_LOW );

811
812
813
814
    p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );

    if( !p_playlist )
    {
gbazin's avatar
   
gbazin committed
815
        if( i_object ) vlc_object_release( p_vlc );
816
        return VLC_ENOOBJ;
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
    }

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

gbazin's avatar
   
gbazin committed
832
    if( i_object ) vlc_object_release( p_vlc );
833
834
835
836
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
837
 * VLC_Stop: stop
838
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
839
int VLC_Stop( int i_object )
840
841
842
843
844
{
    intf_thread_t *   p_intf;
    playlist_t    *   p_playlist;
    vout_thread_t *   p_vout;
    aout_instance_t * p_aout;
Sam Hocevar's avatar
Sam Hocevar committed
845
846
847
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;
848
849

    /* Check that the handle is valid */
850
    if( !p_vlc )
851
    {
852
        return VLC_ENOOBJ;
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
    }

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

    /*
     * Free playlists
     */
    msg_Dbg( p_vlc, "removing all playlists" );
    while( (p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST,
                                          FIND_CHILD )) )
    {
        vlc_object_detach( p_playlist );
        vlc_object_release( p_playlist );
        playlist_Destroy( p_playlist );
    }

    /*
     * Free video outputs
     */
    msg_Dbg( p_vlc, "removing all video outputs" );
    while( (p_vout = vlc_object_find( p_vlc, VLC_OBJECT_VOUT, FIND_CHILD )) )
    {
        vlc_object_detach( p_vout );
        vlc_object_release( p_vout );
887
        vout_Destroy( p_vout );
888
889
890
891
892
893
894
895
896
897
898
899
900
    }

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

gbazin's avatar
   
gbazin committed
901
    if( i_object ) vlc_object_release( p_vlc );
902
903
904
905
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
906
 * VLC_Pause: toggle pause
907
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
908
int VLC_Pause( int i_object )
909
910
{
    input_thread_t *p_input;
Sam Hocevar's avatar
Sam Hocevar committed
911
912
913
914
915
916
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;

    if( !p_vlc )
    {
917
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
918
    }
919
920
921
922
923

    p_input = vlc_object_find( p_vlc, VLC_OBJECT_INPUT, FIND_CHILD );

    if( !p_input )
    {
gbazin's avatar
   
gbazin committed
924
        if( i_object ) vlc_object_release( p_vlc );
925
        return VLC_ENOOBJ;
926
927
928
929
930
    }

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

gbazin's avatar
   
gbazin committed
931
    if( i_object ) vlc_object_release( p_vlc );
932
933
934
935
    return VLC_SUCCESS;
}

/*****************************************************************************
Sam Hocevar's avatar
Sam Hocevar committed
936
 * VLC_FullScreen: toggle fullscreen mode
937
 *****************************************************************************/
Sam Hocevar's avatar
Sam Hocevar committed
938
int VLC_FullScreen( int i_object )
939
940
{
    vout_thread_t *p_vout;
Sam Hocevar's avatar
Sam Hocevar committed
941
942
943
944
945
946
    vlc_t *p_vlc;

    p_vlc = i_object ? vlc_object_get( &libvlc, i_object ) : p_static_vlc;

    if( !p_vlc )
    {
947
        return VLC_ENOOBJ;
Sam Hocevar's avatar
Sam Hocevar committed
948
    }
949
950
951
952
953

    p_vout = vlc_object_find( p_vlc, VLC_OBJECT_VOUT, FIND_CHILD );

    if( !p_vout )
    {
gbazin's avatar
   
gbazin committed
954
        if( i_object ) vlc_object_release( p_vlc );
955
        return VLC_ENOOBJ;
956
957
958
959
960
    }

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

gbazin's avatar
   
gbazin committed
961
    if( i_object ) vlc_object_release( p_vlc );
962
963
964
    return VLC_SUCCESS;
}

965
966
/* following functions are local */

Sam Hocevar's avatar
Sam Hocevar committed
967
968
969
970
971
972
973
974
975
/*****************************************************************************
 * 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 )
{
976
977
978
#if defined( ENABLE_NLS ) \
     && ( defined( HAVE_GETTEXT ) || defined( HAVE_INCLUDED_GETTEXT ) )

979
980
981
982
983
    char *          psz_path;
#ifdef SYS_DARWIN
    char            psz_tmp[1024];
#endif

984
985
#   if defined( HAVE_INCLUDED_GETTEXT ) && !defined( HAVE_LC_MESSAGES )
    if( *psz_lang )
Sam Hocevar's avatar
Sam Hocevar committed
986
    {
987
988
989
990
991
992
993
994
        /* 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];
        snprintf( psz_lcall, 19, "LC_ALL=%s", psz_lang );
        psz_lcall[19] = '\0';
        putenv( psz_lcall );
Sam Hocevar's avatar
Sam Hocevar committed
995
    }
996
#   endif
Sam Hocevar's avatar
Sam Hocevar committed
997

998
999
#   if defined( HAVE_LC_MESSAGES )
    setlocale( LC_MESSAGES, psz_lang );
Sam Hocevar's avatar
Sam Hocevar committed
1000
#   endif