add avahi support to trunk

This commit is contained in:
Ron Pedde 2006-11-28 22:11:18 +00:00
parent 9a6718d22c
commit c4c823affb
3 changed files with 367 additions and 4 deletions

View File

@ -39,6 +39,7 @@ fi
AC_DEFINE_UNQUOTED(CONFFILE,"${CONFFILE}",Where the config file is) AC_DEFINE_UNQUOTED(CONFFILE,"${CONFFILE}",Where the config file is)
rend_posix=true rend_posix=true
rend_avahi=false
db_sqlite=false db_sqlite=false
db_sqlite3=false db_sqlite3=false
use_ffmpeg=false; use_ffmpeg=false;
@ -67,16 +68,25 @@ AC_ARG_ENABLE(sqlite3,[ --enable-sqlite3 Enable sqlite3 db backend],
AC_ARG_ENABLE(mdns,[ --enable-mdns Enable mDNS support], AC_ARG_ENABLE(mdns,[ --enable-mdns Enable mDNS support],
[ case "${enableval}" in [ case "${enableval}" in
yes) ;; yes) ;;
no) rend_posix=false; rend_howl=false; CPPFLAGS="${CPPFLAGS} -DWITHOUT_MDNS";; no) rend_posix=false; rend_howl=false; rend_avahi=false; CPPFLAGS="${CPPFLAGS} -DWITHOUT_MDNS";;
*) AC_MSG_ERROR(bad value ${enableval} for --disable-mdns);; *) AC_MSG_ERROR(bad value ${enableval} for --disable-mdns);;
esac ]) esac ])
AC_ARG_ENABLE(nslu2,[ --enable-nslu2 Build for NSLU2/uNSLUng], AC_ARG_ENABLE(nslu2,[ --enable-nslu2 Build for NSLU2/uNSLUng],
CFLAGS="${CFLAGS} -DNSLU2") CFLAGS="${CFLAGS} -DNSLU2")
AC_ARG_ENABLE(avahi,[ --enable-avahi Use avahi 0.6 or later],
[ case "${enableval}" in
yes) PKG_CHECK_MODULES(AVAHI, [ avahi-client >= 0.6 ]);
rend_avahi=true; rend_howl=false; rend_posix=false; LDFLAGS="${LDFLAGS} $AVAHI_LIBS";
CPPFLAGS="${CPPFLAGS} $AVAHI_CFLAGS -DWITH_AVAHI";;
no) rend_avahi=false;;
*) AC_MSG_ERROR(bad value ${enableval} for --enable-avahi);;
esac ])
AC_ARG_ENABLE(howl,[ --enable-howl Use howl 0.9.2 or later], AC_ARG_ENABLE(howl,[ --enable-howl Use howl 0.9.2 or later],
[ case "${enableval}" in [ case "${enableval}" in
yes) rend_howl=true; rend_posix=false; LIBS="${LIBS} -lhowl"; yes) rend_howl=true; rend_posix=false; rend_avahi=false; LIBS="${LIBS} -lhowl";
CPPFLAGS="${CPPFLAGS} -DWITH_HOWL";; CPPFLAGS="${CPPFLAGS} -DWITH_HOWL";;
no) rend_howl=false;; no) rend_howl=false;;
*) AC_MSG_ERROR(bad value ${enableval} for --enable-howl);; *) AC_MSG_ERROR(bad value ${enableval} for --enable-howl);;
@ -107,6 +117,7 @@ AC_ARG_ENABLE(ffmpeg,[ --enable-ffmpeg Enable ffmpeg transcode support],
AM_CONDITIONAL(COND_REND_HOWL, test x$rend_howl = xtrue) AM_CONDITIONAL(COND_REND_HOWL, test x$rend_howl = xtrue)
AM_CONDITIONAL(COND_REND_POSIX, test x$rend_posix = xtrue) AM_CONDITIONAL(COND_REND_POSIX, test x$rend_posix = xtrue)
AM_CONDITIONAL(COND_REND_AVAHI, test x$rend_avahi = xtrue)
AM_CONDITIONAL(COND_OGGVORBIS, test x$use_oggvorbis = xtrue) AM_CONDITIONAL(COND_OGGVORBIS, test x$use_oggvorbis = xtrue)
AM_CONDITIONAL(COND_FLAC, test x$use_flac = xtrue) AM_CONDITIONAL(COND_FLAC, test x$use_flac = xtrue)
AM_CONDITIONAL(COND_MUSEPACK, test x$use_musepack = xtrue) AM_CONDITIONAL(COND_MUSEPACK, test x$use_musepack = xtrue)

View File

@ -18,6 +18,10 @@ if COND_REND_OSX
ORENDSRC=rend-osx.c rend-unix.c ORENDSRC=rend-osx.c rend-unix.c
endif endif
if COND_REND_AVAHI
ARENDSRC=rend-avahi.c
endif
if COND_OGGVORBIS if COND_OGGVORBIS
OGGVORBISSRC=scan-ogg.c OGGVORBISSRC=scan-ogg.c
endif endif
@ -58,8 +62,8 @@ mt_daapd_SOURCES = main.c daapd.h rend.h uici.c uici.h webserver.c \
smart-parser.c smart-parser.h xml-rpc.c xml-rpc.h \ smart-parser.c smart-parser.h xml-rpc.c xml-rpc.h \
os.h ll.c ll.h conf.c conf.h compat.c compat.h util.c util.h \ os.h ll.c ll.h conf.c conf.h compat.c compat.h util.c util.h \
os-unix.h os-unix.c os.h plugin.c plugin.h db-sql-updates.c \ os-unix.h os-unix.c os.h plugin.c plugin.h db-sql-updates.c \
$(PRENDSRC) $(ORENDSRC) $(HRENDSRC) $(OGGVORBISSRC) $(FLACSRC) \ $(PRENDSRC) $(ORENDSRC) $(HRENDSRC) $(ARENDSRC) $(OGGVORBISSRC) \
$(MUSEPACKSRC) $(SQLITEDB) $(SQLITE3DB) $(SQLDB) $(GDBM) $(FLACSRC) $(MUSEPACKSRC) $(SQLITEDB) $(SQLITE3DB) $(SQLDB) $(GDBM)
EXTRA_DIST = mDNS.c mDNSClientAPI.h mDNSDebug.h mDNSPosix.c \ EXTRA_DIST = mDNS.c mDNSClientAPI.h mDNSDebug.h mDNSPosix.c \
mDNSUNP.c mDNSPlatformFunctions.h mDNSPosix.h mDNSUNP.h \ mDNSUNP.c mDNSPlatformFunctions.h mDNSPosix.h mDNSUNP.h \

348
src/rend-avahi.c Normal file
View File

@ -0,0 +1,348 @@
/*
* 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 "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;
}