aboutsummaryrefslogtreecommitdiff
path: root/win32/statfs.c
blob: 97b3ce679af5a3378d16b0534aeb1eaf31951de4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
#include <sys/statfs.h>
#include "libbb.h"

/*
 * Code from libguestfs (with addition of GetVolumeInformation call)
 */
int statfs(const char *file, struct statfs *buf)
{
	ULONGLONG free_bytes_available;         /* for user - similar to bavail */
	ULONGLONG total_number_of_bytes;
	ULONGLONG total_number_of_free_bytes;   /* for everyone - bfree */
	DWORD serial, namelen, flags;
	char fsname[100];
	struct mntent *mnt;
	/* Valid filesystem names don't seem to be documented.  The following
	 * are present in Wine (dlls/kernel32/volume.c). */
#define FS_NAMES "NTFS\0FAT\0FAT32\0CDFS\0UDF\0"
	int fstypes[] = {0, 0x5346544e, 0x4006, 0x4006, 0x9660, 0x15013346};

	if ( (mnt=find_mount_point(file, 0)) == NULL ) {
		return -1;
	}

	file = mnt->mnt_dir;
	if ( !GetDiskFreeSpaceEx(file, (PULARGE_INTEGER) &free_bytes_available,
			(PULARGE_INTEGER) &total_number_of_bytes,
			(PULARGE_INTEGER) &total_number_of_free_bytes) ) {
		errno = err_win_to_posix();
		return -1;
	}

	if ( !GetVolumeInformation(file, NULL, 0, &serial, &namelen, &flags,
								fsname, 100) ) {
		errno = err_win_to_posix();
		return -1;
	}

	memset(buf, 0, sizeof(*buf));

	/* XXX I couldn't determine how to get block size.  MSDN has a
	 * unhelpful hard-coded list here:
	 * http://support.microsoft.com/kb/140365
	 * but this depends on the filesystem type, the size of the disk and
	 * the version of Windows.  So this code assumes the disk is NTFS
	 * and the version of Windows is >= Win2K.
	 */
	if (total_number_of_bytes < UINT64_C(16) * 1024 * 1024 * 1024 * 1024)
		buf->f_bsize = 4096;
	else if (total_number_of_bytes < UINT64_C(32) * 1024 * 1024 * 1024 * 1024)
		buf->f_bsize = 8192;
	else if (total_number_of_bytes < UINT64_C(64) * 1024 * 1024 * 1024 * 1024)
		buf->f_bsize = 16384;
	else if (total_number_of_bytes < UINT64_C(128) * 1024 * 1024 * 1024 * 1024)
		buf->f_bsize = 32768;
	else
		buf->f_bsize = 65536;

	buf->f_type = fstypes[index_in_strings(FS_NAMES, fsname)+1];
	buf->f_frsize = buf->f_bsize;
	buf->f_blocks = total_number_of_bytes / buf->f_bsize;
	buf->f_bfree = total_number_of_free_bytes / buf->f_bsize;
	buf->f_bavail = free_bytes_available / buf->f_bsize;
	//buf->f_files = 0;
	//buf->f_ffree = 0;
	buf->f_fsid = serial;
	//buf->f_flag = 0;
	buf->f_namelen = namelen;

	return 0;
}