Merge pull request #1398 from owntone/coverity_issues

Fixes for misc minor issues reported by Coverity
This commit is contained in:
ejurgensen 2022-01-21 19:03:31 +01:00 committed by GitHub
commit 119eb46daa
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
32 changed files with 344 additions and 239 deletions

View File

@ -851,7 +851,13 @@ parent_dir_image_find(char *out_path, size_t len, const char *dir)
DPRINTF(E_LOG, L_ART, "Could not find parent dir name (%s)\n", path); DPRINTF(E_LOG, L_ART, "Could not find parent dir name (%s)\n", path);
return -1; return -1;
} }
strcpy(parentdir, ptr + 1);
ret = snprintf(parentdir, sizeof(parentdir), "%s", ptr + 1);
if ((ret < 0) || (ret >= sizeof(parentdir)))
{
DPRINTF(E_LOG, L_ART, "Impossible error occured in parent_dir_image_find(), cause was: %s\n", ptr + 1);
return -1;
}
path_len = strlen(path); path_len = strlen(path);
nextensions = ARRAY_SIZE(cover_extension); nextensions = ARRAY_SIZE(cover_extension);

View File

@ -88,9 +88,9 @@ command_cb_sync(struct commands_base *cmdbase, struct command *cmd)
// Command execution is waiting for pending events before returning to the caller // Command execution is waiting for pending events before returning to the caller
cmdbase->current_cmd = cmd; cmdbase->current_cmd = cmd;
cmd->pending = cmd->ret; cmd->pending = cmd->ret;
return;
} }
else
{
// Command execution finished, execute the bottom half function // Command execution finished, execute the bottom half function
if (cmd->ret == 0 && cmd->func_bh) if (cmd->ret == 0 && cmd->func_bh)
cmd->func_bh(cmd->arg, &cmd->ret); cmd->func_bh(cmd->arg, &cmd->ret);
@ -104,7 +104,6 @@ command_cb_sync(struct commands_base *cmdbase, struct command *cmd)
// Note if cmd->func was cmdloop_exit then cmdbase may be invalid now, // Note if cmd->func was cmdloop_exit then cmdbase may be invalid now,
// because commands_base_destroy() may have freed it // because commands_base_destroy() may have freed it
} }
}
/* /*
* Event callback function * Event callback function
@ -357,7 +356,7 @@ commands_exec_async(struct commands_base *cmdbase, command_function func, void *
struct command *cmd; struct command *cmd;
int ret; int ret;
cmd = calloc(1, sizeof(struct command)); CHECK_NULL(L_MAIN, cmd = calloc(1, sizeof(struct command)));
cmd->func = func; cmd->func = func;
cmd->func_bh = NULL; cmd->func_bh = NULL;
cmd->arg = arg; cmd->arg = arg;

View File

@ -775,6 +775,20 @@ free_di(struct directory_info *di, int content_only)
memset(di, 0, sizeof(struct directory_info)); memset(di, 0, sizeof(struct directory_info));
} }
void
free_wi(struct watch_info *wi, int content_only)
{
if (!wi)
return;
free(wi->path);
if (!content_only)
free(wi);
else
memset(wi, 0, sizeof(struct watch_info));
}
void void
free_query_params(struct query_params *qp, int content_only) free_query_params(struct query_params *qp, int content_only)
{ {
@ -2682,8 +2696,8 @@ db_query_fetch_string_sort(char **string, char **sortstring, struct query_params
int int
db_files_get_count(uint32_t *nitems, uint32_t *nstreams, const char *filter) db_files_get_count(uint32_t *nitems, uint32_t *nstreams, const char *filter)
{ {
sqlite3_stmt *stmt; sqlite3_stmt *stmt = NULL;
char *query; char *query = NULL;
int ret; int ret;
if (!filter && !nstreams) if (!filter && !nstreams)
@ -2698,17 +2712,16 @@ db_files_get_count(uint32_t *nitems, uint32_t *nstreams, const char *filter)
if (!query) if (!query)
{ {
DPRINTF(E_LOG, L_DB, "Out of memory for query string\n"); DPRINTF(E_LOG, L_DB, "Out of memory for query string\n");
return -1; goto error;
} }
DPRINTF(E_DBG, L_DB, "Running query '%s'\n", query); DPRINTF(E_DBG, L_DB, "Running query '%s'\n", query);
ret = db_blocking_prepare_v2(query, -1, &stmt, NULL); ret = db_blocking_prepare_v2(query, -1, &stmt, NULL);
sqlite3_free(query);
if (ret != SQLITE_OK) if (ret != SQLITE_OK)
{ {
DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl));
return -1; goto error;
} }
ret = db_blocking_step(stmt); ret = db_blocking_step(stmt);
@ -2719,8 +2732,7 @@ db_files_get_count(uint32_t *nitems, uint32_t *nstreams, const char *filter)
else else
DPRINTF(E_LOG, L_DB, "Could not step: %s (%s)\n", sqlite3_errmsg(hdl), query); DPRINTF(E_LOG, L_DB, "Could not step: %s (%s)\n", sqlite3_errmsg(hdl), query);
sqlite3_finalize(stmt); goto error;
return -1;
} }
if (nitems) if (nitems)
@ -2733,9 +2745,16 @@ db_files_get_count(uint32_t *nitems, uint32_t *nstreams, const char *filter)
; /* EMPTY */ ; /* EMPTY */
#endif #endif
sqlite3_free(query);
sqlite3_finalize(stmt); sqlite3_finalize(stmt);
return 0; return 0;
error:
if (query)
sqlite3_free(query);
if (stmt)
sqlite3_finalize(stmt);
return -1;
} }
static void static void
@ -6059,10 +6078,10 @@ queue_reshuffle(uint32_t item_id, int queue_version)
int pos; int pos;
uint32_t count; uint32_t count;
struct db_queue_item queue_item; struct db_queue_item queue_item;
int *shuffle_pos; int *shuffle_pos = NULL;
int len; int len;
int i; int i;
struct query_params qp; struct query_params qp = { 0 };
int ret; int ret;
DPRINTF(E_DBG, L_DB, "Reshuffle queue after item with item-id: %d\n", item_id); DPRINTF(E_DBG, L_DB, "Reshuffle queue after item with item-id: %d\n", item_id);
@ -6071,43 +6090,39 @@ queue_reshuffle(uint32_t item_id, int queue_version)
query = sqlite3_mprintf("UPDATE queue SET shuffle_pos = pos, queue_version = %d;", queue_version); query = sqlite3_mprintf("UPDATE queue SET shuffle_pos = pos, queue_version = %d;", queue_version);
ret = db_query_run(query, 1, 0); ret = db_query_run(query, 1, 0);
if (ret < 0) if (ret < 0)
return -1; goto error;
pos = 0; pos = 0;
if (item_id > 0) if (item_id > 0)
{ {
pos = db_queue_get_pos(item_id, 0); pos = db_queue_get_pos(item_id, 0);
if (pos < 0) if (pos < 0)
return -1; goto error;
pos++; // Do not reshuffle the base item pos++; // Do not reshuffle the base item
} }
ret = db_queue_get_count(&count); ret = db_queue_get_count(&count);
if (ret < 0) if (ret < 0)
return -1; goto error;
len = count - pos; len = count - pos;
DPRINTF(E_DBG, L_DB, "Reshuffle %d items off %" PRIu32 " total items, starting from pos %d\n", len, count, pos); DPRINTF(E_DBG, L_DB, "Reshuffle %d items off %" PRIu32 " total items, starting from pos %d\n", len, count, pos);
shuffle_pos = malloc(len * sizeof(int)); CHECK_NULL(L_DB, shuffle_pos = malloc(len * sizeof(int)));
for (i = 0; i < len; i++) for (i = 0; i < len; i++)
{ {
shuffle_pos[i] = i + pos; shuffle_pos[i] = i + pos;
} }
shuffle_int(&shuffle_rng, shuffle_pos, len); rng_shuffle_int(&shuffle_rng, shuffle_pos, len);
memset(&qp, 0, sizeof(struct query_params));
qp.filter = sqlite3_mprintf("pos >= %d", pos); qp.filter = sqlite3_mprintf("pos >= %d", pos);
ret = queue_enum_start(&qp); ret = queue_enum_start(&qp);
if (ret < 0) if (ret < 0)
{ goto error;
sqlite3_free(qp.filter);
return -1;
}
i = 0; i = 0;
while ((ret = queue_enum_fetch(&qp, &queue_item, 0)) == 0 && (queue_item.id > 0) && (i < len)) while ((ret = queue_enum_fetch(&qp, &queue_item, 0)) == 0 && (queue_item.id > 0) && (i < len))
@ -6124,12 +6139,18 @@ queue_reshuffle(uint32_t item_id, int queue_version)
} }
db_query_end(&qp); db_query_end(&qp);
sqlite3_free(qp.filter);
if (ret < 0) if (ret < 0)
return -1; goto error;
sqlite3_free(qp.filter);
free(shuffle_pos);
return 0; return 0;
error:
sqlite3_free(qp.filter);
free(shuffle_pos);
return -1;
} }
/* /*
@ -6216,7 +6237,7 @@ db_watch_delete_bywd(uint32_t wd)
} }
int int
db_watch_delete_bypath(char *path) db_watch_delete_bypath(const char *path)
{ {
#define Q_TMPL "DELETE FROM inotify WHERE path = '%q';" #define Q_TMPL "DELETE FROM inotify WHERE path = '%q';"
char *query; char *query;
@ -6228,7 +6249,7 @@ db_watch_delete_bypath(char *path)
} }
int int
db_watch_delete_bymatch(char *path) db_watch_delete_bymatch(const char *path)
{ {
#define Q_TMPL "DELETE FROM inotify WHERE path LIKE '%q/%%';" #define Q_TMPL "DELETE FROM inotify WHERE path LIKE '%q/%%';"
char *query; char *query;
@ -6316,12 +6337,12 @@ db_watch_get_byquery(struct watch_info *wi, char *query)
} }
int int
db_watch_get_bywd(struct watch_info *wi) db_watch_get_bywd(struct watch_info *wi, int wd)
{ {
#define Q_TMPL "SELECT * FROM inotify WHERE wd = %d;" #define Q_TMPL "SELECT * FROM inotify WHERE wd = %d;"
char *query; char *query;
query = sqlite3_mprintf(Q_TMPL, wi->wd); query = sqlite3_mprintf(Q_TMPL, wd);
if (!query) if (!query)
{ {
DPRINTF(E_LOG, L_DB, "Out of memory for query string\n"); DPRINTF(E_LOG, L_DB, "Out of memory for query string\n");
@ -6333,12 +6354,12 @@ db_watch_get_bywd(struct watch_info *wi)
} }
int int
db_watch_get_bypath(struct watch_info *wi) db_watch_get_bypath(struct watch_info *wi, const char *path)
{ {
#define Q_TMPL "SELECT * FROM inotify WHERE path = '%q';" #define Q_TMPL "SELECT * FROM inotify WHERE path = '%q';"
char *query; char *query;
query = sqlite3_mprintf(Q_TMPL, wi->path); query = sqlite3_mprintf(Q_TMPL, path);
if (!query) if (!query)
{ {
DPRINTF(E_LOG, L_DB, "Out of memory for query string\n"); DPRINTF(E_LOG, L_DB, "Out of memory for query string\n");
@ -6350,7 +6371,7 @@ db_watch_get_bypath(struct watch_info *wi)
} }
void void
db_watch_mark_bypath(char *path, enum strip_type strip, uint32_t cookie) db_watch_mark_bypath(const char *path, enum strip_type strip, uint32_t cookie)
{ {
#define Q_TMPL "UPDATE inotify SET path = substr(path, %d), cookie = %" PRIi64 " WHERE path = '%q';" #define Q_TMPL "UPDATE inotify SET path = substr(path, %d), cookie = %" PRIi64 " WHERE path = '%q';"
char *query; char *query;
@ -6368,7 +6389,7 @@ db_watch_mark_bypath(char *path, enum strip_type strip, uint32_t cookie)
} }
void void
db_watch_mark_bymatch(char *path, enum strip_type strip, uint32_t cookie) db_watch_mark_bymatch(const char *path, enum strip_type strip, uint32_t cookie)
{ {
#define Q_TMPL "UPDATE inotify SET path = substr(path, %d), cookie = %" PRIi64 " WHERE path LIKE '%q/%%';" #define Q_TMPL "UPDATE inotify SET path = substr(path, %d), cookie = %" PRIi64 " WHERE path LIKE '%q/%%';"
char *query; char *query;
@ -6386,7 +6407,7 @@ db_watch_mark_bymatch(char *path, enum strip_type strip, uint32_t cookie)
} }
void void
db_watch_move_bycookie(uint32_t cookie, char *path) db_watch_move_bycookie(uint32_t cookie, const char *path)
{ {
#define Q_TMPL "UPDATE inotify SET path = '%q' || path, cookie = 0 WHERE cookie = %" PRIi64 ";" #define Q_TMPL "UPDATE inotify SET path = '%q' || path, cookie = 0 WHERE cookie = %" PRIi64 ";"
char *query; char *query;
@ -6611,8 +6632,6 @@ db_pragma_get_cache_size()
if (ret != SQLITE_OK) if (ret != SQLITE_OK)
{ {
DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl));
sqlite3_free(query);
return 0; return 0;
} }
@ -6620,13 +6639,11 @@ db_pragma_get_cache_size()
if (ret == SQLITE_DONE) if (ret == SQLITE_DONE)
{ {
DPRINTF(E_DBG, L_DB, "End of query results\n"); DPRINTF(E_DBG, L_DB, "End of query results\n");
sqlite3_free(query);
return 0; return 0;
} }
else if (ret != SQLITE_ROW) else if (ret != SQLITE_ROW)
{ {
DPRINTF(E_LOG, L_DB, "Could not step: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not step: %s\n", sqlite3_errmsg(hdl));
sqlite3_free(query);
return -1; return -1;
} }
@ -6717,8 +6734,6 @@ db_pragma_get_synchronous()
if (ret != SQLITE_OK) if (ret != SQLITE_OK)
{ {
DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl));
sqlite3_free(query);
return 0; return 0;
} }
@ -6726,13 +6741,11 @@ db_pragma_get_synchronous()
if (ret == SQLITE_DONE) if (ret == SQLITE_DONE)
{ {
DPRINTF(E_DBG, L_DB, "End of query results\n"); DPRINTF(E_DBG, L_DB, "End of query results\n");
sqlite3_free(query);
return 0; return 0;
} }
else if (ret != SQLITE_ROW) else if (ret != SQLITE_ROW)
{ {
DPRINTF(E_LOG, L_DB, "Could not step: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not step: %s\n", sqlite3_errmsg(hdl));
sqlite3_free(query);
return -1; return -1;
} }
@ -6781,8 +6794,6 @@ db_pragma_get_mmap_size()
if (ret != SQLITE_OK) if (ret != SQLITE_OK)
{ {
DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not prepare statement: %s\n", sqlite3_errmsg(hdl));
sqlite3_free(query);
return 0; return 0;
} }
@ -6790,13 +6801,11 @@ db_pragma_get_mmap_size()
if (ret == SQLITE_DONE) if (ret == SQLITE_DONE)
{ {
DPRINTF(E_DBG, L_DB, "End of query results\n"); DPRINTF(E_DBG, L_DB, "End of query results\n");
sqlite3_free(query);
return 0; return 0;
} }
else if (ret != SQLITE_ROW) else if (ret != SQLITE_ROW)
{ {
DPRINTF(E_LOG, L_DB, "Could not step: %s\n", sqlite3_errmsg(hdl)); DPRINTF(E_LOG, L_DB, "Could not step: %s\n", sqlite3_errmsg(hdl));
sqlite3_free(query);
return -1; return -1;
} }

View File

@ -573,6 +573,9 @@ free_pli(struct playlist_info *pli, int content_only);
void void
free_di(struct directory_info *di, int content_only); free_di(struct directory_info *di, int content_only);
void
free_wi(struct watch_info *wi, int content_only);
void void
free_query_params(struct query_params *qp, int content_only); free_query_params(struct query_params *qp, int content_only);
@ -960,28 +963,28 @@ int
db_watch_delete_bywd(uint32_t wd); db_watch_delete_bywd(uint32_t wd);
int int
db_watch_delete_bypath(char *path); db_watch_delete_bypath(const char *path);
int int
db_watch_delete_bymatch(char *path); db_watch_delete_bymatch(const char *path);
int int
db_watch_delete_bycookie(uint32_t cookie); db_watch_delete_bycookie(uint32_t cookie);
int int
db_watch_get_bywd(struct watch_info *wi); db_watch_get_bywd(struct watch_info *wi, int wd);
int int
db_watch_get_bypath(struct watch_info *wi); db_watch_get_bypath(struct watch_info *wi, const char *path);
void void
db_watch_mark_bypath(char *path, enum strip_type strip, uint32_t cookie); db_watch_mark_bypath(const char *path, enum strip_type strip, uint32_t cookie);
void void
db_watch_mark_bymatch(char *path, enum strip_type strip, uint32_t cookie); db_watch_mark_bymatch(const char *path, enum strip_type strip, uint32_t cookie);
void void
db_watch_move_bycookie(uint32_t cookie, char *path); db_watch_move_bycookie(uint32_t cookie, const char *path);
int int
db_watch_cookie_known(uint32_t cookie); db_watch_cookie_known(uint32_t cookie);

View File

@ -1702,13 +1702,23 @@ bind_socket_ai(int family, struct addrinfo *ai, int reuse)
} }
#endif #endif
if (family == AF_INET6) if (family == AF_INET6) {
setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on)); r = setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on));
if (r == -1) {
event_warn("IPV6_V6ONLY");
}
}
r = setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, (void *)&on, sizeof(on));
if (r == -1) {
event_warn("SO_KEEPALIVE");
}
setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, (void *)&on, sizeof(on));
if (reuse) { if (reuse) {
setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, r = setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (void *)&on, sizeof(on));
(void *)&on, sizeof(on)); if (r == -1) {
event_warn("SO_REUSEADDR");
}
} }
if (ai != NULL) { if (ai != NULL) {

View File

@ -1310,9 +1310,10 @@ httpd_stream_file(struct evhttp_request *req, int id)
if (!transcode) if (!transcode)
{ {
/* Hint the OS */ /* Hint the OS */
posix_fadvise(st->fd, st->start_offset, st->stream_size, POSIX_FADV_WILLNEED); if ( (ret = posix_fadvise(st->fd, st->start_offset, st->stream_size, POSIX_FADV_WILLNEED)) != 0 ||
posix_fadvise(st->fd, st->start_offset, st->stream_size, POSIX_FADV_SEQUENTIAL); (ret = posix_fadvise(st->fd, st->start_offset, st->stream_size, POSIX_FADV_SEQUENTIAL)) != 0 ||
posix_fadvise(st->fd, st->start_offset, st->stream_size, POSIX_FADV_NOREUSE); (ret = posix_fadvise(st->fd, st->start_offset, st->stream_size, POSIX_FADV_NOREUSE)) != 0 )
DPRINTF(E_DBG, L_HTTPD, "posix_fadvise() failed with error %d\n", ret);
} }
#endif #endif
@ -1353,6 +1354,10 @@ httpd_gzip_deflate(struct evbuffer *in)
strm.zfree = Z_NULL; strm.zfree = Z_NULL;
strm.opaque = Z_NULL; strm.opaque = Z_NULL;
// Just to keep Coverity from complaining about uninitialized values
strm.total_in = 0;
strm.total_out = 0;
// Set up a gzip stream (the "+ 16" in 15 + 16), instead of a zlib stream (default) // Set up a gzip stream (the "+ 16" in 15 + 16), instead of a zlib stream (default)
ret = deflateInit2(&strm, Z_DEFAULT_COMPRESSION, Z_DEFLATED, 15 + 16, 8, Z_DEFAULT_STRATEGY); ret = deflateInit2(&strm, Z_DEFAULT_COMPRESSION, Z_DEFLATED, 15 + 16, 8, Z_DEFAULT_STRATEGY);
if (ret != Z_OK) if (ret != Z_OK)

