]> git.sesse.net Git - vlc/blobdiff - src/libvlc-common.c
Attempt to fix C++ compilation
[vlc] / src / libvlc-common.c
index 4529b4b13dc9a1053712869088eedb7fd1e3a67a..9116c48fbd7e512a208b95e08f95c8056284c0a6 100644 (file)
 #   include <locale.h>
 #endif
 
+#ifdef HAVE_DBUS_3
+#   include <dbus/dbus.h>
+
+/* this is also defined in modules/control/dbus.h */
+/* names registered on the session bus */
+#define VLC_DBUS_SERVICE        "org.videolan.vlc"
+#define VLC_DBUS_INTERFACE      "org.videolan.vlc"
+#define VLC_DBUS_OBJECT_PATH    "/org/videolan/vlc"
+#endif
+
 #ifdef HAVE_HAL
 #   include <hal/libhal.h>
 #endif
@@ -88,6 +98,8 @@
 
 #include "libvlc.h"
 
+#include "playlist/playlist_internal.h"
+
 /*****************************************************************************
  * The evil global variable. We handle it with care, don't worry.
  *****************************************************************************/
@@ -111,8 +123,6 @@ static void Version       ( void );
 #ifdef WIN32
 static void ShowConsole   ( vlc_bool_t );
 static void PauseConsole  ( void );
-extern void __wgetmainargs(int *argc, wchar_t ***wargv, wchar_t ***wenviron,
-                           int expand_wildcards, int *startupinfo);
 #endif
 static int  ConsoleWidth  ( void );
 
@@ -215,6 +225,40 @@ libvlc_int_t * libvlc_InternalCreate( void )
     return p_libvlc;
 }
 
+/*
+ * D-Bus callback needed in libvlc_InternalInit()
+ */
+#ifdef HAVE_DBUS_3
+/* Handling of messages received on / object */
+static DBusHandlerResult handle_root
+    ( DBusConnection *p_conn, DBusMessage *p_from, void *p_data ) 
+{
+    DBusMessage* p_msg = dbus_message_new_method_return( p_from );
+    if( !p_msg ) return DBUS_HANDLER_RESULT_NEED_MEMORY;
+
+    DBusMessageIter args;
+    dbus_message_iter_init_append( p_msg, &args );
+
+    char *p_root = malloc( strlen( "<node name='/'></node>" ) );
+    if (!p_root ) return DBUS_HANDLER_RESULT_NEED_MEMORY;
+    sprintf( p_root, "<node name='/'></node>" );
+
+    if( !dbus_message_iter_append_basic( &args, DBUS_TYPE_STRING, &p_root ) )
+            return DBUS_HANDLER_RESULT_NEED_MEMORY;
+
+    if( !dbus_connection_send( p_conn, p_msg, NULL ) )
+        return DBUS_HANDLER_RESULT_NEED_MEMORY;
+    dbus_connection_flush( p_conn );
+    dbus_message_unref( p_msg );
+    return DBUS_HANDLER_RESULT_HANDLED;
+}
+/* vtable passed to dbus_connection_register_object_path() */
+static DBusObjectPathVTable vlc_dbus_root_vtable = {
+    NULL, handle_root, NULL, NULL, NULL, NULL
+};
+
+#endif
+
 /**
  * Initialize a libvlc instance
  * This function initializes a previously allocated libvlc instance:
@@ -289,7 +333,8 @@ int libvlc_InternalInit( libvlc_int_t *p_libvlc, int i_argc, char *ppsz_argv[] )
     }
     p_help_module->psz_object_name = "help";
     p_help_module->psz_longname = N_("Help options");
-    config_Duplicate( p_help_module, p_help_config );
+    config_Duplicate( p_help_module, p_help_config,
+                      sizeof (p_help_config) / sizeof (p_help_config[0]) );
     vlc_object_attach( p_help_module, libvlc_global.p_module_bank );
     /* End hack */
 
@@ -569,6 +614,166 @@ int libvlc_InternalInit( libvlc_int_t *p_libvlc, int i_argc, char *ppsz_argv[] )
      */
     system_Configure( p_libvlc, &i_argc, ppsz_argv );
 
