From d9d91c1cc42726831a50fd16eadeb21c2a0657b7 Mon Sep 17 00:00:00 2001
From: Doug Lea
Date: Tue, 3 Oct 2017 13:45:11 -0700
Subject: [PATCH] 8186171: HashMap: Entry.setValue may not work after
Iterator.remove() called for previous entries
Reviewed-by: martin, psandoz
---
.../share/classes/java/util/HashMap.java | 48 ++---
.../concurrent/tck/ConcurrentHashMapTest.java | 20 +-
.../tck/ConcurrentSkipListMapTest.java | 15 +-
.../java/util/concurrent/tck/HashMapTest.java | 60 ++++++
.../concurrent/tck/MapImplementation.java | 49 +++++
.../jdk/java/util/concurrent/tck/MapTest.java | 172 ++++++++++++++++++
.../java/util/concurrent/tck/TreeMapTest.java | 15 +-
7 files changed, 350 insertions(+), 29 deletions(-)
create mode 100644 test/jdk/java/util/concurrent/tck/HashMapTest.java
create mode 100644 test/jdk/java/util/concurrent/tck/MapImplementation.java
create mode 100644 test/jdk/java/util/concurrent/tck/MapTest.java
diff --git a/src/java.base/share/classes/java/util/HashMap.java b/src/java.base/share/classes/java/util/HashMap.java
index 4f0e8e6c7d6..4534656cd97 100644
--- a/src/java.base/share/classes/java/util/HashMap.java
+++ b/src/java.base/share/classes/java/util/HashMap.java
@@ -490,7 +490,7 @@ public class HashMap extends AbstractMap
}
/**
- * Implements Map.putAll and Map constructor
+ * Implements Map.putAll and Map constructor.
*
* @param m the map
* @param evict false when initially constructing this map, else
@@ -557,7 +557,7 @@ public class HashMap extends AbstractMap
}
/**
- * Implements Map.get and related methods
+ * Implements Map.get and related methods.
*
* @param hash hash for key
* @param key the key
@@ -612,7 +612,7 @@ public class HashMap extends AbstractMap
}
/**
- * Implements Map.put and related methods
+ * Implements Map.put and related methods.
*
* @param hash hash for key
* @param key the key
@@ -700,7 +700,7 @@ public class HashMap extends AbstractMap
}
threshold = newThr;
@SuppressWarnings({"rawtypes","unchecked"})
- Node[] newTab = (Node[])new Node[newCap];
+ Node[] newTab = (Node[])new Node[newCap];
table = newTab;
if (oldTab != null) {
for (int j = 0; j < oldCap; ++j) {
@@ -800,7 +800,7 @@ public class HashMap extends AbstractMap
}
/**
- * Implements Map.remove and related methods
+ * Implements Map.remove and related methods.
*
* @param hash hash for key
* @param key the key
@@ -875,7 +875,7 @@ public class HashMap extends AbstractMap
public boolean containsValue(Object value) {
Node[] tab; V v;
if ((tab = table) != null && size > 0) {
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next) {
if ((v = e.value) == value ||
(value != null && value.equals(v)))
@@ -927,7 +927,7 @@ public class HashMap extends AbstractMap
throw new NullPointerException();
if (size > 0 && (tab = table) != null) {
int mc = modCount;
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next)
action.accept(e.key);
}
@@ -975,7 +975,7 @@ public class HashMap extends AbstractMap
throw new NullPointerException();
if (size > 0 && (tab = table) != null) {
int mc = modCount;
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next)
action.accept(e.value);
}
@@ -1038,7 +1038,7 @@ public class HashMap extends AbstractMap
throw new NullPointerException();
if (size > 0 && (tab = table) != null) {
int mc = modCount;
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next)
action.accept(e);
}
@@ -1335,7 +1335,7 @@ public class HashMap extends AbstractMap
throw new NullPointerException();
if (size > 0 && (tab = table) != null) {
int mc = modCount;
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next)
action.accept(e.key, e.value);
}
@@ -1351,7 +1351,7 @@ public class HashMap extends AbstractMap
throw new NullPointerException();
if (size > 0 && (tab = table) != null) {
int mc = modCount;
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next) {
e.value = function.apply(e.key, e.value);
}
@@ -1394,9 +1394,10 @@ public class HashMap extends AbstractMap
}
/**
- * Save the state of the {@code HashMap} instance to a stream (i.e.,
- * serialize it).
+ * Saves this map to a stream (that is, serializes it).
*
+ * @param s the stream
+ * @throws IOException if an I/O error occurs
* @serialData The capacity of the HashMap (the length of the
* bucket array) is emitted (int), followed by the
* size (an int, the number of key-value
@@ -1415,8 +1416,11 @@ public class HashMap extends AbstractMap
}
/**
- * Reconstitute the {@code HashMap} instance from a stream (i.e.,
- * deserialize it).
+ * Reconstitutes this map 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 IOException if an I/O error occurs
*/
private void readObject(java.io.ObjectInputStream s)
throws IOException, ClassNotFoundException {
@@ -1445,7 +1449,7 @@ public class HashMap extends AbstractMap
threshold = ((cap < MAXIMUM_CAPACITY && ft < MAXIMUM_CAPACITY) ?
(int)ft : Integer.MAX_VALUE);
@SuppressWarnings({"rawtypes","unchecked"})
- Node[] tab = (Node[])new Node[cap];
+ Node[] tab = (Node[])new Node[cap];
table = tab;
// Read the keys and values, and put the mappings in the HashMap
@@ -1830,7 +1834,7 @@ public class HashMap extends AbstractMap
void internalWriteEntries(java.io.ObjectOutputStream s) throws IOException {
Node[] tab;
if (size > 0 && (tab = table) != null) {
- for (Node e : tab) {
+ for (Node e : tab) {
for (; e != null; e = e.next) {
s.writeObject(e.key);
s.writeObject(e.value);
@@ -1951,7 +1955,6 @@ public class HashMap extends AbstractMap
/**
* Forms tree of the nodes linked from this node.
- * @return root of tree
*/
final void treeify(Node[] tab) {
TreeNode root = null;
@@ -2089,8 +2092,11 @@ public class HashMap extends AbstractMap
return;
if (root.parent != null)
root = root.root();
- if (root == null || root.right == null ||
- (rl = root.left) == null || rl.left == null) {
+ if (root == null
+ || (movable
+ && (root.right == null
+ || (rl = root.left) == null
+ || rl.left == null))) {
tab[index] = first.untreeify(map); // too small
return;
}
@@ -2319,7 +2325,7 @@ public class HashMap extends AbstractMap
static TreeNode balanceDeletion(TreeNode root,
TreeNode x) {
- for (TreeNode xp, xpl, xpr;;) {
+ for (TreeNode xp, xpl, xpr;;) {
if (x == null || x == root)
return root;
else if ((xp = x.parent) == null) {
diff --git a/test/jdk/java/util/concurrent/tck/ConcurrentHashMapTest.java b/test/jdk/java/util/concurrent/tck/ConcurrentHashMapTest.java
index 007d3b6798d..dd54714e74a 100644
--- a/test/jdk/java/util/concurrent/tck/ConcurrentHashMapTest.java
+++ b/test/jdk/java/util/concurrent/tck/ConcurrentHashMapTest.java
@@ -26,8 +26,9 @@
* However, the following notice accompanied the original version of this
* file:
*
- * Written by Doug Lea with assistance from members of JCP JSR-166
- * Expert Group and released to the public domain, as explained at
+ * Written by Doug Lea and Martin Buchholz with assistance from
+ * members of JCP JSR-166 Expert Group and released to the public
+ * domain, as explained at
* http://creativecommons.org/publicdomain/zero/1.0/
* Other contributors include Andrew Wright, Jeffrey Hayes,
* Pat Fisher, Mike Judd.
@@ -45,14 +46,25 @@ import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import junit.framework.Test;
-import junit.framework.TestSuite;
public class ConcurrentHashMapTest extends JSR166TestCase {
public static void main(String[] args) {
main(suite(), args);
}
public static Test suite() {
- return new TestSuite(ConcurrentHashMapTest.class);
+ class Implementation implements MapImplementation {
+ public Class> klazz() { return ConcurrentHashMap.class; }
+ public Map emptyMap() { return new ConcurrentHashMap(); }
+ public Object makeKey(int i) { return i; }
+ public Object makeValue(int i) { return i; }
+ public boolean isConcurrent() { return true; }
+ public boolean permitsNullKeys() { return false; }
+ public boolean permitsNullValues() { return false; }
+ public boolean supportsSetValue() { return true; }
+ }
+ return newTestSuite(
+ ConcurrentHashMapTest.class,
+ MapTest.testSuite(new Implementation()));
}
/**
diff --git a/test/jdk/java/util/concurrent/tck/ConcurrentSkipListMapTest.java b/test/jdk/java/util/concurrent/tck/ConcurrentSkipListMapTest.java
index fdd9892f09a..376f1f7ebb8 100644
--- a/test/jdk/java/util/concurrent/tck/ConcurrentSkipListMapTest.java
+++ b/test/jdk/java/util/concurrent/tck/ConcurrentSkipListMapTest.java
@@ -45,14 +45,25 @@ import java.util.Set;
import java.util.concurrent.ConcurrentSkipListMap;
import junit.framework.Test;
-import junit.framework.TestSuite;
public class ConcurrentSkipListMapTest extends JSR166TestCase {
public static void main(String[] args) {
main(suite(), args);
}
public static Test suite() {
- return new TestSuite(ConcurrentSkipListMapTest.class);
+ class Implementation implements MapImplementation {
+ public Class> klazz() { return ConcurrentSkipListMap.class; }
+ public Map emptyMap() { return new ConcurrentSkipListMap(); }
+ public Object makeKey(int i) { return i; }
+ public Object makeValue(int i) { return i; }
+ public boolean isConcurrent() { return true; }
+ public boolean permitsNullKeys() { return false; }
+ public boolean permitsNullValues() { return false; }
+ public boolean supportsSetValue() { return false; }
+ }
+ return newTestSuite(
+ ConcurrentSkipListMapTest.class,
+ MapTest.testSuite(new Implementation()));
}
/**
diff --git a/test/jdk/java/util/concurrent/tck/HashMapTest.java b/test/jdk/java/util/concurrent/tck/HashMapTest.java
new file mode 100644
index 00000000000..357d28038bb
--- /dev/null
+++ b/test/jdk/java/util/concurrent/tck/HashMapTest.java
@@ -0,0 +1,60 @@
+/*
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+/*
+ * This file is available under and governed by the GNU General Public
+ * License version 2 only, as published by the Free Software Foundation.
+ * However, the following notice accompanied the original version of this
+ * file:
+ *
+ * Written by Doug Lea and Martin Buchholz with assistance from
+ * members of JCP JSR-166 Expert Group and released to the public
+ * domain, as explained at
+ * http://creativecommons.org/publicdomain/zero/1.0/
+ */
+
+import java.util.HashMap;
+import java.util.Map;
+
+import junit.framework.Test;
+
+public class HashMapTest extends JSR166TestCase {
+ public static void main(String[] args) {
+ main(suite(), args);
+ }
+
+ public static Test suite() {
+ class Implementation implements MapImplementation {
+ public Class> klazz() { return HashMap.class; }
+ public Map emptyMap() { return new HashMap(); }
+ public Object makeKey(int i) { return i; }
+ public Object makeValue(int i) { return i; }
+ public boolean isConcurrent() { return false; }
+ public boolean permitsNullKeys() { return true; }
+ public boolean permitsNullValues() { return true; }
+ public boolean supportsSetValue() { return true; }
+ }
+ return newTestSuite(
+ // HashMapTest.class,
+ MapTest.testSuite(new Implementation()));
+ }
+}
diff --git a/test/jdk/java/util/concurrent/tck/MapImplementation.java b/test/jdk/java/util/concurrent/tck/MapImplementation.java
new file mode 100644
index 00000000000..eba9a6609da
--- /dev/null
+++ b/test/jdk/java/util/concurrent/tck/MapImplementation.java
@@ -0,0 +1,49 @@
+/*
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+/*
+ * This file is available under and governed by the GNU General Public
+ * License version 2 only, as published by the Free Software Foundation.
+ * However, the following notice accompanied the original version of this
+ * file:
+ *
+ * Written by Doug Lea and Martin Buchholz with assistance from
+ * members of JCP JSR-166 Expert Group and released to the public
+ * domain, as explained at
+ * http://creativecommons.org/publicdomain/zero/1.0/
+ */
+
+import java.util.Map;
+
+/** Allows tests to work with different Map implementations. */
+public interface MapImplementation {
+ /** Returns the Map implementation class. */
+ public Class> klazz();
+ /** Returns an empty map. */
+ public Map emptyMap();
+ public Object makeKey(int i);
+ public Object makeValue(int i);
+ public boolean isConcurrent();
+ public boolean permitsNullKeys();
+ public boolean permitsNullValues();
+ public boolean supportsSetValue();
+}
diff --git a/test/jdk/java/util/concurrent/tck/MapTest.java b/test/jdk/java/util/concurrent/tck/MapTest.java
new file mode 100644
index 00000000000..5d66954e169
--- /dev/null
+++ b/test/jdk/java/util/concurrent/tck/MapTest.java
@@ -0,0 +1,172 @@
+/*
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+/*
+ * This file is available under and governed by the GNU General Public
+ * License version 2 only, as published by the Free Software Foundation.
+ * However, the following notice accompanied the original version of this
+ * file:
+ *
+ * Written by Doug Lea and Martin Buchholz with assistance from
+ * members of JCP JSR-166 Expert Group and released to the public
+ * domain, as explained at
+ * http://creativecommons.org/publicdomain/zero/1.0/
+ */
+
+import junit.framework.Test;
+
+import java.util.ArrayList;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+import java.util.concurrent.ThreadLocalRandom;
+
+/**
+ * Contains tests applicable to all Map implementations.
+ */
+public class MapTest extends JSR166TestCase {
+ final MapImplementation impl;
+
+ /** Tests are parameterized by a Map implementation. */
+ MapTest(MapImplementation impl, String methodName) {
+ super(methodName);
+ this.impl = impl;
+ }
+
+ public static Test testSuite(MapImplementation impl) {
+ return newTestSuite(
+ parameterizedTestSuite(MapTest.class,
+ MapImplementation.class,
+ impl));
+ }
+
+ public void testImplSanity() {
+ final ThreadLocalRandom rnd = ThreadLocalRandom.current();
+ {
+ Map m = impl.emptyMap();
+ assertTrue(m.isEmpty());
+ assertEquals(0, m.size());
+ Object k = impl.makeKey(rnd.nextInt());
+ Object v = impl.makeValue(rnd.nextInt());
+ m.put(k, v);
+ assertFalse(m.isEmpty());
+ assertEquals(1, m.size());
+ assertTrue(m.containsKey(k));
+ assertTrue(m.containsValue(v));
+ }
+ {
+ Map m = impl.emptyMap();
+ Object v = impl.makeValue(rnd.nextInt());
+ if (impl.permitsNullKeys()) {
+ m.put(null, v);
+ assertTrue(m.containsKey(null));
+ assertTrue(m.containsValue(v));
+ } else {
+ assertThrows(NullPointerException.class, () -> m.put(null, v));
+ }
+ }
+ {
+ Map m = impl.emptyMap();
+ Object k = impl.makeKey(rnd.nextInt());
+ if (impl.permitsNullValues()) {
+ m.put(k, null);
+ assertTrue(m.containsKey(k));
+ assertTrue(m.containsValue(null));
+ } else {
+ assertThrows(NullPointerException.class, () -> m.put(k, null));
+ }
+ }
+ {
+ Map m = impl.emptyMap();
+ Object k = impl.makeKey(rnd.nextInt());
+ Object v1 = impl.makeValue(rnd.nextInt());
+ Object v2 = impl.makeValue(rnd.nextInt());
+ m.put(k, v1);
+ if (impl.supportsSetValue()) {
+ ((Map.Entry)(m.entrySet().iterator().next())).setValue(v2);
+ assertSame(v2, m.get(k));
+ assertTrue(m.containsKey(k));
+ assertTrue(m.containsValue(v2));
+ assertFalse(m.containsValue(v1));
+ } else {
+ assertThrows(UnsupportedOperationException.class,
+ () -> ((Map.Entry)(m.entrySet().iterator().next())).setValue(v2));
+ }
+ }
+ }
+
+ /**
+ * Tests and extends the scenario reported in
+ * https://bugs.openjdk.java.net/browse/JDK-8186171
+ * HashMap: Entry.setValue may not work after Iterator.remove() called for previous entries
+ * ant -Djsr166.tckTestClass=HashMapTest -Djsr166.methodFilter=testBug8186171 -Djsr166.runsPerTest=1000 tck
+ */
+ public void testBug8186171() {
+ if (!impl.supportsSetValue()) return;
+ final ThreadLocalRandom rnd = ThreadLocalRandom.current();
+ final boolean permitsNullValues = impl.permitsNullValues();
+ final Object v1 = (permitsNullValues && rnd.nextBoolean())
+ ? null : impl.makeValue(1);
+ final Object v2 = (permitsNullValues && rnd.nextBoolean() && v1 != null)
+ ? null : impl.makeValue(2);
+
+ /** If true, always lands in first bucket in hash tables. */
+ final boolean poorHash = rnd.nextBoolean();
+ class Key implements Comparable {
+ final int i;
+ Key(int i) { this.i = i; }
+ public int hashCode() { return poorHash ? 0 : super.hashCode(); }
+ public int compareTo(Key x) {
+ return Integer.compare(this.i, x.i);
+ }
+ }
+
+ // Both HashMap and ConcurrentHashMap have:
+ // TREEIFY_THRESHOLD = 8; UNTREEIFY_THRESHOLD = 6;
+ final int size = rnd.nextInt(1, 25);
+
+ List keys = new ArrayList<>();
+ for (int i = size; i-->0; ) keys.add(new Key(i));
+ Key keyToFrob = keys.get(rnd.nextInt(keys.size()));
+
+ Map m = impl.emptyMap();
+ for (Key key : keys) m.put(key, v1);
+
+ for (Iterator> it = m.entrySet().iterator();
+ it.hasNext(); ) {
+ Map.Entry entry = it.next();
+ if (entry.getKey() == keyToFrob)
+ entry.setValue(v2); // does this have the expected effect?
+ else
+ it.remove();
+ }
+
+ assertFalse(m.containsValue(v1));
+ assertTrue(m.containsValue(v2));
+ assertTrue(m.containsKey(keyToFrob));
+ assertEquals(1, m.size());
+ }
+
+// public void testFailsIntentionallyForDebugging() {
+// fail(impl.klazz().getSimpleName());
+// }
+}
diff --git a/test/jdk/java/util/concurrent/tck/TreeMapTest.java b/test/jdk/java/util/concurrent/tck/TreeMapTest.java
index bd24e32e951..ca9c544f9fa 100644
--- a/test/jdk/java/util/concurrent/tck/TreeMapTest.java
+++ b/test/jdk/java/util/concurrent/tck/TreeMapTest.java
@@ -44,14 +44,25 @@ import java.util.Set;
import java.util.TreeMap;
import junit.framework.Test;
-import junit.framework.TestSuite;
public class TreeMapTest extends JSR166TestCase {
public static void main(String[] args) {
main(suite(), args);
}
public static Test suite() {
- return new TestSuite(TreeMapTest.class);
+ class Implementation implements MapImplementation {
+ public Class> klazz() { return TreeMap.class; }
+ public Map emptyMap() { return new TreeMap(); }
+ public Object makeKey(int i) { return i; }
+ public Object makeValue(int i) { return i; }
+ public boolean isConcurrent() { return false; }
+ public boolean permitsNullKeys() { return false; }
+ public boolean permitsNullValues() { return true; }
+ public boolean supportsSetValue() { return true; }
+ }
+ return newTestSuite(
+ TreeMapTest.class,
+ MapTest.testSuite(new Implementation()));
}
/**