Author: markt Date: Tue Jun 19 10:53:53 2018 New Revision: 1833812 URL: http://svn.apache.org/viewvc?rev=1833812&view=rev Log: Fix https://bz.apache.org/bugzilla/show_bug.cgi?id=62458 Update the internal fork of Commons Pool to dfef97b (2018-06-18) to pick up some bug fixes and enhancements.
Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/AbandonedConfig.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseGenericObjectPool.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseObjectPoolConfig.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionPolicy.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionTimer.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPool.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolConfig.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolMXBean.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPool.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPoolConfig.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/LinkedBlockingDeque.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/NoOpCallStack.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/PoolImplUtils.java tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/SoftReferenceObjectPool.java Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/AbandonedConfig.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/AbandonedConfig.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/AbandonedConfig.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/AbandonedConfig.java Tue Jun 19 10:53:53 2018 @@ -200,7 +200,7 @@ public class AbandonedConfig { * @see #getRequireFullStackTrace() * @since 2.5 */ - public void setRequireFullStackTrace(boolean requireFullStackTrace) { + public void setRequireFullStackTrace(final boolean requireFullStackTrace) { this.requireFullStackTrace = requireFullStackTrace; } Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseGenericObjectPool.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseGenericObjectPool.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseGenericObjectPool.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseGenericObjectPool.java Tue Jun 19 10:53:53 2018 @@ -21,10 +21,12 @@ import java.io.StringWriter; import java.io.Writer; import java.lang.management.ManagementFactory; import java.lang.ref.WeakReference; +import java.lang.reflect.InvocationTargetException; import java.util.Arrays; import java.util.Deque; import java.util.Iterator; import java.util.TimerTask; +import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicLong; @@ -38,6 +40,7 @@ import javax.management.ObjectName; import org.apache.tomcat.dbcp.pool2.BaseObject; import org.apache.tomcat.dbcp.pool2.PooledObject; +import org.apache.tomcat.dbcp.pool2.PooledObjectState; import org.apache.tomcat.dbcp.pool2.SwallowedExceptionListener; /** @@ -60,6 +63,8 @@ public abstract class BaseGenericObjectP */ public static final int MEAN_TIMING_STATS_CACHE_SIZE = 100; + private static final String EVICTION_POLICY_TYPE_NAME = EvictionPolicy.class.getName(); + // Configuration attributes private volatile int maxTotal = GenericKeyedObjectPoolConfig.DEFAULT_MAX_TOTAL; @@ -106,7 +111,7 @@ public abstract class BaseGenericObjectP // Monitoring (primarily JMX) attributes - private final ObjectName oname; + private final ObjectName objectName; private final String creationStackTrace; private final AtomicLong borrowedCount = new AtomicLong(0); private final AtomicLong returnedCount = new AtomicLong(0); @@ -130,12 +135,12 @@ public abstract class BaseGenericObjectP * overridden by the config * @param jmxNamePrefix Prefix to be used for JMX name for the new pool */ - public BaseGenericObjectPool(final BaseObjectPoolConfig config, + public BaseGenericObjectPool(final BaseObjectPoolConfig<T> config, final String jmxNameBase, final String jmxNamePrefix) { if (config.getJmxEnabled()) { - this.oname = jmxRegister(config, jmxNameBase, jmxNamePrefix); + this.objectName = jmxRegister(config, jmxNameBase, jmxNamePrefix); } else { - this.oname = null; + this.objectName = null; } // Populate the creation stack trace @@ -446,9 +451,11 @@ public abstract class BaseGenericObjectP } /** - * Sets the number of milliseconds to sleep between runs of the idle - * object evictor thread. When non-positive, no idle object evictor thread - * will be run. + * Sets the number of milliseconds to sleep between runs of the idle object evictor thread. + * <ul> + * <li>When positive, the idle object evictor thread starts.</li> + * <li>When non-positive, no idle object evictor thread runs.</li> + * </ul> * * @param timeBetweenEvictionRunsMillis * number of milliseconds to sleep between evictor runs @@ -588,47 +595,75 @@ public abstract class BaseGenericObjectP } /** - * Sets the name of the {@link EvictionPolicy} implementation that is - * used by this pool. The Pool will attempt to load the class using the - * thread context class loader. If that fails, the Pool will attempt to load - * the class using the class loader that loaded this class. + * Sets the eviction policy for this pool. + * + * @param evictionPolicy + * the eviction policy for this pool. + * @since 2.6.0 + */ + public void setEvictionPolicy(final EvictionPolicy<T> evictionPolicy) { + this.evictionPolicy = evictionPolicy; + } + + /** + * Sets the name of the {@link EvictionPolicy} implementation that is used by this pool. The Pool will attempt to + * load the class using the given class loader. If that fails, use the class loader for the {@link EvictionPolicy} + * interface. * - * @param evictionPolicyClassName the fully qualified class name of the - * new eviction policy + * @param evictionPolicyClassName + * the fully qualified class name of the new eviction policy + * @param classLoader + * the class loader to load the given {@code evictionPolicyClassName}. * * @see #getEvictionPolicyClassName() + * @since 2.6.0 If loading the class using the given class loader fails, use the class loader for the + * {@link EvictionPolicy} interface. */ - public final void setEvictionPolicyClassName( - final String evictionPolicyClassName) { + public final void setEvictionPolicyClassName(final String evictionPolicyClassName, final ClassLoader classLoader) { + // Getting epClass here and now best matches the caller's environment + final Class<?> epClass = EvictionPolicy.class; + final ClassLoader epClassLoader = epClass.getClassLoader(); try { - Class<?> clazz; try { - clazz = Class.forName(evictionPolicyClassName, true, - Thread.currentThread().getContextClassLoader()); - } catch (final ClassNotFoundException e) { - clazz = Class.forName(evictionPolicyClassName); - } - final Object policy = clazz.getConstructor().newInstance(); - if (policy instanceof EvictionPolicy<?>) { - @SuppressWarnings("unchecked") // safe, because we just checked the class - final - EvictionPolicy<T> evicPolicy = (EvictionPolicy<T>) policy; - this.evictionPolicy = evicPolicy; - } else { - throw new IllegalArgumentException("[" + evictionPolicyClassName + - "] does not implement EvictionPolicy"); + setEvictionPolicy(evictionPolicyClassName, classLoader); + } catch (final ClassCastException | ClassNotFoundException e) { + setEvictionPolicy(evictionPolicyClassName, epClassLoader); } - } catch (final ClassNotFoundException e) { - throw new IllegalArgumentException( - "Unable to create EvictionPolicy instance of type " + - evictionPolicyClassName, e); - } catch (final ReflectiveOperationException e) { - throw new IllegalArgumentException( - "Unable to create EvictionPolicy instance of type " + - evictionPolicyClassName, e); + } catch (final ClassCastException e) { + throw new IllegalArgumentException("Class " + evictionPolicyClassName + " from class loaders [" + + classLoader + ", " + epClassLoader + "] do not implement " + EVICTION_POLICY_TYPE_NAME); + } catch (final ClassNotFoundException | InstantiationException | IllegalAccessException + | InvocationTargetException | NoSuchMethodException e) { + final String exMessage = "Unable to create " + EVICTION_POLICY_TYPE_NAME + " instance of type " + + evictionPolicyClassName; + throw new IllegalArgumentException(exMessage, e); } } + @SuppressWarnings("unchecked") + private void setEvictionPolicy(final String className, final ClassLoader classLoader) + throws ClassNotFoundException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException { + final Class<?> clazz = Class.forName(className, true, classLoader); + final Object policy = clazz.getConstructor().newInstance(); + this.evictionPolicy = (EvictionPolicy<T>) policy; + } + + /** + * Sets the name of the {@link EvictionPolicy} implementation that is used by this pool. The Pool will attempt to + * load the class using the thread context class loader. If that fails, the use the class loader for the + * {@link EvictionPolicy} interface. + * + * @param evictionPolicyClassName + * the fully qualified class name of the new eviction policy + * + * @see #getEvictionPolicyClassName() + * @since 2.6.0 If loading the class using the thread context class loader fails, use the class loader for the + * {@link EvictionPolicy} interface. + */ + public final void setEvictionPolicyClassName(final String evictionPolicyClassName) { + setEvictionPolicyClassName(evictionPolicyClassName, Thread.currentThread().getContextClassLoader()); + } + /** * Gets the timeout that will be used when waiting for the Evictor to * shutdown if this pool is closed and it is the only pool still using the @@ -686,8 +721,9 @@ public abstract class BaseGenericObjectP * * @return the eviction policy * @since 2.4 + * @since 2.6.0 Changed access from protected to public. */ - protected EvictionPolicy<T> getEvictionPolicy() { + public EvictionPolicy<T> getEvictionPolicy() { return evictionPolicy; } @@ -742,7 +778,7 @@ public abstract class BaseGenericObjectP * @return the JMX name */ public final ObjectName getJmxName() { - return oname; + return objectName; } /** @@ -883,9 +919,9 @@ public abstract class BaseGenericObjectP * Swallows an exception and notifies the configured listener for swallowed * exceptions queue. * - * @param e exception to be swallowed + * @param swallowException exception to be swallowed */ - final void swallowException(final Exception e) { + final void swallowException(final Exception swallowException) { final SwallowedExceptionListener listener = getSwallowedExceptionListener(); if (listener == null) { @@ -893,11 +929,9 @@ public abstract class BaseGenericObjectP } try { - listener.onSwallowException(e); - } catch (final OutOfMemoryError oome) { - throw oome; - } catch (final VirtualMachineError vme) { - throw vme; + listener.onSwallowException(swallowException); + } catch (final VirtualMachineError e) { + throw e; } catch (final Throwable t) { // Ignore. Enjoy the irony. } @@ -934,16 +968,29 @@ public abstract class BaseGenericObjectP } /** + * Marks the object as returning to the pool. + * @param pooledObject instance to return to the keyed pool + */ + protected void markReturningState(PooledObject<T> pooledObject) { + synchronized(pooledObject) { + final PooledObjectState state = pooledObject.getState(); + if (state != PooledObjectState.ALLOCATED) { + throw new IllegalStateException( + "Object has already been returned to this pool or is invalid"); + } + pooledObject.markReturning(); // Keep from being marked abandoned + } + } + + /** * Unregisters this pool's MBean. */ final void jmxUnregister() { - if (oname != null) { + if (objectName != null) { try { ManagementFactory.getPlatformMBeanServer().unregisterMBean( - oname); - } catch (final MBeanRegistrationException e) { - swallowException(e); - } catch (final InstanceNotFoundException e) { + objectName); + } catch (final MBeanRegistrationException | InstanceNotFoundException e) { swallowException(e); } } @@ -962,7 +1009,7 @@ public abstract class BaseGenericObjectP * @param jmxNamePrefix name prefix * @return registered ObjectName, null if registration fails */ - private ObjectName jmxRegister(final BaseObjectPoolConfig config, + private ObjectName jmxRegister(final BaseObjectPoolConfig<T> config, final String jmxNameBase, String jmxNamePrefix) { ObjectName objectName = null; final MBeanServer mbs = ManagementFactory.getPlatformMBeanServer(); @@ -999,10 +1046,7 @@ public abstract class BaseGenericObjectP } catch (final InstanceAlreadyExistsException e) { // Increment the index and try again i++; - } catch (final MBeanRegistrationException e) { - // Shouldn't happen. Skip registration if it does. - registered = true; - } catch (final NotCompliantMBeanException e) { + } catch (final MBeanRegistrationException | NotCompliantMBeanException e) { // Shouldn't happen. Skip registration if it does. registered = true; } @@ -1032,7 +1076,10 @@ public abstract class BaseGenericObjectP * * @see GenericKeyedObjectPool#setTimeBetweenEvictionRunsMillis */ - class Evictor extends TimerTask { + class Evictor implements Runnable { + + private ScheduledFuture<?> scheduledFuture; + /** * Run pool maintenance. Evict objects qualifying for eviction and then * ensure that the minimum number of idle instances are available. @@ -1080,6 +1127,16 @@ public abstract class BaseGenericObjectP Thread.currentThread().setContextClassLoader(savedClassLoader); } } + + + void setScheduledFuture(final ScheduledFuture<?> scheduledFuture) { + this.scheduledFuture = scheduledFuture; + } + + + void cancel() { + scheduledFuture.cancel(false); + } } /** @@ -1296,7 +1353,7 @@ public abstract class BaseGenericObjectP builder.append(", factoryClassLoader="); builder.append(factoryClassLoader); builder.append(", oname="); - builder.append(oname); + builder.append(objectName); builder.append(", creationStackTrace="); builder.append(creationStackTrace); builder.append(", borrowedCount="); @@ -1323,4 +1380,5 @@ public abstract class BaseGenericObjectP builder.append(swallowedExceptionListener); } + } Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseObjectPoolConfig.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseObjectPoolConfig.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseObjectPoolConfig.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/BaseObjectPoolConfig.java Tue Jun 19 10:53:53 2018 @@ -24,10 +24,12 @@ import org.apache.tomcat.dbcp.pool2.Base * defined by the public constants. * <p> * This class is not thread-safe. + * </p> * + * @param <T> Type of element pooled. * @since 2.0 */ -public abstract class BaseObjectPoolConfig extends BaseObject implements Cloneable { +public abstract class BaseObjectPoolConfig<T> extends BaseObject implements Cloneable { /** * The default value for the {@code lifo} configuration attribute. @@ -159,9 +161,7 @@ public abstract class BaseObjectPoolConf * @see GenericObjectPool#getEvictionPolicyClassName() * @see GenericKeyedObjectPool#getEvictionPolicyClassName() */ - public static final String DEFAULT_EVICTION_POLICY_CLASS_NAME = - "org.apache.tomcat.dbcp.pool2.impl.DefaultEvictionPolicy"; - + public static final String DEFAULT_EVICTION_POLICY_CLASS_NAME = DefaultEvictionPolicy.class.getName(); private boolean lifo = DEFAULT_LIFO; @@ -181,6 +181,8 @@ public abstract class BaseObjectPoolConf private int numTestsPerEvictionRun = DEFAULT_NUM_TESTS_PER_EVICTION_RUN; + private EvictionPolicy<T> evictionPolicy = null; // Only 2.6.0 applications set this + private String evictionPolicyClassName = DEFAULT_EVICTION_POLICY_CLASS_NAME; private boolean testOnCreate = DEFAULT_TEST_ON_CREATE; @@ -553,6 +555,21 @@ public abstract class BaseObjectPoolConf } /** + * Get the value for the {@code evictionPolicyClass} configuration + * attribute for pools created with this configuration instance. + * + * @return The current setting of {@code evictionPolicyClass} for this + * configuration instance + * + * @see GenericObjectPool#getEvictionPolicy() + * @see GenericKeyedObjectPool#getEvictionPolicy() + * @since 2.6.0 + */ + public EvictionPolicy<T> getEvictionPolicy() { + return evictionPolicy; + } + + /** * Get the value for the {@code evictionPolicyClassName} configuration * attribute for pools created with this configuration instance. * @@ -567,6 +584,21 @@ public abstract class BaseObjectPoolConf } /** + * Set the value for the {@code evictionPolicyClass} configuration + * attribute for pools created with this configuration instance. + * + * @param evictionPolicy The new setting of + * {@code evictionPolicyClass} for this configuration instance + * + * @see GenericObjectPool#getEvictionPolicy() + * @see GenericKeyedObjectPool#getEvictionPolicy() + * @since 2.6.0 + */ + public void setEvictionPolicy(final EvictionPolicy<T> evictionPolicy) { + this.evictionPolicy = evictionPolicy; + } + + /** * Set the value for the {@code evictionPolicyClassName} configuration * attribute for pools created with this configuration instance. * Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionPolicy.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionPolicy.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionPolicy.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionPolicy.java Tue Jun 19 10:53:53 2018 @@ -40,6 +40,5 @@ public interface EvictionPolicy<T> { * @return <code>true</code> if the object should be evicted, otherwise * <code>false</code> */ - boolean evict(EvictionConfig config, PooledObject<T> underTest, - int idleCount); + boolean evict(EvictionConfig config, PooledObject<T> underTest, int idleCount); } Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionTimer.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionTimer.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionTimer.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/EvictionTimer.java Tue Jun 19 10:53:53 2018 @@ -18,7 +18,7 @@ package org.apache.tomcat.dbcp.pool2.imp import java.security.AccessController; import java.security.PrivilegedAction; -import java.util.TimerTask; +import java.util.concurrent.ScheduledFuture; import java.util.concurrent.ScheduledThreadPoolExecutor; import java.util.concurrent.ThreadFactory; import java.util.concurrent.TimeUnit; @@ -44,9 +44,6 @@ class EvictionTimer { /** Executor instance */ private static ScheduledThreadPoolExecutor executor; //@GuardedBy("EvictionTimer.class") - /** Static usage count tracker */ - private static int usageCount; //@GuardedBy("EvictionTimer.class") - /** Prevent instantiation */ private EvictionTimer() { // Hide the default constructor @@ -66,19 +63,22 @@ class EvictionTimer { /** * Add the specified eviction task to the timer. Tasks that are added with a - * call to this method *must* call {@link #cancel(TimerTask)} to cancel the - * task to prevent memory and/or thread leaks in application server - * environments. + * call to this method *must* call {@link #cancel(BaseGenericObjectPool.Evictor,long,TimeUnit)} + * to cancel the task to prevent memory and/or thread leaks in application + * server environments. * @param task Task to be scheduled * @param delay Delay in milliseconds before task is executed * @param period Time in milliseconds between executions */ - static synchronized void schedule(final Runnable task, final long delay, final long period) { + static synchronized void schedule( + final BaseGenericObjectPool<?>.Evictor task, final long delay, final long period) { if (null == executor) { executor = new ScheduledThreadPoolExecutor(1, new EvictorThreadFactory()); + executor.setRemoveOnCancelPolicy(true); } - usageCount++; - executor.scheduleWithFixedDelay(task, delay, period, TimeUnit.MILLISECONDS); + final ScheduledFuture<?> scheduledFuture = + executor.scheduleWithFixedDelay(task, delay, period, TimeUnit.MILLISECONDS); + task.setScheduledFuture(scheduledFuture); } /** @@ -90,10 +90,10 @@ class EvictionTimer { * terminate? * @param unit The units for the specified timeout */ - static synchronized void cancel(final TimerTask task, final long timeout, final TimeUnit unit) { + static synchronized void cancel( + final BaseGenericObjectPool<?>.Evictor task, final long timeout, final TimeUnit unit) { task.cancel(); - usageCount--; - if (usageCount == 0) { + if (executor.getQueue().size() == 0) { executor.shutdown(); try { executor.awaitTermination(timeout, unit); Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPool.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPool.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPool.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPool.java Tue Jun 19 10:53:53 2018 @@ -85,7 +85,7 @@ public class GenericKeyedObjectPool<K,T> * @param factory the factory to be used to create entries */ public GenericKeyedObjectPool(final KeyedPooledObjectFactory<K,T> factory) { - this(factory, new GenericKeyedObjectPoolConfig()); + this(factory, new GenericKeyedObjectPoolConfig<T>()); } /** @@ -98,8 +98,8 @@ public class GenericKeyedObjectPool<K,T> * the configuration object will not be reflected in the * pool. */ - public GenericKeyedObjectPool(final KeyedPooledObjectFactory<K,T> factory, - final GenericKeyedObjectPoolConfig config) { + public GenericKeyedObjectPool(final KeyedPooledObjectFactory<K, T> factory, + final GenericKeyedObjectPoolConfig<T> config) { super(config, ONAME_BASE, config.getJmxNamePrefix()); @@ -111,8 +111,6 @@ public class GenericKeyedObjectPool<K,T> this.fairness = config.getFairness(); setConfig(config); - - startEvictor(getTimeBetweenEvictionRunsMillis()); } /** @@ -237,7 +235,7 @@ public class GenericKeyedObjectPool<K,T> * * @see GenericKeyedObjectPoolConfig */ - public void setConfig(final GenericKeyedObjectPoolConfig conf) { + public void setConfig(final GenericKeyedObjectPoolConfig<T> conf) { setLifo(conf.getLifo()); setMaxIdlePerKey(conf.getMaxIdlePerKey()); setMaxTotalPerKey(conf.getMaxTotalPerKey()); @@ -251,11 +249,16 @@ public class GenericKeyedObjectPool<K,T> setTestWhileIdle(conf.getTestWhileIdle()); setNumTestsPerEvictionRun(conf.getNumTestsPerEvictionRun()); setMinEvictableIdleTimeMillis(conf.getMinEvictableIdleTimeMillis()); - setSoftMinEvictableIdleTimeMillis( - conf.getSoftMinEvictableIdleTimeMillis()); - setTimeBetweenEvictionRunsMillis( - conf.getTimeBetweenEvictionRunsMillis()); - setEvictionPolicyClassName(conf.getEvictionPolicyClassName()); + setSoftMinEvictableIdleTimeMillis(conf.getSoftMinEvictableIdleTimeMillis()); + setTimeBetweenEvictionRunsMillis(conf.getTimeBetweenEvictionRunsMillis()); + final EvictionPolicy<T> policy = conf.getEvictionPolicy(); + if (policy == null) { + // Use the class name (pre-2.6.0 compatible) + setEvictionPolicyClassName(conf.getEvictionPolicyClassName()); + } else { + // Otherwise, use the class (2.6.0 feature) + setEvictionPolicy(policy); + } setEvictorShutdownTimeoutMillis(conf.getEvictorShutdownTimeoutMillis()); } @@ -470,14 +473,7 @@ public class GenericKeyedObjectPool<K,T> "Returned object not currently part of this pool"); } - synchronized(p) { - final PooledObjectState state = p.getState(); - if (state != PooledObjectState.ALLOCATED) { - throw new IllegalStateException( - "Object has already been returned to this pool or is invalid"); - } - p.markReturning(); // Keep from being marked abandoned (once GKOP does this) - } + markReturningState(p); final long activeTime = p.getActiveTimeMillis(); @@ -489,13 +485,7 @@ public class GenericKeyedObjectPool<K,T> } catch (final Exception e) { swallowException(e); } - if (objectDeque.idleObjects.hasTakeWaiters()) { - try { - addObject(key); - } catch (final Exception e) { - swallowException(e); - } - } + whenWaitersAddObject(key, objectDeque.idleObjects); return; } } @@ -509,13 +499,7 @@ public class GenericKeyedObjectPool<K,T> } catch (final Exception e) { swallowException(e); } - if (objectDeque.idleObjects.hasTakeWaiters()) { - try { - addObject(key); - } catch (final Exception e) { - swallowException(e); - } - } + whenWaitersAddObject(key, objectDeque.idleObjects); return; } @@ -555,6 +539,20 @@ public class GenericKeyedObjectPool<K,T> } } + /** + * Whether there is at least one thread waiting on this deque, add an pool object. + * @param key + * @param idleObjects + */ + private void whenWaitersAddObject(final K key, LinkedBlockingDeque<PooledObject<T>> idleObjects) { + if (idleObjects.hasTakeWaiters()) { + try { + addObject(key); + } catch (final Exception e) { + swallowException(e); + } + } + } /** * {@inheritDoc} Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolConfig.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolConfig.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolConfig.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolConfig.java Tue Jun 19 10:53:53 2018 @@ -23,10 +23,12 @@ package org.apache.tomcat.dbcp.pool2.imp * <p> * This class is not thread-safe; it is only intended to be used to provide * attributes used when creating a pool. + * </p> * + * @param <T> Type of element pooled. * @since 2.0 */ -public class GenericKeyedObjectPoolConfig extends BaseObjectPoolConfig { +public class GenericKeyedObjectPoolConfig<T> extends BaseObjectPoolConfig<T> { /** * The default value for the {@code maxTotalPerKey} configuration attribute. @@ -171,10 +173,11 @@ public class GenericKeyedObjectPoolConfi this.maxIdlePerKey = maxIdlePerKey; } + @SuppressWarnings("unchecked") @Override - public GenericKeyedObjectPoolConfig clone() { + public GenericKeyedObjectPoolConfig<T> clone() { try { - return (GenericKeyedObjectPoolConfig) super.clone(); + return (GenericKeyedObjectPoolConfig<T>) super.clone(); } catch (final CloneNotSupportedException e) { throw new AssertionError(); // Can't happen } Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolMXBean.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolMXBean.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolMXBean.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericKeyedObjectPoolMXBean.java Tue Jun 19 10:53:53 2018 @@ -34,169 +34,204 @@ import java.util.Map; * @since 2.0 */ public interface GenericKeyedObjectPoolMXBean<K> { + // Expose getters for configuration settings + /** * See {@link GenericKeyedObjectPool#getBlockWhenExhausted()} * @return See {@link GenericKeyedObjectPool#getBlockWhenExhausted()} */ boolean getBlockWhenExhausted(); + /** * See {@link GenericKeyedObjectPool#getFairness()} * @return See {@link GenericKeyedObjectPool#getFairness()} */ boolean getFairness(); + /** * See {@link GenericKeyedObjectPool#getLifo()} * @return See {@link GenericKeyedObjectPool#getLifo()} */ boolean getLifo(); + /** * See {@link GenericKeyedObjectPool#getMaxIdlePerKey()} * @return See {@link GenericKeyedObjectPool#getMaxIdlePerKey()} */ int getMaxIdlePerKey(); + /** * See {@link GenericKeyedObjectPool#getMaxTotal()} * @return See {@link GenericKeyedObjectPool#getMaxTotal()} */ int getMaxTotal(); + /** * See {@link GenericKeyedObjectPool#getMaxTotalPerKey()} * @return See {@link GenericKeyedObjectPool#getMaxTotalPerKey()} */ int getMaxTotalPerKey(); + /** * See {@link GenericKeyedObjectPool#getMaxWaitMillis()} * @return See {@link GenericKeyedObjectPool#getMaxWaitMillis()} */ long getMaxWaitMillis(); + /** * See {@link GenericKeyedObjectPool#getMinEvictableIdleTimeMillis()} * @return See {@link GenericKeyedObjectPool#getMinEvictableIdleTimeMillis()} */ long getMinEvictableIdleTimeMillis(); + /** * See {@link GenericKeyedObjectPool#getMinIdlePerKey()} * @return See {@link GenericKeyedObjectPool#getMinIdlePerKey()} */ int getMinIdlePerKey(); + /** * See {@link GenericKeyedObjectPool#getNumActive()} * @return See {@link GenericKeyedObjectPool#getNumActive()} */ int getNumActive(); + /** * See {@link GenericKeyedObjectPool#getNumIdle()} * @return See {@link GenericKeyedObjectPool#getNumIdle()} */ int getNumIdle(); + /** * See {@link GenericKeyedObjectPool#getNumTestsPerEvictionRun()} * @return See {@link GenericKeyedObjectPool#getNumTestsPerEvictionRun()} */ int getNumTestsPerEvictionRun(); + /** * See {@link GenericKeyedObjectPool#getTestOnCreate()} * @return See {@link GenericKeyedObjectPool#getTestOnCreate()} * @since 2.2 */ boolean getTestOnCreate(); + /** * See {@link GenericKeyedObjectPool#getTestOnBorrow()} * @return See {@link GenericKeyedObjectPool#getTestOnBorrow()} */ boolean getTestOnBorrow(); + /** * See {@link GenericKeyedObjectPool#getTestOnReturn()} * @return See {@link GenericKeyedObjectPool#getTestOnReturn()} */ boolean getTestOnReturn(); + /** * See {@link GenericKeyedObjectPool#getTestWhileIdle()} * @return See {@link GenericKeyedObjectPool#getTestWhileIdle()} */ boolean getTestWhileIdle(); + /** * See {@link GenericKeyedObjectPool#getTimeBetweenEvictionRunsMillis()} * @return See {@link GenericKeyedObjectPool#getTimeBetweenEvictionRunsMillis()} */ long getTimeBetweenEvictionRunsMillis(); + /** * See {@link GenericKeyedObjectPool#isClosed()} * @return See {@link GenericKeyedObjectPool#isClosed()} */ boolean isClosed(); + // Expose getters for monitoring attributes + /** * See {@link GenericKeyedObjectPool#getNumActivePerKey()} * @return See {@link GenericKeyedObjectPool#getNumActivePerKey()} */ Map<String,Integer> getNumActivePerKey(); + /** * See {@link GenericKeyedObjectPool#getBorrowedCount()} * @return See {@link GenericKeyedObjectPool#getBorrowedCount()} */ long getBorrowedCount(); + /** * See {@link GenericKeyedObjectPool#getReturnedCount()} * @return See {@link GenericKeyedObjectPool#getReturnedCount()} */ long getReturnedCount(); + /** * See {@link GenericKeyedObjectPool#getCreatedCount()} * @return See {@link GenericKeyedObjectPool#getCreatedCount()} */ long getCreatedCount(); + /** * See {@link GenericKeyedObjectPool#getDestroyedCount()} * @return See {@link GenericKeyedObjectPool#getDestroyedCount()} */ long getDestroyedCount(); + /** * See {@link GenericKeyedObjectPool#getDestroyedByEvictorCount()} * @return See {@link GenericKeyedObjectPool#getDestroyedByEvictorCount()} */ long getDestroyedByEvictorCount(); + /** * See {@link GenericKeyedObjectPool#getDestroyedByBorrowValidationCount()} * @return See {@link GenericKeyedObjectPool#getDestroyedByBorrowValidationCount()} */ long getDestroyedByBorrowValidationCount(); + /** * See {@link GenericKeyedObjectPool#getMeanActiveTimeMillis()} * @return See {@link GenericKeyedObjectPool#getMeanActiveTimeMillis()} */ long getMeanActiveTimeMillis(); + /** * See {@link GenericKeyedObjectPool#getMeanIdleTimeMillis()} * @return See {@link GenericKeyedObjectPool#getMeanIdleTimeMillis()} */ long getMeanIdleTimeMillis(); + /** * See {@link GenericKeyedObjectPool#getMaxBorrowWaitTimeMillis()} * @return See {@link GenericKeyedObjectPool#getMaxBorrowWaitTimeMillis()} */ long getMeanBorrowWaitTimeMillis(); + /** * See {@link GenericKeyedObjectPool#getMaxBorrowWaitTimeMillis()} * @return See {@link GenericKeyedObjectPool#getMaxBorrowWaitTimeMillis()} */ long getMaxBorrowWaitTimeMillis(); + /** * See {@link GenericKeyedObjectPool#getCreationStackTrace()} * @return See {@link GenericKeyedObjectPool#getCreationStackTrace()} */ String getCreationStackTrace(); + /** * See {@link GenericKeyedObjectPool#getNumWaiters()} * @return See {@link GenericKeyedObjectPool#getNumWaiters()} */ int getNumWaiters(); + /** * See {@link GenericKeyedObjectPool#getNumWaitersByKey()} * @return See {@link GenericKeyedObjectPool#getNumWaitersByKey()} */ Map<String,Integer> getNumWaitersByKey(); + /** * See {@link GenericKeyedObjectPool#listAllObjects()} * @return See {@link GenericKeyedObjectPool#listAllObjects()} Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPool.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPool.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPool.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPool.java Tue Jun 19 10:53:53 2018 @@ -77,18 +77,18 @@ public class GenericObjectPool<T> extend implements ObjectPool<T>, GenericObjectPoolMXBean, UsageTracking<T> { /** - * Create a new <code>GenericObjectPool</code> using defaults from + * Creates a new <code>GenericObjectPool</code> using defaults from * {@link GenericObjectPoolConfig}. * * @param factory The object factory to be used to create object instances * used by this pool */ public GenericObjectPool(final PooledObjectFactory<T> factory) { - this(factory, new GenericObjectPoolConfig()); + this(factory, new GenericObjectPoolConfig<T>()); } /** - * Create a new <code>GenericObjectPool</code> using a specific + * Creates a new <code>GenericObjectPool</code> using a specific * configuration. * * @param factory The object factory to be used to create object instances @@ -99,7 +99,7 @@ public class GenericObjectPool<T> extend * pool. */ public GenericObjectPool(final PooledObjectFactory<T> factory, - final GenericObjectPoolConfig config) { + final GenericObjectPoolConfig<T> config) { super(config, ONAME_BASE, config.getJmxNamePrefix()); @@ -112,12 +112,10 @@ public class GenericObjectPool<T> extend idleObjects = new LinkedBlockingDeque<>(config.getFairness()); setConfig(config); - - startEvictor(getTimeBetweenEvictionRunsMillis()); } /** - * Create a new <code>GenericObjectPool</code> that tracks and destroys + * Creates a new <code>GenericObjectPool</code> that tracks and destroys * objects that are checked out, but never returned to the pool. * * @param factory The object factory to be used to create object instances @@ -130,7 +128,7 @@ public class GenericObjectPool<T> extend * and removal. The configuration is used by value. */ public GenericObjectPool(final PooledObjectFactory<T> factory, - final GenericObjectPoolConfig config, final AbandonedConfig abandonedConfig) { + final GenericObjectPoolConfig<T> config, final AbandonedConfig abandonedConfig) { this(factory, config); setAbandonedConfig(abandonedConfig); } @@ -221,7 +219,7 @@ public class GenericObjectPool<T> extend } /** - * Whether or not abandoned object removal is configured for this pool. + * Gets whether or not abandoned object removal is configured for this pool. * * @return true if this pool is configured to detect and remove * abandoned objects @@ -232,7 +230,7 @@ public class GenericObjectPool<T> extend } /** - * Will this pool identify and log any abandoned objects? + * Gets whether this pool identifies and logs any abandoned objects. * * @return {@code true} if abandoned object removal is configured for this * pool and removal events are to be logged otherwise {@code false} @@ -246,8 +244,8 @@ public class GenericObjectPool<T> extend } /** - * Will a check be made for abandoned objects when an object is borrowed - * from this pool? + * Gets whether a check is made for abandoned objects when an object is borrowed + * from this pool. * * @return {@code true} if abandoned object removal is configured to be * activated by borrowObject otherwise {@code false} @@ -261,7 +259,7 @@ public class GenericObjectPool<T> extend } /** - * Will a check be made for abandoned objects when the evictor runs? + * Gets whether a check is made for abandoned objects when the evictor runs. * * @return {@code true} if abandoned object removal is configured to be * activated when the evictor runs otherwise {@code false} @@ -275,7 +273,7 @@ public class GenericObjectPool<T> extend } /** - * Obtain the timeout before which an object will be considered to be + * Obtains the timeout before which an object will be considered to be * abandoned by this pool. * * @return The abandoned object timeout in seconds if abandoned object @@ -297,7 +295,7 @@ public class GenericObjectPool<T> extend * * @see GenericObjectPoolConfig */ - public void setConfig(final GenericObjectPoolConfig conf) { + public void setConfig(final GenericObjectPoolConfig<T> conf) { setLifo(conf.getLifo()); setMaxIdle(conf.getMaxIdle()); setMinIdle(conf.getMinIdle()); @@ -310,11 +308,16 @@ public class GenericObjectPool<T> extend setTestWhileIdle(conf.getTestWhileIdle()); setNumTestsPerEvictionRun(conf.getNumTestsPerEvictionRun()); setMinEvictableIdleTimeMillis(conf.getMinEvictableIdleTimeMillis()); - setTimeBetweenEvictionRunsMillis( - conf.getTimeBetweenEvictionRunsMillis()); - setSoftMinEvictableIdleTimeMillis( - conf.getSoftMinEvictableIdleTimeMillis()); - setEvictionPolicyClassName(conf.getEvictionPolicyClassName()); + setTimeBetweenEvictionRunsMillis(conf.getTimeBetweenEvictionRunsMillis()); + setSoftMinEvictableIdleTimeMillis(conf.getSoftMinEvictableIdleTimeMillis()); + final EvictionPolicy<T> policy = conf.getEvictionPolicy(); + if (policy == null) { + // Use the class name (pre-2.6.0 compatible) + setEvictionPolicyClassName(conf.getEvictionPolicyClassName()); + } else { + // Otherwise, use the class (2.6.0 feature) + setEvictionPolicy(policy); + } setEvictorShutdownTimeoutMillis(conf.getEvictorShutdownTimeoutMillis()); } @@ -341,7 +344,7 @@ public class GenericObjectPool<T> extend } /** - * Obtain a reference to the factory used to create, destroy and validate + * Obtains a reference to the factory used to create, destroy and validate * the objects used by this pool. * * @return the factory @@ -362,7 +365,7 @@ public class GenericObjectPool<T> extend } /** - * Borrow an object from the pool using the specific waiting time which only + * Borrows an object from the pool using the specific waiting time which only * applies if {@link #getBlockWhenExhausted()} is true. * <p> * If there is one or more idle instance available in the pool, then an @@ -533,14 +536,7 @@ public class GenericObjectPool<T> extend return; // Object was abandoned and removed } - synchronized(p) { - final PooledObjectState state = p.getState(); - if (state != PooledObjectState.ALLOCATED) { - throw new IllegalStateException( - "Object has already been returned to this pool or is invalid"); - } - p.markReturning(); // Keep from being marked abandoned - } + markReturningState(p); final long activeTime = p.getActiveTimeMillis(); @@ -888,7 +884,7 @@ public class GenericObjectPool<T> extend final PooledObject<T> p; try { p = factory.makeObject(); - } catch (final Exception e) { + } catch (final Throwable e) { createCount.decrementAndGet(); throw e; } finally { @@ -976,7 +972,7 @@ public class GenericObjectPool<T> extend } /** - * Create an object, and place it into the pool. addObject() is useful for + * Creates an object, and place it into the pool. addObject() is useful for * "pre-loading" a pool with idle objects. * <p> * If there is no capacity available to add to the pool, this is a no-op @@ -994,7 +990,7 @@ public class GenericObjectPool<T> extend } /** - * Add the provided wrapped pooled object to the set of idle objects for + * Adds the provided wrapped pooled object to the set of idle objects for * this pool. The object must already be part of the pool. If {@code p} * is null, this is a no-op (no exception, but no impact on the pool). * @@ -1014,7 +1010,7 @@ public class GenericObjectPool<T> extend } /** - * Calculate the number of objects to test in a run of the idle object + * Calculates the number of objects to test in a run of the idle object * evictor. * * @return The number of objects to test for validity @@ -1029,7 +1025,7 @@ public class GenericObjectPool<T> extend } /** - * Recover abandoned objects which have been checked out but + * Recovers abandoned objects which have been checked out but * not used since longer than the removeAbandonedTimeout. * * @param ac The configuration to use to identify abandoned objects @@ -1085,7 +1081,7 @@ public class GenericObjectPool<T> extend private volatile String factoryType = null; /** - * Return an estimate of the number of threads currently blocked waiting for + * Returns an estimate of the number of threads currently blocked waiting for * an object from the pool. This is intended for monitoring only, not for * synchronization control. * @@ -1101,7 +1097,7 @@ public class GenericObjectPool<T> extend } /** - * Return the type - including the specific type rather than the generic - + * Returns the type - including the specific type rather than the generic - * of the factory. * * @return A string representation of the factory type Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPoolConfig.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPoolConfig.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPoolConfig.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/GenericObjectPoolConfig.java Tue Jun 19 10:53:53 2018 @@ -23,10 +23,12 @@ package org.apache.tomcat.dbcp.pool2.imp * <p> * This class is not thread-safe; it is only intended to be used to provide * attributes used when creating a pool. + * </p> * + * @param <T> Type of element pooled. * @since 2.0 */ -public class GenericObjectPoolConfig extends BaseObjectPoolConfig { +public class GenericObjectPoolConfig<T> extends BaseObjectPoolConfig<T> { /** * The default value for the {@code maxTotal} configuration attribute. @@ -133,10 +135,11 @@ public class GenericObjectPoolConfig ext this.minIdle = minIdle; } + @SuppressWarnings("unchecked") @Override - public GenericObjectPoolConfig clone() { + public GenericObjectPoolConfig<T> clone() { try { - return (GenericObjectPoolConfig) super.clone(); + return (GenericObjectPoolConfig<T>) super.clone(); } catch (final CloneNotSupportedException e) { throw new AssertionError(); // Can't happen } Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/LinkedBlockingDeque.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/LinkedBlockingDeque.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/LinkedBlockingDeque.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/LinkedBlockingDeque.java Tue Jun 19 10:53:53 2018 @@ -901,7 +901,7 @@ class LinkedBlockingDeque<E> extends Abs } /** - * Empty the queue to the specified collection. + * Drains the queue to the specified collection. * * @param c The collection to add the elements to * @@ -920,7 +920,7 @@ class LinkedBlockingDeque<E> extends Abs } /** - * Empty no more than the specified number of elements from the queue to the + * Drains no more than the specified number of elements from the queue to the * specified collection. * * @param c collection to add the elements to @@ -1330,7 +1330,7 @@ class LinkedBlockingDeque<E> extends Abs } /** - * Save the state of this deque to a stream (that is, serialize it). + * Saves the state of this deque to a stream (that is, serialize it). * * @serialData The capacity (int), followed by elements (each an * {@code Object}) in the proper order, followed by a null @@ -1354,7 +1354,7 @@ class LinkedBlockingDeque<E> extends Abs } /** - * Reconstitute this deque from a stream (that is, + * Reconstitutes this deque from a stream (that is, * deserialize it). * @param s the stream */ @@ -1379,8 +1379,7 @@ class LinkedBlockingDeque<E> extends Abs // Monitoring methods /** - * Returns true if there are threads waiting to take instances from this deque. - * See disclaimer on accuracy in + * Returns true if there are threads waiting to take instances from this deque. See disclaimer on accuracy in * {@link java.util.concurrent.locks.ReentrantLock#hasWaiters(Condition)}. * * @return true if there is at least one thread waiting on this deque's notEmpty condition. @@ -1395,9 +1394,8 @@ class LinkedBlockingDeque<E> extends Abs } /** - * Returns the length of the queue of threads waiting to take instances from this deque. - * See disclaimer on accuracy in - * {@link java.util.concurrent.locks.ReentrantLock#getWaitQueueLength(Condition)}. + * Returns the length of the queue of threads waiting to take instances from this deque. See disclaimer on accuracy + * in {@link java.util.concurrent.locks.ReentrantLock#getWaitQueueLength(Condition)}. * * @return number of threads waiting on this deque's notEmpty condition. */ @@ -1411,8 +1409,7 @@ class LinkedBlockingDeque<E> extends Abs } /** - * Interrupts the threads currently waiting to take an object from the pool. - * See disclaimer on accuracy in + * Interrupts the threads currently waiting to take an object from the pool. See disclaimer on accuracy in * {@link java.util.concurrent.locks.ReentrantLock#getWaitingThreads(Condition)}. */ public void interuptTakeWaiters() { Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/NoOpCallStack.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/NoOpCallStack.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/NoOpCallStack.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/NoOpCallStack.java Tue Jun 19 10:53:53 2018 @@ -26,13 +26,16 @@ import java.io.PrintWriter; */ public class NoOpCallStack implements CallStack { + /** + * Singleton instance. + */ public static final CallStack INSTANCE = new NoOpCallStack(); private NoOpCallStack() { } @Override - public boolean printStackTrace(PrintWriter writer) { + public boolean printStackTrace(final PrintWriter writer) { return false; } Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/PoolImplUtils.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/PoolImplUtils.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/PoolImplUtils.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/PoolImplUtils.java Tue Jun 19 10:53:53 2018 @@ -32,84 +32,115 @@ class PoolImplUtils { /** * Identifies the concrete type of object that an object factory creates. * - * @param factory The factory to examine + * @param factoryClass + * The factory to examine * * @return the type of object the factory creates */ @SuppressWarnings("rawtypes") - static Class<?> getFactoryType(final Class<? extends PooledObjectFactory> factory) { - return (Class<?>) getGenericType(PooledObjectFactory.class, factory); + static Class<?> getFactoryType(final Class<? extends PooledObjectFactory> factoryClass) { + final Class<PooledObjectFactory> type = PooledObjectFactory.class; + final Object genericType = getGenericType(type, factoryClass); + if (genericType instanceof Integer) { + // POOL-324 org.apache.commons.pool2.impl.GenericObjectPool.getFactoryType() throws + // java.lang.ClassCastException + // + // A bit hackish, but we must handle cases when getGenericType() does not return a concrete types. + final ParameterizedType pi = getParameterizedType(type, factoryClass); + if (pi != null) { + final Type[] bounds = ((TypeVariable) pi.getActualTypeArguments()[((Integer) genericType).intValue()]).getBounds(); + if (bounds != null && bounds.length > 0) { + final Type bound0 = bounds[0]; + if (bound0 instanceof Class) { + return (Class<?>) bound0; + } + } + } + // last resort: Always return a Class + return Object.class; + } + return (Class<?>) genericType; } - /** - * Obtain the concrete type used by an implementation of an interface that - * uses a generic type. + * Obtains the concrete type used by an implementation of an interface that uses a generic type. * - * @param type The interface that defines a generic type - * @param clazz The class that implements the interface with a concrete type - * @param <T> The interface type + * @param type + * The interface that defines a generic type + * @param clazz + * The class that implements the interface with a concrete type + * @param <T> + * The interface type * * @return concrete type used by the implementation */ - private static <T> Object getGenericType(final Class<T> type, - final Class<? extends T> clazz) { + private static <T> Object getGenericType(final Class<T> type, final Class<? extends T> clazz) { + if (type == null || clazz == null) { + // Error will be logged further up the call stack + return null; + } // Look to see if this class implements the generic interface - - // Get all the interfaces - final Type[] interfaces = clazz.getGenericInterfaces(); - for (final Type iface : interfaces) { - // Only need to check interfaces that use generics - if (iface instanceof ParameterizedType) { - final ParameterizedType pi = (ParameterizedType) iface; - // Look for the generic interface - if (pi.getRawType() instanceof Class) { - if (type.isAssignableFrom((Class<?>) pi.getRawType())) { - return getTypeParameter( - clazz, pi.getActualTypeArguments()[0]); - } - } - } + final ParameterizedType pi = getParameterizedType(type, clazz); + if (pi != null) { + return getTypeParameter(clazz, pi.getActualTypeArguments()[0]); } // Interface not found on this class. Look at the superclass. @SuppressWarnings("unchecked") - final - Class<? extends T> superClazz = - (Class<? extends T>) clazz.getSuperclass(); + final Class<? extends T> superClass = (Class<? extends T>) clazz.getSuperclass(); - final Object result = getGenericType(type, superClazz); + final Object result = getGenericType(type, superClass); if (result instanceof Class<?>) { - // Superclass implements interface and defines explicit type for - // generic + // Superclass implements interface and defines explicit type for generic return result; } else if (result instanceof Integer) { - // Superclass implements interface and defines unknown type for - // generic + // Superclass implements interface and defines unknown type for generic // Map that unknown type to the generic types defined in this class - final ParameterizedType superClassType = - (ParameterizedType) clazz.getGenericSuperclass(); - return getTypeParameter(clazz, - superClassType.getActualTypeArguments()[ - ((Integer) result).intValue()]); + final ParameterizedType superClassType = (ParameterizedType) clazz.getGenericSuperclass(); + return getTypeParameter(clazz, superClassType.getActualTypeArguments()[((Integer) result).intValue()]); } else { // Error will be logged further up the call stack return null; } } + /** + * Gets the matching parameterized type or null. + * @param type + * The interface that defines a generic type + * @param clazz + * The class that implements the interface with a concrete type + * @param <T> + * The interface type + */ + private static <T> ParameterizedType getParameterizedType(final Class<T> type, final Class<? extends T> clazz) { + for (final Type iface : clazz.getGenericInterfaces()) { + // Only need to check interfaces that use generics + if (iface instanceof ParameterizedType) { + final ParameterizedType pi = (ParameterizedType) iface; + // Look for the generic interface + if (pi.getRawType() instanceof Class) { + if (type.isAssignableFrom((Class<?>) pi.getRawType())) { + return pi; + } + } + } + } + return null; + } /** - * For a generic parameter, return either the Class used or if the type - * is unknown, the index for the type in definition of the class + * For a generic parameter, return either the Class used or if the type is unknown, the index for the type in + * definition of the class * - * @param clazz defining class - * @param argType the type argument of interest + * @param clazz + * defining class + * @param argType + * the type argument of interest * - * @return An instance of {@link Class} representing the type used by the - * type parameter or an instance of {@link Integer} representing - * the index for the type in the definition of the defining class + * @return An instance of {@link Class} representing the type used by the type parameter or an instance of + * {@link Integer} representing the index for the type in the definition of the defining class */ private static Object getTypeParameter(final Class<?> clazz, final Type argType) { if (argType instanceof Class<?>) { Modified: tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/SoftReferenceObjectPool.java URL: http://svn.apache.org/viewvc/tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/SoftReferenceObjectPool.java?rev=1833812&r1=1833811&r2=1833812&view=diff ============================================================================== --- tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/SoftReferenceObjectPool.java (original) +++ tomcat/tc8.0.x/trunk/java/org/apache/tomcat/dbcp/pool2/impl/SoftReferenceObjectPool.java Tue Jun 19 10:53:53 2018 @@ -78,7 +78,7 @@ public class SoftReferenceObjectPool<T> } /** - * Borrow an object from the pool. If there are no idle instances available + * Borrows an object from the pool. If there are no idle instances available * in the pool, the configured factory's * {@link PooledObjectFactory#makeObject()} method is invoked to create a * new instance. @@ -243,7 +243,7 @@ public class SoftReferenceObjectPool<T> } /** - * Create an object, and place it into the pool. addObject() is useful for + * Creates an object, and places it into the pool. addObject() is useful for * "pre-loading" a pool with idle objects. * <p> * Before being added to the pool, the newly created instance is @@ -313,7 +313,7 @@ public class SoftReferenceObjectPool<T> } /** - * Return the number of instances currently borrowed from this pool. + * Returns the number of instances currently borrowed from this pool. * * @return the number of instances currently borrowed from this pool */ @@ -345,7 +345,7 @@ public class SoftReferenceObjectPool<T> } /** - * Close this pool, and free any resources associated with it. Invokes + * Closes this pool, and frees any resources associated with it. Invokes * {@link #clear()} to destroy and remove instances in the pool. * <p> * Calling {@link #addObject} or {@link #borrowObject} after invoking this @@ -382,7 +382,7 @@ public class SoftReferenceObjectPool<T> } /** - * Find the PooledSoftReference in allReferences that points to obj. + * Finds the PooledSoftReference in allReferences that points to obj. * * @param obj returning object * @return PooledSoftReference wrapping a soft reference to obj @@ -399,7 +399,7 @@ public class SoftReferenceObjectPool<T> } /** - * Destroy a {@code PooledSoftReference} and remove it from the idle and all + * Destroys a {@code PooledSoftReference} and removes it from the idle and all * references pools. * * @param toDestroy PooledSoftReference to destroy --------------------------------------------------------------------- To unsubscribe, e-mail: dev-unsubscr...@tomcat.apache.org For additional commands, e-mail: dev-h...@tomcat.apache.org