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

F79659-03

coherence/lang/SynchronizedMemberWriteBlock.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_SYNCHRONIZED_MEMBER_WRITE_BLOCK_HPP
00008 #define COH_SYNCHRONIZED_MEMBER_WRITE_BLOCK_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/SynchronizedMemberReadBlock.hpp"
00013 
00014 #include <iostream>
00015 
00016 COH_OPEN_NAMESPACE2(coherence,lang)
00017 
00018 
00019 /**
00020 * The SynchronizedMemberBlock class allows for easy creation of synchronized
00021 * code blocks based on an Object's member level read/write lock. The
00022 * SynchronizedMemberBlock object will ensure that the locks are acquired and
00023 * released as part of starting and ending the code block.
00024 *
00025 * Member read/write locks are not a general purpose feature, and should not
00026 * be used to protect blocking calls. They are intended to protect reads and
00027 * writes to data member primitives, and other short non-blocking code
00028 * blocks.
00029 *
00030 * Example usage:
00031 * @code
00032 * // outside of sync block
00033 *     {
00034 *     SynchronizedMemberWriteBlock syncWrite(self()); // write lock
00035 *     acquired // critical section goes here
00036 *     // ...
00037 *     } // write lock released
00038 * // outside of sync block
00039 * @endcode
00040 *
00041 * A more friendly form is to use the COH_SYNCHRONIZED_MEMBER_WRITE
00042 * macros. Example usage:
00043 *
00044 * @code
00045 * // outside of sync block
00046 * COH_SYNCHRONIZED_MEMBER_WRITE // write lock acquired
00047 *   {
00048 *   // critical section goes here
00049 *   // ...
00050 *   // ...
00051 *   } // write lock released
00052 * // outside of sync block
00053 * @endcode
00054 *
00055 * The SynchronizedMemberWriteBlock class relies on its creator to ensure that
00056 * the associated Object outlives the scope of the block. The helper macro
00057 * ensures this by only allowing you to create a block for the encompassing
00058 * Object, i.e. "this". If the blocks are manually created then the caller
00059 * must ensure that the associated Object outlives the block.
00060 *
00061 * Note: This class indirectly derives from FinalizableBlock, allowing custom
00062 * finalizers to be registered. The finalizers will execute after the lock has
00063 * been released.
00064 *
00065 * @author mf 2008.01.29
00066 */
00067 class COH_EXPORT SynchronizedMemberWriteBlock
00068         : public SynchronizedMemberReadBlock
00069     {
00070     // ----- constructors ---------------------------------------------------
00071 
00072     public:
00073         /**
00074         * Construct a synchronized SynchronizedMemberWriteBlock, acquiring the
00075         * Object's member write lock.
00076         *
00077         * @param o          the Object to lock
00078         * @param pDelegate  SynchronizedMemberWriteBlock to delegate to, or
00079         *                   NULL for no delegate
00080         *
00081         * @throws IllegalArgumentException if pSyncDelegate is non-NULL and
00082         *         references a different Object.
00083         */
00084         SynchronizedMemberWriteBlock(const Object& o,
00085                 SynchronizedMemberWriteBlock* pDelegate = NULL)
00086             : SynchronizedMemberReadBlock()
00087             {
00088             initialize(o, pDelegate);
00089             if (isTerminal())
00090                 {
00091                 o._acquireMemberWriteLock();
00092                 }
00093             m_cpObject = &o;
00094             }
00095 
00096         /**
00097         * Destroy a SynchronizedMemberWriteBlock.
00098         *
00099         * This is a no-op for a delegating block, otherwise the write lock is
00100         * released.
00101         */
00102         ~SynchronizedMemberWriteBlock()
00103             {
00104             const Object* cpObject = m_cpObject;
00105             if (NULL != cpObject && isTerminal())
00106                 {
00107                 m_cpObject = NULL; // record that the lock has been released
00108                 try
00109                     {
00110                     cpObject->_releaseMemberWriteLock();
00111                     }
00112                 catch (const std::exception& e)
00113                     {
00114                     std::cerr << "Error releasing MemberWriteLock: " <<
00115                         e.what() << std::endl;
00116                     }
00117                 }
00118             }
00119 
00120 
00121     // ----- SynchronizedMemberWriteBlock interface -------------------------
00122 
00123     public:
00124         /**
00125         * Set the specified member to the specified value, without
00126         * obtaining additional synchronization.
00127         *
00128         * This helper function is only supported on "members" which
00129         * supply a custom two-parameter "set" method utilizing the
00130         * SynchronizedMemberWriteBlock facility.
00131         *
00132         * @param member  the member to set
00133         * @param value   the value to set the member to
00134         */
00135         template<class M, class V> void setMember(M& member, V value)
00136             {
00137             member.set(value, this);
00138             }
00139 
00140         /**
00141         * Set the specified member to the specified value so long as
00142         * the current value is equal to the assumed value.
00143         *
00144         * This helper function is only supported on "members" which
00145         * supply a custom parameterized "set" and "get" methods utilizing the
00146         * SynchronizedMemberWriteBlock facility.
00147         *
00148         * @param member        the member to set
00149         * @param valueAssumed  the assumed current value of the member
00150         * @param valueNew      the value to set the member to
00151         *
00152         * @return the actual value prior to the update attempt
00153         */
00154         template<class M, class V> typename M::GetType updateMember(
00155                     M& member, V valueAssumed, V valueNew)
00156             {
00157             V valueActual = member.get(this);
00158             if (valueAssumed == valueActual)
00159                 {
00160                 member.set(valueNew, this);
00161                 }
00162             return valueActual;
00163             }
00164 
00165         /**
00166         * Set the specified member to the specified value and return the
00167         * previous value.
00168         *
00169         * This helper function is only supported on "members" which
00170         * supply a custom parameterized "set" and "get" methods utilizing the
00171         * SynchronizedMemberWriteBlock facility.
00172         *
00173         * @param member        the member to set
00174         * @param valueNew      the value to set the member to
00175         *
00176         * @return the prior value
00177         */
00178         template<class M, class V> typename M::GetType exchangeMember(
00179                     M& member, V valueNew)
00180             {
00181             V valuePre = member.get(this);
00182             member.set(valueNew, this);
00183             return valuePre;
00184             }
00185 
00186 
00187     // ----- nested class: Guard --------------------------------------------
00188 
00189     public:
00190         /**
00191         * Simple write lock structure for use in inlining.
00192         */
00193         class COH_EXPORT Guard
00194             {
00195             public:
00196                 Guard(const Object& o)
00197                     : m_o(o)
00198                     {
00199                     acquireMemberWriteLock(o);
00200                     }
00201 
00202                 ~Guard()
00203                     {
00204                     releaseMemberWriteLock(m_o);
00205                     }
00206 
00207             private:
00208                 const Object& m_o;
00209             };
00210 
00211 
00212     // ----- helper methods -------------------------------------------------
00213 
00214     protected:
00215         /*
00216         * Acquire an Object's write lock
00217         */
00218         static void acquireMemberWriteLock(const Object& o)
00219             {
00220             o._acquireMemberWriteLock();
00221             }
00222 
00223         /*
00224         * Release an Object's write lock
00225         */
00226         static void releaseMemberWriteLock(const Object& o)
00227             {
00228             o._releaseMemberWriteLock();
00229             }
00230     };
00231 
00232 COH_CLOSE_NAMESPACE2
00233 
00234 
00235 /**
00236 * Macro for making more readable synchronized member write code blocks See the
00237 * documentation of SynchronizedMemberWriteBlock for a usage example.
00238 *
00239 * @see coherence::lang::SynchronizedMemberWriteBlock
00240 */
00241 #define COH_SYNCHRONIZED_MEMBER_WRITE \
00242     if (coherence::lang::SynchronizedMemberWriteBlock coh_synchronized_member_write \
00243         = coherence::lang::SynchronizedMemberWriteBlock(Object::self())) \
00244         { \
00245         COH_THROW(coherence::lang::IllegalStateException::create()); \
00246         } \
00247     else
00248 
00249 #endif // COH_SYNCHRONIZED_MEMBER_WRITE_BLOCK_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.