+/* FIXME: could be replaced by using Unix sockets */
+#ifdef HAVE_DBUS_3
+    /* Initialise D-Bus interface, check for other instances */
+    DBusConnection  *p_conn;
+    DBusError       dbus_error;
+    int             i_dbus_service;
+
+    dbus_threads_init_default();
+    dbus_error_init( &dbus_error );
+
+    /* connect to the session bus */
+    p_conn = dbus_bus_get( DBUS_BUS_SESSION, &dbus_error );
+    if( !p_conn )
+    {
+        msg_Err( p_libvlc, "Failed to connect to the D-Bus session daemon: %s",
+                dbus_error.message );
+        dbus_error_free( &dbus_error );
+    }
+    else
+    {
+        /* we request the service org.videolan.vlc */
+        i_dbus_service = dbus_bus_request_name( p_conn, VLC_DBUS_SERVICE, 0, 
+                &dbus_error );
+        if( dbus_error_is_set( &dbus_error ) )
+        { 
+            msg_Err( p_libvlc, "Error requesting %s service: %s\n",
+                    VLC_DBUS_SERVICE, dbus_error.message );
+            dbus_error_free( &dbus_error );
+        }
+        else
+        {
+            if( i_dbus_service != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER )
+            { /* the name is already registered by another instance of vlc */
+                if( config_GetInt( p_libvlc, "one-instance" ) )
+                {
+                    /* check if /org/videolan/vlc exists
+                     * if not: D-Bus control is not enabled on the other
+                     * instance and we can't pass MRLs to it */
+                    DBusMessage *p_test_msg, *p_test_reply;
+                    p_test_msg =  dbus_message_new_method_call(
+                            VLC_DBUS_SERVICE, VLC_DBUS_OBJECT_PATH,
+                            VLC_DBUS_INTERFACE, "Nothing" );
+                    /* block unti a reply arrives */
+                    p_test_reply = dbus_connection_send_with_reply_and_block(
+                            p_conn, p_test_msg, -1, &dbus_error );
+                    dbus_message_unref( p_test_msg );
+                    if( p_test_reply == NULL )
+                    {
+                        dbus_error_free( &dbus_error );
+                        msg_Err( p_libvlc, "one instance mode has been "
+                                "set but D-Bus control interface is not "
+                                "enabled. Enable it and restart vlc, or "
+                                "disable one instance mode." );
+                    }
+                    else
+                    {
+                        dbus_message_unref( p_test_reply );
+                        msg_Warn( p_libvlc,
+                                "Another vlc instance exists: will now exit");
+
+                        int i_input;
+                        DBusMessage* p_dbus_msg;
+                        DBusMessageIter dbus_args;
+                        DBusPendingCall* p_dbus_pending;
+                        dbus_bool_t b_play;
+
+                        for( i_input = optind;i_input < i_argc;i_input++ )
+                        {
+                            msg_Dbg( p_libvlc, "Give %s to other vlc\n",
+                                    ppsz_argv[i_input] );
+
+                            p_dbus_msg = dbus_message_new_method_call(
+                                    VLC_DBUS_SERVICE, VLC_DBUS_OBJECT_PATH,
+                                    VLC_DBUS_INTERFACE, "AddMRL" );
+
+                            if ( NULL == p_dbus_msg )
+                            {
+                                msg_Err( p_libvlc, "D-Bus problem" );
+                                system_End( p_libvlc );
+                                exit( 0 );
+                            }
+
+                            /* append MRLs */
+                            dbus_message_iter_init_append( p_dbus_msg,
+                                    &dbus_args );
+                            if ( !dbus_message_iter_append_basic( &dbus_args, 
+                                        DBUS_TYPE_STRING,
+                                        &ppsz_argv[i_input] ) )
+                            {
+                                msg_Err( p_libvlc, "Out of memory" );
+                                dbus_message_unref( p_dbus_msg );
+                                system_End( p_libvlc );
+                                exit( 0 );
+                            }
+                            b_play = TRUE;
+                            if( config_GetInt( p_libvlc, "playlist-enqueue" ) )
+                                b_play = FALSE;
+                            if ( !dbus_message_iter_append_basic( &dbus_args,
+                                        DBUS_TYPE_BOOLEAN, &b_play ) )
+                            {
+                                msg_Err( p_libvlc, "Out of memory" );
+                                dbus_message_unref( p_dbus_msg );
+                                system_End( p_libvlc );
+                                exit( 0 );
+                            }
+
+                            /* send message and get a handle for a reply */
+                            if ( !dbus_connection_send_with_reply ( p_conn,
+                                        p_dbus_msg, &p_dbus_pending, -1 ) )
+                            {
+                                msg_Err( p_libvlc, "D-Bus problem" );
+                                dbus_message_unref( p_dbus_msg );
+                                system_End( p_libvlc );
+                                exit( 0 );
+                            }
+
+                            if ( NULL == p_dbus_pending )
+                            {
+                                msg_Err( p_libvlc, "D-Bus problem" );
+                                dbus_message_unref( p_dbus_msg );
+                                system_End( p_libvlc );
+                                exit( 0 );
+                            }
+                            dbus_connection_flush( p_conn );
+                            dbus_message_unref( p_dbus_msg );
+                            /* block until we receive a reply */
+                            dbus_pending_call_block( p_dbus_pending );
+                            dbus_pending_call_unref( p_dbus_pending );
+                        } /* processes all command line MRLs */
+
+                        /* bye bye */
+                        system_End( p_libvlc );
+                        exit( 0 );
+                    }
+                } /* we're not in one-instance mode */
+                else
+                {
+                    msg_Dbg( p_libvlc, 
+                            "%s is already registered on the session bus\n",
+                            VLC_DBUS_SERVICE );
+                }
+            } /* the named is owned by something else */
+            else
+            {
+                /* register "/" object */
+                if( !dbus_connection_register_object_path( p_conn, "/", 
+                        &vlc_dbus_root_vtable, NULL ) )
+                {
+                    msg_Err( p_libvlc, "Out of memory" );
+                }
+                msg_Dbg( p_libvlc, 
+                        "We are the primary owner of %s on the session bus",
+                        VLC_DBUS_SERVICE );
+            }
+        } /* no error when requesting the name on the bus */
+        /* we unreference the connection when we've finished with it */
+        dbus_connection_unref( p_conn );
+    } /* ( p_conn != NULL ) */
+#endif
+
     /*
      * Message queue options
      */
