00001
00002
00003
00004
00005
00006
00007 #ifndef COH_MEMBER_VIEW_HPP
00008 #define COH_MEMBER_VIEW_HPP
00009
00010 #include "coherence/lang/compatibility.hpp"
00011
00012 #include "coherence/lang/Object.hpp"
00013 #include "coherence/lang/SmartMember.hpp"
00014 #include "coherence/lang/SynchronizedMemberReadBlock.hpp"
00015 #include "coherence/lang/SynchronizedMemberWriteBlock.hpp"
00016 #include "coherence/lang/TypedHandle.hpp"
00017 #include "coherence/lang/TypedHolder.hpp"
00018
00019 #include <ostream>
00020
00021 COH_OPEN_NAMESPACE2(coherence,lang)
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036 template<class T>
00037 class MemberView
00038 : public SmartMember, public ChainedHandleElement
00039 {
00040
00041
00042 public:
00043
00044
00045
00046 typedef T ValueType;
00047
00048
00049
00050
00051 typedef typename T::View ValueView;
00052
00053
00054
00055
00056 typedef ValueView GetType;
00057
00058
00059
00060
00061 public:
00062
00063
00064
00065
00066
00067 MemberView(const Object& oGuardian)
00068 : SmartMember(oGuardian), ChainedHandleElement( true),
00069 m_cpo(NULL)
00070 {
00071 if (oGuardian._isEscaped())
00072 {
00073 m_prev = m_next = NULL;
00074 }
00075 }
00076
00077
00078
00079
00080
00081
00082
00083 MemberView(const Object& oGuardian, const ValueView& that)
00084 : SmartMember(oGuardian), ChainedHandleElement( true),
00085 m_cpo(NULL)
00086 {
00087 if (oGuardian._isEscaped())
00088 {
00089 m_prev = m_next = NULL;
00090 }
00091 set(that);
00092 }
00093
00094
00095
00096
00097
00098
00099
00100
00101 MemberView(const Object& oGuardian, const ValueView& that, bool fMutable)
00102 : SmartMember(oGuardian), ChainedHandleElement( true),
00103 m_cpo(NULL)
00104 {
00105 if (oGuardian._isEscaped())
00106 {
00107 m_prev = m_next = NULL;
00108 }
00109 set(that);
00110 m_nMutability = fMutable ? forever_mutable : safe_immutable;
00111 }
00112
00113
00114
00115
00116 ~MemberView()
00117 {
00118 try
00119 {
00120 m_nMutability = inherited;
00121 set(NULL);
00122 }
00123 catch (const std::exception& e)
00124 {
00125
00126 std::cerr << "Error during ~MemberView: " << e.what() << std::endl;
00127 return;
00128 }
00129 }
00130
00131 private:
00132
00133
00134
00135 MemberView(const MemberView&);
00136
00137
00138
00139
00140 public:
00141
00142
00143
00144
00145
00146
00147
00148 MemberView& operator=(const ValueView& that)
00149 {
00150 set(that);
00151 return *this;
00152 }
00153
00154
00155
00156
00157
00158
00159
00160
00161 MemberView& operator=(const MemberView& that)
00162 {
00163 set(that);
00164 return *this;
00165 }
00166
00167
00168
00169
00170
00171
00172 operator ValueView() const
00173 {
00174 return get();
00175 }
00176
00177
00178
00179
00180
00181
00182 template<class PT>
00183 operator TypedHandle<const PT>() const
00184 {
00185 return get();
00186 }
00187
00188
00189
00190
00191
00192
00193 template<class PT>
00194 operator TypedHolder<PT>() const
00195 {
00196 return get();
00197 }
00198
00199
00200
00201
00202
00203
00204 ValueView operator->() const
00205 {
00206 return get();
00207 }
00208
00209
00210
00211
00212
00213
00214
00215
00216 const T& operator*() const
00217 {
00218 return *get();
00219 }
00220
00221
00222
00223 protected:
00224
00225
00226
00227 virtual void onEscape(bool fEscaped) const
00228 {
00229 const T* cpo = m_cpo;
00230 if (cpo)
00231 {
00232 cpo->_attach(fEscaped);
00233 }
00234
00235 if (fEscaped)
00236 {
00237 performAction(unlink());
00238 m_prev = m_next = NULL;
00239 }
00240 else
00241 {
00242 if (cpo)
00243 {
00244 cpo->_detach( true);
00245 }
00246 m_prev = m_next = this;
00247 }
00248
00249 SmartMember::onEscape(fEscaped);
00250 }
00251
00252
00253
00254
00255 virtual size64_t retained() const
00256 {
00257 ValueView v = get();
00258 return v == NULL
00259 ? 0
00260 : v->sizeOf( true);
00261 }
00262
00263
00264
00265
00266 protected:
00267
00268
00269
00270
00271
00272
00273
00274 void set(const TypedHolder<T>& that,
00275 SynchronizedMemberWriteBlock* pSync = NULL)
00276 {
00277 if (m_prev == NULL)
00278 {
00279 setEscaped(that, pSync);
00280 }
00281 else if (m_nMutability >= forever_immutable)
00282 {
00283 coh_throw_illegal_state("attempt to set const MemberView");
00284 }
00285 else
00286 {
00287 performAction(link(that));
00288 m_cpo = get_pointer(that);
00289 }
00290 }
00291
00292
00293
00294
00295
00296
00297
00298
00299 void setEscaped(const TypedHolder<T>& that,
00300 SynchronizedMemberWriteBlock* pSync = NULL)
00301 {
00302 const Object& oGuardian = getGuardian();
00303 const T* cpo = get_pointer(that);
00304 const T* cpDetach = NULL;
00305 const Object* cpAttach = NULL == cpo ? NULL : cpo->_attach( true);
00306
00307 if (pSync == NULL)
00308 {
00309
00310 SynchronizedMemberWriteBlock::Guard guard(oGuardian);
00311 if (m_nMutability >= forever_immutable)
00312 {
00313 coh_throw_illegal_state("attempt to set const MemberView");
00314 }
00315
00316 cpDetach = m_cpo;
00317 m_cpo = NULL == cpAttach ? NULL : cpo;
00318 }
00319 else
00320 {
00321
00322 SynchronizedMemberWriteBlock syncWrite(oGuardian, pSync);
00323 if (m_nMutability >= forever_immutable)
00324 {
00325 coh_throw_illegal_state("attempt to set const MemberView");
00326 }
00327
00328 cpDetach = m_cpo;
00329 m_cpo = NULL == cpAttach ? NULL : cpo;
00330 }
00331
00332 if (cpDetach)
00333 {
00334 cpDetach->_detach( true);
00335 }
00336
00337 }
00338
00339
00340
00341
00342
00343
00344
00345
00346
00347 ValueView get(SynchronizedMemberReadBlock* pSync = NULL) const
00348 {
00349 if (m_prev == NULL)
00350 {
00351 return getEscaped(pSync);
00352 }
00353 else
00354 {
00355 return TypedHandle<const T>(m_cpo, *this);
00356 }
00357 }
00358
00359
00360
00361
00362
00363
00364
00365
00366
00367 ValueView getEscaped(SynchronizedMemberReadBlock* pSync = NULL) const
00368 {
00369 const Object& oGuardian = getGuardian();
00370 if (pSync != NULL)
00371 {
00372
00373 SynchronizedMemberReadBlock syncRead(oGuardian, pSync);
00374 return ValueView(m_cpo);
00375 }
00376 else if (m_nMutability == safe_immutable)
00377 {
00378
00379 return ValueView(m_cpo);
00380 }
00381 else
00382 {
00383
00384 SynchronizedMemberReadBlock::Guard guard(oGuardian);
00385 return ValueView(m_cpo);
00386 }
00387 }
00388
00389
00390
00391
00392
00393
00394
00395
00396 void performAction(Action nAction) const
00397 {
00398 const T* cpo = m_cpo;
00399 switch (nAction)
00400 {
00401 case action_error:
00402 case action_flip:
00403 coh_throw_illegal_state("corrupted ChainedHandleElement");
00404 case action_detach:
00405 if (cpo) cpo->_detach( false);
00406
00407 case action_none:
00408 default:
00409 break;
00410 }
00411 }
00412
00413
00414
00415
00416 protected:
00417
00418
00419
00420 const T* m_cpo;
00421
00422
00423
00424
00425
00426
00427 friend class SynchronizedMemberReadBlock;
00428
00429
00430
00431
00432 friend class SynchronizedMemberWriteBlock;
00433 };
00434
00435
00436
00437
00438
00439
00440
00441
00442
00443
00444
00445
00446
00447 template <typename Char, typename Traits, class T>
00448 COH_INLINE std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& out, const MemberView<T>& mv)
00449 {
00450 out << (typename T::View) mv;
00451 return out;
00452 }
00453
00454
00455
00456
00457
00458
00459 template<class T> void clear_handle(MemberView<T>& mv)
00460 {
00461 mv = NULL;
00462 }
00463
00464
00465
00466
00467
00468
00469
00470
00471
00472
00473
00474
00475 template<class D, class T>
00476 D cast(const MemberView<T>& mv, bool fThrow = true)
00477 {
00478 return cast<D>((typename MemberView<T>::ValueView) mv, fThrow);
00479 }
00480
00481
00482
00483
00484
00485
00486
00487
00488 template<class D, class T>
00489 bool instanceof(const MemberView<T>& mv)
00490 {
00491 return NULL != cast<D>(mv, false);
00492 }
00493
00494
00495
00496
00497
00498
00499
00500
00501 template<class T>
00502 bool is_null(const MemberView<T>& mv)
00503 {
00504 return mv == NULL;
00505 }
00506
00507 COH_CLOSE_NAMESPACE2
00508
00509 #endif // COH_MEMBER_VIEW_HPP