00001
00002
00003
00004
00005
00006
00007 #ifndef COH_MEMBER_HOLDER_HPP
00008 #define COH_MEMBER_HOLDER_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
00037
00038
00039
00040
00041
00042
00043 template<class T>
00044 class MemberHolder
00045 : public SmartMember, public ChainedHandleElement
00046 {
00047
00048
00049 public:
00050
00051
00052
00053 typedef const T ValueType;
00054
00055
00056
00057
00058 typedef typename T::Handle ValueHandle;
00059
00060
00061
00062
00063 typedef typename T::View ValueView;
00064
00065
00066
00067
00068 typedef typename T::Holder ValueHolder;
00069
00070
00071
00072
00073 typedef TypedHolder<T> GetType;
00074
00075
00076
00077
00078 public:
00079
00080
00081
00082
00083
00084 MemberHolder(const Object& oGuardian)
00085 : SmartMember(oGuardian), ChainedHandleElement( false),
00086 m_po(NULL)
00087 {
00088 if (oGuardian._isEscaped())
00089 {
00090 m_prev = m_next = NULL;
00091 }
00092 }
00093
00094
00095
00096
00097
00098
00099
00100 MemberHolder(const Object& oGuardian, const TypedHolder<T>& that)
00101 : SmartMember(oGuardian), ChainedHandleElement( false),
00102 m_po(NULL)
00103 {
00104 if (oGuardian._isEscaped())
00105 {
00106 m_prev = m_next = NULL;
00107 }
00108 set(that);
00109 }
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119 MemberHolder(const Object& oGuardian, const TypedHolder<T>& that,
00120 bool fMutable)
00121 : SmartMember(oGuardian), ChainedHandleElement( false),
00122 m_po(NULL)
00123 {
00124 if (oGuardian._isEscaped())
00125 {
00126 m_prev = m_next = NULL;
00127 }
00128 set(that);
00129 m_nMutability = fMutable ? forever_mutable : safe_immutable;
00130 }
00131
00132
00133
00134
00135 ~MemberHolder()
00136 {
00137 try
00138 {
00139 m_nMutability = inherited;
00140 set(NULL);
00141 }
00142 catch (const std::exception& e)
00143 {
00144
00145 std::cerr << "Error during ~MemberHolder: " << e.what() << std::endl;
00146 return;
00147 }
00148 }
00149
00150 protected:
00151
00152
00153
00154
00155
00156 MemberHolder()
00157 : SmartMember(), ChainedHandleElement( false), m_po(NULL)
00158 {
00159 }
00160
00161 private:
00162
00163
00164
00165 MemberHolder(const MemberHolder&);
00166
00167
00168
00169
00170 public:
00171
00172
00173
00174
00175
00176
00177
00178
00179 MemberHolder& operator=(const MemberHolder& that)
00180 {
00181 set(that);
00182 return *this;
00183 }
00184
00185
00186
00187
00188
00189
00190
00191
00192
00193 MemberHolder& operator=(const TypedHolder<T>& that)
00194 {
00195 set(that);
00196 return *this;
00197 }
00198
00199
00200
00201
00202
00203
00204 operator ValueView() const
00205 {
00206 if (m_prev == NULL)
00207 {
00208 if (m_nMutability == safe_immutable)
00209 {
00210
00211 return ValueView(m_po);
00212 }
00213 else
00214 {
00215 SynchronizedMemberReadBlock::Guard guard(getGuardian());
00216 return ValueView(m_po);
00217 }
00218 }
00219 else
00220 {
00221 return TypedHandle<const T>(m_po, *this);
00222 }
00223 }
00224
00225
00226
00227
00228
00229
00230 template<class PT>
00231 operator TypedHandle<const PT>() const
00232 {
00233 return (ValueView) *this;
00234 }
00235
00236
00237
00238
00239
00240
00241 template<class PT>
00242 operator TypedHolder<PT>() const
00243 {
00244 return get();
00245 }
00246
00247
00248
00249
00250
00251
00252 ValueView operator->() const
00253 {
00254 return (ValueView) *this;
00255 }
00256
00257
00258
00259
00260
00261
00262
00263
00264 const T& operator*() const
00265 {
00266 return *get();
00267 }
00268
00269
00270
00271 protected:
00272
00273
00274
00275 virtual void onEscape(bool fEscaped) const
00276 {
00277 T* po = m_po;
00278 if (po)
00279 {
00280 if (m_fView)
00281 {
00282 ((const T*) po)->_attach(fEscaped);
00283 }
00284 else
00285 {
00286 po->_attach(fEscaped);
00287 }
00288 }
00289
00290 if (fEscaped)
00291 {
00292 performAction(unlink());
00293 m_prev = m_next = NULL;
00294 }
00295 else
00296 {
00297 if (po)
00298 {
00299 if (m_fView)
00300 {
00301 ((const T*) po)->_detach( true);
00302 }
00303 else
00304 {
00305 po->_detach( true);
00306 }
00307 }
00308 m_prev = m_next = this;
00309 }
00310
00311 SmartMember::onEscape(fEscaped);
00312 }
00313
00314
00315
00316
00317 virtual size64_t retained() const
00318 {
00319 TypedHolder<T> oh = get();
00320 return oh == NULL
00321 ? 0
00322 : oh->sizeOf( true);
00323 }
00324
00325
00326
00327 protected:
00328
00329
00330
00331
00332
00333
00334 void set(const TypedHolder<T>& that,
00335 SynchronizedMemberWriteBlock *pSync = NULL)
00336 {
00337 if (m_prev == NULL)
00338 {
00339 setEscaped(that, pSync);
00340 }
00341 else if (m_nMutability >= forever_immutable)
00342 {
00343 coh_throw_illegal_state("attempt to set const MemberHolder");
00344 }
00345 else
00346 {
00347 performAction(link(that));
00348 m_po = const_cast<T*>(that.m_cpo);
00349 m_fView = that.m_fView;
00350 }
00351 }
00352
00353
00354
00355
00356
00357
00358
00359 void setEscaped(const TypedHolder<T>& that,
00360 SynchronizedMemberWriteBlock *pSync = NULL)
00361 {
00362 const Object& oGuardian = getGuardian();
00363 T* pNew = const_cast<T*>(that.m_cpo);
00364 bool fNewView = that.m_fView;
00365 T* pOld;
00366 bool fOldView;
00367
00368 if (pNew != NULL)
00369 {
00370 if (fNewView)
00371 {
00372 ((const T*) pNew)->_attach( true);
00373 }
00374 else
00375 {
00376 pNew->_attach( true);
00377 }
00378 }
00379
00380 if (pSync == NULL)
00381 {
00382
00383 SynchronizedMemberWriteBlock::Guard guard(oGuardian);
00384 if (m_nMutability >= forever_immutable)
00385 {
00386 coh_throw_illegal_state("attempt to set const MemberHolder");
00387 }
00388 pOld = m_po;
00389 fOldView = m_fView;
00390
00391 m_po = pNew;
00392 m_fView = fNewView;
00393 }
00394 else
00395 {
00396
00397 SynchronizedMemberWriteBlock syncWrite(oGuardian, pSync);
00398 if (m_nMutability >= forever_immutable)
00399 {
00400 coh_throw_illegal_state("attempt to set const MemberHolder");
00401 }
00402
00403 pOld = m_po;
00404 fOldView = m_fView;
00405
00406 m_po = pNew;
00407 m_fView = fNewView;
00408 }
00409
00410 if (pOld != NULL)
00411 {
00412 if (fOldView)
00413 {
00414 ((const T*) pOld)->_detach( true);
00415 }
00416 else
00417 {
00418 pOld->_detach( true);
00419 }
00420 }
00421 }
00422
00423
00424
00425
00426
00427
00428
00429
00430
00431
00432 TypedHolder<T> get(SynchronizedMemberReadBlock* pSync = NULL) const
00433 {
00434 if (m_prev == NULL)
00435 {
00436 return getEscaped(pSync);
00437 }
00438 else if (m_fView)
00439 {
00440 return TypedHandle<const T>(m_po, *this);
00441 }
00442 else
00443 {
00444 return TypedHandle<T>(m_po, *this);
00445 }
00446 }
00447
00448
00449
00450
00451
00452
00453
00454
00455
00456
00457 TypedHolder<T> getEscaped(SynchronizedMemberReadBlock* pSync = NULL) const
00458 {
00459 const Object& oGuardian = getGuardian();
00460
00461 if (pSync != NULL)
00462 {
00463 SynchronizedMemberReadBlock syncRead(oGuardian, pSync);
00464 if (m_fView)
00465 {
00466 return ValueView(m_po);
00467 }
00468 return ValueHandle(m_po);
00469 }
00470 else if (m_nMutability == safe_immutable)
00471 {
00472
00473 if (m_fView)
00474 {
00475 return ValueView(m_po);
00476 }
00477 return ValueHandle(m_po);
00478 }
00479 else
00480 {
00481 SynchronizedMemberReadBlock::Guard guard(oGuardian);
00482 if (m_fView)
00483 {
00484 return ValueView(m_po);
00485 }
00486 return ValueHandle(m_po);
00487 }
00488 }
00489
00490
00491
00492
00493
00494
00495 void performAction(Action nAction) const
00496 {
00497 T* po = m_po;
00498 switch (nAction)
00499 {
00500 case action_error:
00501 coh_throw_illegal_state("corrupted ChainedHandleElement");
00502 case action_flip:
00503 if (NULL != po) ((const T*) po)->_attach( false);
00504
00505 case action_detach:
00506 if (NULL != po)
00507 {
00508 if (m_fView)
00509 {
00510 ((const T*) po)->_detach( false);
00511 }
00512 else
00513 {
00514 po->_detach( false);
00515 }
00516 }
00517
00518 case action_none:
00519 default:
00520 break;
00521 }
00522 }
00523
00524
00525
00526
00527 protected:
00528
00529
00530
00531 T* m_po;
00532
00533
00534
00535
00536
00537
00538
00539 template<class> friend class Array;
00540
00541
00542
00543
00544 friend class SynchronizedMemberReadBlock;
00545
00546
00547
00548
00549 friend class SynchronizedMemberWriteBlock;
00550 };
00551
00552
00553
00554
00555
00556
00557
00558
00559
00560
00561
00562
00563
00564 template <typename Char, typename Traits, class T>
00565 COH_INLINE std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& out, const MemberHolder<T>& th)
00566 {
00567 out << (typename T::View) th;
00568 return out;
00569 }
00570
00571
00572
00573
00574
00575
00576 template<class T> void clear_handle(MemberHolder<T>& mh)
00577 {
00578 mh = NULL;
00579 }
00580
00581
00582
00583
00584
00585
00586
00587
00588 template<class T>
00589 bool is_null(const MemberHolder<T>& mh)
00590 {
00591 return mh == NULL;
00592 }
00593
00594
00595
00596
00597
00598
00599
00600
00601
00602
00603
00604
00605 template<class D, class T>
00606 D cast(const MemberHolder<T>& mh, bool fThrow = true)
00607 {
00608 return cast<D>((TypedHolder<T>) mh, fThrow);
00609 }
00610
00611
00612
00613
00614
00615
00616
00617
00618 template<class D, class T>
00619 bool instanceof(const MemberHolder<T>& mh)
00620 {
00621 return instanceof<D>((TypedHolder<T>) mh);
00622 }
00623
00624 COH_CLOSE_NAMESPACE2
00625
00626 #endif // COH_MEMBER_HOLDER_HPP