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

F79659-03

coherence/lang/MemberHolder.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_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 * MemberHolder is a thread-safe handle implementation which supports
00026 * referencing Objects as either Handles or Views. MemberHolder can safely be
00027 * used in place of View but adds the ability to attempt a safe down cast to a
00028 * Handle. This differs from the C++ const_cast in that the down cast will only
00029 * succeed if the MemberHolder had been assigned from a Handle, an assignment
00030 * from a View results in a MemberHolder whose down cast operation will throw
00031 * a ClassCastException.
00032 *
00033 * MemberHolder is not for general use, instead it is a convenience handle type
00034 * which can be used for container like classes, which need to be able to
00035 * contain both Handles and Views.
00036 *
00037 * Note: In the rare case that a MemberHolder is declared via the mutable
00038 *       keyword, the MemberHolder must be informed of this fact by setting
00039 *       fMutable to true during construction.
00040 *
00041 * @author mf  2008.01.09
00042 */
00043 template<class T>
00044 class MemberHolder
00045     : public SmartMember, public ChainedHandleElement
00046     {
00047     // ----- typedefs -------------------------------------------------------
00048 
00049     public:
00050         /**
00051         * The type of the values the holder can reference.
00052         */
00053         typedef const T ValueType;
00054 
00055         /**
00056         * The Handle type for the referenced Object.
00057         */
00058         typedef typename T::Handle ValueHandle;
00059 
00060         /**
00061         * The View type for the referenced Object.
00062         */
00063         typedef typename T::View ValueView;
00064 
00065         /**
00066         * The Holder type for the referenced Object.
00067         */
00068         typedef typename T::Holder ValueHolder;
00069 
00070         /**
00071         * Result type for a non-const get operation.
00072         */
00073         typedef TypedHolder<T> GetType;
00074 
00075 
00076     // -------- constructors ------------------------------------------------
00077 
00078     public:
00079         /**
00080         * Construct a new MemberHolder referencing NULL via a handle.
00081         *
00082         * @param oGuardian  the object that protects this member
00083         */
00084         MemberHolder(const Object& oGuardian)
00085             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ false),
00086               m_po(NULL)
00087             {
00088             if (oGuardian._isEscaped())
00089                 {
00090                 m_prev = m_next = NULL;
00091                 }
00092             }
00093 
00094         /**
00095         * Construct a new MemberHolder referencing the specified Object.
00096         *
00097         * @param oGuardian  the object that protects this member
00098         * @param that       the Object to reference
00099         */
00100         MemberHolder(const Object& oGuardian, const TypedHolder<T>& that)
00101             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ 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         * Construct a new MemberHolder referencing the specified Object.
00113         *
00114         * @param oGuardian  the object that protects this member
00115         * @param that       the Object to reference
00116         * @param fMutable   true if the member is declared as mutable, false
00117         *                   if declared as const
00118         */
00119         MemberHolder(const Object& oGuardian, const TypedHolder<T>& that,
00120                 bool fMutable)
00121             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ 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         * Destroy the MemberHolder.
00134         */
00135         ~MemberHolder()
00136             {
00137             try
00138                 {
00139                 m_nMutability = inherited;
00140                 set(NULL);
00141                 }
00142             catch (const std::exception& e)
00143                 {
00144                 // Exception::View is not a known type within this file
00145                 std::cerr << "Error during ~MemberHolder: " << e.what() << std::endl;
00146                 return; // can't re-throw from within destructor
00147                 }
00148             }
00149 
00150     protected:
00151         /**
00152         * Construct a MemberHolder without any associated guardian.
00153         *
00154         * The new MemberHolder is not usable until the guardian is specified.
00155         */
00156         MemberHolder()
00157             : SmartMember(), ChainedHandleElement(/*fView*/ false), m_po(NULL)
00158             {
00159             }
00160 
00161     private:
00162         /**
00163         * Blocked copy constructor.
00164         */
00165         MemberHolder(const MemberHolder&);
00166 
00167 
00168     // ----- operators ------------------------------------------------------
00169 
00170     public:
00171         /**
00172         * Assign this MemberHolder to reference the same Object (and in the
00173         * same manner) as the specified MemberHolder.
00174         *
00175         * @param that  the object to reference
00176         *
00177         * @return a reference to this MemberHolder
00178         */
00179         MemberHolder& operator=(const MemberHolder& that)
00180             {
00181             set(that); // assign from snapshot
00182             return *this;
00183             }
00184 
00185         /**
00186         * Assign this MemberHolder to reference the same Object (and in the
00187         * same manner) as the specified MemberHolder.
00188         *
00189         * @param that  the object to reference
00190         *
00191         * @return a reference to this MemberHolder
00192         */
00193         MemberHolder& operator=(const TypedHolder<T>& that)
00194             {
00195             set(that);
00196             return *this;
00197             }
00198 
00199         /**
00200         * Return a View to the referenced Object.
00201         *
00202         * @return a View to the referenced Object
00203         */
00204         operator ValueView() const
00205             {
00206             if (m_prev == NULL)
00207                 {
00208                 if (m_nMutability == safe_immutable)
00209                     {
00210                     // safe_immutable, synchronization not required
00211                     return ValueView(m_po);
00212                     }
00213                 else
00214                     {
00215                     SynchronizedMemberReadBlock::Guard guard(getGuardian());
00216                     return ValueView(m_po); // must occur in syncRead scope
00217                     }
00218                 }
00219             else
00220                 {
00221                 return TypedHandle<const T>(m_po, *this);
00222                 }
00223             }
00224 
00225         /**
00226         * Return a View to the referenced Object.
00227         *
00228         * @return a View to the referenced Object
00229         */
00230         template<class PT>
00231         operator TypedHandle<const PT>() const
00232             {
00233             return (ValueView) *this;
00234             }
00235 
00236         /**
00237         * Return a TypedHolder to the referenced Object.
00238         *
00239         * @return a TypedHolder to the referenced Object
00240         */
00241         template<class PT>
00242         operator TypedHolder<PT>() const
00243             {
00244             return get();
00245             }
00246 
00247         /**
00248         * Dereference the MemberHolder.
00249         *
00250         * @return a const pointer to the referenced Object
00251         */
00252         ValueView operator->() const
00253             {
00254             return (ValueView) *this;
00255             }
00256 
00257         /**
00258         * Dereference this handle, returning <tt>T&</tt>.
00259         *
00260         * @return a raw <tt>T&</tt> reference to the referenced Object
00261         *
00262         * @throws NullPointerException if the this handle is @c NULL
00263         */
00264         const T& operator*() const
00265             {
00266             return *get();
00267             }
00268 
00269     // ----- SmartMember interface ------------------------------------------
00270 
00271     protected:
00272         /**
00273         * {@inheritDoc}
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(/*fEscaped*/ true);
00302                         }
00303                     else
00304                         {
00305                         po->_detach(/*fEscaped*/ true);
00306                         }
00307                     }
00308                 m_prev = m_next = this;
00309                 }
00310 
00311             SmartMember::onEscape(fEscaped);
00312             }
00313 
00314         /**
00315         * {@inheritDoc}
00316         */
00317         virtual size64_t retained() const
00318             {
00319             TypedHolder<T> oh = get();
00320             return oh == NULL
00321                     ? 0
00322                     : oh->sizeOf(/*fDeep*/ true);
00323             }
00324 
00325     // ----- helper methods -------------------------------------------------
00326 
00327     protected:
00328         /**
00329         * Set the Holder to reference an Object via a View.
00330         *
00331         * @param that   the Object to reference
00332         * @param pSync  the synch block to delegate to or NULL to create one
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         * Set the escaped Holder to reference an Object via a View.
00355         *
00356         * @param that   the Object to reference
00357         * @param pSync  the synch block to delegate to or NULL to create one
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(/*fEscaped*/ true);
00373                     }
00374                 else
00375                     {
00376                     pNew->_attach(/*fEscaped*/ true);
00377                     }
00378                 }
00379 
00380             if (pSync == NULL)
00381                 {
00382                 // sync block
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                 // sync block
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(/*fEscaped*/ true);
00415                     }
00416                 else
00417                     {
00418                     pOld->_detach(/*fEscaped*/ true);
00419                     }
00420                 }
00421             }
00422 
00423         /**
00424          * Return a TypedHolder referencing the same Object as this
00425          * MemberHolder
00426          *
00427          * @param pSync  the sync block to delegate to or NULL to create one
00428          *
00429          * @return a TypedHolder referencing the same Object as this
00430          *         MemberHolder
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          * Return a TypedHolder referencing the same Object as this
00450          * MemberHolder
00451          *
00452          * @param pSync  the sync block to delegate to or NULL to create one
00453          *
00454          * @return a TypedHolder referencing the same Object as this
00455          *         MemberHolder
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); // must occur in syncRead scope
00467                     }
00468                 return ValueHandle(m_po); // must occur in syncRead scope
00469                 }
00470             else if (m_nMutability == safe_immutable)
00471                 {
00472                 // safe_immutable, synchronization not required
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); // must occur in syncRead scope
00485                     }
00486                 return ValueHandle(m_po); // must occur in syncRead scope
00487                 }
00488             }
00489 
00490         /**
00491         * Perform the specified action.
00492         *
00493         * @param nAction  the action to perform
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(/*fEscaped*/ false);
00504                     // fall through
00505                 case action_detach:
00506                     if (NULL != po)
00507                         {
00508                         if (m_fView)
00509                             {
00510                             ((const T*) po)->_detach(/*fEscaped*/ false);
00511                             }
00512                         else
00513                             {
00514                             po->_detach(/*fEscaped*/ false);
00515                             }
00516                         }
00517                     // fall through
00518                 case action_none:
00519                 default:
00520                     break;
00521                 }
00522             }
00523 
00524 
00525     // ----- data members ---------------------------------------------------
00526 
00527     protected:
00528         /**
00529         * The referenced Object.
00530         */
00531         T* m_po;
00532 
00533 
00534     // ----- friends --------------------------------------------------------
00535 
00536     /**
00537     * @internal
00538     */
00539     template<class> friend class Array;
00540 
00541     /**
00542     * @internal
00543     */
00544     friend class SynchronizedMemberReadBlock;
00545 
00546     /**
00547     * @internal
00548     */
00549     friend class SynchronizedMemberWriteBlock;
00550     };
00551 
00552 
00553 // ----- non-member operators and functions ---------------------------------
00554 
00555 /**
00556 * Output a human-readable description of the given MemberHolder to the
00557 * specified stream.
00558 *
00559 * @param out  the stream used to output the description
00560 * @param th   the MemberHolder to describe
00561 *
00562 * @return the supplied stream
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 * Assign the specified holder to NULL.
00573 *
00574 * @param mh the holder to clear
00575 */
00576 template<class T> void clear_handle(MemberHolder<T>& mh)
00577     {
00578     mh = NULL;
00579     }
00580 
00581 /**
00582 * Return true if the supplied holder equals NULL.
00583 *
00584 * @param mh  the holder to test
00585 *
00586 * @return true iff the supplied holder equals NULL
00587 */
00588 template<class T>
00589 bool is_null(const MemberHolder<T>& mh)
00590     {
00591     return mh == NULL;
00592     }
00593 
00594 /**
00595 * Perform a dynamic cast the pointer associated with the MemberHolder
00596 * to a the specified handle/view type.
00597 *
00598 * @param mh      the MemberHolder from which to perform the cast
00599 * @param fThrow  true if an exception is to be thrown on a failed cast
00600 *
00601 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00602 *
00603 * @throws ClassCastException if the cast fails and fThrow is true
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 * Perform an instanceof check on a handle or view.
00613 *
00614 * @param mh  the MemberHolder from which to perform the test
00615 *
00616 * @return true if the supplied handle is an instance of the specified type
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
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.