/hg/icedtea6: 3 new changesets
andrew at icedtea.classpath.org
andrew at icedtea.classpath.org
Thu Apr 25 10:19:17 PDT 2013
changeset ffe30d3918c4 in /hg/icedtea6
details: http://icedtea.classpath.org/hg/icedtea6?cmd=changeset;node=ffe30d3918c4
author: Andrew John Hughes <gnu.andrew at redhat.com>
date: Thu Apr 25 15:01:06 2013 +0100
Add backport of 7036559 and ConcurrentHashMap deserialization reliability fix for 8009063.
changeset 0396c602f40d in /hg/icedtea6
details: http://icedtea.classpath.org/hg/icedtea6?cmd=changeset;node=0396c602f40d
author: Andrew John Hughes <gnu.andrew at redhat.com>
date: Thu Apr 25 15:04:12 2013 +0100
Fix issues with previous commit.
2013-04-12 Andrew John Hughes <gnu.andrew at member.fsf.org>
* Makefile.am:
(SECURITY_PATCHES): Correct path to 7036559; not
a security patch but a backport to enable one to
be applied.
* patches/security/20130416/7036559.patch: Moved to...
* patches/openjdk/7036559-concurrenthashmap_improvements.patch:
...here.
2013-04-11 Jon VanAlten <jon.vanalten at redhat.com>
* Makefile.am:
(SECURITY_PATCHES): Add new patches.
* patches/security/20130416/7036559.patch: Add backport.
* patches/security/20130416/8009063.patch: Add security fix.
changeset d5ea2fe9da2d in /hg/icedtea6
details: http://icedtea.classpath.org/hg/icedtea6?cmd=changeset;node=d5ea2fe9da2d
author: Andrew John Hughes <gnu.andrew at redhat.com>
date: Thu Apr 25 18:18:04 2013 +0100
Use latest hs23 HEAD, bringing in security fixes and aarch64 patch.
2013-04-25 Andrew John Hughes <gnu.andrew at member.fsf.org>
* Makefile.am:
(ICEDTEA_PATCHES): Move aarch64.patch to original
HotSpot only.
* hotspot.map: Sync with latest hs23 HEAD.
diffstat:
ChangeLog | 24 +
Makefile.am | 10 +-
hotspot.map | 2 +-
patches/openjdk/7036559-concurrenthashmap_improvements.patch | 1436 ++++++++++
patches/security/20130416/8009063.patch | 67 +
5 files changed, 1533 insertions(+), 6 deletions(-)
diffs (truncated from 1578 to 500 lines):
diff -r ceab4a096b50 -r d5ea2fe9da2d ChangeLog
--- a/ChangeLog Mon Apr 22 18:11:11 2013 +0100
+++ b/ChangeLog Thu Apr 25 18:18:04 2013 +0100
@@ -1,3 +1,27 @@
+2013-04-25 Andrew John Hughes <gnu.andrew at member.fsf.org>
+
+ * Makefile.am:
+ (ICEDTEA_PATCHES): Move aarch64.patch to original
+ HotSpot only.
+ * hotspot.map: Sync with latest hs23 HEAD.
+
+2013-04-12 Andrew John Hughes <gnu.andrew at member.fsf.org>
+
+ * Makefile.am:
+ (SECURITY_PATCHES): Correct path to 7036559; not
+ a security patch but a backport to enable one to
+ be applied.
+ * patches/security/20130416/7036559.patch: Moved to...
+ * patches/openjdk/7036559-concurrenthashmap_improvements.patch:
+ ...here.
+
+2013-04-11 Jon VanAlten <jon.vanalten at redhat.com>
+
+ * Makefile.am:
+ (SECURITY_PATCHES): Add new patches.
+ * patches/security/20130416/7036559.patch: Add backport.
+ * patches/security/20130416/8009063.patch: Add security fix.
+
2013-04-22 Andrew John Hughes <gnu.andrew at member.fsf.org>
PR1336: Bootstrap failure on Fedora 17/18
diff -r ceab4a096b50 -r d5ea2fe9da2d Makefile.am
--- a/Makefile.am Mon Apr 22 18:11:11 2013 +0100
+++ b/Makefile.am Thu Apr 25 18:18:04 2013 +0100
@@ -301,7 +301,9 @@
patches/security/20130219/8007688.patch \
patches/security/20130304/8007014.patch \
patches/security/20130304/8007675.patch \
- patches/openjdk/8009641-8007675_build_fix.patch
+ patches/openjdk/8009641-8007675_build_fix.patch \
+ patches/openjdk/7036559-concurrenthashmap_improvements.patch \
+ patches/security/20130416/8009063.patch
if !WITH_ALT_HSBUILD
SECURITY_PATCHES += \
@@ -557,12 +559,10 @@
patches/hotspot/original/7197906-handle_32_bit_shifts.patch \
patches/hotspot/original/fix_get_stack_bounds_leak.patch \
patches/hotspot/original/jvmtiEnv.patch \
- patches/hotspot/original/6840152-jvm_crashes_with_heavyweight_monitors.patch
+ patches/hotspot/original/6840152-jvm_crashes_with_heavyweight_monitors.patch \
+ patches/aarch64.patch
endif
-# Needs to be after the addition of SH support to the original HotSpot
-ICEDTEA_PATCHES += patches/aarch64.patch
-
if WITH_RHINO
ICEDTEA_PATCHES += \
patches/rhino.patch
diff -r ceab4a096b50 -r d5ea2fe9da2d hotspot.map
--- a/hotspot.map Mon Apr 22 18:11:11 2013 +0100
+++ b/hotspot.map Thu Apr 25 18:18:04 2013 +0100
@@ -1,2 +1,2 @@
# version url changeset sha256sum
-hs23 http://icedtea.classpath.org/hg/release/icedtea7-forest-2.3/hotspot 23888f3dec52 6d77e26134d47e62621a35b259c70d8e98070724af9a718ec2b85cf84b954614
+hs23 http://icedtea.classpath.org/hg/release/icedtea7-forest-2.3/hotspot 332f7e24a493 da6f849e2b8c0e8c46de4171b9f14ec9d97bac76dd56006d9c33323b23f54f98
diff -r ceab4a096b50 -r d5ea2fe9da2d patches/openjdk/7036559-concurrenthashmap_improvements.patch
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/patches/openjdk/7036559-concurrenthashmap_improvements.patch Thu Apr 25 18:18:04 2013 +0100
@@ -0,0 +1,1436 @@
+# HG changeset patch
+# User dl
+# Date 1303139440 -3600
+# Node ID 005c0c85b0decf18a90ff6c9601d1b9a2c0a3fa4
+# Parent 603e70836e74e5c18fc32279f7e4df5b4c63e0b6
+7036559: ConcurrentHashMap footprint and contention improvements
+Reviewed-by: chegar
+
+diff --git a/src/share/classes/java/util/concurrent/ConcurrentHashMap.java b/src/share/classes/java/util/concurrent/ConcurrentHashMap.java
+--- openjdk/jdk/src/share/classes/java/util/concurrent/ConcurrentHashMap.java
++++ openjdk/jdk/src/share/classes/java/util/concurrent/ConcurrentHashMap.java
+@@ -105,7 +105,25 @@
+
+ /*
+ * The basic strategy is to subdivide the table among Segments,
+- * each of which itself is a concurrently readable hash table.
++ * each of which itself is a concurrently readable hash table. To
++ * reduce footprint, all but one segments are constructed only
++ * when first needed (see ensureSegment). To maintain visibility
++ * in the presence of lazy construction, accesses to segments as
++ * well as elements of segment's table must use volatile access,
++ * which is done via Unsafe within methods segmentAt etc
++ * below. These provide the functionality of AtomicReferenceArrays
++ * but reduce the levels of indirection. Additionally,
++ * volatile-writes of table elements and entry "next" fields
++ * within locked operations use the cheaper "lazySet" forms of
++ * writes (via putOrderedObject) because these writes are always
++ * followed by lock releases that maintain sequential consistency
++ * of table updates.
++ *
++ * Historical note: The previous version of this class relied
++ * heavily on "final" fields, which avoided some volatile reads at
++ * the expense of a large initial footprint. Some remnants of
++ * that design (including forced construction of segment 0) exist
++ * to ensure serialization compatibility.
+ */
+
+ /* ---------------- Constants -------------- */
+@@ -137,8 +155,15 @@
+ static final int MAXIMUM_CAPACITY = 1 << 30;
+
+ /**
++ * The minimum capacity for per-segment tables. Must be a power
++ * of two, at least two to avoid immediate resizing on next use
++ * after lazy construction.
++ */
++ static final int MIN_SEGMENT_TABLE_CAPACITY = 2;
++
++ /**
+ * The maximum number of segments to allow; used to bound
+- * constructor arguments.
++ * constructor arguments. Must be power of two less than 1 << 24.
+ */
+ static final int MAX_SEGMENTS = 1 << 16; // slightly conservative
+
+@@ -164,7 +189,7 @@
+ final int segmentShift;
+
+ /**
+- * The segments, each of which is a specialized hash table
++ * The segments, each of which is a specialized hash table.
+ */
+ final Segment<K,V>[] segments;
+
+@@ -172,7 +197,65 @@
+ transient Set<Map.Entry<K,V>> entrySet;
+ transient Collection<V> values;
+
+- /* ---------------- Small Utilities -------------- */
++ /**
++ * ConcurrentHashMap list entry. Note that this is never exported
++ * out as a user-visible Map.Entry.
++ */
++ static final class HashEntry<K,V> {
++ final int hash;
++ final K key;
++ volatile V value;
++ volatile HashEntry<K,V> next;
++
++ HashEntry(int hash, K key, V value, HashEntry<K,V> next) {
++ this.hash = hash;
++ this.key = key;
++ this.value = value;
++ this.next = next;
++ }
++
++ /**
++ * Sets next field with volatile write semantics. (See above
++ * about use of putOrderedObject.)
++ */
++ final void setNext(HashEntry<K,V> n) {
++ UNSAFE.putOrderedObject(this, nextOffset, n);
++ }
++
++ // Unsafe mechanics
++ static final sun.misc.Unsafe UNSAFE;
++ static final long nextOffset;
++ static {
++ try {
++ UNSAFE = sun.misc.Unsafe.getUnsafe();
++ Class k = HashEntry.class;
++ nextOffset = UNSAFE.objectFieldOffset
++ (k.getDeclaredField("next"));
++ } catch (Exception e) {
++ throw new Error(e);
++ }
++ }
++ }
++
++ /**
++ * Gets the ith element of given table (if nonnull) with volatile
++ * read semantics.
++ */
++ @SuppressWarnings("unchecked")
++ static final <K,V> HashEntry<K,V> entryAt(HashEntry<K,V>[] tab, int i) {
++ return (tab == null) ? null :
++ (HashEntry<K,V>) UNSAFE.getObjectVolatile
++ (tab, ((long)i << TSHIFT) + TBASE);
++ }
++
++ /**
++ * Sets the ith element of given table, with volatile write
++ * semantics. (See above about use of putOrderedObject.)
++ */
++ static final <K,V> void setEntryAt(HashEntry<K,V>[] tab, int i,
++ HashEntry<K,V> e) {
++ UNSAFE.putOrderedObject(tab, ((long)i << TSHIFT) + TBASE, e);
++ }
+
+ /**
+ * Applies a supplemental hash function to a given hashCode, which
+@@ -193,104 +276,67 @@
+ }
+
+ /**
+- * Returns the segment that should be used for key with given hash
+- * @param hash the hash code for the key
+- * @return the segment
+- */
+- final Segment<K,V> segmentFor(int hash) {
+- return segments[(hash >>> segmentShift) & segmentMask];
+- }
+-
+- /* ---------------- Inner Classes -------------- */
+-
+- /**
+- * ConcurrentHashMap list entry. Note that this is never exported
+- * out as a user-visible Map.Entry.
+- *
+- * Because the value field is volatile, not final, it is legal wrt
+- * the Java Memory Model for an unsynchronized reader to see null
+- * instead of initial value when read via a data race. Although a
+- * reordering leading to this is not likely to ever actually
+- * occur, the Segment.readValueUnderLock method is used as a
+- * backup in case a null (pre-initialized) value is ever seen in
+- * an unsynchronized access method.
+- */
+- static final class HashEntry<K,V> {
+- final K key;
+- final int hash;
+- volatile V value;
+- final HashEntry<K,V> next;
+-
+- HashEntry(K key, int hash, HashEntry<K,V> next, V value) {
+- this.key = key;
+- this.hash = hash;
+- this.next = next;
+- this.value = value;
+- }
+-
+- @SuppressWarnings("unchecked")
+- static final <K,V> HashEntry<K,V>[] newArray(int i) {
+- return new HashEntry[i];
+- }
+- }
+-
+- /**
+ * Segments are specialized versions of hash tables. This
+ * subclasses from ReentrantLock opportunistically, just to
+ * simplify some locking and avoid separate construction.
+ */
+ static final class Segment<K,V> extends ReentrantLock implements Serializable {
+ /*
+- * Segments maintain a table of entry lists that are ALWAYS
+- * kept in a consistent state, so can be read without locking.
+- * Next fields of nodes are immutable (final). All list
+- * additions are performed at the front of each bin. This
+- * makes it easy to check changes, and also fast to traverse.
+- * When nodes would otherwise be changed, new nodes are
+- * created to replace them. This works well for hash tables
+- * since the bin lists tend to be short. (The average length
+- * is less than two for the default load factor threshold.)
++ * Segments maintain a table of entry lists that are always
++ * kept in a consistent state, so can be read (via volatile
++ * reads of segments and tables) without locking. This
++ * requires replicating nodes when necessary during table
++ * resizing, so the old lists can be traversed by readers
++ * still using old version of table.
+ *
+- * Read operations can thus proceed without locking, but rely
+- * on selected uses of volatiles to ensure that completed
+- * write operations performed by other threads are
+- * noticed. For most purposes, the "count" field, tracking the
+- * number of elements, serves as that volatile variable
+- * ensuring visibility. This is convenient because this field
+- * needs to be read in many read operations anyway:
+- *
+- * - All (unsynchronized) read operations must first read the
+- * "count" field, and should not look at table entries if
+- * it is 0.
+- *
+- * - All (synchronized) write operations should write to
+- * the "count" field after structurally changing any bin.
+- * The operations must not take any action that could even
+- * momentarily cause a concurrent read operation to see
+- * inconsistent data. This is made easier by the nature of
+- * the read operations in Map. For example, no operation
+- * can reveal that the table has grown but the threshold
+- * has not yet been updated, so there are no atomicity
+- * requirements for this with respect to reads.
+- *
+- * As a guide, all critical volatile reads and writes to the
+- * count field are marked in code comments.
++ * This class defines only mutative methods requiring locking.
++ * Except as noted, the methods of this class perform the
++ * per-segment versions of ConcurrentHashMap methods. (Other
++ * methods are integrated directly into ConcurrentHashMap
++ * methods.) These mutative methods use a form of controlled
++ * spinning on contention via methods scanAndLock and
++ * scanAndLockForPut. These intersperse tryLocks with
++ * traversals to locate nodes. The main benefit is to absorb
++ * cache misses (which are very common for hash tables) while
++ * obtaining locks so that traversal is faster once
++ * acquired. We do not actually use the found nodes since they
++ * must be re-acquired under lock anyway to ensure sequential
++ * consistency of updates (and in any case may be undetectably
++ * stale), but they will normally be much faster to re-locate.
++ * Also, scanAndLockForPut speculatively creates a fresh node
++ * to use in put if no node is found.
+ */
+
+ private static final long serialVersionUID = 2249069246763182397L;
+
+ /**
+- * The number of elements in this segment's region.
++ * The maximum number of times to tryLock in a prescan before
++ * possibly blocking on acquire in preparation for a locked
++ * segment operation. On multiprocessors, using a bounded
++ * number of retries maintains cache acquired while locating
++ * nodes.
+ */
+- transient volatile int count;
++ static final int MAX_SCAN_RETRIES =
++ Runtime.getRuntime().availableProcessors() > 1 ? 64 : 1;
+
+ /**
+- * Number of updates that alter the size of the table. This is
+- * used during bulk-read methods to make sure they see a
+- * consistent snapshot: If modCounts change during a traversal
+- * of segments computing size or checking containsValue, then
+- * we might have an inconsistent view of state so (usually)
+- * must retry.
++ * The per-segment table. Elements are accessed via
++ * entryAt/setEntryAt providing volatile semantics.
++ */
++ transient volatile HashEntry<K,V>[] table;
++
++ /**
++ * The number of elements. Accessed only either within locks
++ * or among other volatile reads that maintain visibility.
++ */
++ transient int count;
++
++ /**
++ * The total number of mutative operations in this segment.
++ * Even though this may overflows 32 bits, it provides
++ * sufficient accuracy for stability checks in CHM isEmpty()
++ * and size() methods. Accessed only either within locks or
++ * among other volatile reads that maintain visibility.
+ */
+ transient int modCount;
+
+@@ -302,11 +348,6 @@
+ transient int threshold;
+
+ /**
+- * The per-segment table.
+- */
+- transient volatile HashEntry<K,V>[] table;
+-
+- /**
+ * The load factor for the hash table. Even though this value
+ * is same for all segments, it is replicated to avoid needing
+ * links to outer object.
+@@ -314,202 +355,94 @@
+ */
+ final float loadFactor;
+
+- Segment(int initialCapacity, float lf) {
+- loadFactor = lf;
+- setTable(HashEntry.<K,V>newArray(initialCapacity));
++ Segment(float lf, int threshold, HashEntry<K,V>[] tab) {
++ this.loadFactor = lf;
++ this.threshold = threshold;
++ this.table = tab;
+ }
+
+- @SuppressWarnings("unchecked")
+- static final <K,V> Segment<K,V>[] newArray(int i) {
+- return new Segment[i];
++ final V put(K key, int hash, V value, boolean onlyIfAbsent) {
++ HashEntry<K,V> node = tryLock() ? null :
++ scanAndLockForPut(key, hash, value);
++ V oldValue;
++ try {
++ HashEntry<K,V>[] tab = table;
++ int index = (tab.length - 1) & hash;
++ HashEntry<K,V> first = entryAt(tab, index);
++ for (HashEntry<K,V> e = first;;) {
++ if (e != null) {
++ K k;
++ if ((k = e.key) == key ||
++ (e.hash == hash && key.equals(k))) {
++ oldValue = e.value;
++ if (!onlyIfAbsent) {
++ e.value = value;
++ ++modCount;
++ }
++ break;
++ }
++ e = e.next;
++ }
++ else {
++ if (node != null)
++ node.setNext(first);
++ else
++ node = new HashEntry<K,V>(hash, key, value, first);
++ int c = count + 1;
++ if (c > threshold && first != null &&
++ tab.length < MAXIMUM_CAPACITY)
++ rehash(node);
++ else
++ setEntryAt(tab, index, node);
++ ++modCount;
++ count = c;
++ oldValue = null;
++ break;
++ }
++ }
++ } finally {
++ unlock();
++ }
++ return oldValue;
+ }
+
+ /**
+- * Sets table to new HashEntry array.
+- * Call only while holding lock or in constructor.
++ * Doubles size of table and repacks entries, also adding the
++ * given node to new table
+ */
+- void setTable(HashEntry<K,V>[] newTable) {
+- threshold = (int)(newTable.length * loadFactor);
+- table = newTable;
+- }
+-
+- /**
+- * Returns properly casted first entry of bin for given hash.
+- */
+- HashEntry<K,V> getFirst(int hash) {
+- HashEntry<K,V>[] tab = table;
+- return tab[hash & (tab.length - 1)];
+- }
+-
+- /**
+- * Reads value field of an entry under lock. Called if value
+- * field ever appears to be null. This is possible only if a
+- * compiler happens to reorder a HashEntry initialization with
+- * its table assignment, which is legal under memory model
+- * but is not known to ever occur.
+- */
+- V readValueUnderLock(HashEntry<K,V> e) {
+- lock();
+- try {
+- return e.value;
+- } finally {
+- unlock();
+- }
+- }
+-
+- /* Specialized implementations of map methods */
+-
+- V get(Object key, int hash) {
+- if (count != 0) { // read-volatile
+- HashEntry<K,V> e = getFirst(hash);
+- while (e != null) {
+- if (e.hash == hash && key.equals(e.key)) {
+- V v = e.value;
+- if (v != null)
+- return v;
+- return readValueUnderLock(e); // recheck
+- }
+- e = e.next;
+- }
+- }
+- return null;
+- }
+-
+- boolean containsKey(Object key, int hash) {
+- if (count != 0) { // read-volatile
+- HashEntry<K,V> e = getFirst(hash);
+- while (e != null) {
+- if (e.hash == hash && key.equals(e.key))
+- return true;
+- e = e.next;
+- }
+- }
+- return false;
+- }
+-
+- boolean containsValue(Object value) {
+- if (count != 0) { // read-volatile
+- HashEntry<K,V>[] tab = table;
+- int len = tab.length;
+- for (int i = 0 ; i < len; i++) {
+- for (HashEntry<K,V> e = tab[i]; e != null; e = e.next) {
+- V v = e.value;
+- if (v == null) // recheck
+- v = readValueUnderLock(e);
More information about the distro-pkg-dev
mailing list