mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-27 22:46:02 -05:00
18a80f15dd
Using worker threads instead of httpd threads means that we, not libevent, decide which requests get handled by which threads. This means that we can make sure blocking requests (e.g. volume changes) don't get in the way of realtime(ish) stuff like mp3 streaming. Includes refactor of httpd_stream_file() since it was a bit of a monster.
190 lines
3.8 KiB
C
190 lines
3.8 KiB
C
/*
|
|
* Copyright (C) 2014 Espen Jürgensen <espenjurgensen@gmail.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 <inttypes.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <time.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <sys/queue.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <event2/event.h>
|
|
|
|
#include "db.h"
|
|
#include "logger.h"
|
|
#include "worker.h"
|
|
#include "evthr.h"
|
|
#include "misc.h"
|
|
|
|
#define THREADPOOL_NTHREADS 4
|
|
|
|
static struct evthr_pool *worker_threadpool;
|
|
static __thread struct evthr *worker_thr;
|
|
|
|
|
|
/* ----------------------------- CALLBACK EXECUTION ------------------------- */
|
|
/* Worker threads */
|
|
|
|
struct worker_arg
|
|
{
|
|
void (*cb)(void *);
|
|
void *cb_arg;
|
|
int delay;
|
|
struct event *timer;
|
|
};
|
|
|
|
|
|
static void
|
|
execute_cb(int fd, short what, void *arg)
|
|
{
|
|
struct worker_arg *cmdarg = arg;
|
|
|
|
cmdarg->cb(cmdarg->cb_arg);
|
|
|
|
event_free(cmdarg->timer);
|
|
free(cmdarg->cb_arg);
|
|
free(cmdarg);
|
|
}
|
|
|
|
static void
|
|
execute(struct evthr *thr, void *arg, void *shared)
|
|
{
|
|
struct worker_arg *cmdarg = arg;
|
|
struct timeval tv = { cmdarg->delay, 0 };
|
|
struct event_base *evbase;
|
|
|
|
if (cmdarg->delay)
|
|
{
|
|
evbase = evthr_get_base(thr);
|
|
cmdarg->timer = evtimer_new(evbase, execute_cb, cmdarg);
|
|
evtimer_add(cmdarg->timer, &tv);
|
|
return;
|
|
}
|
|
|
|
cmdarg->cb(cmdarg->cb_arg);
|
|
free(cmdarg->cb_arg);
|
|
free(cmdarg);
|
|
}
|
|
|
|
static void
|
|
init_cb(struct evthr *thr, void *shared)
|
|
{
|
|
CHECK_ERR(L_MAIN, db_perthread_init());
|
|
|
|
worker_thr = thr;
|
|
|
|
thread_setname(pthread_self(), "worker");
|
|
}
|
|
|
|
static void
|
|
exit_cb(struct evthr *thr, void *shared)
|
|
{
|
|
worker_thr = NULL;
|
|
|
|
db_perthread_deinit();
|
|
}
|
|
|
|
|
|
/* ---------------------------- Our worker API --------------------------- */
|
|
|
|
void
|
|
worker_execute(void (*cb)(void *), void *cb_arg, size_t arg_size, int delay)
|
|
{
|
|
struct worker_arg *cmdarg;
|
|
void *argcpy;
|
|
|
|
cmdarg = calloc(1, sizeof(struct worker_arg));
|
|
if (!cmdarg)
|
|
{
|
|
DPRINTF(E_LOG, L_MAIN, "Could not allocate worker_arg\n");
|
|
return;
|
|
}
|
|
|
|
if (arg_size > 0)
|
|
{
|
|
argcpy = malloc(arg_size);
|
|
if (!argcpy)
|
|
{
|
|
DPRINTF(E_LOG, L_MAIN, "Out of memory\n");
|
|
free(cmdarg);
|
|
return;
|
|
}
|
|
|
|
memcpy(argcpy, cb_arg, arg_size);
|
|
}
|
|
else
|
|
argcpy = NULL;
|
|
|
|
cmdarg->cb = cb;
|
|
cmdarg->cb_arg = argcpy;
|
|
cmdarg->delay = delay;
|
|
|
|
evthr_pool_defer(worker_threadpool, execute, cmdarg);
|
|
}
|
|
|
|
struct event_base *
|
|
worker_evbase_get(void)
|
|
{
|
|
return evthr_get_base(worker_thr);
|
|
}
|
|
|
|
int
|
|
worker_init(void)
|
|
{
|
|
int ret;
|
|
|
|
worker_threadpool = evthr_pool_wexit_new(THREADPOOL_NTHREADS, init_cb, exit_cb, NULL);
|
|
if (!worker_threadpool)
|
|
{
|
|
DPRINTF(E_LOG, L_MAIN, "Could not create worker thread pool\n");
|
|
goto error;
|
|
}
|
|
|
|
ret = evthr_pool_start(worker_threadpool);
|
|
if (ret < 0)
|
|
{
|
|
DPRINTF(E_LOG, L_MAIN, "Could not spawn worker threads\n");
|
|
goto error;
|
|
}
|
|
|
|
return 0;
|
|
|
|
error:
|
|
worker_deinit();
|
|
return -1;
|
|
}
|
|
|
|
void
|
|
worker_deinit(void)
|
|
{
|
|
evthr_pool_stop(worker_threadpool);
|
|
evthr_pool_free(worker_threadpool);
|
|
}
|