From 6bb71d9e25f04dbde13d010e2e6279c8059c2cdb Mon Sep 17 00:00:00 2001 From: Brian Burkhalter Date: Thu, 29 Apr 2021 15:34:04 +0000 Subject: [PATCH] 8264762: ByteBuffer.byteOrder(BIG_ENDIAN).asXBuffer.put(Xarray) and ByteBuffer.byteOrder(nativeOrder()).asXBuffer.put(Xarray) are slow Reviewed-by: alanb, psandoz, chegar --- .../java/nio/Direct-X-Buffer.java.template | 169 ------------------ .../classes/java/nio/X-Buffer.java.template | 144 +++++++++++---- 2 files changed, 109 insertions(+), 204 deletions(-) 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 5d7e4d9e374..4c540d324a0 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 @@ -344,82 +344,6 @@ class Direct$Type$Buffer$RW$$BO$ } } #end[streamableType] - - public $Type$Buffer get($type$[] dst, int offset, int length) { -#if[rw] - if (((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_TO_ARRAY_THRESHOLD) { - Objects.checkFromIndexSize(offset, length, dst.length); - int pos = position(); - int lim = limit(); - assert (pos <= lim); - int rem = (pos <= lim ? lim - pos : 0); - if (length > rem) - throw new BufferUnderflowException(); - - long dstOffset = ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); - try { -#if[!byte] - if (order() != ByteOrder.nativeOrder()) - SCOPED_MEMORY_ACCESS.copySwapMemory(scope(), null, null, - ix(pos), - dst, - dstOffset, - (long)length << $LG_BYTES_PER_VALUE$, - (long)1 << $LG_BYTES_PER_VALUE$); - else -#end[!byte] - SCOPED_MEMORY_ACCESS.copyMemory(scope(), null, null, - ix(pos), - dst, - dstOffset, - (long)length << $LG_BYTES_PER_VALUE$); - } finally { - Reference.reachabilityFence(this); - } - position(pos + length); - } else { - super.get(dst, offset, length); - } - return this; -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } - - public $Type$Buffer get(int index, $type$[] dst, int offset, int length) { -#if[rw] - if (((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_TO_ARRAY_THRESHOLD) { - Objects.checkFromIndexSize(index, length, limit()); - Objects.checkFromIndexSize(offset, length, dst.length); - - long dstOffset = ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); - try { -#if[!byte] - if (order() != ByteOrder.nativeOrder()) - SCOPED_MEMORY_ACCESS.copySwapMemory(scope(), null, null, - ix(index), - dst, - dstOffset, - (long)length << $LG_BYTES_PER_VALUE$, - (long)1 << $LG_BYTES_PER_VALUE$); - else -#end[!byte] - SCOPED_MEMORY_ACCESS.copyMemory(scope(), null, null, - ix(index), - dst, - dstOffset, - (long)length << $LG_BYTES_PER_VALUE$); - } finally { - Reference.reachabilityFence(this); - } - } else { - super.get(index, dst, offset, length); - } - return this; -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } #end[rw] public $Type$Buffer put($type$ x) { @@ -448,99 +372,6 @@ class Direct$Type$Buffer$RW$$BO$ #end[rw] } - public $Type$Buffer put($Type$Buffer src) { -#if[rw] - super.put(src); - return this; -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } - - public $Type$Buffer put(int index, $Type$Buffer src, int offset, int length) { -#if[rw] - super.put(index, src, offset, length); - return this; -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } - - public $Type$Buffer put($type$[] src, int offset, int length) { -#if[rw] - if (((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_FROM_ARRAY_THRESHOLD) { - Objects.checkFromIndexSize(offset, length, src.length); - int pos = position(); - int lim = limit(); - assert (pos <= lim); - int rem = (pos <= lim ? lim - pos : 0); - if (length > rem) - throw new BufferOverflowException(); - - long srcOffset = ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); - try { -#if[!byte] - if (order() != ByteOrder.nativeOrder()) - SCOPED_MEMORY_ACCESS.copySwapMemory(null, scope(), src, - srcOffset, - null, - ix(pos), - (long)length << $LG_BYTES_PER_VALUE$, - (long)1 << $LG_BYTES_PER_VALUE$); - else -#end[!byte] - SCOPED_MEMORY_ACCESS.copyMemory(null, scope(), src, - srcOffset, - null, - ix(pos), - (long)length << $LG_BYTES_PER_VALUE$); - } finally { - Reference.reachabilityFence(this); - } - position(pos + length); - } else { - super.put(src, offset, length); - } - return this; -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } - - public $Type$Buffer put(int index, $type$[] src, int offset, int length) { -#if[rw] - if (((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_FROM_ARRAY_THRESHOLD) { - Objects.checkFromIndexSize(index, length, limit()); - Objects.checkFromIndexSize(offset, length, src.length); - - - long srcOffset = ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); - try { -#if[!byte] - if (order() != ByteOrder.nativeOrder()) - SCOPED_MEMORY_ACCESS.copySwapMemory(null, scope(), src, - srcOffset, - null, - ix(index), - (long)length << $LG_BYTES_PER_VALUE$, - (long)1 << $LG_BYTES_PER_VALUE$); - else -#end[!byte] - SCOPED_MEMORY_ACCESS.copyMemory( - null, scope(), src, - srcOffset, null, ix(index), (long)length << $LG_BYTES_PER_VALUE$); - } finally { - Reference.reachabilityFence(this); - } - } else { - super.put(index, src, offset, length); - } - return this; -#else[rw] - throw new ReadOnlyBufferException(); -#end[rw] - } - public {#if[byte]?Mapped$Type$Buffer:$Type$Buffer} compact() { #if[rw] int pos = position(); diff --git a/src/java.base/share/classes/java/nio/X-Buffer.java.template b/src/java.base/share/classes/java/nio/X-Buffer.java.template index 5cce454aff1..6950e519a85 100644 --- a/src/java.base/share/classes/java/nio/X-Buffer.java.template +++ b/src/java.base/share/classes/java/nio/X-Buffer.java.template @@ -268,6 +268,8 @@ public abstract class $Type$Buffer extends Buffer implements Comparable<$Type$Buffer>{#if[char]?, Appendable, CharSequence, Readable} { + // Cached array base offset + private static final long ARRAY_BASE_OFFSET = UNSAFE.arrayBaseOffset($type$[].class); // These fields are declared here rather than in Heap-X-Buffer in order to // reduce the number of virtual method invocations needed to access these @@ -791,11 +793,13 @@ public abstract class $Type$Buffer */ public $Type$Buffer get($type$[] dst, int offset, int length) { Objects.checkFromIndexSize(offset, length, dst.length); - if (length > remaining()) + int pos = position(); + if (length > limit() - pos) throw new BufferUnderflowException(); - int end = offset + length; - for (int i = offset; i < end; i++) - dst[i] = get(); + + getArray(pos, dst, offset, length); + + position(pos + length); return this; } @@ -869,9 +873,9 @@ public abstract class $Type$Buffer public $Type$Buffer get(int index, $type$[] dst, int offset, int length) { Objects.checkFromIndexSize(index, length, limit()); Objects.checkFromIndexSize(offset, length, dst.length); - int end = offset + length; - for (int i = offset, j = index; i < end; i++, j++) - dst[i] = get(j); + + getArray(index, dst, offset, length); + return this; } @@ -906,6 +910,40 @@ public abstract class $Type$Buffer return get(index, dst, 0, dst.length); } + private $Type$Buffer getArray(int index, $type$[] dst, int offset, int length) { + if ( +#if[char] + isAddressable() && +#end[char] + ((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_TO_ARRAY_THRESHOLD) { + long bufAddr = address + ((long)index << $LG_BYTES_PER_VALUE$); + long dstOffset = + ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); + long len = (long)length << $LG_BYTES_PER_VALUE$; + + try { +#if[!byte] + if (order() != ByteOrder.nativeOrder()) + SCOPED_MEMORY_ACCESS.copySwapMemory( + scope(), null, base(), bufAddr, + dst, dstOffset, len, $Fulltype$.BYTES); + else +#end[!byte] + SCOPED_MEMORY_ACCESS.copyMemory( + scope(), null, base(), bufAddr, + dst, dstOffset, len); + } finally { + Reference.reachabilityFence(this); + } + } else { + int end = offset + length; + for (int i = offset, j = index; i < end; i++, j++) { + dst[i] = get(j); + } + } + return this; + } + // -- Bulk put operations -- /** @@ -1038,8 +1076,8 @@ public abstract class $Type$Buffer } void putBuffer(int pos, $Type$Buffer src, int srcPos, int n) { +#if[rw] Object srcBase = src.base(); - #if[char] if (src.isAddressable()) { #else[char] @@ -1053,29 +1091,21 @@ public abstract class $Type$Buffer long addr = address + ((long)pos << $LG_BYTES_PER_VALUE$); long len = (long)n << $LG_BYTES_PER_VALUE$; + try { #if[!byte] - if (this.order() == src.order()) { -#end[!byte] - try { - SCOPED_MEMORY_ACCESS.copyMemory( - src.scope(), scope(), srcBase, - srcAddr, base, addr, len); - } finally { - Reference.reachabilityFence(src); - Reference.reachabilityFence(this); - } -#if[!byte] - } else { - try { + if (this.order() != src.order()) SCOPED_MEMORY_ACCESS.copySwapMemory( - src.scope(), scope(), srcBase, - srcAddr, base, addr, len, (long)1 << $LG_BYTES_PER_VALUE$); - } finally { - Reference.reachabilityFence(src); - Reference.reachabilityFence(this); - } - } + src.scope(), scope(), srcBase, srcAddr, + base, addr, len, $Fulltype$.BYTES); + else #end[!byte] + SCOPED_MEMORY_ACCESS.copyMemory( + src.scope(), scope(), srcBase, srcAddr, + base, addr, len); + } finally { + Reference.reachabilityFence(src); + Reference.reachabilityFence(this); + } #if[char] } else { // src.isAddressable() == false assert StringCharBuffer.class.isInstance(src); @@ -1084,6 +1114,9 @@ public abstract class $Type$Buffer put(i, src.get(j)); } #end[char] +#else[rw] + throw new ReadOnlyBufferException(); +#end[rw] } /** @@ -1138,12 +1171,16 @@ public abstract class $Type$Buffer * If this buffer is read-only */ public $Type$Buffer put($type$[] src, int offset, int length) { + if (isReadOnly()) + throw new ReadOnlyBufferException(); Objects.checkFromIndexSize(offset, length, src.length); - if (length > remaining()) + int pos = position(); + if (length > limit() - pos) throw new BufferOverflowException(); - int end = offset + length; - for (int i = offset; i < end; i++) - this.put(src[i]); + + putArray(pos, src, offset, length); + + position(pos + length); return this; } @@ -1223,9 +1260,9 @@ public abstract class $Type$Buffer throw new ReadOnlyBufferException(); Objects.checkFromIndexSize(index, length, limit()); Objects.checkFromIndexSize(offset, length, src.length); - int end = offset + length; - for (int i = offset, j = index; i < end; i++, j++) - this.put(j, src[i]); + + putArray(index, src, offset, length); + return this; } @@ -1262,6 +1299,43 @@ public abstract class $Type$Buffer return put(index, src, 0, src.length); } + private $Type$Buffer putArray(int index, $type$[] src, int offset, int length) { +#if[rw] + if ( +#if[char] + isAddressable() && +#end[char] + ((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_FROM_ARRAY_THRESHOLD) { + long bufAddr = address + ((long)index << $LG_BYTES_PER_VALUE$); + long srcOffset = + ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$); + long len = (long)length << $LG_BYTES_PER_VALUE$; + + try { +#if[!byte] + if (order() != ByteOrder.nativeOrder()) + SCOPED_MEMORY_ACCESS.copySwapMemory( + null, scope(), src, srcOffset, + base(), bufAddr, len, $Fulltype$.BYTES); + else +#end[!byte] + SCOPED_MEMORY_ACCESS.copyMemory( + null, scope(), src, srcOffset, + base(), bufAddr, len); + } finally { + Reference.reachabilityFence(this); + } + } else { + int end = offset + length; + for (int i = offset, j = index; i < end; i++, j++) + this.put(j, src[i]); + } + return this; +#else[rw] + throw new ReadOnlyBufferException(); +#end[rw] + } + #if[char] /**