From de3604ea0cfaf32c348450d769ea68fea0a10414 Mon Sep 17 00:00:00 2001 From: Ben Walsh Date: Mon, 12 Mar 2018 16:09:18 -0700 Subject: [PATCH] 8199462: Use Reference.reachabilityFence in direct ByteBuffer methods Reviewed-by: psandoz, vlivanov, plevart --- .../classes/java/lang/ref/Reference.java | 12 +- .../nio/Direct-X-Buffer-bin.java.template | 30 +++-- .../java/nio/Direct-X-Buffer.java.template | 112 +++++++++++------- .../classes/java/nio/MappedByteBuffer.java | 13 +- .../classes/java/nio/X-Buffer.java.template | 5 - .../java/lang/ref/ReachabilityFenceTest.java | 8 +- 6 files changed, 113 insertions(+), 67 deletions(-) diff --git a/src/java.base/share/classes/java/lang/ref/Reference.java b/src/java.base/share/classes/java/lang/ref/Reference.java index 042eb474b45..4fdeb452c78 100644 --- a/src/java.base/share/classes/java/lang/ref/Reference.java +++ b/src/java.base/share/classes/java/lang/ref/Reference.java @@ -25,7 +25,7 @@ package java.lang.ref; -import jdk.internal.vm.annotation.DontInline; +import jdk.internal.vm.annotation.ForceInline; import jdk.internal.HotSpotIntrinsicCandidate; import jdk.internal.misc.JavaLangRefAccess; import jdk.internal.misc.SharedSecrets; @@ -420,10 +420,12 @@ public abstract class Reference { * @param ref the reference. If {@code null}, this method has no effect. * @since 9 */ - @DontInline + @ForceInline public static void reachabilityFence(Object ref) { - // Does nothing, because this method is annotated with @DontInline - // HotSpot needs to retain the ref and not GC it before a call to this - // method + // Does nothing. This method is annotated with @ForceInline to eliminate + // most of the overhead that using @DontInline would cause with the + // HotSpot JVM, when this fence is used in a wide variety of situations. + // HotSpot JVM retains the ref and does not GC it before a call to + // this method, because the JIT-compilers do not have GC-only safepoints. } } diff --git a/src/java.base/share/classes/java/nio/Direct-X-Buffer-bin.java.template b/src/java.base/share/classes/java/nio/Direct-X-Buffer-bin.java.template index cacadb101df..a6fed522fa4 100644 --- a/src/java.base/share/classes/java/nio/Direct-X-Buffer-bin.java.template +++ b/src/java.base/share/classes/java/nio/Direct-X-Buffer-bin.java.template @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2018, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -32,24 +32,40 @@ class XXX { #if[rw] private $type$ get$Type$(long a) { - $memtype$ x = UNSAFE.get$Memtype$Unaligned(null, a, bigEndian); - return $fromBits$(x); + try { + $memtype$ x = UNSAFE.get$Memtype$Unaligned(null, a, bigEndian); + return $fromBits$(x); + } finally { + Reference.reachabilityFence(this); + } } public $type$ get$Type$() { - return get$Type$(ix(nextGetIndex($BYTES_PER_VALUE$))); + try { + return get$Type$(ix(nextGetIndex($BYTES_PER_VALUE$))); + } finally { + Reference.reachabilityFence(this); + } } public $type$ get$Type$(int i) { - return get$Type$(ix(checkIndex(i, $BYTES_PER_VALUE$))); + try { + return get$Type$(ix(checkIndex(i, $BYTES_PER_VALUE$))); + } finally { + Reference.reachabilityFence(this); + } } #end[rw] private ByteBuffer put$Type$(long a, $type$ x) { #if[rw] - $memtype$ y = $toBits$(x); - UNSAFE.put$Memtype$Unaligned(null, a, y, bigEndian); + try { + $memtype$ y = $toBits$(x); + UNSAFE.put$Memtype$Unaligned(null, a, y, bigEndian); + } finally { + Reference.reachabilityFence(this); + } return this; #else[rw] throw new ReadOnlyBufferException(); diff --git a/src/java.base/share/classes/java/nio/Direct-X-Buffer.java.template b/src/java.base/share/classes/java/nio/Direct-X-Buffer.java.template index 5fa9be47b20..fc4582995e5 100644 --- a/src/java.base/share/classes/java/nio/Direct-X-Buffer.java.template +++ b/src/java.base/share/classes/java/nio/Direct-X-Buffer.java.template @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2018, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -28,6 +28,7 @@ package java.nio; import java.io.FileDescriptor; +import java.lang.ref.Reference; import jdk.internal.misc.VM; import jdk.internal.ref.Cleaner; import sun.nio.ch.DirectBuffer; @@ -257,16 +258,28 @@ class Direct$Type$Buffer$RW$$BO$ } public $type$ get() { - return $fromBits$($swap$(UNSAFE.get$Swaptype$(ix(nextGetIndex())))); + try { + return $fromBits$($swap$(UNSAFE.get$Swaptype$(ix(nextGetIndex())))); + } finally { + Reference.reachabilityFence(this); + } } public $type$ get(int i) { - return $fromBits$($swap$(UNSAFE.get$Swaptype$(ix(checkIndex(i))))); + try { + return $fromBits$($swap$(UNSAFE.get$Swaptype$(ix(checkIndex(i))))); + } finally { + Reference.reachabilityFence(this); + } } #if[streamableType] $type$ getUnchecked(int i) { - return $fromBits$($swap$(UNSAFE.get$Swaptype$(ix(i)))); + try { + return $fromBits$($swap$(UNSAFE.get$Swaptype$(ix(i)))); + } finally { + Reference.reachabilityFence(this); + } } #end[streamableType] @@ -282,21 +295,25 @@ class Direct$Type$Buffer$RW$$BO$ throw new BufferUnderflowException(); long dstOffset = ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); + try { #if[!byte] - if (order() != ByteOrder.nativeOrder()) - UNSAFE.copySwapMemory(null, + if (order() != ByteOrder.nativeOrder()) + UNSAFE.copySwapMemory(null, + ix(pos), + dst, + dstOffset, + (long)length << $LG_BYTES_PER_VALUE$, + (long)1 << $LG_BYTES_PER_VALUE$); + else +#end[!byte] + UNSAFE.copyMemory(null, ix(pos), dst, dstOffset, - (long)length << $LG_BYTES_PER_VALUE$, - (long)1 << $LG_BYTES_PER_VALUE$); - else -#end[!byte] - UNSAFE.copyMemory(null, - ix(pos), - dst, - dstOffset, - (long)length << $LG_BYTES_PER_VALUE$); + (long)length << $LG_BYTES_PER_VALUE$); + } finally { + Reference.reachabilityFence(this); + } position(pos + length); } else { super.get(dst, offset, length); @@ -311,7 +328,11 @@ class Direct$Type$Buffer$RW$$BO$ public $Type$Buffer put($type$ x) { #if[rw] - UNSAFE.put$Swaptype$(ix(nextPutIndex()), $swap$($toBits$(x))); + try { + UNSAFE.put$Swaptype$(ix(nextPutIndex()), $swap$($toBits$(x))); + } finally { + Reference.reachabilityFence(this); + } return this; #else[rw] throw new ReadOnlyBufferException(); @@ -320,7 +341,11 @@ class Direct$Type$Buffer$RW$$BO$ public $Type$Buffer put(int i, $type$ x) { #if[rw] - UNSAFE.put$Swaptype$(ix(checkIndex(i)), $swap$($toBits$(x))); + try { + UNSAFE.put$Swaptype$(ix(checkIndex(i)), $swap$($toBits$(x))); + } finally { + Reference.reachabilityFence(this); + } return this; #else[rw] throw new ReadOnlyBufferException(); @@ -346,7 +371,12 @@ class Direct$Type$Buffer$RW$$BO$ if (srem > rem) throw new BufferOverflowException(); - UNSAFE.copyMemory(sb.ix(spos), ix(pos), (long)srem << $LG_BYTES_PER_VALUE$); + try { + UNSAFE.copyMemory(sb.ix(spos), ix(pos), (long)srem << $LG_BYTES_PER_VALUE$); + } finally { + Reference.reachabilityFence(sb); + Reference.reachabilityFence(this); + } sb.position(spos + srem); position(pos + srem); } else if (src.hb != null) { @@ -380,21 +410,25 @@ class Direct$Type$Buffer$RW$$BO$ throw new BufferOverflowException(); long srcOffset = ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); + try { #if[!byte] - if (order() != ByteOrder.nativeOrder()) - UNSAFE.copySwapMemory(src, + if (order() != ByteOrder.nativeOrder()) + UNSAFE.copySwapMemory(src, + srcOffset, + null, + ix(pos), + (long)length << $LG_BYTES_PER_VALUE$, + (long)1 << $LG_BYTES_PER_VALUE$); + else +#end[!byte] + UNSAFE.copyMemory(src, srcOffset, null, ix(pos), - (long)length << $LG_BYTES_PER_VALUE$, - (long)1 << $LG_BYTES_PER_VALUE$); - else -#end[!byte] - UNSAFE.copyMemory(src, - srcOffset, - null, - ix(pos), - (long)length << $LG_BYTES_PER_VALUE$); + (long)length << $LG_BYTES_PER_VALUE$); + } finally { + Reference.reachabilityFence(this); + } position(pos + length); } else { super.put(src, offset, length); @@ -411,8 +445,11 @@ class Direct$Type$Buffer$RW$$BO$ int lim = limit(); assert (pos <= lim); int rem = (pos <= lim ? lim - pos : 0); - - UNSAFE.copyMemory(ix(pos), ix(0), (long)rem << $LG_BYTES_PER_VALUE$); + try { + UNSAFE.copyMemory(ix(pos), ix(0), (long)rem << $LG_BYTES_PER_VALUE$); + } finally { + Reference.reachabilityFence(this); + } position(rem); limit(capacity()); discardMark(); @@ -497,19 +534,6 @@ class Direct$Type$Buffer$RW$$BO$ #if[byte] - - byte _get(int i) { // package-private - return UNSAFE.getByte(address + i); - } - - void _put(int i, byte b) { // package-private -#if[rw] - UNSAFE.putByte(address + i, b); -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } - // #BIN // // Binary-data access methods for short, char, int, long, float, diff --git a/src/java.base/share/classes/java/nio/MappedByteBuffer.java b/src/java.base/share/classes/java/nio/MappedByteBuffer.java index 2f84cbf4549..d9ead79c092 100644 --- a/src/java.base/share/classes/java/nio/MappedByteBuffer.java +++ b/src/java.base/share/classes/java/nio/MappedByteBuffer.java @@ -26,6 +26,7 @@ package java.nio; import java.io.FileDescriptor; +import java.lang.ref.Reference; import jdk.internal.misc.Unsafe; @@ -166,9 +167,15 @@ public abstract class MappedByteBuffer int count = Bits.pageCount(length); long a = mappingAddress(offset); byte x = 0; - for (int i=0; i