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