On 5/13/11 11:25 PM, Phil Steitz wrote: > On 5/13/11 12:05 PM, ma...@apache.org wrote: >> Author: markt >> Date: Fri May 13 19:05:20 2011 >> New Revision: 1102868 >> >> URL: http://svn.apache.org/viewvc?rev=1102868&view=rev >> Log: >> Add the ability to register an interest in a key to prevent the associated >> ObjectDeque being removed from under methods that expect it to remain >> available for the life of the method. >> Fixes the issues found by commons-performance > Not quite. I am still getting the IllegalStateException > sporadically. The Deque is there, but empty (as before this change > - if it were not there, we would get NPE not ISE) when a return > fails. Once this happens, the pool becomes strangely completely > exhausted. A single failed return causes all threads to end up > waiting on borrow. I could understand how maxActive failed returns > could cause this, but I have seen it with just one ISE thrown. > > I see this > > [java] java.lang.IllegalStateException: Returned object not > currently part of this pool > [java] at > org.apache.commons.pool2.impl.GenericKeyedObjectPool.returnObject(GenericKeyedObjectPool.java:1162) > > And then execution stalls and a thread dump shows all threads like this: > > [java] "pool-1-thread-99" prio=5 tid=101b6c000 nid=0x11086f000 > waiting on condition [11086e000] > [java] java.lang.Thread.State: WAITING (parking) > [java] at sun.misc.Unsafe.park(Native Method) > [java] - parking to wait for <7f4351a08> (a > java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject) > [java] at > java.util.concurrent.locks.LockSupport.park(LockSupport.java:158) > [java] at > java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:1987) > [java] at > org.apache.commons.pool2.impl.LinkedBlockingDeque.takeFirst(LinkedBlockingDeque.java:469) > [java] at > org.apache.commons.pool2.impl.GenericKeyedObjectPool.borrowObject(GenericKeyedObjectPool.java:1061) > > The config-pool.xml in [performance] src/pool with > GenericKeyedObjectPool2 as the pool type leads to this about one in > every three runs for me. If you set the number of iterations to > 10000, you are almost sure to hit it.
Here is one explanation. I will try to put together a unit test and convince myself that the fix is correct. The painful-to-maintain-but-I-guess-useful feature of GKOP that it tries to clear the oldest elements from the pool when it needs to create new ones (really more a feature of an LRU cache than a pool) is enabled by calling clearOldest, now from within create. There is no synch in either of these methods. Inside clearOldest, a map sorted by idle time is created and idle instances are placed in that map to be killed off until enough have been removed. Since there is no synch, an instance could be borrowed while this is going on. The destroy method removes its victim from both allOjbects and idleObjects. My current theory is that this problem will be fixed if destroy only removes the instance from allObjects if it has been found in idleOjbects. This likely means that destroy has to distinguish "active" (returnObject, invalidateObject) vs "idle" (evict, clearOldest, clear) instance destruction. Phil > Phil >> Modified: >> >> commons/proper/pool/trunk/src/java/org/apache/commons/pool2/impl/GenericKeyedObjectPool.java >> >> Modified: >> commons/proper/pool/trunk/src/java/org/apache/commons/pool2/impl/GenericKeyedObjectPool.java >> URL: >> http://svn.apache.org/viewvc/commons/proper/pool/trunk/src/java/org/apache/commons/pool2/impl/GenericKeyedObjectPool.java?rev=1102868&r1=1102867&r2=1102868&view=diff >> ============================================================================== >> --- >> commons/proper/pool/trunk/src/java/org/apache/commons/pool2/impl/GenericKeyedObjectPool.java >> (original) >> +++ >> commons/proper/pool/trunk/src/java/org/apache/commons/pool2/impl/GenericKeyedObjectPool.java >> Fri May 13 19:05:20 2011 >> @@ -28,6 +28,10 @@ import java.util.TreeMap; >> import java.util.concurrent.ConcurrentHashMap; >> import java.util.concurrent.TimeUnit; >> import java.util.concurrent.atomic.AtomicInteger; >> +import java.util.concurrent.atomic.AtomicLong; >> +import java.util.concurrent.locks.Lock; >> +import java.util.concurrent.locks.ReadWriteLock; >> +import java.util.concurrent.locks.ReentrantReadWriteLock; >> >> import org.apache.commons.pool2.BaseKeyedObjectPool; >> import org.apache.commons.pool2.KeyedPoolableObjectFactory; >> @@ -1025,103 +1029,106 @@ public class GenericKeyedObjectPool<K,T> >> long maxWait = _maxWait; >> >> boolean create; >> - ObjectDeque<T> objectDeque = poolMap.get(key); >> + ObjectDeque<T> objectDeque = register(key); >> >> - while (p == null) { >> - create = false; >> - if (whenExhaustedAction == WhenExhaustedAction.FAIL) { >> - if (objectDeque != null) { >> - p = objectDeque.getIdleObjects().pollFirst(); >> - } >> - if (p == null) { >> - create = true; >> - p = create(key, false); >> - } >> - if (p == null) { >> - throw new NoSuchElementException("Pool exhausted"); >> - } >> - if (!p.allocate()) { >> - p = null; >> - } >> - } else if (whenExhaustedAction == WhenExhaustedAction.BLOCK) { >> - if (objectDeque != null) { >> - p = objectDeque.getIdleObjects().pollFirst(); >> - } >> - if (p == null) { >> - create = true; >> - p = create(key, false); >> - } >> - if (p == null && objectDeque != null) { >> - if (maxWait < 1) { >> - p = objectDeque.getIdleObjects().takeFirst(); >> - } else { >> - p = objectDeque.getIdleObjects().pollFirst(maxWait, >> - TimeUnit.MILLISECONDS); >> + try { >> + while (p == null) { >> + create = false; >> + if (whenExhaustedAction == WhenExhaustedAction.FAIL) { >> + if (objectDeque != null) { >> + p = objectDeque.getIdleObjects().pollFirst(); >> } >> - } >> - if (p == null) { >> - throw new NoSuchElementException( >> - "Timeout waiting for idle object"); >> - } >> - if (!p.allocate()) { >> - p = null; >> - } >> - } else if (whenExhaustedAction == WhenExhaustedAction.GROW) { >> - if (objectDeque != null) { >> - p = objectDeque.getIdleObjects().pollFirst(); >> - } >> - if (p == null) { >> - create = true; >> - p = create(key, true); >> - } >> - if (p != null && !p.allocate()) { >> - p = null; >> - } >> - } >> - >> - if (p != null) { >> - try { >> - _factory.activateObject(key, p.getObject()); >> - } catch (Exception e) { >> - try { >> - destroy(key, p); >> - } catch (Exception e1) { >> - // Ignore - activation failure is more important >> + if (p == null) { >> + create = true; >> + p = create(key, false); >> + } >> + if (p == null) { >> + throw new NoSuchElementException("Pool exhausted"); >> + } >> + if (!p.allocate()) { >> + p = null; >> + } >> + } else if (whenExhaustedAction == >> WhenExhaustedAction.BLOCK) { >> + if (objectDeque != null) { >> + p = objectDeque.getIdleObjects().pollFirst(); >> + } >> + if (p == null) { >> + create = true; >> + p = create(key, false); >> + } >> + if (p == null && objectDeque != null) { >> + if (maxWait < 1) { >> + p = objectDeque.getIdleObjects().takeFirst(); >> + } else { >> + p = >> objectDeque.getIdleObjects().pollFirst(maxWait, >> + TimeUnit.MILLISECONDS); >> + } >> + } >> + if (p == null) { >> + throw new NoSuchElementException( >> + "Timeout waiting for idle object"); >> + } >> + if (!p.allocate()) { >> + p = null; >> + } >> + } else if (whenExhaustedAction == WhenExhaustedAction.GROW) >> { >> + if (objectDeque != null) { >> + p = objectDeque.getIdleObjects().pollFirst(); >> + } >> + if (p == null) { >> + create = true; >> + p = create(key, true); >> } >> - p = null; >> - if (create) { >> - NoSuchElementException nsee = new >> NoSuchElementException( >> - "Unable to activate object"); >> - nsee.initCause(e); >> - throw nsee; >> + if (p != null && !p.allocate()) { >> + p = null; >> } >> } >> - if (p != null && getTestOnBorrow()) { >> - boolean validate = false; >> - Throwable validationThrowable = null; >> + >> + if (p != null) { >> try { >> - validate = _factory.validateObject(key, >> p.getObject()); >> - } catch (Throwable t) { >> - PoolUtils.checkRethrow(t); >> - } >> - if (!validate) { >> + _factory.activateObject(key, p.getObject()); >> + } catch (Exception e) { >> try { >> destroy(key, p); >> - } catch (Exception e) { >> - // Ignore - validation failure is more important >> + } catch (Exception e1) { >> + // Ignore - activation failure is more important >> } >> p = null; >> if (create) { >> NoSuchElementException nsee = new >> NoSuchElementException( >> - "Unable to validate object"); >> - nsee.initCause(validationThrowable); >> + "Unable to activate object"); >> + nsee.initCause(e); >> throw nsee; >> } >> } >> + if (p != null && getTestOnBorrow()) { >> + boolean validate = false; >> + Throwable validationThrowable = null; >> + try { >> + validate = _factory.validateObject(key, >> p.getObject()); >> + } catch (Throwable t) { >> + PoolUtils.checkRethrow(t); >> + } >> + if (!validate) { >> + try { >> + destroy(key, p); >> + } catch (Exception e) { >> + // Ignore - validation failure is more >> important >> + } >> + p = null; >> + if (create) { >> + NoSuchElementException nsee = new >> NoSuchElementException( >> + "Unable to validate object"); >> + nsee.initCause(validationThrowable); >> + throw nsee; >> + } >> + } >> + } >> } >> } >> + } finally { >> + deregister(key); >> } >> - >> return p.getObject(); >> } >> >> @@ -1258,22 +1265,28 @@ public class GenericKeyedObjectPool<K,T> >> @Override >> public void clear(K key) { >> >> - ObjectDeque<T> objectDeque = poolMap.get(key); >> - if (objectDeque == null) { >> - return; >> - } >> - LinkedBlockingDeque<PooledObject<T>> idleObjects = >> - objectDeque.getIdleObjects(); >> + register(key); >> >> - PooledObject<T> p = idleObjects.poll(); >> - >> - while (p != null) { >> - try { >> - destroy(key, p); >> - } catch (Exception e) { >> - // TODO - Ignore? >> + try { >> + ObjectDeque<T> objectDeque = poolMap.get(key); >> + if (objectDeque == null) { >> + return; >> + } >> + LinkedBlockingDeque<PooledObject<T>> idleObjects = >> + objectDeque.getIdleObjects(); >> + >> + PooledObject<T> p = idleObjects.poll(); >> + >> + while (p != null) { >> + try { >> + destroy(key, p); >> + } catch (Exception e) { >> + // TODO - Ignore? >> + } >> + p = idleObjects.poll(); >> } >> - p = idleObjects.poll(); >> + } finally { >> + deregister(key); >> } >> } >> >> @@ -1563,26 +1576,11 @@ public class GenericKeyedObjectPool<K,T> >> } >> } >> >> - // Make sure the key exists in the poolMap >> - ObjectDeque<T> objectDeque; >> - int newNumActive; >> - synchronized (poolMap) { >> - // This all has to be in the sync block to ensure that the key >> is >> - // not removed by destroy >> - objectDeque = poolMap.get(key); >> - if (objectDeque == null) { >> - objectDeque = new ObjectDeque<T>(); >> - newNumActive = objectDeque.getNumActive().incrementAndGet(); >> - poolMap.put(key, objectDeque); >> - poolKeyList.add(key); >> - } else { >> - newNumActive = objectDeque.getNumActive().incrementAndGet(); >> - } >> - } >> + ObjectDeque<T> objectDeque = poolMap.get(key); >> + int newNumActive = objectDeque.getNumActive().incrementAndGet(); >> >> // Check against the per key limit >> if (!force && maxActive > -1 && newNumActive > maxActive) { >> - cleanObjectDeque(key, objectDeque); >> numTotal.decrementAndGet(); >> return null; >> } >> @@ -1592,7 +1590,6 @@ public class GenericKeyedObjectPool<K,T> >> try { >> t = _factory.makeObject(key); >> } catch (Exception e) { >> - cleanObjectDeque(key, objectDeque); >> numTotal.decrementAndGet(); >> throw e; >> } >> @@ -1603,33 +1600,76 @@ public class GenericKeyedObjectPool<K,T> >> } >> >> private void destroy(K key, PooledObject<T> toDestory) throws Exception >> { >> - >> - ObjectDeque<T> objectDeque = poolMap.get(key); >> - objectDeque.getIdleObjects().remove(toDestory); >> - objectDeque.getAllObjects().remove(toDestory.getObject()); >> + >> + register(key); >> + >> + try { >> + ObjectDeque<T> objectDeque = poolMap.get(key); >> + objectDeque.getIdleObjects().remove(toDestory); >> + objectDeque.getAllObjects().remove(toDestory.getObject()); >> + >> + try { >> + _factory.destroyObject(key, toDestory.getObject()); >> + } finally { >> + objectDeque.getNumActive().decrementAndGet(); >> + numTotal.decrementAndGet(); >> + } >> + } finally { >> + deregister(key); >> + } >> + } >> >> + private ObjectDeque<T> register(K k) { >> + Lock lock = keyLock.readLock(); >> + ObjectDeque<T> objectDeque = null; >> try { >> - _factory.destroyObject(key, toDestory.getObject()); >> + lock.lock(); >> + objectDeque = poolMap.get(k); >> + if (objectDeque == null) { >> + // Upgrade to write lock >> + lock.unlock(); >> + lock = keyLock.writeLock(); >> + lock.lock(); >> + objectDeque = poolMap.get(k); >> + if (objectDeque == null) { >> + objectDeque = new ObjectDeque<T>(); >> + objectDeque.getNumInterested().incrementAndGet(); >> + poolMap.put(k, objectDeque); >> + poolKeyList.add(k); >> + } else { >> + objectDeque.getNumInterested().incrementAndGet(); >> + } >> + } else { >> + objectDeque.getNumInterested().incrementAndGet(); >> + } >> } finally { >> - cleanObjectDeque(key, objectDeque); >> - numTotal.decrementAndGet(); >> + lock.unlock(); >> } >> + return objectDeque; >> } >> + >> + private void deregister(K k) { >> + ObjectDeque<T> objectDeque; >> >> - private void cleanObjectDeque(K key, ObjectDeque<T> objectDeque) { >> - int newNumActive = objectDeque.getNumActive().decrementAndGet(); >> - if (newNumActive == 0) { >> - synchronized (poolMap) { >> - newNumActive = objectDeque.getNumActive().get(); >> - if (newNumActive == 0) { >> - poolMap.remove(key); >> - poolKeyList.remove(key); >> + // TODO Think carefully about when a read lock is required >> + objectDeque = poolMap.get(k); >> + long numInterested = >> objectDeque.getNumInterested().decrementAndGet(); >> + if (numInterested == 0 && objectDeque.getNumActive().get() == 0) { >> + // Potential to remove key >> + Lock lock = keyLock.writeLock(); >> + lock.lock(); >> + try { >> + if (objectDeque.getNumActive().get() == 0 && >> + objectDeque.getNumInterested().get() == 0) { >> + poolMap.remove(k); >> + poolKeyList.remove(k); >> } >> + } finally { >> + lock.unlock(); >> } >> } >> } >> >> - >> /** >> * Iterates through all the known keys and creates any necessary >> objects to maintain >> * the minimum level of pooled objects. >> @@ -1726,8 +1766,13 @@ public class GenericKeyedObjectPool<K,T> >> if (_factory == null) { >> throw new IllegalStateException("Cannot add objects without a >> factory."); >> } >> - PooledObject<T> p = create(key, false); >> - addIdleObject(key, p); >> + register(key); >> + try { >> + PooledObject<T> p = create(key, false); >> + addIdleObject(key, p); >> + } finally { >> + deregister(key); >> + } >> } >> >> /** >> @@ -1843,6 +1888,8 @@ public class GenericKeyedObjectPool<K,T> >> private Map<S, PooledObject<S>> allObjects = >> new ConcurrentHashMap<S, PooledObject<S>>(); >> >> + private AtomicLong numInterested = new AtomicLong(0); >> + >> public LinkedBlockingDeque<PooledObject<S>> getIdleObjects() { >> return idleObjects; >> } >> @@ -1851,6 +1898,10 @@ public class GenericKeyedObjectPool<K,T> >> return numActive; >> } >> >> + public AtomicLong getNumInterested() { >> + return numInterested; >> + } >> + >> public Map<S, PooledObject<S>> getAllObjects() { >> return allObjects; >> } >> @@ -2104,6 +2155,9 @@ public class GenericKeyedObjectPool<K,T> >> /** List of pool keys - used to control eviction order */ >> private List<K> poolKeyList = new ArrayList<K>(); >> >> + /** Lock used to manage adding/removing of keys */ >> + private ReadWriteLock keyLock = new ReentrantReadWriteLock(true); >> + >> /** >> * The combined count of the currently active objects for all keys and >> those >> * in the process of being created. Under load, it may exceed >> >> >> --------------------------------------------------------------------- To unsubscribe, e-mail: dev-unsubscr...@commons.apache.org For additional commands, e-mail: dev-h...@commons.apache.org