/* * Copyright (c) 2000, 2024, 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. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * 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. */ #include #include #include #include #include #if defined(_ALLBSD_SOURCE) #define fdatasync fsync #endif #if defined(__linux__) #include #include #endif #include "jni.h" #include "nio.h" #include "nio_util.h" #include "sun_nio_ch_UnixFileDispatcherImpl.h" #include "java_lang_Integer.h" #include "java_lang_Long.h" #include #include "io_util_md.h" #if defined(_AIX) #define statvfs statvfs64 #define fstatvfs fstatvfs64 #endif JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_read0(JNIEnv *env, jclass clazz, jobject fdo, jlong address, jint len) { jint fd = fdval(env, fdo); void *buf = (void *)jlong_to_ptr(address); return convertReturnVal(env, read(fd, buf, len), JNI_TRUE); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_pread0(JNIEnv *env, jclass clazz, jobject fdo, jlong address, jint len, jlong offset) { jint fd = fdval(env, fdo); void *buf = (void *)jlong_to_ptr(address); return convertReturnVal(env, pread(fd, buf, len, offset), JNI_TRUE); } JNIEXPORT jlong JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_readv0(JNIEnv *env, jclass clazz, jobject fdo, jlong address, jint len) { jint fd = fdval(env, fdo); struct iovec *iov = (struct iovec *)jlong_to_ptr(address); return convertLongReturnVal(env, readv(fd, iov, len), JNI_TRUE); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_write0(JNIEnv *env, jclass clazz, jobject fdo, jlong address, jint len) { jint fd = fdval(env, fdo); void *buf = (void *)jlong_to_ptr(address); return convertReturnVal(env, write(fd, buf, len), JNI_FALSE); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_pwrite0(JNIEnv *env, jclass clazz, jobject fdo, jlong address, jint len, jlong offset) { jint fd = fdval(env, fdo); void *buf = (void *)jlong_to_ptr(address); return convertReturnVal(env, pwrite(fd, buf, len, offset), JNI_FALSE); } JNIEXPORT jlong JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_writev0(JNIEnv *env, jclass clazz, jobject fdo, jlong address, jint len) { jint fd = fdval(env, fdo); struct iovec *iov = (struct iovec *)jlong_to_ptr(address); return convertLongReturnVal(env, writev(fd, iov, len), JNI_FALSE); } static jlong handle(JNIEnv *env, jlong rv, char *msg) { if (rv >= 0) return rv; if (errno == EINTR) return IOS_INTERRUPTED; JNU_ThrowIOExceptionWithLastError(env, msg); return IOS_THROWN; } JNIEXPORT jlong JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_seek0(JNIEnv *env, jclass clazz, jobject fdo, jlong offset) { jint fd = fdval(env, fdo); off_t result; if (offset < 0) { result = lseek(fd, 0, SEEK_CUR); } else { result = lseek(fd, offset, SEEK_SET); } return handle(env, (jlong)result, "lseek failed"); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_force0(JNIEnv *env, jobject this, jobject fdo, jboolean md) { jint fd = fdval(env, fdo); int result = 0; if (md == JNI_FALSE) { result = fdatasync(fd); } else { result = fsync(fd); } return handle(env, result, "Force failed"); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_truncate0(JNIEnv *env, jobject this, jobject fdo, jlong size) { return handle(env, ftruncate(fdval(env, fdo), size), "Truncation failed"); } JNIEXPORT jlong JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_size0(JNIEnv *env, jobject this, jobject fdo) { jint fd = fdval(env, fdo); struct stat fbuf; if (fstat(fd, &fbuf) < 0) return handle(env, -1, "Size failed"); #if defined(__linux__) if (S_ISBLK(fbuf.st_mode)) { uint64_t size; if (ioctl(fd, BLKGETSIZE64, &size) < 0) return handle(env, -1, "Size failed"); return (jlong)size; } #endif return fbuf.st_size; } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_available0(JNIEnv *env, jobject this, jobject fdo) { jint fd = fdval(env, fdo); struct stat fbuf; jlong size = -1; if (fstat(fd, &fbuf) != -1) { int mode = fbuf.st_mode; if (S_ISCHR(mode) || S_ISFIFO(mode) || S_ISSOCK(mode)) { int n = ioctl(fd, FIONREAD, &n); if (n >= 0) { return n; } } else if (S_ISREG(mode)) { size = fbuf.st_size; } } jlong position; if ((position = lseek(fd, 0, SEEK_CUR)) == -1) { return 0; } if (size < position) { if ((size = lseek(fd, 0, SEEK_END)) == -1) return 0; else if (lseek(fd, position, SEEK_SET) == -1) return 0; } jlong available = size - position; return available > java_lang_Integer_MAX_VALUE ? java_lang_Integer_MAX_VALUE : (jint)available; } JNIEXPORT jboolean JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_isOther0(JNIEnv *env, jobject this, jobject fdo) { jint fd = fdval(env, fdo); struct stat fbuf; if (fstat(fd, &fbuf) == -1) handle(env, -1, "isOther failed"); if (S_ISREG(fbuf.st_mode) || S_ISDIR(fbuf.st_mode) || S_ISLNK(fbuf.st_mode)) return JNI_FALSE; return JNI_TRUE; } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_lock0(JNIEnv *env, jobject this, jobject fdo, jboolean block, jlong pos, jlong size, jboolean shared) { jint fd = fdval(env, fdo); jint lockResult = 0; int cmd = 0; struct flock fl; fl.l_whence = SEEK_SET; if (size == (jlong)java_lang_Long_MAX_VALUE) { fl.l_len = (off_t)0; } else { fl.l_len = (off_t)size; } fl.l_start = (off_t)pos; if (shared == JNI_TRUE) { fl.l_type = F_RDLCK; } else { fl.l_type = F_WRLCK; } if (block == JNI_TRUE) { cmd = F_SETLKW; } else { cmd = F_SETLK; } lockResult = fcntl(fd, cmd, &fl); if (lockResult < 0) { if ((cmd == F_SETLK) && (errno == EAGAIN || errno == EACCES)) return sun_nio_ch_UnixFileDispatcherImpl_NO_LOCK; if (errno == EINTR) return sun_nio_ch_UnixFileDispatcherImpl_INTERRUPTED; JNU_ThrowIOExceptionWithLastError(env, "Lock failed"); } return 0; } JNIEXPORT void JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_release0(JNIEnv *env, jobject this, jobject fdo, jlong pos, jlong size) { jint fd = fdval(env, fdo); jint lockResult = 0; struct flock fl; int cmd = F_SETLK; fl.l_whence = SEEK_SET; if (size == (jlong)java_lang_Long_MAX_VALUE) { fl.l_len = (off_t)0; } else { fl.l_len = (off_t)size; } fl.l_start = (off_t)pos; fl.l_type = F_UNLCK; lockResult = fcntl(fd, cmd, &fl); if (lockResult < 0) { JNU_ThrowIOExceptionWithLastError(env, "Release failed"); } } static void closeFileDescriptor(JNIEnv *env, int fd) { if (fd != -1) { int result = close(fd); if (result < 0) JNU_ThrowIOExceptionWithLastError(env, "Close failed"); } } JNIEXPORT void JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_closeIntFD(JNIEnv *env, jclass clazz, jint fd) { closeFileDescriptor(env, fd); } JNIEXPORT jlong JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_allocationGranularity0(JNIEnv *env, jclass klass) { jlong pageSize = sysconf(_SC_PAGESIZE); return pageSize; } JNIEXPORT jlong JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_map0(JNIEnv *env, jclass klass, jobject fdo, jint prot, jlong off, jlong len, jboolean map_sync) { void *mapAddress = 0; jint fd = fdval(env, fdo); int protections = 0; int flags = 0; // should never be called with map_sync and prot == PRIVATE assert((prot != sun_nio_ch_UnixFileDispatcherImpl_MAP_PV) || !map_sync); if (prot == sun_nio_ch_UnixFileDispatcherImpl_MAP_RO) { protections = PROT_READ; flags = MAP_SHARED; } else if (prot == sun_nio_ch_UnixFileDispatcherImpl_MAP_RW) { protections = PROT_WRITE | PROT_READ; flags = MAP_SHARED; } else if (prot == sun_nio_ch_UnixFileDispatcherImpl_MAP_PV) { protections = PROT_WRITE | PROT_READ; flags = MAP_PRIVATE; } // if MAP_SYNC and MAP_SHARED_VALIDATE are not defined then it is // best to define them here. This ensures the code compiles on old // OS releases which do not provide the relevant headers. If run // on the same machine then it will work if the kernel contains // the necessary support otherwise mmap should fail with an // invalid argument error #ifndef MAP_SYNC #define MAP_SYNC 0x80000 #endif #ifndef MAP_SHARED_VALIDATE #define MAP_SHARED_VALIDATE 0x03 #endif if (map_sync) { // ensure // 1) this is Linux on AArch64, x86_64, or PPC64 LE // 2) the mmap APIs are available at compile time #if !defined(LINUX) || ! (defined(aarch64) || (defined(amd64) && defined(_LP64)) || defined(ppc64le)) // TODO - implement for solaris/AIX/BSD/WINDOWS and for 32 bit JNU_ThrowInternalError(env, "should never call map on platform where MAP_SYNC is unimplemented"); return IOS_THROWN; #else flags |= MAP_SYNC | MAP_SHARED_VALIDATE; #endif } mapAddress = mmap( 0, /* Let OS decide location */ len, /* Number of bytes to map */ protections, /* File permissions */ flags, /* Changes are shared */ fd, /* File descriptor of mapped file */ off); /* Offset into file */ if (mapAddress == MAP_FAILED) { if (map_sync && errno == ENOTSUP) { JNU_ThrowIOExceptionWithLastError(env, "map with mode MAP_SYNC unsupported"); return IOS_THROWN; } if (errno == ENOMEM) { JNU_ThrowOutOfMemoryError(env, "Map failed"); return IOS_THROWN; } return handle(env, -1, "Map failed"); } return ((jlong) (unsigned long) mapAddress); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_unmap0(JNIEnv *env, jclass klass, jlong address, jlong len) { void *a = (void *)jlong_to_ptr(address); return handle(env, munmap(a, (size_t)len), "Unmap failed"); } JNIEXPORT jint JNICALL Java_sun_nio_ch_UnixFileDispatcherImpl_setDirect0(JNIEnv *env, jclass clazz, jobject fdo) { jint fd = fdval(env, fdo); jint result; struct statvfs file_stat; #if defined(O_DIRECT) || defined(F_NOCACHE) || defined(DIRECTIO_ON) #ifdef O_DIRECT jint orig_flag; orig_flag = fcntl(fd, F_GETFL); if (orig_flag == -1) { JNU_ThrowIOExceptionWithLastError(env, "DirectIO setup failed"); return -1; } result = fcntl(fd, F_SETFL, orig_flag | O_DIRECT); if (result == -1) { JNU_ThrowIOExceptionWithLastError(env, "DirectIO setup failed"); return result; } #elif defined(F_NOCACHE) result = fcntl(fd, F_NOCACHE, 1); if (result == -1) { JNU_ThrowIOExceptionWithLastError(env, "DirectIO setup failed"); return result; } #elif defined(DIRECTIO_ON) result = directio(fd, DIRECTIO_ON); if (result == -1) { JNU_ThrowIOExceptionWithLastError(env, "DirectIO setup failed"); return result; } #endif result = fstatvfs(fd, &file_stat); if(result == -1) { JNU_ThrowIOExceptionWithLastError(env, "DirectIO setup failed"); return result; } else { result = (int)file_stat.f_frsize; } #else result = -1; #endif return result; }