owntone-server/src/rend-avahi.c

350 lines
10 KiB
C

/*
* Rendezvous support with avahi
*
* Copyright (C) 2005 Sebastian Dröge <slomo@ubuntu.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-1307 USA
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <assert.h>
#include <pthread.h>
#include <net/if.h>
#include <avahi-client/client.h>
#include <avahi-client/publish.h>
#include <avahi-client/client.h>
#include <avahi-common/alternative.h>
#include <avahi-common/error.h>
#include <avahi-common/simple-watch.h>
#include <avahi-common/timeval.h>
#include <avahi-common/malloc.h>
#include "daapd.h"
#include "err.h"
static AvahiClient *mdns_client = NULL;
static AvahiEntryGroup *mdns_group = NULL;
static AvahiSimplePoll *simple_poll = NULL;
typedef struct tag_rend_avahi_group_entry {
char *name;
char *type;
int port;
char *iface;
char *txt;
struct tag_rend_avahi_group_entry *next;
} REND_AVAHI_GROUP_ENTRY;
static REND_AVAHI_GROUP_ENTRY rend_avahi_entries = { NULL, NULL, 0, NULL };
static pthread_t rend_tid;
static pthread_cond_t rend_avahi_cond;
static pthread_mutex_t rend_avahi_mutex;
static void _rend_avahi_signal(void);
static void _rend_avahi_wait_on(void *what);
static void _rend_avahi_lock(void);
static void _rend_avahi_unlock(void);
static int _rend_avahi_create_services(void);
/* add a new group entry node */
static int _rend_avahi_add_group_entry(char *name, char *type, int port, char *iface, char *txt) {
REND_AVAHI_GROUP_ENTRY *pge;
pge = (REND_AVAHI_GROUP_ENTRY *)malloc(sizeof(REND_AVAHI_GROUP_ENTRY));
if(!pge)
return 0;
pge->name = strdup(name);
pge->type = strdup(type);
pge->iface = strdup(iface);
pge->txt = strdup(txt);
pge->port = port;
_rend_avahi_lock();
pge->next = rend_avahi_entries.next;
rend_avahi_entries.next = pge;
_rend_avahi_unlock();
return 1;
}
static void *rend_poll(void *arg) {
int ret;
while((ret = avahi_simple_poll_iterate(simple_poll,-1)) == 0);
if(ret < 0) {
DPRINTF(E_WARN,L_REND,"Avahi poll thread quit iwth error: %s\n",
avahi_strerror(avahi_client_errno(mdns_client)));
} else {
DPRINTF(E_DBG,L_REND,"Avahi poll thread quit\n");
}
return NULL;
}
static void entry_group_callback(AvahiEntryGroup *g, AvahiEntryGroupState state, AVAHI_GCC_UNUSED void *userdata) {
// assert(g == mdns_group);
switch (state) {
case AVAHI_ENTRY_GROUP_ESTABLISHED:
DPRINTF(E_DBG, L_REND, "Successfully added mdns services\n");
_rend_avahi_signal();
break;
case AVAHI_ENTRY_GROUP_COLLISION:
DPRINTF(E_DBG, L_REND, "Group collision\n");
/*
new_name = avahi_alternative_service_name(mdns_name);
DPRINTF(E_WARN, L_REND, "mdns service name collision. Renamed service %s -> %s\n", mdns_name, new_name);
free(mdns_name);
mdns_name = new_name;
add_services(avahi_entry_group_get_client(g));
*/
break;
case AVAHI_ENTRY_GROUP_FAILURE :
avahi_simple_poll_quit(simple_poll);
break;
case AVAHI_ENTRY_GROUP_UNCOMMITED:
case AVAHI_ENTRY_GROUP_REGISTERING:
break;
}
}
void _rend_avahi_lock(void) {
if(pthread_mutex_lock(&rend_avahi_mutex))
DPRINTF(E_FATAL,L_REND,"Could not lock mutex\n");
}
void _rend_avahi_unlock(void) {
pthread_mutex_unlock(&rend_avahi_mutex);
}
void _rend_avahi_signal(void) {
/* kick the condition for waiters */
_rend_avahi_lock();
pthread_cond_signal(&rend_avahi_cond);
_rend_avahi_unlock();
}
void _rend_avahi_wait_on(void *what) {
DPRINTF(E_DBG,L_REND,"Waiting on something...\n");
if(pthread_mutex_lock(&rend_avahi_mutex))
DPRINTF(E_FATAL,L_REND,"Could not lock mutex\n");
while(!what) {
pthread_cond_wait(&rend_avahi_cond,&rend_avahi_mutex);
}
_rend_avahi_unlock();
DPRINTF(E_DBG,L_REND,"Done waiting.\n");
}
int rend_register(char *name, char *type, int port, char *iface, char *txt) {
DPRINTF(E_DBG,L_REND,"Adding %s/%s\n",name,type);
_rend_avahi_add_group_entry(name,type,port,iface,txt);
if(mdns_group) {
DPRINTF(E_DBG,L_MISC,"Resetting mdns group\n");
avahi_entry_group_reset(mdns_group);
}
DPRINTF(E_DBG,L_REND,"Creating service group (again?)\n");
_rend_avahi_create_services();
return 0;
}
/**
* register the block of services
*
* @returns true if successful, false otherwise
*/
int _rend_avahi_create_services(void) {
int ret = 0;
REND_AVAHI_GROUP_ENTRY *pentry;
AvahiStringList *psl;
unsigned char count=0;
unsigned char *key,*nextkey;
unsigned char *newtxt;
DPRINTF(E_DBG,L_REND,"Creting service group\n");
if(!rend_avahi_entries.next) {
DPRINTF(E_DBG,L_REND,"No entries yet... skipping service create\n");
return 1;
}
if (mdns_group == NULL) {
if (!(mdns_group = avahi_entry_group_new(mdns_client,
entry_group_callback,
NULL))) {
DPRINTF(E_WARN, L_REND, "Could not create AvahiEntryGroup: %s\n",
avahi_strerror(avahi_client_errno(mdns_client)));
return 0;
}
}
/* wait for entry group to be created */
_rend_avahi_wait_on(mdns_group);
_rend_avahi_lock();
pentry = rend_avahi_entries.next;
while(pentry) {
/* TODO: honor iface parameter */
DPRINTF(E_DBG,L_REND,"Re-registering %s/%s\n",pentry->name,pentry->type);
/* build a string list */
psl = NULL;
newtxt = (unsigned char *)strdup(pentry->txt);
if(!newtxt)
DPRINTF(E_FATAL,L_REND,"malloc\n");
key=nextkey=newtxt;
if(*nextkey)
count = *nextkey;
DPRINTF(E_DBG,L_REND,"Found key of size %d\n",count);
while((*nextkey)&&(nextkey < (newtxt + strlen(pentry->txt)))) {
key = nextkey + 1;
nextkey += (count+1);
count = *nextkey;
*nextkey = '\0';
psl=avahi_string_list_add(psl,(char*)key);
DPRINTF(E_DBG,L_REND,"Added key %s\n",key);
*nextkey=count;
}
free(newtxt);
if ((ret = avahi_entry_group_add_service_strlst(mdns_group,
AVAHI_IF_UNSPEC,
AVAHI_PROTO_UNSPEC, 0,
avahi_strdup(pentry->name),
avahi_strdup(pentry->type),
NULL, NULL,pentry->port,
psl)) < 0) {
DPRINTF(E_WARN, L_REND, "Could not add mdns services: %s\n", avahi_strerror(ret));
avahi_string_list_free(psl);
_rend_avahi_unlock();
return 0;
}
pentry = pentry->next;
}
_rend_avahi_unlock();
if ((ret = avahi_entry_group_commit(mdns_group)) < 0) {
DPRINTF(E_WARN, L_REND, "Could not commit mdns services: %s\n",
avahi_strerror(avahi_client_errno(mdns_client)));
return 0;
}
return 1;
}
static void client_callback(AvahiClient *c, AvahiClientState state, AVAHI_GCC_UNUSED void * userdata) {
assert(c);
switch(state) {
case AVAHI_CLIENT_S_RUNNING:
DPRINTF(E_LOG,L_REND,"Client running\n");
if(!mdns_group)
_rend_avahi_create_services();
break;
case AVAHI_CLIENT_S_COLLISION:
DPRINTF(E_LOG,L_REND,"Client collision\n");
if(mdns_group)
avahi_entry_group_reset(mdns_group);
break;
case AVAHI_CLIENT_FAILURE:
DPRINTF(E_LOG,L_REND,"Client failure\n");
avahi_simple_poll_quit(simple_poll);
break;
case AVAHI_CLIENT_S_REGISTERING:
DPRINTF(E_LOG,L_REND,"Client registering\n");
if(mdns_group)
avahi_entry_group_reset(mdns_group);
break;
case AVAHI_CLIENT_CONNECTING:
break;
}
}
int rend_init(char *user) {
int error;
if(pthread_cond_init(&rend_avahi_cond,NULL)) {
DPRINTF(E_LOG,L_REND,"Could not initialize rendezvous condition\n");
return -1;
}
if(pthread_mutex_init(&rend_avahi_mutex,NULL)) {
DPRINTF(E_LOG,L_REND,"Could not initialize rendezvous mutex\n");
return -1;
}
DPRINTF(E_DBG, L_REND, "Initializing avahi\n");
if(!(simple_poll = avahi_simple_poll_new())) {
DPRINTF(E_LOG,L_REND,"Error starting poll thread\n");
return -1;
}
/*
mdns_name = strdup(name);
mdns_port = port;
// if ((interface != NULL) && (if_nametoindex(interface) != 0))
// mdns_interface = if_nametoindex(interface);
// else
mdns_interface = AVAHI_IF_UNSPEC;
*/
if (!(mdns_client = avahi_client_new(avahi_simple_poll_get(simple_poll),
0,client_callback,NULL,&error))) {
DPRINTF(E_WARN, L_REND, "avahi_client_new: Error in avahi: %s\n",
avahi_strerror(avahi_client_errno(mdns_client)));
avahi_simple_poll_free(simple_poll);
return -1;
}
DPRINTF(E_DBG, L_REND, "Starting avahi polling thread\n");
if(pthread_create(&rend_tid, NULL, rend_poll, NULL)) {
DPRINTF(E_FATAL,L_REND,"Could not start avahi polling thread.\n");
}
return 0;
}
int rend_stop() {
avahi_simple_poll_quit(simple_poll);
if (mdns_client != NULL)
avahi_client_free(mdns_client);
return 0;
}
int rend_running(void) {
return 1;
}
int rend_unregister(char *name, char *type, int port) {
return 0;
}