/* * Copyright (c) 2020, 2021, 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. */ package org.openjdk.bench.java.io; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; import org.openjdk.jmh.annotations.Measurement; import org.openjdk.jmh.annotations.Mode; import org.openjdk.jmh.annotations.OutputTimeUnit; import org.openjdk.jmh.annotations.Param; import org.openjdk.jmh.annotations.Scope; import org.openjdk.jmh.annotations.Setup; import org.openjdk.jmh.annotations.State; import org.openjdk.jmh.annotations.Warmup; import java.io.ByteArrayInputStream; import java.io.IOException; import java.io.InputStreamReader; import java.nio.charset.Charset; import java.util.concurrent.TimeUnit; /** * Tests the overheads of reading encoded byte arrays via StreamDecoder */ @BenchmarkMode(Mode.AverageTime) @OutputTimeUnit(TimeUnit.NANOSECONDS) @State(Scope.Thread) @Warmup(time=2, iterations=5) @Measurement(time=3, iterations=5) @Fork(value=2, jvmArgs="-Xmx1g") public class ByteStreamDecoder { @Param({"US-ASCII", "ISO-8859-1", "UTF-8", "ISO-8859-6", "MS932"}) private String charsetName; @Param({"256", "4096", "25000"}) private int length; private byte[] bytes; private byte[] nonASCIIBytesEnd; private byte[] nonASCIIBytesStart; private byte[] nonASCIIBytesEveryOther; private char[] chars; private Charset cs; @Setup public void setup() throws IOException { var s = """ Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed non magna augue. Sed tristique ante id maximus interdum. Suspendisse potenti. Aliquam molestie metus vitae magna gravida egestas. Phasellus eleifend tortor sit amet neque euismod, vitae luctus ante viverra. Sed quis justo ultrices, eleifend dui sed, egestas lorem. Mauris ipsum ex, interdum eu turpis sed, fermentum efficitur lorem. Sed vel imperdiet libero, eget ullamcorper sem. Praesent gravida arcu quis ipsum viverra tristique. Quisque maximus elit nec nisi vulputate tempor. Integer aliquet tortor vel vehicula efficitur. Sed neque felis, ultricies eu leo ultricies, egestas placerat dolor. Etiam iaculis magna quis lacinia tincidunt. Donec in tellus volutpat, semper nunc ornare, tempus erat. Donec volutpat mauris in arcu mattis sollicitudin. Morbi vestibulum ipsum sed erat porta, mollis commodo nisi gravida. """; int n = length / s.length(); String full = ""; if (n > 0) { full = s.repeat(n); } n = length % s.length(); if (n > 0) { full += s.substring(0, n); } cs = Charset.forName(charsetName); bytes = full.getBytes(cs); nonASCIIBytesEnd = (full + "\u00e5").getBytes(cs); nonASCIIBytesStart = ("\u00e5" + full).getBytes(cs); // string hostile to ASCII fast-path optimizations: every other char is ASCII StringBuilder sb = new StringBuilder(); for (int i = 0; i < full.length(); i++) { sb.append(i % 2 == 0 ? full.charAt(i) : '\u00e5'); } nonASCIIBytesEveryOther = sb.toString().getBytes(cs); chars = new char[full.length() + 2]; try { if (!readStringDirect().equals(readStringReader())) { System.out.println("direct: " + readStringDirect()); System.out.println("reader: " + readStringReader()); throw new RuntimeException("Unexpectedly different"); } if (!readStringDirect_NonASCIIEnd().equals(readStringReader_NonASCIIEnd())) { throw new RuntimeException("Unexpectedly different"); } if (!readStringDirect_NonASCIIStart().equals(readStringReader_NonASCIIStart())) { throw new RuntimeException("Unexpectedly different"); } if (!readStringDirect_NonASCIIEveryOther().equals(readStringReader_NonASCIIEveryOther())) { throw new RuntimeException("Unexpectedly different"); } } catch (Exception e) { e.printStackTrace(); } } @Benchmark public String readStringReader() throws Exception { int len = new InputStreamReader(new ByteArrayInputStream(bytes), cs).read(chars); return new String(chars, 0, len); } @Benchmark public String readStringReader_NonASCIIEnd() throws Exception { int len = new InputStreamReader(new ByteArrayInputStream(nonASCIIBytesEnd), cs).read(chars); return new String(chars, 0, len); } @Benchmark public String readStringReader_NonASCIIStart() throws Exception { int len = new InputStreamReader(new ByteArrayInputStream(nonASCIIBytesStart), cs).read(chars); return new String(chars, 0, len); } @Benchmark public String readStringReader_NonASCIIEveryOther() throws Exception { int len = new InputStreamReader(new ByteArrayInputStream(nonASCIIBytesEveryOther), cs).read(chars); return new String(chars, 0, len); } @Benchmark public String readStringDirect() throws Exception { return new String(bytes, cs); } @Benchmark public String readStringDirect_NonASCIIEnd() throws Exception { return new String(nonASCIIBytesEnd, cs); } @Benchmark public String readStringDirect_NonASCIIStart() throws Exception { return new String(nonASCIIBytesStart, cs); } @Benchmark public String readStringDirect_NonASCIIEveryOther() throws Exception { return new String(nonASCIIBytesEveryOther, cs); } }