2017-01-08 12:13:18 +03:00
|
|
|
#ifndef __LINUX_SEQLOCK_H
|
|
|
|
#define __LINUX_SEQLOCK_H
|
|
|
|
|
|
|
|
#include <linux/compiler.h>
|
|
|
|
|
|
|
|
typedef struct seqcount {
|
|
|
|
unsigned sequence;
|
|
|
|
} seqcount_t;
|
|
|
|
|
2018-12-20 04:34:24 +03:00
|
|
|
static inline void seqcount_init(seqcount_t *s)
|
2017-01-08 12:13:18 +03:00
|
|
|
{
|
|
|
|
s->sequence = 0;
|
|
|
|
}
|
|
|
|
|
2018-12-20 04:34:24 +03:00
|
|
|
static inline unsigned read_seqcount_begin(const seqcount_t *s)
|
2017-01-08 12:13:18 +03:00
|
|
|
{
|
|
|
|
unsigned ret;
|
|
|
|
|
|
|
|
repeat:
|
|
|
|
ret = READ_ONCE(s->sequence);
|
|
|
|
if (unlikely(ret & 1)) {
|
|
|
|
cpu_relax();
|
|
|
|
goto repeat;
|
|
|
|
}
|
|
|
|
smp_rmb();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int read_seqcount_retry(const seqcount_t *s, unsigned start)
|
|
|
|
{
|
|
|
|
smp_rmb();
|
2018-12-20 04:34:24 +03:00
|
|
|
return unlikely(s->sequence != start);
|
2017-01-08 12:13:18 +03:00
|
|
|
}
|
|
|
|
|
2018-12-20 04:34:24 +03:00
|
|
|
static inline void write_seqcount_begin(seqcount_t *s)
|
2017-01-08 12:13:18 +03:00
|
|
|
{
|
|
|
|
s->sequence++;
|
|
|
|
smp_wmb();
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void write_seqcount_end(seqcount_t *s)
|
|
|
|
{
|
|
|
|
smp_wmb();
|
2018-12-20 04:34:24 +03:00
|
|
|
s->sequence++;
|
2017-01-08 12:13:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* __LINUX_SEQLOCK_H */
|