/* * Copyright (c) 2020, 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 * 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. */ import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.nio.Buffer; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.nio.CharBuffer; import java.nio.DoubleBuffer; import java.nio.FloatBuffer; import java.nio.IntBuffer; import java.nio.LongBuffer; import java.nio.ReadOnlyBufferException; import java.nio.ShortBuffer; import java.util.ArrayList; import java.util.Arrays; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.Random; import org.testng.Assert; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; /* * @test * @bug 8245121 * @summary Ensure that a bulk put of a buffer into another is correct. * @compile --enable-preview -source ${jdk.version} BulkPutBuffer.java * @run testng/othervm --enable-preview BulkPutBuffer */ public class BulkPutBuffer { static final long SEED = System.nanoTime(); static final MyRandom RND = new MyRandom(SEED); static final int ITERATIONS = 100; static final int MAX_CAPACITY = 1024; static class MyRandom extends Random { MyRandom(long seed) { super(seed); } public byte nextByte() { return (byte)next(8); } public char nextChar() { return (char)next(16); } public short nextShort() { return (short)next(16); } } enum BufferKind { HEAP, HEAP_VIEW, DIRECT, STRING; } static final Map,TypeAttr> typeToAttr; static record TypeAttr(Class type, int bytes, String name) {} static { typeToAttr = Map.of( byte.class, new TypeAttr(ByteBuffer.class, Byte.BYTES, "Byte"), char.class, new TypeAttr(CharBuffer.class, Character.BYTES, "Char"), short.class, new TypeAttr(ShortBuffer.class, Short.BYTES, "Short"), int.class, new TypeAttr(IntBuffer.class, Integer.BYTES, "Int"), float.class, new TypeAttr(FloatBuffer.class, Float.BYTES, "Float"), long.class, new TypeAttr(LongBuffer.class, Long.BYTES, "Long"), double.class, new TypeAttr(DoubleBuffer.class, Double.BYTES, "Double") ); } static BufferKind[] getKinds(Class elementType) { BufferKind[] kinds; if (elementType == byte.class) kinds = new BufferKind[] { BufferKind.DIRECT, BufferKind.HEAP }; else if (elementType == char.class) kinds = BufferKind.values(); else kinds = new BufferKind[] { BufferKind.DIRECT, BufferKind.HEAP, BufferKind.HEAP_VIEW }; return kinds; } static ByteOrder[] getOrders(BufferKind kind, Class elementType) { switch (kind) { case HEAP: return new ByteOrder[] { ByteOrder.nativeOrder() }; default: if (elementType == byte.class) return new ByteOrder[] { ByteOrder.nativeOrder() }; else return new ByteOrder[] { ByteOrder.BIG_ENDIAN, ByteOrder.LITTLE_ENDIAN }; } } public static class BufferProxy { final Class elementType; final BufferKind kind; final ByteOrder order; // Buffer methods MethodHandle alloc; MethodHandle allocBB; MethodHandle allocDirect; MethodHandle asTypeBuffer; MethodHandle putAbs; MethodHandle getAbs; MethodHandle putBufRel; MethodHandle equals; // MyRandom method MethodHandle nextType; BufferProxy(Class elementType, BufferKind kind, ByteOrder order) { this.elementType = elementType; this.kind = kind; this.order = order; Class bufferType = typeToAttr.get(elementType).type; var lookup = MethodHandles.lookup(); try { String name = typeToAttr.get(elementType).name; alloc = lookup.findStatic(bufferType, "allocate", MethodType.methodType(bufferType, int.class)); allocBB = lookup.findStatic(ByteBuffer.class, "allocate", MethodType.methodType(ByteBuffer.class, int.class)); allocDirect = lookup.findStatic(ByteBuffer.class, "allocateDirect", MethodType.methodType(ByteBuffer.class, int.class)); if (elementType != byte.class) { asTypeBuffer = lookup.findVirtual(ByteBuffer.class, "as" + name + "Buffer", MethodType.methodType(bufferType)); } putAbs = lookup.findVirtual(bufferType, "put", MethodType.methodType(bufferType, int.class, elementType)); getAbs = lookup.findVirtual(bufferType, "get", MethodType.methodType(elementType, int.class)); putBufRel = lookup.findVirtual(bufferType, "put", MethodType.methodType(bufferType, bufferType)); equals = lookup.findVirtual(bufferType, "equals", MethodType.methodType(boolean.class, Object.class)); nextType = lookup.findVirtual(MyRandom.class, "next" + name, MethodType.methodType(elementType)); } catch (IllegalAccessException | NoSuchMethodException e) { throw new AssertionError(e); } } Buffer create(int capacity) throws Throwable { Class bufferType = typeToAttr.get(elementType).type; try { if (bufferType == ByteBuffer.class || kind == BufferKind.DIRECT || kind == BufferKind.HEAP_VIEW) { int len = capacity*typeToAttr.get(elementType).bytes; ByteBuffer bb = (ByteBuffer)allocBB.invoke(len); byte[] bytes = new byte[len]; RND.nextBytes(bytes); bb.put(0, bytes); if (bufferType == ByteBuffer.class) { return (Buffer)bb; } else { bb.order(order); return (Buffer)asTypeBuffer.invoke(bb); } } else if (bufferType == CharBuffer.class && kind == BufferKind.STRING) { char[] array = new char[capacity]; for (int i = 0; i < capacity; i++) { array[i] = RND.nextChar(); } return CharBuffer.wrap(new String(array)); } else { Buffer buf = (Buffer)alloc.invoke(capacity); for (int i = 0; i < capacity; i++) { putAbs.invoke(buf, i, nextType.invoke(RND)); } return buf; } } catch (Exception e) { throw new AssertionError(e); } } void copy(Buffer src, int srcOff, Buffer dst, int dstOff, int length) throws Throwable { try { for (int i = 0; i < length; i++) { putAbs.invoke(dst, dstOff + i, getAbs.invoke(src, srcOff + i)); } } catch (ReadOnlyBufferException ro) { throw ro; } catch (Exception e) { throw new AssertionError(e); } } void put(Buffer src, Buffer dst) throws Throwable { try { putBufRel.invoke(dst, src); } catch (ReadOnlyBufferException ro) { throw ro; } catch (Exception e) { throw new AssertionError(e); } } boolean equals(Buffer src, Buffer dst) throws Throwable { try { return Boolean.class.cast(equals.invoke(dst, src)); } catch (Exception e) { throw new AssertionError(e); } } } static List getProxies(Class type) { List proxies = new ArrayList(); for (BufferKind kind : getKinds(type)) { for (ByteOrder order : getOrders(kind, type)) { proxies.add(new BufferProxy(type, kind, order)); } } return proxies; } @DataProvider static Object[][] proxies() { ArrayList args = new ArrayList<>(); for (Class type : typeToAttr.keySet()) { List proxies = getProxies(type); for (BufferProxy proxy : proxies) { args.add(new Object[] {proxy}); } } return args.toArray(Object[][]::new); } @DataProvider static Object[][] proxyPairs() { List args = new ArrayList<>(); for (Class type : typeToAttr.keySet()) { List proxies = getProxies(type); for (BufferProxy proxy1 : proxies) { for (BufferProxy proxy2 : proxies) { args.add(new Object[] {proxy1, proxy2}); } } } return args.toArray(Object[][]::new); } @Test(dataProvider = "proxies") public static void testSelf(BufferProxy bp) throws Throwable { for (int i = 0; i < ITERATIONS; i++) { int cap = RND.nextInt(MAX_CAPACITY); Buffer buf = bp.create(cap); int lowerOffset = RND.nextInt(1 + cap/10); int lowerLength = RND.nextInt(1 + cap/2); if (lowerLength < 2) continue; Buffer lower = buf.slice(lowerOffset, lowerLength); Buffer lowerCopy = bp.create(lowerLength); if (lowerCopy.isReadOnly()) { Assert.expectThrows(ReadOnlyBufferException.class, () -> bp.copy(lower, 0, lowerCopy, 0, lowerLength)); break; } else { bp.copy(lower, 0, lowerCopy, 0, lowerLength); } int middleOffset = RND.nextInt(1 + cap/2); Buffer middle = buf.slice(middleOffset, lowerLength); if (middle.isReadOnly()) { Assert.expectThrows(ReadOnlyBufferException.class, () -> bp.put(lower, middle)); break; } else { bp.put(lower, middle); } middle.flip(); Assert.assertTrue(bp.equals(lowerCopy, middle), String.format("%d %s %d %d %d %d%n", SEED, buf.getClass().getName(), cap, lowerOffset, lowerLength, middleOffset)); } } @Test(dataProvider = "proxyPairs") public static void testPairs(BufferProxy bp, BufferProxy sbp) throws Throwable { for (int i = 0; i < ITERATIONS; i++) { int cap = Math.max(4, RND.nextInt(MAX_CAPACITY)); int cap2 = cap/2; Buffer buf = bp.create(cap); int pos = RND.nextInt(Math.max(1, cap2)); buf.position(pos); buf.mark(); int lim = pos + Math.max(1, cap - pos); buf.limit(lim); int scap = Math.max(buf.remaining(), RND.nextInt(1024)); Buffer src = sbp.create(scap); int diff = scap - buf.remaining(); int spos = diff > 0 ? RND.nextInt(diff) : 0; src.position(spos); src.mark(); int slim = spos + buf.remaining(); src.limit(slim); if (buf.isReadOnly()) { Assert.expectThrows(ReadOnlyBufferException.class, () -> bp.put(src, buf)); break; } else { bp.put(src, buf); } buf.reset(); src.reset(); Assert.assertTrue(bp.equals(src, buf), String.format("%d %s %d %d %d %s %d %d %d%n", SEED, buf.getClass().getName(), cap, pos, lim, src.getClass().getName(), scap, spos, slim)); } } }