View File

@ -136,9 +136,6 @@ static struct timeval daap_update_refresh_tv = { DAAP_UPDATE_REFRESH, 0 };
static void static void
daap_session_free(struct daap_session *s) daap_session_free(struct daap_session *s)
{ {
if (!s)
return;
free(s); free(s);
} }
@ -637,7 +634,7 @@ parse_meta(const struct dmap_field ***out_meta, const char *param)
CHECK_NULL(L_DAAP, meta = calloc(nmeta, sizeof(const struct dmap_field *))); CHECK_NULL(L_DAAP, meta = calloc(nmeta, sizeof(const struct dmap_field *)));
field = strtok_r(metastr, ",", &ptr); field = strtok_r(metastr, ",", &ptr);
for (i = 0; i < nmeta; i++) for (i = 0; field != NULL && i < nmeta; i++)
{ {
for (n = 0; (n < i) && (strcmp(field, meta[n]->desc) != 0); n++); for (n = 0; (n < i) && (strcmp(field, meta[n]->desc) != 0); n++);
@ -662,8 +659,6 @@ parse_meta(const struct dmap_field ***out_meta, const char *param)
} }
field = strtok_r(NULL, ",", &ptr); field = strtok_r(NULL, ",", &ptr);
if (!field)
break;
} }
free(metastr); free(metastr);
@ -1185,14 +1180,14 @@ daap_reply_songlist_generic(struct httpd_request *hreq, int playlist)
struct evbuffer *songlist; struct evbuffer *songlist;
struct evkeyvalq *headers; struct evkeyvalq *headers;
struct daap_session *s; struct daap_session *s;
const struct dmap_field **meta; const struct dmap_field **meta = NULL;
struct sort_ctx *sctx; struct sort_ctx *sctx;
const char *param; const char *param;
const char *client_codecs; const char *client_codecs;
const char *tag; const char *tag;
char *last_codectype; char *last_codectype;
size_t len; size_t len;
int nmeta; int nmeta = 0;
int sort_headers; int sort_headers;
int nsongs; int nsongs;
int transcode; int transcode;
@ -1246,18 +1241,12 @@ daap_reply_songlist_generic(struct httpd_request *hreq, int playlist)
goto error; goto error;
} }
} }
else
{
meta = NULL;
nmeta = 0;
}
ret = db_query_start(&qp); ret = db_query_start(&qp);
if (ret < 0) if (ret < 0)
{ {
DPRINTF(E_LOG, L_DAAP, "Could not start query\n"); DPRINTF(E_LOG, L_DAAP, "Could not start query\n");
free(meta);
dmap_error_make(hreq->reply, tag, "Could not start query"); dmap_error_make(hreq->reply, tag, "Could not start query");
goto error; goto error;
} }
@ -1320,7 +1309,6 @@ daap_reply_songlist_generic(struct httpd_request *hreq, int playlist)
DPRINTF(E_DBG, L_DAAP, "Done with song list, %d songs\n", nsongs); DPRINTF(E_DBG, L_DAAP, "Done with song list, %d songs\n", nsongs);
free(last_codectype); free(last_codectype);
free(meta);
db_query_end(&qp); db_query_end(&qp);
if (ret == -100) if (ret == -100)
@ -1361,6 +1349,7 @@ daap_reply_songlist_generic(struct httpd_request *hreq, int playlist)
CHECK_ERR(L_DAAP, evbuffer_add_buffer(hreq->reply, sctx->headerlist)); CHECK_ERR(L_DAAP, evbuffer_add_buffer(hreq->reply, sctx->headerlist));
} }
free(meta);
daap_sort_context_free(sctx); daap_sort_context_free(sctx);
evbuffer_free(song); evbuffer_free(song);
evbuffer_free(songlist); evbuffer_free(songlist);
@ -1369,6 +1358,7 @@ daap_reply_songlist_generic(struct httpd_request *hreq, int playlist)
return DAAP_REPLY_OK; return DAAP_REPLY_OK;
error: error:
free(meta);
daap_sort_context_free(sctx); daap_sort_context_free(sctx);
evbuffer_free(song); evbuffer_free(song);
evbuffer_free(songlist); evbuffer_free(songlist);
@ -1417,7 +1407,7 @@ daap_reply_playlists(struct httpd_request *hreq)
struct evbuffer *playlist; struct evbuffer *playlist;
const struct dmap_field_map *dfm; const struct dmap_field_map *dfm;
const struct dmap_field *df; const struct dmap_field *df;
const struct dmap_field **meta; const struct dmap_field **meta = NULL;
const char *param; const char *param;
char **strval; char **strval;
size_t len; size_t len;
@ -1473,7 +1463,6 @@ daap_reply_playlists(struct httpd_request *hreq)
{ {
DPRINTF(E_LOG, L_DAAP, "Could not start query\n"); DPRINTF(E_LOG, L_DAAP, "Could not start query\n");
free(meta);
dmap_error_make(hreq->reply, "aply", "Could not start query"); dmap_error_make(hreq->reply, "aply", "Could not start query");
goto error; goto error;
} }
@ -1585,7 +1574,6 @@ daap_reply_playlists(struct httpd_request *hreq)
} }
db_query_end(&qp); db_query_end(&qp);
free(meta);
DPRINTF(E_DBG, L_DAAP, "Done with playlist list, %d playlists\n", npls); DPRINTF(E_DBG, L_DAAP, "Done with playlist list, %d playlists\n", npls);
@ -1612,6 +1600,7 @@ daap_reply_playlists(struct httpd_request *hreq)
CHECK_ERR(L_DAAP, evbuffer_add_buffer(hreq->reply, playlistlist)); CHECK_ERR(L_DAAP, evbuffer_add_buffer(hreq->reply, playlistlist));
free(meta);
evbuffer_free(playlist); evbuffer_free(playlist);
evbuffer_free(playlistlist); evbuffer_free(playlistlist);
free_query_params(&qp, 1); free_query_params(&qp, 1);
@ -1619,6 +1608,7 @@ daap_reply_playlists(struct httpd_request *hreq)
return DAAP_REPLY_OK; return DAAP_REPLY_OK;
error: error:
free(meta);
evbuffer_free(playlist); evbuffer_free(playlist);
evbuffer_free(playlistlist); evbuffer_free(playlistlist);
free_query_params(&qp, 1); free_query_params(&qp, 1);
@ -1635,7 +1625,7 @@ daap_reply_groups(struct httpd_request *hreq)
struct evbuffer *grouplist; struct evbuffer *grouplist;
const struct dmap_field_map *dfm; const struct dmap_field_map *dfm;
const struct dmap_field *df; const struct dmap_field *df;
const struct dmap_field **meta; const struct dmap_field **meta = NULL;
struct sort_ctx *sctx; struct sort_ctx *sctx;
cfg_t *lib; cfg_t *lib;
const char *param; const char *param;
@ -1699,7 +1689,6 @@ daap_reply_groups(struct httpd_request *hreq)
{ {
DPRINTF(E_LOG, L_DAAP, "Could not start query\n"); DPRINTF(E_LOG, L_DAAP, "Could not start query\n");
free(meta);
dmap_error_make(hreq->reply, tag, "Could not start query"); dmap_error_make(hreq->reply, tag, "Could not start query");
goto error; goto error;
} }
@ -1787,7 +1776,6 @@ daap_reply_groups(struct httpd_request *hreq)
} }
db_query_end(&qp); db_query_end(&qp);
free(meta);
DPRINTF(E_DBG, L_DAAP, "Done with group list, %d groups\n", ngrp); DPRINTF(E_DBG, L_DAAP, "Done with group list, %d groups\n", ngrp);
@ -1829,6 +1817,7 @@ daap_reply_groups(struct httpd_request *hreq)
CHECK_ERR(L_DAAP, evbuffer_add_buffer(hreq->reply, sctx->headerlist)); CHECK_ERR(L_DAAP, evbuffer_add_buffer(hreq->reply, sctx->headerlist));
} }
free(meta);
daap_sort_context_free(sctx); daap_sort_context_free(sctx);
evbuffer_free(group); evbuffer_free(group);
evbuffer_free(grouplist); evbuffer_free(grouplist);
@ -1837,6 +1826,7 @@ daap_reply_groups(struct httpd_request *hreq)
return DAAP_REPLY_OK; return DAAP_REPLY_OK;
error: error:
free(meta);
daap_sort_context_free(sctx); daap_sort_context_free(sctx);
evbuffer_free(group); evbuffer_free(group);
evbuffer_free(grouplist); evbuffer_free(grouplist);

