You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
557 lines
18 KiB
557 lines
18 KiB
/*****************************************************************************
|
|
* avahi.c: Bonjour services discovery module
|
|
*****************************************************************************
|
|
* Copyright (C) 2005-2009, 2016 VideoLAN and VLC authors
|
|
*
|
|
* Authors: Jon Lech Johansen <jon@nanocrew.net>
|
|
* Jean-Baptiste Kempf <jb@videolan.org>
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
* Includes
|
|
*****************************************************************************/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#define VLC_MODULE_LICENSE VLC_LICENSE_GPL_2_PLUS
|
|
#include <vlc_common.h>
|
|
#include <vlc_plugin.h>
|
|
#include <vlc_services_discovery.h>
|
|
#include <vlc_renderer_discovery.h>
|
|
|
|
#include <avahi-client/client.h>
|
|
#include <avahi-client/publish.h>
|
|
#include <avahi-client/lookup.h>
|
|
#include <avahi-common/thread-watch.h>
|
|
#include <avahi-common/malloc.h>
|
|
#include <avahi-common/error.h>
|
|
|
|
/*****************************************************************************
|
|
* Module descriptor
|
|
*****************************************************************************/
|
|
|
|
/* Callbacks */
|
|
static int OpenSD ( vlc_object_t * );
|
|
static void CloseSD( vlc_object_t * );
|
|
static int OpenRD ( vlc_object_t * );
|
|
static void CloseRD( vlc_object_t * );
|
|
|
|
VLC_SD_PROBE_HELPER("avahi", N_("Zeroconf network services"), SD_CAT_LAN)
|
|
VLC_RD_PROBE_HELPER( "avahi_renderer", "Avahi Zeroconf renderer Discovery" )
|
|
|
|
vlc_module_begin ()
|
|
set_shortname( "Avahi" )
|
|
set_description( N_("Zeroconf services") )
|
|
set_category( CAT_PLAYLIST )
|
|
set_subcategory( SUBCAT_PLAYLIST_SD )
|
|
set_capability( "services_discovery", 0 )
|
|
set_callbacks( OpenSD, CloseSD )
|
|
add_shortcut( "mdns", "avahi" )
|
|
|
|
VLC_SD_PROBE_SUBMODULE
|
|
add_submodule() \
|
|
set_description( N_( "Avahi Renderer Discovery" ) )
|
|
set_category( CAT_SOUT )
|
|
set_subcategory( SUBCAT_SOUT_RENDERER )
|
|
set_capability( "renderer_discovery", 0 )
|
|
set_callbacks( OpenRD, CloseRD )
|
|
add_shortcut( "mdns_renderer", "avahi_renderer" )
|
|
VLC_RD_PROBE_SUBMODULE
|
|
vlc_module_end ()
|
|
|
|
/*****************************************************************************
|
|
* Local structures
|
|
*****************************************************************************/
|
|
|
|
typedef struct
|
|
{
|
|
AvahiThreadedPoll *poll;
|
|
AvahiClient *client;
|
|
vlc_dictionary_t services_name_to_input_item;
|
|
vlc_object_t *parent;
|
|
bool renderer;
|
|
} discovery_sys_t;
|
|
|
|
static const struct
|
|
{
|
|
const char *psz_protocol;
|
|
const char *psz_service_name;
|
|
bool b_renderer;
|
|
} protocols[] = {
|
|
{ "ftp", "_ftp._tcp", false },
|
|
{ "smb", "_smb._tcp", false },
|
|
{ "nfs", "_nfs._tcp", false },
|
|
{ "sftp", "_sftp-ssh._tcp", false },
|
|
{ "rtsp", "_rtsp._tcp", false },
|
|
{ "chromecast", "_googlecast._tcp", true },
|
|
};
|
|
#define NB_PROTOCOLS (sizeof(protocols) / sizeof(*protocols))
|
|
|
|
static char* get_string_list_value( AvahiStringList* txt, const char* key )
|
|
{
|
|
AvahiStringList *asl = avahi_string_list_find( txt, key );
|
|
if( asl == NULL )
|
|
return NULL;
|
|
char* res = NULL;
|
|
char *sl_key = NULL;
|
|
char *sl_value = NULL;
|
|
if( avahi_string_list_get_pair( asl, &sl_key, &sl_value, NULL ) == 0 &&
|
|
sl_value != NULL )
|
|
{
|
|
res = strdup( sl_value );
|
|
}
|
|
|
|
if( sl_key != NULL )
|
|
avahi_free( (void *)sl_key );
|
|
if( sl_value != NULL )
|
|
avahi_free( (void *)sl_value );
|
|
return res;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* helpers
|
|
*****************************************************************************/
|
|
static void add_renderer( const char *psz_protocol, const char *psz_name,
|
|
const char *psz_addr, uint16_t i_port,
|
|
AvahiStringList *txt, discovery_sys_t *p_sys )
|
|
{
|
|
vlc_renderer_discovery_t *p_rd = ( vlc_renderer_discovery_t* )(p_sys->parent);
|
|
AvahiStringList *asl = NULL;
|
|
char *friendly_name = NULL;
|
|
char *icon_uri = NULL;
|
|
char *uri = NULL;
|
|
char *model = NULL;
|
|
const char *demux = NULL;
|
|
const char *extra_uri = NULL;
|
|
int renderer_flags = 0;
|
|
|
|
if( !strcmp( "chromecast", psz_protocol ) ) {
|
|
/* Capabilities */
|
|
asl = avahi_string_list_find( txt, "ca" );
|
|
if( asl != NULL ) {
|
|
char *key = NULL;
|
|
char *value = NULL;
|
|
if( avahi_string_list_get_pair( asl, &key, &value, NULL ) == 0 &&
|
|
value != NULL )
|
|
{
|
|
int ca = atoi( value );
|
|
|
|
if( ( ca & 0x01 ) != 0 )
|
|
renderer_flags |= VLC_RENDERER_CAN_VIDEO;
|
|
if( ( ca & 0x04 ) != 0 )
|
|
renderer_flags |= VLC_RENDERER_CAN_AUDIO;
|
|
}
|
|
|
|
if( key != NULL )
|
|
avahi_free( (void *)key );
|
|
if( value != NULL )
|
|
avahi_free( (void *)value );
|
|
}
|
|
|
|
/* Friendly name */
|
|
friendly_name = get_string_list_value( txt, "fn" );
|
|
|
|
/* Icon */
|
|
char* icon_raw = get_string_list_value( txt, "ic" );
|
|
if( icon_raw != NULL ) {
|
|
if( asprintf( &icon_uri, "http://%s:8008%s", psz_addr, icon_raw) < 0 )
|
|
icon_uri = NULL;
|
|
free( icon_raw );
|
|
}
|
|
|
|
model = get_string_list_value( txt, "md" );
|
|
|
|
if( asprintf( &uri, "%s://%s:%u", psz_protocol, psz_addr, i_port ) < 0 )
|
|
goto error;
|
|
|
|
extra_uri = renderer_flags & VLC_RENDERER_CAN_VIDEO ? NULL : "no-video";
|
|
demux = "cc_demux";
|
|
}
|
|
|
|
if ( friendly_name && model ) {
|
|
char* combined;
|
|
if ( asprintf( &combined, "%s (%s)", friendly_name, model ) == -1 )
|
|
combined = NULL;
|
|
if ( combined != NULL ) {
|
|
free(friendly_name);
|
|
friendly_name = combined;
|
|
}
|
|
}
|
|
|
|
vlc_renderer_item_t *p_renderer_item =
|
|
vlc_renderer_item_new( psz_protocol, friendly_name ? friendly_name : psz_name, uri, extra_uri,
|
|
demux, icon_uri, renderer_flags );
|
|
if( p_renderer_item == NULL )
|
|
goto error;
|
|
|
|
vlc_dictionary_insert( &p_sys->services_name_to_input_item,
|
|
psz_name, p_renderer_item);
|
|
vlc_rd_add_item( p_rd, p_renderer_item );
|
|
|
|
error:
|
|
free( friendly_name );
|
|
free( icon_uri );
|
|
free( uri );
|
|
free( model );
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* client_callback
|
|
*****************************************************************************/
|
|
static void client_callback( AvahiClient *c, AvahiClientState state,
|
|
void * userdata )
|
|
{
|
|
discovery_sys_t *p_sys = userdata;
|
|
|
|
if( state == AVAHI_CLIENT_FAILURE &&
|
|
(avahi_client_errno(c) == AVAHI_ERR_DISCONNECTED) )
|
|
{
|
|
msg_Err( p_sys->parent, "avahi client disconnected" );
|
|
avahi_threaded_poll_quit( p_sys->poll );
|
|
}
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* 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,
|
|
AvahiLookupResultFlags flags,
|
|
void* userdata )
|
|
{
|
|
discovery_sys_t *p_sys = userdata;
|
|
|
|
VLC_UNUSED(interface); VLC_UNUSED(host_name);
|
|
VLC_UNUSED(flags);
|
|
|
|
if( event == AVAHI_RESOLVER_FAILURE )
|
|
{
|
|
msg_Err( p_sys->parent,
|
|
"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;
|
|
char *psz_addr = NULL;
|
|
AvahiStringList *asl = NULL;
|
|
input_item_t *p_input = NULL;
|
|
|
|
msg_Info( p_sys->parent, "service '%s' of type '%s' in domain '%s' port %i",
|
|
name, type, domain, port );
|
|
|
|
avahi_address_snprint(a, (sizeof(a)/sizeof(a[0]))-1, address);
|
|
if( protocol == AVAHI_PROTO_INET6 )
|
|
if( asprintf( &psz_addr, "[%s]", a ) == -1 )
|
|
{
|
|
avahi_service_resolver_free( r );
|
|
return;
|
|
}
|
|
|
|
const char *psz_protocol = NULL;
|
|
bool is_renderer = false;
|
|
for( unsigned int i = 0; i < NB_PROTOCOLS; i++ )
|
|
{
|
|
if( !strcmp(type, protocols[i].psz_service_name) )
|
|
{
|
|
psz_protocol = protocols[i].psz_protocol;
|
|
is_renderer = protocols[i].b_renderer;
|
|
break;
|
|
}
|
|
}
|
|
if( psz_protocol == NULL )
|
|
{
|
|
free( psz_addr );
|
|
avahi_service_resolver_free( r );
|
|
return;
|
|
}
|
|
|
|
if( txt != NULL && is_renderer )
|
|
{
|
|
const char* addr_v4v6 = psz_addr != NULL ? psz_addr : a;
|
|
add_renderer( psz_protocol, name, addr_v4v6, port, txt, p_sys );
|
|
free( psz_addr );
|
|
avahi_service_resolver_free( r );
|
|
return;
|
|
}
|
|
|
|
if( txt != NULL )
|
|
asl = avahi_string_list_find( txt, "path" );
|
|
if( asl != NULL )
|
|
{
|
|
size_t size;
|
|
char *key = NULL;
|
|
char *value = NULL;
|
|
if( avahi_string_list_get_pair( asl, &key, &value, &size ) == 0 &&
|
|
value != NULL )
|
|
{
|
|
if( asprintf( &psz_uri, "%s://%s:%d%s",
|
|
psz_protocol,
|
|
psz_addr != NULL ? psz_addr : a,
|
|
port, value ) == -1 )
|
|
{
|
|
free( psz_addr );
|
|
avahi_service_resolver_free( r );
|
|
return;
|
|
}
|
|
}
|
|
if( key != NULL )
|
|
avahi_free( (void *)key );
|
|
if( value != NULL )
|
|
avahi_free( (void *)value );
|
|
}
|
|
else
|
|
{
|
|
if( asprintf( &psz_uri, "%s://%s:%d",
|
|
psz_protocol,
|
|
psz_addr != NULL ? psz_addr : a, port ) == -1 )
|
|
{
|
|
free( psz_addr );
|
|
avahi_service_resolver_free( r );
|
|
return;
|
|
}
|
|
}
|
|
|
|
free( psz_addr );
|
|
|
|
if( psz_uri != NULL )
|
|
{
|
|
p_input = input_item_NewDirectory( psz_uri, name, ITEM_NET );
|
|
free( psz_uri );
|
|
}
|
|
if( p_input != NULL )
|
|
{
|
|
services_discovery_t *p_sd = ( services_discovery_t* )(p_sys->parent);
|
|
vlc_dictionary_insert( &p_sys->services_name_to_input_item,
|
|
name, p_input );
|
|
services_discovery_AddItem( p_sd, p_input );
|
|
}
|
|
}
|
|
|
|
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,
|
|
AvahiLookupResultFlags flags,
|
|
void* userdata )
|
|
{
|
|
VLC_UNUSED(b);
|
|
VLC_UNUSED(flags);
|
|
discovery_sys_t *p_sys = userdata;
|
|
if( event == AVAHI_BROWSER_NEW )
|
|
{
|
|
if( avahi_service_resolver_new( p_sys->client, interface, protocol,
|
|
name, type, domain, AVAHI_PROTO_UNSPEC,
|
|
0,
|
|
resolve_callback, userdata ) == NULL )
|
|
{
|
|
msg_Err( p_sys->parent, "failed to resolve service '%s': %s", name,
|
|
avahi_strerror( avahi_client_errno( p_sys->client ) ) );
|
|
}
|
|
}
|
|
else if( event == AVAHI_BROWSER_REMOVE && name )
|
|
{
|
|
/** \todo Store the input id and search it, rather than searching the items */
|
|
void *p_item;
|
|
p_item = vlc_dictionary_value_for_key(
|
|
&p_sys->services_name_to_input_item,
|
|
name );
|
|
if( !p_item )
|
|
msg_Err( p_sys->parent, "failed to find service '%s' in playlist", name );
|
|
else
|
|
{
|
|
if( p_sys->renderer )
|
|
{
|
|
vlc_renderer_discovery_t *p_rd = ( vlc_renderer_discovery_t* )(p_sys->parent);
|
|
vlc_rd_remove_item( p_rd, p_item );
|
|
vlc_renderer_item_release( p_item );
|
|
}
|
|
else
|
|
{
|
|
services_discovery_t *p_sd = ( services_discovery_t* )(p_sys->parent);
|
|
services_discovery_RemoveItem( p_sd, p_item );
|
|
input_item_Release( p_item );
|
|
}
|
|
vlc_dictionary_remove_value_for_key(
|
|
&p_sys->services_name_to_input_item,
|
|
name, NULL, NULL );
|
|
}
|
|
}
|
|
}
|
|
|
|
static void clear_input_item( void* p_item, void* p_obj )
|
|
{
|
|
VLC_UNUSED( p_obj );
|
|
input_item_Release( p_item );
|
|
}
|
|
|
|
static void clear_renderer_item( void* p_item, void* p_obj )
|
|
{
|
|
VLC_UNUSED( p_obj );
|
|
vlc_renderer_item_release( p_item );
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Open: initialize and create stuff
|
|
*****************************************************************************/
|
|
static int OpenCommon( discovery_sys_t *p_sys )
|
|
{
|
|
int err;
|
|
|
|
vlc_dictionary_init( &p_sys->services_name_to_input_item, 1 );
|
|
|
|
p_sys->poll = avahi_threaded_poll_new();
|
|
if( p_sys->poll == NULL )
|
|
{
|
|
msg_Err( p_sys->parent, "failed to create Avahi threaded poll" );
|
|
goto error;
|
|
}
|
|
|
|
p_sys->client = avahi_client_new( avahi_threaded_poll_get(p_sys->poll),
|
|
0, client_callback, p_sys, &err );
|
|
if( p_sys->client == NULL )
|
|
{
|
|
msg_Err( p_sys->parent, "failed to create avahi client: %s",
|
|
avahi_strerror( err ) );
|
|
goto error;
|
|
}
|
|
|
|
for( unsigned i = 0; i < NB_PROTOCOLS; i++ )
|
|
{
|
|
if( protocols[i].b_renderer != p_sys->renderer )
|
|
continue;
|
|
|
|
AvahiServiceBrowser *sb;
|
|
sb = avahi_service_browser_new( p_sys->client, AVAHI_IF_UNSPEC,
|
|
AVAHI_PROTO_UNSPEC,
|
|
protocols[i].psz_service_name, NULL,
|
|
0, browse_callback, p_sys );
|
|
if( sb == NULL )
|
|
{
|
|
msg_Err( p_sys->parent, "failed to create avahi service browser %s", avahi_strerror( avahi_client_errno(p_sys->client) ) );
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
avahi_threaded_poll_start( p_sys->poll );
|
|
|
|
return VLC_SUCCESS;
|
|
|
|
error:
|
|
if( p_sys->client != NULL )
|
|
avahi_client_free( p_sys->client );
|
|
if( p_sys->poll != NULL )
|
|
avahi_threaded_poll_free( p_sys->poll );
|
|
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
static int OpenSD( vlc_object_t *p_this )
|
|
{
|
|
services_discovery_t *p_sd = ( services_discovery_t* )p_this;
|
|
p_sd->description = _("Zeroconf network services");
|
|
|
|
discovery_sys_t *p_sys = p_sd->p_sys = calloc( 1, sizeof( discovery_sys_t ) );
|
|
if( !p_sd->p_sys )
|
|
return VLC_ENOMEM;
|
|
p_sys->parent = p_this;
|
|
p_sys->renderer = false;
|
|
|
|
int ret = OpenCommon( p_sys );
|
|
if( ret != VLC_SUCCESS )
|
|
{
|
|
vlc_dictionary_clear( &p_sys->services_name_to_input_item,
|
|
clear_input_item, NULL );
|
|
free( p_sys );
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int OpenRD( vlc_object_t *p_this )
|
|
{
|
|
vlc_renderer_discovery_t *p_rd = (vlc_renderer_discovery_t *)p_this;
|
|
|
|
discovery_sys_t *p_sys = p_rd->p_sys = calloc( 1, sizeof( discovery_sys_t ) );
|
|
if( !p_rd->p_sys )
|
|
return VLC_ENOMEM;
|
|
p_sys->parent = p_this;
|
|
p_sys->renderer = true;
|
|
|
|
int ret = OpenCommon( p_sys );
|
|
if( ret != VLC_SUCCESS )
|
|
{
|
|
vlc_dictionary_clear( &p_sys->services_name_to_input_item,
|
|
clear_renderer_item, NULL );
|
|
free( p_sys );
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Close: cleanup
|
|
*****************************************************************************/
|
|
static void CloseCommon( discovery_sys_t *p_sys )
|
|
{
|
|
avahi_threaded_poll_stop( p_sys->poll );
|
|
|
|
avahi_client_free( p_sys->client );
|
|
avahi_threaded_poll_free( p_sys->poll );
|
|
|
|
}
|
|
|
|
static void CloseSD( vlc_object_t *p_this )
|
|
{
|
|
services_discovery_t *p_sd = ( services_discovery_t* )p_this;
|
|
discovery_sys_t *p_sys = p_sd->p_sys;
|
|
CloseCommon( p_sys );
|
|
vlc_dictionary_clear( &p_sys->services_name_to_input_item,
|
|
clear_input_item, NULL );
|
|
free( p_sys );
|
|
}
|
|
|
|
static void CloseRD( vlc_object_t *p_this )
|
|
{
|
|
vlc_renderer_discovery_t *p_rd = (vlc_renderer_discovery_t *)p_this;
|
|
discovery_sys_t *p_sys = p_rd->p_sys;
|
|
CloseCommon( p_sys );
|
|
vlc_dictionary_clear( &p_sys->services_name_to_input_item,
|
|
clear_renderer_item, NULL );
|
|
free( p_sys );
|
|
}
|
|
|