@@ -678,7 +883,7 @@ int libvlc_InternalInit( libvlc_int_t *p_libvlc, int i_argc, char *ppsz_argv[] )
     if( psz_modules && *psz_modules )
     {
         /* Add service discovery modules */
-        playlist_AddSDModules( p_playlist, psz_modules );
+        playlist_ServicesDiscoveryAdd( p_playlist, psz_modules );
     }
     if( psz_modules ) free( psz_modules );
 
@@ -747,7 +952,7 @@ int libvlc_InternalInit( libvlc_int_t *p_libvlc, int i_argc, char *ppsz_argv[] )
 #ifdef HAVE_SYSLOG_H
     if( config_GetInt( p_libvlc, "syslog" ) == 1 )
     {
-        char *psz_logmode = "logmode=syslog";
+        const char *psz_logmode = "logmode=syslog";
         libvlc_InternalAddIntf( 0, "logger,none", VLC_FALSE, VLC_FALSE,
                                 1, &psz_logmode );
     }
@@ -922,7 +1127,7 @@ int libvlc_InternalDestroy( libvlc_int_t *p_libvlc, vlc_bool_t b_release )
 int libvlc_InternalAddIntf( libvlc_int_t *p_libvlc,
                             char const *psz_module,
                             vlc_bool_t b_block, vlc_bool_t b_play,
-                            int i_options, char **ppsz_options )
+                            int i_options, const char *const *ppsz_options )
 {
     int i_err;
     intf_thread_t *p_intf;
@@ -979,7 +1184,7 @@ static void SetLanguage ( char const *psz_lang )
 #if defined( ENABLE_NLS ) \
      && ( defined( HAVE_GETTEXT ) || defined( HAVE_INCLUDED_GETTEXT ) )
 
-    char *          psz_path;
+    const char *          psz_path;
 #if defined( __APPLE__ ) || defined ( WIN32 ) || defined( SYS_BEOS )
     char            psz_tmp[1024];
 #endif
@@ -1041,17 +1246,6 @@ static int GetFilenames( libvlc_int_t *p_vlc, int i_argc, char *ppsz_argv[] )
 {
     int i_opt, i_options;
 
-#ifdef WIN32
-    wchar_t **wargv, **wenvp;
-    int si = { 0 };
-
-    if( GetVersion() < 0x80000000 )
-    {
-        /* fetch unicode argv[] for Windows NT and above */
-        __wgetmainargs(&i_opt, &wargv, &wenvp, 0, &si);
-    }
-#endif
-
     /* We assume that the remaining parameters are filenames
      * and their input options */
     for( i_opt = i_argc - 1; i_opt >= optind; i_opt-- )
@@ -1073,12 +1267,10 @@ static int GetFilenames( libvlc_int_t *p_vlc, int i_argc, char *ppsz_argv[] )
 #ifdef WIN32
         if( GetVersion() < 0x80000000 )
         {
-            psz_target = FromWide( wargv[ i_opt ] );
-            VLC_AddTarget( p_vlc->i_object_id, psz_target,
+            VLC_AddTarget( p_vlc->i_object_id, ppsz_argv[i_opt],
                        (char const **)( i_options ? &ppsz_argv[i_opt + 1] :
                                         NULL ), i_options,
                        PLAYLIST_INSERT, 0 );
-            free( psz_target );
         }
         else
 #endif
@@ -1155,7 +1347,6 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
 #endif
     vlc_list_t *p_list;
     module_t *p_parser;
-    module_config_t *p_item;
     char psz_spaces_text[PADDING_SPACES+LINE_START+1];
     char psz_spaces_longtext[LINE_START+3];
     char psz_format[sizeof(FORMAT_STRING)];
@@ -1180,8 +1371,9 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
     for( i_index = 0; i_index < p_list->i_count; i_index++ )
     {
         vlc_bool_t b_help_module;
-
-        p_parser = (module_t *)p_list->p_values[i_index].p_object ;
+        module_t *p_parser = (module_t *)p_list->p_values[i_index].p_object;
+        module_config_t *p_item,
+                        *p_end = p_parser->p_config + p_parser->confsize;
 
         if( psz_module_name && strcmp( psz_module_name,
                                        p_parser->psz_object_name ) )
@@ -1199,13 +1391,12 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
         if( !b_advanced )
         {
             for( p_item = p_parser->p_config;
-                 p_item->i_type != CONFIG_HINT_END;
+                 p_item < p_end;
                  p_item++ )
             {
                 if( (p_item->i_type & CONFIG_ITEM) &&
                     !p_item->b_advanced ) break;
             }
-            if( p_item->i_type == CONFIG_HINT_END ) continue;
         }
 
         /* Print name of module */
@@ -1216,12 +1407,12 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
 
         /* Print module options */
         for( p_item = p_parser->p_config;
-             p_item->i_type != CONFIG_HINT_END;
+             p_item < p_end;
              p_item++ )
         {
             char *psz_text, *psz_spaces = psz_spaces_text;
-            char *psz_bra = NULL, *psz_type = NULL, *psz_ket = NULL;
-            char *psz_suf = "", *psz_prefix = NULL;
+            const char *psz_bra = NULL, *psz_type = NULL, *psz_ket = NULL;
+            const char *psz_suf = "", *psz_prefix = NULL;
             signed int i;
 
             /* Skip deprecated options */
@@ -1258,11 +1449,11 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
                 {
                     psz_bra = OPTION_VALUE_SEP "{";
                     psz_type = psz_buffer;
-                    psz_type[0] = '\0';
+                    psz_buffer[0] = '\0';
                     for( i = 0; p_item->ppsz_list[i]; i++ )
                     {
-                        if( i ) strcat( psz_type, "," );
-                        strcat( psz_type, p_item->ppsz_list[i] );
+                        if( i ) strcat( psz_buffer, "," );
+                        strcat( psz_buffer, p_item->ppsz_list[i] );
                     }
                     psz_ket = "}";
                 }
@@ -1277,11 +1468,11 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
                 {
                     psz_bra = OPTION_VALUE_SEP "{";
                     psz_type = psz_buffer;
-                    psz_type[0] = '\0';
+                    psz_buffer[0] = '\0';
                     for( i = 0; p_item->ppsz_list_text[i]; i++ )
                     {
-                        if( i ) strcat( psz_type, ", " );
-                        sprintf( psz_type + strlen(psz_type), "%i (%s)",
+                        if( i ) strcat( psz_buffer, ", " );
+                        sprintf( psz_buffer + strlen(psz_buffer), "%i (%s)",
                                  p_item->pi_list[i],
                                  p_item->ppsz_list_text[i] );
                     }
@@ -1297,7 +1488,7 @@ static void Usage( libvlc_int_t *p_this, char const *psz_module_name )
                 psz_bra = ""; psz_type = ""; psz_ket = "";
                 if( !b_help_module )
                 {
-                    psz_suf = p_item->i_value ? _(" (default enabled)") :
+                    psz_suf = p_item->value.i ? _(" (default enabled)") :
                                                 _(" (default disabled)");
                 }
                 break;
@@ -1682,6 +1873,7 @@ static void InitDeviceValues( libvlc_int_t *p_vlc )
 
 #ifdef HAVE_HAL_1
         libhal_ctx_shutdown( ctx, NULL );
+        dbus_connection_unref( p_connection );
 #else
         hal_shutdown( ctx );
 #endif