View File

@ -1410,7 +1410,12 @@ dacp_reply_play(struct httpd_request *hreq)
if (ret < 0) if (ret < 0)
return -1; return -1;
player_playback_start(); ret = player_playback_start();
if (ret < 0)
{
httpd_send_error(hreq->req, 500, "Internal Server Error");
return -1;
}
/* 204 No Content is the canonical reply */ /* 204 No Content is the canonical reply */
httpd_send_reply(hreq->req, HTTP_NOCONTENT, "No Content", hreq->reply, HTTPD_SEND_NO_GZIP); httpd_send_reply(hreq->req, HTTP_NOCONTENT, "No Content", hreq->reply, HTTPD_SEND_NO_GZIP);
@ -2731,10 +2736,12 @@ dacp_reply_mutetoggle(struct httpd_request *hreq)
} }
// We don't actually mute, because the player doesn't currently support unmuting // We don't actually mute, because the player doesn't currently support unmuting
if (speaker_info.selected) ret = speaker_info.selected ? player_speaker_disable(speaker_info.id) : player_speaker_enable(speaker_info.id);
player_speaker_disable(speaker_info.id); if (ret < 0)
else {
player_speaker_enable(speaker_info.id); httpd_send_error(hreq->req, 500, "Internal Server Error");
return -1;
}
/* 204 No Content is the canonical reply */ /* 204 No Content is the canonical reply */
httpd_send_reply(hreq->req, HTTP_NOCONTENT, "No Content", hreq->reply, HTTPD_SEND_NO_GZIP); httpd_send_reply(hreq->req, HTTP_NOCONTENT, "No Content", hreq->reply, HTTPD_SEND_NO_GZIP);

