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

F79659-03

coherence/lang/MemberHandle.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_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 * MemberHandle is a thread-safe handle used by an Object to reference its
00027 * non-const child Objects.
00028 *
00029 * MemberHandles transfer the constness of their guardian Object. When a
00030 * MemberHandle is accessed from within a const method of the enclosing
00031 * "parent" class, it will only provide const access to the Object which it
00032 * references. If the enclosing Object becomes only accessed via views the
00033 * MemberHandle will also automatically, and permanently switch its reference
00034 * type from a handle to a view.
00035 *
00036 * Note: In the rare case that a MemberHandle is declared via the mutable
00037 *       keyword, the MemberHandle must be informed of this fact by setting
00038 *       fMutable to true during construction.
00039 *
00040 * @author mf  2007.07.05
00041 *
00042 * @see MemberView
00043 * @see MemberHolder
00044 */
00045 template<class T>
00046 class MemberHandle
00047     : public SmartMember, public ChainedHandleElement
00048     {
00049     // ----- typedefs -------------------------------------------------------
00050 
00051     public:
00052         /**
00053         * The type of the values the handle can reference.
00054         */
00055         typedef T ValueType;
00056 
00057         /**
00058         * The Handle type for the referenced Object.
00059         */
00060         typedef typename T::Handle ValueHandle;
00061 
00062         /**
00063         * The View type for thekil referenced Object.
00064         */
00065         typedef typename T::View ValueView;
00066 
00067         /**
00068         * Result type for a non-const get operation.
00069         */
00070         typedef ValueHandle GetType;
00071 
00072         /**
00073         * Result type for a non-const get operation.
00074         */
00075         typedef ValueView ConstGetType;
00076 
00077 
00078     // -------- constructors ------------------------------------------------
00079 
00080     public:
00081         /**
00082         * Construct a new MemberHandle referencing NULL via a handle.
00083         *
00084         * @param oGuardian  the object that protects this member
00085         */
00086         MemberHandle(const Object& oGuardian)
00087             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ false),
00088               m_po(NULL)
00089             {
00090             if (oGuardian._isEscaped())
00091                 {
00092                 m_prev = m_next = NULL;
00093                 }
00094             }
00095 
00096         /**
00097         * Construct a new MemberHandle referencing the specified Object via
00098         * a handle.
00099         *
00100         * @param oGuardian   the object that protects this member
00101         * @param that        the Object to reference
00102         */
00103         MemberHandle(const Object& oGuardian, const ValueHandle& that)
00104             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ 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         * Construct a new MemberHandle referencing the specified Object via
00116         * a handle.
00117         *
00118         * @param oGuardian  the object that protects this member
00119         * @param that       the Object to reference
00120         * @param fMutable   true if the member is declared as mutable, false
00121         *                   if declared as const
00122         */
00123         MemberHandle(const Object& oGuardian, const ValueHandle& that,
00124                 bool fMutable)
00125             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ 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         * Destroy the MemberHandle.
00138         */
00139         ~MemberHandle()
00140             {
00141             try
00142                 {
00143                 const T* cpo = m_po;
00144                 if (cpo && m_fView && m_prev == NULL)
00145                     {
00146                     // escaped and const (view based ref); set() does not
00147                     // handle this case as it can only occur during destruction
00148                     cpo->_detach(/*fEscaped*/ true);
00149                     }
00150                 else
00151                     {
00152                     // even if cpo == NULL we still may need to call unlink()
00153                     m_nMutability = inherited;
00154                     set(NULL);
00155                     }
00156                 }
00157             catch (const std::exception& e)
00158                 {
00159                 // Exception::View is not a known type within this file
00160                 std::cerr << "Error during ~MemberHandle: " << e.what() << std::endl;
00161                 return; // can't re-throw from within destructor
00162                 }
00163             }
00164 
00165     private:
00166         /**
00167         * Blocked copy constructor.
00168         */
00169         MemberHandle(const MemberHandle&);
00170 
00171 
00172     // ----- operators ------------------------------------------------------
00173 
00174     public:
00175         /**
00176         * Assign the MemberHandle to reference another object.
00177         *
00178         * @param that  the object to reference
00179         *
00180         * @return a reference to this MemberHandle
00181         */
00182         MemberHandle& operator=(const ValueHandle& that)
00183             {
00184             set(that);
00185             return *this;
00186             }
00187 
00188         /**
00189         * Assign this MemberHandle to reference the same Object as the
00190         * specified MemberHandle.
00191         *
00192         * @param that  the object to reference
00193         *
00194         * @return a reference to this MemberHandle
00195         */
00196         MemberHandle& operator=(MemberHandle& that)
00197             {
00198             operator=((ValueHandle) that); // assign from snapshot
00199             return *this;
00200             }
00201 
00202         /**
00203         * Return a View to the referenced Object.
00204         *
00205         * @return a View to the referenced Object
00206         */
00207         operator ValueView() const
00208             {
00209             return get();
00210             }
00211 
00212         /**
00213         * Return a Handle to the referenced Object.
00214         *
00215         * @return a Handle to the referenced Object
00216         */
00217         operator ValueHandle()
00218             {
00219             return get();
00220             }
00221 
00222         /**
00223         * Return a View to the referenced Object.
00224         *
00225         * @return a View to the referenced Object
00226         */
00227         template<class PT>
00228         operator TypedHandle<const PT>() const
00229             {
00230             return get();
00231             }
00232 
00233         /**
00234         * Return a Handle to the referenced Object.
00235         *
00236         * @return a Handle to the referenced Object
00237         */
00238         template<class PT>
00239         operator TypedHandle<PT>()
00240             {
00241             return get();
00242             }
00243 
00244         /**
00245         * Return a TypedHolder to the referenced Object.
00246         *
00247         * @return a TypedHolder to the referenced Object
00248         *
00249         */
00250         template<class PT>
00251         operator TypedHolder<PT>() const
00252             {
00253             return get();
00254             }
00255 
00256         /**
00257         * Return a TypedHolder to the referenced Object.
00258         *
00259         * @return a TypedHolder to the referenced Object
00260         */
00261         template<class PT>
00262         operator TypedHolder<PT>()
00263             {
00264             return get();
00265             }
00266 
00267         /**
00268         * Dereference the MemberHandle.
00269         *
00270         * @return a const pointer to the referenced Object
00271         */
00272         ValueView operator->() const
00273             {
00274             return get();
00275             }
00276 
00277         /**
00278         * Dereference the MemberHandle.
00279         *
00280         * @return a const pointer to the referenced Object
00281         */
00282         ValueHandle operator->()
00283             {
00284             return get();
00285             }
00286 
00287         /**
00288         * Dereference this handle, returning <tt>T&</tt>.
00289         *
00290         * @return a raw <tt>T&</tt> reference to the referenced Object
00291         *
00292         * @throws NullPointerException if the this handle is @c NULL
00293         */
00294         const T& operator*() const
00295             {
00296             return *get();
00297             }
00298 
00299         /**
00300         * Dereference this handle, returning <tt>T&</tt>.
00301         *
00302         * @return a raw <tt>T&</tt> reference to the referenced Object
00303         *
00304         * @throws NullPointerException if the this handle is @c NULL
00305         */
00306         T& operator*()
00307             {
00308             return *get();
00309             }
00310 
00311     private:
00312         /**
00313         * Blocked assignment operator.
00314         */
00315         MemberHandle<T>& operator=(const MemberHandle<T>&);
00316 
00317 
00318     // ----- helper methods -------------------------------------------------
00319 
00320     protected:
00321         /**
00322         * Set the MemberHandle to reference the same Object as the supplied
00323         * Handle.
00324         *
00325         * @param that   the handle to copy
00326         * @param pSync  optional external SyncornizedMemberBlock to use
00327         *               to avoid internal synchronization.
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         * Set the MemberHandle to reference the same Object as the supplied
00349         * Handle.
00350         *
00351         * @param that   the handle to copy
00352         * @param pSync  optional external SyncornizedMemberBlock to use
00353         *               to avoid internal synchronization.
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(/*fEscaped*/ true);
00362 
00363             if (pSync == NULL)
00364                 {
00365                 // sync block
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                 // sync block
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(/*fEscaped*/ true);
00389                 }
00390             }
00391 
00392         /**
00393         * Return a View to the referenced Object.
00394         *
00395         * @param pSync  optional external SyncornizedMemberReadBlock to use
00396         *               to avoid internal synchronization.
00397         *
00398         * @return a View to the referenced Object
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         * Return a View to the referenced Object.
00414         *
00415         * @param pSync  optional external SyncornizedMemberReadBlock to use
00416         *               to avoid internal synchronization.
00417         *
00418         * @return a View to the referenced Object
00419         */
00420         ValueView getEscaped(SynchronizedMemberReadBlock* pSync = NULL) const
00421             {
00422             if (pSync != NULL)
00423                 {
00424                 SynchronizedMemberReadBlock syncRead(getGuardian(), pSync);
00425                 // return of escaped must occur within scope of syncRead
00426                 return ValueView((const T*) m_po);
00427                 }
00428             else if (m_nMutability == safe_immutable)
00429                 {
00430                 // safe_immutable, synchronization not required; also implies view
00431                 return ValueView((const T*) m_po);
00432                 }
00433             else
00434                 {
00435                 SynchronizedMemberReadBlock::Guard guard(getGuardian());
00436                 // return of escaped must occur within scope of guard
00437                 return ValueView((const T*) m_po);
00438                 }
00439             }
00440 
00441         /**
00442         * Return a Handle to the referenced Object.
00443         *
00444         * @param pSync  optional external SyncornizedMemberReadBlock to use
00445         *               to avoid internal synchronization.
00446         *
00447         * @return a Handle to the referenced Object
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         * Return a Handle to the referenced Object.
00463         *
00464         * @param pSync  optional external SyncornizedMemberReadBlock to use
00465         *               to avoid internal synchronization.
00466         *
00467         * @return a Handle to the referenced Object
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); // must occur within scope of syncRead
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); // must occur within scope of syncRead
00491                 }
00492             }
00493 
00494         /**
00495         * Perform the specified action.
00496         *
00497         * @param nAction  the action to perform
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(/*fEscaped*/ false);
00508                     // fall through
00509                 case action_detach:
00510                     if (NULL != po)
00511                         {
00512                         if (m_fView)
00513                             {
00514                             ((const T*) po)->_detach(/*fEscaped*/ false);
00515                             }
00516                         else
00517                             {
00518                             po->_detach(/*fEscaped*/ false);
00519                             }
00520                         }
00521                     // fall through
00522                 case action_none:
00523                 default:
00524                     break;
00525                 }
00526             }
00527 
00528     // ----- SmartMember interface ------------------------------------------
00529 
00530     protected:
00531         /**
00532         * {@inheritDoc}
00533         *
00534         * The MemberHandle is automatically flipped when the guardian Object
00535         * becomes only referenced via consts.
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                 // re-attach as const, then detach the old reference
00555                 if (fEscaped)
00556                     {
00557                     ((const T*) po)->_attach(/*fEscaped*/ true);
00558                     po->_detach(/*fEscaped*/ 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         * {@inheritDoc}
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);  // new attach
00582                     }
00583                 else
00584                     {
00585                     po->_attach(fEscaped);  // new attach
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(/*fEscaped*/ true); // old detach
00602                         }
00603                     else
00604                         {
00605                         po->_detach(/*fEscaped*/ true); // old detach
00606                         }
00607                     }
00608                 m_prev = m_next = this;
00609                 }
00610 
00611             SmartMember::onEscape(fEscaped);
00612             }
00613 
00614         /**
00615         * {@inheritDoc}
00616         */
00617         virtual size64_t retained() const
00618             {
00619             ValueView v = get();
00620             return v == NULL
00621                     ? 0
00622                     : v->sizeOf(/*fDeep*/ true);
00623             }
00624 
00625     // ----- data members ---------------------------------------------------
00626 
00627     protected:
00628        /**
00629         * The referenced Object.
00630         */
00631         T* m_po;
00632 
00633 
00634     // ----- friends --------------------------------------------------------
00635 
00636     /**
00637     * @internal
00638     */
00639     template<class D, class H> friend D cast(MemberHandle<T>& mh, bool fThrow);
00640 
00641     /**
00642     * @internal
00643     */
00644     friend class SynchronizedMemberReadBlock;
00645 
00646     /**
00647     * @internal
00648     */
00649     friend class SynchronizedMemberWriteBlock;
00650     };
00651 
00652 
00653 // ----- non-member operators and functions ---------------------------------
00654 
00655 /**
00656 * Output a human-readable description of the given MemberHandle to the
00657 * specified stream.
00658 *
00659 * @param out  the stream used to output the description
00660 * @param mh   the MemberHandle to describe
00661 *
00662 * @return the supplied stream
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 * Assign the specified handle to NULL.
00673 *
00674 * @param mh the handle to clear
00675 */
00676 template<class T> void clear_handle(MemberHandle<T>& mh)
00677     {
00678     mh = NULL;
00679     }
00680 
00681 /**
00682 * Return true if the supplied handle equals NULL.
00683 *
00684 * @param mh  the member handle to test
00685 *
00686 * @return true iff the supplied handle equals NULL
00687 */
00688 template<class T>
00689 bool is_null(const MemberHandle<T>& mh)
00690     {
00691     return mh == NULL;
00692     }
00693 
00694 /**
00695 * Perform a dynamic cast the pointer associated with the MemberHandle
00696 * to a the specified handle/view type.
00697 *
00698 * @param mh      the MemberHandle from which to perform the cast
00699 * @param fThrow  true if an exception is to be thrown on a failed cast
00700 *
00701 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00702 *
00703 * @throws ClassCastException if the cast fails and fThrow is true
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 * Perform a dynamic cast the pointer associated with the MemberHandle
00713 * to a the specified handle/view type.
00714 *
00715 * @param mh      the MemberHandle from which to perform the cast
00716 * @param fThrow  true if an exception is to be thrown on a failed cast
00717 *
00718 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00719 *
00720 * @throws ClassCastException if the cast fails and fThrow is true
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 * Perform an instanceof check on a MemberHandle.
00730 *
00731 * @param mh  the MemberHandle from which to perform the test
00732 *
00733 * @return true if the supplied handle is an instance of the specified type
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 * Perform an instanceof check on a MemberHandle.
00743 *
00744 * @param mh  the MemberHandle from which to perform the test
00745 *
00746 * @return true if the supplied handle is an instance of the specified type
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
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.