mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2024-11-08 20:28:43 +00:00
56 lines
1.3 KiB
C
56 lines
1.3 KiB
C
|
/*
|
||
|
* 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
|