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