1*38fd1498Szrj // <atomic> compatibility -*- C++ -*- 2*38fd1498Szrj 3*38fd1498Szrj // Copyright (C) 2008-2018 Free Software Foundation, Inc. 4*38fd1498Szrj // 5*38fd1498Szrj // This file is part of the GNU ISO C++ Library. This library is free 6*38fd1498Szrj // software; you can redistribute it and/or modify it under the 7*38fd1498Szrj // terms of the GNU General Public License as published by the 8*38fd1498Szrj // Free Software Foundation; either version 3, or (at your option) 9*38fd1498Szrj // any later version. 10*38fd1498Szrj 11*38fd1498Szrj // This library is distributed in the hope that it will be useful, 12*38fd1498Szrj // but WITHOUT ANY WARRANTY; without even the implied warranty of 13*38fd1498Szrj // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 14*38fd1498Szrj // GNU General Public License for more details. 15*38fd1498Szrj 16*38fd1498Szrj // Under Section 7 of GPL version 3, you are granted additional 17*38fd1498Szrj // permissions described in the GCC Runtime Library Exception, version 18*38fd1498Szrj // 3.1, as published by the Free Software Foundation. 19*38fd1498Szrj 20*38fd1498Szrj // You should have received a copy of the GNU General Public License and 21*38fd1498Szrj // a copy of the GCC Runtime Library Exception along with this program; 22*38fd1498Szrj // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see 23*38fd1498Szrj // <http://www.gnu.org/licenses/>. 24*38fd1498Szrj 25*38fd1498Szrj #include "gstdint.h" 26*38fd1498Szrj #include <atomic> 27*38fd1498Szrj #include <mutex> 28*38fd1498Szrj 29*38fd1498Szrj // XXX GLIBCXX_ABI Deprecated 30*38fd1498Szrj // gcc-4.7.0 31*38fd1498Szrj 32*38fd1498Szrj #ifdef _GLIBCXX_SHARED 33*38fd1498Szrj 34*38fd1498Szrj #define LOGSIZE 4 35*38fd1498Szrj 36*38fd1498Szrj namespace 37*38fd1498Szrj { 38*38fd1498Szrj #if defined(_GLIBCXX_HAS_GTHREADS) && defined(_GLIBCXX_USE_C99_STDINT_TR1) 39*38fd1498Szrj std::mutex& get_atomic_mutex()40*38fd1498Szrj get_atomic_mutex() 41*38fd1498Szrj { 42*38fd1498Szrj static std::mutex atomic_mutex; 43*38fd1498Szrj return atomic_mutex; 44*38fd1498Szrj } 45*38fd1498Szrj #endif 46*38fd1498Szrj 47*38fd1498Szrj std::__atomic_flag_base flag_table[ 1 << LOGSIZE ] = 48*38fd1498Szrj { 49*38fd1498Szrj ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, 50*38fd1498Szrj ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, 51*38fd1498Szrj ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, 52*38fd1498Szrj ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, ATOMIC_FLAG_INIT, 53*38fd1498Szrj }; 54*38fd1498Szrj } // anonymous namespace 55*38fd1498Szrj 56*38fd1498Szrj namespace std _GLIBCXX_VISIBILITY(default) 57*38fd1498Szrj { 58*38fd1498Szrj _GLIBCXX_BEGIN_NAMESPACE_VERSION 59*38fd1498Szrj 60*38fd1498Szrj namespace __atomic0 61*38fd1498Szrj { 62*38fd1498Szrj 63*38fd1498Szrj struct atomic_flag : public __atomic_flag_base 64*38fd1498Szrj { 65*38fd1498Szrj bool 66*38fd1498Szrj test_and_set(memory_order) noexcept; 67*38fd1498Szrj 68*38fd1498Szrj void 69*38fd1498Szrj clear(memory_order) noexcept; 70*38fd1498Szrj }; 71*38fd1498Szrj 72*38fd1498Szrj bool test_and_set(memory_order)73*38fd1498Szrj atomic_flag::test_and_set(memory_order) noexcept 74*38fd1498Szrj { 75*38fd1498Szrj #if defined(_GLIBCXX_HAS_GTHREADS) && defined(_GLIBCXX_USE_C99_STDINT_TR1) 76*38fd1498Szrj lock_guard<mutex> __lock(get_atomic_mutex()); 77*38fd1498Szrj #endif 78*38fd1498Szrj bool result = _M_i; 79*38fd1498Szrj _M_i = true; 80*38fd1498Szrj return result; 81*38fd1498Szrj } 82*38fd1498Szrj 83*38fd1498Szrj void clear(memory_order)84*38fd1498Szrj atomic_flag::clear(memory_order) noexcept 85*38fd1498Szrj { 86*38fd1498Szrj #if defined(_GLIBCXX_HAS_GTHREADS) && defined(_GLIBCXX_USE_C99_STDINT_TR1) 87*38fd1498Szrj lock_guard<mutex> __lock(get_atomic_mutex()); 88*38fd1498Szrj #endif 89*38fd1498Szrj _M_i = false; 90*38fd1498Szrj } 91*38fd1498Szrj } // namespace __atomic0 92*38fd1498Szrj 93*38fd1498Szrj _GLIBCXX_BEGIN_EXTERN_C 94*38fd1498Szrj 95*38fd1498Szrj bool atomic_flag_test_and_set_explicit(__atomic_flag_base * __a,memory_order __m)96*38fd1498Szrj atomic_flag_test_and_set_explicit(__atomic_flag_base* __a, 97*38fd1498Szrj memory_order __m) _GLIBCXX_NOTHROW 98*38fd1498Szrj { 99*38fd1498Szrj atomic_flag* d = static_cast<atomic_flag*>(__a); 100*38fd1498Szrj return d->test_and_set(__m); 101*38fd1498Szrj } 102*38fd1498Szrj 103*38fd1498Szrj void atomic_flag_clear_explicit(__atomic_flag_base * __a,memory_order __m)104*38fd1498Szrj atomic_flag_clear_explicit(__atomic_flag_base* __a, 105*38fd1498Szrj memory_order __m) _GLIBCXX_NOTHROW 106*38fd1498Szrj { 107*38fd1498Szrj atomic_flag* d = static_cast<atomic_flag*>(__a); 108*38fd1498Szrj return d->clear(__m); 109*38fd1498Szrj } 110*38fd1498Szrj 111*38fd1498Szrj void __atomic_flag_wait_explicit(__atomic_flag_base * __a,memory_order __x)112*38fd1498Szrj __atomic_flag_wait_explicit(__atomic_flag_base* __a, 113*38fd1498Szrj memory_order __x) _GLIBCXX_NOTHROW 114*38fd1498Szrj { 115*38fd1498Szrj while (atomic_flag_test_and_set_explicit(__a, __x)) 116*38fd1498Szrj { }; 117*38fd1498Szrj } 118*38fd1498Szrj 119*38fd1498Szrj _GLIBCXX_CONST __atomic_flag_base* __atomic_flag_for_address(const volatile void * __z)120*38fd1498Szrj __atomic_flag_for_address(const volatile void* __z) _GLIBCXX_NOTHROW 121*38fd1498Szrj { 122*38fd1498Szrj uintptr_t __u = reinterpret_cast<uintptr_t>(__z); 123*38fd1498Szrj __u += (__u >> 2) + (__u << 4); 124*38fd1498Szrj __u += (__u >> 7) + (__u << 5); 125*38fd1498Szrj __u += (__u >> 17) + (__u << 13); 126*38fd1498Szrj if (sizeof(uintptr_t) > 4) 127*38fd1498Szrj __u += (__u >> 31); 128*38fd1498Szrj __u &= ~((~uintptr_t(0)) << LOGSIZE); 129*38fd1498Szrj return flag_table + __u; 130*38fd1498Szrj } 131*38fd1498Szrj 132*38fd1498Szrj _GLIBCXX_END_EXTERN_C 133*38fd1498Szrj 134*38fd1498Szrj _GLIBCXX_END_NAMESPACE_VERSION 135*38fd1498Szrj } // namespace std 136*38fd1498Szrj 137*38fd1498Szrj #endif 138*38fd1498Szrj 139*38fd1498Szrj // XXX GLIBCXX_ABI Deprecated 140*38fd1498Szrj // gcc-4.5.0 141*38fd1498Szrj // <atomic> signature changes 142*38fd1498Szrj 143*38fd1498Szrj // The rename syntax for default exported names is 144*38fd1498Szrj // asm (".symver name1,exportedname@GLIBCXX_3.4") 145*38fd1498Szrj // asm (".symver name2,exportedname@@GLIBCXX_3.4.5") 146*38fd1498Szrj // In the future, GLIBCXX_ABI > 6 should remove all uses of 147*38fd1498Szrj // _GLIBCXX_*_SYMVER macros in this file. 148*38fd1498Szrj 149*38fd1498Szrj #if defined(_GLIBCXX_SYMVER_GNU) && defined(_GLIBCXX_SHARED) \ 150*38fd1498Szrj && defined(_GLIBCXX_HAVE_AS_SYMVER_DIRECTIVE) \ 151*38fd1498Szrj && defined(_GLIBCXX_HAVE_SYMVER_SYMBOL_RENAMING_RUNTIME_SUPPORT) 152*38fd1498Szrj 153*38fd1498Szrj #define _GLIBCXX_ASM_SYMVER(cur, old, version) \ 154*38fd1498Szrj asm (".symver " #cur "," #old "@@" #version); 155*38fd1498Szrj 156*38fd1498Szrj _GLIBCXX_ASM_SYMVER(_ZNSt9__atomic011atomic_flag5clearESt12memory_order, _ZNVSt9__atomic011atomic_flag5clearESt12memory_order, GLIBCXX_3.4.11) 157*38fd1498Szrj 158*38fd1498Szrj _GLIBCXX_ASM_SYMVER(_ZNSt9__atomic011atomic_flag12test_and_setESt12memory_order, _ZNVSt9__atomic011atomic_flag12test_and_setESt12memory_order, GLIBCXX_3.4.11) 159*38fd1498Szrj 160*38fd1498Szrj #endif 161