From 38008f1a2106a6c7b34dcc44b684f62b5e7a5267 Mon Sep 17 00:00:00 2001
From: Doug Lea
Date: Mon, 22 Jul 2013 15:26:11 +0100
Subject: [PATCH] 8020976: Ensure consistent insertion for ConcurrentHashMap
Reviewed-by: chegar
---
.../util/concurrent/ConcurrentHashMap.java | 127 ++++++++++++------
1 file changed, 87 insertions(+), 40 deletions(-)
diff --git a/jdk/src/share/classes/java/util/concurrent/ConcurrentHashMap.java b/jdk/src/share/classes/java/util/concurrent/ConcurrentHashMap.java
index 08e2bd38239..edd788afe25 100644
--- a/jdk/src/share/classes/java/util/concurrent/ConcurrentHashMap.java
+++ b/jdk/src/share/classes/java/util/concurrent/ConcurrentHashMap.java
@@ -265,7 +265,8 @@ import java.util.stream.Stream;
* @param the type of keys maintained by this map
* @param the type of mapped values
*/
-public class ConcurrentHashMap extends AbstractMap implements ConcurrentMap, Serializable {
+public class ConcurrentHashMap extends AbstractMap
+ implements ConcurrentMap, Serializable {
private static final long serialVersionUID = 7249069246763182397L;
/*
@@ -439,16 +440,18 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* related operations (which is the main reason we cannot use
* existing collections such as TreeMaps). TreeBins contain
* Comparable elements, but may contain others, as well as
- * elements that are Comparable but not necessarily Comparable
- * for the same T, so we cannot invoke compareTo among them. To
- * handle this, the tree is ordered primarily by hash value, then
- * by Comparable.compareTo order if applicable. On lookup at a
- * node, if elements are not comparable or compare as 0 then both
- * left and right children may need to be searched in the case of
- * tied hash values. (This corresponds to the full list search
- * that would be necessary if all elements were non-Comparable and
- * had tied hashes.) The red-black balancing code is updated from
- * pre-jdk-collections
+ * elements that are Comparable but not necessarily Comparable for
+ * the same T, so we cannot invoke compareTo among them. To handle
+ * this, the tree is ordered primarily by hash value, then by
+ * Comparable.compareTo order if applicable. On lookup at a node,
+ * if elements are not comparable or compare as 0 then both left
+ * and right children may need to be searched in the case of tied
+ * hash values. (This corresponds to the full list search that
+ * would be necessary if all elements were non-Comparable and had
+ * tied hashes.) On insertion, to keep a total ordering (or as
+ * close as is required here) across rebalancings, we compare
+ * classes and identityHashCodes as tie-breakers. The red-black
+ * balancing code is updated from pre-jdk-collections
* (http://gee.cs.oswego.edu/dl/classes/collections/RBCell.java)
* based in turn on Cormen, Leiserson, and Rivest "Introduction to
* Algorithms" (CLR).
@@ -478,6 +481,10 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* unused "Segment" class that is instantiated in minimal form
* only when serializing.
*
+ * Also, solely for compatibility with previous versions of this
+ * class, it extends AbstractMap, even though all of its methods
+ * are overridden, so it is just useless baggage.
+ *
* This file is organized to make things a little easier to follow
* while reading than they might otherwise: First the main static
* declarations and utilities, then fields, then main public
@@ -1352,6 +1359,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* Saves the state of the {@code ConcurrentHashMap} instance to a
* stream (i.e., serializes it).
* @param s the stream
+ * @throws java.io.IOException if an I/O error occurs
* @serialData
* the key (Object) and value (Object)
* for each key-value mapping, followed by a null pair.
@@ -1394,6 +1402,9 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
/**
* Reconstitutes the instance from a stream (that is, deserializes it).
* @param s the stream
+ * @throws ClassNotFoundException if the class of a serialized object
+ * could not be found
+ * @throws java.io.IOException if an I/O error occurs
*/
private void readObject(java.io.ObjectInputStream s)
throws java.io.IOException, ClassNotFoundException {
@@ -2080,6 +2091,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* Creates a new {@link Set} backed by a ConcurrentHashMap
* from the given type to {@code Boolean.TRUE}.
*
+ * @param the element type of the returned set
* @return the new set
* @since 1.8
*/
@@ -2094,9 +2106,10 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
*
* @param initialCapacity The implementation performs internal
* sizing to accommodate this many elements.
+ * @param the element type of the returned set
+ * @return the new set
* @throws IllegalArgumentException if the initial capacity of
* elements is negative
- * @return the new set
* @since 1.8
*/
public static KeySetView newKeySet(int initialCapacity) {
@@ -2643,19 +2656,18 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
p = pr;
else if ((pk = p.key) == k || (pk != null && k.equals(pk)))
return p;
- else if (pl == null && pr == null)
- break;
+ else if (pl == null)
+ p = pr;
+ else if (pr == null)
+ p = pl;
else if ((kc != null ||
(kc = comparableClassFor(k)) != null) &&
(dir = compareComparables(kc, k, pk)) != 0)
p = (dir < 0) ? pl : pr;
- else if (pl == null)
- p = pr;
- else if (pr == null ||
- (q = pr.findTreeNode(h, k, kc)) == null)
- p = pl;
- else
+ else if ((q = pr.findTreeNode(h, k, kc)) != null)
return q;
+ else
+ p = pl;
} while (p != null);
}
return null;
@@ -2681,6 +2693,23 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
static final int WAITER = 2; // set when waiting for write lock
static final int READER = 4; // increment value for setting read lock
+ /**
+ * Tie-breaking utility for ordering insertions when equal
+ * hashCodes and non-comparable. We don't require a total
+ * order, just a consistent insertion rule to maintain
+ * equivalence across rebalancings. Tie-breaking further than
+ * necessary simplifies testing a bit.
+ */
+ static int tieBreakOrder(Object a, Object b) {
+ int d;
+ if (a == null || b == null ||
+ (d = a.getClass().getName().
+ compareTo(b.getClass().getName())) == 0)
+ d = (System.identityHashCode(a) <= System.identityHashCode(b) ?
+ -1 : 1);
+ return d;
+ }
+
/**
* Creates bin with initial set of nodes headed by b.
*/
@@ -2697,21 +2726,21 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
r = x;
}
else {
- Object key = x.key;
- int hash = x.hash;
+ K k = x.key;
+ int h = x.hash;
Class> kc = null;
for (TreeNode p = r;;) {
int dir, ph;
- if ((ph = p.hash) > hash)
+ K pk = p.key;
+ if ((ph = p.hash) > h)
dir = -1;
- else if (ph < hash)
+ else if (ph < h)
dir = 1;
- else if ((kc != null ||
- (kc = comparableClassFor(key)) != null))
- dir = compareComparables(kc, key, p.key);
- else
- dir = 0;
- TreeNode xp = p;
+ else if ((kc == null &&
+ (kc = comparableClassFor(k)) == null) ||
+ (dir = compareComparables(kc, k, pk)) == 0)
+ dir = tieBreakOrder(k, pk);
+ TreeNode xp = p;
if ((p = (dir <= 0) ? p.left : p.right) == null) {
x.parent = xp;
if (dir <= 0)
@@ -2725,6 +2754,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
}
}
this.root = r;
+ assert checkInvariants(root);
}
/**
@@ -2805,8 +2835,9 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
*/
final TreeNode putTreeVal(int h, K k, V v) {
Class> kc = null;
+ boolean searched = false;
for (TreeNode p = root;;) {
- int dir, ph; K pk; TreeNode q, pr;
+ int dir, ph; K pk;
if (p == null) {
first = root = new TreeNode(h, k, v, null, null);
break;
@@ -2820,21 +2851,25 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
else if ((kc == null &&
(kc = comparableClassFor(k)) == null) ||
(dir = compareComparables(kc, k, pk)) == 0) {
- if (p.left == null)
- dir = 1;
- else if ((pr = p.right) == null ||
- (q = pr.findTreeNode(h, k, kc)) == null)
- dir = -1;
- else
- return q;
+ if (!searched) {
+ TreeNode q, ch;
+ searched = true;
+ if (((ch = p.left) != null &&
+ (q = ch.findTreeNode(h, k, kc)) != null) ||
+ ((ch = p.right) != null &&
+ (q = ch.findTreeNode(h, k, kc)) != null))
+ return q;
+ }
+ dir = tieBreakOrder(k, pk);
}
+
TreeNode xp = p;
- if ((p = (dir < 0) ? p.left : p.right) == null) {
+ if ((p = (dir <= 0) ? p.left : p.right) == null) {
TreeNode x, f = first;
first = x = new TreeNode(h, k, v, f, xp);
if (f != null)
f.prev = x;
- if (dir < 0)
+ if (dir <= 0)
xp.left = x;
else
xp.right = x;
@@ -3546,6 +3581,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case the action is not applied)
* @param action the action
+ * @param the return type of the transformer
* @since 1.8
*/
public void forEach(long parallelismThreshold,
@@ -3569,6 +3605,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* needed for this operation to be executed in parallel
* @param searchFunction a function returning a non-null
* result on success, else null
+ * @param the return type of the search function
* @return a non-null result from applying the given search
* function on each (key, value), or null if none
* @since 1.8
@@ -3592,6 +3629,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case it is not combined)
* @param reducer a commutative associative combining function
+ * @param the return type of the transformer
* @return the result of accumulating the given transformation
* of all (key, value) pairs
* @since 1.8
@@ -3710,6 +3748,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case the action is not applied)
* @param action the action
+ * @param the return type of the transformer
* @since 1.8
*/
public void forEachKey(long parallelismThreshold,
@@ -3733,6 +3772,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* needed for this operation to be executed in parallel
* @param searchFunction a function returning a non-null
* result on success, else null
+ * @param the return type of the search function
* @return a non-null result from applying the given search
* function on each key, or null if none
* @since 1.8
@@ -3775,6 +3815,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case it is not combined)
* @param reducer a commutative associative combining function
+ * @param the return type of the transformer
* @return the result of accumulating the given transformation
* of all keys
* @since 1.8
@@ -3894,6 +3935,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case the action is not applied)
* @param action the action
+ * @param the return type of the transformer
* @since 1.8
*/
public void forEachValue(long parallelismThreshold,
@@ -3917,6 +3959,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* needed for this operation to be executed in parallel
* @param searchFunction a function returning a non-null
* result on success, else null
+ * @param the return type of the search function
* @return a non-null result from applying the given search
* function on each value, or null if none
* @since 1.8
@@ -3958,6 +4001,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case it is not combined)
* @param reducer a commutative associative combining function
+ * @param the return type of the transformer
* @return the result of accumulating the given transformation
* of all values
* @since 1.8
@@ -4075,6 +4119,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case the action is not applied)
* @param action the action
+ * @param the return type of the transformer
* @since 1.8
*/
public void forEachEntry(long parallelismThreshold,
@@ -4098,6 +4143,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* needed for this operation to be executed in parallel
* @param searchFunction a function returning a non-null
* result on success, else null
+ * @param the return type of the search function
* @return a non-null result from applying the given search
* function on each entry, or null if none
* @since 1.8
@@ -4139,6 +4185,7 @@ public class ConcurrentHashMap extends AbstractMap implements Concurre
* for an element, or null if there is no transformation (in
* which case it is not combined)
* @param reducer a commutative associative combining function
+ * @param the return type of the transformer
* @return the result of accumulating the given transformation
* of all entries
* @since 1.8