Path: blob/main/contrib/llvm-project/libcxx/src/include/atomic_support.h
35230 views
//===----------------------------------------------------------------------===////1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===////78#ifndef ATOMIC_SUPPORT_H9#define ATOMIC_SUPPORT_H1011#include <__config>12#include <memory> // for __libcpp_relaxed_load1314#if defined(__clang__) && __has_builtin(__atomic_load_n) && __has_builtin(__atomic_store_n) && \15__has_builtin(__atomic_add_fetch) && __has_builtin(__atomic_exchange_n) && \16__has_builtin(__atomic_compare_exchange_n) && defined(__ATOMIC_RELAXED) && defined(__ATOMIC_CONSUME) && \17defined(__ATOMIC_ACQUIRE) && defined(__ATOMIC_RELEASE) && defined(__ATOMIC_ACQ_REL) && defined(__ATOMIC_SEQ_CST)18# define _LIBCPP_HAS_ATOMIC_BUILTINS19#elif defined(_LIBCPP_COMPILER_GCC)20# define _LIBCPP_HAS_ATOMIC_BUILTINS21#endif2223#if !defined(_LIBCPP_HAS_ATOMIC_BUILTINS) && !defined(_LIBCPP_HAS_NO_THREADS)24# if defined(_LIBCPP_WARNING)25_LIBCPP_WARNING("Building libc++ without __atomic builtins is unsupported")26# else27# warning Building libc++ without __atomic builtins is unsupported28# endif29#endif3031_LIBCPP_BEGIN_NAMESPACE_STD3233namespace {3435#if defined(_LIBCPP_HAS_ATOMIC_BUILTINS) && !defined(_LIBCPP_HAS_NO_THREADS)3637enum __libcpp_atomic_order {38_AO_Relaxed = __ATOMIC_RELAXED,39_AO_Consume = __ATOMIC_CONSUME,40_AO_Acquire = __ATOMIC_ACQUIRE,41_AO_Release = __ATOMIC_RELEASE,42_AO_Acq_Rel = __ATOMIC_ACQ_REL,43_AO_Seq = __ATOMIC_SEQ_CST44};4546template <class _ValueType, class _FromType>47inline _LIBCPP_HIDE_FROM_ABI void __libcpp_atomic_store(_ValueType* __dest, _FromType __val, int __order = _AO_Seq) {48__atomic_store_n(__dest, __val, __order);49}5051template <class _ValueType, class _FromType>52inline _LIBCPP_HIDE_FROM_ABI void __libcpp_relaxed_store(_ValueType* __dest, _FromType __val) {53__atomic_store_n(__dest, __val, _AO_Relaxed);54}5556template <class _ValueType>57inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_load(_ValueType const* __val, int __order = _AO_Seq) {58return __atomic_load_n(__val, __order);59}6061template <class _ValueType, class _AddType>62inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_add(_ValueType* __val, _AddType __a, int __order = _AO_Seq) {63return __atomic_add_fetch(__val, __a, __order);64}6566template <class _ValueType>67inline _LIBCPP_HIDE_FROM_ABI _ValueType68__libcpp_atomic_exchange(_ValueType* __target, _ValueType __value, int __order = _AO_Seq) {69return __atomic_exchange_n(__target, __value, __order);70}7172template <class _ValueType>73inline _LIBCPP_HIDE_FROM_ABI bool __libcpp_atomic_compare_exchange(74_ValueType* __val,75_ValueType* __expected,76_ValueType __after,77int __success_order = _AO_Seq,78int __fail_order = _AO_Seq) {79return __atomic_compare_exchange_n(__val, __expected, __after, true, __success_order, __fail_order);80}8182#else // _LIBCPP_HAS_NO_THREADS8384enum __libcpp_atomic_order { _AO_Relaxed, _AO_Consume, _AO_Acquire, _AO_Release, _AO_Acq_Rel, _AO_Seq };8586template <class _ValueType, class _FromType>87inline _LIBCPP_HIDE_FROM_ABI void __libcpp_atomic_store(_ValueType* __dest, _FromType __val, int = 0) {88*__dest = __val;89}9091template <class _ValueType, class _FromType>92inline _LIBCPP_HIDE_FROM_ABI void __libcpp_relaxed_store(_ValueType* __dest, _FromType __val) {93*__dest = __val;94}9596template <class _ValueType>97inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_load(_ValueType const* __val, int = 0) {98return *__val;99}100101template <class _ValueType, class _AddType>102inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_add(_ValueType* __val, _AddType __a, int = 0) {103return *__val += __a;104}105106template <class _ValueType>107inline _LIBCPP_HIDE_FROM_ABI _ValueType108__libcpp_atomic_exchange(_ValueType* __target, _ValueType __value, int = _AO_Seq) {109_ValueType old = *__target;110*__target = __value;111return old;112}113114template <class _ValueType>115inline _LIBCPP_HIDE_FROM_ABI bool116__libcpp_atomic_compare_exchange(_ValueType* __val, _ValueType* __expected, _ValueType __after, int = 0, int = 0) {117if (*__val == *__expected) {118*__val = __after;119return true;120}121*__expected = *__val;122return false;123}124125#endif // _LIBCPP_HAS_NO_THREADS126127} // end namespace128129_LIBCPP_END_NAMESPACE_STD130131#endif // ATOMIC_SUPPORT_H132133134