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

F79659-03

coherence/lang/ChainedHandleElement.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_CHAINED_HANDLE_ELEMENT_HPP
00008 #define COH_CHAINED_HANDLE_ELEMENT_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include <stdlib.h>
00013 
00014 COH_OPEN_NAMESPACE2(coherence,lang)
00015 
00016 /**
00017 * @internal
00018 *
00019 * The base class for TypedHandles, allowing for reference chaining.  Reference
00020 * chaining is not thread-safe and thus may only be used for handles which are
00021 * not shared across threads.
00022 *
00023 * @author mf  2008.10.20
00024 */
00025 class ChainedHandleElement
00026     {
00027     protected:
00028         /**
00029         * Action to take on unlinking
00030         */
00031         typedef enum
00032             {
00033             action_none   = 0, // no action required, chain still exists
00034             action_flip   = 1, // unlink last handle from handle->view chain
00035             action_detach = 2, // unlink final element of chain
00036             action_error  = 3  // corrupted chain, raise error
00037             } Action;
00038 
00039 
00040     // ----- constructors ---------------------------------------------------
00041 
00042     protected:
00043         /**
00044         * Create an independent element.
00045         */
00046         ChainedHandleElement(bool fView)
00047             : m_fView(fView)
00048             {
00049             // new chain
00050             m_prev = m_next = this;
00051             }
00052 
00053         /**
00054         * Create a dependent element, adding it to a chain.
00055         */
00056         ChainedHandleElement(const ChainedHandleElement& that, bool fView)
00057             : m_prev(&that), m_next(that.m_next), m_fView(fView)
00058             {
00059             // add self to chain
00060             m_next->m_prev = m_prev->m_next = this;
00061             }
00062 
00063         /**
00064         * Create an optionally dependent element.
00065         */
00066         ChainedHandleElement(const ChainedHandleElement* that, bool fView)
00067             : m_fView(fView)
00068             {
00069             if (that)
00070                 {
00071                 // add self to chain
00072                 m_prev = that;
00073                 m_next = that->m_next;
00074                 m_next->m_prev = m_prev->m_next = this;
00075                 }
00076             else
00077                 {
00078                 // new chain
00079                 m_prev = m_next = this;
00080                 }
00081             }
00082 
00083         /**
00084         * No destructor, derived class must perform unlink and take resulting
00085         * action.
00086         */
00087         //~ChainedHandleElement()
00088 
00089 
00090     // ----- ChainedHandleElement interface ---------------------------------
00091 
00092     protected:
00093         /**
00094         * Unlink this element from the chain.
00095         *
00096         * @return the action to be taken
00097         */
00098         Action unlink() const
00099             {
00100             Action nAction =
00101                     this == m_prev /* implies && this == m_next */ ? action_detach
00102                     // COH-9499 - removed m_prev and m_next m_fView pre-checks due to
00103                     // compiler issues with GCC 4.1
00104                   : !m_fView && scan()                             ? action_flip
00105                                                                    : action_none;
00106 
00107             // unlink self from chain
00108             m_prev->m_next = m_next;
00109             m_next->m_prev = m_prev;
00110 
00111             // empty chain
00112             m_prev = m_next = this;
00113 
00114             return nAction;
00115             }
00116 
00117         /**
00118         * Link this element to a new chain.
00119         *
00120         * @param that  the element to link to
00121         *
00122         * @return the action to be taken
00123         */
00124         Action link(const ChainedHandleElement& that) const
00125             {
00126             Action nAction =
00127                     this == m_prev /* implies && this == m_next */ ? action_detach
00128                     // COH-9499 - removed m_prev and m_next m_fView pre-checks due to
00129                     // compiler issues with GCC 4.1
00130                   : !m_fView && scan()                             ? action_flip
00131                                                                    : action_none;
00132 
00133             // unlink self from chain
00134             m_prev->m_next = m_next;
00135             m_next->m_prev = m_prev;
00136 
00137             // link self into new chain
00138             m_prev = &that;
00139             m_next = that.m_next;
00140 
00141             m_next->m_prev = that.m_next = this;
00142 
00143             return nAction;
00144             }
00145 
00146         /**
00147         * Scan the chain for additional handles.
00148         *
00149         * The caller must be a handle, and have found that m_prev != m_next != this
00150         *
00151         * @return true if this is the only handle in the chain
00152         */
00153         bool scan() const
00154             {
00155             // The current element must be a handle, so we can scan the list
00156             // until we encounter a handle; if it is this element then there
00157             // are no other handles present. While this is a potentially
00158             // expensive call it is also rarely occurs on a non-trivial chain.
00159             // Since you cannot assign a Handle from a View this will only
00160             // happen if multiple Views are injected in the middle of a handle
00161             // chain. Overall this minimal rare cost is more than worth it as
00162             // it allows chaining of Handles and Views, which removes the far
00163             // more expensive (potentially atomic) reference count increment
00164             // for the common pattern of passing a Handle to a function which
00165             // takes a View.
00166             ChainedHandleElement const* pElm = m_prev;
00167             if (pElm->m_fView && m_next->m_fView)
00168                 for (pElm = pElm->m_prev; pElm->m_fView; pElm = pElm->m_prev);
00169             // else; common case handle->this handle
00170             return pElm == this;
00171             }
00172 
00173 
00174     // ----- blocked methods ------------------------------------------------
00175 
00176     private:
00177         /**
00178         * Blocked dynamic allocation.
00179         */
00180         static void* operator new(size_t cb);
00181 
00182 
00183     // ----- data members ---------------------------------------------------
00184 
00185     protected:
00186         /**
00187         * The previous element in the chain.
00188         */
00189         mutable ChainedHandleElement const* m_prev;
00190 
00191         /**
00192         * The next element in the chain.
00193         */
00194         mutable ChainedHandleElement const* m_next;
00195 
00196         /**
00197         * Flag indicating if the associated handle is a view.
00198         */
00199         bool m_fView;
00200 
00201 
00202     // ----- friends --------------------------------------------------------
00203 
00204     /**
00205     * @internal
00206     */
00207     template<class> friend class TypedHandle;
00208     };
00209 
00210 COH_CLOSE_NAMESPACE2
00211 
00212 #endif // COH_CHAINED_HANDLE_ELEMENT_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.