Skip to content

Commit 369611e

Browse files
author
Brian Burkhalter
committedMay 12, 2022
8286677: [BACKOUT] (fc) Tune FileChannel.transferFrom()
Reviewed-by: dholmes, mikael
1 parent 4b8a66a commit 369611e

File tree

4 files changed

+15
-132
lines changed

4 files changed

+15
-132
lines changed
 

‎src/java.base/share/classes/sun/nio/ch/FileChannelImpl.java

+9-70
Original file line numberDiff line numberDiff line change
@@ -541,9 +541,9 @@ public void force(boolean metaData) throws IOException {
541541
}
542542

543543
// Assume at first that the underlying kernel supports sendfile();
544-
// set this to true if we find out later that it doesn't
544+
// set this to false if we find out later that it doesn't
545545
//
546-
private static volatile boolean transferToNotSupported;
546+
private static volatile boolean transferSupported = true;
547547

548548
// Assume that the underlying kernel sendfile() will work if the target
549549
// fd is a pipe; set this to false if we find out later that it doesn't
@@ -587,7 +587,7 @@ private long transferToDirectlyInternal(long position, int icount,
587587
}
588588
if (n == IOStatus.UNSUPPORTED) {
589589
// Don't bother trying again
590-
transferToNotSupported = true;
590+
transferSupported = false;
591591
return IOStatus.UNSUPPORTED;
592592
}
593593
return IOStatus.normalize(n);
@@ -601,7 +601,7 @@ private long transferToDirectly(long position, int icount,
601601
WritableByteChannel target)
602602
throws IOException
603603
{
604-
if (transferToNotSupported)
604+
if (!transferSupported)
605605
return IOStatus.UNSUPPORTED;
606606

607607
FileDescriptor targetFD = null;
@@ -646,9 +646,8 @@ private long transferToDirectly(long position, int icount,
646646
}
647647

648648
// Size threshold above which to use a mapped buffer;
649-
// transferToArbitraryChannel() and transferFromArbitraryChannel()
650-
// are faster for smaller transfers
651-
private static final long MAPPED_TRANSFER_THRESHOLD = 16L*1024L;
649+
// transferToArbitraryChannel() is faster for smaller transfers
650+
private static final long TRUSTED_TRANSFER_THRESHOLD = 16L*1024L;
652651

653652
// Maximum size to map when using a mapped buffer
654653
private static final long MAPPED_TRANSFER_SIZE = 8L*1024L*1024L;
@@ -657,7 +656,7 @@ private long transferToTrustedChannel(long position, long count,
657656
WritableByteChannel target)
658657
throws IOException
659658
{
660-
if (count < MAPPED_TRANSFER_THRESHOLD)
659+
if (count < TRUSTED_TRANSFER_THRESHOLD)
661660
return IOStatus.UNSUPPORTED_CASE;
662661

663662
boolean isSelChImpl = (target instanceof SelChImpl);
@@ -782,66 +781,12 @@ public long transferTo(long position, long count,
782781
return transferToArbitraryChannel(position, count, target);
783782
}
784783

785-
// Assume at first that the underlying kernel supports copy_file_range();
786-
// set this to true if we find out later that it doesn't
787-
//
788-
private static volatile boolean transferFromNotSupported;
789-
790-
private long transferFromDirectlyInternal(FileDescriptor srcFD,
791-
long position, long count)
792-
throws IOException
793-
{
794-
long n = -1;
795-
int ti = -1;
796-
try {
797-
beginBlocking();
798-
ti = threads.add();
799-
if (!isOpen())
800-
return -1;
801-
do {
802-
long comp = Blocker.begin();
803-
try {
804-
n = transferFrom0(srcFD, fd, position, count);
805-
} finally {
806-
Blocker.end(comp);
807-
}
808-
} while ((n == IOStatus.INTERRUPTED) && isOpen());
809-
if (n == IOStatus.UNSUPPORTED) {
810-
// Don't bother trying again
811-
transferFromNotSupported = true;
812-
return IOStatus.UNSUPPORTED;
813-
}
814-
return IOStatus.normalize(n);
815-
} finally {
816-
threads.remove(ti);
817-
end (n > -1);
818-
}
819-
}
820-
821-
private long transferFromDirectly(FileChannelImpl src,
822-
long position, long count)
823-
throws IOException
824-
{
825-
if (!src.readable)
826-
throw new NonReadableChannelException();
827-
if (transferFromNotSupported)
828-
return IOStatus.UNSUPPORTED;
829-
FileDescriptor srcFD = src.fd;
830-
if (srcFD == null)
831-
return IOStatus.UNSUPPORTED_CASE;
832-
833-
return transferFromDirectlyInternal(srcFD, position, count);
834-
}
835-
836784
private long transferFromFileChannel(FileChannelImpl src,
837785
long position, long count)
838786
throws IOException
839787
{
840788
if (!src.readable)
841789
throw new NonReadableChannelException();
842-
if (count < MAPPED_TRANSFER_THRESHOLD)
843-
return IOStatus.UNSUPPORTED_CASE;
844-
845790
synchronized (src.positionLock) {
846791
long pos = src.position();
847792
long max = Math.min(count, src.size() - pos);
@@ -931,10 +876,8 @@ public long transferFrom(ReadableByteChannel src,
931876
return 0;
932877

933878
if (src instanceof FileChannelImpl fci) {
934-
long n;
935-
if ((n = transferFromDirectly(fci, position, count)) >= 0)
936-
return n;
937-
if ((n = transferFromFileChannel(fci, position, count)) >= 0)
879+
long n = transferFromFileChannel(fci, position, count);
880+
if (n >= 0)
938881
return n;
939882
}
940883

@@ -1568,10 +1511,6 @@ private native long map0(int prot, long position, long length, boolean isSync)
15681511
private static native long transferTo0(FileDescriptor src, long position,
15691512
long count, FileDescriptor dst);
15701513

1571-
private static native long transferFrom0(FileDescriptor src,
1572-
FileDescriptor dst,
1573-
long position, long count);
1574-
15751514
// Retrieves the maximum size of a transfer
15761515
private static native int maxDirectTransferSize0();
15771516

‎src/java.base/unix/native/libnio/ch/FileChannelImpl.c

-43
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,6 @@
3131

3232
#if defined(__linux__)
3333
#include <sys/sendfile.h>
34-
#include <dlfcn.h>
3534
#elif defined(_AIX)
3635
#include <string.h>
3736
#include <sys/socket.h>
@@ -53,21 +52,11 @@
5352

5453
static jfieldID chan_fd; /* jobject 'fd' in sun.nio.ch.FileChannelImpl */
5554

56-
#if defined(__linux__)
57-
typedef ssize_t copy_file_range_func(int, loff_t*, int, loff_t*, size_t,
58-
unsigned int);
59-
static copy_file_range_func* my_copy_file_range_func = NULL;
60-
#endif
61-
6255
JNIEXPORT jlong JNICALL
6356
Java_sun_nio_ch_FileChannelImpl_initIDs(JNIEnv *env, jclass clazz)
6457
{
6558
jlong pageSize = sysconf(_SC_PAGESIZE);
6659
chan_fd = (*env)->GetFieldID(env, clazz, "fd", "Ljava/io/FileDescriptor;");
67-
#if defined(__linux__)
68-
my_copy_file_range_func =
69-
(copy_file_range_func*) dlsym(RTLD_DEFAULT, "copy_file_range");
70-
#endif
7160
return pageSize;
7261
}
7362

@@ -263,38 +252,6 @@ Java_sun_nio_ch_FileChannelImpl_transferTo0(JNIEnv *env, jobject this,
263252
#endif
264253
}
265254

266-
JNIEXPORT jlong JNICALL
267-
Java_sun_nio_ch_FileChannelImpl_transferFrom0(JNIEnv *env, jobject this,
268-
jobject srcFDO, jobject dstFDO,
269-
jlong position, jlong count)
270-
{
271-
#if defined(__linux__)
272-
if (my_copy_file_range_func == NULL)
273-
return IOS_UNSUPPORTED;
274-
275-
jint srcFD = fdval(env, srcFDO);
276-
jint dstFD = fdval(env, dstFDO);
277-
278-
off64_t offset = (off64_t)position;
279-
jlong n = my_copy_file_range_func(srcFD, NULL, dstFD, &offset, count, 0);
280-
if (n < 0) {
281-
if (errno == EAGAIN)
282-
return IOS_UNAVAILABLE;
283-
if ((errno == EINVAL || errno == EXDEV) && ((ssize_t)count >= 0))
284-
return IOS_UNSUPPORTED_CASE;
285-
if (errno == EINTR) {
286-
return IOS_INTERRUPTED;
287-
}
288-
JNU_ThrowIOExceptionWithLastError(env, "Transfer failed");
289-
return IOS_THROWN;
290-
}
291-
return n;
292-
#else
293-
return IOS_UNSUPPORTED;
294-
#endif
295-
}
296-
297-
298255
JNIEXPORT jint JNICALL
299256
Java_sun_nio_ch_FileChannelImpl_maxDirectTransferSize0(JNIEnv* env, jobject this)
300257
{

‎src/java.base/windows/native/libnio/ch/FileChannelImpl.c

+1-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2000, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2000, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -193,13 +193,6 @@ Java_sun_nio_ch_FileChannelImpl_transferTo0(JNIEnv *env, jobject this,
193193
return chunkSize;
194194
}
195195

196-
JNIEXPORT jlong JNICALL
197-
Java_sun_nio_ch_FileChannelImpl_transferFrom0(JNIEnv *env, jobject this,
198-
jobject srcFDO, jobject dstFDO,
199-
jlong position, jlong count)
200-
{
201-
return IOS_UNSUPPORTED;
202-
}
203196

204197
JNIEXPORT jint JNICALL
205198
Java_sun_nio_ch_FileChannelImpl_maxDirectTransferSize0(JNIEnv* env, jobject this)

‎test/jdk/java/nio/channels/FileChannel/Transfers.java

+5-11
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
/* @test
2525
* @summary Comprehensive test for FileChannel.transfer{From,To}
26-
* @bug 4708120 8274113
26+
* @bug 4708120
2727
* @author Mark Reinhold
2828
* @run main/timeout=300 Transfers
2929
*/
@@ -468,16 +468,10 @@ static void testFrom(long seed, Source src, FileChannel fc, int off, int len)
468468
int pos = (int)seed & 0xfff;
469469
fc.position(pos);
470470

471-
long position = off;
472-
long count = len;
473-
while (count > 0) {
474-
int n = (int)fc.transferFrom(src.channel(), position, count);
475-
if (n < 0 || n > count)
476-
throw new Failure("Incorrect transfer length n = : " + n
477-
+ " (expected 0 <= n <= " + len + ")");
478-
position += n;
479-
count -= n;
480-
}
471+
int n = (int)fc.transferFrom(src.channel(), off, len);
472+
if (n != len)
473+
throw new Failure("Incorrect transfer length: " + n
474+
+ " (expected " + len + ")");
481475

482476
// Check that source didn't change, and was read correctly
483477
src.verify();

0 commit comments

Comments
 (0)
Please sign in to comment.