bonjour.c 10.4 KB
Newer Older
1 2 3
/*****************************************************************************
 * bonjour.c: Bonjour services discovery module
 *****************************************************************************
4
 * Copyright (C) 2005-2008 the VideoLAN team
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
 * $Id$
 *
 * Authors: Jon Lech Johansen <jon@nanocrew.net>
 *
 * 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
Antoine Cellerier's avatar
Antoine Cellerier committed
21
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 23 24 25 26 27
 *****************************************************************************/

/*****************************************************************************
 * Includes
 *****************************************************************************/

28 29 30 31
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif

32
#include <vlc_common.h>
33
#include <vlc_plugin.h>
Clément Stenac's avatar
Clément Stenac committed
34
#include <vlc_playlist.h>
35
#include <vlc_arrays.h>
36 37

#include <avahi-client/client.h>
38 39
#include <avahi-client/publish.h>
#include <avahi-client/lookup.h>
40
#include <avahi-common/thread-watch.h>
41 42 43 44 45 46 47 48 49 50 51 52 53
#include <avahi-common/malloc.h>
#include <avahi-common/error.h>

/*****************************************************************************
 * Module descriptor
 *****************************************************************************/

/* Callbacks */
    static int  Open ( vlc_object_t * );
    static void Close( vlc_object_t * );

vlc_module_begin();
    set_shortname( "Bonjour" );
54
    set_description( N_("Bonjour services") );
55 56 57 58 59 60 61 62 63 64 65 66
    set_category( CAT_PLAYLIST );
    set_subcategory( SUBCAT_PLAYLIST_SD );
    set_capability( "services_discovery", 0 );
    set_callbacks( Open, Close );
vlc_module_end();

/*****************************************************************************
 * Local structures
 *****************************************************************************/

struct services_discovery_sys_t
{
67
    AvahiThreadedPoll   *poll;
68 69
    AvahiClient         *client;
    AvahiServiceBrowser *sb;
70
    vlc_dictionary_t    services_name_to_input_item;
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
};

/*****************************************************************************
 * Local prototypes
 *****************************************************************************/

/*****************************************************************************
 * client_callback
 *****************************************************************************/
static void client_callback( AvahiClient *c, AvahiClientState state,
                             void * userdata )
{
    services_discovery_t *p_sd = ( services_discovery_t* )userdata;
    services_discovery_sys_t *p_sys = p_sd->p_sys;

86 87
    if( state == AVAHI_CLIENT_FAILURE &&
        (avahi_client_errno(c) == AVAHI_ERR_DISCONNECTED) )
88 89
    {
        msg_Err( p_sd, "avahi client disconnected" );
90
        avahi_threaded_poll_quit( p_sys->poll );
91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
    }
}

/*****************************************************************************
 * resolve_callback
 *****************************************************************************/
static void resolve_callback(
    AvahiServiceResolver *r,
    AvahiIfIndex interface,
    AvahiProtocol protocol,
    AvahiResolverEvent event,
    const char *name,
    const char *type,
    const char *domain,
    const char *host_name,
    const AvahiAddress *address,
    uint16_t port,
    AvahiStringList *txt,
109
    AvahiLookupResultFlags flags,
110 111 112 113
    void* userdata )
{
    services_discovery_t *p_sd = ( services_discovery_t* )userdata;
    services_discovery_sys_t *p_sys = p_sd->p_sys;
114 115 116
    
    VLC_UNUSED(interface); VLC_UNUSED(host_name);
    VLC_UNUSED(flags);
117

118
    if( event == AVAHI_RESOLVER_FAILURE )
119 120 121 122 123 124 125 126 127
    {
        msg_Err( p_sd,
                 "failed to resolve service '%s' of type '%s' in domain '%s'",
                 name, type, domain );
    }
    else if( event == AVAHI_RESOLVER_FOUND )
    {
        char a[128];
        char *psz_uri = NULL;
128 129
        char *psz_addr = NULL;
        AvahiStringList *asl = NULL;
130
        input_item_t *p_input = NULL;
131 132 133 134 135

        msg_Dbg( p_sd, "service '%s' of type '%s' in domain '%s'",
                 name, type, domain );

        avahi_address_snprint(a, (sizeof(a)/sizeof(a[0]))-1, address);
136
        if( protocol == AVAHI_PROTO_INET6 )
137 138
            if( asprintf( &psz_addr, "[%s]", a ) == -1 )
                return;
139

140 141
        if( txt != NULL )
            asl = avahi_string_list_find( txt, "path" );
142 143 144 145 146
        if( asl != NULL )
        {
            size_t size;
            char *key = NULL;
            char *value = NULL;
147 148 149
            if( avahi_string_list_get_pair( asl, &key, &value, &size ) == 0 &&
                value != NULL )
            {
150 151 152 153 154 155
                if( asprintf( &psz_uri, "http://%s:%d%s",
                          psz_addr != NULL ? psz_addr : a, port, value ) == -1 )
                {
                    free( psz_addr );
                    return;
                }
156 157 158 159 160
            }
            if( key != NULL )
                avahi_free( (void *)key );
            if( value != NULL )
                avahi_free( (void *)value );
161 162 163
        }
        else
        {
164 165 166 167 168 169
            if( asprintf( &psz_uri, "http://%s:%d",
                      psz_addr != NULL ? psz_addr : a, port ) == -1 )
            {
                free( psz_addr );
                return;
            }
170 171
        }

172 173 174
        if( psz_addr != NULL )
            free( (void *)psz_addr );

175
        if( psz_uri != NULL )
176
        {
177
            p_input = input_item_NewExt( p_sd, psz_uri, name, 0, NULL, -1 );
178 179
            free( (void *)psz_uri );
        }
180
        if( p_input != NULL )
181
        {
182 183 184
            vlc_dictionary_insert( &p_sys->services_name_to_input_item,
                name, p_input );
            services_discovery_AddItem( p_sd, p_input, NULL /* no category */ );
Pierre d'Herbemont's avatar
Pierre d'Herbemont committed
185
            vlc_gc_decref( p_input );
186
       }
187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202
    }

    avahi_service_resolver_free( r );
}

