2017-04-21 10:43:26 +00:00
|
|
|
/*-
|
|
|
|
* Copyright (c) 2013 Andrew Turner <andrew@freebsd.org>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* $FreeBSD$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _MACHINE_ATOMIC_H_
|
|
|
|
#define _MACHINE_ATOMIC_H_
|
|
|
|
|
|
|
|
#define isb() __asm __volatile("isb" : : : "memory")
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Options for DMB and DSB:
|
|
|
|
* oshld Outer Shareable, load
|
|
|
|
* oshst Outer Shareable, store
|
|
|
|
* osh Outer Shareable, all
|
|
|
|
* nshld Non-shareable, load
|
|
|
|
* nshst Non-shareable, store
|
|
|
|
* nsh Non-shareable, all
|
|
|
|
* ishld Inner Shareable, load
|
|
|
|
* ishst Inner Shareable, store
|
|
|
|
* ish Inner Shareable, all
|
|
|
|
* ld Full system, load
|
|
|
|
* st Full system, store
|
|
|
|
* sy Full system, all
|
|
|
|
*/
|
|
|
|
#define dsb(opt) __asm __volatile("dsb " __STRING(opt) : : : "memory")
|
|
|
|
#define dmb(opt) __asm __volatile("dmb " __STRING(opt) : : : "memory")
|
|
|
|
|
|
|
|
#define mb() dmb(sy) /* Full system memory barrier all */
|
|
|
|
#define wmb() dmb(st) /* Full system memory barrier store */
|
|
|
|
#define rmb() dmb(ld) /* Full system memory barrier load */
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#if defined(KCSAN) && !defined(KCSAN_RUNTIME)
|
|
|
|
#include <sys/_cscan_atomic.h>
|
|
|
|
#else
|
|
|
|
|
|
|
|
#include <sys/atomic_common.h>
|
|
|
|
|
|
|
|
#ifdef _KERNEL
|
|
|
|
extern bool lse_supported;
|
|
|
|
|
|
|
|
#ifdef LSE_ATOMICS
|
|
|
|
#define _ATOMIC_LSE_SUPPORTED 1
|
|
|
|
#else
|
|
|
|
#define _ATOMIC_LSE_SUPPORTED lse_supported
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define _ATOMIC_LSE_SUPPORTED 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define _ATOMIC_OP_PROTO(t, op, bar, flav) \
|
2017-04-21 10:43:26 +00:00
|
|
|
static __inline void \
|
2021-08-31 11:00:09 +00:00
|
|
|
atomic_##op##_##bar##t##flav(volatile uint##t##_t *p, uint##t##_t val)
|
|
|
|
|
|
|
|
#define _ATOMIC_OP_IMPL(t, w, s, op, llsc_asm_op, lse_asm_op, pre, bar, a, l) \
|
|
|
|
_ATOMIC_OP_PROTO(t, op, bar, _llsc) \
|
2017-04-21 10:43:26 +00:00
|
|
|
{ \
|
2021-08-31 11:00:09 +00:00
|
|
|
uint##t##_t tmp; \
|
2017-04-21 10:43:26 +00:00
|
|
|
int res; \
|
|
|
|
\
|
2021-08-31 11:00:09 +00:00
|
|
|
pre; \
|
2017-04-21 10:43:26 +00:00
|
|
|
__asm __volatile( \
|
2021-08-31 11:00:09 +00:00
|
|
|
"1: ld"#a"xr"#s" %"#w"0, [%2]\n" \
|
|
|
|
" "#llsc_asm_op" %"#w"0, %"#w"0, %"#w"3\n" \
|
|
|
|
" st"#l"xr"#s" %w1, %"#w"0, [%2]\n" \
|
|
|
|
" cbnz %w1, 1b\n" \
|
2017-04-21 10:43:26 +00:00
|
|
|
: "=&r"(tmp), "=&r"(res) \
|
|
|
|
: "r" (p), "r" (val) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
} \
|
|
|
|
\
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_OP_PROTO(t, op, bar, _lse) \
|
2017-04-21 10:43:26 +00:00
|
|
|
{ \
|
2021-08-31 11:00:09 +00:00
|
|
|
uint##t##_t tmp; \
|
2017-04-21 10:43:26 +00:00
|
|
|
\
|
2021-08-31 11:00:09 +00:00
|
|
|
pre; \
|
2017-04-21 10:43:26 +00:00
|
|
|
__asm __volatile( \
|
2021-08-31 11:00:09 +00:00
|
|
|
".arch_extension lse\n" \
|
|
|
|
"ld"#lse_asm_op#a#l#s" %"#w"2, %"#w"0, [%1]\n" \
|
|
|
|
".arch_extension nolse\n" \
|
|
|
|
: "=r" (tmp) \
|
2017-04-21 10:43:26 +00:00
|
|
|
: "r" (p), "r" (val) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
2021-08-31 11:00:09 +00:00
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_OP_PROTO(t, op, bar, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
atomic_##op##_##bar##t##_lse(p, val); \
|
|
|
|
else \
|
|
|
|
atomic_##op##_##bar##t##_llsc(p, val); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define __ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, bar, a, l) \
|
|
|
|
_ATOMIC_OP_IMPL(8, w, b, op, llsc_asm_op, lse_asm_op, pre, \
|
|
|
|
bar, a, l) \
|
|
|
|
_ATOMIC_OP_IMPL(16, w, h, op, llsc_asm_op, lse_asm_op, pre, \
|
|
|
|
bar, a, l) \
|
|
|
|
_ATOMIC_OP_IMPL(32, w, , op, llsc_asm_op, lse_asm_op, pre, \
|
|
|
|
bar, a, l) \
|
|
|
|
_ATOMIC_OP_IMPL(64, , , op, llsc_asm_op, lse_asm_op, pre, \
|
|
|
|
bar, a, l)
|
|
|
|
|
|
|
|
#define _ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre) \
|
|
|
|
__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, , , ) \
|
|
|
|
__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, acq_, a, ) \
|
|
|
|
__ATOMIC_OP(op, llsc_asm_op, lse_asm_op, pre, rel_, , l)
|
|
|
|
|
|
|
|
_ATOMIC_OP(add, add, add, )
|
|
|
|
_ATOMIC_OP(clear, bic, clr, )
|
|
|
|
_ATOMIC_OP(set, orr, set, )
|
|
|
|
_ATOMIC_OP(subtract, add, add, val = -val)
|
|
|
|
|
|
|
|
#define _ATOMIC_CMPSET_PROTO(t, bar, flav) \
|
|
|
|
static __inline int \
|
|
|
|
atomic_cmpset_##bar##t##flav(volatile uint##t##_t *p, \
|
|
|
|
uint##t##_t cmpval, uint##t##_t newval)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_FCMPSET_PROTO(t, bar, flav) \
|
2017-04-21 10:43:26 +00:00
|
|
|
static __inline int \
|
2021-08-31 11:00:09 +00:00
|
|
|
atomic_fcmpset_##bar##t##flav(volatile uint##t##_t *p, \
|
|
|
|
uint##t##_t *cmpval, uint##t##_t newval)
|
|
|
|
|
|
|
|
#define _ATOMIC_CMPSET_IMPL(t, w, s, bar, a, l) \
|
|
|
|
_ATOMIC_CMPSET_PROTO(t, bar, _llsc) \
|
2017-04-21 10:43:26 +00:00
|
|
|
{ \
|
2021-08-31 11:00:09 +00:00
|
|
|
uint##t##_t tmp; \
|
2017-04-21 10:43:26 +00:00
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
2021-08-31 11:00:09 +00:00
|
|
|
"1: mov %w1, #1\n" \
|
|
|
|
" ld"#a"xr"#s" %"#w"0, [%2]\n" \
|
|
|
|
" cmp %"#w"0, %"#w"3\n" \
|
|
|
|
" b.ne 2f\n" \
|
|
|
|
" st"#l"xr"#s" %w1, %"#w"4, [%2]\n" \
|
|
|
|
" cbnz %w1, 1b\n" \
|
2017-04-21 10:43:26 +00:00
|
|
|
"2:" \
|
|
|
|
: "=&r"(tmp), "=&r"(res) \
|
|
|
|
: "r" (p), "r" (cmpval), "r" (newval) \
|
2021-08-31 11:00:09 +00:00
|
|
|
: "cc", "memory" \
|
2017-04-21 10:43:26 +00:00
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (!res); \
|
|
|
|
} \
|
|
|
|
\
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_CMPSET_PROTO(t, bar, _lse) \
|
2017-04-21 10:43:26 +00:00
|
|
|
{ \
|
2021-08-31 11:00:09 +00:00
|
|
|
uint##t##_t oldval; \
|
2017-04-21 10:43:26 +00:00
|
|
|
int res; \
|
|
|
|
\
|
2021-08-31 11:00:09 +00:00
|
|
|
oldval = cmpval; \
|
2017-04-21 10:43:26 +00:00
|
|
|
__asm __volatile( \
|
2021-08-31 11:00:09 +00:00
|
|
|
".arch_extension lse\n" \
|
|
|
|
"cas"#a#l#s" %"#w"1, %"#w"4, [%3]\n" \
|
|
|
|
"cmp %"#w"1, %"#w"2\n" \
|
|
|
|
"cset %w0, eq\n" \
|
|
|
|
".arch_extension nolse\n" \
|
|
|
|
: "=r" (res), "+&r" (cmpval) \
|
|
|
|
: "r" (oldval), "r" (p), "r" (newval) \
|
|
|
|
: "cc", "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (res); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_CMPSET_PROTO(t, bar, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
return (atomic_cmpset_##bar##t##_lse(p, cmpval, \
|
|
|
|
newval)); \
|
|
|
|
else \
|
|
|
|
return (atomic_cmpset_##bar##t##_llsc(p, cmpval, \
|
|
|
|
newval)); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_FCMPSET_PROTO(t, bar, _llsc) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t _cmpval, tmp; \
|
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
_cmpval = *cmpval; \
|
|
|
|
__asm __volatile( \
|
|
|
|
" mov %w1, #1\n" \
|
|
|
|
" ld"#a"xr"#s" %"#w"0, [%2]\n" \
|
|
|
|
" cmp %"#w"0, %"#w"3\n" \
|
|
|
|
" b.ne 1f\n" \
|
|
|
|
" st"#l"xr"#s" %w1, %"#w"4, [%2]\n" \
|
|
|
|
"1:" \
|
2017-04-21 10:43:26 +00:00
|
|
|
: "=&r"(tmp), "=&r"(res) \
|
2021-08-31 11:00:09 +00:00
|
|
|
: "r" (p), "r" (_cmpval), "r" (newval) \
|
|
|
|
: "cc", "memory" \
|
2017-04-21 10:43:26 +00:00
|
|
|
); \
|
2021-08-31 11:00:09 +00:00
|
|
|
*cmpval = tmp; \
|
2017-04-21 10:43:26 +00:00
|
|
|
\
|
|
|
|
return (!res); \
|
2021-08-31 11:00:09 +00:00
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_FCMPSET_PROTO(t, bar, _lse) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t _cmpval, tmp; \
|
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
_cmpval = tmp = *cmpval; \
|
|
|
|
__asm __volatile( \
|
|
|
|
".arch_extension lse\n" \
|
|
|
|
"cas"#a#l#s" %"#w"1, %"#w"4, [%3]\n" \
|
|
|
|
"cmp %"#w"1, %"#w"2\n" \
|
|
|
|
"cset %w0, eq\n" \
|
|
|
|
".arch_extension nolse\n" \
|
|
|
|
: "=r" (res), "+&r" (tmp) \
|
|
|
|
: "r" (_cmpval), "r" (p), "r" (newval) \
|
|
|
|
: "cc", "memory" \
|
|
|
|
); \
|
|
|
|
*cmpval = tmp; \
|
|
|
|
\
|
|
|
|
return (res); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_FCMPSET_PROTO(t, bar, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
return (atomic_fcmpset_##bar##t##_lse(p, cmpval, \
|
|
|
|
newval)); \
|
|
|
|
else \
|
|
|
|
return (atomic_fcmpset_##bar##t##_llsc(p, cmpval, \
|
|
|
|
newval)); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_CMPSET(bar, a, l) \
|
|
|
|
_ATOMIC_CMPSET_IMPL(8, w, b, bar, a, l) \
|
|
|
|
_ATOMIC_CMPSET_IMPL(16, w, h, bar, a, l) \
|
|
|
|
_ATOMIC_CMPSET_IMPL(32, w, , bar, a, l) \
|
|
|
|
_ATOMIC_CMPSET_IMPL(64, , , bar, a, l)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_cmpset_8 atomic_cmpset_8
|
|
|
|
#define atomic_fcmpset_8 atomic_fcmpset_8
|
|
|
|
#define atomic_cmpset_16 atomic_cmpset_16
|
|
|
|
#define atomic_fcmpset_16 atomic_fcmpset_16
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_CMPSET( , , )
|
|
|
|
_ATOMIC_CMPSET(acq_, a, )
|
|
|
|
_ATOMIC_CMPSET(rel_, ,l)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_FETCHADD_PROTO(t, flav) \
|
|
|
|
static __inline uint##t##_t \
|
|
|
|
atomic_fetchadd_##t##flav(volatile uint##t##_t *p, uint##t##_t val)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_FETCHADD_IMPL(t, w) \
|
|
|
|
_ATOMIC_FETCHADD_PROTO(t, _llsc) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t ret, tmp; \
|
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
|
|
|
"1: ldxr %"#w"2, [%3]\n" \
|
|
|
|
" add %"#w"0, %"#w"2, %"#w"4\n" \
|
|
|
|
" stxr %w1, %"#w"0, [%3]\n" \
|
|
|
|
" cbnz %w1, 1b\n" \
|
|
|
|
: "=&r" (tmp), "=&r" (res), "=&r" (ret) \
|
|
|
|
: "r" (p), "r" (val) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (ret); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_FETCHADD_PROTO(t, _lse) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t ret; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
|
|
|
".arch_extension lse\n" \
|
|
|
|
"ldadd %"#w"2, %"#w"0, [%1]\n" \
|
|
|
|
".arch_extension nolse\n" \
|
|
|
|
: "=r" (ret) \
|
|
|
|
: "r" (p), "r" (val) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (ret); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_FETCHADD_PROTO(t, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
return (atomic_fetchadd_##t##_lse(p, val)); \
|
|
|
|
else \
|
|
|
|
return (atomic_fetchadd_##t##_llsc(p, val)); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_FETCHADD_IMPL(32, w)
|
|
|
|
_ATOMIC_FETCHADD_IMPL(64, )
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_SWAP_PROTO(t, flav) \
|
|
|
|
static __inline uint##t##_t \
|
|
|
|
atomic_swap_##t##flav(volatile uint##t##_t *p, uint##t##_t val)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_READANDCLEAR_PROTO(t, flav) \
|
|
|
|
static __inline uint##t##_t \
|
|
|
|
atomic_readandclear_##t##flav(volatile uint##t##_t *p)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_SWAP_IMPL(t, w, zreg) \
|
|
|
|
_ATOMIC_SWAP_PROTO(t, _llsc) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t ret; \
|
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
|
|
|
"1: ldxr %"#w"1, [%2]\n" \
|
|
|
|
" stxr %w0, %"#w"3, [%2]\n" \
|
|
|
|
" cbnz %w0, 1b\n" \
|
|
|
|
: "=&r" (res), "=&r" (ret) \
|
|
|
|
: "r" (p), "r" (val) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (ret); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_SWAP_PROTO(t, _lse) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t ret; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
|
|
|
".arch_extension lse\n" \
|
|
|
|
"swp %"#w"2, %"#w"0, [%1]\n" \
|
|
|
|
".arch_extension nolse\n" \
|
|
|
|
: "=r" (ret) \
|
|
|
|
: "r" (p), "r" (val) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (ret); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_SWAP_PROTO(t, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
return (atomic_swap_##t##_lse(p, val)); \
|
|
|
|
else \
|
|
|
|
return (atomic_swap_##t##_llsc(p, val)); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_READANDCLEAR_PROTO(t, _llsc) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t ret; \
|
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
|
|
|
"1: ldxr %"#w"1, [%2]\n" \
|
|
|
|
" stxr %w0, "#zreg", [%2]\n" \
|
|
|
|
" cbnz %w0, 1b\n" \
|
|
|
|
: "=&r" (res), "=&r" (ret) \
|
|
|
|
: "r" (p) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return (ret); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_READANDCLEAR_PROTO(t, _lse) \
|
|
|
|
{ \
|
|
|
|
return (atomic_swap_##t##_lse(p, 0)); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_READANDCLEAR_PROTO(t, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
return (atomic_readandclear_##t##_lse(p)); \
|
|
|
|
else \
|
|
|
|
return (atomic_readandclear_##t##_llsc(p)); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_SWAP_IMPL(32, w, wzr)
|
|
|
|
_ATOMIC_SWAP_IMPL(64, , xzr)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_TEST_OP_PROTO(t, op, flav) \
|
|
|
|
static __inline int \
|
|
|
|
atomic_testand##op##_##t##flav(volatile uint##t##_t *p, u_int val)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_TEST_OP_IMPL(t, w, op, llsc_asm_op, lse_asm_op) \
|
|
|
|
_ATOMIC_TEST_OP_PROTO(t, op, _llsc) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t mask, old, tmp; \
|
|
|
|
int res; \
|
|
|
|
\
|
|
|
|
mask = ((uint##t##_t)1) << (val & (t - 1)); \
|
|
|
|
__asm __volatile( \
|
|
|
|
"1: ldxr %"#w"2, [%3]\n" \
|
|
|
|
" "#llsc_asm_op" %"#w"0, %"#w"2, %"#w"4\n" \
|
|
|
|
" stxr %w1, %"#w"0, [%3]\n" \
|
|
|
|
" cbnz %w1, 1b\n" \
|
|
|
|
: "=&r" (tmp), "=&r" (res), "=&r" (old) \
|
|
|
|
: "r" (p), "r" (mask) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return ((old & mask) != 0); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_TEST_OP_PROTO(t, op, _lse) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t mask, old; \
|
|
|
|
\
|
|
|
|
mask = ((uint##t##_t)1) << (val & (t - 1)); \
|
|
|
|
__asm __volatile( \
|
|
|
|
".arch_extension lse\n" \
|
|
|
|
"ld"#lse_asm_op" %"#w"2, %"#w"0, [%1]\n" \
|
|
|
|
".arch_extension nolse\n" \
|
|
|
|
: "=r" (old) \
|
|
|
|
: "r" (p), "r" (mask) \
|
|
|
|
: "memory" \
|
|
|
|
); \
|
|
|
|
\
|
|
|
|
return ((old & mask) != 0); \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
_ATOMIC_TEST_OP_PROTO(t, op, ) \
|
|
|
|
{ \
|
|
|
|
if (_ATOMIC_LSE_SUPPORTED) \
|
|
|
|
return (atomic_testand##op##_##t##_lse(p, val)); \
|
|
|
|
else \
|
|
|
|
return (atomic_testand##op##_##t##_llsc(p, val)); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_TEST_OP(op, llsc_asm_op, lse_asm_op) \
|
|
|
|
_ATOMIC_TEST_OP_IMPL(32, w, op, llsc_asm_op, lse_asm_op) \
|
|
|
|
_ATOMIC_TEST_OP_IMPL(64, , op, llsc_asm_op, lse_asm_op)
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_TEST_OP(clear, bic, clr)
|
|
|
|
_ATOMIC_TEST_OP(set, orr, set)
|
|
|
|
|
|
|
|
#define _ATOMIC_LOAD_ACQ_IMPL(t, w, s) \
|
|
|
|
static __inline uint##t##_t \
|
|
|
|
atomic_load_acq_##t(volatile uint##t##_t *p) \
|
|
|
|
{ \
|
|
|
|
uint##t##_t ret; \
|
|
|
|
\
|
|
|
|
__asm __volatile( \
|
|
|
|
"ldar"#s" %"#w"0, [%1]\n" \
|
|
|
|
: "=&r" (ret) \
|
|
|
|
: "r" (p) \
|
|
|
|
: "memory"); \
|
|
|
|
\
|
|
|
|
return (ret); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_load_acq_8 atomic_load_acq_8
|
|
|
|
#define atomic_load_acq_16 atomic_load_acq_16
|
|
|
|
_ATOMIC_LOAD_ACQ_IMPL(8, w, b)
|
|
|
|
_ATOMIC_LOAD_ACQ_IMPL(16, w, h)
|
|
|
|
_ATOMIC_LOAD_ACQ_IMPL(32, w, )
|
|
|
|
_ATOMIC_LOAD_ACQ_IMPL(64, , )
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#define _ATOMIC_STORE_REL_IMPL(t, w, s) \
|
|
|
|
static __inline void \
|
|
|
|
atomic_store_rel_##t(volatile uint##t##_t *p, uint##t##_t val) \
|
|
|
|
{ \
|
|
|
|
__asm __volatile( \
|
|
|
|
"stlr"#s" %"#w"0, [%1]\n" \
|
|
|
|
: \
|
|
|
|
: "r" (val), "r" (p) \
|
|
|
|
: "memory"); \
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
_ATOMIC_STORE_REL_IMPL(8, w, b)
|
|
|
|
_ATOMIC_STORE_REL_IMPL(16, w, h)
|
|
|
|
_ATOMIC_STORE_REL_IMPL(32, w, )
|
|
|
|
_ATOMIC_STORE_REL_IMPL(64, , )
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
#define atomic_add_int atomic_add_32
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_int atomic_fcmpset_32
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_clear_int atomic_clear_32
|
|
|
|
#define atomic_cmpset_int atomic_cmpset_32
|
|
|
|
#define atomic_fetchadd_int atomic_fetchadd_32
|
|
|
|
#define atomic_readandclear_int atomic_readandclear_32
|
|
|
|
#define atomic_set_int atomic_set_32
|
|
|
|
#define atomic_swap_int atomic_swap_32
|
|
|
|
#define atomic_subtract_int atomic_subtract_32
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_testandclear_int atomic_testandclear_32
|
|
|
|
#define atomic_testandset_int atomic_testandset_32
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
#define atomic_add_acq_int atomic_add_acq_32
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_acq_int atomic_fcmpset_acq_32
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_clear_acq_int atomic_clear_acq_32
|
|
|
|
#define atomic_cmpset_acq_int atomic_cmpset_acq_32
|
|
|
|
#define atomic_load_acq_int atomic_load_acq_32
|
|
|
|
#define atomic_set_acq_int atomic_set_acq_32
|
|
|
|
#define atomic_subtract_acq_int atomic_subtract_acq_32
|
|
|
|
|
|
|
|
#define atomic_add_rel_int atomic_add_rel_32
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_rel_int atomic_fcmpset_rel_32
|
|
|
|
#define atomic_clear_rel_int atomic_clear_rel_32
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_cmpset_rel_int atomic_cmpset_rel_32
|
|
|
|
#define atomic_set_rel_int atomic_set_rel_32
|
|
|
|
#define atomic_subtract_rel_int atomic_subtract_rel_32
|
|
|
|
#define atomic_store_rel_int atomic_store_rel_32
|
|
|
|
|
|
|
|
#define atomic_add_long atomic_add_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_long atomic_fcmpset_64
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_clear_long atomic_clear_64
|
|
|
|
#define atomic_cmpset_long atomic_cmpset_64
|
|
|
|
#define atomic_fetchadd_long atomic_fetchadd_64
|
|
|
|
#define atomic_readandclear_long atomic_readandclear_64
|
|
|
|
#define atomic_set_long atomic_set_64
|
|
|
|
#define atomic_swap_long atomic_swap_64
|
|
|
|
#define atomic_subtract_long atomic_subtract_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_testandclear_long atomic_testandclear_64
|
|
|
|
#define atomic_testandset_long atomic_testandset_64
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
#define atomic_add_ptr atomic_add_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_ptr atomic_fcmpset_64
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_clear_ptr atomic_clear_64
|
|
|
|
#define atomic_cmpset_ptr atomic_cmpset_64
|
|
|
|
#define atomic_fetchadd_ptr atomic_fetchadd_64
|
|
|
|
#define atomic_readandclear_ptr atomic_readandclear_64
|
|
|
|
#define atomic_set_ptr atomic_set_64
|
|
|
|
#define atomic_swap_ptr atomic_swap_64
|
|
|
|
#define atomic_subtract_ptr atomic_subtract_64
|
|
|
|
|
|
|
|
#define atomic_add_acq_long atomic_add_acq_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_acq_long atomic_fcmpset_acq_64
|
|
|
|
#define atomic_clear_acq_long atomic_clear_acq_64
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_cmpset_acq_long atomic_cmpset_acq_64
|
|
|
|
#define atomic_load_acq_long atomic_load_acq_64
|
|
|
|
#define atomic_set_acq_long atomic_set_acq_64
|
|
|
|
#define atomic_subtract_acq_long atomic_subtract_acq_64
|
|
|
|
|
|
|
|
#define atomic_add_acq_ptr atomic_add_acq_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_acq_ptr atomic_fcmpset_acq_64
|
|
|
|
#define atomic_clear_acq_ptr atomic_clear_acq_64
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_cmpset_acq_ptr atomic_cmpset_acq_64
|
|
|
|
#define atomic_load_acq_ptr atomic_load_acq_64
|
|
|
|
#define atomic_set_acq_ptr atomic_set_acq_64
|
|
|
|
#define atomic_subtract_acq_ptr atomic_subtract_acq_64
|
|
|
|
|
|
|
|
#define atomic_add_rel_long atomic_add_rel_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_rel_long atomic_fcmpset_rel_64
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_clear_rel_long atomic_clear_rel_64
|
|
|
|
#define atomic_cmpset_rel_long atomic_cmpset_rel_64
|
|
|
|
#define atomic_set_rel_long atomic_set_rel_64
|
|
|
|
#define atomic_subtract_rel_long atomic_subtract_rel_64
|
|
|
|
#define atomic_store_rel_long atomic_store_rel_64
|
|
|
|
|
|
|
|
#define atomic_add_rel_ptr atomic_add_rel_64
|
2021-08-31 11:00:09 +00:00
|
|
|
#define atomic_fcmpset_rel_ptr atomic_fcmpset_rel_64
|
2017-04-21 10:43:26 +00:00
|
|
|
#define atomic_clear_rel_ptr atomic_clear_rel_64
|
|
|
|
#define atomic_cmpset_rel_ptr atomic_cmpset_rel_64
|
|
|
|
#define atomic_set_rel_ptr atomic_set_rel_64
|
|
|
|
#define atomic_subtract_rel_ptr atomic_subtract_rel_64
|
|
|
|
#define atomic_store_rel_ptr atomic_store_rel_64
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
atomic_thread_fence_acq(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
dmb(ld);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
atomic_thread_fence_rel(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
dmb(sy);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
atomic_thread_fence_acq_rel(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
dmb(sy);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
atomic_thread_fence_seq_cst(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
dmb(sy);
|
|
|
|
}
|
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#include <sys/_atomic_subword.h>
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-08-31 11:00:09 +00:00
|
|
|
#endif /* KCSAN && !KCSAN_RUNTIME */
|
|
|
|
#endif /* _MACHINE_ATOMIC_H_ */
|