00001
00002
00003
00004
00005
00006
00007 #ifndef COH_MEMBER_HANDLE_HPP
00008 #define COH_MEMBER_HANDLE_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
00018 #include <ostream>
00019
00020 COH_OPEN_NAMESPACE2(coherence,lang)
00021
00022 template<class> class MemberView;
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039
00040
00041
00042
00043
00044
00045 template<class T>
00046 class MemberHandle
00047 : public SmartMember, public ChainedHandleElement
00048 {
00049
00050
00051 public:
00052
00053
00054
00055 typedef T ValueType;
00056
00057
00058
00059
00060 typedef typename T::Handle ValueHandle;
00061
00062
00063
00064
00065 typedef typename T::View ValueView;
00066
00067
00068
00069
00070 typedef ValueHandle GetType;
00071
00072
00073
00074
00075 typedef ValueView ConstGetType;
00076
00077
00078
00079
00080 public:
00081
00082
00083
00084
00085
00086 MemberHandle(const Object& oGuardian)
00087 : SmartMember(oGuardian), ChainedHandleElement( false),
00088 m_po(NULL)
00089 {
00090 if (oGuardian._isEscaped())
00091 {
00092 m_prev = m_next = NULL;
00093 }
00094 }
00095
00096
00097
00098
00099
00100
00101
00102
00103 MemberHandle(const Object& oGuardian, const ValueHandle& that)
00104 : SmartMember(oGuardian), ChainedHandleElement( false),
00105 m_po(NULL)
00106 {
00107 if (oGuardian._isEscaped())
00108 {
00109 m_prev = m_next = NULL;
00110 }
00111 set(that);
00112 }
00113
00114
00115
00116
00117
00118
00119
00120
00121
00122
00123 MemberHandle(const Object& oGuardian, const ValueHandle& that,
00124 bool fMutable)
00125 : SmartMember(oGuardian), ChainedHandleElement( false),
00126 m_po(NULL)
00127 {
00128 if (oGuardian._isEscaped())
00129 {
00130 m_prev = m_next = NULL;
00131 }
00132 set(that);
00133 m_nMutability = fMutable ? forever_mutable : safe_immutable;
00134 }
00135
00136
00137
00138
00139 ~MemberHandle()
00140 {
00141 try
00142 {
00143 const T* cpo = m_po;
00144 if (cpo && m_fView && m_prev == NULL)
00145 {
00146
00147
00148 cpo->_detach( true);
00149 }
00150 else
00151 {
00152
00153 m_nMutability = inherited;
00154 set(NULL);
00155 }
00156 }
00157 catch (const std::exception& e)
00158 {
00159
00160 std::cerr << "Error during ~MemberHandle: " << e.what() << std::endl;
00161 return;
00162 }
00163 }
00164
00165 private:
00166
00167
00168
00169 MemberHandle(const MemberHandle&);
00170
00171
00172
00173
00174 public:
00175
00176
00177
00178
00179
00180
00181
00182 MemberHandle& operator=(const ValueHandle& that)
00183 {
00184 set(that);
00185 return *this;
00186 }
00187
00188
00189
00190
00191
00192
00193
00194
00195
00196 MemberHandle& operator=(MemberHandle& that)
00197 {
00198 operator=((ValueHandle) that);
00199 return *this;
00200 }
00201
00202
00203
00204
00205
00206
00207 operator ValueView() const
00208 {
00209 return get();
00210 }
00211
00212
00213
00214
00215
00216
00217 operator ValueHandle()
00218 {
00219 return get();
00220 }
00221
00222
00223
00224
00225
00226
00227 template<class PT>
00228 operator TypedHandle<const PT>() const
00229 {
00230 return get();
00231 }
00232
00233
00234
00235
00236
00237
00238 template<class PT>
00239 operator TypedHandle<PT>()
00240 {
00241 return get();
00242 }
00243
00244
00245
00246
00247
00248
00249
00250 template<class PT>
00251 operator TypedHolder<PT>() const
00252 {
00253 return get();
00254 }
00255
00256
00257
00258
00259
00260
00261 template<class PT>
00262 operator TypedHolder<PT>()
00263 {
00264 return get();
00265 }
00266
00267
00268
00269
00270
00271
00272 ValueView operator->() const
00273 {
00274 return get();
00275 }
00276
00277
00278
00279
00280
00281
00282 ValueHandle operator->()
00283 {
00284 return get();
00285 }
00286
00287
00288
00289
00290
00291
00292
00293
00294 const T& operator*() const
00295 {
00296 return *get();
00297 }
00298
00299
00300
00301
00302
00303
00304
00305
00306 T& operator*()
00307 {
00308 return *get();
00309 }
00310
00311 private:
00312
00313
00314
00315 MemberHandle<T>& operator=(const MemberHandle<T>&);
00316
00317
00318
00319
00320 protected:
00321
00322
00323
00324
00325
00326
00327
00328
00329 void set(const TypedHandle<T>& that,
00330 SynchronizedMemberWriteBlock* pSync = NULL)
00331 {
00332 if (m_prev == NULL)
00333 {
00334 setEscaped(that, pSync);
00335 }
00336 else if (m_nMutability >= forever_immutable)
00337 {
00338 coh_throw_illegal_state("attempt to set const MemberHandle");
00339 }
00340 else
00341 {
00342 performAction(link(that));
00343 m_po = get_pointer(that);
00344 }
00345 }
00346
00347
00348
00349
00350
00351
00352
00353
00354
00355 void setEscaped(const TypedHandle<T>& that,
00356 SynchronizedMemberWriteBlock* pSync = NULL)
00357 {
00358 const Object& oGuardian = getGuardian();
00359 T* po = get_pointer(that);
00360 T* pDetach = NULL;
00361 Object* pAttach = po == NULL ? NULL : po->_attach( true);
00362
00363 if (pSync == NULL)
00364 {
00365
00366 SynchronizedMemberWriteBlock::Guard guard(oGuardian);
00367 if (m_nMutability >= forever_immutable)
00368 {
00369 coh_throw_illegal_state("attempt to set const MemberHandle");
00370 }
00371 pDetach = m_po;
00372 m_po = NULL == pAttach ? NULL : po;
00373 }
00374 else
00375 {
00376
00377 SynchronizedMemberWriteBlock syncWrite(oGuardian, pSync);
00378 if (m_nMutability >= forever_immutable)
00379 {
00380 coh_throw_illegal_state("attempt to set const MemberHandle");
00381 }
00382 pDetach = m_po;
00383 m_po = NULL == pAttach ? NULL : po;
00384 }
00385
00386 if (pDetach)
00387 {
00388 pDetach->_detach( true);
00389 }
00390 }
00391
00392
00393
00394
00395
00396
00397
00398
00399
00400 ValueView get(SynchronizedMemberReadBlock* pSync = NULL) const
00401 {
00402 if (m_prev == NULL)
00403 {
00404 return getEscaped(pSync);
00405 }
00406 else
00407 {
00408 return TypedHandle<const T>(m_po, *this);
00409 }
00410 }
00411
00412
00413
00414
00415
00416
00417
00418
00419
00420 ValueView getEscaped(SynchronizedMemberReadBlock* pSync = NULL) const
00421 {
00422 if (pSync != NULL)
00423 {
00424 SynchronizedMemberReadBlock syncRead(getGuardian(), pSync);
00425
00426 return ValueView((const T*) m_po);
00427 }
00428 else if (m_nMutability == safe_immutable)
00429 {
00430
00431 return ValueView((const T*) m_po);
00432 }
00433 else
00434 {
00435 SynchronizedMemberReadBlock::Guard guard(getGuardian());
00436
00437 return ValueView((const T*) m_po);
00438 }
00439 }
00440
00441
00442
00443
00444
00445
00446
00447
00448
00449 ValueHandle get(SynchronizedMemberReadBlock* pSync = NULL)
00450 {
00451 if (m_prev == NULL)
00452 {
00453 return getEscaped(pSync);
00454 }
00455 else
00456 {
00457 return TypedHandle<T>(m_po, *this);
00458 }
00459 }
00460
00461
00462
00463
00464
00465
00466
00467
00468
00469 ValueHandle getEscaped(SynchronizedMemberReadBlock* pSync = NULL)
00470 {
00471 const Object& oGuardian = getGuardian();
00472 if (pSync == NULL)
00473 {
00474 SynchronizedMemberReadBlock::Guard guard(oGuardian);
00475 if (m_fView)
00476 {
00477 coh_throw_illegal_state(
00478 "attempt to access handle from const MemberHandle");
00479 }
00480 return ValueHandle(m_po);
00481 }
00482 else
00483 {
00484 SynchronizedMemberReadBlock syncRead(oGuardian, pSync);
00485 if (m_fView)
00486 {
00487 coh_throw_illegal_state(
00488 "attempt to access handle from const MemberHandle");
00489 }
00490 return ValueHandle(m_po);
00491 }
00492 }
00493
00494
00495
00496
00497
00498
00499 void performAction(Action nAction) const
00500 {
00501 T* po = m_po;
00502 switch (nAction)
00503 {
00504 case action_error:
00505 coh_throw_illegal_state("corrupted ChainedHandleElement");
00506 case action_flip:
00507 if (NULL != po) ((const T*) po)->_attach( false);
00508
00509 case action_detach:
00510 if (NULL != po)
00511 {
00512 if (m_fView)
00513 {
00514 ((const T*) po)->_detach( false);
00515 }
00516 else
00517 {
00518 po->_detach( false);
00519 }
00520 }
00521
00522 case action_none:
00523 default:
00524 break;
00525 }
00526 }
00527
00528
00529
00530 protected:
00531
00532
00533
00534
00535
00536
00537 virtual void onConst()
00538 {
00539 bool fEscaped = m_prev == NULL;
00540 T* po = NULL;
00541
00542 if (fEscaped)
00543 {
00544 SynchronizedMemberWriteBlock::Guard guard(getGuardian());
00545 po = m_po;
00546 }
00547 else
00548 {
00549 po = m_po;
00550 }
00551
00552 if (NULL != po)
00553 {
00554
00555 if (fEscaped)
00556 {
00557 ((const T*) po)->_attach( true);
00558 po->_detach( true);
00559 }
00560 else if (scan())
00561 {
00562 performAction(action_flip);
00563 }
00564 }
00565
00566 m_fView = true;
00567 SmartMember::onConst();
00568 }
00569
00570
00571
00572
00573 virtual void onEscape(bool fEscaped) const
00574 {
00575 T* po = m_po;
00576 if (po != NULL)
00577 {
00578 if (m_fView)
00579 {
00580 const T* cpo = po;
00581 cpo->_attach(fEscaped);
00582 }
00583 else
00584 {
00585 po->_attach(fEscaped);
00586 }
00587 }
00588
00589 if (fEscaped)
00590 {
00591 performAction(unlink());
00592 m_prev = m_next = NULL;
00593 }
00594 else
00595 {
00596 if (po)
00597 {
00598 if (m_fView)
00599 {
00600 const T* cpo = po;
00601 cpo->_detach( true);
00602 }
00603 else
00604 {
00605 po->_detach( true);
00606 }
00607 }
00608 m_prev = m_next = this;
00609 }
00610
00611 SmartMember::onEscape(fEscaped);
00612 }
00613
00614
00615
00616
00617 virtual size64_t retained() const
00618 {
00619 ValueView v = get();
00620 return v == NULL
00621 ? 0
00622 : v->sizeOf( true);
00623 }
00624
00625
00626
00627 protected:
00628
00629
00630
00631 T* m_po;
00632
00633
00634
00635
00636
00637
00638
00639 template<class D, class H> friend D cast(MemberHandle<T>& mh, bool fThrow);
00640
00641
00642
00643
00644 friend class SynchronizedMemberReadBlock;
00645
00646
00647
00648
00649 friend class SynchronizedMemberWriteBlock;
00650 };
00651
00652
00653
00654
00655
00656
00657
00658
00659
00660
00661
00662
00663
00664 template <typename Char, typename Traits, class T>
00665 COH_INLINE std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& out, const MemberHandle<T>& mh)
00666 {
00667 out << (typename T::View) mh;
00668 return out;
00669 }
00670
00671
00672
00673
00674
00675
00676 template<class T> void clear_handle(MemberHandle<T>& mh)
00677 {
00678 mh = NULL;
00679 }
00680
00681
00682
00683
00684
00685
00686
00687
00688 template<class T>
00689 bool is_null(const MemberHandle<T>& mh)
00690 {
00691 return mh == NULL;
00692 }
00693
00694
00695
00696
00697
00698
00699
00700
00701
00702
00703
00704
00705 template<class D, class T>
00706 D cast(MemberHandle<T>& mh, bool fThrow = true)
00707 {
00708 return cast<D>((typename MemberHandle<T>::ValueHandle) mh, fThrow);
00709 }
00710
00711
00712
00713
00714
00715
00716
00717
00718
00719
00720
00721
00722 template<class D, class T>
00723 D cast(const MemberHandle<T>& mh, bool fThrow = true)
00724 {
00725 return cast<D>((typename MemberHandle<T>::ValueView) mh, fThrow);
00726 }
00727
00728
00729
00730
00731
00732
00733
00734
00735 template<class D, class T>
00736 bool instanceof(MemberHandle<T>& mh)
00737 {
00738 return NULL != cast<D>(mh, false);
00739 }
00740
00741
00742
00743
00744
00745
00746
00747
00748 template<class D, class T>
00749 bool instanceof(const MemberHandle<T>& mh)
00750 {
00751 return NULL != cast<D>(mh, false);
00752 }
00753
00754 COH_CLOSE_NAMESPACE2
00755
00756 #endif // COH_MEMBER_HANDLE_HPP