8169465: Deadlock in com.sun.jndi.ldap.pool.Connections

Reviewed-by: dfuchs, vtewari
This commit is contained in:
Rob McKenna 2016-12-16 19:15:37 +00:00
parent 4484fde336
commit 013944c3f4
2 changed files with 36 additions and 24 deletions

View File

@ -27,7 +27,6 @@ package com.sun.jndi.ldap.pool;
import java.util.ArrayList; // JDK 1.2 import java.util.ArrayList; // JDK 1.2
import java.util.List; import java.util.List;
import java.util.Iterator;
import java.lang.ref.Reference; import java.lang.ref.Reference;
import java.lang.ref.SoftReference; import java.lang.ref.SoftReference;
@ -290,24 +289,29 @@ final class Connections implements PoolCallback {
* @param threshold an entry idle since this time has expired. * @param threshold an entry idle since this time has expired.
* @return true if no more connections in list * @return true if no more connections in list
*/ */
synchronized boolean expire(long threshold) { boolean expire(long threshold) {
Iterator<ConnectionDesc> iter = conns.iterator(); List<ConnectionDesc> clonedConns;
ConnectionDesc entry; synchronized(this) {
while (iter.hasNext()) { clonedConns = new ArrayList<>(conns);
entry = iter.next(); }
List<ConnectionDesc> expired = new ArrayList<>();
for (ConnectionDesc entry : clonedConns) {
d("expire(): ", entry);
if (entry.expire(threshold)) { if (entry.expire(threshold)) {
d("expire(): removing ", entry); expired.add(entry);
td("Expired ", entry); td("expire(): Expired ", entry);
}
iter.remove(); // remove from pool }
synchronized (this) {
conns.removeAll(expired);
// Don't need to call notify() because we're // Don't need to call notify() because we're
// removing only idle connections. If there were // removing only idle connections. If there were
// idle connections, then there should be no waiters. // idle connections, then there should be no waiters.
}
}
return conns.isEmpty(); // whether whole list has 'expired' return conns.isEmpty(); // whether whole list has 'expired'
} }
}
/** /**
* Called when this instance of Connections has been removed from Pool. * Called when this instance of Connections has been removed from Pool.

View File

@ -25,11 +25,11 @@
package com.sun.jndi.ldap.pool; package com.sun.jndi.ldap.pool;
import java.util.ArrayList;
import java.util.Map; import java.util.Map;
import java.util.WeakHashMap; import java.util.WeakHashMap;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.Iterator;
import java.util.LinkedList; import java.util.LinkedList;
import java.io.PrintStream; import java.io.PrintStream;
@ -166,17 +166,25 @@ final public class Pool {
* and removed. * and removed.
*/ */
public void expire(long threshold) { public void expire(long threshold) {
Collection<ConnectionsRef> copy;
synchronized (map) { synchronized (map) {
Iterator<ConnectionsRef> iter = map.values().iterator(); copy = new ArrayList<>(map.values());
}
ArrayList<ConnectionsRef> removed = new ArrayList<>();
Connections conns; Connections conns;
while (iter.hasNext()) { for (ConnectionsRef ref : copy) {
conns = iter.next().getConnections(); conns = ref.getConnections();
if (conns.expire(threshold)) { if (conns.expire(threshold)) {
d("expire(): removing ", conns); d("expire(): removing ", conns);
iter.remove(); removed.add(ref);
} }
} }
synchronized (map) {
map.values().removeAll(removed);
} }
expungeStaleConnections(); expungeStaleConnections();
} }