View File

@ -1829,7 +1829,7 @@ jsonapi_reply_outputs_set(struct httpd_request *hreq)
{ {
nspk = json_object_array_length(outputs); nspk = json_object_array_length(outputs);
ids = calloc((nspk + 1), sizeof(uint64_t)); CHECK_NULL(L_WEB, ids = calloc((nspk + 1), sizeof(uint64_t)));
ids[0] = nspk; ids[0] = nspk;
ret = 0; ret = 0;
@ -2502,8 +2502,8 @@ jsonapi_reply_queue_tracks_add(struct httpd_request *hreq)
const char *param_uris; const char *param_uris;
const char *param_expression; const char *param_expression;
const char *param; const char *param;
int pos = -1; int pos;
int limit = -1; int limit;
bool shuffle; bool shuffle;
int total_count = 0; int total_count = 0;
json_object *reply; json_object *reply;
@ -2522,6 +2522,8 @@ jsonapi_reply_queue_tracks_add(struct httpd_request *hreq)
DPRINTF(E_DBG, L_WEB, "Add tracks starting at position '%d\n", pos); DPRINTF(E_DBG, L_WEB, "Add tracks starting at position '%d\n", pos);
} }
else
pos = -1;
param_uris = evhttp_find_header(hreq->query, "uris"); param_uris = evhttp_find_header(hreq->query, "uris");
param_expression = evhttp_find_header(hreq->query, "expression"); param_expression = evhttp_find_header(hreq->query, "expression");
@ -2557,9 +2559,10 @@ jsonapi_reply_queue_tracks_add(struct httpd_request *hreq)
{ {
// This overrides the value specified in query // This overrides the value specified in query
param = evhttp_find_header(hreq->query, "limit"); param = evhttp_find_header(hreq->query, "limit");
if (param) if (param && safe_atoi32(param, &limit) == 0)
safe_atoi32(param, &limit);
ret = queue_tracks_add_byexpression(param_expression, pos, limit, &total_count); ret = queue_tracks_add_byexpression(param_expression, pos, limit, &total_count);
else
ret = queue_tracks_add_byexpression(param_expression, pos, -1, &total_count);
} }
if (ret == 0) if (ret == 0)
@ -2579,9 +2582,12 @@ jsonapi_reply_queue_tracks_add(struct httpd_request *hreq)
if (param && strcmp(param, "start") == 0) if (param && strcmp(param, "start") == 0)
{ {
if ((param = evhttp_find_header(hreq->query, "playback_from_position"))) if ((param = evhttp_find_header(hreq->query, "playback_from_position")))
play_item_at_position(param); ret = (play_item_at_position(param) == HTTP_NOCONTENT) ? 0 : -1;
else else
player_playback_start(); ret = player_playback_start();
if (ret < 0)
return HTTP_INTERNAL;
} }
return HTTP_OK; return HTTP_OK;
@ -2610,10 +2616,11 @@ update_pos(uint32_t item_id, const char *new, char shuffle)
} }
static inline void static inline void
update_str(char **str, const char *new) update_str(bool *is_changed, char **str, const char *new)
{ {
free(*str); free(*str);
*str = strdup(new); *str = strdup(new);
*is_changed = true;
} }
static int static int
@ -2629,11 +2636,19 @@ jsonapi_reply_queue_tracks_update(struct httpd_request *hreq)
player_get_status(&status); player_get_status(&status);
if (strcmp(hreq->uri_parsed->path_parts[3], "now_playing") != 0) if (strcmp(hreq->uri_parsed->path_parts[3], "now_playing") != 0)
safe_atou32(hreq->uri_parsed->path_parts[3], &item_id); {
ret = safe_atou32(hreq->uri_parsed->path_parts[3], &item_id);
if (ret < 0)
{
DPRINTF(E_LOG, L_WEB, "No valid item id given: '%s'\n", hreq->uri_parsed->path);
return HTTP_BADREQUEST;
}
}
else else
item_id = status.item_id; item_id = status.item_id;
if (!item_id || !(queue_item = db_queue_fetch_byitemid(item_id))) queue_item = db_queue_fetch_byitemid(item_id);
if (!queue_item)
{ {
DPRINTF(E_LOG, L_WEB, "No valid item id given, or now_playing given but not playing: '%s'\n", hreq->uri_parsed->path); DPRINTF(E_LOG, L_WEB, "No valid item id given, or now_playing given but not playing: '%s'\n", hreq->uri_parsed->path);
return HTTP_BADREQUEST; return HTTP_BADREQUEST;
@ -2643,20 +2658,20 @@ jsonapi_reply_queue_tracks_update(struct httpd_request *hreq)
is_changed = false; is_changed = false;
if ((param = evhttp_find_header(hreq->query, "new_position"))) if ((param = evhttp_find_header(hreq->query, "new_position")))
ret = update_pos(item_id, param, status.shuffle); ret = update_pos(item_id, param, status.shuffle);
if ((param = evhttp_find_header(hreq->query, "title")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "title")))
update_str(&queue_item->title, param); update_str(&is_changed, &queue_item->title, param);
if ((param = evhttp_find_header(hreq->query, "album")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "album")))
update_str(&queue_item->album, param); update_str(&is_changed, &queue_item->album, param);
if ((param = evhttp_find_header(hreq->query, "artist")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "artist")))
update_str(&queue_item->artist, param); update_str(&is_changed, &queue_item->artist, param);
if ((param = evhttp_find_header(hreq->query, "album_artist")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "album_artist")))
update_str(&queue_item->album_artist, param); update_str(&is_changed, &queue_item->album_artist, param);
if ((param = evhttp_find_header(hreq->query, "composer")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "composer")))
update_str(&queue_item->composer, param); update_str(&is_changed, &queue_item->composer, param);
if ((param = evhttp_find_header(hreq->query, "genre")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "genre")))
update_str(&queue_item->genre, param); update_str(&is_changed, &queue_item->genre, param);
if ((param = evhttp_find_header(hreq->query, "artwork_url")) && (is_changed = true)) if ((param = evhttp_find_header(hreq->query, "artwork_url")))
update_str(&queue_item->artwork_url, param); update_str(&is_changed, &queue_item->artwork_url, param);
if (ret != HTTP_OK) if (ret != HTTP_OK)
return ret; return ret;

View File

