a4f2078bd6
Reviewed-by: alanb, bpb
377 lines
13 KiB
Java
377 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2000, 2022, 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 4429043 8002180
|
|
* @summary Test file mapping with FileChannel
|
|
* @run main/othervm/timeout=240 MapTest
|
|
* @key randomness
|
|
*/
|
|
|
|
import java.io.*;
|
|
import java.nio.MappedByteBuffer;
|
|
import java.nio.channels.*;
|
|
import java.nio.channels.FileChannel.MapMode;
|
|
import java.nio.file.Files;
|
|
import static java.nio.file.StandardOpenOption.*;
|
|
import static java.nio.charset.StandardCharsets.*;
|
|
import java.util.Random;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
|
|
/**
|
|
* Testing FileChannel's mapping capabilities.
|
|
*/
|
|
|
|
public class MapTest {
|
|
|
|
private static PrintStream out = System.out;
|
|
private static PrintStream err = System.err;
|
|
|
|
private static Random generator = new Random();
|
|
|
|
private static int CHARS_PER_LINE = File.separatorChar == '/' ? 5 : 6;
|
|
|
|
private static File blah;
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
blah = File.createTempFile("blah", null);
|
|
blah.deleteOnExit();
|
|
long t0 = System.nanoTime();
|
|
initTestFile(blah);
|
|
long t1 = System.nanoTime();
|
|
out.printf("Test file %s initialized in %d ns (%d ms) %n",
|
|
blah, t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
t0 = t1;
|
|
try {
|
|
testZero();
|
|
t1 = System.nanoTime();
|
|
out.printf("Zero size: done in %d ns (%d ms) %n",
|
|
t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
t0 = t1;
|
|
testRead();
|
|
t1 = System.nanoTime();
|
|
out.printf("Read: done in %d ns (%d ms) %n",
|
|
t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
t0 = t1;
|
|
testWrite();
|
|
t1 = System.nanoTime();
|
|
out.printf("Write: done in %d ns (%d ms) %n",
|
|
t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
t0 = t1;
|
|
testHighOffset();
|
|
t1 = System.nanoTime();
|
|
out.printf("High offset: done in %d ns (%d ms) %n",
|
|
t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
t0 = t1;
|
|
testForce();
|
|
t1 = System.nanoTime();
|
|
out.printf("Force: done in %d ns (%d ms) %n",
|
|
t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
t0 = t1;
|
|
testExceptions();
|
|
t1 = System.nanoTime();
|
|
out.printf("Exceptions: done in %d ns (%d ms) %n",
|
|
t1 - t0, TimeUnit.NANOSECONDS.toMillis(t1 - t0));
|
|
} finally {
|
|
blah.delete();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates file blah:
|
|
* 0000
|
|
* 0001
|
|
* 0002
|
|
* 0003
|
|
* .
|
|
* .
|
|
* .
|
|
* 3999
|
|
*
|
|
* Blah extends beyond a single page of memory so that the
|
|
* ability to index into a file of multiple pages is tested.
|
|
*/
|
|
private static void initTestFile(File blah) throws Exception {
|
|
try (BufferedWriter writer = Files.newBufferedWriter(blah.toPath(), ISO_8859_1)) {
|
|
for (int i=0; i<4000; i++) {
|
|
String number = new Integer(i).toString();
|
|
for (int h=0; h<4-number.length(); h++)
|
|
writer.write("0");
|
|
writer.write(""+i);
|
|
writer.newLine();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Tests zero size file mapping
|
|
*/
|
|
private static void testZero() throws Exception {
|
|
try (FileInputStream fis = new FileInputStream(blah)) {
|
|
FileChannel fc = fis.getChannel();
|
|
MappedByteBuffer b = fc.map(MapMode.READ_ONLY, 0, 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Maps blah file with a random offset and checks to see if read
|
|
* from the ByteBuffer gets the right line number
|
|
*/
|
|
private static void testRead() throws Exception {
|
|
StringBuilder sb = new StringBuilder();
|
|
sb.setLength(4);
|
|
|
|
for (int x=0; x<1000; x++) {
|
|
try (FileInputStream fis = new FileInputStream(blah)) {
|
|
FileChannel fc = fis.getChannel();
|
|
|
|
long offset = generator.nextInt(10000);
|
|
long expectedResult = offset / CHARS_PER_LINE;
|
|
offset = expectedResult * CHARS_PER_LINE;
|
|
|
|
MappedByteBuffer b = fc.map(MapMode.READ_ONLY,
|
|
offset, 100);
|
|
|
|
for (int i=0; i<4; i++) {
|
|
byte aByte = b.get(i);
|
|
sb.setCharAt(i, (char)aByte);
|
|
}
|
|
|
|
int result = Integer.parseInt(sb.toString());
|
|
if (result != expectedResult) {
|
|
err.println("I expected "+expectedResult);
|
|
err.println("I got "+result);
|
|
throw new Exception("Read test failed");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Maps blah file with a random offset and checks to see if data
|
|
* written out to the file can be read back in
|
|
*/
|
|
private static void testWrite() throws Exception {
|
|
StringBuilder sb = new StringBuilder();
|
|
sb.setLength(4);
|
|
|
|
for (int x=0; x<1000; x++) {
|
|
try (RandomAccessFile raf = new RandomAccessFile(blah, "rw")) {
|
|
FileChannel fc = raf.getChannel();
|
|
|
|
long offset = generator.nextInt(1000);
|
|
MappedByteBuffer b = fc.map(MapMode.READ_WRITE,
|
|
offset, 100);
|
|
|
|
for (int i=0; i<4; i++) {
|
|
b.put(i, (byte)('0' + i));
|
|
}
|
|
|
|
for (int i=0; i<4; i++) {
|
|
byte aByte = b.get(i);
|
|
sb.setCharAt(i, (char)aByte);
|
|
}
|
|
if (!sb.toString().equals("0123"))
|
|
throw new Exception("Write test failed");
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void testHighOffset() throws Exception {
|
|
StringBuilder sb = new StringBuilder();
|
|
sb.setLength(4);
|
|
|
|
for (int x=0; x<1000; x++) {
|
|
try (RandomAccessFile raf = new RandomAccessFile(blah, "rw")) {
|
|
FileChannel fc = raf.getChannel();
|
|
long offset = 66000;
|
|
MappedByteBuffer b = fc.map(MapMode.READ_WRITE,
|
|
offset, 100);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Maps blah file, writes some data and forcing writeback of
|
|
* the data exercising various valid and invalid writeback ranges.
|
|
*/
|
|
private static void testForce() throws Exception {
|
|
try (RandomAccessFile raf = new RandomAccessFile(blah, "rw")) {
|
|
FileChannel fc = raf.getChannel();
|
|
final int BLOCK_SIZE = 64;
|
|
final int BLOCK_COUNT = (4096 * 2)/ BLOCK_SIZE;
|
|
int offset = 0;
|
|
MappedByteBuffer b = fc.map(MapMode.READ_WRITE,
|
|
0, BLOCK_SIZE * (BLOCK_COUNT + 1));
|
|
|
|
for (int blocks = 0; blocks < BLOCK_COUNT; blocks++) {
|
|
for (int i = 0; i < BLOCK_SIZE; i++) {
|
|
b.put(offset + i, (byte)('0' + i));
|
|
}
|
|
b.force(offset, BLOCK_SIZE);
|
|
offset += BLOCK_SIZE;
|
|
}
|
|
|
|
Exception exc = null;
|
|
try {
|
|
// start and end are out of range
|
|
b.force(offset + BLOCK_SIZE, BLOCK_SIZE);
|
|
} catch (IndexOutOfBoundsException e) {
|
|
exc = e;
|
|
}
|
|
if (exc == null) {
|
|
throw new RuntimeException("expected Exception for force beyond buffer extent");
|
|
}
|
|
|
|
exc = null;
|
|
try {
|
|
// start is in range but end is out of range
|
|
b.force(offset, 2 * BLOCK_SIZE);
|
|
} catch (IndexOutOfBoundsException e) {
|
|
exc = e;
|
|
}
|
|
if (exc == null) {
|
|
throw new RuntimeException("expected Exception for force beyond write limit");
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Test exceptions specified by map method
|
|
*/
|
|
private static void testExceptions() throws Exception {
|
|
// check exceptions when channel opened for read access
|
|
try (FileChannel fc = FileChannel.open(blah.toPath(), READ)) {
|
|
testExceptions(fc);
|
|
|
|
checkException(fc, MapMode.READ_WRITE, 0L, fc.size(),
|
|
NonWritableChannelException.class);
|
|
|
|
checkException(fc, MapMode.READ_WRITE, -1L, fc.size(),
|
|
NonWritableChannelException.class, IllegalArgumentException.class);
|
|
|
|
checkException(fc, MapMode.READ_WRITE, 0L, -1L,
|
|
NonWritableChannelException.class, IllegalArgumentException.class);
|
|
|
|
checkException(fc, MapMode.PRIVATE, 0L, fc.size(),
|
|
NonWritableChannelException.class);
|
|
|
|
checkException(fc, MapMode.PRIVATE, -1L, fc.size(),
|
|
NonWritableChannelException.class, IllegalArgumentException.class);
|
|
|
|
checkException(fc, MapMode.PRIVATE, 0L, -1L,
|
|
NonWritableChannelException.class, IllegalArgumentException.class);
|
|
}
|
|
|
|
// check exceptions when channel opened for write access
|
|
try (FileChannel fc = FileChannel.open(blah.toPath(), WRITE)) {
|
|
testExceptions(fc);
|
|
|
|
checkException(fc, MapMode.READ_ONLY, 0L, fc.size(),
|
|
NonReadableChannelException.class);
|
|
|
|
checkException(fc, MapMode.READ_ONLY, -1L, fc.size(),
|
|
NonReadableChannelException.class, IllegalArgumentException.class);
|
|
|
|
/*
|
|
* implementation/spec mismatch, these tests disabled for now
|
|
*/
|
|
//checkException(fc, MapMode.READ_WRITE, 0L, fc.size(),
|
|
// NonWritableChannelException.class);
|
|
//checkException(fc, MapMode.PRIVATE, 0L, fc.size(),
|
|
// NonWritableChannelException.class);
|
|
}
|
|
|
|
// check exceptions when channel opened for read and write access
|
|
try (FileChannel fc = FileChannel.open(blah.toPath(), READ, WRITE)) {
|
|
testExceptions(fc);
|
|
}
|
|
}
|
|
|
|
private static void testExceptions(FileChannel fc) throws IOException {
|
|
checkException(fc, null, 0L, fc.size(),
|
|
NullPointerException.class);
|
|
|
|
checkException(fc, MapMode.READ_ONLY, -1L, fc.size(),
|
|
IllegalArgumentException.class);
|
|
|
|
checkException(fc, null, -1L, fc.size(),
|
|
IllegalArgumentException.class, NullPointerException.class);
|
|
|
|
checkException(fc, MapMode.READ_ONLY, 0L, -1L,
|
|
IllegalArgumentException.class);
|
|
|
|
checkException(fc, null, 0L, -1L,
|
|
IllegalArgumentException.class, NullPointerException.class);
|
|
|
|
checkException(fc, MapMode.READ_ONLY, 0L, Integer.MAX_VALUE + 1L,
|
|
IllegalArgumentException.class);
|
|
|
|
checkException(fc, null, 0L, Integer.MAX_VALUE + 1L,
|
|
IllegalArgumentException.class, NullPointerException.class);
|
|
|
|
checkException(fc, MapMode.READ_ONLY, Long.MAX_VALUE, 1L,
|
|
IllegalArgumentException.class);
|
|
|
|
checkException(fc, null, Long.MAX_VALUE, 1L,
|
|
IllegalArgumentException.class, NullPointerException.class);
|
|
|
|
}
|
|
|
|
/**
|
|
* Checks that FileChannel map throws one of the expected exceptions
|
|
* when invoked with the given inputs.
|
|
*/
|
|
private static void checkException(FileChannel fc,
|
|
MapMode mode,
|
|
long position,
|
|
long size,
|
|
Class<?>... expected)
|
|
throws IOException
|
|
{
|
|
Exception exc = null;
|
|
try {
|
|
fc.map(mode, position, size);
|
|
} catch (Exception actual) {
|
|
exc = actual;
|
|
}
|
|
if (exc != null) {
|
|
for (Class<?> clazz: expected) {
|
|
if (clazz.isInstance(exc)) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
System.err.println("Expected one of");
|
|
for (Class<?> clazz: expected) {
|
|
System.out.println(clazz);
|
|
}
|
|
if (exc == null) {
|
|
throw new RuntimeException("No expection thrown");
|
|
} else {
|
|
throw new RuntimeException("Unexpected exception thrown", exc);
|
|
}
|
|
}
|
|
}
|