/*
 *	BIRD Library -- Read-Copy-Update Basic Operations
 *
 *	(c) 2021 Maria Matejka <mq@jmq.cz>
 *	(c) 2021 CZ.NIC z.s.p.o.
 *
 *	Can be freely distributed and used under the terms of the GNU GPL.
 *	Note: all the relevant patents shall be expired.
 */

#ifndef _BIRD_RCU_H_
#define _BIRD_RCU_H_

#include "lib/birdlib.h"
#include "lib/lists.h"
#include <stdatomic.h>

#define RCU_GP_PHASE  0x100000
#define RCU_NEST_MASK 0x0fffff
#define RCU_NEST_CNT  0x000001

extern _Atomic uint rcu_gp_ctl;

struct rcu_birdloop {
  node n;
  _Atomic uint ctl;
};

extern _Thread_local struct rcu_birdloop *this_rcu_birdloop;

static inline void rcu_read_lock(void)
{
  uint cmp = atomic_load_explicit(&this_rcu_birdloop->ctl, memory_order_acquire);

  if (cmp & RCU_NEST_MASK)
    atomic_store_explicit(&this_rcu_birdloop->ctl, cmp + RCU_NEST_CNT, memory_order_relaxed);
  else
    atomic_store(&this_rcu_birdloop->ctl, atomic_load_explicit(&rcu_gp_ctl, memory_order_acquire));
}

static inline void rcu_read_unlock(void)
{
  atomic_fetch_sub(&this_rcu_birdloop->ctl, RCU_NEST_CNT);
}

void synchronize_rcu(void);

/* Registering and unregistering a birdloop. To be called from birdloop implementation */
void rcu_birdloop_start(struct rcu_birdloop *);
void rcu_birdloop_stop(struct rcu_birdloop *);

/* Run this from resource init */
void rcu_init(void);

#endif