diff --git a/test/lib-test/jdk/test/lib/hexdump/HexPrinterTest.java b/test/lib-test/jdk/test/lib/hexdump/HexPrinterTest.java index bad85a1e624..fff06dd0c99 100644 --- a/test/lib-test/jdk/test/lib/hexdump/HexPrinterTest.java +++ b/test/lib-test/jdk/test/lib/hexdump/HexPrinterTest.java @@ -77,10 +77,9 @@ public class HexPrinterTest { Object[][] builtinParams() { return new Object[][]{ {"minimal", "", "%02x", 16, "", 64, HexPrinter.Formatters.NONE, ""}, - {"canonical", "%08x ", "%02x ", 16, "|", 31, HexPrinter.Formatters.ASCII, "|\n"}, - {"simple", "%5d: ", "%02x ", 16, " // ", 64, HexPrinter.Formatters.PRINTABLE, "\n"}, - {"source", " ", "(byte)%3d, ", 8, " // ", 64, HexPrinter.Formatters.PRINTABLE, - "\n"}, + {"canonical", "%08x ", "%02x ", 16, "|", 31, HexPrinter.Formatters.PRINTABLE, "|\n"}, + {"simple", "%5d: ", "%02x ", 16, " // ", 64, HexPrinter.Formatters.ASCII, "\n"}, + {"source", " ", "(byte)%3d, ", 8, " // ", 64, HexPrinter.Formatters.PRINTABLE, "\n"}, }; } @@ -162,6 +161,52 @@ public class HexPrinterTest { } } + @Test + static void testPrintable() { + String expected = + "................................" + + " !\"#$%&'()*+,-./0123456789:;<=>?" + + "@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_" + + "`abcdefghijklmnopqrstuvwxyz{|}~." + + "................................" + + "................................" + + "................................" + + "................................"; + byte[] bytes = new byte[256]; + for (int i = 0; i < bytes.length; i++) + bytes[i] = (byte)i; + HexPrinter p = HexPrinter.minimal() + .withBytesFormat("", 256) + .formatter(HexPrinter.Formatters.PRINTABLE, "", 512); + String actual = p.toString(bytes); + Assert.assertEquals(actual, expected, "Formatters.Printable mismatch"); + } + + @Test + static void testASCII() { + String expected = "\\nul\\soh\\stx\\etx\\eot\\enq\\ack\\bel\\b\\t\\n\\vt\\f\\r\\so\\si\\dle" + + "\\dc1\\dc2\\dc3\\dc4\\nak\\syn\\etb\\can\\em\\sub\\esc\\fs\\gs\\rs\\us" + + " !\"#$%&'()*+,-./0123456789:;<=>?" + + "@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_" + + "`abcdefghijklmnopqrstuvwxyz{|}~\\127" + + "\\128\\129\\130\\131\\132\\133\\134\\135\\136\\137\\138\\139\\140\\141\\142\\143" + + "\\144\\145\\146\\147\\148\\149\\150\\151\\152\\153\\154\\155\\156\\157\\158\\159" + + "\\160\\161\\162\\163\\164\\165\\166\\167\\168\\169\\170\\171\\172\\173\\174\\175" + + "\\176\\177\\178\\179\\180\\181\\182\\183\\184\\185\\186\\187\\188\\189\\190\\191" + + "\\192\\193\\194\\195\\196\\197\\198\\199\\200\\201\\202\\203\\204\\205\\206\\207" + + "\\208\\209\\210\\211\\212\\213\\214\\215\\216\\217\\218\\219\\220\\221\\222\\223" + + "\\224\\225\\226\\227\\228\\229\\230\\231\\232\\233\\234\\235\\236\\237\\238\\239" + + "\\240\\241\\242\\243\\244\\245\\246\\247\\248\\249\\250\\251\\252\\253\\254\\255"; + byte[] bytes = new byte[256]; + for (int i = 0; i < bytes.length; i++) + bytes[i] = (byte)i; + HexPrinter p = HexPrinter.minimal() + .withBytesFormat("", 256) + .formatter(HexPrinter.Formatters.ASCII, "", 256); + String actual = p.toString(bytes); + Assert.assertEquals(actual, expected, "Formatters.ASCII mismatch"); + } + @DataProvider(name = "PrimitiveFormatters") Object[][] formatterParams() { return new Object[][]{ @@ -252,6 +297,13 @@ public class HexPrinterTest { }; } + @DataProvider(name = "badsources") + Object[][] badSources() { + return new Object[][]{ + {genBytes(21), 5, 22}, + }; + } + public static byte[] genData(int len) { // Create a byte array with data for two lines byte[] bytes = new byte[len]; @@ -312,6 +364,26 @@ public class HexPrinterTest { Assert.assertEquals(r.replace("00", "").length(), 0, "contents not all zeros"); } + @Test(dataProvider = "badsources", + expectedExceptions = java.lang.IndexOutOfBoundsException.class) + public void testBadToStringByteBuffer(byte[] bytes, int offset, int length) { + if (length < 0) + length = bytes.length - offset; + ByteBuffer bb = ByteBuffer.wrap(bytes, 0, bytes.length); + System.out.printf("Source: %s, off: %d, len: %d%n", + bytes.getClass().getName(), offset, length); + String actual; + if (offset == 0 && length < 0) { + bb.position(offset); + bb.limit(length); + actual = HexPrinter.simple().toString(bb); + } else + actual = HexPrinter.simple().toString(bb, offset, length); + System.out.println(actual); + String expected = HexPrinter.simple().toString(bytes, offset, length); + Assert.assertEquals(actual, expected, "mismatch in format()"); + } + @Test(dataProvider = "sources") public void testToStringByteBuffer(byte[] bytes, int offset, int length) { if (length < 0) diff --git a/test/lib/jdk/test/lib/hexdump/HexPrinter.java b/test/lib/jdk/test/lib/hexdump/HexPrinter.java index 91de0498e55..850f1735923 100644 --- a/test/lib/jdk/test/lib/hexdump/HexPrinter.java +++ b/test/lib/jdk/test/lib/hexdump/HexPrinter.java @@ -91,7 +91,7 @@ import java.util.Objects; * {@linkplain Formatter Formatter} functions read and interpret the bytes to show the * structure and content of a protocol or data stream. * Built-in formatters include {@link HexPrinter#formatter(Class, String) primitives}, - * {@link Formatters#PRINTABLE printable ascii}, + * {@link Formatters#PRINTABLE printable bytes}, * and {@link Formatters#utf8Parser(DataInputStream, Appendable) UTF-8 strings}. * The {@link #formatter(Formatter, String, int) formatter} method sets the * formatting function, the delimiter, and the width. @@ -238,7 +238,7 @@ public final class HexPrinter { *
  • each byte value is formatted as 2 hex digits and a space: {@code "%02x "}, *
  • maximum number of byte values per line: {@value initBytesCount}, *
  • delimiter for the annotation: {@code "|"}, - *
  • formatter: {@link Formatters#ASCII ASCII bytes}, and + *
  • formatter: {@link Formatters#PRINTABLE printable bytes}, and *
  • line separator: "|" + {@link System#lineSeparator()}, *
  • destination: {@link System#out System.out}. * @@ -254,7 +254,7 @@ public final class HexPrinter { * @return a new HexPrinter */ public static HexPrinter canonical() { - return new HexPrinter(Formatters.ASCII, "%08x ", + return new HexPrinter(Formatters.PRINTABLE, "%08x ", "%02x ", initBytesCount, "|", 31, "|" + System.lineSeparator(), System.out); @@ -271,7 +271,7 @@ public final class HexPrinter { *
  • delimiter for the annotation: {@code " // "}, *
  • width for the annotation: {@value initAnnoWidth}, *
  • line separator: {@link System#lineSeparator()}, - *
  • formatter: {@link Formatters#PRINTABLE printable ASCII} + *
  • formatter: {@link Formatters#ASCII ASCII bytes} * showing printable characters, mnemonics for control chars, and * otherwise the decimal byte values, *
  • destination default: {@link System#out System.out}. @@ -288,7 +288,7 @@ public final class HexPrinter { * @return a new HexPrinter */ public static HexPrinter simple() { - return new HexPrinter(Formatters.PRINTABLE, initOffsetFormat, + return new HexPrinter(Formatters.ASCII, initOffsetFormat, initBytesFormat, initBytesCount, initAnnoDelim, initAnnoWidth, System.lineSeparator(), System.out); @@ -305,9 +305,8 @@ public final class HexPrinter { *
  • delimiter for the annotation: {@code " // "}, *
  • width for the annotation: {@value initAnnoWidth}, *
  • line separator: {@link System#lineSeparator()}, - *
  • formatter: {@link Formatters#PRINTABLE printable ASCII} - * showing printable characters, mnemonics for control chars, and - * otherwise the decimal byte values, + *
  • formatter: {@link Formatters#PRINTABLE printable bytes} + * showing printable characters and otherwise ".", *
  • destination default: {@link System#out System.out}. * * @@ -426,34 +425,30 @@ public final class HexPrinter { } /** - * The formatter function is called repeatedly to read the bytes - * from the offset for the length and append the output. + * The formatter function is called for the range of the ByteBuffer's contents. * All annotation output is appended and flushed to the output destination. - * The ByteBuffer position and limit are unused and not modified. + * The ByteBuffer position is not used and not modified. * * @param source a ByteBuffer - * @param offset the offset in the ByteBuffer - * @param length the length in the ByteBuffer + * @param index the index in the ByteBuffer, must be non-negative and + * less than {@code limit()}. + * @param length the length in the ByteBuffer must be non-negative and + * no larger than {@code source.limit() - index} * @return this HexPrinter * @throws java.io.UncheckedIOException if an I/O error occurs + * @throws java.lang.IndexOutOfBoundsException if the preconditions on + * {@code index} and {@code length} do not hold */ - public HexPrinter format(ByteBuffer source, int offset, int length) { + public HexPrinter format(ByteBuffer source, int index, int length) { Objects.requireNonNull(source, "ByteBuffer must be non-null"); - ByteArrayInputStream bais; - if (source.hasArray() && !source.isReadOnly()) { - bais = new ByteArrayInputStream(source.array(), offset, length); - } else { - int size = source.limit() - source.position(); - byte[] bytes = new byte[size]; - source.get(bytes, offset, length); - bais = new ByteArrayInputStream(bytes); - } - return format(bais, offset); + byte[] bytes = new byte[length]; + source.get(index, bytes, 0, length); + ByteArrayInputStream bais = new ByteArrayInputStream(bytes); + return format(bais, index); } /** - * The formatter function is called repeatedly to read all of the bytes - * in the source and append the output. + * The formatter function is called for the ByteBuffer's contents. * The source bytes are from the {@code ByteBuffer.position()} * to the {@code ByteBuffer.limit()}. * The position is not modified. @@ -464,7 +459,7 @@ public final class HexPrinter { * @throws java.io.UncheckedIOException if an I/O error occurs */ public HexPrinter format(ByteBuffer source) { - return format(source, source.position(), source.limit()); + return format(source, source.position(), source.limit() - source.position()); } /** @@ -544,37 +539,36 @@ public final class HexPrinter { } /** - * The formatter function is called repeatedly to read the bytes - * from the offset for the length and return a String. - * The ByteBuffer position and limit are unused and not modified. + * The formatter function is called for the range of the ByteBuffer contents + * and returned as a string. + * The ByteBuffer position is not used and not modified. * * @param source a ByteBuffer - * @param offset the offset in the ByteBuffer - * @param length the length in the ByteBuffer + * @param index the index in the ByteBuffer, must be non-negative and + * less than {@code limit()}. + * @param length the length in the ByteBuffer must be non-negative and + * no larger than {@code source.limit() - index} * @return the output as a non-null {@code String} * @throws java.io.UncheckedIOException if an I/O error occurs + * @throws java.lang.IndexOutOfBoundsException if the preconditions on + * {@code index} and {@code length} do not hold */ - public String toString(ByteBuffer source, int offset, int length) { + public String toString(ByteBuffer source, int index, int length) { Objects.requireNonNull(source, "ByteBuffer must be non-null"); + byte[] bytes = new byte[length]; + source.get(index, bytes, 0, length); + ByteArrayInputStream bais = new ByteArrayInputStream(bytes); StringBuilder sb = new StringBuilder(); - ByteArrayInputStream bais; - if (source.hasArray() && !source.isReadOnly()) { - bais = new ByteArrayInputStream(source.array(), offset, length); - } else { - byte[] bytes = new byte[length]; - source.get(bytes, offset, length); - bais = new ByteArrayInputStream(bytes); - } try (AnnotationWriter writer = - new AnnotationWriter(this, bais, offset, sb)) { + new AnnotationWriter(this, bais, index, sb)) { writer.flush(); return sb.toString(); } } /** - * The formatter function is called repeatedly to read all of the bytes - * in the source and return a String. + * The formatter function is called for the ByteBuffer contents + * and returned as a string. * The source bytes are from the {@code ByteBuffer.position()} * to the {@code ByteBuffer.limit()}. * The position is not modified. @@ -584,7 +578,7 @@ public final class HexPrinter { * @throws java.io.UncheckedIOException if an I/O error occurs */ public String toString(ByteBuffer source) { - return toString(source, source.position(), source.limit()); + return toString(source, source.position(), source.limit() - source.position()); } /** @@ -830,18 +824,20 @@ public final class HexPrinter { } /** - * Built-in formatters for printable byte, ASCII, UTF-8 and primitive types. + * Built-in formatters for printable byte, ASCII byte, UTF-8 and primitive types. * Formatters for primitive types and different formatting options * can be found by calling {@link #ofPrimitive(Class, String)}. */ public enum Formatters implements Formatter { /** - * Read a byte and if it is ASCII write it, - * otherwise, write its mnemonic or its decimal value. + * Read a byte, return the value as a single character string + * if it is printable, otherwise return ".". */ PRINTABLE, /** - * Read a byte, if it is ASCII write it, otherwise write a ".". + * Read a byte and return it as a string. + * Return the character if it is ASCII, return its mnemonic if it + * is a control character, otherwise return its decimal value as a string. */ ASCII, /** @@ -863,10 +859,8 @@ public final class HexPrinter { } /** - * Read a byte and write it as ASCII if it is printable, - * print its mnemonic if it is a control character, - * and print its decimal value otherwise. - * A space separator character is appended for control and decimal values. + * Read a byte and return it as a single character string if it is printable, + * otherwise return ".". * * @param in a DataInputStream * @param out an Appendable to write to @@ -874,17 +868,17 @@ public final class HexPrinter { */ static void bytePrintable(DataInputStream in, Appendable out) throws IOException { int v = in.readUnsignedByte(); - if (v < 32) { - out.append("\\").append(CONTROL_MNEMONICS[v]); - } else if (v < 126 && Character.isDefined(v)) { + if (!Character.isISOControl(v) && v < 127) { out.append((char) v); } else { - out.append("\\").append(Integer.toString(v, 10)); + out.append('.'); } } /** - * Read a byte and write it as ASCII if it is printable, otherwise print ".". + * Read a byte and return it as a string. + * Append the byte if it is ASCII, its mnemonic if it + * is a control character, and otherwise its decimal value. * * @param in a DataInputStream * @param out an Appendable to write to @@ -892,10 +886,12 @@ public final class HexPrinter { */ static void byteASCII(DataInputStream in, Appendable out) throws IOException { int v = in.readUnsignedByte(); - if (Character.isDefined(v)) { + if (v < 32) { + out.append('\\').append(CONTROL_MNEMONICS[v]); + } else if (v < 127) { out.append((char) v); } else { - out.append('.'); + out.append('\\').append(Integer.toString(v, 10)); } }