/*****************************************************************************
 * browser_callback
 *****************************************************************************/
static void browse_callback(
    AvahiServiceBrowser *b,
    AvahiIfIndex interface,
    AvahiProtocol protocol,
    AvahiBrowserEvent event,
    const char *name,
    const char *type,
    const char *domain,
203
    AvahiLookupResultFlags flags,
204 205
    void* userdata )
{
206 207
    VLC_UNUSED(b);
    VLC_UNUSED(flags);
208 209 210 211 212 213
    services_discovery_t *p_sd = ( services_discovery_t* )userdata;
    services_discovery_sys_t *p_sys = p_sd->p_sys;
    if( event == AVAHI_BROWSER_NEW )
    {
        if( avahi_service_resolver_new( p_sys->client, interface, protocol,
                                        name, type, domain, AVAHI_PROTO_UNSPEC,
214
                                        0,
215 216 217 218 219 220
                                        resolve_callback, userdata ) == NULL )
        {
            msg_Err( p_sd, "failed to resolve service '%s': %s", name,
                     avahi_strerror( avahi_client_errno( p_sys->client ) ) );
        }
    }
221
    else if( name )
222
    {
223
        /** \todo Store the input id and search it, rather than searching the items */
224 225 226 227 228
        input_item_t *p_item;
        p_item = vlc_dictionary_value_for_key(
                        &p_sys->services_name_to_input_item,
                        name );
        if( !p_item )
229 230 231
            msg_Err( p_sd, "failed to find service '%s' in playlist", name );
        else
        {
232 233 234
            services_discovery_RemoveItem( p_sd, p_item );
            vlc_dictionary_remove_value_for_key(
                        &p_sys->services_name_to_input_item,
235
                        name, NULL, NULL );
236 237 238 239 240 241 242 243 244 245 246 247 248 249 250
        }
    }
}

/*****************************************************************************
 * Open: initialize and create stuff
 *****************************************************************************/
static int Open( vlc_object_t *p_this )
{
    services_discovery_t *p_sd = ( services_discovery_t* )p_this;
    services_discovery_sys_t *p_sys;
    int err;

    p_sd->p_sys = p_sys = (services_discovery_sys_t *)malloc(
        sizeof( services_discovery_sys_t ) );
251 252 253

    if( !p_sys )
        return VLC_ENOMEM;
254 255 256

    memset( p_sys, 0, sizeof(*p_sys) );

257 258
    vlc_dictionary_init( &p_sys->services_name_to_input_item, 1 );

259 260
    p_sys->poll = avahi_threaded_poll_new();
    if( p_sys->poll == NULL )
261
    {
262
        msg_Err( p_sd, "failed to create Avahi threaded poll" );
263 264 265
        goto error;
    }

266
    p_sys->client = avahi_client_new( avahi_threaded_poll_get(p_sys->poll),
267
                                      0, client_callback, p_sd, &err );
268 269 270 271 272 273 274 275 276 277
    if( p_sys->client == NULL )
    {
        msg_Err( p_sd, "failed to create avahi client: %s",
                 avahi_strerror( err ) );
        goto error;
    }

    p_sys->sb = avahi_service_browser_new( p_sys->client, AVAHI_IF_UNSPEC,
                                           AVAHI_PROTO_UNSPEC,
                                           "_vlc-http._tcp", NULL,
278
                                           0, browse_callback, p_sd );
279 280 281 282 283 284
    if( p_sys->sb == NULL )
    {
        msg_Err( p_sd, "failed to create avahi service browser" );
        goto error;
    }

285
    services_discovery_SetLocalizedName( p_sd, _("Bonjour") );
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
286

287 288 289 290 291 292 293
    return VLC_SUCCESS;

error:
    if( p_sys->sb != NULL )
        avahi_service_browser_free( p_sys->sb );
    if( p_sys->client != NULL )
        avahi_client_free( p_sys->client );
294 295
    if( p_sys->poll != NULL )
        avahi_threaded_poll_free( p_sys->poll );
296

297
    vlc_dictionary_clear( &p_sys->services_name_to_input_item, NULL, NULL );
298
    free( p_sys );
299 300 301 302 303 304 305 306 307 308 309 310 311 312

    return VLC_EGENERIC;
}

/*****************************************************************************
 * Close: cleanup
 *****************************************************************************/
static void Close( vlc_object_t *p_this )
{
    services_discovery_t *p_sd = ( services_discovery_t* )p_this;
    services_discovery_sys_t *p_sys = p_sd->p_sys;

    avahi_service_browser_free( p_sys->sb );
    avahi_client_free( p_sys->client );
313
    avahi_threaded_poll_free( p_sys->poll );
314

315
    vlc_dictionary_clear( &p_sys->services_name_to_input_item, NULL, NULL );
316 317
    free( p_sys );
}