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));
}
}