summaryrefslogtreecommitdiffstats
path: root/mp_fifo.c
blob: 4bda7209e0c8b23a81f553ac2040f09e9a64c1cf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
/*
 * This file is part of MPlayer.
 *
 * MPlayer 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.
 *
 * MPlayer 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 MPlayer; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
#include "osdep/timer.h"
#include "input/input.h"
#include "input/keycodes.h"
#include "mp_fifo.h"
#include "talloc.h"
#include "options.h"


struct mp_fifo {
    struct MPOpts *opts;
    int *data;
    int readpos;
    int size;
    int num_entries;
    int max_up;
    int num_up;
    int last_key_down;
    unsigned last_down_time;
};

struct mp_fifo *mp_fifo_create(struct MPOpts *opts)
{
    struct mp_fifo *fifo = talloc_zero(NULL, struct mp_fifo);
    fifo->opts = opts;
    /* Typical mouse wheel use will generate a sequence repeating 3 events:
     * down, doubleclick, up, down, doubleclick, up, ...
     * Normally only one of those event types triggers a command,
     * so allow opts->key_fifo_size such repeats.
     */
    fifo->max_up = opts->key_fifo_size;
    fifo->size = opts->key_fifo_size * 3;
    fifo->data = talloc_array_ptrtype(fifo, fifo->data, fifo->size);
    return fifo;
}

static bool is_up(int code)
{
    return code > 0 && !(code & MP_KEY_DOWN)
        && !(code >= MOUSE_BTN0_DBL && code < MOUSE_BTN_DBL_END);
}

static int fifo_peek(struct mp_fifo *fifo, int offset)
{
    return fifo->data[(fifo->readpos + offset) % fifo->size];
}

static int fifo_read(struct mp_fifo *fifo)
{
    int code = fifo_peek(fifo, 0);
    fifo->readpos += 1;
    fifo->readpos %= fifo->size;
    fifo->num_entries--;
    fifo->num_up -= is_up(code);
    assert(fifo->num_entries >= 0);
    assert(fifo->num_up >= 0);
    return code;
}

static void fifo_write(struct mp_fifo *fifo, int code)
{
    fifo->data[(fifo->readpos + fifo->num_entries) % fifo->size] = code;
    fifo->num_entries++;
    fifo->num_up += is_up(code);
    assert(fifo->num_entries <= fifo->size);
    assert(fifo->num_up <= fifo->max_up);
}

static void mplayer_put_key_internal(struct mp_fifo *fifo, int code)
{
    // Clear key-down state if we're forced to drop entries
    if (fifo->num_entries >= fifo->size - 1
        || fifo->num_up >= fifo->max_up) {
        if (fifo_peek(fifo, fifo->num_entries - 1) != MP_INPUT_RELEASE_ALL)
            fifo_write(fifo, MP_INPUT_RELEASE_ALL);
    } else
        fifo_write(fifo, code);
}

int mplayer_get_key(void *ctx, int fd)
{
    struct mp_fifo *fifo = ctx;
    if (!fifo->num_entries)
        return MP_INPUT_NOTHING;
    return fifo_read(fifo);
}

static void put_double(struct mp_fifo *fifo, int code)
{
  if (code >= MOUSE_BTN0 && code < MOUSE_BTN_END)
      mplayer_put_key_internal(fifo, code - MOUSE_BTN0 + MOUSE_BTN0_DBL);
}

void mplayer_put_key(struct mp_fifo *fifo, int code)
{
    unsigned now = GetTimerMS();
    int doubleclick_time = fifo->opts->doubleclick_time;
    // ignore system-doubleclick if we generate these events ourselves
    if (doubleclick_time
        && (code & ~MP_KEY_DOWN) >= MOUSE_BTN0_DBL
        && (code & ~MP_KEY_DOWN) < MOUSE_BTN_DBL_END)
        return;
    mplayer_put_key_internal(fifo, code);
    if (code & MP_KEY_DOWN) {
        code &= ~MP_KEY_DOWN;
        if (fifo->last_key_down == code
            && now - fifo->last_down_time < doubleclick_time)
            put_double(fifo, code);
        fifo->last_key_down = code;
        fifo->last_down_time = now;
    }
}