1/*-
  2 * Copyright (c) 2013 Andrew Turner <andrew@freebsd.org>
  3 * All rights reserved.
  4 *
  5 * Redistribution and use in source and binary forms, with or without
  6 * modification, are permitted provided that the following conditions
  7 * are met:
  8 * 1. Redistributions of source code must retain the above copyright
  9 *    notice, this list of conditions and the following disclaimer.
 10 * 2. Redistributions in binary form must reproduce the above copyright
 11 *    notice, this list of conditions and the following disclaimer in the
 12 *    documentation and/or other materials provided with the distribution.
 13 *
 14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
 15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 17 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
 18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 24 * SUCH DAMAGE.
 25 */
 26
 27#ifdef __arm__
 28#include <arm/atomic.h>
 29#else /* !__arm__ */
 30
 31#ifndef	_MACHINE_ATOMIC_H_
 32#define	_MACHINE_ATOMIC_H_
 33
 34#define	isb()		__asm __volatile("isb" : : : "memory")
 35
 36/*
 37 * Options for DMB and DSB:
 38 *	oshld	Outer Shareable, load
 39 *	oshst	Outer Shareable, store
 40 *	osh	Outer Shareable, all
 41 *	nshld	Non-shareable, load
 42 *	nshst	Non-shareable, store
 43 *	nsh	Non-shareable, all
 44 *	ishld	Inner Shareable, load
 45 *	ishst	Inner Shareable, store
 46 *	ish	Inner Shareable, all
 47 *	ld	Full system, load
 48 *	st	Full system, store
 49 *	sy	Full system, all
 50 */
 51#define	dsb(opt)	__asm __volatile("dsb " __STRING(opt) : : : "memory")
 52#define	dmb(opt)	__asm __volatile("dmb " __STRING(opt) : : : "memory")
 53
 54#define	mb()	dmb(sy)	/* Full system memory barrier all */
 55#define	wmb()	dmb(st)	/* Full system memory barrier store */
 56#define	rmb()	dmb(ld)	/* Full system memory barrier load */
 57
 58#ifdef _KERNEL
 59extern _Bool lse_supported;
 60#endif
 61
 62#if defined(SAN_NEEDS_INTERCEPTORS) && !defined(SAN_RUNTIME)
 63#include <sys/atomic_san.h>
 64#else
 65
 66#include <sys/atomic_common.h>
 67
 68#if defined(__ARM_FEATURE_ATOMICS)
 69#define	_ATOMIC_LSE_SUPPORTED	1
 70#elif defined(_KERNEL)
 71#ifdef LSE_ATOMICS
 72#define	_ATOMIC_LSE_SUPPORTED	1
 73#else
 74#define	_ATOMIC_LSE_SUPPORTED	lse_supported
 75#endif
 76#else
 77#define	_ATOMIC_LSE_SUPPORTED	0
 78#endif
 79
 80#define	_ATOMIC_OP_PROTO(t, op, bar, flav)				\
 81static __inline void							\
 82atomic_##op##_##bar##t##flav(volatile uint##t##_t *p, uint##t##_t val)
 83
 84#define	_ATOMIC_OP_IMPL(t, w, s, op, llsc_asm_op, lse_asm_op, pre, bar, a, l) \
 85_ATOMIC_OP_PROTO(t, op, bar, _llsc)					\
 86{									\
 87	uint##t##_t tmp;						\
 88	int res;							\
 89									\
 90	pre;								\
 91	__asm __volatile(						\
 92	    "1: ld"#a"xr"#s"	%"#w"0, [%2]\n"				\
 93	    "   "#llsc_asm_op"	%"#w"0, %"#w"0, %"#w"3\n"		\
 94	    "   st"#l"xr"#s"	%w1, %"#w"0, [%2]\n"			\
 95	    "   cbnz		%w1, 1b\n"				\
 96	    : "=&r"(tmp), "=&r"(res)					\
 97	    : "r" (p), "r" (val)					\
 98	    : "memory"							\
 99	);								\
100}									\
101									\
102_ATOMIC_OP_PROTO(t, op, bar, _lse)					\
103{									\
104	uint##t##_t tmp;						\
105									\
106	pre;								\
107	__asm __volatile(						\
108	    ".arch_extension lse\n"					\
109	    "ld"#lse_asm_op#a#l#s"	%"#w"2, %"#w"0, [%1]\n"		\
110	    ".arch_extension nolse\n"					\
111	    : "=r" (tmp)						\
112	    : "r" (p), "r" (val)					\
113	    : "memory"							\
114	);								\
115}									\
116									\
117_ATOMIC_OP_PROTO(t, op, bar, )						\
118{									\
119	if (_ATOMIC_LSE_SUPPORTED)					\
120		atomic_##op##_##bar##t##_lse(p, val);			\
121	else								\
122		atomic_##op##_##bar##t##_llsc(p, val);			\
123}
124
125#define	__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, bar, a, l)	\
126	_ATOMIC_OP_IMPL(8,  w, b, op, llsc_asm_op, lse_asm_op, pre,	\
127	    bar, a, l)							\
128	_ATOMIC_OP_IMPL(16, w, h, op, llsc_asm_op, lse_asm_op, pre,	\
129	    bar, a, l)							\
130	_ATOMIC_OP_IMPL(32, w,  , op, llsc_asm_op, lse_asm_op, pre,	\
131	    bar, a, l)							\
132	_ATOMIC_OP_IMPL(64,  ,  , op, llsc_asm_op, lse_asm_op, pre,	\
133	    bar, a, l)
134
135#define	_ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre)			\
136	__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre,     ,  ,  )	\
137	__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, acq_, a,  )	\
138	__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, rel_,  , l)
139
140_ATOMIC_OP(add,      add, add, )
141_ATOMIC_OP(clear,    bic, clr, )
142_ATOMIC_OP(set,      orr, set, )
143_ATOMIC_OP(subtract, add, add, val = -val)
144
145#define	_ATOMIC_CMPSET_PROTO(t, bar, flav)				\
146static __inline int							\
147atomic_cmpset_##bar##t##flav(volatile uint##t##_t *p,			\
148    uint##t##_t cmpval, uint##t##_t newval)
149
150#define	_ATOMIC_FCMPSET_PROTO(t, bar, flav)				\
151static __inline int							\
152atomic_fcmpset_##bar##t##flav(volatile uint##t##_t *p,			\
153    uint##t##_t *cmpval, uint##t##_t newval)
154
155#define	_ATOMIC_CMPSET_IMPL(t, w, s, bar, a, l)				\
156_ATOMIC_CMPSET_PROTO(t, bar, _llsc)					\
157{									\
158	uint##t##_t tmp;						\
159	int res;							\
160									\
161	__asm __volatile(						\
162	    "1: mov		%w1, #1\n"				\
163	    "   ld"#a"xr"#s"	%"#w"0, [%2]\n"				\
164	    "   cmp		%"#w"0, %"#w"3\n"			\
165	    "   b.ne		2f\n"					\
166	    "   st"#l"xr"#s"	%w1, %"#w"4, [%2]\n"			\
167	    "   cbnz		%w1, 1b\n"				\
168	    "2:"							\
169	    : "=&r"(tmp), "=&r"(res)					\
170	    : "r" (p), "r" (cmpval), "r" (newval)			\
171	    : "cc", "memory"						\
172	);								\
173									\
174	return (!res);							\
175}									\
176									\
177_ATOMIC_CMPSET_PROTO(t, bar, _lse)					\
178{									\
179	uint##t##_t oldval;						\
180	int res;							\
181									\
182	oldval = cmpval;						\
183	__asm __volatile(						\
184	    ".arch_extension lse\n"					\
185	    "cas"#a#l#s"	%"#w"1, %"#w"4, [%3]\n"			\
186	    "cmp		%"#w"1, %"#w"2\n"			\
187	    "cset		%w0, eq\n"				\
188	    ".arch_extension nolse\n"					\
189	    : "=r" (res), "+&r" (cmpval)				\
190	    : "r" (oldval), "r" (p), "r" (newval)			\
191	    : "cc", "memory"						\
192	);								\
193									\
194	return (res);							\
195}									\
196									\
197_ATOMIC_CMPSET_PROTO(t, bar, )						\
198{									\
199	if (_ATOMIC_LSE_SUPPORTED)					\
200		return (atomic_cmpset_##bar##t##_lse(p, cmpval,		\
201		    newval));						\
202	else								\
203		return (atomic_cmpset_##bar##t##_llsc(p, cmpval,	\
204		    newval));						\
205}									\
206									\
207_ATOMIC_FCMPSET_PROTO(t, bar, _llsc)					\
208{									\
209	uint##t##_t _cmpval, tmp;					\
210	int res;							\
211									\
212	_cmpval = *cmpval;						\
213	__asm __volatile(						\
214	    "   mov		%w1, #1\n"				\
215	    "   ld"#a"xr"#s"	%"#w"0, [%2]\n"				\
216	    "   cmp		%"#w"0, %"#w"3\n"			\
217	    "   b.ne		1f\n"					\
218	    "   st"#l"xr"#s"	%w1, %"#w"4, [%2]\n"			\
219	    "1:"							\
220	    : "=&r"(tmp), "=&r"(res)					\
221	    : "r" (p), "r" (_cmpval), "r" (newval)			\
222	    : "cc", "memory"						\
223	);								\
224	*cmpval = tmp;							\
225									\
226	return (!res);							\
227}									\
228									\
229_ATOMIC_FCMPSET_PROTO(t, bar, _lse)					\
230{									\
231	uint##t##_t _cmpval, tmp;					\
232	int res;							\
233									\
234	_cmpval = tmp = *cmpval;					\
235	__asm __volatile(						\
236	    ".arch_extension lse\n"					\
237	    "cas"#a#l#s"	%"#w"1, %"#w"4, [%3]\n"			\
238	    "cmp		%"#w"1, %"#w"2\n"			\
239	    "cset		%w0, eq\n"				\
240	    ".arch_extension nolse\n"					\
241	    : "=r" (res), "+&r" (tmp)					\
242	    : "r" (_cmpval), "r" (p), "r" (newval)			\
243	    : "cc", "memory"						\
244	);								\
245	*cmpval = tmp;							\
246									\
247	return (res);							\
248}									\
249									\
250_ATOMIC_FCMPSET_PROTO(t, bar, )						\
251{									\
252	if (_ATOMIC_LSE_SUPPORTED)					\
253		return (atomic_fcmpset_##bar##t##_lse(p, cmpval,	\
254		    newval));						\
255	else								\
256		return (atomic_fcmpset_##bar##t##_llsc(p, cmpval,	\
257		    newval));						\
258}
259
260#define	_ATOMIC_CMPSET(bar, a, l)					\
261	_ATOMIC_CMPSET_IMPL(8,  w, b, bar, a, l)			\
262	_ATOMIC_CMPSET_IMPL(16, w, h, bar, a, l)			\
263	_ATOMIC_CMPSET_IMPL(32, w,  , bar, a, l)			\
264	_ATOMIC_CMPSET_IMPL(64,  ,  , bar, a, l)
265
266#define	atomic_cmpset_8		atomic_cmpset_8
267#define	atomic_fcmpset_8	atomic_fcmpset_8
268#define	atomic_cmpset_16	atomic_cmpset_16
269#define	atomic_fcmpset_16	atomic_fcmpset_16
270
271_ATOMIC_CMPSET(    ,  , )
272_ATOMIC_CMPSET(acq_, a, )
273_ATOMIC_CMPSET(rel_,  ,l)
274
275#define	_ATOMIC_FETCHADD_PROTO(t, flav)					\
276static __inline uint##t##_t						\
277atomic_fetchadd_##t##flav(volatile uint##t##_t *p, uint##t##_t val)
278
279#define	_ATOMIC_FETCHADD_IMPL(t, w)					\
280_ATOMIC_FETCHADD_PROTO(t, _llsc)					\
281{									\
282	uint##t##_t ret, tmp;						\
283	int res;							\
284									\
285	__asm __volatile(						\
286	    "1: ldxr	%"#w"2, [%3]\n"					\
287	    "   add	%"#w"0, %"#w"2, %"#w"4\n"			\
288	    "   stxr	%w1, %"#w"0, [%3]\n"				\
289            "   cbnz	%w1, 1b\n"					\
290	    : "=&r" (tmp), "=&r" (res), "=&r" (ret)			\
291	    : "r" (p), "r" (val)					\
292	    : "memory"							\
293	);								\
294									\
295	return (ret);							\
296}									\
297									\
298_ATOMIC_FETCHADD_PROTO(t, _lse)						\
299{									\
300	uint##t##_t ret;						\
301									\
302	__asm __volatile(						\
303	    ".arch_extension lse\n"					\
304	    "ldadd	%"#w"2, %"#w"0, [%1]\n"				\
305	    ".arch_extension nolse\n"					\
306	    : "=r" (ret)						\
307	    : "r" (p), "r" (val)					\
308	    : "memory"							\
309	);								\
310									\
311	return (ret);							\
312}									\
313									\
314_ATOMIC_FETCHADD_PROTO(t, )						\
315{									\
316	if (_ATOMIC_LSE_SUPPORTED)					\
317		return (atomic_fetchadd_##t##_lse(p, val));		\
318	else								\
319		return (atomic_fetchadd_##t##_llsc(p, val));		\
320}
321
322_ATOMIC_FETCHADD_IMPL(32, w)
323_ATOMIC_FETCHADD_IMPL(64,  )
324
325#define	_ATOMIC_SWAP_PROTO(t, flav)					\
326static __inline uint##t##_t						\
327atomic_swap_##t##flav(volatile uint##t##_t *p, uint##t##_t val)
328
329#define	_ATOMIC_READANDCLEAR_PROTO(t, flav)				\
330static __inline uint##t##_t						\
331atomic_readandclear_##t##flav(volatile uint##t##_t *p)
332
333#define	_ATOMIC_SWAP_IMPL(t, w, zreg)					\
334_ATOMIC_SWAP_PROTO(t, _llsc)						\
335{									\
336	uint##t##_t ret;						\
337	int res;							\
338									\
339	__asm __volatile(						\
340	    "1: ldxr	%"#w"1, [%2]\n"					\
341	    "   stxr	%w0, %"#w"3, [%2]\n"				\
342            "   cbnz	%w0, 1b\n"					\
343	    : "=&r" (res), "=&r" (ret)					\
344	    : "r" (p), "r" (val)					\
345	    : "memory"							\
346	);								\
347									\
348	return (ret);							\
349}									\
350									\
351_ATOMIC_SWAP_PROTO(t, _lse)						\
352{									\
353	uint##t##_t ret;						\
354									\
355	__asm __volatile(						\
356	    ".arch_extension lse\n"					\
357	    "swp	%"#w"2, %"#w"0, [%1]\n"				\
358	    ".arch_extension nolse\n"					\
359	    : "=r" (ret)						\
360	    : "r" (p), "r" (val)					\
361	    : "memory"							\
362	);								\
363									\
364	return (ret);							\
365}									\
366									\
367_ATOMIC_SWAP_PROTO(t, )							\
368{									\
369	if (_ATOMIC_LSE_SUPPORTED)					\
370		return (atomic_swap_##t##_lse(p, val));			\
371	else								\
372		return (atomic_swap_##t##_llsc(p, val));		\
373}									\
374									\
375_ATOMIC_READANDCLEAR_PROTO(t, _llsc)					\
376{									\
377	uint##t##_t ret;						\
378	int res;							\
379									\
380	__asm __volatile(						\
381	    "1: ldxr	%"#w"1, [%2]\n"					\
382	    "   stxr	%w0, "#zreg", [%2]\n"				\
383	    "   cbnz	%w0, 1b\n"					\
384	    : "=&r" (res), "=&r" (ret)					\
385	    : "r" (p)							\
386	    : "memory"							\
387	);								\
388									\
389	return (ret);							\
390}									\
391									\
392_ATOMIC_READANDCLEAR_PROTO(t, _lse)					\
393{									\
394	return (atomic_swap_##t##_lse(p, 0));				\
395}									\
396									\
397_ATOMIC_READANDCLEAR_PROTO(t, )						\
398{									\
399	if (_ATOMIC_LSE_SUPPORTED)					\
400		return (atomic_readandclear_##t##_lse(p));		\
401	else								\
402		return (atomic_readandclear_##t##_llsc(p));		\
403}
404
405_ATOMIC_SWAP_IMPL(32, w, wzr)
406_ATOMIC_SWAP_IMPL(64,  , xzr)
407
408#define	_ATOMIC_TEST_OP_PROTO(t, op, bar, flav)				\
409static __inline int							\
410atomic_testand##op##_##bar##t##flav(volatile uint##t##_t *p, u_int val)
411
412#define	_ATOMIC_TEST_OP_IMPL(t, w, op, llsc_asm_op, lse_asm_op, bar, a)	\
413_ATOMIC_TEST_OP_PROTO(t, op, bar, _llsc)				\
414{									\
415	uint##t##_t mask, old, tmp;					\
416	int res;							\
417									\
418	mask = ((uint##t##_t)1) << (val & (t - 1));			\
419	__asm __volatile(						\
420	    "1: ld"#a"xr	%"#w"2, [%3]\n"				\
421	    "  "#llsc_asm_op"	%"#w"0, %"#w"2, %"#w"4\n"		\
422	    "   stxr		%w1, %"#w"0, [%3]\n"			\
423	    "   cbnz		%w1, 1b\n"				\
424	    : "=&r" (tmp), "=&r" (res), "=&r" (old)			\
425	    : "r" (p), "r" (mask)					\
426	    : "memory"							\
427	);								\
428									\
429	return ((old & mask) != 0);					\
430}									\
431									\
432_ATOMIC_TEST_OP_PROTO(t, op, bar, _lse)					\
433{									\
434	uint##t##_t mask, old;						\
435									\
436	mask = ((uint##t##_t)1) << (val & (t - 1));			\
437	__asm __volatile(						\
438	    ".arch_extension lse\n"					\
439	    "ld"#lse_asm_op#a"	%"#w"2, %"#w"0, [%1]\n"			\
440	    ".arch_extension nolse\n"					\
441	    : "=r" (old)						\
442	    : "r" (p), "r" (mask)					\
443	    : "memory"							\
444	);								\
445									\
446	return ((old & mask) != 0);					\
447}									\
448									\
449_ATOMIC_TEST_OP_PROTO(t, op, bar, )					\
450{									\
451	if (_ATOMIC_LSE_SUPPORTED)					\
452		return (atomic_testand##op##_##bar##t##_lse(p, val));	\
453	else								\
454		return (atomic_testand##op##_##bar##t##_llsc(p, val));	\
455}
456
457#define	_ATOMIC_TEST_OP(op, llsc_asm_op, lse_asm_op)			\
458	_ATOMIC_TEST_OP_IMPL(32, w, op, llsc_asm_op, lse_asm_op,     ,  ) \
459	_ATOMIC_TEST_OP_IMPL(32, w, op, llsc_asm_op, lse_asm_op, acq_, a) \
460	_ATOMIC_TEST_OP_IMPL(64,  , op, llsc_asm_op, lse_asm_op,     ,  ) \
461	_ATOMIC_TEST_OP_IMPL(64,  , op, llsc_asm_op, lse_asm_op, acq_, a)
462
463_ATOMIC_TEST_OP(clear, bic, clr)
464_ATOMIC_TEST_OP(set,   orr, set)
465
466#define	_ATOMIC_LOAD_ACQ_IMPL(t, w, s)					\
467static __inline uint##t##_t						\
468atomic_load_acq_##t(volatile uint##t##_t *p)				\
469{									\
470	uint##t##_t ret;						\
471									\
472	__asm __volatile(						\
473	    "ldar"#s"	%"#w"0, [%1]\n"					\
474	    : "=&r" (ret)						\
475	    : "r" (p)							\
476	    : "memory");						\
477									\
478	return (ret);							\
479}
480
481#define	atomic_load_acq_8	atomic_load_acq_8
482#define	atomic_load_acq_16	atomic_load_acq_16
483_ATOMIC_LOAD_ACQ_IMPL(8,  w, b)
484_ATOMIC_LOAD_ACQ_IMPL(16, w, h)
485_ATOMIC_LOAD_ACQ_IMPL(32, w,  )
486_ATOMIC_LOAD_ACQ_IMPL(64,  ,  )
487
488#define	_ATOMIC_STORE_REL_IMPL(t, w, s)					\
489static __inline void							\
490atomic_store_rel_##t(volatile uint##t##_t *p, uint##t##_t val)		\
491{									\
492	__asm __volatile(						\
493	    "stlr"#s"	%"#w"0, [%1]\n"					\
494	    :								\
495	    : "r" (val), "r" (p)					\
496	    : "memory");						\
497}
498
499_ATOMIC_STORE_REL_IMPL(8,  w, b)
500_ATOMIC_STORE_REL_IMPL(16, w, h)
501_ATOMIC_STORE_REL_IMPL(32, w,  )
502_ATOMIC_STORE_REL_IMPL(64,  ,  )
503
504#define	atomic_add_char			atomic_add_8
505#define	atomic_fcmpset_char		atomic_fcmpset_8
506#define	atomic_clear_char		atomic_clear_8
507#define	atomic_cmpset_char		atomic_cmpset_8
508#define	atomic_fetchadd_char		atomic_fetchadd_8
509#define	atomic_readandclear_char	atomic_readandclear_8
510#define	atomic_set_char			atomic_set_8
511#define	atomic_swap_char		atomic_swap_8
512#define	atomic_subtract_char		atomic_subtract_8
513#define	atomic_testandclear_char	atomic_testandclear_8
514#define	atomic_testandset_char		atomic_testandset_8
515
516#define	atomic_add_acq_char		atomic_add_acq_8
517#define	atomic_fcmpset_acq_char		atomic_fcmpset_acq_8
518#define	atomic_clear_acq_char		atomic_clear_acq_8
519#define	atomic_cmpset_acq_char		atomic_cmpset_acq_8
520#define	atomic_load_acq_char		atomic_load_acq_8
521#define	atomic_set_acq_char		atomic_set_acq_8
522#define	atomic_subtract_acq_char	atomic_subtract_acq_8
523#define	atomic_testandset_acq_char	atomic_testandset_acq_8
524
525#define	atomic_add_rel_char		atomic_add_rel_8
526#define	atomic_fcmpset_rel_char		atomic_fcmpset_rel_8
527#define	atomic_clear_rel_char		atomic_clear_rel_8
528#define	atomic_cmpset_rel_char		atomic_cmpset_rel_8
529#define	atomic_set_rel_char		atomic_set_rel_8
530#define	atomic_subtract_rel_char	atomic_subtract_rel_8
531#define	atomic_store_rel_char		atomic_store_rel_8
532
533#define	atomic_add_short		atomic_add_16
534#define	atomic_fcmpset_short		atomic_fcmpset_16
535#define	atomic_clear_short		atomic_clear_16
536#define	atomic_cmpset_short		atomic_cmpset_16
537#define	atomic_fetchadd_short		atomic_fetchadd_16
538#define	atomic_readandclear_short	atomic_readandclear_16
539#define	atomic_set_short		atomic_set_16
540#define	atomic_swap_short		atomic_swap_16
541#define	atomic_subtract_short		atomic_subtract_16
542#define	atomic_testandclear_short	atomic_testandclear_16
543#define	atomic_testandset_short		atomic_testandset_16
544
545#define	atomic_add_acq_short		atomic_add_acq_16
546#define	atomic_fcmpset_acq_short	atomic_fcmpset_acq_16
547#define	atomic_clear_acq_short		atomic_clear_acq_16
548#define	atomic_cmpset_acq_short		atomic_cmpset_acq_16
549#define	atomic_load_acq_short		atomic_load_acq_16
550#define	atomic_set_acq_short		atomic_set_acq_16
551#define	atomic_subtract_acq_short	atomic_subtract_acq_16
552#define	atomic_testandset_acq_short	atomic_testandset_acq_16
553
554#define	atomic_add_rel_short		atomic_add_rel_16
555#define	atomic_fcmpset_rel_short	atomic_fcmpset_rel_16
556#define	atomic_clear_rel_short		atomic_clear_rel_16
557#define	atomic_cmpset_rel_short		atomic_cmpset_rel_16
558#define	atomic_set_rel_short		atomic_set_rel_16
559#define	atomic_subtract_rel_short	atomic_subtract_rel_16
560#define	atomic_store_rel_short		atomic_store_rel_16
561
562#define	atomic_add_int			atomic_add_32
563#define	atomic_fcmpset_int		atomic_fcmpset_32
564#define	atomic_clear_int		atomic_clear_32
565#define	atomic_cmpset_int		atomic_cmpset_32
566#define	atomic_fetchadd_int		atomic_fetchadd_32
567#define	atomic_readandclear_int		atomic_readandclear_32
568#define	atomic_set_int			atomic_set_32
569#define	atomic_swap_int			atomic_swap_32
570#define	atomic_subtract_int		atomic_subtract_32
571#define	atomic_testandclear_int		atomic_testandclear_32
572#define	atomic_testandset_int		atomic_testandset_32
573
574#define	atomic_add_acq_int		atomic_add_acq_32
575#define	atomic_fcmpset_acq_int		atomic_fcmpset_acq_32
576#define	atomic_clear_acq_int		atomic_clear_acq_32
577#define	atomic_cmpset_acq_int		atomic_cmpset_acq_32
578#define	atomic_load_acq_int		atomic_load_acq_32
579#define	atomic_set_acq_int		atomic_set_acq_32
580#define	atomic_subtract_acq_int		atomic_subtract_acq_32
581#define	atomic_testandset_acq_int	atomic_testandset_acq_32
582
583#define	atomic_add_rel_int		atomic_add_rel_32
584#define	atomic_fcmpset_rel_int		atomic_fcmpset_rel_32
585#define	atomic_clear_rel_int		atomic_clear_rel_32
586#define	atomic_cmpset_rel_int		atomic_cmpset_rel_32
587#define	atomic_set_rel_int		atomic_set_rel_32
588#define	atomic_subtract_rel_int		atomic_subtract_rel_32
589#define	atomic_store_rel_int		atomic_store_rel_32
590
591#define	atomic_add_long			atomic_add_64
592#define	atomic_fcmpset_long		atomic_fcmpset_64
593#define	atomic_clear_long		atomic_clear_64
594#define	atomic_cmpset_long		atomic_cmpset_64
595#define	atomic_fetchadd_long		atomic_fetchadd_64
596#define	atomic_readandclear_long	atomic_readandclear_64
597#define	atomic_set_long			atomic_set_64
598#define	atomic_swap_long		atomic_swap_64
599#define	atomic_subtract_long		atomic_subtract_64
600#define	atomic_testandclear_long	atomic_testandclear_64
601#define	atomic_testandset_long		atomic_testandset_64
602
603#define	atomic_add_ptr			atomic_add_64
604#define	atomic_fcmpset_ptr		atomic_fcmpset_64
605#define	atomic_clear_ptr		atomic_clear_64
606#define	atomic_cmpset_ptr		atomic_cmpset_64
607#define	atomic_fetchadd_ptr		atomic_fetchadd_64
608#define	atomic_readandclear_ptr		atomic_readandclear_64
609#define	atomic_set_ptr			atomic_set_64
610#define	atomic_swap_ptr			atomic_swap_64
611#define	atomic_subtract_ptr		atomic_subtract_64
612
613#define	atomic_add_acq_long		atomic_add_acq_64
614#define	atomic_fcmpset_acq_long		atomic_fcmpset_acq_64
615#define	atomic_clear_acq_long		atomic_clear_acq_64
616#define	atomic_cmpset_acq_long		atomic_cmpset_acq_64
617#define	atomic_load_acq_long		atomic_load_acq_64
618#define	atomic_set_acq_long		atomic_set_acq_64
619#define	atomic_subtract_acq_long	atomic_subtract_acq_64
620#define	atomic_testandset_acq_long	atomic_testandset_acq_64
621
622#define	atomic_add_acq_ptr		atomic_add_acq_64
623#define	atomic_fcmpset_acq_ptr		atomic_fcmpset_acq_64
624#define	atomic_clear_acq_ptr		atomic_clear_acq_64
625#define	atomic_cmpset_acq_ptr		atomic_cmpset_acq_64
626#define	atomic_load_acq_ptr		atomic_load_acq_64
627#define	atomic_set_acq_ptr		atomic_set_acq_64
628#define	atomic_subtract_acq_ptr		atomic_subtract_acq_64
629
630#define	atomic_add_rel_long		atomic_add_rel_64
631#define	atomic_fcmpset_rel_long		atomic_fcmpset_rel_64
632#define	atomic_clear_rel_long		atomic_clear_rel_64
633#define	atomic_cmpset_rel_long		atomic_cmpset_rel_64
634#define	atomic_set_rel_long		atomic_set_rel_64
635#define	atomic_subtract_rel_long	atomic_subtract_rel_64
636#define	atomic_store_rel_long		atomic_store_rel_64
637
638#define	atomic_add_rel_ptr		atomic_add_rel_64
639#define	atomic_fcmpset_rel_ptr		atomic_fcmpset_rel_64
640#define	atomic_clear_rel_ptr		atomic_clear_rel_64
641#define	atomic_cmpset_rel_ptr		atomic_cmpset_rel_64
642#define	atomic_set_rel_ptr		atomic_set_rel_64
643#define	atomic_subtract_rel_ptr		atomic_subtract_rel_64
644#define	atomic_store_rel_ptr		atomic_store_rel_64
645
646static __inline void
647atomic_thread_fence_acq(void)
648{
649
650	dmb(ld);
651}
652
653static __inline void
654atomic_thread_fence_rel(void)
655{
656
657	dmb(sy);
658}
659
660static __inline void
661atomic_thread_fence_acq_rel(void)
662{
663
664	dmb(sy);
665}
666
667static __inline void
668atomic_thread_fence_seq_cst(void)
669{
670
671	dmb(sy);
672}
673
674#endif /* KCSAN && !KCSAN_RUNTIME */
675#endif /* _MACHINE_ATOMIC_H_ */
676
677#endif /* !__arm__ */