mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-15 08:45:02 -05:00
Replace AVL tree in DACP by a static hash using gperf
This commit is contained in:
parent
7e35068d46
commit
cbb1669ee1
1
src/.gitignore
vendored
1
src/.gitignore
vendored
@ -10,3 +10,4 @@ RSP2SQL.[ch]
|
|||||||
|
|
||||||
daap_query_hash.c
|
daap_query_hash.c
|
||||||
rsp_query_hash.c
|
rsp_query_hash.c
|
||||||
|
dacp_prop_hash.c
|
||||||
|
@ -23,11 +23,13 @@ endif
|
|||||||
|
|
||||||
GPERF_FILES = \
|
GPERF_FILES = \
|
||||||
daap_query.gperf \
|
daap_query.gperf \
|
||||||
rsp_query.gperf
|
rsp_query.gperf \
|
||||||
|
dacp_prop.gperf
|
||||||
|
|
||||||
GPERF_PRODUCTS = \
|
GPERF_PRODUCTS = \
|
||||||
daap_query_hash.c \
|
daap_query_hash.c \
|
||||||
rsp_query_hash.c
|
rsp_query_hash.c \
|
||||||
|
dacp_prop_hash.c
|
||||||
|
|
||||||
ANTLR_GRAMMARS = \
|
ANTLR_GRAMMARS = \
|
||||||
RSP.g RSP2SQL.g \
|
RSP.g RSP2SQL.g \
|
||||||
|
22
src/dacp_prop.gperf
Normal file
22
src/dacp_prop.gperf
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
%language=ANSI-C
|
||||||
|
%readonly-tables
|
||||||
|
%enum
|
||||||
|
%switch=1
|
||||||
|
%compare-lengths
|
||||||
|
%define hash-function-name dacp_hash_prop
|
||||||
|
%define lookup-function-name dacp_find_prop
|
||||||
|
%define slot-name desc
|
||||||
|
%struct-type
|
||||||
|
%omit-struct-type
|
||||||
|
struct dacp_prop_map;
|
||||||
|
%%
|
||||||
|
"dmcp.volume", dacp_propget_volume, dacp_propset_volume
|
||||||
|
"dacp.playerstate", dacp_propget_playerstate, NULL
|
||||||
|
"dacp.nowplaying", dacp_propget_nowplaying, NULL
|
||||||
|
"dacp.playingtime", dacp_propget_playingtime, dacp_propset_playingtime
|
||||||
|
"dacp.volumecontrollable", dacp_propget_volumecontrollable, NULL
|
||||||
|
"dacp.availableshufflestates", dacp_propget_availableshufflestates, NULL
|
||||||
|
"dacp.availablerepeatstates", dacp_propget_availablerepeatstates, NULL
|
||||||
|
"dacp.shufflestate", dacp_propget_shufflestate, dacp_propset_shufflestate
|
||||||
|
"dacp.repeatstate", dacp_propget_repeatstate, dacp_propset_repeatstate
|
||||||
|
"dacp.userrating", NULL, dacp_propset_userrating
|
198
src/httpd_dacp.c
198
src/httpd_dacp.c
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2010 Julien BLACHE <jb@jblache.org>
|
* Copyright (C) 2010-2011 Julien BLACHE <jb@jblache.org>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -39,7 +39,6 @@
|
|||||||
|
|
||||||
#include <event.h>
|
#include <event.h>
|
||||||
#include "evhttp/evhttp.h"
|
#include "evhttp/evhttp.h"
|
||||||
#include <avl.h>
|
|
||||||
|
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
#include "misc.h"
|
#include "misc.h"
|
||||||
@ -78,7 +77,6 @@ typedef void (*dacp_propget)(struct evbuffer *evbuf, struct player_status *statu
|
|||||||
typedef void (*dacp_propset)(const char *value, struct evkeyvalq *query);
|
typedef void (*dacp_propset)(const char *value, struct evkeyvalq *query);
|
||||||
|
|
||||||
struct dacp_prop_map {
|
struct dacp_prop_map {
|
||||||
uint32_t hash;
|
|
||||||
char *desc;
|
char *desc;
|
||||||
dacp_propget propget;
|
dacp_propget propget;
|
||||||
dacp_propset propset;
|
dacp_propset propset;
|
||||||
@ -117,21 +115,9 @@ dacp_propset_repeatstate(const char *value, struct evkeyvalq *query);
|
|||||||
static void
|
static void
|
||||||
dacp_propset_userrating(const char *value, struct evkeyvalq *query);
|
dacp_propset_userrating(const char *value, struct evkeyvalq *query);
|
||||||
|
|
||||||
static struct dacp_prop_map dacp_props[] =
|
|
||||||
{
|
|
||||||
{ 0, "dmcp.volume", dacp_propget_volume, dacp_propset_volume },
|
|
||||||
{ 0, "dacp.playerstate", dacp_propget_playerstate, NULL },
|
|
||||||
{ 0, "dacp.nowplaying", dacp_propget_nowplaying, NULL },
|
|
||||||
{ 0, "dacp.playingtime", dacp_propget_playingtime, dacp_propset_playingtime },
|
|
||||||
{ 0, "dacp.volumecontrollable", dacp_propget_volumecontrollable, NULL },
|
|
||||||
{ 0, "dacp.availableshufflestates", dacp_propget_availableshufflestates, NULL },
|
|
||||||
{ 0, "dacp.availablerepeatstates", dacp_propget_availablerepeatstates, NULL },
|
|
||||||
{ 0, "dacp.shufflestate", dacp_propget_shufflestate, dacp_propset_shufflestate },
|
|
||||||
{ 0, "dacp.repeatstate", dacp_propget_repeatstate, dacp_propset_repeatstate },
|
|
||||||
{ 0, "dacp.userrating", NULL, dacp_propset_userrating },
|
|
||||||
|
|
||||||
{ 0, NULL, NULL, NULL }
|
/* gperf static hash, dacp_prop.gperf */
|
||||||
};
|
#include "dacp_prop_hash.c"
|
||||||
|
|
||||||
|
|
||||||
/* Play status update */
|
/* Play status update */
|
||||||
@ -146,9 +132,6 @@ static int current_rev;
|
|||||||
/* Play status update requests */
|
/* Play status update requests */
|
||||||
static struct dacp_update_request *update_requests;
|
static struct dacp_update_request *update_requests;
|
||||||
|
|
||||||
/* Properties */
|
|
||||||
static avl_tree_t *dacp_props_hash;
|
|
||||||
|
|
||||||
/* Seek timer */
|
/* Seek timer */
|
||||||
static struct event seek_timer;
|
static struct event seek_timer;
|
||||||
static int seek_target;
|
static int seek_target;
|
||||||
@ -394,95 +377,6 @@ update_fail_cb(struct evhttp_connection *evcon, void *arg)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Properties helpers */
|
|
||||||
static int
|
|
||||||
dacp_prop_map_compare(const void *aa, const void *bb)
|
|
||||||
{
|
|
||||||
struct dacp_prop_map *a = (struct dacp_prop_map *)aa;
|
|
||||||
struct dacp_prop_map *b = (struct dacp_prop_map *)bb;
|
|
||||||
|
|
||||||
if (a->hash < b->hash)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
if (a->hash > b->hash)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct dacp_prop_map *
|
|
||||||
dacp_find_prop(uint32_t hash)
|
|
||||||
{
|
|
||||||
struct dacp_prop_map dpm;
|
|
||||||
avl_node_t *node;
|
|
||||||
|
|
||||||
dpm.hash = hash;
|
|
||||||
|
|
||||||
node = avl_search(dacp_props_hash, &dpm);
|
|
||||||
if (!node)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
return (struct dacp_prop_map *)node->item;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
parse_properties(struct evhttp_request *req, char *tag, const char *param, uint32_t **out_prop, int *out_nprop)
|
|
||||||
{
|
|
||||||
char *ptr;
|
|
||||||
char *prop;
|
|
||||||
char *propstr;
|
|
||||||
uint32_t *hashes;
|
|
||||||
int nprop;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
*out_nprop = -1;
|
|
||||||
|
|
||||||
propstr = strdup(param);
|
|
||||||
if (!propstr)
|
|
||||||
{
|
|
||||||
DPRINTF(E_LOG, L_DACP, "Could not duplicate properties parameter; out of memory\n");
|
|
||||||
|
|
||||||
dmap_send_error(req, tag, "Out of memory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
nprop = 1;
|
|
||||||
ptr = propstr;
|
|
||||||
while ((ptr = strchr(ptr + 1, ',')))
|
|
||||||
nprop++;
|
|
||||||
|
|
||||||
DPRINTF(E_DBG, L_DACP, "Asking for %d properties\n", nprop);
|
|
||||||
|
|
||||||
hashes = (uint32_t *)malloc((nprop + 1) * sizeof(uint32_t));
|
|
||||||
if (!hashes)
|
|
||||||
{
|
|
||||||
DPRINTF(E_LOG, L_DACP, "Could not allocate properties array; out of memory\n");
|
|
||||||
|
|
||||||
dmap_send_error(req, tag, "Out of memory");
|
|
||||||
|
|
||||||
free(propstr);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
memset(hashes, 0, (nprop + 1) * sizeof(uint32_t));
|
|
||||||
|
|
||||||
prop = strtok_r(propstr, ",", &ptr);
|
|
||||||
for (i = 0; i < nprop; i++)
|
|
||||||
{
|
|
||||||
hashes[i] = djb_hash(prop, strlen(prop));
|
|
||||||
|
|
||||||
prop = strtok_r(NULL, ",", &ptr);
|
|
||||||
if (!prop)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
DPRINTF(E_DBG, L_DACP, "Found %d properties\n", nprop);
|
|
||||||
|
|
||||||
*out_nprop = nprop;
|
|
||||||
*out_prop = hashes;
|
|
||||||
|
|
||||||
free(propstr);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Properties getters */
|
/* Properties getters */
|
||||||
static void
|
static void
|
||||||
dacp_propget_volume(struct evbuffer *evbuf, struct player_status *status, struct media_file_info *mfi)
|
dacp_propget_volume(struct evbuffer *evbuf, struct player_status *status, struct media_file_info *mfi)
|
||||||
@ -1306,13 +1200,13 @@ dacp_reply_getproperty(struct evhttp_request *req, struct evbuffer *evbuf, char
|
|||||||
{
|
{
|
||||||
struct player_status status;
|
struct player_status status;
|
||||||
struct daap_session *s;
|
struct daap_session *s;
|
||||||
struct dacp_prop_map *dpm;
|
const struct dacp_prop_map *dpm;
|
||||||
struct media_file_info *mfi;
|
struct media_file_info *mfi;
|
||||||
struct evbuffer *proplist;
|
struct evbuffer *proplist;
|
||||||
const char *param;
|
const char *param;
|
||||||
uint32_t *prop;
|
char *ptr;
|
||||||
int nprop;
|
char *prop;
|
||||||
int i;
|
char *propstr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
s = daap_session_find(req, query, evbuf);
|
s = daap_session_find(req, query, evbuf);
|
||||||
@ -1328,11 +1222,12 @@ dacp_reply_getproperty(struct evhttp_request *req, struct evbuffer *evbuf, char
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
parse_properties(req, "cmgt", param, &prop, &nprop);
|
propstr = strdup(param);
|
||||||
if (nprop < 0)
|
if (!propstr)
|
||||||
{
|
{
|
||||||
DPRINTF(E_LOG, L_DACP, "Failed to parse properties parameter in getproperty call\n");
|
DPRINTF(E_LOG, L_DACP, "Could not duplicate properties parameter; out of memory\n");
|
||||||
|
|
||||||
|
dmap_send_error(req, "cmgt", "Out of memory");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1342,7 +1237,7 @@ dacp_reply_getproperty(struct evhttp_request *req, struct evbuffer *evbuf, char
|
|||||||
DPRINTF(E_LOG, L_DACP, "Could not allocate evbuffer for properties list\n");
|
DPRINTF(E_LOG, L_DACP, "Could not allocate evbuffer for properties list\n");
|
||||||
|
|
||||||
dmap_send_error(req, "cmgt", "Out of memory");
|
dmap_send_error(req, "cmgt", "Out of memory");
|
||||||
goto out_free_prop;
|
goto out_free_propstr;
|
||||||
}
|
}
|
||||||
|
|
||||||
player_get_status(&status);
|
player_get_status(&status);
|
||||||
@ -1361,27 +1256,29 @@ dacp_reply_getproperty(struct evhttp_request *req, struct evbuffer *evbuf, char
|
|||||||
else
|
else
|
||||||
mfi = NULL;
|
mfi = NULL;
|
||||||
|
|
||||||
for (i = 0; i < nprop; i++)
|
prop = strtok_r(propstr, ",", &ptr);
|
||||||
|
while (prop)
|
||||||
{
|
{
|
||||||
dpm = dacp_find_prop(prop[i]);
|
dpm = dacp_find_prop(prop, strlen(prop));
|
||||||
if (!dpm)
|
if (!dpm)
|
||||||
{
|
{
|
||||||
DPRINTF(E_LOG, L_DACP, "Could not find requested property (%d)\n", i + 1);
|
DPRINTF(E_LOG, L_DACP, "Could not find requested property '%s'\n", prop);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dpm->propget)
|
if (dpm->propget)
|
||||||
dpm->propget(proplist, &status, mfi);
|
dpm->propget(proplist, &status, mfi);
|
||||||
else
|
else
|
||||||
DPRINTF(E_WARN, L_DACP, "No getter method for DACP property %s\n", dpm->desc);
|
DPRINTF(E_WARN, L_DACP, "No getter method for DACP property %s\n", prop);
|
||||||
|
|
||||||
|
prop = strtok_r(NULL, ",", &ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
free(propstr);
|
||||||
|
|
||||||
if (mfi)
|
if (mfi)
|
||||||
free_mfi(mfi, 0);
|
free_mfi(mfi, 0);
|
||||||
|
|
||||||
if (nprop > 0)
|
|
||||||
free(prop);
|
|
||||||
|
|
||||||
dmap_add_container(evbuf, "cmgt", 12 + EVBUFFER_LENGTH(proplist)); /* 8 + len */
|
dmap_add_container(evbuf, "cmgt", 12 + EVBUFFER_LENGTH(proplist)); /* 8 + len */
|
||||||
dmap_add_int(evbuf, "mstt", 200); /* 12 */
|
dmap_add_int(evbuf, "mstt", 200); /* 12 */
|
||||||
|
|
||||||
@ -1402,18 +1299,16 @@ dacp_reply_getproperty(struct evhttp_request *req, struct evbuffer *evbuf, char
|
|||||||
out_free_proplist:
|
out_free_proplist:
|
||||||
evbuffer_free(proplist);
|
evbuffer_free(proplist);
|
||||||
|
|
||||||
out_free_prop:
|
out_free_propstr:
|
||||||
if (nprop > 0)
|
free(propstr);
|
||||||
free(prop);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
dacp_reply_setproperty(struct evhttp_request *req, struct evbuffer *evbuf, char **uri, struct evkeyvalq *query)
|
dacp_reply_setproperty(struct evhttp_request *req, struct evbuffer *evbuf, char **uri, struct evkeyvalq *query)
|
||||||
{
|
{
|
||||||
struct daap_session *s;
|
struct daap_session *s;
|
||||||
struct dacp_prop_map *dpm;
|
const struct dacp_prop_map *dpm;
|
||||||
struct evkeyval *param;
|
struct evkeyval *param;
|
||||||
uint32_t hash;
|
|
||||||
|
|
||||||
s = daap_session_find(req, query, evbuf);
|
s = daap_session_find(req, query, evbuf);
|
||||||
if (!s)
|
if (!s)
|
||||||
@ -1430,9 +1325,7 @@ dacp_reply_setproperty(struct evhttp_request *req, struct evbuffer *evbuf, char
|
|||||||
|
|
||||||
TAILQ_FOREACH(param, query, next)
|
TAILQ_FOREACH(param, query, next)
|
||||||
{
|
{
|
||||||
hash = djb_hash(param->key, strlen(param->key));
|
dpm = dacp_find_prop(param->key, strlen(param->key));
|
||||||
|
|
||||||
dpm = dacp_find_prop(hash);
|
|
||||||
|
|
||||||
if (!dpm)
|
if (!dpm)
|
||||||
{
|
{
|
||||||
@ -1795,8 +1688,6 @@ int
|
|||||||
dacp_init(void)
|
dacp_init(void)
|
||||||
{
|
{
|
||||||
char buf[64];
|
char buf[64];
|
||||||
avl_node_t *node;
|
|
||||||
struct dacp_prop_map *dpm;
|
|
||||||
int i;
|
int i;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
@ -1837,38 +1728,6 @@ dacp_init(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dacp_props_hash = avl_alloc_tree(dacp_prop_map_compare, NULL);
|
|
||||||
if (!dacp_props_hash)
|
|
||||||
{
|
|
||||||
DPRINTF(E_FATAL, L_DACP, "DACP init could not allocate DACP props AVL tree\n");
|
|
||||||
|
|
||||||
goto dacp_avl_alloc_fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; dacp_props[i].desc; i++)
|
|
||||||
{
|
|
||||||
dacp_props[i].hash = djb_hash(dacp_props[i].desc, strlen(dacp_props[i].desc));
|
|
||||||
|
|
||||||
node = avl_insert(dacp_props_hash, &dacp_props[i]);
|
|
||||||
if (!node)
|
|
||||||
{
|
|
||||||
if (errno != EEXIST)
|
|
||||||
DPRINTF(E_FATAL, L_DACP, "DACP init failed; AVL insert error: %s\n", strerror(errno));
|
|
||||||
else
|
|
||||||
{
|
|
||||||
node = avl_search(dacp_props_hash, &dacp_props[i]);
|
|
||||||
dpm = node->item;
|
|
||||||
|
|
||||||
DPRINTF(E_FATAL, L_DACP, "DACP init failed; WARNING: duplicate hash key\n");
|
|
||||||
DPRINTF(E_FATAL, L_DACP, "Hash %x, string %s\n", dacp_props[i].hash, dacp_props[i].desc);
|
|
||||||
|
|
||||||
DPRINTF(E_FATAL, L_DACP, "Hash %x, string %s\n", dpm->hash, dpm->desc);
|
|
||||||
}
|
|
||||||
|
|
||||||
goto dacp_avl_insert_fail;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef USE_EVENTFD
|
#ifdef USE_EVENTFD
|
||||||
event_set(&updateev, update_efd, EV_READ, playstatusupdate_cb, NULL);
|
event_set(&updateev, update_efd, EV_READ, playstatusupdate_cb, NULL);
|
||||||
#else
|
#else
|
||||||
@ -1881,11 +1740,6 @@ dacp_init(void)
|
|||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
dacp_avl_insert_fail:
|
|
||||||
avl_free_tree(dacp_props_hash);
|
|
||||||
dacp_avl_alloc_fail:
|
|
||||||
for (i = 0; dacp_handlers[i].handler; i++)
|
|
||||||
regfree(&dacp_handlers[i].preg);
|
|
||||||
regexp_fail:
|
regexp_fail:
|
||||||
#ifdef USE_EVENTFD
|
#ifdef USE_EVENTFD
|
||||||
close(update_efd);
|
close(update_efd);
|
||||||
@ -1922,8 +1776,6 @@ dacp_deinit(void)
|
|||||||
|
|
||||||
event_del(&updateev);
|
event_del(&updateev);
|
||||||
|
|
||||||
avl_free_tree(dacp_props_hash);
|
|
||||||
|
|
||||||
#ifdef USE_EVENTFD
|
#ifdef USE_EVENTFD
|
||||||
close(update_efd);
|
close(update_efd);
|
||||||
#else
|
#else
|
||||||
|
Loading…
x
Reference in New Issue
Block a user