atomic_prv_basic.h
1 /***************************************************************************
2  Copyright (C) 2002-2016 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13  ***************************************************************************/
14 #ifndef ATOMIC_PRV_BASIC_H_
15 #define ATOMIC_PRV_BASIC_H_
16 
17 #ifndef USE_STD_ATOMIC
18  #ifdef _MSC_VER
19  #define USE_STD_ATOMIC
20  #endif
21  #if defined __clang__
22  #define USE_STD_ATOMIC
23  #endif
24  #if defined __GNUC__ && !defined __clang__
25  #if __GNUC__ >= 5 && __GNUC_MINOR__ >= 1
26  #define USE_STD_ATOMIC
27  #endif
28  #endif
29 #endif
30 
31 #include <stdint.h>
32 
33 template <typename T, class Enable = void > class atomic;
34 
35 #ifdef USE_STD_ATOMIC
36  #include "atomic_prv_std.h"
37 #else
38 
39 #if defined __i386__ || defined __i486__ || defined __i586__ || defined __i686__\
40  || defined __x86_64__ || defined _M_IX86 || defined _M_X64
41  #include "atomic_prv_x86.h"
42 #else
43  #if defined __ppc__ || defined __POWERPC__ || defined __powerpc__
44  #include "atomic_prv_ppc.h"
45  #else
46  #error Unsupported processor
47  #endif // __ppc__
48 #endif // __i386__
49 
50 #include <type_traits>
51 
52 //! atomic access to POD type capable of CAS.
53 template <typename T>
55 public:
56  atomic_pod_cas() noexcept = default;
57  atomic_pod_cas(T t) noexcept : m_var(t) {}
58  atomic_pod_cas(const atomic_pod_cas &t) noexcept : m_var(t) {}
59  operator T() const noexcept { T x = m_var; readBarrier(); return x;}
60  atomic_pod_cas &operator=(T t) noexcept {
61  writeBarrier(); m_var = t; return *this;
62  }
63  atomic_pod_cas &operator=(const atomic_pod_cas &x) noexcept {
64  writeBarrier(); m_var = x.m_var; return *this;
65  }
66  T exchange(T newv) noexcept {
67  T old = atomicSwap(newv, &m_var);
68  return old;
69  }
70  bool compare_set_strong(T oldv, T newv) noexcept {
71  bool ret = atomicCompareAndSet(oldv, newv, &m_var);
72  return ret;
73  }
74 protected:
75  T m_var;
76 };
77 
78 //! atomic access to POD type capable of CAS2.
79 template <typename T>
81 public:
82  atomic_pod_cas2() noexcept = default;
83  atomic_pod_cas2(T t) noexcept : m_var(t) {}
84  atomic_pod_cas2(const atomic_pod_cas2 &t) noexcept : m_var(t) {}
85  operator T() const noexcept {
86  for(;;) {
87  T oldv = m_var;
88  if(atomicCompareAndSet(oldv, oldv, &m_var)) {
89  return oldv;
90  }
91  }
92  }
93  atomic_pod_cas2 &operator=(T t) noexcept {
94  writeBarrier();
95  for(;;) {
96  T oldv = m_var;
97  if(atomicCompareAndSet(oldv, t, &m_var))
98  break;
99  }
100  return *this;
101  }
102  atomic_pod_cas2 &operator=(const atomic_pod_cas2 &x) noexcept {
103  *this = (T)x;
104  return *this;
105  }
106  T exchange(T newv) noexcept {
107  for(;;) {
108  T oldv = m_var;
109  if(atomicCompareAndSet(oldv, newv, &m_var)) {
110  return oldv;
111  }
112  }
113  }
114  bool compare_set_strong(T oldv, T newv) noexcept {
115  bool ret = atomicCompareAndSet(oldv, newv, &m_var);
116  return ret;
117  }
118 protected:
119  T m_var
120 #ifdef _MSC_VER
121  __declspec(align(8));
122 #else
123  __attribute__((aligned(8)));
124 #endif
125 };
126 
127 //! atomic access to POD type capable of CAS2.
128 template <typename T>
129 class atomic<T, typename std::enable_if<
130 (sizeof(int_cas2) * 2 == sizeof(T)) && std::is_pod<T>::value>::type>
131 : public atomic_pod_cas2<T> {
132 public:
133  atomic() noexcept = default;
134  atomic(T t) noexcept : atomic_pod_cas2<T>(t) {}
135  atomic(const atomic &t) noexcept = default;
136 };
137 
138 //! atomic access to POD type capable of CAS.
139 template <typename T>
140 class atomic<T, typename std::enable_if<
141 (sizeof(int_cas_max) >= sizeof(T)) && std::is_pod<T>::value &&
142 !std::is_integral<T>::value>::type>
143 : public atomic_pod_cas<T> {
144 public:
145  atomic() noexcept = default;
146  atomic(T t) noexcept : atomic_pod_cas<T>(t) {}
147  atomic(const atomic &t) noexcept = default;
148 };
149 
150 //! atomic access to integer-POD-type capable of CAS.
151 template <typename T>
152 class atomic<T, typename std::enable_if<
153 (sizeof(int_cas_max) >= sizeof(T)) && std::is_integral<T>::value>::type >
154 : public atomic_pod_cas<T> {
155 public:
156  atomic() noexcept : atomic_pod_cas<T>((T)0) {}
157  atomic(T t) noexcept : atomic_pod_cas<T>(t) {}
158  atomic(const atomic &t) = default;
159  //! Note that the return value is atomically given.
160  atomic &operator++() noexcept {writeBarrier(); atomicInc( &this->m_var); return *this;}
161  //! Note that the return value is atomically given.
162  atomic &operator--() noexcept {writeBarrier(); atomicDecAndTest( &this->m_var); return *this;}
163  //! Note that the return value is atomically given.
164  atomic &operator+=(T t) noexcept {writeBarrier(); atomicAdd( &this->m_var, t); return *this;}
165  //! Note that the return value is atomically given.
166  atomic &operator-=(T t) noexcept {writeBarrier(); atomicAdd( &this->m_var, -t); return *this;}
167  bool decAndTest() noexcept {
168  bool ret = atomicDecAndTest( &this->m_var);
169  return ret;
170  }
171  bool addAndTest(T t) noexcept {
172  bool ret = atomicAddAndTest( &this->m_var, t);
173  return ret;
174  }
175 };
176 
177 #endif //!USE_STD_ATOMIC
178 
179 #endif /*ATOMIC_PRV_BASIC_H_*/

Generated for KAME4 by  doxygen 1.8.3