From 7baccd9ee24bcbc2ca5e449d8d3e5b80860cc088 Mon Sep 17 00:00:00 2001
From: Brian Burkhalter This method does not modify this channel's position. If the given
- * position is greater than or equal to the file's current size then no bytes are
- * transferred. If the target channel has a position then bytes are
- * written starting at that position and then the position is incremented
- * by the number of bytes written.
+ * position is greater than or equal to the file's current size then no
+ * bytes are transferred. If the target channel has a position then bytes
+ * are written starting at that position and then the position
+ * is incremented by the number of bytes written.
*
* This method is potentially much more efficient than a simple loop
* that reads from this channel and writes to the target channel. Many
@@ -701,8 +701,10 @@ public abstract class FileChannel
* source has reached end-of-stream.
*
* This method does not modify this channel's position. If the given
- * position is greater than the file's current size then no bytes are
- * transferred. If the source channel has a position then bytes are read
+ * position is greater than or equal to the file's current size then the
+ * file will be grown to accommodate the new bytes; the values of any bytes
+ * between the previous end-of-file and the newly-written bytes are
+ * unspecified. If the source channel has a position then bytes are read
* starting at that position and then the position is incremented by the
* number of bytes read.
*
@@ -715,7 +717,7 @@ public abstract class FileChannel
* The source channel
*
* @param position
- * The position within the file at which the transfer is to begin;
+ * The file position at which the transfer is to begin;
* must be non-negative
*
* @param count
@@ -761,7 +763,8 @@ public abstract class FileChannel
* #read(ByteBuffer)} method, except that bytes are read starting at the
* given file position rather than at the channel's current position. This
* method does not modify this channel's position. If the given position
- * is greater than or equal to the file's current size then no bytes are read.
If the file is open in append mode, then
* the effect of invoking this method is unspecified.
diff --git a/src/java.base/share/classes/sun/nio/ch/FileChannelImpl.java b/src/java.base/share/classes/sun/nio/ch/FileChannelImpl.java
index 4aeafa2b460..72fb1824323 100644
--- a/src/java.base/share/classes/sun/nio/ch/FileChannelImpl.java
+++ b/src/java.base/share/classes/sun/nio/ch/FileChannelImpl.java
@@ -935,8 +935,6 @@ public class FileChannelImpl
throw new NonWritableChannelException();
if ((position < 0) || (count < 0))
throw new IllegalArgumentException();
- if (position > size())
- return 0;
// System calls supporting fast transfers might not work on files
// which advertise zero size such as those in Linux /proc
diff --git a/test/jdk/java/nio/channels/FileChannel/TransferFromExtend.java b/test/jdk/java/nio/channels/FileChannel/TransferFromExtend.java
new file mode 100644
index 00000000000..538964e2fcd
--- /dev/null
+++ b/test/jdk/java/nio/channels/FileChannel/TransferFromExtend.java
@@ -0,0 +1,161 @@
+/*
+ * Copyright (c) 2023, 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 8303260
+ * @summary Test transferFrom to a position greater than the file size
+ * @library .. /test/lib
+ * @build jdk.test.lib.RandomFactory
+ * @run junit TransferFromExtend
+ * @key randomness
+ */
+
+import java.io.ByteArrayInputStream;
+import java.io.IOException;
+import java.nio.ByteBuffer;
+import java.nio.channels.Channels;
+import java.nio.channels.FileChannel;
+import java.nio.channels.ReadableByteChannel;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+import java.util.Random;
+import java.util.stream.Stream;
+
+import static java.nio.file.StandardOpenOption.*;
+
+import jdk.test.lib.RandomFactory;
+
+import org.junit.jupiter.params.ParameterizedTest;
+import org.junit.jupiter.params.provider.Arguments;
+import org.junit.jupiter.params.provider.MethodSource;
+
+import static org.junit.jupiter.api.Assertions.*;
+
+public class TransferFromExtend {
+ private static final Random RND = RandomFactory.getRandom();
+
+ private static final Path DIR = Path.of(System.getProperty("test.dir", "."));
+
+ private static Stream