1*22ce4affSfengbojiang /*-
2*22ce4affSfengbojiang * Copyright (c) 2014 Mateusz Guzik <[email protected]>
3*22ce4affSfengbojiang *
4*22ce4affSfengbojiang * Redistribution and use in source and binary forms, with or without
5*22ce4affSfengbojiang * modification, are permitted provided that the following conditions
6*22ce4affSfengbojiang * are met:
7*22ce4affSfengbojiang * 1. Redistributions of source code must retain the above copyright
8*22ce4affSfengbojiang * notice, this list of conditions and the following disclaimer.
9*22ce4affSfengbojiang * 2. Redistributions in binary form must reproduce the above copyright
10*22ce4affSfengbojiang * notice, this list of conditions and the following disclaimer in the
11*22ce4affSfengbojiang * documentation and/or other materials provided with the distribution.
12*22ce4affSfengbojiang *
13*22ce4affSfengbojiang * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14*22ce4affSfengbojiang * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15*22ce4affSfengbojiang * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16*22ce4affSfengbojiang * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17*22ce4affSfengbojiang * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18*22ce4affSfengbojiang * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19*22ce4affSfengbojiang * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20*22ce4affSfengbojiang * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21*22ce4affSfengbojiang * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22*22ce4affSfengbojiang * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23*22ce4affSfengbojiang * SUCH DAMAGE.
24*22ce4affSfengbojiang *
25*22ce4affSfengbojiang * $FreeBSD$
26*22ce4affSfengbojiang */
27*22ce4affSfengbojiang
28*22ce4affSfengbojiang #ifndef _SYS_SEQC_H_
29*22ce4affSfengbojiang #define _SYS_SEQC_H_
30*22ce4affSfengbojiang
31*22ce4affSfengbojiang #ifdef _KERNEL
32*22ce4affSfengbojiang #include <sys/systm.h>
33*22ce4affSfengbojiang #endif
34*22ce4affSfengbojiang #include <sys/types.h>
35*22ce4affSfengbojiang
36*22ce4affSfengbojiang /*
37*22ce4affSfengbojiang * seqc_t may be included in structs visible to userspace
38*22ce4affSfengbojiang */
39*22ce4affSfengbojiang #include <sys/_seqc.h>
40*22ce4affSfengbojiang
41*22ce4affSfengbojiang #ifdef _KERNEL
42*22ce4affSfengbojiang
43*22ce4affSfengbojiang /* A hack to get MPASS macro */
44*22ce4affSfengbojiang #include <sys/lock.h>
45*22ce4affSfengbojiang
46*22ce4affSfengbojiang #include <machine/cpu.h>
47*22ce4affSfengbojiang
48*22ce4affSfengbojiang /*
49*22ce4affSfengbojiang * Predicts from inline functions are not honored by clang.
50*22ce4affSfengbojiang */
51*22ce4affSfengbojiang #define seqc_in_modify(seqc) ({ \
52*22ce4affSfengbojiang seqc_t __seqc = (seqc); \
53*22ce4affSfengbojiang \
54*22ce4affSfengbojiang __predict_false(__seqc & 1); \
55*22ce4affSfengbojiang })
56*22ce4affSfengbojiang
57*22ce4affSfengbojiang static __inline void
seqc_write_begin(seqc_t * seqcp)58*22ce4affSfengbojiang seqc_write_begin(seqc_t *seqcp)
59*22ce4affSfengbojiang {
60*22ce4affSfengbojiang
61*22ce4affSfengbojiang critical_enter();
62*22ce4affSfengbojiang MPASS(!seqc_in_modify(*seqcp));
63*22ce4affSfengbojiang *seqcp += 1;
64*22ce4affSfengbojiang atomic_thread_fence_rel();
65*22ce4affSfengbojiang }
66*22ce4affSfengbojiang
67*22ce4affSfengbojiang static __inline void
seqc_write_end(seqc_t * seqcp)68*22ce4affSfengbojiang seqc_write_end(seqc_t *seqcp)
69*22ce4affSfengbojiang {
70*22ce4affSfengbojiang
71*22ce4affSfengbojiang atomic_thread_fence_rel();
72*22ce4affSfengbojiang *seqcp += 1;
73*22ce4affSfengbojiang MPASS(!seqc_in_modify(*seqcp));
74*22ce4affSfengbojiang critical_exit();
75*22ce4affSfengbojiang }
76*22ce4affSfengbojiang
77*22ce4affSfengbojiang static __inline seqc_t
seqc_read_any(const seqc_t * seqcp)78*22ce4affSfengbojiang seqc_read_any(const seqc_t *seqcp)
79*22ce4affSfengbojiang {
80*22ce4affSfengbojiang
81*22ce4affSfengbojiang return (atomic_load_acq_int(__DECONST(seqc_t *, seqcp)));
82*22ce4affSfengbojiang }
83*22ce4affSfengbojiang
84*22ce4affSfengbojiang static __inline seqc_t
seqc_read_notmodify(const seqc_t * seqcp)85*22ce4affSfengbojiang seqc_read_notmodify(const seqc_t *seqcp)
86*22ce4affSfengbojiang {
87*22ce4affSfengbojiang
88*22ce4affSfengbojiang return (atomic_load_acq_int(__DECONST(seqc_t *, seqcp)) & ~1);
89*22ce4affSfengbojiang }
90*22ce4affSfengbojiang
91*22ce4affSfengbojiang static __inline seqc_t
seqc_read(const seqc_t * seqcp)92*22ce4affSfengbojiang seqc_read(const seqc_t *seqcp)
93*22ce4affSfengbojiang {
94*22ce4affSfengbojiang seqc_t ret;
95*22ce4affSfengbojiang
96*22ce4affSfengbojiang for (;;) {
97*22ce4affSfengbojiang ret = seqc_read_any(seqcp);
98*22ce4affSfengbojiang if (seqc_in_modify(ret)) {
99*22ce4affSfengbojiang cpu_spinwait();
100*22ce4affSfengbojiang continue;
101*22ce4affSfengbojiang }
102*22ce4affSfengbojiang break;
103*22ce4affSfengbojiang }
104*22ce4affSfengbojiang
105*22ce4affSfengbojiang return (ret);
106*22ce4affSfengbojiang }
107*22ce4affSfengbojiang
108*22ce4affSfengbojiang #define seqc_consistent_nomb(seqcp, oldseqc) ({ \
109*22ce4affSfengbojiang const seqc_t *__seqcp = (seqcp); \
110*22ce4affSfengbojiang seqc_t __oldseqc = (oldseqc); \
111*22ce4affSfengbojiang \
112*22ce4affSfengbojiang MPASS(!(seqc_in_modify(__oldseqc))); \
113*22ce4affSfengbojiang __predict_true(*__seqcp == __oldseqc); \
114*22ce4affSfengbojiang })
115*22ce4affSfengbojiang
116*22ce4affSfengbojiang #define seqc_consistent(seqcp, oldseqc) ({ \
117*22ce4affSfengbojiang atomic_thread_fence_acq(); \
118*22ce4affSfengbojiang seqc_consistent_nomb(seqcp, oldseqc); \
119*22ce4affSfengbojiang })
120*22ce4affSfengbojiang
121*22ce4affSfengbojiang /*
122*22ce4affSfengbojiang * Variant which does not critical enter/exit.
123*22ce4affSfengbojiang */
124*22ce4affSfengbojiang static __inline void
seqc_sleepable_write_begin(seqc_t * seqcp)125*22ce4affSfengbojiang seqc_sleepable_write_begin(seqc_t *seqcp)
126*22ce4affSfengbojiang {
127*22ce4affSfengbojiang
128*22ce4affSfengbojiang MPASS(!seqc_in_modify(*seqcp));
129*22ce4affSfengbojiang *seqcp += 1;
130*22ce4affSfengbojiang atomic_thread_fence_rel();
131*22ce4affSfengbojiang }
132*22ce4affSfengbojiang
133*22ce4affSfengbojiang static __inline void
seqc_sleepable_write_end(seqc_t * seqcp)134*22ce4affSfengbojiang seqc_sleepable_write_end(seqc_t *seqcp)
135*22ce4affSfengbojiang {
136*22ce4affSfengbojiang
137*22ce4affSfengbojiang atomic_thread_fence_rel();
138*22ce4affSfengbojiang *seqcp += 1;
139*22ce4affSfengbojiang MPASS(!seqc_in_modify(*seqcp));
140*22ce4affSfengbojiang }
141*22ce4affSfengbojiang
142*22ce4affSfengbojiang #endif /* _KERNEL */
143*22ce4affSfengbojiang #endif /* _SYS_SEQC_H_ */
144