@ -645,6 +645,7 @@ streaming_init(void)
default: default:
DPRINTF(E_LOG, L_STREAMING, "Unsuppported streaming bit_rate=%d, supports: 64/96/128/192/320, defaulting\n", val); DPRINTF(E_LOG, L_STREAMING, "Unsuppported streaming bit_rate=%d, supports: 64/96/128/192/320, defaulting\n", val);
} }
DPRINTF(E_INFO, L_STREAMING, "Streaming quality: %d/%d/%d @ %dkbps\n", streaming_quality_out.sample_rate, streaming_quality_out.bits_per_sample, streaming_quality_out.channels, streaming_quality_out.bit_rate/1000); DPRINTF(E_INFO, L_STREAMING, "Streaming quality: %d/%d/%d @ %dkbps\n", streaming_quality_out.sample_rate, streaming_quality_out.bits_per_sample, streaming_quality_out.channels, streaming_quality_out.bit_rate/1000);
val = cfg_getint(cfgsec, "icy_metaint"); val = cfg_getint(cfgsec, "icy_metaint");
@ -654,7 +655,12 @@ streaming_init(void)
else else
DPRINTF(E_INFO, L_STREAMING, "Unsupported icy_metaint=%d, supported range: 4096..131072, defaulting to %d\n", val, streaming_icy_metaint); DPRINTF(E_INFO, L_STREAMING, "Unsupported icy_metaint=%d, supported range: 4096..131072, defaulting to %d\n", val, streaming_icy_metaint);
pthread_mutex_init(&streaming_sessions_lck, NULL); ret = mutex_init(&streaming_sessions_lck);
if (ret < 0)
{
DPRINTF(E_FATAL, L_STREAMING, "Could not initialize mutex (%d): %s\n", ret, strerror(ret));
goto error;
}
// Non-blocking because otherwise httpd and player thread may deadlock // Non-blocking because otherwise httpd and player thread may deadlock
#ifdef HAVE_PIPE2 #ifdef HAVE_PIPE2

View File

@ -217,7 +217,7 @@ metadata_get(struct input_source *source)
if (!inputs[source->type]->metadata_get) if (!inputs[source->type]->metadata_get)
return NULL; return NULL;
metadata = calloc(1, sizeof(struct input_metadata)); CHECK_NULL(L_PLAYER, metadata = calloc(1, sizeof(struct input_metadata)));
ret = inputs[source->type]->metadata_get(metadata, source); ret = inputs[source->type]->metadata_get(metadata, source);
if (ret < 0) if (ret < 0)
@ -890,8 +890,8 @@ input_init(void)
int i; int i;
// Prepare input buffer // Prepare input buffer
pthread_mutex_init(&input_buffer.mutex, NULL); CHECK_ERR(L_PLAYER, mutex_init(&input_buffer.mutex));
pthread_cond_init(&input_buffer.cond, NULL); CHECK_ERR(L_PLAYER, pthread_cond_init(&input_buffer.cond, NULL));
CHECK_NULL(L_PLAYER, evbase_input = event_base_new()); CHECK_NULL(L_PLAYER, evbase_input = event_base_new());
CHECK_NULL(L_PLAYER, input_buffer.evbuf = evbuffer_new()); CHECK_NULL(L_PLAYER, input_buffer.evbuf = evbuffer_new());

View File

@ -51,7 +51,7 @@ path_to_media_id_and_type(struct sp_file *file)
struct sp_channel * struct sp_channel *
channel_get(uint32_t channel_id, struct sp_session *session) channel_get(uint32_t channel_id, struct sp_session *session)
{ {
if (channel_id > sizeof(session->channels)/sizeof(session->channels)[0]) if (channel_id >= sizeof(session->channels)/sizeof(session->channels)[0])
return NULL; return NULL;
if (session->channels[channel_id].state == SP_CHANNEL_STATE_UNALLOCATED) if (session->channels[channel_id].state == SP_CHANNEL_STATE_UNALLOCATED)
@ -148,6 +148,7 @@ channel_flush(struct sp_channel *channel)
int fd = channel->audio_fd[0]; int fd = channel->audio_fd[0];
int flags; int flags;
int got; int got;
int ret;
evbuffer_drain(channel->audio_buf, -1); evbuffer_drain(channel->audio_buf, -1);
@ -157,13 +158,18 @@ channel_flush(struct sp_channel *channel)
if (flags == -1) if (flags == -1)
return -1; return -1;
fcntl(fd, F_SETFL, flags | O_NONBLOCK); ret = fcntl(fd, F_SETFL, flags | O_NONBLOCK);
if (ret < 0)
return -1;
do do
got = read(fd, buf, sizeof(buf)); got = read(fd, buf, sizeof(buf));
while (got > 0); while (got > 0);
fcntl(fd, F_SETFL, flags); ret = fcntl(fd, F_SETFL, flags);
if (ret < 0)
return -1;
return 0; return 0;
} }

View File

