1 #ifndef FIO_OS_WINDOWS_H
\r
2 #define FIO_OS_WINDOWS_H
\r
4 #include <sys/types.h>
\r
9 #include "../smalloc.h"
\r
10 #include "../file.h"
\r
13 #define FIO_HAVE_ODIRECT
\r
14 #define FIO_HAVE_CPU_AFFINITY
\r
15 #define FIO_HAVE_CHARDEV_SIZE
\r
16 #define FIO_HAVE_FALLOCATE
\r
17 #define FIO_HAVE_FDATASYNC
\r
18 #define FIO_HAVE_WINDOWSAIO
\r
19 #define FIO_HAVE_GETTID
\r
21 #define FIO_USE_GENERIC_RAND
\r
23 #define OS_MAP_ANON MAP_ANON
\r
25 #define OS_CLOCK CLOCK_REALTIME
\r
27 #define FIO_PREFERRED_ENGINE "windowsaio"
\r
29 typedef off_t off64_t;
\r
32 LARGE_INTEGER Length;
\r
33 } GET_LENGTH_INFORMATION;
\r
35 #define IOCTL_DISK_GET_LENGTH_INFO 0x7405C
\r
37 pid_t cygwin_winpid_to_pid(int winpid);
\r
39 static inline int blockdev_size(struct fio_file *f, unsigned long long *bytes)
\r
44 if (f->hFile == NULL) {
\r
45 hFile = CreateFile(f->file_name, GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE,
\r
46 NULL, OPEN_EXISTING, 0, NULL);
\r
51 GET_LENGTH_INFORMATION info;
\r
55 if (DeviceIoControl(hFile, IOCTL_DISK_GET_LENGTH_INFO, NULL, 0, &info, sizeof(info), &outBytes, NULL))
\r
56 *bytes = info.Length.QuadPart;
\r
60 /* If we were passed a POSIX fd,
\r
61 * close the HANDLE we created via CreateFile */
\r
62 if (hFile != INVALID_HANDLE_VALUE && f->hFile == NULL)
\r
68 static inline int chardev_size(struct fio_file *f, unsigned long long *bytes)
\r
70 return blockdev_size(f, bytes);
\r
73 static inline int blockdev_invalidate_cache(struct fio_file *f)
\r
75 /* There's no way to invalidate the cache in Windows
\r
76 * so just pretend to succeed */
\r
80 static inline unsigned long long os_phys_mem(void)
\r
82 SYSTEM_INFO sysInfo;
\r
84 GetSystemInfo(&sysInfo);
\r
85 addr = (unsigned long)sysInfo.lpMaximumApplicationAddress;
\r
89 static inline void os_get_tmpdir(char *path, int len)
\r
91 GetTempPath(len, path);
\r
94 typedef DWORD_PTR os_cpu_mask_t;
\r
96 static inline int gettid(void)
\r
98 return GetCurrentThreadId();
\r
101 static inline int pid_to_winpid(int pid)
\r
104 DWORD outbytes = 0;
\r
108 allocsize = sizeof(DWORD) * 1024;
\r
111 if (allocsize == outbytes)
\r
114 ids = realloc(ids, allocsize);
\r
115 EnumProcesses(ids, allocsize, &outbytes);
\r
116 } while (allocsize == outbytes);
\r
118 for (int i = 0; i < (outbytes/sizeof(DWORD)); i++) {
\r
119 if (cygwin_winpid_to_pid(ids[i]) == pid) {
\r
129 HANDLE WINAPI OpenThread(
\r
130 DWORD dwDesiredAccess,
\r
131 BOOL bInheritHandle,
\r
134 DWORD WINAPI GetProcessIdOfThread(HANDLE Thread);
\r
136 static inline int fio_setaffinity(int pid, os_cpu_mask_t cpumask)
\r
142 h = OpenThread(THREAD_QUERY_INFORMATION | THREAD_SET_INFORMATION, TRUE, pid);
\r
144 bSuccess = SetThreadAffinityMask(h, cpumask);
\r
146 // then we might have a process id instead of a thread id
\r
147 winpid = pid_to_winpid(pid);
\r
148 h = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_SET_INFORMATION, TRUE, winpid);
\r
152 bSuccess = SetProcessAffinityMask(h, cpumask);
\r
157 return (bSuccess)? 0 : -1;
\r
160 static inline void fio_getaffinity(int pid, os_cpu_mask_t *mask)
\r
162 os_cpu_mask_t systemMask;
\r
165 HANDLE h = OpenThread(THREAD_QUERY_INFORMATION, TRUE, pid);
\r
167 winpid = GetProcessIdOfThread(h);
\r
169 winpid = pid_to_winpid(pid);
\r
171 h = OpenProcess(PROCESS_QUERY_INFORMATION, TRUE, winpid);
\r
174 GetProcessAffinityMask(h, mask, &systemMask);
\r
177 fprintf(stderr, "fio_getaffinity failed: failed to get handle for pid %d\n", pid);
\r
182 static inline void fio_cpu_clear(os_cpu_mask_t *mask, int cpu)
\r
184 *mask ^= 1 << (cpu-1);
\r
187 static inline void fio_cpu_set(os_cpu_mask_t *mask, int cpu)
\r
189 *mask |= 1 << (cpu-1);
\r
192 static inline int fio_cpuset_init(os_cpu_mask_t *mask)
\r
198 static inline int fio_cpuset_exit(os_cpu_mask_t *mask)
\r
203 #define FIO_MAX_CPUS MAXIMUM_PROCESSORS
\r
206 #define FIO_MADV_FREE MADV_FREE
\r
209 #endif /* FIO_OS_WINDOWS_H */
\r