00001 /* 00002 * Copyright (c) 2000, 2020, Oracle and/or its affiliates. 00003 * 00004 * Licensed under the Universal Permissive License v 1.0 as shown at 00005 * http://oss.oracle.com/licenses/upl. 00006 */ 00007 #ifndef COH_NATIVE_ATOMIC32_HPP 00008 #define COH_NATIVE_ATOMIC32_HPP 00009 00010 #include "coherence/lang/compatibility.hpp" 00011 00012 COH_OPEN_NAMESPACE2(coherence,native) 00013 00014 00015 /** 00016 * @internal 00017 * 00018 * 32-bit atomic integer, providing JSR-133 volatile memory semantics. 00019 * 00020 * @author mf 2015.02.11 00021 */ 00022 class COH_EXPORT NativeAtomic32 00023 { 00024 // ----- constructors --------------------------------------------------- 00025 00026 public: 00027 /** 00028 * Create a new Atomic32, as Atomic32 is not an Object 00029 * it is the responsibility of the caller to eventually 00030 * destroy this object. 00031 * 00032 * @param nValue the initial value 00033 * @param fAtomic true if the initial assignment needs synchronization 00034 * 00035 * @return a pointer to the new Atomic32 00036 */ 00037 NativeAtomic32(int32_t nValue = 0, bool fAtomic = true) 00038 { 00039 if (fAtomic) 00040 { 00041 set(nValue); 00042 } 00043 else 00044 { 00045 poke(nValue); 00046 } 00047 } 00048 00049 00050 // ----- Atomic32 interface --------------------------------------------- 00051 00052 public: 00053 /** 00054 * Return the current value. 00055 * 00056 * @return the current value 00057 */ 00058 int32_t get() const; 00059 00060 /** 00061 * Return the assumed value without performing any memory 00062 * synchronization. This value should not be trusted, but is suitable 00063 * as an assumed value to be passed to update, which will validate if 00064 * it is correct before allowing any associated change to occur. 00065 * 00066 * @return the assumed value 00067 */ 00068 int32_t peek() const 00069 { 00070 return m_nAtomic; 00071 } 00072 00073 /** 00074 * Unconditionally set the value in a thread-safe manor. 00075 * 00076 * @param nValue the new value 00077 */ 00078 void set(int32_t nValue); 00079 00080 /** 00081 * Unconditionally (and in a non thread-safe manor) set the value. 00082 * 00083 * @param nValue the new value 00084 */ 00085 void poke(int32_t nValue) 00086 { 00087 m_nAtomic = nValue; 00088 } 00089 00090 /** 00091 * Set the value so long as the current value matches the expected value. 00092 * 00093 * @param nAssume the expected current value 00094 * @param nValue the new value 00095 * 00096 * @return the prior actual value, if the returned value 00097 * does is not equal to the supplied assumed 00098 * value then update did not take place 00099 */ 00100 int32_t update(int32_t nAssume, int32_t nValue); 00101 00102 /** 00103 * Set the value so long as the current value matches the expected value. 00104 * 00105 * This version of the update method supports conditionally performing 00106 * the update in an unsafe manor. The unsafe update is performed 00107 * without any form of synchronization and should only be used when it 00108 * is known that the object is referenced by a single thread. 00109 * 00110 * @param nAssume the expected current value 00111 * @param nValue the new value 00112 * @param fAtomic perform a safe update operation which can be used 00113 * across threads 00114 * 00115 * @return the prior actual value, if the returned value 00116 * does is not equal to the supplied assumed 00117 * value then update did not take place 00118 */ 00119 int32_t update(int32_t nAssume, int32_t nValue, bool fAtomic) 00120 { 00121 if (fAtomic) 00122 { 00123 return update(nAssume, nValue); 00124 } 00125 00126 int32_t nOld = peek(); 00127 if (nOld == nAssume) 00128 { 00129 poke(nValue); 00130 } 00131 return nOld; 00132 } 00133 00134 /** 00135 * Adjust the NativeAtomic32 by the specified value. 00136 * 00137 * @param c the amount to adjust the atomic by 00138 * @param fSafe to provide overflow protection 00139 * 00140 * @return the old value 00141 */ 00142 int32_t postAdjust(int32_t c, bool fSafe = true) 00143 { 00144 int32_t cAssume; 00145 int32_t cActual = peek(); 00146 do 00147 { 00148 cAssume = cActual; 00149 if (fSafe && 00150 ((c > 0 && cAssume > max_value - c) || 00151 (c < 0 && cAssume < min_value - c ))) 00152 { 00153 // would overflow 00154 coh_throw_illegal_argument("NativeAtomic32 overflow"); 00155 } 00156 cActual = update(cAssume, cAssume + c); 00157 } 00158 while (cAssume != cActual); 00159 return cAssume; 00160 } 00161 00162 /** 00163 * Adjust the NativeAtomic32 by the specified value. 00164 * 00165 * @param c the amount to adjust the atomic by 00166 * @param fSafe to provide overflow protection 00167 * 00168 * @return the adjusted value 00169 */ 00170 int32_t adjust(int32_t c, bool fSafe = true) 00171 { 00172 return postAdjust(c, fSafe) + c; 00173 } 00174 00175 00176 // ----- data members --------------------------------------------------- 00177 00178 private: 00179 /** 00180 * A constant for the minimum representable int32_t value. 00181 * 00182 * Note: This is defined in Integer32.hpp, but we can not include 00183 * Integer32.hpp, so define it here. 00184 */ 00185 static const int32_t min_value = 0x80000000U; 00186 00187 /** 00188 * A constant for the maximum representable int32_t value. 00189 * 00190 * Note: This is defined in Integer32.hpp, but we can not include 00191 * Integer32.hpp, so define it here. 00192 */ 00193 static const int32_t max_value = 0x7FFFFFFFU; 00194 00195 /** 00196 * The atomically accessed value, in general this should be word aligned. 00197 */ 00198 COH_ALIGN(4, int32_t, m_nAtomic); 00199 }; 00200 00201 COH_CLOSE_NAMESPACE2 00202 00203 #endif // COH_NATIVE_ATOMIC32_HPP