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

F79659-03

coherence/lang/TimeoutBlock.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_TIMEOUT_BLOCK_HPP
00008 #define COH_TIMEOUT_BLOCK_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/Thread.hpp"
00013 
00014 COH_OPEN_NAMESPACE2(coherence, lang)
00015 
00016 /**
00017  * TimeoutBlock provides a mechanism for allowing a thread to interrupt itself if it doesn't return
00018  * to a specific call site within a given timeout.  TimeoutBlock instances are intended to be
00019  * used only with the COH_TIMEOUT_AFTER helper macro.  Once constructed a TimeoutBlock attempts to ensure that
00020  * the corresponding block completes within the specified timeout and if it does not the thread will
00021  * self-interrupt.  Exiting the timeout block will automatically clear any interrupt present on the thread
00022  * and in such a case an InterruptedException will be thrown.
00023  *
00024  * <pre>
00025  * try
00026  *     {
00027  *     COH_TIMEOUT_AFTER(5000)
00028  *         {
00029  *         doSomething();
00030  *         }
00031  *     } // this thread will self-interrupt if it doesn't reach this line within 5 seconds
00032  * catch (InterruptedException::View vEx)
00033  *     {
00034  *     // thread timed out or was otherwise interrupted
00035  *     }
00036  * </pre>
00037  *
00038  * Note that when catching the InterruptedException the preferred form is to surround the COH_TIMEOUT block with a try/catch
00039  * rather then embed the try/catch within the block.  This allows handling of cases where the thread gets interrupted but
00040  * the doSomething() method does not choose to throw the InterruptedException, in which case the TimeoutBlock destructor will
00041  * throw it, but this happens as the block terminates and thus would not be catchable within the block.
00042  *
00043  * Note that TimeoutBlock can only self-interrupt at interruptible points, and does not defend against
00044  * CPU bound loops for example.
00045  *
00046  * @author  mf 2015.03.03
00047  */
00048 class COH_EXPORT TimeoutBlock
00049     {
00050     // ----- constructors ---------------------------------------------------
00051 
00052     public:
00053         /**
00054          * Specify a new timeout.
00055          *
00056          * This constructor variant allows the caller to override a parent timeout.  This is
00057          * rarely needed, and is roughly the equivalent of silently consuming a thread interrupt
00058          * without rethrowing the InterruptedException.
00059          *
00060          * @param cMillis         the new timeout.
00061          * @param fForceOverride  true if this timeout is allowed to extend a parent timeout.
00062          */
00063         TimeoutBlock(int64_t cMillis, bool fForceOverride = false);
00064 
00065         /**
00066         * Copy constructor for COH_TIMEOUT macros.
00067         */
00068         TimeoutBlock(const TimeoutBlock& that);
00069 
00070         /**
00071         * Destroy a TimeoutBlock object.
00072         *
00073         * This will automatically reset the timeout to any former value.
00074         */
00075         ~TimeoutBlock();
00076 
00077 
00078     // ----- operators ------------------------------------------------------
00079 
00080     public:
00081         /*
00082         * Boolean conversion for use in COH_TIMEOUT macros.
00083         *
00084         * @return false always
00085         */
00086         operator bool() const
00087             {
00088             return false;
00089             }
00090 
00091     private:
00092         /**
00093         * Blocked assignment operator.
00094         */
00095         const TimeoutBlock& operator=(const TimeoutBlock&);
00096 
00097         /**
00098         * Blocked dynamic allocation.
00099         */
00100         static void* operator new(size_t);
00101 
00102 
00103     // ----- data members ---------------------------------------------------
00104 
00105     protected:
00106         /**
00107          * This TimeoutBlock's timeout.
00108          */
00109         int64_t m_cMillisTimeout;
00110 
00111         /**
00112          * The original timeout before this instance changed it.
00113          */
00114         int64_t m_lTimeoutOrig;
00115 
00116         /**
00117          * The current thread, or NULL if this block is inactive.
00118          */
00119         mutable Thread::Handle m_hThread;
00120     };
00121 
00122 COH_CLOSE_NAMESPACE2
00123 
00124 /**
00125 * Macro for making more readable timeout code blocks See the
00126 * documentation of TimeoutBlock for a usage example.
00127 *
00128 * @see coherence::lang::TimeoutBlock
00129 */
00130 #define COH_TIMEOUT_AFTER(CMILLIS) \
00131     if (coherence::lang::TimeoutBlock COH_UNIQUE_IDENTIFIER(_coh_timeout_) \
00132         = coherence::lang::TimeoutBlock(CMILLIS)) \
00133         { \
00134         COH_THROW(coherence::lang::IllegalStateException::create()); \
00135         } \
00136     else
00137 
00138 /**
00139 * Macro for making more readable timeout code blocks which is allowed to
00140 * extend an already active timeout.
00141 *
00142 * This variant allows the caller to extend a parent timeout.  This is rarely
00143 * needed, and is roughly the equivalent of silently consuming a thread interrupt
00144 * without rethrowing the InterruptedException.  Use of this method should
00145 * be extremely limited.
00146 *
00147 * @see coherence::lang::TimeoutBlock
00148 */
00149 #define COH_TIMEOUT_OVERRIDE(CMILLIS) \
00150     if (coherence::lang::TimeoutBlock COH_UNIQUE_IDENTIFIER(_coh_timeout_) \
00151         = coherence::lang::TimeoutBlock(CMILLIS, /*fForceOverride*/ true)) \
00152         { \
00153         COH_THROW(coherence::lang::IllegalStateException::create()); \
00154         } \
00155     else
00156 
00157 #endif // COH_TIMEOUT_BLOCK_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.