From 8774d700441268f9c04da7292a3b67ce09451e6f Mon Sep 17 00:00:00 2001 From: Ivan Gerasimov Date: Fri, 29 Jun 2018 17:35:04 -0700 Subject: [PATCH] 8204310: Simpler RandomAccessFile.setLength() on Windows Reviewed-by: alanb --- .../windows/native/libjava/io_util_md.c | 17 +- .../windows/native/libjava/io_util_md.h | 2 +- .../java/io/RandomAccessFile/SetLength.java | 165 +++++++++++++----- .../nio/channels/FileChannel/TruncateRAF.java | 155 ++++++++++++++++ 4 files changed, 289 insertions(+), 50 deletions(-) create mode 100644 test/jdk/java/nio/channels/FileChannel/TruncateRAF.java diff --git a/src/java.base/windows/native/libjava/io_util_md.c b/src/java.base/windows/native/libjava/io_util_md.c index c077e2e6742..db8ef0d18cf 100644 --- a/src/java.base/windows/native/libjava/io_util_md.c +++ b/src/java.base/windows/native/libjava/io_util_md.c @@ -458,19 +458,20 @@ handleSync(FD fd) { return 0; } - -int +jint handleSetLength(FD fd, jlong length) { HANDLE h = (HANDLE)fd; - long high = (long)(length >> 32); - DWORD ret; + FILE_END_OF_FILE_INFO eofInfo; - if (h == (HANDLE)(-1)) return -1; - ret = SetFilePointer(h, (long)(length), &high, FILE_BEGIN); - if (ret == 0xFFFFFFFF && GetLastError() != NO_ERROR) { + eofInfo.EndOfFile.QuadPart = length; + + if (h == INVALID_HANDLE_VALUE) { + return -1; + } + if (!SetFileInformationByHandle(h, FileEndOfFileInfo, &eofInfo, + sizeof(FILE_END_OF_FILE_INFO))) { return -1; } - if (SetEndOfFile(h) == FALSE) return -1; return 0; } diff --git a/src/java.base/windows/native/libjava/io_util_md.h b/src/java.base/windows/native/libjava/io_util_md.h index 66f9ab5b214..d0a811794da 100644 --- a/src/java.base/windows/native/libjava/io_util_md.h +++ b/src/java.base/windows/native/libjava/io_util_md.h @@ -43,7 +43,7 @@ WCHAR* currentDir(int di); int currentDirLength(const WCHAR* path, int pathlen); int handleAvailable(FD fd, jlong *pbytes); int handleSync(FD fd); -int handleSetLength(FD fd, jlong length); +jint handleSetLength(FD fd, jlong length); jlong handleGetLength(FD fd); JNIEXPORT jint handleRead(FD fd, void *buf, jint len); jint handleWrite(FD fd, const void *buf, jint len); diff --git a/test/jdk/java/io/RandomAccessFile/SetLength.java b/test/jdk/java/io/RandomAccessFile/SetLength.java index 10098794aae..826b1ee5814 100644 --- a/test/jdk/java/io/RandomAccessFile/SetLength.java +++ b/test/jdk/java/io/RandomAccessFile/SetLength.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -22,60 +22,143 @@ */ /* @test - @summary General tests of the setLength method -- Should migrate to 1.2 JCK + * @bug 8204310 + * @summary General tests of the setLength method + * @library /test/lib + * @build jdk.test.lib.RandomFactory + * @run main SetLength + * @key randomness */ -import java.io.*; +import java.io.IOException; +import java.io.File; +import java.io.RandomAccessFile; +import jdk.test.lib.RandomFactory; public class SetLength { - static void fail(String s) { - throw new RuntimeException(s); + static void checkState(RandomAccessFile f, long expectedFilePointer, + long expectedLength) + throws IOException + { + long filePointer = f.getFilePointer(); + long length = f.length(); + if (length != expectedLength) { + throw new RuntimeException("File length " + length + " != expected " + + expectedLength); + } + if (filePointer != expectedFilePointer) { + throw new RuntimeException("File pointer " + filePointer + + " != expected " + expectedFilePointer); + } } - static void go(File fn, int max) throws IOException { - int chunk = max / 4; - long i; - RandomAccessFile f; + static void test(RandomAccessFile f, long quarterLength) + throws IOException + { + long halfLength = 2 * quarterLength; + long threeQuarterLength = 3 * quarterLength; + long fullLength = 4 * quarterLength; - f = new RandomAccessFile(fn, "rw"); - f.setLength(2 * chunk); - if (f.length() != 2 * chunk) fail("Length not increased to " + (2 * chunk)); - if ((i = f.getFilePointer()) != 0) fail("File pointer shifted to " + i); - byte[] buf = new byte[max]; - f.write(buf); - if (f.length() != max) fail("Write didn't work"); - if (f.getFilePointer() != max) fail("File pointer inconsistent"); - f.setLength(3 * chunk); - if (f.length() != 3 * chunk) fail("Length not reduced to " + 3 * chunk); - if (f.getFilePointer() != 3 * chunk) fail("File pointer not shifted to " + (3 * chunk)); - f.seek(1 * chunk); - if (f.getFilePointer() != 1 * chunk) fail("File pointer not shifted to " + (1 * chunk)); - f.setLength(2 * chunk); - if (f.length() != 2 * chunk) fail("Length not reduced to " + (2 * chunk)); - if (f.getFilePointer() != 1 * chunk) fail("File pointer not shifted to " + (1 * chunk)); + // initially, empty file + checkState(f, 0, 0); + + // extending the file size + f.setLength(halfLength); + checkState(f, 0, halfLength); + + // writing from the begining + f.write(new byte[(int)fullLength]); + checkState(f, fullLength, fullLength); + + // setting to the same length + f.setLength(fullLength); + checkState(f, fullLength, fullLength); + + // truncating the file + f.setLength(threeQuarterLength); + checkState(f, threeQuarterLength, threeQuarterLength); + + // changing the file pointer + f.seek(quarterLength); + checkState(f, quarterLength, threeQuarterLength); + + // truncating the file again + f.setLength(halfLength); + checkState(f, quarterLength, halfLength); + + // writing from the middle with extending the file + f.write(new byte[(int)halfLength]); + checkState(f, threeQuarterLength, threeQuarterLength); + + // changing the file pointer + f.seek(quarterLength); + checkState(f, quarterLength, threeQuarterLength); + + // writing from the middle without extending the file + f.write(new byte[(int)quarterLength]); + checkState(f, halfLength, threeQuarterLength); + + // changing the file pointer to the end of file + f.seek(threeQuarterLength); + checkState(f, threeQuarterLength, threeQuarterLength); + + // writing to the end of file + f.write(new byte[(int)quarterLength]); + checkState(f, fullLength, fullLength); + + // truncating the file to zero + f.setLength(0); + checkState(f, 0, 0); + + // changing the file pointer beyond the end of file + f.seek(threeQuarterLength); + checkState(f, threeQuarterLength, 0); + + // writing beyont the end of file + f.write(new byte[(int)quarterLength]); + checkState(f, fullLength, fullLength); + + // negative file pointer + try { + f.seek(-1); + throw new RuntimeException("IOE not thrown"); + } catch (IOException expected) { + } + checkState(f, fullLength, fullLength); + + // truncating the file after failed seek + f.setLength(halfLength); + checkState(f, halfLength, halfLength); + + // truncating after closing f.close(); + try { + f.setLength(halfLength); + throw new RuntimeException("IOE not thrown"); + } catch (IOException expected) { + } } public static void main(String[] args) throws IOException { - File fn = new File("x.SetLength"); - try { - go(fn, 20); - fn.delete(); - go(fn, 64 * 1024); - RandomAccessFile f = new RandomAccessFile(fn, "r"); - boolean thrown = false; - try { - f.setLength(3); - } catch (IOException x) { - thrown = true; - } - if (!thrown) fail("setLength succeeded on a file opened read-only"); - f.close(); + File f28b = new File("f28b"); + File f28K = new File("f28K"); + File frnd = new File("frnd"); + + try (RandomAccessFile raf28b = new RandomAccessFile(f28b, "rw"); + RandomAccessFile raf28K = new RandomAccessFile(f28K, "rw"); + RandomAccessFile rafrnd = new RandomAccessFile(frnd, "rw")) { + test(raf28b, 7); + test(raf28K, 7 * 1024); + test(rafrnd, 1 + RandomFactory.getRandom().nextInt(16000)); } - finally { - fn.delete(); + + // truncating read-only file + try (RandomAccessFile raf28b = new RandomAccessFile(f28b, "r")) { + raf28b.setLength(42); + throw new RuntimeException("IOE not thrown"); + } catch (IOException expected) { } } diff --git a/test/jdk/java/nio/channels/FileChannel/TruncateRAF.java b/test/jdk/java/nio/channels/FileChannel/TruncateRAF.java new file mode 100644 index 00000000000..5d6a7b3b3ef --- /dev/null +++ b/test/jdk/java/nio/channels/FileChannel/TruncateRAF.java @@ -0,0 +1,155 @@ +/* + * Copyright (c) 2018, 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 8204310 + * @summary Check how FileChannel behaves if the file size/offset change via + * RAF.setLength() and other methods. + * @run main TruncateRAF + */ + +import java.io.File; +import java.io.RandomAccessFile; +import java.io.IOException; +import java.nio.ByteBuffer; +import java.nio.channels.FileChannel; + +public class TruncateRAF { + + static void checkState(RandomAccessFile raf, FileChannel fch, + long expectedOffset, long expectedLength) + throws IOException + { + long rafLength = raf.length(); + long rafOffset = raf.getFilePointer(); + long fchLength = fch.size(); + long fchOffset = fch.position(); + + if (rafLength != expectedLength) + throw new RuntimeException("rafLength (" + rafLength + ") != " + + "expectedLength (" + expectedLength + ")"); + if (rafOffset != expectedOffset) + throw new RuntimeException("rafOffset (" + rafOffset + ") != " + + "expectedOffset (" + expectedOffset + ")"); + if (fchLength != expectedLength) + throw new RuntimeException("fchLength (" + fchLength + ") != " + + "expectedLength (" + expectedLength + ")"); + if (fchOffset != expectedOffset) + throw new RuntimeException("fchOffset (" + fchOffset + ") != " + + "expectedOffset (" + expectedOffset + ")"); + } + + public static void main(String[] args) throws Throwable { + File file = new File("tmp"); + try (RandomAccessFile raf = new RandomAccessFile(file, "rw"); + FileChannel fch = raf.getChannel()) { + + // initially empty + checkState(raf, fch, 0, 0); + + // seeking beyond EOF + raf.seek(42); + checkState(raf, fch, 42, 0); + + // seeking beyond EOF + fch.position(84); + checkState(raf, fch, 84, 0); + + // writing at offset beyond EOF + raf.write(1); + checkState(raf, fch, 85, 85); + + // truncating + raf.setLength(63); + checkState(raf, fch, 63, 63); + + // writing at EOF + fch.write(ByteBuffer.wrap(new byte[1])); + checkState(raf, fch, 64, 64); + + // seeking at the middle + fch.position(32); + checkState(raf, fch, 32, 64); + + // truncating beyond offset + fch.truncate(42); + checkState(raf, fch, 32, 42); + + // truncating before offset + fch.truncate(16); + checkState(raf, fch, 16, 16); + + // writing at position beyond EOF + fch.write(ByteBuffer.wrap(new byte[1]), 127); + checkState(raf, fch, 16, 128); + + // writing at position before EOF + fch.write(ByteBuffer.wrap(new byte[1]), 42); + checkState(raf, fch, 16, 128); + + // truncating + raf.setLength(64); + checkState(raf, fch, 16, 64); + + // changing offset + raf.seek(21); + checkState(raf, fch, 21, 64); + + // skipping should change offset + raf.skipBytes(4); + checkState(raf, fch, 25, 64); + + // reading should change offset + raf.read(); + checkState(raf, fch, 26, 64); + + // truncating to zero + raf.setLength(0); + checkState(raf, fch, 0, 0); + + // FileChannel cannot expand size + fch.truncate(42); + checkState(raf, fch, 0, 0); + + // expanding + raf.setLength(42); + checkState(raf, fch, 0, 42); + + // seeking beyond EOF + raf.seek(512); + checkState(raf, fch, 512, 42); + + // truncating to the same size + fch.truncate(256); + checkState(raf, fch, 256, 42); + + // truncating to the same size + fch.truncate(42); + checkState(raf, fch, 42, 42); + + // truncating to zero + fch.truncate(0); + checkState(raf, fch, 0, 0); + } + } +}