4b8f7db6be
Reviewed-by: vromero
640 lines
22 KiB
Java
640 lines
22 KiB
Java
/*
|
|
* Copyright (c) 2008, 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 4607272 5041655 6822643 6830721 6842687
|
|
* @summary Unit test for AsynchronousFileChannel
|
|
* @key randomness
|
|
*/
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.nio.ByteBuffer;
|
|
import java.nio.channels.AsynchronousCloseException;
|
|
import java.nio.channels.AsynchronousFileChannel;
|
|
import java.nio.channels.ClosedChannelException;
|
|
import java.nio.channels.CompletionHandler;
|
|
import java.nio.channels.FileLock;
|
|
import java.nio.channels.NonWritableChannelException;
|
|
import java.nio.channels.OverlappingFileLockException;
|
|
import java.nio.file.Path;
|
|
import java.nio.file.StandardOpenOption;
|
|
import java.util.ArrayList;
|
|
import java.util.EnumSet;
|
|
import java.util.List;
|
|
import java.util.Random;
|
|
import java.util.Set;
|
|
import java.util.concurrent.CancellationException;
|
|
import java.util.concurrent.CountDownLatch;
|
|
import java.util.concurrent.ExecutionException;
|
|
import java.util.concurrent.Executors;
|
|
import java.util.concurrent.ExecutorService;
|
|
import java.util.concurrent.Future;
|
|
import java.util.concurrent.ThreadFactory;
|
|
import java.util.concurrent.TimeoutException;
|
|
import java.util.concurrent.TimeUnit;
|
|
import java.util.concurrent.atomic.AtomicReference;
|
|
import static java.nio.file.StandardOpenOption.*;
|
|
|
|
public class Basic {
|
|
|
|
private static final Random rand = new Random();
|
|
|
|
public static void main(String[] args) throws IOException {
|
|
// create temporary file
|
|
File blah = File.createTempFile("blah", null);
|
|
blah.deleteOnExit();
|
|
|
|
AsynchronousFileChannel ch = AsynchronousFileChannel
|
|
.open(blah.toPath(), READ, WRITE);
|
|
try {
|
|
// run tests
|
|
testUsingCompletionHandlers(ch);
|
|
testUsingWaitOnResult(ch);
|
|
testInterruptHandlerThread(ch);
|
|
} finally {
|
|
ch.close();
|
|
}
|
|
|
|
// run test that expects channel to be closed
|
|
testClosedChannel(ch);
|
|
|
|
// these tests open the file themselves
|
|
testLocking(blah.toPath());
|
|
testCustomThreadPool(blah.toPath());
|
|
testAsynchronousClose(blah.toPath());
|
|
testCancel(blah.toPath());
|
|
testTruncate(blah.toPath());
|
|
|
|
// eagerly clean-up
|
|
blah.delete();
|
|
}
|
|
|
|
/*
|
|
* Generate buffer with random contents
|
|
* Writes buffer to file using a CompletionHandler to consume the result
|
|
* of each write operation
|
|
* Reads file to EOF to a new buffer using a CompletionHandler to consume
|
|
* the result of each read operation
|
|
* Compares buffer contents
|
|
*/
|
|
static void testUsingCompletionHandlers(AsynchronousFileChannel ch)
|
|
throws IOException
|
|
{
|
|
System.out.println("testUsingCompletionHandlers");
|
|
|
|
ch.truncate(0L);
|
|
|
|
// generate buffer with random elements and write it to file
|
|
ByteBuffer src = genBuffer();
|
|
writeFully(ch, src, 0L);
|
|
|
|
// read to EOF or buffer is full
|
|
ByteBuffer dst = (rand.nextBoolean()) ?
|
|
ByteBuffer.allocateDirect(src.capacity()) :
|
|
ByteBuffer.allocate(src.capacity());
|
|
readAll(ch, dst, 0L);
|
|
|
|
// check buffers are the same
|
|
src.flip();
|
|
dst.flip();
|
|
if (!src.equals(dst)) {
|
|
throw new RuntimeException("Contents differ");
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Generate buffer with random contents
|
|
* Writes buffer to file, invoking the Future's get method to wait for
|
|
* each write operation to complete
|
|
* Reads file to EOF to a new buffer, invoking the Future's get method to
|
|
* wait for each write operation to complete
|
|
* Compares buffer contents
|
|
*/
|
|
static void testUsingWaitOnResult(AsynchronousFileChannel ch)
|
|
throws IOException
|
|
{
|
|
System.out.println("testUsingWaitOnResult");
|
|
|
|
ch.truncate(0L);
|
|
|
|
// generate buffer
|
|
ByteBuffer src = genBuffer();
|
|
|
|
// write buffer completely to file
|
|
long position = 0L;
|
|
while (src.hasRemaining()) {
|
|
Future<Integer> result = ch.write(src, position);
|
|
try {
|
|
int n = result.get();
|
|
// update position
|
|
position += n;
|
|
} catch (ExecutionException x) {
|
|
throw new RuntimeException(x.getCause());
|
|
} catch (InterruptedException x) {
|
|
throw new RuntimeException(x);
|
|
}
|
|
}
|
|
|
|
// read file into new buffer
|
|
ByteBuffer dst = (rand.nextBoolean()) ?
|
|
ByteBuffer.allocateDirect(src.capacity()) :
|
|
ByteBuffer.allocate(src.capacity());
|
|
position = 0L;
|
|
int n;
|
|
do {
|
|
Future<Integer> result = ch.read(dst, position);
|
|
try {
|
|
n = result.get();
|
|
|
|
// update position
|
|
if (n > 0) position += n;
|
|
} catch (ExecutionException x) {
|
|
throw new RuntimeException(x.getCause());
|
|
} catch (InterruptedException x) {
|
|
throw new RuntimeException(x);
|
|
}
|
|
} while (n > 0);
|
|
|
|
// check buffers are the same
|
|
src.flip();
|
|
dst.flip();
|
|
if (!src.equals(dst)) {
|
|
throw new RuntimeException("Contents differ");
|
|
}
|
|
}
|
|
|
|
// exercise lock methods
|
|
static void testLocking(Path file) throws IOException {
|
|
System.out.println("testLocking");
|
|
|
|
AsynchronousFileChannel ch = AsynchronousFileChannel
|
|
.open(file, READ, WRITE);
|
|
FileLock fl;
|
|
try {
|
|
// test 1 - acquire lock and check that tryLock throws
|
|
// OverlappingFileLockException
|
|
try {
|
|
long pos = rand.nextInt(Integer.MAX_VALUE);
|
|
fl = ch.lock(pos, 0, false).get();
|
|
long expectedSize = Long.MAX_VALUE - pos;
|
|
if(fl.size() != expectedSize)
|
|
throw new RuntimeException("Lock size " + fl.size() +
|
|
" != " + expectedSize + " for position " + pos);
|
|
} catch (ExecutionException x) {
|
|
throw new RuntimeException(x);
|
|
} catch (InterruptedException x) {
|
|
throw new RuntimeException("Should not be interrupted");
|
|
}
|
|
if (!fl.acquiredBy().equals(ch))
|
|
throw new RuntimeException("FileLock#acquiredBy returned incorrect channel");
|
|
try {
|
|
ch.tryLock();
|
|
throw new RuntimeException("OverlappingFileLockException expected");
|
|
} catch (OverlappingFileLockException x) {
|
|
}
|
|
fl.release();
|
|
|
|
// test 2 - acquire try and check that lock throws OverlappingFileLockException
|
|
long pos = rand.nextInt(Integer.MAX_VALUE);
|
|
fl = ch.tryLock(pos, 0, false);
|
|
long expectedSize = Long.MAX_VALUE - pos;
|
|
if(fl.size() != expectedSize)
|
|
throw new RuntimeException("Lock size " + fl.size() + " != " +
|
|
expectedSize + " for position " + pos);
|
|
if (fl == null)
|
|
throw new RuntimeException("Unable to acquire lock");
|
|
try {
|
|
ch.lock((Void)null, new CompletionHandler<FileLock,Void> () {
|
|
public void completed(FileLock result, Void att) {
|
|
}
|
|
public void failed(Throwable exc, Void att) {
|
|
}
|
|
});
|
|
throw new RuntimeException("OverlappingFileLockException expected");
|
|
} catch (OverlappingFileLockException x) {
|
|
}
|
|
} finally {
|
|
ch.close();
|
|
}
|
|
|
|
// test 3 - channel is closed so FileLock should no longer be valid
|
|
if (fl.isValid())
|
|
throw new RuntimeException("FileLock expected to be invalid");
|
|
}
|
|
|
|
// interrupt should not close channel
|
|
static void testInterruptHandlerThread(final AsynchronousFileChannel ch) {
|
|
System.out.println("testInterruptHandlerThread");
|
|
|
|
ByteBuffer buf = ByteBuffer.allocateDirect(100);
|
|
final CountDownLatch latch = new CountDownLatch(1);
|
|
|
|
ch.read(buf, 0L, (Void)null, new CompletionHandler<Integer,Void>() {
|
|
public void completed(Integer result, Void att) {
|
|
try {
|
|
Thread.currentThread().interrupt();
|
|
long size = ch.size();
|
|
latch.countDown();
|
|
} catch (IOException x) {
|
|
x.printStackTrace();
|
|
}
|
|
}
|
|
public void failed(Throwable exc, Void att) {
|
|
}
|
|
});
|
|
|
|
// wait for handler to complete
|
|
await(latch);
|
|
}
|
|
|
|
// invoke method on closed channel
|
|
static void testClosedChannel(AsynchronousFileChannel ch) {
|
|
System.out.println("testClosedChannel");
|
|
|
|
if (ch.isOpen())
|
|
throw new RuntimeException("Channel should be closed");
|
|
|
|
ByteBuffer buf = ByteBuffer.allocateDirect(100);
|
|
|
|
// check read fails with ClosedChannelException
|
|
try {
|
|
ch.read(buf, 0L).get();
|
|
throw new RuntimeException("ExecutionException expected");
|
|
} catch (ExecutionException x) {
|
|
if (!(x.getCause() instanceof ClosedChannelException))
|
|
throw new RuntimeException("Cause of ClosedChannelException expected");
|
|
} catch (InterruptedException x) {
|
|
}
|
|
|
|
// check write fails with ClosedChannelException
|
|
try {
|
|
ch.write(buf, 0L).get();
|
|
throw new RuntimeException("ExecutionException expected");
|
|
} catch (ExecutionException x) {
|
|
if (!(x.getCause() instanceof ClosedChannelException))
|
|
throw new RuntimeException("Cause of ClosedChannelException expected");
|
|
} catch (InterruptedException x) {
|
|
}
|
|
|
|
// check lock fails with ClosedChannelException
|
|
try {
|
|
ch.lock().get();
|
|
throw new RuntimeException("ExecutionException expected");
|
|
} catch (ExecutionException x) {
|
|
if (!(x.getCause() instanceof ClosedChannelException))
|
|
throw new RuntimeException("Cause of ClosedChannelException expected");
|
|
} catch (InterruptedException x) {
|
|
}
|
|
}
|
|
|
|
|
|
// exercise custom thread pool
|
|
static void testCustomThreadPool(Path file) throws IOException {
|
|
System.out.println("testCustomThreadPool");
|
|
|
|
// records threads that are created
|
|
final List<Thread> threads = new ArrayList<Thread>();
|
|
|
|
ThreadFactory threadFactory = new ThreadFactory() {
|
|
@Override
|
|
public Thread newThread(Runnable r) {
|
|
Thread t = new Thread(r);
|
|
t.setDaemon(true);
|
|
synchronized (threads) {
|
|
threads.add(t);
|
|
}
|
|
return t;
|
|
}
|
|
};
|
|
|
|
// exercise tests with varied number of threads
|
|
for (int nThreads=1; nThreads<=5; nThreads++) {
|
|
synchronized (threads) {
|
|
threads.clear();
|
|
}
|
|
ExecutorService executor = Executors.newFixedThreadPool(nThreads, threadFactory);
|
|
Set<StandardOpenOption> opts = EnumSet.of(WRITE);
|
|
AsynchronousFileChannel ch = AsynchronousFileChannel.open(file, opts, executor);
|
|
try {
|
|
for (int i=0; i<10; i++) {
|
|
// do I/O operation to see which thread invokes the completion handler
|
|
final AtomicReference<Thread> invoker = new AtomicReference<Thread>();
|
|
final CountDownLatch latch = new CountDownLatch(1);
|
|
|
|
ch.write(genBuffer(), 0L, (Void)null, new CompletionHandler<Integer,Void>() {
|
|
public void completed(Integer result, Void att) {
|
|
invoker.set(Thread.currentThread());
|
|
latch.countDown();
|
|
}
|
|
public void failed(Throwable exc, Void att) {
|
|
}
|
|
});
|
|
await(latch);
|
|
|
|
// check invoker
|
|
boolean found = false;
|
|
synchronized (threads) {
|
|
for (Thread t: threads) {
|
|
if (t == invoker.get()) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (!found)
|
|
throw new RuntimeException("Invoker thread not found");
|
|
}
|
|
} finally {
|
|
ch.close();
|
|
executor.shutdown();
|
|
}
|
|
}
|
|
|
|
|
|
// test sharing a thread pool between many channels
|
|
ExecutorService executor = Executors
|
|
.newFixedThreadPool(1+rand.nextInt(10), threadFactory);
|
|
final int n = 50 + rand.nextInt(50);
|
|
AsynchronousFileChannel[] channels = new AsynchronousFileChannel[n];
|
|
try {
|
|
for (int i=0; i<n; i++) {
|
|
Set<StandardOpenOption> opts = EnumSet.of(WRITE);
|
|
channels[i] = AsynchronousFileChannel.open(file, opts, executor);
|
|
final CountDownLatch latch = new CountDownLatch(1);
|
|
channels[i].write(genBuffer(), 0L, (Void)null, new CompletionHandler<Integer,Void>() {
|
|
public void completed(Integer result, Void att) {
|
|
latch.countDown();
|
|
}
|
|
public void failed(Throwable exc, Void att) {
|
|
}
|
|
});
|
|
await(latch);
|
|
|
|
// close ~half the channels
|
|
if (rand.nextBoolean())
|
|
channels[i].close();
|
|
}
|
|
} finally {
|
|
// close remaining channels
|
|
for (int i=0; i<n; i++) {
|
|
if (channels[i] != null) channels[i].close();
|
|
}
|
|
executor.shutdown();
|
|
}
|
|
}
|
|
|
|
// exercise asynchronous close
|
|
static void testAsynchronousClose(Path file) throws IOException {
|
|
System.out.println("testAsynchronousClose");
|
|
|
|
// create file
|
|
AsynchronousFileChannel ch = AsynchronousFileChannel
|
|
.open(file, WRITE, TRUNCATE_EXISTING);
|
|
long size = 0L;
|
|
do {
|
|
ByteBuffer buf = genBuffer();
|
|
int n = buf.remaining();
|
|
writeFully(ch, buf, size);
|
|
size += n;
|
|
} while (size < (50L * 1024L * 1024L));
|
|
|
|
ch.close();
|
|
|
|
ch = AsynchronousFileChannel.open(file, WRITE, SYNC);
|
|
|
|
// randomize number of writers, buffer size, and positions
|
|
|
|
int nwriters = 1 + rand.nextInt(8);
|
|
ByteBuffer[] buf = new ByteBuffer[nwriters];
|
|
long[] position = new long[nwriters];
|
|
for (int i=0; i<nwriters; i++) {
|
|
buf[i] = genBuffer();
|
|
position[i] = rand.nextInt((int)size);
|
|
}
|
|
|
|
// initiate I/O
|
|
Future[] result = new Future[nwriters];
|
|
for (int i=0; i<nwriters; i++) {
|
|
result[i] = ch.write(buf[i], position[i]);
|
|
}
|
|
|
|
// close file
|
|
ch.close();
|
|
|
|
// write operations should complete or fail with AsynchronousCloseException
|
|
for (int i=0; i<nwriters; i++) {
|
|
try {
|
|
result[i].get();
|
|
} catch (ExecutionException x) {
|
|
Throwable cause = x.getCause();
|
|
if (!(cause instanceof AsynchronousCloseException))
|
|
throw new RuntimeException(cause);
|
|
} catch (CancellationException x) {
|
|
throw new RuntimeException(x); // should not happen
|
|
} catch (InterruptedException x) {
|
|
throw new RuntimeException(x); // should not happen
|
|
}
|
|
}
|
|
}
|
|
|
|
// exercise cancel method
|
|
static void testCancel(Path file) throws IOException {
|
|
System.out.println("testCancel");
|
|
|
|
for (int i=0; i<2; i++) {
|
|
boolean mayInterruptIfRunning = (i == 0) ? false : true;
|
|
|
|
// open with SYNC option to improve chances that write will not
|
|
// complete immediately
|
|
AsynchronousFileChannel ch = AsynchronousFileChannel
|
|
.open(file, WRITE, SYNC);
|
|
|
|
// start write operation
|
|
Future<Integer> res = ch.write(genBuffer(), 0L);
|
|
|
|
// cancel operation
|
|
boolean cancelled = res.cancel(mayInterruptIfRunning);
|
|
|
|
// check post-conditions
|
|
if (!res.isDone())
|
|
throw new RuntimeException("isDone should return true");
|
|
if (res.isCancelled() != cancelled)
|
|
throw new RuntimeException("isCancelled not consistent");
|
|
try {
|
|
res.get();
|
|
if (cancelled)
|
|
throw new RuntimeException("CancellationException expected");
|
|
} catch (CancellationException x) {
|
|
if (!cancelled)
|
|
throw new RuntimeException("CancellationException not expected");
|
|
} catch (ExecutionException x) {
|
|
throw new RuntimeException(x);
|
|
} catch (InterruptedException x) {
|
|
throw new RuntimeException(x);
|
|
}
|
|
try {
|
|
res.get(1, TimeUnit.SECONDS);
|
|
if (cancelled)
|
|
throw new RuntimeException("CancellationException expected");
|
|
} catch (CancellationException x) {
|
|
if (!cancelled)
|
|
throw new RuntimeException("CancellationException not expected");
|
|
} catch (ExecutionException x) {
|
|
throw new RuntimeException(x);
|
|
} catch (TimeoutException x) {
|
|
throw new RuntimeException(x);
|
|
} catch (InterruptedException x) {
|
|
throw new RuntimeException(x);
|
|
}
|
|
|
|
ch.close();
|
|
}
|
|
}
|
|
|
|
// exercise truncate method
|
|
static void testTruncate(Path file) throws IOException {
|
|
System.out.println("testTruncate");
|
|
|
|
// basic tests
|
|
AsynchronousFileChannel ch = AsynchronousFileChannel
|
|
.open(file, CREATE, WRITE, TRUNCATE_EXISTING);
|
|
try {
|
|
writeFully(ch, genBuffer(), 0L);
|
|
long size = ch.size();
|
|
|
|
// attempt to truncate to a size greater than the current size
|
|
if (ch.truncate(size + 1L).size() != size)
|
|
throw new RuntimeException("Unexpected size after truncation");
|
|
|
|
// truncate file
|
|
if (ch.truncate(size - 1L).size() != (size - 1L))
|
|
throw new RuntimeException("Unexpected size after truncation");
|
|
|
|
// invalid size
|
|
try {
|
|
ch.truncate(-1L);
|
|
throw new RuntimeException("IllegalArgumentException expected");
|
|
} catch (IllegalArgumentException e) { }
|
|
|
|
} finally {
|
|
ch.close();
|
|
}
|
|
|
|
// channel is closed
|
|
try {
|
|
ch.truncate(0L);
|
|
throw new RuntimeException("ClosedChannelException expected");
|
|
} catch (ClosedChannelException e) { }
|
|
|
|
// channel is read-only
|
|
ch = AsynchronousFileChannel.open(file, READ);
|
|
try {
|
|
try {
|
|
ch.truncate(0L);
|
|
throw new RuntimeException("NonWritableChannelException expected");
|
|
} catch (NonWritableChannelException e) { }
|
|
} finally {
|
|
ch.close();
|
|
}
|
|
}
|
|
|
|
// returns ByteBuffer with random bytes
|
|
static ByteBuffer genBuffer() {
|
|
int size = 1024 + rand.nextInt(16000);
|
|
byte[] buf = new byte[size];
|
|
boolean useDirect = rand.nextBoolean();
|
|
if (useDirect) {
|
|
ByteBuffer bb = ByteBuffer.allocateDirect(buf.length);
|
|
bb.put(buf);
|
|
bb.flip();
|
|
return bb;
|
|
} else {
|
|
return ByteBuffer.wrap(buf);
|
|
}
|
|
}
|
|
|
|
// writes all remaining bytes in the buffer to the given channel at the
|
|
// given position
|
|
static void writeFully(final AsynchronousFileChannel ch,
|
|
final ByteBuffer src,
|
|
long position)
|
|
{
|
|
final CountDownLatch latch = new CountDownLatch(1);
|
|
|
|
// use position as attachment
|
|
ch.write(src, position, position, new CompletionHandler<Integer,Long>() {
|
|
public void completed(Integer result, Long position) {
|
|
int n = result;
|
|
if (src.hasRemaining()) {
|
|
long p = position + n;
|
|
ch.write(src, p, p, this);
|
|
} else {
|
|
latch.countDown();
|
|
}
|
|
}
|
|
public void failed(Throwable exc, Long position) {
|
|
}
|
|
});
|
|
|
|
// wait for writes to complete
|
|
await(latch);
|
|
}
|
|
|
|
static void readAll(final AsynchronousFileChannel ch,
|
|
final ByteBuffer dst,
|
|
long position)
|
|
{
|
|
final CountDownLatch latch = new CountDownLatch(1);
|
|
|
|
// use position as attachment
|
|
ch.read(dst, position, position, new CompletionHandler<Integer,Long>() {
|
|
public void completed(Integer result, Long position) {
|
|
int n = result;
|
|
if (n > 0) {
|
|
long p = position + n;
|
|
ch.read(dst, p, p, this);
|
|
} else {
|
|
latch.countDown();
|
|
}
|
|
}
|
|
public void failed(Throwable exc, Long position) {
|
|
}
|
|
});
|
|
|
|
// wait for reads to complete
|
|
await(latch);
|
|
}
|
|
|
|
static void await(CountDownLatch latch) {
|
|
// wait until done
|
|
boolean done = false;
|
|
while (!done) {
|
|
try {
|
|
latch.await();
|
|
done = true;
|
|
} catch (InterruptedException x) { }
|
|
}
|
|
}
|
|
}
|