--- /dev/null 2016-11-14 12:26:28.251338568 -0800 +++ new/test/java/nio/channels/FileChannel/PwriteDirect.java 2016-11-16 10:28:21.729497933 -0800 @@ -0,0 +1,159 @@ +/* + * Copyright (c) 2000, 2010, 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 4862411 + * @summary Test positional write method of FileChannel with DirectIO + * @key randomness + */ + +import java.io.*; +import java.nio.ByteBuffer; +import java.nio.CharBuffer; +import java.nio.channels.*; +import java.nio.channels.FileChannel; +import java.util.Random; +import java.nio.file.Paths; +import java.nio.file.Path; +import java.nio.file.Files; +import java.nio.file.FileStore; +import java.nio.file.StandardOpenOption; +import com.sun.nio.file.ExtendedOpenOption; + +/** + * Testing FileChannel's positional write method. + */ +public class PwriteDirect { + + private static Random generator = new Random(); + + private static File blah; + + public static void main(String[] args) throws Exception { + genericTest(); + testUnwritableChannel(); + } + + // This test for bug 4862411 + private static void testUnwritableChannel() throws Exception { + File blah = File.createTempFile("blah2", null); + blah.deleteOnExit(); + FileOutputStream fos = new FileOutputStream(blah); + fos.write(new byte[4096]); + fos.close(); + + String path = blah.getAbsolutePath(); + Path p = Paths.get(path); + FileChannel fc = FileChannel.open(p, ExtendedOpenOption.DIRECT); + + try { + fc.write(ByteBuffer.allocate(4096),0); + throw new RuntimeException("Expected exception not thrown"); + } catch(NonWritableChannelException e) { + // Correct result + } finally { + fc.close(); + blah.delete(); + } + } + + private static void genericTest() throws Exception { + blah = File.createTempFile("blah", null); + blah.deleteOnExit(); + initTestFile(blah); + + String path = blah.getAbsolutePath(); + Path p = Paths.get(path); + FileChannel c = FileChannel.open(p, StandardOpenOption.WRITE, StandardOpenOption.READ, ExtendedOpenOption.DIRECT); + FileStore fs = Files.getFileStore(p); + int alignment = fs.getBlockSize(); + + for (int x=0; x<100; x++) { + long offset = generator.nextInt(100) * 4096; + ByteBuffer block = ByteBuffer.allocateDirect(4096 + alignment - 1).alignedSlice(alignment); + + // Write known sequence out + for (byte i=0; i<4; i++) { + block.put(i); + } + block.position(0); + long originalPosition = c.position(); + int totalWritten = 0; + while (totalWritten < 4096) { + int written = c.write(block, offset); + if (written < 0) + throw new Exception("Write failed"); + totalWritten += written; + } + + long newPosition = c.position(); + + // Ensure that file pointer position has not changed + if (originalPosition != newPosition) + throw new Exception("File position modified"); + + // Attempt to read sequence back in + block = ByteBuffer.allocateDirect(4096 + alignment - 1).alignedSlice(alignment); + originalPosition = c.position(); + int totalRead = 0; + while (totalRead < 4096) { + int read = c.read(block, offset); + if (read < 0) + throw new Exception("Read failed"); + totalRead += read; + } + newPosition = c.position(); + + // Ensure that file pointer position has not changed + if (originalPosition != newPosition) + throw new Exception("File position modified"); + + for (byte i=0; i<4; i++) { + if (block.get(i) != i) + throw new Exception("Write test failed"); + } + } + c.close(); + blah.delete(); + } + + /** + * Creates file blah: + */ + private static void initTestFile(File blah) throws Exception { + FileOutputStream fos = new FileOutputStream(blah); + BufferedWriter awriter + = new BufferedWriter(new OutputStreamWriter(fos, "8859_1")); + + for(int i=0; i<100; i++) { + String number = new Integer(i).toString(); + for (int h=0; h<4-number.length(); h++) + awriter.write("0"); + awriter.write(""+i); + for (int j=0; j < 4092; j++) + awriter.write("0"); + } + awriter.flush(); + awriter.close(); + } +}