mirror of
https://github.com/mpv-player/mpv
synced 2025-01-19 13:51:14 +00:00
input: make input queue thread safe
If pthreads are enabled the input queue accesses are regulated by acquiring a mutex. This is useful for platforms like OS X, where the events are created in the cocoa thread and added to the queue to then be dequeued in the playloop thread.
This commit is contained in:
parent
87ce0c3b8d
commit
36586dd7b7
@ -66,6 +66,16 @@
|
|||||||
|
|
||||||
#define MP_MAX_KEY_DOWN 4
|
#define MP_MAX_KEY_DOWN 4
|
||||||
|
|
||||||
|
#if HAVE_PTHREADS
|
||||||
|
#include <pthread.h>
|
||||||
|
static pthread_mutex_t queue_mutex = PTHREAD_MUTEX_INITIALIZER;
|
||||||
|
#define queue_lock() pthread_mutex_lock(&queue_mutex)
|
||||||
|
#define queue_unlock() pthread_mutex_unlock(&queue_mutex)
|
||||||
|
#else
|
||||||
|
#define queue_lock() 0
|
||||||
|
#define queue_lock() 0
|
||||||
|
#endif
|
||||||
|
|
||||||
struct cmd_bind {
|
struct cmd_bind {
|
||||||
int keys[MP_MAX_KEY_DOWN];
|
int keys[MP_MAX_KEY_DOWN];
|
||||||
int num_keys;
|
int num_keys;
|
||||||
@ -653,23 +663,30 @@ bool mp_input_is_abort_cmd(int cmd_id)
|
|||||||
|
|
||||||
static int queue_count_cmds(struct cmd_queue *queue)
|
static int queue_count_cmds(struct cmd_queue *queue)
|
||||||
{
|
{
|
||||||
|
queue_lock();
|
||||||
int res = 0;
|
int res = 0;
|
||||||
for (struct mp_cmd *cmd = queue->first; cmd; cmd = cmd->queue_next)
|
for (struct mp_cmd *cmd = queue->first; cmd; cmd = cmd->queue_next)
|
||||||
res++;
|
res++;
|
||||||
|
queue_unlock();
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool queue_has_abort_cmds(struct cmd_queue *queue)
|
static bool queue_has_abort_cmds(struct cmd_queue *queue)
|
||||||
{
|
{
|
||||||
for (struct mp_cmd *cmd = queue->first; cmd; cmd = cmd->queue_next) {
|
queue_lock();
|
||||||
if (mp_input_is_abort_cmd(cmd->id))
|
bool ret = false;
|
||||||
return true;
|
for (struct mp_cmd *cmd = queue->first; cmd; cmd = cmd->queue_next)
|
||||||
|
if (mp_input_is_abort_cmd(cmd->id)) {
|
||||||
|
ret = true;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return false;
|
queue_unlock();
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void queue_remove(struct cmd_queue *queue, struct mp_cmd *cmd)
|
static void queue_remove(struct cmd_queue *queue, struct mp_cmd *cmd)
|
||||||
{
|
{
|
||||||
|
queue_lock();
|
||||||
struct mp_cmd **p_prev = &queue->first;
|
struct mp_cmd **p_prev = &queue->first;
|
||||||
while (*p_prev != cmd) {
|
while (*p_prev != cmd) {
|
||||||
p_prev = &(*p_prev)->queue_next;
|
p_prev = &(*p_prev)->queue_next;
|
||||||
@ -677,11 +694,13 @@ static void queue_remove(struct cmd_queue *queue, struct mp_cmd *cmd)
|
|||||||
// if this fails, cmd was not in the queue
|
// if this fails, cmd was not in the queue
|
||||||
assert(*p_prev == cmd);
|
assert(*p_prev == cmd);
|
||||||
*p_prev = cmd->queue_next;
|
*p_prev = cmd->queue_next;
|
||||||
|
queue_unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void queue_add(struct cmd_queue *queue, struct mp_cmd *cmd,
|
static void queue_add(struct cmd_queue *queue, struct mp_cmd *cmd,
|
||||||
bool at_head)
|
bool at_head)
|
||||||
{
|
{
|
||||||
|
queue_lock();
|
||||||
if (at_head) {
|
if (at_head) {
|
||||||
cmd->queue_next = queue->first;
|
cmd->queue_next = queue->first;
|
||||||
queue->first = cmd;
|
queue->first = cmd;
|
||||||
@ -692,6 +711,16 @@ static void queue_add(struct cmd_queue *queue, struct mp_cmd *cmd,
|
|||||||
*p_prev = cmd;
|
*p_prev = cmd;
|
||||||
cmd->queue_next = NULL;
|
cmd->queue_next = NULL;
|
||||||
}
|
}
|
||||||
|
queue_unlock();
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct mp_cmd *queue_peek(struct cmd_queue *queue)
|
||||||
|
{
|
||||||
|
struct mp_cmd *ret = NULL;
|
||||||
|
queue_lock();
|
||||||
|
ret = queue->first;
|
||||||
|
queue_unlock();
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct input_fd *mp_input_add_fd(struct input_ctx *ictx)
|
static struct input_fd *mp_input_add_fd(struct input_ctx *ictx)
|
||||||
@ -1752,7 +1781,7 @@ mp_cmd_t *mp_input_get_cmd(struct input_ctx *ictx, int time, int peek_only)
|
|||||||
if (repeated)
|
if (repeated)
|
||||||
queue_add(queue, repeated, false);
|
queue_add(queue, repeated, false);
|
||||||
}
|
}
|
||||||
struct mp_cmd *ret = queue->first;
|
struct mp_cmd *ret = queue_peek(queue);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user