aboutsummaryrefslogtreecommitdiff
path: root/src/os/event.c
blob: 798bf03946258a6fde556c1c8e011236bdaecf70 (plain) (blame)
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
#include <stdint.h>
#include <stdbool.h>

#include <uv.h>

#include "os/event.h"
#include "os/input.h"

static uv_timer_t timer;
static uv_prepare_t timer_prepare;
static void timer_cb(uv_timer_t *handle, int);
static void timer_prepare_cb(uv_prepare_t *, int);

void event_init()
{
  /* Initialize input events */
  input_init();
  /* Timer to wake the event loop if a timeout argument is passed to
   * `event_poll` */
  uv_timer_init(uv_default_loop(), &timer);
  /* This prepare handle that actually starts the timer */
  uv_prepare_init(uv_default_loop(), &timer_prepare);
}

/* Wait for some event */
bool event_poll(int32_t ms)
{
  bool timed_out;
  uv_run_mode run_mode = UV_RUN_ONCE;

  if (input_ready()) {
    /* If there's a pending input event to be consumed, do it now */
    return true;
  }

  input_start();
  timed_out = false;

  if (ms > 0) {
    /* Timeout passed as argument to the timer */
    timer.data = &timed_out;
    /* We only start the timer after the loop is running, for that we
     * use an prepare handle(pass the interval as data to it) */
    timer_prepare.data = &ms;
    uv_prepare_start(&timer_prepare, timer_prepare_cb);
  } else if (ms == 0) {
    /* 
     * For ms == 0, we need to do a non-blocking event poll by
     * setting the run mode to UV_RUN_NOWAIT.
     */
    run_mode = UV_RUN_NOWAIT;
  }

  do {
    /* Run one event loop iteration, blocking for events if run_mode is
     * UV_RUN_ONCE */
    uv_run(uv_default_loop(), run_mode);
  } while (
      /* Continue running if ... */
      !input_ready() && /* ... we have no input */
      run_mode != UV_RUN_NOWAIT && /* ... ms != 0 */
      !timed_out  /* ... we didn't get a timeout */
      );

  input_stop();

  if (ms > 0) {
    /* Stop the timer */
    uv_timer_stop(&timer);
  }

  return input_ready();
}

/* Set a flag in the `event_poll` loop for signaling of a timeout */
static void timer_cb(uv_timer_t *handle, int status)
{
  *((bool *)handle->data) = true;
}

static void timer_prepare_cb(uv_prepare_t *handle, int status)
{
  uv_timer_start(&timer, timer_cb, *(uint32_t *)timer_prepare.data, 0);
  uv_prepare_stop(&timer_prepare);
}