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_ATOMIC64_HPP 00008 #define COH_NATIVE_ATOMIC64_HPP 00009 00010 #include "coherence/lang/compatibility.hpp" 00011 00012 COH_OPEN_NAMESPACE2(coherence,native) 00013 00014 00015 /** 00016 * @internal 00017 * 00018 * 64-bit atomic integer, providing JSR-133 volatile memory semantics. 00019 * 00020 * @author mf 2007.12.01 00021 */ 00022 class COH_EXPORT NativeAtomic64 00023 { 00024 // ----- constructors --------------------------------------------------- 00025 00026 public: 00027 /** 00028 * Create a new Atomic64, as Atomic64 is not an Object 00029 * it is the responsibility of the caller to eventually 00030 * destroy this object. 00031 * 00032 * @param lValue the initial value 00033 * @param fAtomic true if the initial assignment needs synchronization 00034 * 00035 * @return a pointer to the new Atomic64 00036 */ 00037 NativeAtomic64(int64_t lValue = 0, bool fAtomic = true) 00038 { 00039 if (fAtomic) 00040 { 00041 set(lValue); 00042 } 00043 else 00044 { 00045 poke(lValue); 00046 } 00047 } 00048 00049 00050 // ----- Atomic64 interface --------------------------------------------- 00051 00052 public: 00053 /** 00054 * Return the current value. 00055 * 00056 * @return the current value 00057 */ 00058 int64_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 int64_t peek() const 00069 { 00070 return m_lAtomic; 00071 } 00072 00073 /** 00074 * Unconditionally set the value in a thread-safe manor. 00075 * 00076 * @param lValue the new value 00077 */ 00078 void set(int64_t lValue); 00079 00080 /** 00081 * Unconditionally (and in a non thread-safe manor) set the value. 00082 * 00083 * @param lValue the new value 00084 */ 00085 void poke(int64_t lValue) 00086 { 00087 m_lAtomic = lValue; 00088 } 00089 00090 /** 00091 * Set the value so long as the current value matches the expected value. 00092 * 00093 * @param lAssume the expected current value 00094 * @param lValue 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 int64_t update(int64_t lAssume, int64_t lValue); 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 lAssume the expected current value 00111 * @param lValue 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 int64_t update(int64_t lAssume, int64_t lValue, bool fAtomic) 00120 { 00121 if (fAtomic) 00122 { 00123 return update(lAssume, lValue); 00124 } 00125 00126 int64_t lOld = peek(); 00127 if (lOld == lAssume) 00128 { 00129 poke(lValue); 00130 } 00131 return lOld; 00132 } 00133 00134 /** 00135 * Adjust the NativeAtomic64 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 int64_t postAdjust(int64_t c, bool fSafe = true) 00143 { 00144 int64_t cAssume; 00145 int64_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("NativeAtomic64 overflow"); 00155 } 00156 cActual = update(cAssume, cAssume + c); 00157 } 00158 while (cAssume != cActual); 00159 return cAssume; 00160 } 00161 00162 /** 00163 * Adjust the NativeAtomic64 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 int64_t adjust(int64_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 int64_t value. 00181 * 00182 * Note: This is defined in Integer64.hpp, but we can not include 00183 * Integer64.hpp, so define it here. 00184 */ 00185 static const int64_t min_value = COH_INT64(0x80000000U, 0x0); 00186 00187 /** 00188 * A constant for the maximum representable int64_t value. 00189 * 00190 * Note: This is defined in Integer64.hpp, but we can not include 00191 * Integer64.hpp, so define it here. 00192 */ 00193 static const int64_t max_value = COH_INT64(0x7FFFFFFFU, 0xFFFFFFFFU); 00194 00195 /** 00196 * The atomically accessed value, in general this should be word aligned. 00197 */ 00198 COH_ALIGN(8, int64_t, m_lAtomic); 00199 }; 00200 00201 COH_CLOSE_NAMESPACE2 00202 00203 #endif // COH_NATIVE_ATOMIC64_HPP