@ -207,6 +207,10 @@ commands_base_new(struct event_base *evbase, command_exit_cb exit_cb)
int ret; int ret;
cmdbase = calloc(1, sizeof(struct commands_base)); cmdbase = calloc(1, sizeof(struct commands_base));
if (!cmdbase)
{
return NULL;
}
#ifdef HAVE_PIPE2 #ifdef HAVE_PIPE2
ret = pipe2(cmdbase->command_pipe, O_CLOEXEC); ret = pipe2(cmdbase->command_pipe, O_CLOEXEC);
@ -370,6 +374,9 @@ commands_exec_async(struct commands_base *cmdbase, command_function func, void *
int ret; int ret;
cmd = calloc(1, sizeof(struct command)); cmd = calloc(1, sizeof(struct command));
if (!cmd)
return -1;
cmd->func = func; cmd->func = func;
cmd->func_bh = NULL; cmd->func_bh = NULL;
cmd->arg = arg; cmd->arg = arg;

View File

@ -620,7 +620,7 @@ response_aplogin_failed(uint8_t *payload, size_t payload_len, struct sp_session
} }
sp_errmsg = "(unknown login error)"; sp_errmsg = "(unknown login error)";
for (int i = 0; i < sizeof(sp_login_errors); i++) for (int i = 0; i < sizeof(sp_login_errors)/sizeof(sp_login_errors[0]); i++)
{ {
if (sp_login_errors[i].errorcode != aplogin_failed->error_code) if (sp_login_errors[i].errorcode != aplogin_failed->error_code)
continue; continue;

View File

@ -48,6 +48,7 @@ events for proceeding are activated directly.
*/ */
#include <pthread.h> #include <pthread.h>
#include <assert.h>
#include "librespot-c-internal.h" #include "librespot-c-internal.h"
#include "commands.h" #include "commands.h"
@ -833,6 +834,8 @@ librespotc_write(int fd, sp_progress_cb progress_cb, void *cb_arg)
cmdargs = calloc(1, sizeof(struct sp_cmdargs)); cmdargs = calloc(1, sizeof(struct sp_cmdargs));
assert(cmdargs);
cmdargs->fd_read = fd; cmdargs->fd_read = fd;
cmdargs->progress_cb = progress_cb; cmdargs->progress_cb = progress_cb;
cmdargs->cb_arg = cb_arg; cmdargs->cb_arg = cb_arg;

View File

@ -180,7 +180,7 @@ pipe_create(const char *path, int id, enum pipetype type, event_callback_fn cb)
{ {
struct pipe *pipe; struct pipe *pipe;
pipe = calloc(1, sizeof(struct pipe)); CHECK_NULL(L_PLAYER, pipe = calloc(1, sizeof(struct pipe)));
pipe->path = strdup(path); pipe->path = strdup(path);
pipe->id = id; pipe->id = id;
pipe->fd = -1; pipe->fd = -1;
@ -916,9 +916,6 @@ pipe_metadata_watch_add(void *arg)
pipe_metadata_watch_del(NULL); // Just in case we somehow already have a metadata pipe open pipe_metadata_watch_del(NULL); // Just in case we somehow already have a metadata pipe open
pipe_metadata.pipe = pipe_create(path, 0, PIPE_METADATA, pipe_metadata_read_cb); pipe_metadata.pipe = pipe_create(path, 0, PIPE_METADATA, pipe_metadata_read_cb);
if (!pipe_metadata.pipe)
return;
pipe_metadata.evbuf = evbuffer_new(); pipe_metadata.evbuf = evbuffer_new();
ret = watch_add(pipe_metadata.pipe); ret = watch_add(pipe_metadata.pipe);
@ -948,13 +945,18 @@ pipe_thread_start(void)
static void static void
pipe_thread_stop(void) pipe_thread_stop(void)
{ {
int ret;
if (!tid_pipe) if (!tid_pipe)
return; return;
commands_exec_sync(cmdbase, pipe_watch_update, NULL, NULL); commands_exec_sync(cmdbase, pipe_watch_update, NULL, NULL);
commands_base_destroy(cmdbase); commands_base_destroy(cmdbase);
pthread_join(tid_pipe, NULL);
ret = pthread_join(tid_pipe, NULL);
if (ret != 0)
DPRINTF(E_LOG, L_PLAYER, "Could not join pipe thread: %s\n", strerror(errno));
event_base_free(evbase_pipe); event_base_free(evbase_pipe);
tid_pipe = 0; tid_pipe = 0;
} }
@ -1037,9 +1039,10 @@ setup(struct input_source *source)
if (fd < 0) if (fd < 0)
return -1; return -1;
CHECK_NULL(L_PLAYER, pipe = pipe_create(source->path, source->id, PIPE_PCM, NULL));
CHECK_NULL(L_PLAYER, source->evbuf = evbuffer_new()); CHECK_NULL(L_PLAYER, source->evbuf = evbuffer_new());
pipe = pipe_create(source->path, source->id, PIPE_PCM, NULL);
pipe->fd = fd; pipe->fd = fd;
pipe->is_autostarted = (source->id == pipe_autostart_id); pipe->is_autostarted = (source->id == pipe_autostart_id);

View File

@ -816,7 +816,6 @@ process_directory(char *path, int parent_id, int flags)
if (!dirp) if (!dirp)
{ {
DPRINTF(E_LOG, L_SCAN, "Could not open directory %s: %s\n", path, strerror(errno)); DPRINTF(E_LOG, L_SCAN, "Could not open directory %s: %s\n", path, strerror(errno));
return; return;
} }
@ -824,7 +823,10 @@ process_directory(char *path, int parent_id, int flags)
ret = virtual_path_make(virtual_path, sizeof(virtual_path), path); ret = virtual_path_make(virtual_path, sizeof(virtual_path), path);
if (ret < 0) if (ret < 0)
{
closedir(dirp);
return; return;
}
dir_id = db_directory_addorupdate(virtual_path, path, 0, parent_id); dir_id = db_directory_addorupdate(virtual_path, path, 0, parent_id);
if (dir_id <= 0) if (dir_id <= 0)
@ -853,7 +855,6 @@ process_directory(char *path, int parent_id, int flags)
if (errno) if (errno)
{ {
DPRINTF(E_LOG, L_SCAN, "readdir error in %s: %s\n", path, strerror(errno)); DPRINTF(E_LOG, L_SCAN, "readdir error in %s: %s\n", path, strerror(errno));
break; break;
} }
@ -879,7 +880,6 @@ process_directory(char *path, int parent_id, int flags)
if (ret < 0) if (ret < 0)
{ {
DPRINTF(E_LOG, L_SCAN, "Skipping %s, read_attributes() failed\n", entry); DPRINTF(E_LOG, L_SCAN, "Skipping %s, read_attributes() failed\n", entry);
continue; continue;
} }
@ -916,7 +916,6 @@ process_directory(char *path, int parent_id, int flags)
if (wi.wd < 0) if (wi.wd < 0)
{ {
DPRINTF(E_WARN, L_SCAN, "Could not create inotify watch for %s: %s\n", path, strerror(errno)); DPRINTF(E_WARN, L_SCAN, "Could not create inotify watch for %s: %s\n", path, strerror(errno));
return; return;
} }
@ -1112,8 +1111,8 @@ static void
process_inotify_dir(struct watch_info *wi, char *path, struct inotify_event *ie) process_inotify_dir(struct watch_info *wi, char *path, struct inotify_event *ie)
{ {
struct watch_enum we; struct watch_enum we;
struct watch_info dummy_wi;
uint32_t rm_wd; uint32_t rm_wd;
char *s;
int flags = 0; int flags = 0;
int ret; int ret;
int parent_id; int parent_id;
@ -1204,12 +1203,9 @@ process_inotify_dir(struct watch_info *wi, char *path, struct inotify_event *ie)
DPRINTF(E_DBG, L_SCAN, "Directory permissions changed (%s): %s\n", wi->path, path); DPRINTF(E_DBG, L_SCAN, "Directory permissions changed (%s): %s\n", wi->path, path);
// Find out if we are already watching the dir (ret will be 0) // Find out if we are already watching the dir (ret will be 0)
s = wi->path; ret = db_watch_get_bypath(&dummy_wi, path);
wi->path = path;
ret = db_watch_get_bypath(wi);
if (ret == 0) if (ret == 0)
free(wi->path); free_wi(&dummy_wi, 1);
wi->path = s;
// We don't use access() or euidaccess() because they don't work with ACL's // We don't use access() or euidaccess() because they don't work with ACL's
// - this also means we can't check for executable permission, which stat() // - this also means we can't check for executable permission, which stat()
@ -1237,7 +1233,7 @@ process_inotify_dir(struct watch_info *wi, char *path, struct inotify_event *ie)
DPRINTF(E_INFO, L_SCAN, "Directory event, but '%s' already being watched\n", path); DPRINTF(E_INFO, L_SCAN, "Directory event, but '%s' already being watched\n", path);
} }
if (fd > 0) if (fd >= 0)
close(fd); close(fd);
} }
@ -1321,7 +1317,7 @@ process_inotify_file(struct watch_info *wi, char *path, struct inotify_event *ie
ie->mask |= IN_CLOSE_WRITE; ie->mask |= IN_CLOSE_WRITE;
} }
if (fd > 0) if (fd >= 0)
close(fd); close(fd);
} }
@ -1336,7 +1332,7 @@ process_inotify_file(struct watch_info *wi, char *path, struct inotify_event *ie
ret = virtual_path_make(dir_vpath, sizeof(dir_vpath), path); ret = virtual_path_make(dir_vpath, sizeof(dir_vpath), path);
if (ret >= 0) if (ret >= 0)
{ {
ptr = strrchr(dir_vpath, '/'); CHECK_NULL(L_SCAN, ptr = strrchr(dir_vpath, '/'));
*ptr = '\0'; *ptr = '\0';
dir_id = db_directory_id_byvirtualpath(dir_vpath); dir_id = db_directory_id_byvirtualpath(dir_vpath);
@ -1539,8 +1535,7 @@ inotify_cb(int fd, short event, void *arg)
* the memory space for ie[1+] contains the name of the file * the memory space for ie[1+] contains the name of the file
* see the inotify documentation * see the inotify documentation
*/ */
wi.wd = ie->wd; ret = db_watch_get_bywd(&wi, ie->wd);
ret = db_watch_get_bywd(&wi);
if (ret < 0) if (ret < 0)
{ {
if (!(ie->mask & IN_IGNORED)) if (!(ie->mask & IN_IGNORED))
@ -1554,30 +1549,30 @@ inotify_cb(int fd, short event, void *arg)
DPRINTF(E_DBG, L_SCAN, "%s deleted or backing filesystem unmounted!\n", wi.path); DPRINTF(E_DBG, L_SCAN, "%s deleted or backing filesystem unmounted!\n", wi.path);
db_watch_delete_bywd(ie->wd); db_watch_delete_bywd(ie->wd);
free(wi.path); free_wi(&wi, 1);
continue; continue;
} }
path[0] = '\0'; path[0] = '\0';
ret = snprintf(path, PATH_MAX, "%s", wi.path); ret = snprintf(path, sizeof(path), "%s", wi.path);
if ((ret < 0) || (ret >= PATH_MAX)) if ((ret < 0) || (ret >= sizeof(path)))
{ {
DPRINTF(E_LOG, L_SCAN, "Skipping event under %s, PATH_MAX exceeded\n", wi.path); DPRINTF(E_LOG, L_SCAN, "Skipping event under %s, PATH_MAX exceeded\n", wi.path);
free(wi.path); free_wi(&wi, 1);
continue; continue;
} }
if (ie->len > 0) if (ie->len > 0)
{ {
namelen = PATH_MAX - ret; namelen = sizeof(path) - ret;
ret = snprintf(path + ret, namelen, "/%s", ie->name); ret = snprintf(path + ret, namelen, "/%s", ie->name);
if ((ret < 0) || (ret >= namelen)) if ((ret < 0) || (ret >= namelen))
{ {
DPRINTF(E_LOG, L_SCAN, "Skipping %s/%s, PATH_MAX exceeded\n", wi.path, ie->name); DPRINTF(E_LOG, L_SCAN, "Skipping %s/%s, PATH_MAX exceeded\n", wi.path, ie->name);
free(wi.path); free_wi(&wi, 1);
continue; continue;
} }
} }
@ -1595,7 +1590,7 @@ inotify_cb(int fd, short event, void *arg)
#else #else
process_inotify_file_defer(&wi, path, ie); process_inotify_file_defer(&wi, path, ie);
#endif #endif
free(wi.path); free_wi(&wi, 1);
} }
free(buf); free(buf);

View File

@ -749,40 +749,40 @@ process_pl_items(plist_t items, int pl_id, const char *name, struct itml_to_db_m
db_transaction_end(); db_transaction_end();
} }
static int static bool
ignore_pl(plist_t pl, const char *name) ignore_pl(plist_t pl, const char *name)
{ {
uint64_t kind; uint64_t kind;
int smart;
uint8_t master; uint8_t master;
uint8_t party; uint8_t party;
kind = 0; if (get_dictval_int_from_key(pl, "Distinguished Kind", &kind) == 0 && kind > 0)
smart = 0; {
master = 0; DPRINTF(E_INFO, L_SCAN, "Ignoring iTunes builtin playlist '%s' (k %" PRIu64 ")\n", name, kind);
party = 0; return true;
}
/* Special (builtin) playlists */ if (get_dictval_bool_from_key(pl, "Master", &master) == 0 && master)
get_dictval_int_from_key(pl, "Distinguished Kind", &kind); {
DPRINTF(E_INFO, L_SCAN, "Ignoring iTunes Master playlist '%s'\n", name);
return true;
}
if (get_dictval_bool_from_key(pl, "Party Shuffle", &party) == 0 && party)
{
DPRINTF(E_INFO, L_SCAN, "Ignoring iTunes Party Shuffle playlist '%s'\n", name);
return true;
}
/* Import smart playlists (optional) */ /* Import smart playlists (optional) */
if (!cfg_getbool(cfg_getsec(cfg, "library"), "itunes_smartpl") if (!cfg_getbool(cfg_getsec(cfg, "library"), "itunes_smartpl")
&& (plist_dict_get_item(pl, "Smart Info") || plist_dict_get_item(pl, "Smart Criteria"))) && (plist_dict_get_item(pl, "Smart Info") || plist_dict_get_item(pl, "Smart Criteria")))
smart = 1;
/* Not interested in the Master playlist */
get_dictval_bool_from_key(pl, "Master", &master);
/* Not interested in Party Shuffle playlists */
get_dictval_bool_from_key(pl, "Party Shuffle", &party);
if ((kind > 0) || smart || party || master)
{ {
DPRINTF(E_INFO, L_SCAN, "Ignoring playlist '%s' (k %" PRIu64 " s%d p%d m%d)\n", name, kind, smart, party, master); DPRINTF(E_INFO, L_SCAN, "Ignoring iTunes smart playlist as set in config '%s'\n", name);
return true;
return 1;
} }
return 0; return false;
} }
static void static void

