C++ Client API Reference for Oracle Coherence
14c (14.1.2.0.0)

F79659-03

coherence/native/NativeAtomic32.hpp

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
Copyright © 2000, 2025, Oracle and/or its affiliates. Licensed under the Universal Permissive License v 1.0 as shown at https://oss.oracle.com/licenses/upl.