a3137c750f
Reviewed-by: alanb, shade
147 lines
5.3 KiB
C
147 lines
5.3 KiB
C
/*
|
|
* 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 <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
|