2478158815
Reviewed-by: alanb, lancea
352 lines
13 KiB
Java
352 lines
13 KiB
Java
/*
|
|
* Copyright (c) 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.
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @bug 8268435 8274780
|
|
* @summary Verify ChannelInputStream methods readAllBytes and readNBytes
|
|
* @requires (sun.arch.data.model == "64" & os.maxMemory >= 16g)
|
|
* @library ..
|
|
* @library /test/lib
|
|
* @build jdk.test.lib.RandomFactory
|
|
* @modules java.base/jdk.internal.util
|
|
* @run testng/othervm/timeout=900 -Xmx12G ReadXBytes
|
|
* @key randomness
|
|
*/
|
|
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.FilterInputStream;
|
|
import java.io.InputStream;
|
|
import java.io.IOException;
|
|
import java.nio.ByteBuffer;
|
|
import java.nio.channels.Channel;
|
|
import java.nio.channels.Channels;
|
|
import java.nio.channels.FileChannel;
|
|
import java.nio.channels.ReadableByteChannel;
|
|
import java.nio.channels.SeekableByteChannel;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.util.List;
|
|
import java.util.Random;
|
|
import jdk.internal.util.ArraysSupport;
|
|
|
|
import static java.nio.file.StandardOpenOption.*;
|
|
|
|
import jdk.test.lib.RandomFactory;
|
|
|
|
import org.testng.Assert;
|
|
import static org.testng.Assert.assertEquals;
|
|
import static org.testng.Assert.assertNotNull;
|
|
import static org.testng.Assert.assertThrows;
|
|
import static org.testng.Assert.assertFalse;
|
|
import static org.testng.Assert.assertTrue;
|
|
import org.testng.annotations.DataProvider;
|
|
import org.testng.annotations.Test;
|
|
|
|
public class ReadXBytes {
|
|
|
|
private static final Random RAND = RandomFactory.getRandom();
|
|
|
|
// The largest source from which to read all bytes
|
|
private static final int BIG_LENGTH = ArraysSupport.SOFT_MAX_ARRAY_LENGTH;
|
|
|
|
// A length greater than a 32-bit integer can accommodate
|
|
private static final long HUGE_LENGTH = Integer.MAX_VALUE + 27L;
|
|
|
|
// Current directory
|
|
private static final Path DIR = Path.of(System.getProperty("test.dir", "."));
|
|
|
|
// --- Framework ---
|
|
|
|
// Create a temporary file path
|
|
static Path createFilePath() {
|
|
String name = String.format("ReadXBytes%d.tmp", System.nanoTime());
|
|
return DIR.resolve(name);
|
|
}
|
|
|
|
// Creates a temporary file of a specified length with undefined content
|
|
static Path createFile(long length) throws IOException {
|
|
Path path = createFilePath();
|
|
path.toFile().deleteOnExit();
|
|
try (FileChannel fc = FileChannel.open(path, CREATE_NEW, SPARSE, WRITE)) {
|
|
if (length > 0) {
|
|
fc.position(length - 1);
|
|
fc.write(ByteBuffer.wrap(new byte[] {27}));
|
|
}
|
|
}
|
|
return path;
|
|
}
|
|
|
|
// Creates a temporary file of a specified length with random content
|
|
static Path createFileWithRandomContent(long length) throws IOException {
|
|
Path file = createFile(length);
|
|
try (FileChannel fc = FileChannel.open(file, WRITE);) {
|
|
long pos = 0L;
|
|
// if the length exceeds 2 GB, skip the first 2 GB - 1 MB bytes
|
|
if (length >= 2L*1024*1024*1024) {
|
|
// write the last (length - 2GB - 1MB) bytes
|
|
pos = 2047L*1024*1024;
|
|
} else if (length > 0) {
|
|
// write either the first or last bytes only
|
|
long p = Math.min(Math.abs(RAND.nextLong()), length - 1);
|
|
pos = RAND.nextBoolean() ? p : length - 1 - p;
|
|
}
|
|
fc.position(pos);
|
|
int bufLength = Math.min(32768, (int)Math.min(length - pos, BIG_LENGTH));
|
|
byte[] buf = new byte[bufLength];
|
|
while (pos < length) {
|
|
RAND.nextBytes(buf);
|
|
int len = (int)Math.min(bufLength, length - pos);
|
|
pos += fc.write(ByteBuffer.wrap(buf, 0, len));
|
|
}
|
|
}
|
|
return file;
|
|
}
|
|
|
|
// Creates a file of a specified length
|
|
@FunctionalInterface
|
|
interface FileCreator {
|
|
Path create(long length) throws IOException;
|
|
}
|
|
|
|
// Performs a test for checking edge cases
|
|
@FunctionalInterface
|
|
interface EdgeTest {
|
|
void test(long length, InputStream source) throws IOException;
|
|
}
|
|
|
|
// Performs a test for evaluating correctness of content
|
|
@FunctionalInterface
|
|
interface DataTest {
|
|
void test(long length, InputStream source, InputStream reference)
|
|
throws IOException;
|
|
}
|
|
|
|
// Construct for testing zero length, EOF, and IAE
|
|
public void edgeTest(long length, FileCreator c, EdgeTest f)
|
|
throws IOException {
|
|
Path file = c.create(length);
|
|
try (FileChannel fc = FileChannel.open(file, READ);
|
|
InputStream cis = Channels.newInputStream(fc)) {
|
|
f.test(length, cis);
|
|
} finally {
|
|
Files.delete(file);
|
|
}
|
|
}
|
|
|
|
// Construct for testing correctness of content
|
|
public void dataTest(long length, FileCreator c, DataTest f)
|
|
throws IOException {
|
|
Path file = c.create(length);
|
|
try {
|
|
for (boolean seekable : List.of(false, true)) {
|
|
try (FileInputStream fis = new FileInputStream(file.toFile())) {
|
|
ReadableByteChannel ch;
|
|
if (seekable) {
|
|
ch = FileChannel.open(file, READ);
|
|
} else {
|
|
InputStream fis2 = new FileInputStream(file.toFile());
|
|
ch = Channels.newChannel(new FilterInputStream(fis2) {});
|
|
assertFalse(ch instanceof SeekableByteChannel);
|
|
}
|
|
try (InputStream cis = Channels.newInputStream(ch)) {
|
|
f.test(length, cis, fis);
|
|
}
|
|
}
|
|
}
|
|
} finally {
|
|
Files.delete(file);
|
|
}
|
|
}
|
|
|
|
// --- readAllBytes tests ---
|
|
|
|
// Verifies readAllBytes() behavior for an empty file
|
|
@Test
|
|
public void readAllBytesFromEmptyFile() throws IOException {
|
|
edgeTest(0L, (length) -> createFile(length),
|
|
(length, cis) -> {
|
|
byte[] bytes = cis.readAllBytes();
|
|
assertNotNull(bytes);
|
|
assertEquals(bytes.length, 0L);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readAllBytes() behavior at EOF
|
|
@Test
|
|
public void readAllBytesAtEOF() throws IOException {
|
|
edgeTest(RAND.nextInt(Short.MAX_VALUE), (length) -> createFile(length),
|
|
(length, cis) -> {
|
|
cis.skipNBytes(length);
|
|
byte[] bytes = cis.readAllBytes();
|
|
assertNotNull(bytes);
|
|
assertEquals(bytes.length, 0);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readAllBytes() behavior for a maximal length source
|
|
@Test
|
|
public void readAllBytesFromMaxLengthFile() throws IOException {
|
|
dataTest(BIG_LENGTH, (length) -> createFileWithRandomContent(length),
|
|
(length, cis, fis) -> {
|
|
byte[] cisBytes = cis.readAllBytes();
|
|
assertNotNull(cisBytes);
|
|
assertEquals(cisBytes.length, (long)length);
|
|
byte[] fisBytes = fis.readAllBytes();
|
|
assertEquals(cisBytes, fisBytes);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readAllBytes() throws OOME if the source is too large
|
|
@Test
|
|
public void readAllBytesFromBeyondMaxLengthFile() throws IOException {
|
|
dataTest(HUGE_LENGTH, (length) -> createFile(length),
|
|
(length, cis, fis) -> {
|
|
assertThrows(OutOfMemoryError.class,
|
|
() -> {cis.readAllBytes();});
|
|
}
|
|
);
|
|
}
|
|
|
|
// Provides an array of lengths
|
|
@DataProvider
|
|
public Object[][] lengthProvider() throws IOException {
|
|
return new Object[][] {
|
|
{1 + RAND.nextInt(1)},
|
|
{1 + RAND.nextInt(Byte.MAX_VALUE)},
|
|
{1 + RAND.nextInt(Short.MAX_VALUE)},
|
|
{1 + RAND.nextInt(1_000_000)},
|
|
{1 + RAND.nextInt(BIG_LENGTH)}
|
|
};
|
|
}
|
|
|
|
// Verifies readAllBytes() accuracy for random lengths and initial positions
|
|
@Test(dataProvider = "lengthProvider")
|
|
public void readAllBytes(int len) throws IOException {
|
|
dataTest(len, (length) -> createFileWithRandomContent(length),
|
|
(length, cis, fis) -> {
|
|
long position = RAND.nextInt(Math.toIntExact(length));
|
|
cis.skipNBytes(position);
|
|
byte[] cisBytes = cis.readAllBytes();
|
|
assertNotNull(cisBytes);
|
|
assertEquals(cisBytes.length, length - position);
|
|
fis.skipNBytes(position);
|
|
byte[] fisBytes = fis.readAllBytes();
|
|
assertEquals(cisBytes, fisBytes);
|
|
}
|
|
);
|
|
}
|
|
|
|
// --- readNBytes tests ---
|
|
|
|
// Verifies readNBytes() behavior for a negative length
|
|
@Test
|
|
public void readNBytesWithNegativeLength() throws IOException {
|
|
edgeTest(0L, (length) -> createFile(length),
|
|
(length, cis) -> {
|
|
assertThrows(IllegalArgumentException.class,
|
|
() -> {cis.readNBytes(-1);});
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readNBytes() for an empty file
|
|
@Test
|
|
public void readNBytesFromEmptyFile() throws IOException {
|
|
edgeTest(0L, (length) -> createFile(length),
|
|
(length, cis) -> {
|
|
byte[] bytes = cis.readNBytes(1);
|
|
assertNotNull(bytes);
|
|
assertEquals(bytes.length, 0);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readNBytes() behavior at EOF
|
|
@Test
|
|
public void readNBytesAtEOF() throws IOException {
|
|
edgeTest(RAND.nextInt(Short.MAX_VALUE), (length) -> createFile(length),
|
|
(length, cis) -> {
|
|
cis.skipNBytes(length);
|
|
byte[] bytes = cis.readNBytes(1);
|
|
assertNotNull(bytes);
|
|
assertEquals(bytes.length, 0);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readNBytes() behavior for a maximal length source
|
|
@Test
|
|
public void readNBytesFromMaxLengthFile() throws IOException {
|
|
dataTest(BIG_LENGTH, (length) -> createFileWithRandomContent(length),
|
|
(length, cis, fis) -> {
|
|
byte[] cisBytes = cis.readNBytes(BIG_LENGTH);
|
|
assertNotNull(cisBytes);
|
|
assertEquals(cisBytes.length, (long)length);
|
|
byte[] fisBytes = fis.readNBytes(BIG_LENGTH);
|
|
assertEquals(cisBytes, fisBytes);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readNBytes() beyond the maximum length source
|
|
@Test
|
|
public void readNBytesFromBeyondMaxLengthFile() throws IOException {
|
|
dataTest(HUGE_LENGTH, (length) -> createFileWithRandomContent(length),
|
|
(length, cis, fis) -> {
|
|
assertTrue(BIG_LENGTH < length, length + " >= " + HUGE_LENGTH);
|
|
int n = Math.toIntExact(length - BIG_LENGTH);
|
|
cis.skipNBytes(BIG_LENGTH);
|
|
byte[] cisBytes = cis.readNBytes(n);
|
|
assertNotNull(cisBytes);
|
|
assertEquals(cisBytes.length, n);
|
|
fis.skipNBytes(BIG_LENGTH);
|
|
byte[] fisBytes = fis.readNBytes(n);
|
|
assertEquals(cisBytes, fisBytes);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Verifies readNBytes() accuracy for random lengths and initial positions
|
|
@Test(dataProvider = "lengthProvider")
|
|
public void readNBytes(int len) throws IOException {
|
|
dataTest(len, (length) -> createFileWithRandomContent(length),
|
|
(length, cis, fis) -> {
|
|
int ilen = Math.toIntExact(len);
|
|
int position = RAND.nextInt(ilen);
|
|
int n = RAND.nextInt(ilen - position);
|
|
cis.skipNBytes(position);
|
|
byte[] cisBytes = cis.readNBytes(n);
|
|
assertNotNull(cisBytes);
|
|
assertEquals(cisBytes.length, n);
|
|
fis.skipNBytes(position);
|
|
byte[] fisBytes = fis.readNBytes(n);
|
|
assertEquals(cisBytes, fisBytes);
|
|
}
|
|
);
|
|
}
|
|
}
|