Home | History | Annotate | Download | only in include
      1 // -*- C++ -*- header.
      2 
      3 // Copyright (C) 2008-2013 Free Software Foundation, Inc.
      4 //
      5 // This file is part of the GNU ISO C++ Library.  This library is free
      6 // software; you can redistribute it and/or modify it under the
      7 // terms of the GNU General Public License as published by the
      8 // Free Software Foundation; either version 3, or (at your option)
      9 // any later version.
     10 
     11 // This library is distributed in the hope that it will be useful,
     12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
     13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     14 // GNU General Public License for more details.
     15 
     16 // Under Section 7 of GPL version 3, you are granted additional
     17 // permissions described in the GCC Runtime Library Exception, version
     18 // 3.1, as published by the Free Software Foundation.
     19 
     20 // You should have received a copy of the GNU General Public License and
     21 // a copy of the GCC Runtime Library Exception along with this program;
     22 // see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
     23 // <http://www.gnu.org/licenses/>.
     24 
     25 /** @file include/atomic
     26  *  This is a Standard C++ Library header.
     27  */
     28 
     29 // Based on "C++ Atomic Types and Operations" by Hans Boehm and Lawrence Crowl.
     30 // http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2007/n2427.html
     31 
     32 #ifndef _GLIBCXX_ATOMIC
     33 #define _GLIBCXX_ATOMIC 1
     34 
     35 #pragma GCC system_header
     36 
     37 #if __cplusplus < 201103L
     38 # include <bits/c++0x_warning.h>
     39 #endif
     40 
     41 #include <bits/atomic_base.h>
     42 
     43 namespace std _GLIBCXX_VISIBILITY(default)
     44 {
     45 _GLIBCXX_BEGIN_NAMESPACE_VERSION
     46 
     47   /**
     48    * @addtogroup atomics
     49    * @{
     50    */
     51 
     52   /// atomic_bool
     53   // NB: No operators or fetch-operations for this type.
     54   struct atomic_bool
     55   {
     56   private:
     57     __atomic_base<bool>	_M_base;
     58 
     59   public:
     60     atomic_bool() noexcept = default;
     61     ~atomic_bool() noexcept = default;
     62     atomic_bool(const atomic_bool&) = delete;
     63     atomic_bool& operator=(const atomic_bool&) = delete;
     64     atomic_bool& operator=(const atomic_bool&) volatile = delete;
     65 
     66     constexpr atomic_bool(bool __i) noexcept : _M_base(__i) { }
     67 
     68     bool
     69     operator=(bool __i) noexcept
     70     { return _M_base.operator=(__i); }
     71 
     72     bool
     73     operator=(bool __i) volatile noexcept
     74     { return _M_base.operator=(__i); }
     75 
     76     operator bool() const noexcept
     77     { return _M_base.load(); }
     78 
     79     operator bool() const volatile noexcept
     80     { return _M_base.load(); }
     81 
     82     bool
     83     is_lock_free() const noexcept { return _M_base.is_lock_free(); }
     84 
     85     bool
     86     is_lock_free() const volatile noexcept { return _M_base.is_lock_free(); }
     87 
     88     void
     89     store(bool __i, memory_order __m = memory_order_seq_cst) noexcept
     90     { _M_base.store(__i, __m); }
     91 
     92     void
     93     store(bool __i, memory_order __m = memory_order_seq_cst) volatile noexcept
     94     { _M_base.store(__i, __m); }
     95 
     96     bool
     97     load(memory_order __m = memory_order_seq_cst) const noexcept
     98     { return _M_base.load(__m); }
     99 
    100     bool
    101     load(memory_order __m = memory_order_seq_cst) const volatile noexcept
    102     { return _M_base.load(__m); }
    103 
    104     bool
    105     exchange(bool __i, memory_order __m = memory_order_seq_cst) noexcept
    106     { return _M_base.exchange(__i, __m); }
    107 
    108     bool
    109     exchange(bool __i,
    110 	     memory_order __m = memory_order_seq_cst) volatile noexcept
    111     { return _M_base.exchange(__i, __m); }
    112 
    113     bool
    114     compare_exchange_weak(bool& __i1, bool __i2, memory_order __m1,
    115 			  memory_order __m2) noexcept
    116     { return _M_base.compare_exchange_weak(__i1, __i2, __m1, __m2); }
    117 
    118     bool
    119     compare_exchange_weak(bool& __i1, bool __i2, memory_order __m1,
    120 			  memory_order __m2) volatile noexcept
    121     { return _M_base.compare_exchange_weak(__i1, __i2, __m1, __m2); }
    122 
    123     bool
    124     compare_exchange_weak(bool& __i1, bool __i2,
    125 			  memory_order __m = memory_order_seq_cst) noexcept
    126     { return _M_base.compare_exchange_weak(__i1, __i2, __m); }
    127 
    128     bool
    129     compare_exchange_weak(bool& __i1, bool __i2,
    130 		     memory_order __m = memory_order_seq_cst) volatile noexcept
    131     { return _M_base.compare_exchange_weak(__i1, __i2, __m); }
    132 
    133     bool
    134     compare_exchange_strong(bool& __i1, bool __i2, memory_order __m1,
    135 			    memory_order __m2) noexcept
    136     { return _M_base.compare_exchange_strong(__i1, __i2, __m1, __m2); }
    137 
    138     bool
    139     compare_exchange_strong(bool& __i1, bool __i2, memory_order __m1,
    140 			    memory_order __m2) volatile noexcept
    141     { return _M_base.compare_exchange_strong(__i1, __i2, __m1, __m2); }
    142 
    143     bool
    144     compare_exchange_strong(bool& __i1, bool __i2,
    145 			    memory_order __m = memory_order_seq_cst) noexcept
    146     { return _M_base.compare_exchange_strong(__i1, __i2, __m); }
    147 
    148     bool
    149     compare_exchange_strong(bool& __i1, bool __i2,
    150 		    memory_order __m = memory_order_seq_cst) volatile noexcept
    151     { return _M_base.compare_exchange_strong(__i1, __i2, __m); }
    152   };
    153 
    154 
    155   /**
    156    *  @brief Generic atomic type, primary class template.
    157    *
    158    *  @tparam _Tp  Type to be made atomic, must be trivally copyable.
    159    */
    160   template<typename _Tp>
    161     struct atomic
    162     {
    163     private:
    164       _Tp _M_i;
    165 
    166     public:
    167       atomic() noexcept = default;
    168       ~atomic() noexcept = default;
    169       atomic(const atomic&) = delete;
    170       atomic& operator=(const atomic&) = delete;
    171       atomic& operator=(const atomic&) volatile = delete;
    172 
    173       constexpr atomic(_Tp __i) noexcept : _M_i(__i) { }
    174 
    175       operator _Tp() const noexcept
    176       { return load(); }
    177 
    178       operator _Tp() const volatile noexcept
    179       { return load(); }
    180 
    181       _Tp
    182       operator=(_Tp __i) noexcept 
    183       { store(__i); return __i; }
    184 
    185       _Tp
    186       operator=(_Tp __i) volatile noexcept 
    187       { store(__i); return __i; }
    188 
    189       bool
    190       is_lock_free() const noexcept
    191       { return __atomic_is_lock_free(sizeof(_M_i), nullptr); }
    192 
    193       bool
    194       is_lock_free() const volatile noexcept
    195       { return __atomic_is_lock_free(sizeof(_M_i), nullptr); }
    196 
    197       void
    198       store(_Tp __i, memory_order _m = memory_order_seq_cst) noexcept
    199       { __atomic_store(&_M_i, &__i, _m); }
    200 
    201       void
    202       store(_Tp __i, memory_order _m = memory_order_seq_cst) volatile noexcept
    203       { __atomic_store(&_M_i, &__i, _m); }
    204 
    205       _Tp
    206       load(memory_order _m = memory_order_seq_cst) const noexcept
    207       { 
    208         _Tp tmp;
    209 	__atomic_load(&_M_i, &tmp, _m); 
    210 	return tmp;
    211       }
    212 
    213       _Tp
    214       load(memory_order _m = memory_order_seq_cst) const volatile noexcept
    215       { 
    216         _Tp tmp;
    217 	__atomic_load(&_M_i, &tmp, _m); 
    218 	return tmp;
    219       }
    220 
    221       _Tp
    222       exchange(_Tp __i, memory_order _m = memory_order_seq_cst) noexcept
    223       { 
    224         _Tp tmp;
    225 	__atomic_exchange(&_M_i, &__i, &tmp, _m); 
    226 	return tmp;
    227       }
    228 
    229       _Tp
    230       exchange(_Tp __i, 
    231 	       memory_order _m = memory_order_seq_cst) volatile noexcept
    232       { 
    233         _Tp tmp;
    234 	__atomic_exchange(&_M_i, &__i, &tmp, _m); 
    235 	return tmp;
    236       }
    237 
    238       bool
    239       compare_exchange_weak(_Tp& __e, _Tp __i, memory_order __s, 
    240 			    memory_order __f) noexcept
    241       {
    242 	return __atomic_compare_exchange(&_M_i, &__e, &__i, true, __s, __f); 
    243       }
    244 
    245       bool
    246       compare_exchange_weak(_Tp& __e, _Tp __i, memory_order __s, 
    247 			    memory_order __f) volatile noexcept
    248       {
    249 	return __atomic_compare_exchange(&_M_i, &__e, &__i, true, __s, __f); 
    250       }
    251 
    252       bool
    253       compare_exchange_weak(_Tp& __e, _Tp __i,
    254 			    memory_order __m = memory_order_seq_cst) noexcept
    255       { return compare_exchange_weak(__e, __i, __m, __m); }
    256 
    257       bool
    258       compare_exchange_weak(_Tp& __e, _Tp __i,
    259 		     memory_order __m = memory_order_seq_cst) volatile noexcept
    260       { return compare_exchange_weak(__e, __i, __m, __m); }
    261 
    262       bool
    263       compare_exchange_strong(_Tp& __e, _Tp __i, memory_order __s, 
    264 			      memory_order __f) noexcept
    265       {
    266 	return __atomic_compare_exchange(&_M_i, &__e, &__i, false, __s, __f); 
    267       }
    268 
    269       bool
    270       compare_exchange_strong(_Tp& __e, _Tp __i, memory_order __s, 
    271 			      memory_order __f) volatile noexcept
    272       {
    273 	return __atomic_compare_exchange(&_M_i, &__e, &__i, false, __s, __f); 
    274       }
    275 
    276       bool
    277       compare_exchange_strong(_Tp& __e, _Tp __i,
    278 			       memory_order __m = memory_order_seq_cst) noexcept
    279       { return compare_exchange_strong(__e, __i, __m, __m); }
    280 
    281       bool
    282       compare_exchange_strong(_Tp& __e, _Tp __i,
    283 		     memory_order __m = memory_order_seq_cst) volatile noexcept
    284       { return compare_exchange_strong(__e, __i, __m, __m); }
    285     };
    286 
    287 
    288   /// Partial specialization for pointer types.
    289   template<typename _Tp>
    290     struct atomic<_Tp*>
    291     {
    292       typedef _Tp* 			__pointer_type;
    293       typedef __atomic_base<_Tp*>	__base_type;
    294       __base_type			_M_b;
    295 
    296       atomic() noexcept = default;
    297       ~atomic() noexcept = default;
    298       atomic(const atomic&) = delete;
    299       atomic& operator=(const atomic&) = delete;
    300       atomic& operator=(const atomic&) volatile = delete;
    301 
    302       constexpr atomic(__pointer_type __p) noexcept : _M_b(__p) { }
    303 
    304       operator __pointer_type() const noexcept
    305       { return __pointer_type(_M_b); }
    306 
    307       operator __pointer_type() const volatile noexcept
    308       { return __pointer_type(_M_b); }
    309 
    310       __pointer_type
    311       operator=(__pointer_type __p) noexcept
    312       { return _M_b.operator=(__p); }
    313 
    314       __pointer_type
    315       operator=(__pointer_type __p) volatile noexcept
    316       { return _M_b.operator=(__p); }
    317 
    318       __pointer_type
    319       operator++(int) noexcept
    320       { return _M_b++; }
    321 
    322       __pointer_type
    323       operator++(int) volatile noexcept
    324       { return _M_b++; }
    325 
    326       __pointer_type
    327       operator--(int) noexcept
    328       { return _M_b--; }
    329 
    330       __pointer_type
    331       operator--(int) volatile noexcept
    332       { return _M_b--; }
    333 
    334       __pointer_type
    335       operator++() noexcept
    336       { return ++_M_b; }
    337 
    338       __pointer_type
    339       operator++() volatile noexcept
    340       { return ++_M_b; }
    341 
    342       __pointer_type
    343       operator--() noexcept
    344       { return --_M_b; }
    345 
    346       __pointer_type
    347       operator--() volatile noexcept
    348       { return --_M_b; }
    349 
    350       __pointer_type
    351       operator+=(ptrdiff_t __d) noexcept
    352       { return _M_b.operator+=(__d); }
    353 
    354       __pointer_type
    355       operator+=(ptrdiff_t __d) volatile noexcept
    356       { return _M_b.operator+=(__d); }
    357 
    358       __pointer_type
    359       operator-=(ptrdiff_t __d) noexcept
    360       { return _M_b.operator-=(__d); }
    361 
    362       __pointer_type
    363       operator-=(ptrdiff_t __d) volatile noexcept
    364       { return _M_b.operator-=(__d); }
    365 
    366       bool
    367       is_lock_free() const noexcept
    368       { return _M_b.is_lock_free(); }
    369 
    370       bool
    371       is_lock_free() const volatile noexcept
    372       { return _M_b.is_lock_free(); }
    373 
    374       void
    375       store(__pointer_type __p,
    376 	    memory_order __m = memory_order_seq_cst) noexcept
    377       { return _M_b.store(__p, __m); }
    378 
    379       void
    380       store(__pointer_type __p,
    381 	    memory_order __m = memory_order_seq_cst) volatile noexcept
    382       { return _M_b.store(__p, __m); }
    383 
    384       __pointer_type
    385       load(memory_order __m = memory_order_seq_cst) const noexcept
    386       { return _M_b.load(__m); }
    387 
    388       __pointer_type
    389       load(memory_order __m = memory_order_seq_cst) const volatile noexcept
    390       { return _M_b.load(__m); }
    391 
    392       __pointer_type
    393       exchange(__pointer_type __p,
    394 	       memory_order __m = memory_order_seq_cst) noexcept
    395       { return _M_b.exchange(__p, __m); }
    396 
    397       __pointer_type
    398       exchange(__pointer_type __p,
    399 	       memory_order __m = memory_order_seq_cst) volatile noexcept
    400       { return _M_b.exchange(__p, __m); }
    401 
    402       bool
    403       compare_exchange_weak(__pointer_type& __p1, __pointer_type __p2,
    404 			    memory_order __m1, memory_order __m2) noexcept
    405       { return _M_b.compare_exchange_strong(__p1, __p2, __m1, __m2); }
    406 
    407       bool
    408       compare_exchange_weak(__pointer_type& __p1, __pointer_type __p2,
    409 			    memory_order __m1,
    410 			    memory_order __m2) volatile noexcept
    411       { return _M_b.compare_exchange_strong(__p1, __p2, __m1, __m2); }
    412 
    413       bool
    414       compare_exchange_weak(__pointer_type& __p1, __pointer_type __p2,
    415 			    memory_order __m = memory_order_seq_cst) noexcept
    416       {
    417 	return compare_exchange_weak(__p1, __p2, __m,
    418 				     __cmpexch_failure_order(__m));
    419       }
    420 
    421       bool
    422       compare_exchange_weak(__pointer_type& __p1, __pointer_type __p2,
    423 		    memory_order __m = memory_order_seq_cst) volatile noexcept
    424       {
    425 	return compare_exchange_weak(__p1, __p2, __m,
    426 				     __cmpexch_failure_order(__m));
    427       }
    428 
    429       bool
    430       compare_exchange_strong(__pointer_type& __p1, __pointer_type __p2,
    431 			      memory_order __m1, memory_order __m2) noexcept
    432       { return _M_b.compare_exchange_strong(__p1, __p2, __m1, __m2); }
    433 
    434       bool
    435       compare_exchange_strong(__pointer_type& __p1, __pointer_type __p2,
    436 			      memory_order __m1,
    437 			      memory_order __m2) volatile noexcept
    438       { return _M_b.compare_exchange_strong(__p1, __p2, __m1, __m2); }
    439 
    440       bool
    441       compare_exchange_strong(__pointer_type& __p1, __pointer_type __p2,
    442 			      memory_order __m = memory_order_seq_cst) noexcept
    443       {
    444 	return _M_b.compare_exchange_strong(__p1, __p2, __m,
    445 					    __cmpexch_failure_order(__m));
    446       }
    447 
    448       bool
    449       compare_exchange_strong(__pointer_type& __p1, __pointer_type __p2,
    450 		    memory_order __m = memory_order_seq_cst) volatile noexcept
    451       {
    452 	return _M_b.compare_exchange_strong(__p1, __p2, __m,
    453 					    __cmpexch_failure_order(__m));
    454       }
    455 
    456       __pointer_type
    457       fetch_add(ptrdiff_t __d,
    458 		memory_order __m = memory_order_seq_cst) noexcept
    459       { return _M_b.fetch_add(__d, __m); }
    460 
    461       __pointer_type
    462       fetch_add(ptrdiff_t __d,
    463 		memory_order __m = memory_order_seq_cst) volatile noexcept
    464       { return _M_b.fetch_add(__d, __m); }
    465 
    466       __pointer_type
    467       fetch_sub(ptrdiff_t __d,
    468 		memory_order __m = memory_order_seq_cst) noexcept
    469       { return _M_b.fetch_sub(__d, __m); }
    470 
    471       __pointer_type
    472       fetch_sub(ptrdiff_t __d,
    473 		memory_order __m = memory_order_seq_cst) volatile noexcept
    474       { return _M_b.fetch_sub(__d, __m); }
    475     };
    476 
    477 
    478   /// Explicit specialization for bool.
    479   template<>
    480     struct atomic<bool> : public atomic_bool
    481     {
    482       typedef bool 			__integral_type;
    483       typedef atomic_bool 		__base_type;
    484 
    485       atomic() noexcept = default;
    486       ~atomic() noexcept = default;
    487       atomic(const atomic&) = delete;
    488       atomic& operator=(const atomic&) = delete;
    489       atomic& operator=(const atomic&) volatile = delete;
    490 
    491       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    492 
    493       using __base_type::operator __integral_type;
    494       using __base_type::operator=;
    495     };
    496 
    497   /// Explicit specialization for char.
    498   template<>
    499     struct atomic<char> : public atomic_char
    500     {
    501       typedef char 			__integral_type;
    502       typedef atomic_char 		__base_type;
    503 
    504       atomic() noexcept = default;
    505       ~atomic() noexcept = default;
    506       atomic(const atomic&) = delete;
    507       atomic& operator=(const atomic&) = delete;
    508       atomic& operator=(const atomic&) volatile = delete;
    509 
    510       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    511 
    512       using __base_type::operator __integral_type;
    513       using __base_type::operator=;
    514     };
    515 
    516   /// Explicit specialization for signed char.
    517   template<>
    518     struct atomic<signed char> : public atomic_schar
    519     {
    520       typedef signed char 		__integral_type;
    521       typedef atomic_schar 		__base_type;
    522 
    523       atomic() noexcept= default;
    524       ~atomic() noexcept = default;
    525       atomic(const atomic&) = delete;
    526       atomic& operator=(const atomic&) = delete;
    527       atomic& operator=(const atomic&) volatile = delete;
    528 
    529       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    530 
    531       using __base_type::operator __integral_type;
    532       using __base_type::operator=;
    533     };
    534 
    535   /// Explicit specialization for unsigned char.
    536   template<>
    537     struct atomic<unsigned char> : public atomic_uchar
    538     {
    539       typedef unsigned char 		__integral_type;
    540       typedef atomic_uchar 		__base_type;
    541 
    542       atomic() noexcept= default;
    543       ~atomic() noexcept = default;
    544       atomic(const atomic&) = delete;
    545       atomic& operator=(const atomic&) = delete;
    546       atomic& operator=(const atomic&) volatile = delete;
    547 
    548       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    549 
    550       using __base_type::operator __integral_type;
    551       using __base_type::operator=;
    552     };
    553 
    554   /// Explicit specialization for short.
    555   template<>
    556     struct atomic<short> : public atomic_short
    557     {
    558       typedef short 			__integral_type;
    559       typedef atomic_short 		__base_type;
    560 
    561       atomic() noexcept = default;
    562       ~atomic() noexcept = default;
    563       atomic(const atomic&) = delete;
    564       atomic& operator=(const atomic&) = delete;
    565       atomic& operator=(const atomic&) volatile = delete;
    566 
    567       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    568 
    569       using __base_type::operator __integral_type;
    570       using __base_type::operator=;
    571     };
    572 
    573   /// Explicit specialization for unsigned short.
    574   template<>
    575     struct atomic<unsigned short> : public atomic_ushort
    576     {
    577       typedef unsigned short 	      	__integral_type;
    578       typedef atomic_ushort 		__base_type;
    579 
    580       atomic() noexcept = default;
    581       ~atomic() noexcept = default;
    582       atomic(const atomic&) = delete;
    583       atomic& operator=(const atomic&) = delete;
    584       atomic& operator=(const atomic&) volatile = delete;
    585 
    586       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    587 
    588       using __base_type::operator __integral_type;
    589       using __base_type::operator=;
    590     };
    591 
    592   /// Explicit specialization for int.
    593   template<>
    594     struct atomic<int> : atomic_int
    595     {
    596       typedef int 			__integral_type;
    597       typedef atomic_int 		__base_type;
    598 
    599       atomic() noexcept = default;
    600       ~atomic() noexcept = default;
    601       atomic(const atomic&) = delete;
    602       atomic& operator=(const atomic&) = delete;
    603       atomic& operator=(const atomic&) volatile = delete;
    604 
    605       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    606 
    607       using __base_type::operator __integral_type;
    608       using __base_type::operator=;
    609     };
    610 
    611   /// Explicit specialization for unsigned int.
    612   template<>
    613     struct atomic<unsigned int> : public atomic_uint
    614     {
    615       typedef unsigned int		__integral_type;
    616       typedef atomic_uint 		__base_type;
    617 
    618       atomic() noexcept = default;
    619       ~atomic() noexcept = default;
    620       atomic(const atomic&) = delete;
    621       atomic& operator=(const atomic&) = delete;
    622       atomic& operator=(const atomic&) volatile = delete;
    623 
    624       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    625 
    626       using __base_type::operator __integral_type;
    627       using __base_type::operator=;
    628     };
    629 
    630   /// Explicit specialization for long.
    631   template<>
    632     struct atomic<long> : public atomic_long
    633     {
    634       typedef long 			__integral_type;
    635       typedef atomic_long 		__base_type;
    636 
    637       atomic() noexcept = default;
    638       ~atomic() noexcept = default;
    639       atomic(const atomic&) = delete;
    640       atomic& operator=(const atomic&) = delete;
    641       atomic& operator=(const atomic&) volatile = delete;
    642 
    643       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    644 
    645       using __base_type::operator __integral_type;
    646       using __base_type::operator=;
    647     };
    648 
    649   /// Explicit specialization for unsigned long.
    650   template<>
    651     struct atomic<unsigned long> : public atomic_ulong
    652     {
    653       typedef unsigned long 		__integral_type;
    654       typedef atomic_ulong 		__base_type;
    655 
    656       atomic() noexcept = default;
    657       ~atomic() noexcept = default;
    658       atomic(const atomic&) = delete;
    659       atomic& operator=(const atomic&) = delete;
    660       atomic& operator=(const atomic&) volatile = delete;
    661 
    662       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    663 
    664       using __base_type::operator __integral_type;
    665       using __base_type::operator=;
    666     };
    667 
    668   /// Explicit specialization for long long.
    669   template<>
    670     struct atomic<long long> : public atomic_llong
    671     {
    672       typedef long long 		__integral_type;
    673       typedef atomic_llong 		__base_type;
    674 
    675       atomic() noexcept = default;
    676       ~atomic() noexcept = default;
    677       atomic(const atomic&) = delete;
    678       atomic& operator=(const atomic&) = delete;
    679       atomic& operator=(const atomic&) volatile = delete;
    680 
    681       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    682 
    683       using __base_type::operator __integral_type;
    684       using __base_type::operator=;
    685     };
    686 
    687   /// Explicit specialization for unsigned long long.
    688   template<>
    689     struct atomic<unsigned long long> : public atomic_ullong
    690     {
    691       typedef unsigned long long       	__integral_type;
    692       typedef atomic_ullong 		__base_type;
    693 
    694       atomic() noexcept = default;
    695       ~atomic() noexcept = default;
    696       atomic(const atomic&) = delete;
    697       atomic& operator=(const atomic&) = delete;
    698       atomic& operator=(const atomic&) volatile = delete;
    699 
    700       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    701 
    702       using __base_type::operator __integral_type;
    703       using __base_type::operator=;
    704     };
    705 
    706   /// Explicit specialization for wchar_t.
    707   template<>
    708     struct atomic<wchar_t> : public atomic_wchar_t
    709     {
    710       typedef wchar_t 			__integral_type;
    711       typedef atomic_wchar_t 		__base_type;
    712 
    713       atomic() noexcept = default;
    714       ~atomic() noexcept = default;
    715       atomic(const atomic&) = delete;
    716       atomic& operator=(const atomic&) = delete;
    717       atomic& operator=(const atomic&) volatile = delete;
    718 
    719       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    720 
    721       using __base_type::operator __integral_type;
    722       using __base_type::operator=;
    723     };
    724 
    725   /// Explicit specialization for char16_t.
    726   template<>
    727     struct atomic<char16_t> : public atomic_char16_t
    728     {
    729       typedef char16_t 			__integral_type;
    730       typedef atomic_char16_t 		__base_type;
    731 
    732       atomic() noexcept = default;
    733       ~atomic() noexcept = default;
    734       atomic(const atomic&) = delete;
    735       atomic& operator=(const atomic&) = delete;
    736       atomic& operator=(const atomic&) volatile = delete;
    737 
    738       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    739 
    740       using __base_type::operator __integral_type;
    741       using __base_type::operator=;
    742     };
    743 
    744   /// Explicit specialization for char32_t.
    745   template<>
    746     struct atomic<char32_t> : public atomic_char32_t
    747     {
    748       typedef char32_t 			__integral_type;
    749       typedef atomic_char32_t 		__base_type;
    750 
    751       atomic() noexcept = default;
    752       ~atomic() noexcept = default;
    753       atomic(const atomic&) = delete;
    754       atomic& operator=(const atomic&) = delete;
    755       atomic& operator=(const atomic&) volatile = delete;
    756 
    757       constexpr atomic(__integral_type __i) noexcept : __base_type(__i) { }
    758 
    759       using __base_type::operator __integral_type;
    760       using __base_type::operator=;
    761     };
    762 
    763 
    764   // Function definitions, atomic_flag operations.
    765   inline bool
    766   atomic_flag_test_and_set_explicit(atomic_flag* __a,
    767 				    memory_order __m) noexcept
    768   { return __a->test_and_set(__m); }
    769 
    770   inline bool
    771   atomic_flag_test_and_set_explicit(volatile atomic_flag* __a,
    772 				    memory_order __m) noexcept
    773   { return __a->test_and_set(__m); }
    774 
    775   inline void
    776   atomic_flag_clear_explicit(atomic_flag* __a, memory_order __m) noexcept
    777   { __a->clear(__m); }
    778 
    779   inline void
    780   atomic_flag_clear_explicit(volatile atomic_flag* __a,
    781 			     memory_order __m) noexcept
    782   { __a->clear(__m); }
    783 
    784   inline bool
    785   atomic_flag_test_and_set(atomic_flag* __a) noexcept
    786   { return atomic_flag_test_and_set_explicit(__a, memory_order_seq_cst); }
    787 
    788   inline bool
    789   atomic_flag_test_and_set(volatile atomic_flag* __a) noexcept
    790   { return atomic_flag_test_and_set_explicit(__a, memory_order_seq_cst); }
    791 
    792   inline void
    793   atomic_flag_clear(atomic_flag* __a) noexcept
    794   { atomic_flag_clear_explicit(__a, memory_order_seq_cst); }
    795 
    796   inline void
    797   atomic_flag_clear(volatile atomic_flag* __a) noexcept
    798   { atomic_flag_clear_explicit(__a, memory_order_seq_cst); }
    799 
    800 
    801   // Function templates generally applicable to atomic types.
    802   template<typename _ITp>
    803     inline bool
    804     atomic_is_lock_free(const atomic<_ITp>* __a) noexcept
    805     { return __a->is_lock_free(); }
    806 
    807   template<typename _ITp>
    808     inline bool
    809     atomic_is_lock_free(const volatile atomic<_ITp>* __a) noexcept
    810     { return __a->is_lock_free(); }
    811 
    812   template<typename _ITp>
    813     inline void
    814     atomic_init(atomic<_ITp>* __a, _ITp __i) noexcept;
    815 
    816   template<typename _ITp>
    817     inline void
    818     atomic_init(volatile atomic<_ITp>* __a, _ITp __i) noexcept;
    819 
    820   template<typename _ITp>
    821     inline void
    822     atomic_store_explicit(atomic<_ITp>* __a, _ITp __i,
    823 			  memory_order __m) noexcept
    824     { __a->store(__i, __m); }
    825 
    826   template<typename _ITp>
    827     inline void
    828     atomic_store_explicit(volatile atomic<_ITp>* __a, _ITp __i,
    829 			  memory_order __m) noexcept
    830     { __a->store(__i, __m); }
    831 
    832   template<typename _ITp>
    833     inline _ITp
    834     atomic_load_explicit(const atomic<_ITp>* __a, memory_order __m) noexcept
    835     { return __a->load(__m); }
    836 
    837   template<typename _ITp>
    838     inline _ITp
    839     atomic_load_explicit(const volatile atomic<_ITp>* __a,
    840 			 memory_order __m) noexcept
    841     { return __a->load(__m); }
    842 
    843   template<typename _ITp>
    844     inline _ITp
    845     atomic_exchange_explicit(atomic<_ITp>* __a, _ITp __i,
    846 			     memory_order __m) noexcept
    847     { return __a->exchange(__i, __m); }
    848 
    849   template<typename _ITp>
    850     inline _ITp
    851     atomic_exchange_explicit(volatile atomic<_ITp>* __a, _ITp __i,
    852 			     memory_order __m) noexcept
    853     { return __a->exchange(__i, __m); }
    854 
    855   template<typename _ITp>
    856     inline bool
    857     atomic_compare_exchange_weak_explicit(atomic<_ITp>* __a,
    858 					  _ITp* __i1, _ITp __i2,
    859 					  memory_order __m1,
    860 					  memory_order __m2) noexcept
    861     { return __a->compare_exchange_weak(*__i1, __i2, __m1, __m2); }
    862 
    863   template<typename _ITp>
    864     inline bool
    865     atomic_compare_exchange_weak_explicit(volatile atomic<_ITp>* __a,
    866 					  _ITp* __i1, _ITp __i2,
    867 					  memory_order __m1,
    868 					  memory_order __m2) noexcept
    869     { return __a->compare_exchange_weak(*__i1, __i2, __m1, __m2); }
    870 
    871   template<typename _ITp>
    872     inline bool
    873     atomic_compare_exchange_strong_explicit(atomic<_ITp>* __a,
    874 					    _ITp* __i1, _ITp __i2,
    875 					    memory_order __m1,
    876 					    memory_order __m2) noexcept
    877     { return __a->compare_exchange_strong(*__i1, __i2, __m1, __m2); }
    878 
    879   template<typename _ITp>
    880     inline bool
    881     atomic_compare_exchange_strong_explicit(volatile atomic<_ITp>* __a,
    882 					    _ITp* __i1, _ITp __i2,
    883 					    memory_order __m1,
    884 					    memory_order __m2) noexcept
    885     { return __a->compare_exchange_strong(*__i1, __i2, __m1, __m2); }
    886 
    887 
    888   template<typename _ITp>
    889     inline void
    890     atomic_store(atomic<_ITp>* __a, _ITp __i) noexcept
    891     { atomic_store_explicit(__a, __i, memory_order_seq_cst); }
    892 
    893   template<typename _ITp>
    894     inline void
    895     atomic_store(volatile atomic<_ITp>* __a, _ITp __i) noexcept
    896     { atomic_store_explicit(__a, __i, memory_order_seq_cst); }
    897 
    898   template<typename _ITp>
    899     inline _ITp
    900     atomic_load(const atomic<_ITp>* __a) noexcept
    901     { return atomic_load_explicit(__a, memory_order_seq_cst); }
    902 
    903   template<typename _ITp>
    904     inline _ITp
    905     atomic_load(const volatile atomic<_ITp>* __a) noexcept
    906     { return atomic_load_explicit(__a, memory_order_seq_cst); }
    907 
    908   template<typename _ITp>
    909     inline _ITp
    910     atomic_exchange(atomic<_ITp>* __a, _ITp __i) noexcept
    911     { return atomic_exchange_explicit(__a, __i, memory_order_seq_cst); }
    912 
    913   template<typename _ITp>
    914     inline _ITp
    915     atomic_exchange(volatile atomic<_ITp>* __a, _ITp __i) noexcept
    916     { return atomic_exchange_explicit(__a, __i, memory_order_seq_cst); }
    917 
    918   template<typename _ITp>
    919     inline bool
    920     atomic_compare_exchange_weak(atomic<_ITp>* __a,
    921 				 _ITp* __i1, _ITp __i2) noexcept
    922     {
    923       return atomic_compare_exchange_weak_explicit(__a, __i1, __i2,
    924 						   memory_order_seq_cst,
    925 						   memory_order_seq_cst);
    926     }
    927 
    928   template<typename _ITp>
    929     inline bool
    930     atomic_compare_exchange_weak(volatile atomic<_ITp>* __a,
    931 				 _ITp* __i1, _ITp __i2) noexcept
    932     {
    933       return atomic_compare_exchange_weak_explicit(__a, __i1, __i2,
    934 						   memory_order_seq_cst,
    935 						   memory_order_seq_cst);
    936     }
    937 
    938   template<typename _ITp>
    939     inline bool
    940     atomic_compare_exchange_strong(atomic<_ITp>* __a,
    941 				   _ITp* __i1, _ITp __i2) noexcept
    942     {
    943       return atomic_compare_exchange_strong_explicit(__a, __i1, __i2,
    944 						     memory_order_seq_cst,
    945 						     memory_order_seq_cst);
    946     }
    947 
    948   template<typename _ITp>
    949     inline bool
    950     atomic_compare_exchange_strong(volatile atomic<_ITp>* __a,
    951 				   _ITp* __i1, _ITp __i2) noexcept
    952     {
    953       return atomic_compare_exchange_strong_explicit(__a, __i1, __i2,
    954 						     memory_order_seq_cst,
    955 						     memory_order_seq_cst);
    956     }
    957 
    958   // Function templates for atomic_integral operations only, using
    959   // __atomic_base. Template argument should be constricted to
    960   // intergral types as specified in the standard, excluding address
    961   // types.
    962   template<typename _ITp>
    963     inline _ITp
    964     atomic_fetch_add_explicit(__atomic_base<_ITp>* __a, _ITp __i,
    965 			      memory_order __m) noexcept
    966     { return __a->fetch_add(__i, __m); }
    967 
    968   template<typename _ITp>
    969     inline _ITp
    970     atomic_fetch_add_explicit(volatile __atomic_base<_ITp>* __a, _ITp __i,
    971 			      memory_order __m) noexcept
    972     { return __a->fetch_add(__i, __m); }
    973 
    974   template<typename _ITp>
    975     inline _ITp
    976     atomic_fetch_sub_explicit(__atomic_base<_ITp>* __a, _ITp __i,
    977 			      memory_order __m) noexcept
    978     { return __a->fetch_sub(__i, __m); }
    979 
    980   template<typename _ITp>
    981     inline _ITp
    982     atomic_fetch_sub_explicit(volatile __atomic_base<_ITp>* __a, _ITp __i,
    983 			      memory_order __m) noexcept
    984     { return __a->fetch_sub(__i, __m); }
    985 
    986   template<typename _ITp>
    987     inline _ITp
    988     atomic_fetch_and_explicit(__atomic_base<_ITp>* __a, _ITp __i,
    989 			      memory_order __m) noexcept
    990     { return __a->fetch_and(__i, __m); }
    991 
    992   template<typename _ITp>
    993     inline _ITp
    994     atomic_fetch_and_explicit(volatile __atomic_base<_ITp>* __a, _ITp __i,
    995 			      memory_order __m) noexcept
    996     { return __a->fetch_and(__i, __m); }
    997 
    998   template<typename _ITp>
    999     inline _ITp
   1000     atomic_fetch_or_explicit(__atomic_base<_ITp>* __a, _ITp __i,
   1001 			     memory_order __m) noexcept
   1002     { return __a->fetch_or(__i, __m); }
   1003 
   1004   template<typename _ITp>
   1005     inline _ITp
   1006     atomic_fetch_or_explicit(volatile __atomic_base<_ITp>* __a, _ITp __i,
   1007 			     memory_order __m) noexcept
   1008     { return __a->fetch_or(__i, __m); }
   1009 
   1010   template<typename _ITp>
   1011     inline _ITp
   1012     atomic_fetch_xor_explicit(__atomic_base<_ITp>* __a, _ITp __i,
   1013 			      memory_order __m) noexcept
   1014     { return __a->fetch_xor(__i, __m); }
   1015 
   1016   template<typename _ITp>
   1017     inline _ITp
   1018     atomic_fetch_xor_explicit(volatile __atomic_base<_ITp>* __a, _ITp __i,
   1019 			      memory_order __m) noexcept
   1020     { return __a->fetch_xor(__i, __m); }
   1021 
   1022   template<typename _ITp>
   1023     inline _ITp
   1024     atomic_fetch_add(__atomic_base<_ITp>* __a, _ITp __i) noexcept
   1025     { return atomic_fetch_add_explicit(__a, __i, memory_order_seq_cst); }
   1026 
   1027   template<typename _ITp>
   1028     inline _ITp
   1029     atomic_fetch_add(volatile __atomic_base<_ITp>* __a, _ITp __i) noexcept
   1030     { return atomic_fetch_add_explicit(__a, __i, memory_order_seq_cst); }
   1031 
   1032   template<typename _ITp>
   1033     inline _ITp
   1034     atomic_fetch_sub(__atomic_base<_ITp>* __a, _ITp __i) noexcept
   1035     { return atomic_fetch_sub_explicit(__a, __i, memory_order_seq_cst); }
   1036 
   1037   template<typename _ITp>
   1038     inline _ITp
   1039     atomic_fetch_sub(volatile __atomic_base<_ITp>* __a, _ITp __i) noexcept
   1040     { return atomic_fetch_sub_explicit(__a, __i, memory_order_seq_cst); }
   1041 
   1042   template<typename _ITp>
   1043     inline _ITp
   1044     atomic_fetch_and(__atomic_base<_ITp>* __a, _ITp __i) noexcept
   1045     { return atomic_fetch_and_explicit(__a, __i, memory_order_seq_cst); }
   1046 
   1047   template<typename _ITp>
   1048     inline _ITp
   1049     atomic_fetch_and(volatile __atomic_base<_ITp>* __a, _ITp __i) noexcept
   1050     { return atomic_fetch_and_explicit(__a, __i, memory_order_seq_cst); }
   1051 
   1052   template<typename _ITp>
   1053     inline _ITp
   1054     atomic_fetch_or(__atomic_base<_ITp>* __a, _ITp __i) noexcept
   1055     { return atomic_fetch_or_explicit(__a, __i, memory_order_seq_cst); }
   1056 
   1057   template<typename _ITp>
   1058     inline _ITp
   1059     atomic_fetch_or(volatile __atomic_base<_ITp>* __a, _ITp __i) noexcept
   1060     { return atomic_fetch_or_explicit(__a, __i, memory_order_seq_cst); }
   1061 
   1062   template<typename _ITp>
   1063     inline _ITp
   1064     atomic_fetch_xor(__atomic_base<_ITp>* __a, _ITp __i) noexcept
   1065     { return atomic_fetch_xor_explicit(__a, __i, memory_order_seq_cst); }
   1066 
   1067   template<typename _ITp>
   1068     inline _ITp
   1069     atomic_fetch_xor(volatile __atomic_base<_ITp>* __a, _ITp __i) noexcept
   1070     { return atomic_fetch_xor_explicit(__a, __i, memory_order_seq_cst); }
   1071 
   1072 
   1073   // Partial specializations for pointers.
   1074   template<typename _ITp>
   1075     inline _ITp*
   1076     atomic_fetch_add_explicit(atomic<_ITp*>* __a, ptrdiff_t __d,
   1077 			      memory_order __m) noexcept
   1078     { return __a->fetch_add(__d, __m); }
   1079 
   1080   template<typename _ITp>
   1081     inline _ITp*
   1082     atomic_fetch_add_explicit(volatile atomic<_ITp*>* __a, ptrdiff_t __d,
   1083 			      memory_order __m) noexcept
   1084     { return __a->fetch_add(__d, __m); }
   1085 
   1086   template<typename _ITp>
   1087     inline _ITp*
   1088     atomic_fetch_add(volatile atomic<_ITp*>* __a, ptrdiff_t __d) noexcept
   1089     { return __a->fetch_add(__d); }
   1090 
   1091   template<typename _ITp>
   1092     inline _ITp*
   1093     atomic_fetch_add(atomic<_ITp*>* __a, ptrdiff_t __d) noexcept
   1094     { return __a->fetch_add(__d); }
   1095 
   1096   template<typename _ITp>
   1097     inline _ITp*
   1098     atomic_fetch_sub_explicit(volatile atomic<_ITp*>* __a,
   1099 			      ptrdiff_t __d, memory_order __m) noexcept
   1100     { return __a->fetch_sub(__d, __m); }
   1101 
   1102   template<typename _ITp>
   1103     inline _ITp*
   1104     atomic_fetch_sub_explicit(atomic<_ITp*>* __a, ptrdiff_t __d,
   1105 			      memory_order __m) noexcept
   1106     { return __a->fetch_sub(__d, __m); }
   1107 
   1108   template<typename _ITp>
   1109     inline _ITp*
   1110     atomic_fetch_sub(volatile atomic<_ITp*>* __a, ptrdiff_t __d) noexcept
   1111     { return __a->fetch_sub(__d); }
   1112 
   1113   template<typename _ITp>
   1114     inline _ITp*
   1115     atomic_fetch_sub(atomic<_ITp*>* __a, ptrdiff_t __d) noexcept
   1116     { return __a->fetch_sub(__d); }
   1117   // @} group atomics
   1118 
   1119 _GLIBCXX_END_NAMESPACE_VERSION
   1120 } // namespace
   1121 
   1122 #endif
   1123