/*
 *	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 = TYPE_CAST(void (*)(struct settle *), void (*)(struct timer *), (_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