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_DUAL_QUEUE_HPP 00008 #define COH_DUAL_QUEUE_HPP 00009 00010 #include "coherence/lang.ns" 00011 00012 #include "coherence/util/AbstractConcurrentQueue.hpp" 00013 #include "coherence/util/Iterator.hpp" 00014 #include "coherence/util/List.hpp" 00015 00016 COH_OPEN_NAMESPACE2(coherence,util) 00017 00018 00019 /** 00020 * The DualQueue is optimized for the producer consumer use case. 00021 * 00022 * Producers work on the tail of the queue, consumers operate on the head of 00023 * the queue. The two portions of the queue are maintained as separate lists, 00024 * and protected by separate locks. 00025 * 00026 * When a consumer looks at the head of the queue, if it is empty, the head 00027 * and tail will be swapped. 00028 * 00029 * @author nsa 2008.02.13 00030 */ 00031 class COH_EXPORT DualQueue 00032 : public class_spec<DualQueue, 00033 extends<AbstractConcurrentQueue> > 00034 { 00035 friend class factory<DualQueue>; 00036 00037 // ----- constructors --------------------------------------------------- 00038 00039 protected: 00040 /** 00041 * Create a new DualQueue. 00042 * 00043 * @return a new DualQueue 00044 */ 00045 DualQueue(); 00046 00047 00048 // ----- DualQueue interface -------------------------------------------- 00049 00050 protected: 00051 /** 00052 * Swap the head and the tail, but only if the head is empty and the 00053 * tail is not. The calling thread must already hold a the head lock. 00054 * 00055 * @return true iff the head and tail were swapped 00056 */ 00057 virtual bool swapNoWait(); 00058 00059 /** 00060 * Return the head element list. 00061 * 00062 * @return the head element list 00063 */ 00064 virtual List::Handle getHeadElementList(); 00065 00066 /** 00067 * Set the head element list. 00068 * 00069 * @param hList the new head list to set 00070 */ 00071 virtual void setHeadElementList(List::Handle hList); 00072 00073 /** 00074 * Return the element list (tail). 00075 * 00076 * @return the element list 00077 */ 00078 virtual List::Handle getElementList(); 00079 00080 /** 00081 * Set the element list (tail) 00082 * 00083 * @param hList the new list to set 00084 */ 00085 virtual void setElementList(List::Handle hList); 00086 00087 /** 00088 * Return the head lock. 00089 * 00090 * @return the head lock 00091 */ 00092 virtual Object::Handle getHeadLock(); 00093 00094 00095 // ----- Queue interface ------------------------------------------------ 00096 00097 public: 00098 /** 00099 * {@inheritDoc} 00100 */ 00101 virtual bool add(Object::Holder oh); 00102 00103 /** 00104 * {@inheritDoc} 00105 */ 00106 virtual bool addHead(Object::Holder oh); 00107 00108 /** 00109 * {@inheritDoc} 00110 */ 00111 virtual bool isEmpty() const; 00112 00113 /** 00114 * {@inheritDoc} 00115 */ 00116 virtual Object::Holder peekNoWait(); 00117 00118 /** 00119 * {@inheritDoc} 00120 */ 00121 virtual Object::Holder removeNoWait(); 00122 00123 00124 // ----- data members --------------------------------------------------- 00125 00126 protected: 00127 /** 00128 * Lock protecting operations on the head of the queue, and tail 00129 * swapping. We cannot simply lock on the head element list as it 00130 * gets swapped with the tail. 00131 * 00132 * To avoid deadlock issues the Queue lock should never be obtained 00133 * while holding the head lock. 00134 * 00135 * For example: 00136 * <pre> 00137 * COH_SYNCHRONIZED(getHeadLock()) 00138 * { 00139 * COH_SYNCHRONIZED(this) 00140 * { 00141 * // this is NOT ok 00142 * } 00143 * } 00144 * COH_SYNCHRONIZED(this) 00145 * { 00146 * COH_SYNCHRONIZED(getHeadLock()) 00147 * { 00148 * // this is ok 00149 * } 00150 * } 00151 * </pre> 00152 * 00153 * The latter approach was chosen as it allows users of the DualQueue 00154 * to perform external synchronization without risking deadlock. 00155 */ 00156 FinalHandle<Object> f_hHeadLock; 00157 00158 /** 00159 * The List that backs the queue. For a dual queue the ElementList is 00160 * the tail. 00161 */ 00162 MemberHandle<List> m_hElementList; 00163 00164 /** 00165 * The storage for the head of the queue. 00166 */ 00167 MemberHandle<List> m_hHeadElementList; 00168 }; 00169 00170 COH_CLOSE_NAMESPACE2 00171 00172 #endif // COH_DUAL_QUEUE_HPP