[RFC] 4950302: (fs spec) Random write produces different results on Linux vs Windows from same .class
Brian Burkhalter
brian.burkhalter at oracle.com
Wed Nov 20 00:06:27 UTC 2019
This issue [1] is caused by a Linux bug documented in the pwrite(2) man page [2]:
POSIX requires that opening a file with the O_APPEND flag should have
no effect on the location at which pwrite() writes data. However, on
Linux, if a file is opened with O_APPEND, pwrite() appends data to the
end of the file, regardless of the value of offset.
One possible fix is [3] where if O_APPEND is set, it is unset to make the pwrite() call and then reset. This of course could be problematic if another thread were writing to the same file descriptor simultaneously: not all uses of IOUtil.write() use exclusion locks. But maybe this is better than the current situation? It was verified to fix the problem shown by the reproducer included in the issue description.
Thanks,
Brian
[1] https://bugs.openjdk.java.net/browse/JDK-4950302
[2] https://linux.die.net/man/2/pwrite
[3]
--- a/src/java.base/unix/native/libnio/ch/FileDispatcherImpl.c
+++ b/src/java.base/unix/native/libnio/ch/FileDispatcherImpl.c
@@ -121,6 +121,19 @@
jint fd = fdval(env, fdo);
void *buf = (void *)jlong_to_ptr(address);
+#if defined(__linux__)
+ // If file descriptor is in append mode then a positional write will
+ // actually append to the file. Temporarily suppress append mode so
+ // the write will occur at the requested position.
+ int flags = fcntl(fd, F_GETFL);
+ if (flags != -1 && flags & O_APPEND) {
+ if (fcntl(fd, F_SETFL, flags & ~O_APPEND) != -1) {
+ int result = pwrite64(fd, buf, len, offset);
+ fcntl(fd, F_SETFL, flags);
+ return convertReturnVal(env, result, JNI_FALSE);
+ }
+ }
+#endif
return convertReturnVal(env, pwrite64(fd, buf, len, offset), JNI_FALSE);
}
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.java.net/pipermail/nio-dev/attachments/20191119/85a98f80/attachment-0001.html>
More information about the nio-dev
mailing list