xref: /f-stack/freebsd/sys/seqc.h (revision 22ce4aff)
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