summaryrefslogtreecommitdiff
path: root/lib/settle.h
blob: d274599d8e1b2106d21a447dd3e977c3ffba66e4 (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
/*
 *	BIRD -- Settle timer
 *
 *	(c) 2022 Maria Matejka <mq@jmq.cz>
 *	(c) 2022 CZ.NIC z.s.p.o.
 *
 *	Can be freely distributed and used under the terms of the GNU GPL.
 */

#ifndef _BIRD_SETTLE_H_
#define _BIRD_SETTLE_H_

#include "lib/birdlib.h"
#include "lib/timer.h"

struct settle_config {
  btime min, max;
};

struct settle {
  union {
    /* Timer hook polymorphism. */
    struct {
      resource _r;
      void (*hook)(struct settle *);
    };
    timer tm;
  };
  struct settle_config cf;
  btime started;
};

STATIC_ASSERT(OFFSETOF(struct settle, hook) == OFFSETOF(struct settle, tm) + OFFSETOF(timer, hook));

#define SETTLE_INIT(_cfp, _hook, _data) (struct settle) { .tm = { .data = (_data), }, .hook = (_hook), .cf = ({ASSERT_DIE((_cfp)->min <= (_cfp)->max); *(_cfp); }), }


static inline void settle_init(struct settle *s, struct settle_config *cf, void (*hook)(struct settle *), void *data)
{
  *s = SETTLE_INIT(cf, hook, data);
}

#define settle_active(s) tm_active(&(s)->tm)

static inline void settle_kick(struct settle *s, struct birdloop *loop)
{
  if (!tm_active(&s->tm))
  {
    s->started = current_time();
    tm_set_in(&s->tm, s->started + s->cf.min, loop);
  }
  else
  {
    btime now = current_time();
    tm_set_in(&s->tm, MIN_(now + s->cf.min, s->started + s->cf.max), loop);
  }
}

static inline void settle_cancel(struct settle *s)
{
  tm_stop(&s->tm);
}

#endif