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 59 extern _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) \ 81 static __inline void \ 82 atomic_##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) \ 146 static __inline int \ 147 atomic_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) \ 151 static __inline int \ 152 atomic_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) \ 276 static __inline uint##t##_t \ 277 atomic_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) \ 326 static __inline uint##t##_t \ 327 atomic_swap_##t##flav(volatile uint##t##_t *p, uint##t##_t val) 328 329 #define _ATOMIC_READANDCLEAR_PROTO(t, flav) \ 330 static __inline uint##t##_t \ 331 atomic_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) \ 409 static __inline int \ 410 atomic_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) \ 467 static __inline uint##t##_t \ 468 atomic_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) \ 489 static __inline void \ 490 atomic_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 646 static __inline void 647 atomic_thread_fence_acq(void) 648 { 649 650 dmb(ld); 651 } 652 653 static __inline void 654 atomic_thread_fence_rel(void) 655 { 656 657 dmb(sy); 658 } 659 660 static __inline void 661 atomic_thread_fence_acq_rel(void) 662 { 663 664 dmb(sy); 665 } 666 667 static __inline void 668 atomic_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__ */ 678