8298619: java/io/File/GetXSpace.java is failing
Reviewed-by: rriggs
This commit is contained in:
parent
c59411929d
commit
3ef834fc6f
@ -79,10 +79,12 @@ ifeq ($(call isTargetOs, windows), true)
|
||||
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libTracePinnedThreads := jvm.lib
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libNewDirectByteBuffer := $(WIN_LIB_JAVA)
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libGetXSpace := $(WIN_LIB_JAVA)
|
||||
else
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libstringPlatformChars := -ljava
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libDirectIO := -ljava
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libNewDirectByteBuffer := -ljava
|
||||
BUILD_JDK_JTREG_LIBRARIES_LIBS_libGetXSpace := -ljava
|
||||
BUILD_JDK_JTREG_LIBRARIES_LDFLAGS_libNativeThread := -pthread
|
||||
|
||||
# java.lang.foreign tests
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2005, 2022, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2005, 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
|
||||
@ -29,7 +29,7 @@
|
||||
* @summary Basic functionality of File.get-X-Space methods.
|
||||
* @library .. /test/lib
|
||||
* @build jdk.test.lib.Platform
|
||||
* @run main/othervm -Djava.security.manager=allow GetXSpace
|
||||
* @run main/othervm/native -Djava.security.manager=allow GetXSpace
|
||||
*/
|
||||
|
||||
import java.io.BufferedReader;
|
||||
@ -52,13 +52,13 @@ import static java.lang.System.out;
|
||||
|
||||
@SuppressWarnings("removal")
|
||||
public class GetXSpace {
|
||||
static {
|
||||
System.loadLibrary("GetXSpace");
|
||||
}
|
||||
|
||||
private static SecurityManager [] sma = { null, new Allow(), new DenyFSA(),
|
||||
new DenyRead() };
|
||||
|
||||
// FileSystem Total Used Available Use% MountedOn
|
||||
private static final Pattern DF_PATTERN = Pattern.compile("([^\\s]+)\\s+(\\d+)\\s+(\\d+)\\s+(\\d+)\\s+\\d+%\\s+([^\\s].*)\n");
|
||||
|
||||
private static int fail = 0;
|
||||
private static int pass = 0;
|
||||
private static Throwable first;
|
||||
@ -100,34 +100,36 @@ public class GetXSpace {
|
||||
}
|
||||
|
||||
private static class Space {
|
||||
private static final long KSIZE = 1024;
|
||||
private final String name;
|
||||
private final long size;
|
||||
private final long total;
|
||||
private final long used;
|
||||
private final long free;
|
||||
private final long available;
|
||||
|
||||
Space(String name, String total, String used, String available) {
|
||||
Space(String name) {
|
||||
this.name = name;
|
||||
try {
|
||||
this.total = Long.parseLong(total) * KSIZE;
|
||||
this.used = Long.parseLong(used) * KSIZE;
|
||||
this.available = Long.parseLong(available) * KSIZE;
|
||||
} catch (NumberFormatException x) {
|
||||
throw new RuntimeException("the regex should have caught this", x);
|
||||
}
|
||||
long[] sizes = new long[4];
|
||||
if (getSpace0(name, sizes))
|
||||
System.err.println("WARNING: total space is estimated");
|
||||
this.size = sizes[0];
|
||||
this.total = sizes[1];
|
||||
this.free = sizes[2];
|
||||
this.available = sizes[3];
|
||||
}
|
||||
|
||||
String name() { return name; }
|
||||
long size() { return size; }
|
||||
long total() { return total; }
|
||||
long used() { return used; }
|
||||
long available() { return available; }
|
||||
long free() { return total - used; }
|
||||
long free() { return free; }
|
||||
|
||||
boolean woomFree(long freeSpace) {
|
||||
return ((freeSpace >= (available / 10)) &&
|
||||
(freeSpace <= (available * 10)));
|
||||
}
|
||||
|
||||
public String toString() {
|
||||
return String.format("%s (%d/%d/%d)", name, total, used, available);
|
||||
return String.format("%s (%d/%d/%d)", name, total, free, available);
|
||||
}
|
||||
}
|
||||
|
||||
@ -149,49 +151,16 @@ public class GetXSpace {
|
||||
out.println(sb);
|
||||
}
|
||||
|
||||
private static ArrayList<Space> space(String f) throws IOException {
|
||||
ArrayList<Space> al = new ArrayList<>();
|
||||
private static ArrayList<String> paths() throws IOException {
|
||||
ArrayList<String> al = new ArrayList<>();
|
||||
|
||||
String cmd = "df -k -P" + (f == null ? "" : " " + f);
|
||||
StringBuilder sb = new StringBuilder();
|
||||
Process p = Runtime.getRuntime().exec(cmd);
|
||||
try (BufferedReader in = new BufferedReader(new InputStreamReader(p.getInputStream()))) {
|
||||
String s;
|
||||
int i = 0;
|
||||
while ((s = in.readLine()) != null) {
|
||||
// skip header
|
||||
if (i++ == 0) continue;
|
||||
sb.append(s).append("\n");
|
||||
}
|
||||
}
|
||||
out.println(sb);
|
||||
|
||||
Matcher m = DF_PATTERN.matcher(sb);
|
||||
int j = 0;
|
||||
while (j < sb.length()) {
|
||||
if (m.find(j)) {
|
||||
// swap can change while this test is running
|
||||
if (!m.group(1).equals("swap")) {
|
||||
String name = f;
|
||||
if (name == null) {
|
||||
// cygwin's df lists windows path as FileSystem (1st group)
|
||||
name = Platform.isWindows() ? m.group(1) : m.group(5);
|
||||
}
|
||||
al.add(new Space(name, m.group(2), m.group(3), m.group(4)));
|
||||
}
|
||||
j = m.end();
|
||||
} else {
|
||||
throw new RuntimeException("unrecognized df output format: "
|
||||
+ "charAt(" + j + ") = '"
|
||||
+ sb.charAt(j) + "'");
|
||||
}
|
||||
File[] roots = File.listRoots();
|
||||
long[] space = new long[4];
|
||||
for (File root : roots) {
|
||||
String path = root.toString();
|
||||
al.add(path);
|
||||
}
|
||||
|
||||
if (al.size() == 0) {
|
||||
// df did not produce output
|
||||
String name = (f == null ? "" : f);
|
||||
al.add(new Space(name, "0", "0", "0"));
|
||||
}
|
||||
return al;
|
||||
}
|
||||
|
||||
@ -231,14 +200,15 @@ public class GetXSpace {
|
||||
long fs = f.getFreeSpace();
|
||||
long us = f.getUsableSpace();
|
||||
|
||||
out.format("%s:%n", s.name());
|
||||
out.format("%s (%d):%n", s.name(), s.size());
|
||||
String fmt = " %-4s total = %12d free = %12d usable = %12d%n";
|
||||
out.format(fmt, "df", s.total(), s.free(), s.available());
|
||||
out.format(fmt, "getX", ts, fs, us);
|
||||
out.format(fmt, "getSpace0", s.total(), s.free(), s.available());
|
||||
out.format(fmt, "getXSpace", ts, fs, us);
|
||||
|
||||
// If the file system can dynamically change size, this check will fail.
|
||||
// This can happen on macOS for the /dev files system.
|
||||
if (ts != s.total() && (!Platform.isOSX() || !s.name().equals("/dev"))) {
|
||||
if (ts != s.total()
|
||||
&& (!Platform.isOSX() || !s.name().equals("/dev"))) {
|
||||
long blockSize = 1;
|
||||
long numBlocks = 0;
|
||||
try {
|
||||
@ -255,32 +225,18 @@ public class GetXSpace {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
|
||||
// On macOS, the number of 1024 byte blocks might be incorrectly
|
||||
// calculated by 'df' using integer division by 2 of the number of
|
||||
// 512 byte blocks, resulting in a size smaller than the actual
|
||||
// value when the number of blocks is odd.
|
||||
if (!Platform.isOSX() || blockSize != 512 || numBlocks % 2 == 0 ||
|
||||
ts - s.total() != 512) {
|
||||
if (Platform.isWindows()) {
|
||||
//
|
||||
// In Cygwin, 'df' has been observed to account for quotas
|
||||
// when reporting the total disk size, but the total size
|
||||
// reported by GetDiskFreeSpaceExW() has been observed not
|
||||
// to account for the quota in which case the latter value
|
||||
// should be larger.
|
||||
//
|
||||
if (s.total() > ts) {
|
||||
fail(s.name() + " total space", s.total(), ">", ts);
|
||||
}
|
||||
} else {
|
||||
fail(s.name() + " total space", s.total(), "!=", ts);
|
||||
if (ts > s.total()) {
|
||||
fail(s.name() + " total space", ts, ">", s.total());
|
||||
}
|
||||
} else if (ts != s.total()) {
|
||||
fail(s.name() + " total space", ts, "!=", s.total());
|
||||
}
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
|
||||
// unix df returns statvfs.f_bavail
|
||||
// unix usable space is from statvfs.f_bavail
|
||||
long tsp = (!Platform.isWindows() ? us : fs);
|
||||
if (!s.woomFree(tsp)) {
|
||||
fail(s.name(), s.available(), "??", tsp);
|
||||
@ -288,14 +244,57 @@ public class GetXSpace {
|
||||
pass();
|
||||
}
|
||||
|
||||
if (fs > s.total()) {
|
||||
fail(s.name(), s.total(), ">", fs);
|
||||
//
|
||||
// Invariants are:
|
||||
// total space <= size
|
||||
// total space == size (Unix)
|
||||
// free space <= total space (if no quotas in effect) (Windows)
|
||||
// free space < size (if quotas in effect) (Windows)
|
||||
// usable space <= total space
|
||||
// usable space <= free space
|
||||
//
|
||||
|
||||
// total space <= size
|
||||
if (ts > s.size()) {
|
||||
fail(s.name() + " size", ts, ">", s.size());
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
|
||||
// On Unix the total space should always be the volume size
|
||||
if (Platform.isWindows()) {
|
||||
// ts != s.size() indicates that quotas are in effect
|
||||
if (ts == s.size() && fs > s.total()) {
|
||||
fail(s.name() + " free space", fs, ">", s.total());
|
||||
} else if (ts < s.size() && fs > s.size()) {
|
||||
fail(s.name() + " free space (quota)", fs, ">", s.size());
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
} else { // not Windows
|
||||
if (ts != s.size()) {
|
||||
fail(s.name() + " total space", ts, "!=", s.size());
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
}
|
||||
|
||||
// usable space <= total space
|
||||
if (us > s.total()) {
|
||||
fail(s.name(), s.total(), ">", us);
|
||||
fail(s.name() + " usable space", us, ">", s.total());
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
|
||||
// usable space <= free space
|
||||
if (us > s.free()) {
|
||||
// free and usable change dynamically
|
||||
System.err.println("Warning: us > s.free()");
|
||||
if (1.0 - Math.abs((double)s.free()/(double)us) > 0.01) {
|
||||
fail(s.name() + " usable vs. free space", us, ">", s.free());
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
} else {
|
||||
pass();
|
||||
}
|
||||
@ -391,13 +390,7 @@ public class GetXSpace {
|
||||
private static int testFile(Path dir) {
|
||||
String dirName = dir.toString();
|
||||
out.format("--- Testing %s%n", dirName);
|
||||
ArrayList<Space> l;
|
||||
try {
|
||||
l = space(dirName);
|
||||
} catch (IOException x) {
|
||||
throw new RuntimeException(dirName + " can't get file system information", x);
|
||||
}
|
||||
compare(l.get(0));
|
||||
compare(new Space(dir.getRoot().toString()));
|
||||
|
||||
if (fail != 0) {
|
||||
err.format("%d tests: %d failure(s); first: %s%n",
|
||||
@ -409,13 +402,13 @@ public class GetXSpace {
|
||||
return fail != 0 ? 1 : 0;
|
||||
}
|
||||
|
||||
private static int testDF() {
|
||||
out.println("--- Testing df");
|
||||
// Find all of the partitions on the machine and verify that the size
|
||||
// returned by "df" is equivalent to File.getXSpace() values.
|
||||
ArrayList<Space> l;
|
||||
private static int testVolumes() {
|
||||
out.println("--- Testing volumes");
|
||||
// Find all of the partitions on the machine and verify that the sizes
|
||||
// returned by File::getXSpace are equivalent to those from getSpace0
|
||||
ArrayList<String> l;
|
||||
try {
|
||||
l = space(null);
|
||||
l = paths();
|
||||
if (Platform.isWindows()) {
|
||||
diskFree();
|
||||
}
|
||||
@ -434,7 +427,8 @@ public class GetXSpace {
|
||||
|
||||
out.format("%nSecurityManager = %s%n" ,
|
||||
(sm == null ? "null" : sm.getClass().getName()));
|
||||
for (var s : l) {
|
||||
for (var p : l) {
|
||||
Space s = new Space(p);
|
||||
if (sm instanceof Deny) {
|
||||
tryCatch(s);
|
||||
} else {
|
||||
@ -472,7 +466,7 @@ public class GetXSpace {
|
||||
}
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
int failedTests = testDF();
|
||||
int failedTests = testVolumes();
|
||||
reset();
|
||||
|
||||
Path tmpDir = Files.createTempDirectory(null);
|
||||
@ -491,4 +485,13 @@ public class GetXSpace {
|
||||
throw new RuntimeException(failedTests + " test(s) failed");
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
// root the root of the volume
|
||||
// size[0] total size: number of bytes in the volume
|
||||
// size[1] total space: number of bytes visible to the caller
|
||||
// size[2] free space: number of free bytes in the volume
|
||||
// size[3] usable space: number of bytes available to the caller
|
||||
//
|
||||
private static native boolean getSpace0(String root, long[] space);
|
||||
}
|
||||
|
146
test/jdk/java/io/File/libGetXSpace.c
Normal file
146
test/jdk/java/io/File/libGetXSpace.c
Normal file
@ -0,0 +1,146 @@
|
||||
/*
|
||||
* 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.
|
||||
*/
|
||||
#include <stdlib.h>
|
||||
#include "jni.h"
|
||||
#include "jni_util.h"
|
||||
#ifdef _WIN64
|
||||
#include <windows.h>
|
||||
#include <fileapi.h>
|
||||
#include <winerror.h>
|
||||
#else
|
||||
#include <sys/errno.h>
|
||||
#include <string.h>
|
||||
#if __APPLE__
|
||||
#include <sys/param.h>
|
||||
#include <sys/mount.h>
|
||||
#else
|
||||
#include <sys/statfs.h>
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#ifdef _WIN64
|
||||
jboolean initialized = JNI_FALSE;
|
||||
BOOL(WINAPI * pfnGetDiskSpaceInformation)(LPCWSTR, LPVOID) = NULL;
|
||||
#endif
|
||||
|
||||
//
|
||||
// root the root of the volume
|
||||
// sizes[0] total size: number of bytes in the volume
|
||||
// sizes[1] total space: number of bytes visible to the caller
|
||||
// sizes[2] free space: number of free bytes in the volume
|
||||
// sizes[3] usable space: number of bytes available to the caller
|
||||
//
|
||||
JNIEXPORT jboolean JNICALL
|
||||
Java_GetXSpace_getSpace0
|
||||
(JNIEnv *env, jclass cls, jstring root, jlongArray sizes)
|
||||
{
|
||||
jboolean totalSpaceIsEstimated = JNI_FALSE;
|
||||
jlong array[4];
|
||||
const jchar* chars = (*env)->GetStringChars(env, root, NULL);
|
||||
if (chars == NULL) {
|
||||
JNU_ThrowByNameWithLastError(env, "java/lang/RuntimeException",
|
||||
"GetStringChars");
|
||||
return JNI_FALSE;
|
||||
}
|
||||
|
||||
#ifdef _WIN64
|
||||
if (initialized == JNI_FALSE) {
|
||||
initialized = JNI_TRUE;
|
||||
HMODULE hmod = GetModuleHandleW(L"kernel32");
|
||||
if (hmod != NULL) {
|
||||
*(FARPROC*)&pfnGetDiskSpaceInformation =
|
||||
GetProcAddress(hmod, "GetDiskSpaceInformationW");
|
||||
}
|
||||
}
|
||||
|
||||
LPCWSTR path = (LPCWSTR)chars;
|
||||
|
||||
if (pfnGetDiskSpaceInformation != NULL) {
|
||||
// use GetDiskSpaceInformationW
|
||||
DISK_SPACE_INFORMATION diskSpaceInfo;
|
||||
BOOL hres = pfnGetDiskSpaceInformation(path, &diskSpaceInfo);
|
||||
if (FAILED(hres)) {
|
||||
JNU_ThrowByNameWithLastError(env, "java/lang/RuntimeException",
|
||||
"GetDiskSpaceInformationW");
|
||||
return totalSpaceIsEstimated;
|
||||
}
|
||||
|
||||
ULONGLONG bytesPerAllocationUnit =
|
||||
diskSpaceInfo.SectorsPerAllocationUnit*diskSpaceInfo.BytesPerSector;
|
||||
array[0] = (jlong)(diskSpaceInfo.ActualTotalAllocationUnits*
|
||||
bytesPerAllocationUnit);
|
||||
array[1] = (jlong)(diskSpaceInfo.CallerTotalAllocationUnits*
|
||||
bytesPerAllocationUnit);
|
||||
array[2] = (jlong)(diskSpaceInfo.ActualAvailableAllocationUnits*
|
||||
bytesPerAllocationUnit);
|
||||
array[3] = (jlong)(diskSpaceInfo.CallerAvailableAllocationUnits*
|
||||
bytesPerAllocationUnit);
|
||||
} else {
|
||||
totalSpaceIsEstimated = JNI_TRUE;
|
||||
|
||||
// if GetDiskSpaceInformationW is unavailable ("The specified
|
||||
// procedure could not be found"), fall back to GetDiskFreeSpaceExW
|
||||
ULARGE_INTEGER freeBytesAvailable;
|
||||
ULARGE_INTEGER totalNumberOfBytes;
|
||||
ULARGE_INTEGER totalNumberOfFreeBytes;
|
||||
|
||||
if (GetDiskFreeSpaceExW(path, &freeBytesAvailable, &totalNumberOfBytes,
|
||||
&totalNumberOfFreeBytes) == 0) {
|
||||
JNU_ThrowByNameWithLastError(env, "java/lang/RuntimeException",
|
||||
"GetDiskFreeSpaceExW");
|
||||
return totalSpaceIsEstimated;
|
||||
}
|
||||
|
||||
// If quotas are in effect, it is impossible to obtain the volume size,
|
||||
// so estimate it as free + used = free + (visible - available)
|
||||
ULONGLONG used = totalNumberOfBytes.QuadPart - freeBytesAvailable.QuadPart;
|
||||
array[0] = (jlong)(totalNumberOfFreeBytes.QuadPart + used);
|
||||
array[1] = (jlong)totalNumberOfBytes.QuadPart;
|
||||
array[2] = (jlong)totalNumberOfFreeBytes.QuadPart;
|
||||
array[3] = (jlong)freeBytesAvailable.QuadPart;
|
||||
}
|
||||
#else
|
||||
struct statfs buf;
|
||||
int result = statfs((const char*)chars, &buf);
|
||||
(*env)->ReleaseStringChars(env, root, chars);
|
||||
if (result < 0) {
|
||||
JNU_ThrowByNameWithLastError(env, "java/lang/RuntimeException",
|
||||
strerror(errno));
|
||||
return totalSpaceIsEstimated;
|
||||
}
|
||||
|
||||
array[0] = (jlong)(buf.f_blocks*buf.f_bsize);
|
||||
array[1] = array[0]; // number visible is the same as the total size
|
||||
array[2] = (jlong)(buf.f_bfree*buf.f_bsize);
|
||||
array[3] = (jlong)(buf.f_bavail*buf.f_bsize);
|
||||
#endif
|
||||
(*env)->SetLongArrayRegion(env, sizes, 0, 4, array);
|
||||
return totalSpaceIsEstimated;
|
||||
}
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
Loading…
Reference in New Issue
Block a user