View File

@ -351,9 +351,10 @@ request_endpoint(const char *uri)
json_object *json_response = NULL; json_object *json_response = NULL;
int ret; int ret;
ctx = calloc(1, sizeof(struct http_client_ctx)); CHECK_NULL(L_SPOTIFY, ctx = calloc(1, sizeof(struct http_client_ctx)));
ctx->output_headers = calloc(1, sizeof(struct keyval)); CHECK_NULL(L_SPOTIFY, ctx->output_headers = calloc(1, sizeof(struct keyval)));
ctx->input_body = evbuffer_new(); CHECK_NULL(L_SPOTIFY, ctx->input_body = evbuffer_new());
ctx->url = uri; ctx->url = uri;
snprintf(bearer_token, sizeof(bearer_token), "Bearer %s", spotify_credentials.access_token); snprintf(bearer_token, sizeof(bearer_token), "Bearer %s", spotify_credentials.access_token);
@ -1091,7 +1092,9 @@ spotifywebapi_oauth_uri_get(const char *redirect_uri)
if (param) if (param)
{ {
uri_len = strlen(spotify_auth_uri) + strlen(param) + 3; uri_len = strlen(spotify_auth_uri) + strlen(param) + 3;
uri = calloc(uri_len, sizeof(char));
CHECK_NULL(L_SPOTIFY, uri = calloc(uri_len, sizeof(char)));
snprintf(uri, uri_len, "%s/?%s", spotify_auth_uri, param); snprintf(uri, uri_len, "%s/?%s", spotify_auth_uri, param);
free(param); free(param);

View File

@ -364,7 +364,7 @@ signal_signalfd_cb(int fd, short event, void *arg)
struct signalfd_siginfo info; struct signalfd_siginfo info;
int status; int status;
while (read(fd, &info, sizeof(struct signalfd_siginfo)) > 0) while (read(fd, &info, sizeof(struct signalfd_siginfo)) == sizeof(struct signalfd_siginfo))
{ {
switch (info.ssi_signo) switch (info.ssi_signo)
{ {

View File

@ -258,7 +258,7 @@ net_bind(short unsigned *port, int type, const char *log_service_name)
char strport[8]; char strport[8];
int yes = 1; int yes = 1;
int no = 0; int no = 0;
int fd; int fd = -1;
int ret; int ret;
cfgaddr = cfg_getstr(cfg_getsec(cfg, "general"), "bind_address"); cfgaddr = cfg_getstr(cfg_getsec(cfg, "general"), "bind_address");
@ -275,7 +275,7 @@ net_bind(short unsigned *port, int type, const char *log_service_name)
return -1; return -1;
} }
for (ptr = servinfo, fd = -1; ptr != NULL; ptr = ptr->ai_next) for (ptr = servinfo; ptr != NULL; ptr = ptr->ai_next)
{ {
if (fd >= 0) if (fd >= 0)
close(fd); close(fd);
@ -338,6 +338,7 @@ net_bind(short unsigned *port, int type, const char *log_service_name)
return fd; return fd;
error: error:
if (fd >= 0)
close(fd); close(fd);
return -1; return -1;
} }

View File

@ -811,7 +811,8 @@ parse_group_params(int argc, char **argv, bool group_in_listcommand, struct quer
return 0; return 0;
*groupsize = (argc - first_group) / 2; *groupsize = (argc - first_group) / 2;
*group = calloc(*groupsize, sizeof(struct mpd_tagtype *));
CHECK_NULL(L_MPD, *group = calloc(*groupsize, sizeof(struct mpd_tagtype *)));
// Now process all group/field arguments // Now process all group/field arguments
for (j = 0; j < (*groupsize); j++) for (j = 0; j < (*groupsize); j++)
@ -3557,7 +3558,7 @@ output_get_cb(struct player_speaker_info *spk, void *arg)
if (!param->output if (!param->output
&& param->shortid == (unsigned short) spk->id) && param->shortid == (unsigned short) spk->id)
{ {
param->output = calloc(1, sizeof(struct output)); CHECK_NULL(L_MPD, param->output = calloc(1, sizeof(struct output)));
param->output->id = spk->id; param->output->id = spk->id;
param->output->shortid = (unsigned short) spk->id; param->output->shortid = (unsigned short) spk->id;

View File

@ -559,7 +559,8 @@ device_id_colon_parse(uint64_t *id, const char *id_str)
char *ptr; char *ptr;
int ret; int ret;
s = calloc(1, strlen(id_str) + 1); CHECK_NULL(L_AIRPLAY, s = calloc(1, strlen(id_str) + 1));
for (ptr = s; *id_str != '\0'; id_str++) for (ptr = s; *id_str != '\0'; id_str++)
{ {
if (*id_str == ':') if (*id_str == ':')

View File

@ -948,6 +948,7 @@ sync_correct(struct alsa_playback_session *pb, double drift, double latency, str
{ {
int step; int step;
int sign; int sign;
int ret;
// We change the sample_rate in steps that are a multiple of 50. So we might // We change the sample_rate in steps that are a multiple of 50. So we might
// step 44100 -> 44000 -> 40900 -> 44000 -> 44100. If we used percentages to // step 44100 -> 44000 -> 40900 -> 44000 -> 44100. If we used percentages to
@ -975,7 +976,14 @@ sync_correct(struct alsa_playback_session *pb, double drift, double latency, str
pb->quality.sample_rate += sign * step; pb->quality.sample_rate += sign * step;
if (pb->sync_resample_step != 0) if (pb->sync_resample_step != 0)
outputs_quality_subscribe(&pb->quality); {
ret = outputs_quality_subscribe(&pb->quality);
if (ret < 0)
{
DPRINTF(E_LOG, L_LAUDIO, "Error adjusting sample rate to %d to maintain sync\n", pb->quality.sample_rate);
return;
}
}
// Reset position so next sync_correct latency correction is only based on // Reset position so next sync_correct latency correction is only based on
// what has elapsed since our correction // what has elapsed since our correction

View File

@ -931,7 +931,7 @@ static int
raop_parse_auth(struct raop_session *rs, struct evrtsp_request *req) raop_parse_auth(struct raop_session *rs, struct evrtsp_request *req)
{ {
const char *param; const char *param;
char *auth; char *auth = NULL;
char *token; char *token;
char *ptr; char *ptr;
@ -951,8 +951,7 @@ raop_parse_auth(struct raop_session *rs, struct evrtsp_request *req)
if (!param) if (!param)
{ {
DPRINTF(E_LOG, L_RAOP, "WWW-Authenticate header not found\n"); DPRINTF(E_LOG, L_RAOP, "WWW-Authenticate header not found\n");
goto error;
return -1;
} }
DPRINTF(E_DBG, L_RAOP, "WWW-Authenticate: %s\n", param); DPRINTF(E_DBG, L_RAOP, "WWW-Authenticate: %s\n", param);
@ -960,19 +959,23 @@ raop_parse_auth(struct raop_session *rs, struct evrtsp_request *req)
if (strncmp(param, "Digest ", strlen("Digest ")) != 0) if (strncmp(param, "Digest ", strlen("Digest ")) != 0)
{ {
DPRINTF(E_LOG, L_RAOP, "Unsupported authentication method: %s\n", param); DPRINTF(E_LOG, L_RAOP, "Unsupported authentication method: %s\n", param);
goto error;
return -1;
} }
auth = strdup(param); auth = strdup(param);
if (!auth) if (!auth)
{ {
DPRINTF(E_LOG, L_RAOP, "Out of memory for WWW-Authenticate header copy\n"); DPRINTF(E_LOG, L_RAOP, "Out of memory for WWW-Authenticate header copy\n");
goto error;
return -1;
} }
token = strchr(auth, ' '); token = strchr(auth, ' ');
if (!token)
{
DPRINTF(E_LOG, L_RAOP, "Unexpected WWW-Authenticate auth\n");
goto error;
}
token++; token++;
token = strtok_r(token, " =", &ptr); token = strtok_r(token, " =", &ptr);
@ -998,8 +1001,6 @@ raop_parse_auth(struct raop_session *rs, struct evrtsp_request *req)
token = strtok_r(NULL, " =", &ptr); token = strtok_r(NULL, " =", &ptr);
} }
free(auth);
if (!rs->realm || !rs->nonce) if (!rs->realm || !rs->nonce)
{ {
DPRINTF(E_LOG, L_RAOP, "Could not find realm/nonce in WWW-Authenticate header\n"); DPRINTF(E_LOG, L_RAOP, "Could not find realm/nonce in WWW-Authenticate header\n");
@ -1016,12 +1017,17 @@ raop_parse_auth(struct raop_session *rs, struct evrtsp_request *req)
rs->nonce = NULL; rs->nonce = NULL;
} }
return -1; goto error;
} }
DPRINTF(E_DBG, L_RAOP, "Found realm: [%s], nonce: [%s]\n", rs->realm, rs->nonce); DPRINTF(E_DBG, L_RAOP, "Found realm: [%s], nonce: [%s]\n", rs->realm, rs->nonce);
free(auth);
return 0; return 0;
error:
free(auth);
return -1;
} }
static int static int
@ -3363,6 +3369,14 @@ raop_cb_startup_setup(struct evrtsp_request *req, void *arg)
} }
token = strchr(transport, ';'); token = strchr(transport, ';');
if (!token)
{
DPRINTF(E_LOG, L_RAOP, "Missing semicolon in Transport header: %s\n", transport);
free(transport);
goto cleanup;
}
token++; token++;
token = strtok_r(token, ";=", &ptr); token = strtok_r(token, ";=", &ptr);

View File

@ -70,6 +70,9 @@ pair_tlv_new() {
void void
pair_tlv_free(pair_tlv_values_t *values) { pair_tlv_free(pair_tlv_values_t *values) {
if (!values)
return;
pair_tlv_t *t = values->head; pair_tlv_t *t = values->head;
while (t) { while (t) {
pair_tlv_t *t2 = t; pair_tlv_t *t2 = t;

View File

@ -347,7 +347,7 @@ srp_user_process_challenge(struct SRPUser *usr, const unsigned char *bytes_s, in
bnum u, x; bnum u, x;
*len_M = 0; *len_M = 0;
*bytes_M = 0; *bytes_M = NULL;
bnum_bin2bn(s, bytes_s, len_s); bnum_bin2bn(s, bytes_s, len_s);
bnum_bin2bn(B, bytes_B, len_B); bnum_bin2bn(B, bytes_B, len_B);
@ -384,15 +384,8 @@ srp_user_process_challenge(struct SRPUser *usr, const unsigned char *bytes_s, in
calculate_H_AMK(usr->alg, usr->H_AMK, usr->A, usr->M, usr->session_key, usr->session_key_len); calculate_H_AMK(usr->alg, usr->H_AMK, usr->A, usr->M, usr->session_key, usr->session_key_len);
*bytes_M = usr->M; *bytes_M = usr->M;
if (len_M)
*len_M = hash_length(usr->alg); *len_M = hash_length(usr->alg);
} }
else
{
*bytes_M = NULL;
if (len_M)
*len_M = 0;
}
cleanup2: cleanup2:
bnum_free(x); bnum_free(x);

View File

@ -55,6 +55,8 @@
#define REQUEST_BUFSIZE 4096 #define REQUEST_BUFSIZE 4096
#define ENCRYPTED_LEN_MAX 0x400 #define ENCRYPTED_LEN_MAX 0x400
// #define DEBUG_SHORT_A 1
enum pair_keys enum pair_keys
{ {
PAIR_SETUP_MSG01 = 0, PAIR_SETUP_MSG01 = 0,
@ -418,6 +420,14 @@ srp_user_get_session_key(struct SRPUser *usr, int *key_length)
return usr->session_key; return usr->session_key;
} }
#ifdef DEBUG_SHORT_A
// This value of "a" will yield a 383 byte A
static uint8_t short_a[] = {
0xef, 0xb5, 0x93, 0xf5, 0x03, 0x97, 0x69, 0x8e, 0x15, 0xed, 0xee, 0x5b, 0xf2, 0xf9, 0x23, 0x6c,
0xf0, 0x59, 0x6c, 0xe2, 0x77, 0xf2, 0x14, 0x16, 0xac, 0x99, 0xfa, 0x31, 0xae, 0x2b, 0xd3, 0x41,
};
#endif
/* Output: username, bytes_A, len_A */ /* Output: username, bytes_A, len_A */
static void static void
srp_user_start_authentication(struct SRPUser *usr, const char **username, srp_user_start_authentication(struct SRPUser *usr, const char **username,
@ -425,6 +435,9 @@ srp_user_start_authentication(struct SRPUser *usr, const char **username,
{ {
// BN_hex2bn(&(usr->a), "D929DFB605687233C9E9030C2280156D03BDB9FDCF3CCE3BC27D9CCFCB5FF6A1"); // BN_hex2bn(&(usr->a), "D929DFB605687233C9E9030C2280156D03BDB9FDCF3CCE3BC27D9CCFCB5FF6A1");
bnum_random(usr->a, 256); bnum_random(usr->a, 256);
#ifdef DEBUG_SHORT_A
bnum_bin2bn(usr->a, short_a, sizeof(short_a));
#endif
#ifdef DEBUG_PAIR #ifdef DEBUG_PAIR
bnum_dump("Random value of usr->a:\n", usr->a); bnum_dump("Random value of usr->a:\n", usr->a);
#endif #endif
@ -459,7 +472,7 @@ srp_user_process_challenge(struct SRPUser *usr, const unsigned char *bytes_s, in
bnum u, x; bnum u, x;
*len_M = 0; *len_M = 0;
*bytes_M = 0; *bytes_M = NULL;
bnum_bin2bn(s, bytes_s, len_s); bnum_bin2bn(s, bytes_s, len_s);
bnum_bin2bn(B, bytes_B, len_B); bnum_bin2bn(B, bytes_B, len_B);
@ -499,15 +512,8 @@ srp_user_process_challenge(struct SRPUser *usr, const unsigned char *bytes_s, in
calculate_H_AMK(usr->alg, usr->H_AMK, usr->A, usr->M, usr->session_key, usr->session_key_len); calculate_H_AMK(usr->alg, usr->H_AMK, usr->A, usr->M, usr->session_key, usr->session_key_len);
*bytes_M = usr->M; *bytes_M = usr->M;
if (len_M)
*len_M = hash_length(usr->alg); *len_M = hash_length(usr->alg);
} }
else
{
*bytes_M = NULL;
if (len_M)
*len_M = 0;
}
cleanup2: cleanup2:
bnum_free(x); bnum_free(x);
@ -1936,6 +1942,7 @@ client_verify_response2(struct pair_verify_context *handle, const uint8_t *data,
handle->status = PAIR_STATUS_COMPLETED; handle->status = PAIR_STATUS_COMPLETED;
pair_tlv_free(response);
return 0; return 0;
} }

View File

@ -764,7 +764,7 @@ remote_pairing_pair(const char *pin)
return REMOTE_INVALID_PIN; return REMOTE_INVALID_PIN;
} }
strncpy(cmdarg, pin, sizeof(cmdarg)); snprintf(cmdarg, sizeof(cmdarg), "%s", pin);
return commands_exec_sync(cmdbase, pairing_pair, NULL, &cmdarg); return commands_exec_sync(cmdbase, pairing_pair, NULL, &cmdarg);
} }

View File

@ -128,7 +128,7 @@ rng_rand_range(struct rng_ctx *ctx, int32_t min, int32_t max)
* Durstenfeld in-place shuffling variant * Durstenfeld in-place shuffling variant
*/ */
void void
shuffle_int(struct rng_ctx *ctx, int *values, int len) rng_shuffle_int(struct rng_ctx *ctx, int *values, int len)
{ {
int i; int i;
int32_t j; int32_t j;

View File

@ -19,7 +19,7 @@ int32_t
rng_rand_range(struct rng_ctx *ctx, int32_t min, int32_t max); rng_rand_range(struct rng_ctx *ctx, int32_t min, int32_t max);
void void
shuffle_int(struct rng_ctx *ctx, int *values, int len); rng_shuffle_int(struct rng_ctx *ctx, int *values, int len);
#endif /* !__RNG_H__ */ #endif /* !__RNG_H__ */

View File

@ -23,7 +23,6 @@
#include <json.h> #include <json.h>
#include <libwebsockets.h> #include <libwebsockets.h>
#include <pthread.h> #include <pthread.h>
#include <errno.h>
#include <stdbool.h> #include <stdbool.h>
#include <stdio.h> #include <stdio.h>
#include <string.h> #include <string.h>
@ -511,13 +510,19 @@ websocket_init(void)
return -1; return -1;
} }
pthread_mutex_init(&websocket_write_event_lock, NULL); ret = mutex_init(&websocket_write_event_lock);
websocket_write_events = 0; if (ret < 0)
{
DPRINTF(E_LOG, L_WEB, "Failed to initialize mutex: %s\n", strerror(ret));
lws_context_destroy(context);
return -1;
}
ret = pthread_create(&tid_websocket, NULL, websocket, NULL); ret = pthread_create(&tid_websocket, NULL, websocket, NULL);
if (ret < 0) if (ret < 0)
{ {
DPRINTF(E_LOG, L_WEB, "Could not spawn websocket thread: %s\n", strerror(errno)); DPRINTF(E_LOG, L_WEB, "Could not spawn websocket thread (%d): %s\n", ret, strerror(ret));
pthread_mutex_destroy(&websocket_write_event_lock);
lws_context_destroy(context); lws_context_destroy(context);
return -1; return -1;
} }
@ -530,10 +535,15 @@ websocket_init(void)
void void
websocket_deinit(void) websocket_deinit(void)
{ {
if (websocket_port > 0) int ret;
{
if (websocket_port <= 0)
return;
websocket_exit = true; websocket_exit = true;
pthread_join(tid_websocket, NULL); ret = pthread_join(tid_websocket, NULL);
if (ret < 0)
DPRINTF(E_LOG, L_WEB, "Error joining websocket thread (%d): %s\n", ret, strerror(ret));
pthread_mutex_destroy(&websocket_write_event_lock); pthread_mutex_destroy(&websocket_write_event_lock);
} }
}