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

F79659-03

coherence/lang/FinalHandle.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_FINAL_HANDLE_HPP
00008 #define COH_FINAL_HANDLE_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/MemberHandle.hpp"
00013 #include "coherence/lang/TypedHandle.hpp"
00014 #include "coherence/lang/SynchronizedMemberWriteBlock.hpp"
00015 
00016 
00017 COH_OPEN_NAMESPACE2(coherence,lang)
00018 
00019 /**
00020 * FinalHandle is an immutable thread-safe handle used by an Object to
00021 * reference its non-const child Objects.
00022 *
00023 * FinalHandles may either be initialized with a value at
00024 * construction-time, or thereafter via the initialize() function.  A
00025 * FinalHandle may be initialized only if the following conditions
00026 * hold:
00027 * <ul>
00028 *   <li> The FinalHandle has not already been initialized
00029 *   (either during construction, or via <tt>initialize()</tt>)
00030 * </ul>
00031 * For example:
00032 *
00033 * @code
00034 * class Foo
00035 *   : public class_spec<Foo>
00036 *   {
00037 *   friend class factory<Foo>;
00038 *
00039 *   protected:
00040 *     Foo()
00041 *       : f_hBar(self()) // construction without initialization
00042 *       {}
00043 *
00044 *   public:
00045 *     void setBar(Bar::Handle hBar)
00046 *       {
00047 *       initialize(f_hBar, hBar); // post-construction initialization
00048 *       }
00049 *
00050 *   private:
00051 *     FinalHandle<Bar> f_hBar;
00052 *   };
00053 * @endcode
00054 *
00055 * FinalHandles transfer the constness of their parent Object. When a
00056 * FinalHandle is accessed from within a const method of the enclosing
00057 * "parent" class, it will only provide const access to the Object which it
00058 * references. If the enclosing Object becomes only accessed via views the
00059 * FinalHandle will also automatically, and permanently switch its reference
00060 * type from a handle to a view.
00061 *
00062 * Note: In the rare case that a FinalHandle is declared via the mutable
00063 *       keyword, the FinalHandle must be informed of this fact by setting
00064 *       fMutable to true during construction.  A mutable FinalHandle will
00065 *       will return Handles from const methods of the class it is a data
00066 *       member of.  The FinalHandle cannot be reassigned to reference another
00067 *       object.
00068 *
00069 * @author mf  2008.12.01
00070 *
00071 * @see FinalView
00072 * @see FinalHolder
00073 */
00074 template<class T>
00075 class FinalHandle
00076     : public MemberHandle<T>
00077     {
00078     // ----- typedefs -------------------------------------------------------
00079 
00080     public:
00081         /**
00082         * The type of the values the holder can reference.
00083         */
00084         typedef T ValueType;
00085 
00086         /**
00087         * The Handle type for the referenced Object.
00088         */
00089         typedef typename T::Handle ValueHandle;
00090 
00091         /**
00092         * The View type for the referenced Object.
00093         */
00094         typedef typename T::View ValueView;
00095 
00096 
00097     // -------- constructors ------------------------------------------------
00098 
00099     public:
00100         /**
00101         * Construct a new FinalHandle referencing NULL via a handle.
00102         *
00103         * @param oGuardian  the object that protects this member
00104         */
00105         FinalHandle(const Object& oGuardian)
00106             : MemberHandle<T>(oGuardian)
00107             {
00108             }
00109 
00110         /**
00111         * Construct a new FinalHandle referencing specified Object.
00112         *
00113         * @param oGuardian  the object that protects this member
00114         * @param that       the object to reference
00115         */
00116         FinalHandle(const Object& oGuardian, const ValueHandle& that)
00117             : MemberHandle<T>(oGuardian, that)
00118             {
00119             // Note: The underlying MemberHandle is initialized in the
00120             // inherited mutability state, allowing the FinalHandle to act
00121             // as a handle in non-const methods.
00122             if (MemberHandle<T>::m_po)
00123                 {
00124                 MemberHandle<T>::m_nMutability = MemberHandle<T>::safe_immutable;
00125                 }
00126             }
00127 
00128         /**
00129         * Construct a new FinalHandle referencing specified Object.
00130         *
00131         * @param oGuardian  the object that protects this member
00132         * @param that       the object to reference
00133         * @param fMutable   true if the member is declared as mutable, false
00134         *                   if declared as const
00135         */
00136         FinalHandle(const Object& oGuardian, const ValueHandle& that, bool fMutable)
00137             : MemberHandle<T>(oGuardian, that, fMutable)
00138             {
00139             if (MemberHandle<T>::m_po && !fMutable)
00140                 {
00141                 MemberHandle<T>::m_nMutability = MemberHandle<T>::safe_immutable;
00142                 }
00143             }
00144 
00145     private:
00146         /**
00147         * Blocked copy constructor
00148         */
00149         FinalHandle(const FinalHandle&);
00150 
00151 
00152     // ----- operators ------------------------------------------------------
00153 
00154     public:
00155         /**
00156         * Return a View to the referenced Object.
00157         *
00158         * @return a View to the referenced Object
00159         */
00160         operator ValueView() const
00161             {
00162             return getFinal();
00163             }
00164 
00165         /**
00166         * Return a Handle to the referenced Object.
00167         *
00168         * @return a Handle to the referenced Object
00169         */
00170         operator ValueHandle()
00171             {
00172             return getFinal();
00173             }
00174 
00175         /**
00176         * Return a View to the referenced Object.
00177         *
00178         * @return a View to the referenced Object
00179         */
00180         template<class PT>
00181         operator TypedHandle<const PT>() const
00182             {
00183             return getFinal();
00184             }
00185 
00186         /**
00187         * Return a Handle to the referenced Object.
00188         *
00189         * @return a Handle to the referenced Object
00190         */
00191         template<class PT>
00192         operator TypedHandle<PT>()
00193             {
00194             return getFinal();
00195             }
00196 
00197         /**
00198         * Return a TypedHolder to the referenced Object.
00199         *
00200         * @return a TypedHolder to the referenced Object
00201         *
00202         */
00203         template<class PT>
00204         operator TypedHolder<PT>() const
00205             {
00206             return getFinal();
00207             }
00208 
00209         /**
00210         * Return a TypedHolder to the referenced Object.
00211         *
00212         * @return a TypedHolder to the referenced Object
00213         */
00214         template<class PT>
00215         operator TypedHolder<PT>()
00216             {
00217             return getFinal();
00218             }
00219 
00220         /**
00221         * Dereference the FinalHandle.
00222         *
00223         * @return a const pointer to the referenced Object
00224         */
00225         const T* operator->() const
00226             {
00227             const T* cpo = MemberHandle<T>::m_po;
00228             if (MemberHandle<T>::m_nMutability < MemberHandle<T>::safe_immutable)
00229                 {
00230                 MemberHandle<T>::readBarrier();
00231                 if (cpo == NULL)
00232                     {
00233                     cpo = MemberHandle<T>::m_po;
00234                     MemberHandle<T>::readBarrier();
00235                     }
00236                 }
00237 
00238             if (NULL == cpo)
00239                 {
00240                 coh_throw_npe(typeid(const T));
00241                 }
00242             return cpo;
00243             }
00244 
00245         /**
00246         * Dereference the FinalHandle.
00247         *
00248         * @return a const pointer to the referenced Object
00249         */
00250         T* operator->()
00251             {
00252             T* po = MemberHandle<T>::m_po;
00253             if (po == NULL &&
00254                 MemberHandle<T>::m_nMutability < MemberHandle<T>::safe_immutable)
00255                 {
00256                 MemberHandle<T>::readBarrier();
00257                 po = MemberHandle<T>::m_po;
00258                 }
00259             if (MemberHandle<T>::m_fView)
00260                 {
00261                 coh_throw_illegal_state(
00262                     "attempt to access handle from const FinalHandle");
00263                 }
00264 
00265             if (NULL == po)
00266                 {
00267                 coh_throw_npe(typeid(T));
00268                 }
00269             return po;
00270             }
00271 
00272     private:
00273         /**
00274         * Blocked assignment operator.
00275         */
00276         FinalHandle& operator=(const ValueHandle& that);
00277 
00278         /**
00279         * Blocked assignment operator.
00280         */
00281         FinalHandle& operator=(const FinalHandle<T>& that);
00282 
00283         /**
00284         * Blocked assignment operator.
00285         */
00286         FinalHandle& operator=(FinalHandle<T>& that);
00287 
00288     // ----- SmartMember interface ------------------------------------------
00289 
00290     protected:
00291         /**
00292         * {@inheritDoc}
00293         */
00294         virtual size64_t retained() const
00295             {
00296             const T* cpo = MemberHandle<T>::m_po;
00297             if (cpo == NULL &&
00298                 MemberHandle<T>::m_nMutability < MemberHandle<T>::safe_immutable)
00299                 {
00300                 MemberHandle<T>::readBarrier();
00301                 cpo = MemberHandle<T>::m_po;
00302                 }
00303 
00304             return cpo == NULL
00305                     ? 0
00306                     : cpo->sizeOf(/*fDeep*/ true);
00307             }
00308 
00309 
00310     // ----- helper methods -------------------------------------------------
00311 
00312     protected:
00313         ValueView getFinal() const
00314             {
00315             const T* cpo = MemberHandle<T>::m_po;
00316             if (MemberHandle<T>::m_prev == NULL)
00317                 {
00318                 if (cpo == NULL &&
00319                     MemberHandle<T>::m_nMutability < MemberHandle<T>::safe_immutable)
00320                     {
00321                     MemberHandle<T>::readBarrier();
00322                     cpo = MemberHandle<T>::m_po;
00323                     }
00324                 return ValueView(cpo);
00325                 }
00326             else
00327                 {
00328                 return TypedHandle<const T>(cpo, *this);
00329                 }
00330             }
00331 
00332         ValueHandle getFinal()
00333             {
00334             T* po = MemberHandle<T>::m_po;
00335             if (MemberHandle<T>::m_fView)
00336                 {
00337                 coh_throw_illegal_state(
00338                     "attempt to access handle from const FinalHandle");
00339                 }
00340 
00341             if (MemberHandle<T>::m_prev == NULL)
00342                 {
00343                 if (po == NULL &&
00344                     MemberHandle<T>::m_nMutability < MemberHandle<T>::safe_immutable)
00345                     {
00346                     MemberHandle<T>::readBarrier();
00347                     }
00348                 return ValueHandle(po);
00349                 }
00350             else
00351                 {
00352                 return TypedHandle<T>(po, *this);
00353                 }
00354             }
00355 
00356 
00357         /**
00358         * Initialize the value of a FinalHandle after it is
00359         * constructed.  A FinalHandle may be initialized only if the
00360         * following conditions hold:
00361         * <ul>
00362         *   <li> The FinalHandle has not already been initialized
00363         *   (either during construction, or via <tt>initialize()</tt>)
00364         *   <li> The FinalHandle has not escaped
00365         * </ul>
00366         *
00367         * @param that  the value to initialize this FinalHandle to
00368         */
00369         void initialize(ValueHandle that)
00370             {
00371             if (MemberHandle<T>::m_nMutability >= MemberHandle<T>::forever_immutable)
00372                 {
00373                 coh_throw_illegal_state(
00374                     "attempt to initialize const FinalHandle");
00375                 }
00376             else if (MemberHandle<T>::m_prev == NULL)
00377                 {
00378                 SynchronizedMemberWriteBlock block(MemberHandle<T>::getGuardian());
00379                 if (MemberHandle<T>::m_po != NULL)
00380                     {
00381                     coh_throw_illegal_state(
00382                         "attempt to multiply initialize FinalHandle");
00383                     }
00384                 MemberHandle<T>::setEscaped(that, &block);
00385                 }
00386             else if (MemberHandle<T>::m_po != NULL)
00387                 {
00388                 coh_throw_illegal_state(
00389                     "attempt to multiply initialize FinalHandle");
00390                 }
00391             else if (that != NULL)
00392                 {
00393                 T* po = get_pointer(that);
00394                 MemberHandle<T>::m_po =
00395                     NULL == po || NULL == po->_attach(/*fEscaped*/ false)
00396                         ? NULL : po;
00397                 if (MemberHandle<T>::m_nMutability != MemberHandle<T>::forever_mutable)
00398                     {
00399                     MemberHandle<T>::m_nMutability = MemberHandle<T>::safe_immutable;
00400                     }
00401                 }
00402             }
00403 
00404     // ----- friends --------------------------------------------------------
00405 
00406     /**
00407     * @internal
00408     */
00409     template<class _T, class H> friend void initialize(FinalHandle<_T>& fh,
00410                                               H that);
00411     };
00412 
00413 /**
00414 * Perform a dynamic cast the pointer associated with the FinalHandle
00415 * to a the specified handle/view type.
00416 *
00417 * @param h       the FinalHandle from which to perform the cast
00418 * @param fThrow  true if an exception is to be thrown on a failed cast
00419 *
00420 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00421 *
00422 * @throws ClassCastException if the cast fails and fThrow is true
00423 */
00424 template<class D, class T>
00425 D cast(FinalHandle<T>& h, bool fThrow = true)
00426     {
00427     return cast<D>((typename FinalHandle<T>::ValueHandle) h, fThrow);
00428     }
00429 
00430 /**
00431 * Perform a dynamic cast the pointer associated with the FinalHandle
00432 * to a the specified handle/view type.
00433 *
00434 * @param h       the FinalHandle from which to perform the cast
00435 * @param fThrow  true if an exception is to be thrown on a failed cast
00436 *
00437 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00438 *
00439 * @throws ClassCastException if the cast fails and fThrow is true
00440 */
00441 template<class D, class T>
00442 D cast(const FinalHandle<T>& h, bool fThrow = true)
00443     {
00444     return cast<D>((typename FinalHandle<T>::ValueView) h, fThrow);
00445     }
00446 
00447 /**
00448 * Perform an instanceof check on a FinalHandle.
00449 *
00450 * @param h   the FinalHandle from which to perform the test
00451 *
00452 * @return true if the supplied handle is an instance of the specified type
00453 */
00454 template<class D, class T>
00455 bool instanceof(FinalHandle<T>& h)
00456     {
00457     return NULL != cast<D>(h, false);
00458     }
00459 
00460 /**
00461 * Perform an instanceof check on a FinalHandle.
00462 *
00463 * @param h   the FinalHandle from which to perform the test
00464 *
00465 * @return true if the supplied handle is an instance of the specified type
00466 */
00467 template<class D, class T>
00468 bool instanceof(const FinalHandle<T>& h)
00469     {
00470     return NULL != cast<D>(h, false);
00471     }
00472 
00473 /**
00474 * Initialize the value of a FinalHandle after it is constructed.  A
00475 * FinalHandle may be initialized only if the following conditions
00476 * hold:
00477 * <ul>
00478 *   <li> The FinalHandle has not already been initialized to a
00479 *   non-NULL value (either during construction, or via
00480 *   <tt>initialize()</tt>)
00481 * </ul>
00482 *
00483 * @param h     the FinalHandle to initialize
00484 * @param that  the value to initialize this FinalHandle to
00485 */
00486 template<class T, class H>
00487 void initialize(FinalHandle<T>& h, H that)
00488     {
00489     h.initialize(that);
00490     }
00491 
00492 COH_CLOSE_NAMESPACE2
00493 
00494 #endif // COH_FINAL_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.