22#include <winpr/config.h>
23#include <winpr/debug.h>
24#include <winpr/assert.h>
26#include <winpr/wtypes.h>
28#include <winpr/file.h>
37#define TAG WINPR_TAG("file")
39#include <winpr/wlog.h>
40#include <winpr/string.h>
52#include <sys/statvfs.h>
56#define MIN(x, y) (((x) < (y)) ? (x) : (y))
59static WINPR_FILE* pStdHandleFile = NULL;
61static void GetStdHandle_Uninit(
void) __attribute__((destructor));
63static BOOL FileIsHandled(HANDLE handle)
65 return WINPR_HANDLE_IS_HANDLED(handle, HANDLE_TYPE_FILE, FALSE);
68static int FileGetFd(HANDLE handle)
70 WINPR_FILE* file = (WINPR_FILE*)handle;
72 if (!FileIsHandled(handle))
75 return fileno(file->fp);
78static BOOL FileCloseHandleInt(HANDLE handle, BOOL force)
80 WINPR_FILE* file = (WINPR_FILE*)handle;
82 if (!FileIsHandled(handle))
87 if (handle == pStdHandleFile)
96 if (fileno(file->fp) > 2)
98 (void)fclose(file->fp);
103 free(file->lpFileName);
108static BOOL FileCloseHandle(HANDLE handle)
110 return FileCloseHandleInt(handle, FALSE);
113static BOOL FileSetEndOfFile(HANDLE hFile)
115 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
120 const INT64 size = _ftelli64(pFile->fp);
124 if (ftruncate(fileno(pFile->fp), (off_t)size) < 0)
126 char ebuffer[256] = { 0 };
127 WLog_ERR(TAG,
"ftruncate %s failed with %s [0x%08X]", pFile->lpFileName,
128 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
129 SetLastError(map_posix_err(errno));
137static DWORD FileSetFilePointer(HANDLE hFile, LONG lDistanceToMove,
138 const PLONG lpDistanceToMoveHigh, DWORD dwMoveMethod)
141 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
146 return INVALID_SET_FILE_POINTER;
150 if (lpDistanceToMoveHigh)
152 offset = (INT64)(((UINT64)*lpDistanceToMoveHigh << 32U) | (UINT64)lDistanceToMove);
155 offset = lDistanceToMove;
157 switch (dwMoveMethod)
169 return INVALID_SET_FILE_POINTER;
172 if (_fseeki64(pFile->fp, offset, whence))
174 char ebuffer[256] = { 0 };
175 WLog_ERR(TAG,
"_fseeki64(%s) failed with %s [0x%08X]", pFile->lpFileName,
176 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
177 return INVALID_SET_FILE_POINTER;
180 return (DWORD)_ftelli64(pFile->fp);
183static BOOL FileSetFilePointerEx(HANDLE hFile,
LARGE_INTEGER liDistanceToMove,
184 PLARGE_INTEGER lpNewFilePointer, DWORD dwMoveMethod)
186 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
192 switch (dwMoveMethod)
207 if (_fseeki64(pFile->fp, liDistanceToMove.QuadPart, whence))
209 char ebuffer[256] = { 0 };
210 WLog_ERR(TAG,
"_fseeki64(%s) failed with %s [0x%08X]", pFile->lpFileName,
211 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
215 if (lpNewFilePointer)
216 lpNewFilePointer->QuadPart = _ftelli64(pFile->fp);
221static BOOL FileRead(PVOID Object, LPVOID lpBuffer, DWORD nNumberOfBytesToRead,
224 size_t io_status = 0;
225 WINPR_FILE* file = NULL;
230 WLog_ERR(TAG,
"WinPR does not support the lpOverlapped parameter");
231 SetLastError(ERROR_NOT_SUPPORTED);
238 file = (WINPR_FILE*)Object;
240 io_status = fread(lpBuffer, 1, nNumberOfBytesToRead, file->fp);
242 if (io_status == 0 && ferror(file->fp))
249 SetLastError(ERROR_NO_DATA);
252 SetLastError(map_posix_err(errno));
256 if (lpNumberOfBytesRead)
257 *lpNumberOfBytesRead = (DWORD)io_status;
262static BOOL FileWrite(PVOID Object, LPCVOID lpBuffer, DWORD nNumberOfBytesToWrite,
263 LPDWORD lpNumberOfBytesWritten,
LPOVERLAPPED lpOverlapped)
265 size_t io_status = 0;
266 WINPR_FILE* file = NULL;
270 WLog_ERR(TAG,
"WinPR does not support the lpOverlapped parameter");
271 SetLastError(ERROR_NOT_SUPPORTED);
278 file = (WINPR_FILE*)Object;
281 io_status = fwrite(lpBuffer, 1, nNumberOfBytesToWrite, file->fp);
282 if (io_status == 0 && ferror(file->fp))
284 SetLastError(map_posix_err(errno));
288 *lpNumberOfBytesWritten = (DWORD)io_status;
292static DWORD FileGetFileSize(HANDLE Object, LPDWORD lpFileSizeHigh)
294 WINPR_FILE* file = NULL;
301 file = (WINPR_FILE*)Object;
303 cur = _ftelli64(file->fp);
307 char ebuffer[256] = { 0 };
308 WLog_ERR(TAG,
"_ftelli64(%s) failed with %s [0x%08X]", file->lpFileName,
309 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
310 return INVALID_FILE_SIZE;
313 if (_fseeki64(file->fp, 0, SEEK_END) != 0)
315 char ebuffer[256] = { 0 };
316 WLog_ERR(TAG,
"_fseeki64(%s) failed with %s [0x%08X]", file->lpFileName,
317 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
318 return INVALID_FILE_SIZE;
321 size = _ftelli64(file->fp);
325 char ebuffer[256] = { 0 };
326 WLog_ERR(TAG,
"_ftelli64(%s) failed with %s [0x%08X]", file->lpFileName,
327 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
328 return INVALID_FILE_SIZE;
331 if (_fseeki64(file->fp, cur, SEEK_SET) != 0)
333 char ebuffer[256] = { 0 };
334 WLog_ERR(TAG,
"_ftelli64(%s) failed with %s [0x%08X]", file->lpFileName,
335 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
336 return INVALID_FILE_SIZE;
340 *lpFileSizeHigh = (UINT32)(size >> 32);
342 return (UINT32)(size & 0xFFFFFFFF);
345static BOOL FileGetFileInformationByHandle(HANDLE hFile,
348 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
351 const char* lastSep = NULL;
355 if (!lpFileInformation)
358 if (fstat(fileno(pFile->fp), &st) == -1)
360 char ebuffer[256] = { 0 };
361 WLog_ERR(TAG,
"fstat failed with %s [%#08X]", errno,
362 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)));
366 lpFileInformation->dwFileAttributes = 0;
368 if (S_ISDIR(st.st_mode))
369 lpFileInformation->dwFileAttributes |= FILE_ATTRIBUTE_DIRECTORY;
371 if (lpFileInformation->dwFileAttributes == 0)
372 lpFileInformation->dwFileAttributes = FILE_ATTRIBUTE_ARCHIVE;
374 lastSep = strrchr(pFile->lpFileName,
'/');
378 const char* name = lastSep + 1;
379 const size_t namelen = strlen(name);
381 if ((namelen > 1) && (name[0] ==
'.') && (name[1] !=
'.'))
382 lpFileInformation->dwFileAttributes |= FILE_ATTRIBUTE_HIDDEN;
385 if (!(st.st_mode & S_IWUSR))
386 lpFileInformation->dwFileAttributes |= FILE_ATTRIBUTE_READONLY;
388#ifdef _DARWIN_FEATURE_64_BIT_INODE
389 ft = STAT_TIME_TO_FILETIME(st.st_birthtime);
391 ft = STAT_TIME_TO_FILETIME(st.st_ctime);
393 lpFileInformation->ftCreationTime.dwHighDateTime = (ft) >> 32ULL;
394 lpFileInformation->ftCreationTime.dwLowDateTime = ft & 0xFFFFFFFF;
395 ft = STAT_TIME_TO_FILETIME(st.st_mtime);
396 lpFileInformation->ftLastWriteTime.dwHighDateTime = (ft) >> 32ULL;
397 lpFileInformation->ftLastWriteTime.dwLowDateTime = ft & 0xFFFFFFFF;
398 ft = STAT_TIME_TO_FILETIME(st.st_atime);
399 lpFileInformation->ftLastAccessTime.dwHighDateTime = (ft) >> 32ULL;
400 lpFileInformation->ftLastAccessTime.dwLowDateTime = ft & 0xFFFFFFFF;
401 lpFileInformation->nFileSizeHigh = ((UINT64)st.st_size) >> 32ULL;
402 lpFileInformation->nFileSizeLow = st.st_size & 0xFFFFFFFF;
403 lpFileInformation->dwVolumeSerialNumber = (UINT32)st.st_dev;
404 lpFileInformation->nNumberOfLinks = (UINT32)st.st_nlink;
405 lpFileInformation->nFileIndexHigh = (st.st_ino >> 4) & 0xFFFFFFFF;
406 lpFileInformation->nFileIndexLow = st.st_ino & 0xFFFFFFFF;
410static BOOL FileLockFileEx(HANDLE hFile, DWORD dwFlags, WINPR_ATTR_UNUSED DWORD dwReserved,
411 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToLockLow,
412 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToLockHigh,
421 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
425 WLog_ERR(TAG,
"WinPR does not support the lpOverlapped parameter");
426 SetLastError(ERROR_NOT_SUPPORTED);
435 WLog_ERR(TAG,
"File %s already locked!", pFile->lpFileName);
442 lock.l_whence = SEEK_SET;
444 if (dwFlags & LOCKFILE_EXCLUSIVE_LOCK)
445 lock.l_type = F_WRLCK;
447 lock.l_type = F_WRLCK;
449 if (dwFlags & LOCKFILE_FAIL_IMMEDIATELY)
454 if (fcntl(fileno(pFile->fp), lckcmd, &lock) == -1)
456 char ebuffer[256] = { 0 };
457 WLog_ERR(TAG,
"F_SETLK failed with %s [0x%08X]",
458 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
462 if (dwFlags & LOCKFILE_EXCLUSIVE_LOCK)
467 if (dwFlags & LOCKFILE_FAIL_IMMEDIATELY)
470 if (flock(fileno(pFile->fp), lock) < 0)
472 char ebuffer[256] = { 0 };
473 WLog_ERR(TAG,
"flock failed with %s [0x%08X]",
474 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
479 pFile->bLocked = TRUE;
484static BOOL FileUnlockFile(HANDLE hFile, WINPR_ATTR_UNUSED DWORD dwFileOffsetLow,
485 WINPR_ATTR_UNUSED DWORD dwFileOffsetHigh,
486 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockLow,
487 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockHigh)
489 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
499 WLog_ERR(TAG,
"File %s is not locked!", pFile->lpFileName);
506 lock.l_whence = SEEK_SET;
507 lock.l_type = F_UNLCK;
508 if (fcntl(fileno(pFile->fp), F_GETLK, &lock) == -1)
510 char ebuffer[256] = { 0 };
511 WLog_ERR(TAG,
"F_UNLCK on %s failed with %s [0x%08X]", pFile->lpFileName,
512 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
517 if (flock(fileno(pFile->fp), LOCK_UN) < 0)
519 char ebuffer[256] = { 0 };
520 WLog_ERR(TAG,
"flock(LOCK_UN) %s failed with %s [0x%08X]", pFile->lpFileName,
521 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
529static BOOL FileUnlockFileEx(HANDLE hFile, WINPR_ATTR_UNUSED DWORD dwReserved,
530 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockLow,
531 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockHigh,
534 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
541 WLog_ERR(TAG,
"WinPR does not support the lpOverlapped parameter");
542 SetLastError(ERROR_NOT_SUPPORTED);
551 WLog_ERR(TAG,
"File %s is not locked!", pFile->lpFileName);
558 lock.l_whence = SEEK_SET;
559 lock.l_type = F_UNLCK;
560 if (fcntl(fileno(pFile->fp), F_GETLK, &lock) == -1)
562 char ebuffer[256] = { 0 };
563 WLog_ERR(TAG,
"F_UNLCK on %s failed with %s [0x%08X]", pFile->lpFileName,
564 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
568 if (flock(fileno(pFile->fp), LOCK_UN) < 0)
570 char ebuffer[256] = { 0 };
571 WLog_ERR(TAG,
"flock(LOCK_UN) %s failed with %s [0x%08X]", pFile->lpFileName,
572 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
580static INT64 FileTimeToUS(
const FILETIME* ft)
582 const INT64 EPOCH_DIFF_US = EPOCH_DIFF * 1000000LL;
583 INT64 tmp = ((INT64)ft->dwHighDateTime) << 32 | ft->dwLowDateTime;
585 tmp -= EPOCH_DIFF_US;
589#if defined(_POSIX_C_SOURCE) && (_POSIX_C_SOURCE >= 200809L)
590static struct timespec filetimeToTimespec(const
FILETIME* ftime)
593 INT64 tmp = FileTimeToUS(ftime);
594 struct timespec ts = { 0 };
595 ts.tv_sec = tmp / 1000000LL;
596 ts.tv_nsec = (tmp % 1000000LL) * 1000LL;
600static BOOL FileSetFileTime(HANDLE hFile, WINPR_ATTR_UNUSED
const FILETIME* lpCreationTime,
603 struct timespec times[2] = { { UTIME_OMIT, UTIME_OMIT },
604 { UTIME_OMIT, UTIME_OMIT } };
605 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
610 if (lpLastAccessTime)
611 times[0] = filetimeToTimespec(lpLastAccessTime);
614 times[1] = filetimeToTimespec(lpLastWriteTime);
617 const int rc = futimens(fileno(pFile->fp), times);
623#elif defined(__APPLE__) || defined(ANDROID) || defined(__FreeBSD__) || defined(KFREEBSD)
624static struct timeval filetimeToTimeval(const
FILETIME* ftime)
627 UINT64 tmp = FileTimeToUS(ftime);
628 struct timeval tv = { 0 };
629 tv.tv_sec = tmp / 1000000ULL;
630 tv.tv_usec = tmp % 1000000ULL;
634static struct timeval statToTimeval(const struct stat* sval)
637 struct timeval tv = { 0 };
638#if defined(__FreeBSD__) || defined(__APPLE__) || defined(KFREEBSD)
639 tv.tv_sec = sval->st_atime;
641 TIMESPEC_TO_TIMEVAL(&tv, &sval->st_atim);
643 TIMESPEC_TO_TIMEVAL(&tv, &sval->st_atimespec);
645#elif defined(ANDROID)
646 tv.tv_sec = sval->st_atime;
647 tv.tv_usec = sval->st_atimensec / 1000UL;
652static BOOL FileSetFileTime(HANDLE hFile,
const FILETIME* lpCreationTime,
655 struct stat buf = { 0 };
657 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
662 const int rc = fstat(fileno(pFile->fp), &buf);
666 struct timeval timevals[2] = { statToTimeval(&buf), statToTimeval(&buf) };
667 if (lpLastAccessTime)
668 timevals[0] = filetimeToTimeval(lpLastAccessTime);
671 timevals[1] = filetimeToTimeval(lpLastWriteTime);
675 const int res = utimes(pFile->lpFileName, timevals);
683static BOOL FileSetFileTime(HANDLE hFile,
const FILETIME* lpCreationTime,
686 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
691 WLog_WARN(TAG,
"TODO: Creation, Access and Write time can not be handled!");
693 "TODO: Define _POSIX_C_SOURCE >= 200809L or implement a platform specific handler!");
713 FileSetFilePointerEx,
719 FileGetFileInformationByHandle,
743 FileGetFileInformationByHandle,
746static const char* FileGetMode(DWORD dwDesiredAccess, DWORD dwCreationDisposition, BOOL* create)
748 BOOL writeable = (dwDesiredAccess & (GENERIC_WRITE | FILE_WRITE_DATA | FILE_APPEND_DATA)) != 0;
750 switch (dwCreationDisposition)
754 return (writeable) ?
"wb+" :
"rwb";
763 return (writeable) ?
"rb+" :
"rb";
764 case TRUNCATE_EXISTING:
773UINT32 map_posix_err(
int fs_errno)
789 rc = ERROR_FILE_NOT_FOUND;
795 rc = ERROR_ACCESS_DENIED;
799 rc = ERROR_FILE_NOT_FOUND;
803 rc = ERROR_BUSY_DRIVE;
807 rc = ERROR_FILE_EXISTS;
811 rc = STATUS_FILE_IS_A_DIRECTORY;
815 rc = STATUS_DIRECTORY_NOT_EMPTY;
819 rc = STATUS_TOO_MANY_OPENED_FILES;
824 char ebuffer[256] = { 0 };
825 WLog_ERR(TAG,
"Missing ERRNO mapping %s [%d]",
826 winpr_strerror(fs_errno, ebuffer,
sizeof(ebuffer)), fs_errno);
827 rc = STATUS_UNSUCCESSFUL;
835static HANDLE FileCreateFileA(LPCSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode,
836 LPSECURITY_ATTRIBUTES lpSecurityAttributes,
837 DWORD dwCreationDisposition, DWORD dwFlagsAndAttributes,
838 HANDLE hTemplateFile)
840 WINPR_FILE* pFile = NULL;
842 const char* mode = FileGetMode(dwDesiredAccess, dwCreationDisposition, &create);
851 if (dwFlagsAndAttributes & FILE_FLAG_OVERLAPPED)
853 WLog_ERR(TAG,
"WinPR does not support the FILE_FLAG_OVERLAPPED flag");
854 SetLastError(ERROR_NOT_SUPPORTED);
855 return INVALID_HANDLE_VALUE;
858 pFile = (WINPR_FILE*)calloc(1,
sizeof(WINPR_FILE));
861 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
862 return INVALID_HANDLE_VALUE;
865 WINPR_HANDLE_SET_TYPE_AND_MODE(pFile, HANDLE_TYPE_FILE, WINPR_FD_READ);
866 pFile->common.ops = &fileOps;
868 pFile->lpFileName = _strdup(lpFileName);
869 if (!pFile->lpFileName)
871 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
873 return INVALID_HANDLE_VALUE;
876 pFile->dwOpenMode = dwDesiredAccess;
877 pFile->dwShareMode = dwShareMode;
878 pFile->dwFlagsAndAttributes = dwFlagsAndAttributes;
879 pFile->lpSecurityAttributes = lpSecurityAttributes;
880 pFile->dwCreationDisposition = dwCreationDisposition;
881 pFile->hTemplateFile = hTemplateFile;
885 if (dwCreationDisposition == CREATE_NEW)
887 if (stat(pFile->lpFileName, &st) == 0)
889 SetLastError(ERROR_FILE_EXISTS);
890 free(pFile->lpFileName);
892 return INVALID_HANDLE_VALUE;
896 fp = winpr_fopen(pFile->lpFileName,
"ab");
899 SetLastError(map_posix_err(errno));
900 free(pFile->lpFileName);
902 return INVALID_HANDLE_VALUE;
905 fp = freopen(pFile->lpFileName, mode, fp);
909 if (stat(pFile->lpFileName, &st) != 0)
911 SetLastError(map_posix_err(errno));
912 free(pFile->lpFileName);
914 return INVALID_HANDLE_VALUE;
920 if (S_ISFIFO(st.st_mode))
922 SetLastError(ERROR_FILE_NOT_FOUND);
923 free(pFile->lpFileName);
925 return INVALID_HANDLE_VALUE;
930 fp = winpr_fopen(pFile->lpFileName, mode);
937 SetLastError(map_posix_err(errno));
938 free(pFile->lpFileName);
940 return INVALID_HANDLE_VALUE;
943 (void)setvbuf(fp, NULL, _IONBF, 0);
948 lock.l_whence = SEEK_SET;
950 if (dwShareMode & FILE_SHARE_READ)
951 lock.l_type = F_RDLCK;
952 if (dwShareMode & FILE_SHARE_WRITE)
953 lock.l_type = F_RDLCK;
955 if (dwShareMode & FILE_SHARE_READ)
957 if (dwShareMode & FILE_SHARE_WRITE)
961 if (dwShareMode & (FILE_SHARE_READ | FILE_SHARE_WRITE))
964 if (fcntl(fileno(pFile->fp), F_SETLKW, &lock) == -1)
966 if (flock(fileno(pFile->fp), lock) < 0)
969 char ebuffer[256] = { 0 };
971 WLog_ERR(TAG,
"F_SETLKW failed with %s [0x%08X]",
972 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
974 WLog_ERR(TAG,
"flock failed with %s [0x%08X]",
975 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
978 SetLastError(map_posix_err(errno));
979 FileCloseHandle(pFile);
980 return INVALID_HANDLE_VALUE;
983 pFile->bLocked = TRUE;
986 if (fstat(fileno(pFile->fp), &st) == 0 && dwFlagsAndAttributes & FILE_ATTRIBUTE_READONLY)
988 st.st_mode &= WINPR_ASSERTING_INT_CAST(mode_t, (mode_t)(~(S_IWUSR | S_IWGRP | S_IWOTH)));
989 fchmod(fileno(pFile->fp), st.st_mode);
992 SetLastError(STATUS_SUCCESS);
996static BOOL IsFileDevice(WINPR_ATTR_UNUSED LPCTSTR lpDeviceName)
1001static const HANDLE_CREATOR FileHandleCreator = { IsFileDevice, FileCreateFileA };
1005 return &FileHandleCreator;
1008static WINPR_FILE* FileHandle_New(FILE* fp)
1010 WINPR_FILE* pFile = NULL;
1011 char name[MAX_PATH] = { 0 };
1013 (void)_snprintf(name,
sizeof(name),
"device_%d", fileno(fp));
1014 pFile = (WINPR_FILE*)calloc(1,
sizeof(WINPR_FILE));
1017 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1021 pFile->common.ops = &shmOps;
1022 pFile->lpFileName = _strdup(name);
1024 WINPR_HANDLE_SET_TYPE_AND_MODE(pFile, HANDLE_TYPE_FILE, WINPR_FD_READ);
1028void GetStdHandle_Uninit(
void)
1030 FileCloseHandleInt(pStdHandleFile, TRUE);
1033HANDLE GetStdHandle(DWORD nStdHandle)
1039 case STD_INPUT_HANDLE:
1042 case STD_OUTPUT_HANDLE:
1045 case STD_ERROR_HANDLE:
1049 return INVALID_HANDLE_VALUE;
1051 if (!pStdHandleFile)
1052 pStdHandleFile = FileHandle_New(fp);
1054 if (!pStdHandleFile)
1055 return INVALID_HANDLE_VALUE;
1057 return (HANDLE)pStdHandleFile;
1060BOOL SetStdHandle(WINPR_ATTR_UNUSED DWORD nStdHandle, WINPR_ATTR_UNUSED HANDLE hHandle)
1065BOOL SetStdHandleEx(WINPR_ATTR_UNUSED DWORD dwStdHandle, WINPR_ATTR_UNUSED HANDLE hNewHandle,
1066 WINPR_ATTR_UNUSED HANDLE* phOldHandle)
1071BOOL GetDiskFreeSpaceA(LPCSTR lpRootPathName, LPDWORD lpSectorsPerCluster, LPDWORD lpBytesPerSector,
1072 LPDWORD lpNumberOfFreeClusters, LPDWORD lpTotalNumberOfClusters)
1075#define STATVFS statfs
1077#define STATVFS statvfs
1080 struct STATVFS svfst = { 0 };
1081 STATVFS(lpRootPathName, &svfst);
1082 *lpSectorsPerCluster = (UINT32)MIN(svfst.f_frsize, UINT32_MAX);
1083 *lpBytesPerSector = 1;
1084 *lpNumberOfFreeClusters = (UINT32)MIN(svfst.f_bavail, UINT32_MAX);
1085 *lpTotalNumberOfClusters = (UINT32)MIN(svfst.f_blocks, UINT32_MAX);
1089BOOL GetDiskFreeSpaceW(LPCWSTR lpRootPathName, LPDWORD lpSectorsPerCluster,
1090 LPDWORD lpBytesPerSector, LPDWORD lpNumberOfFreeClusters,
1091 LPDWORD lpTotalNumberOfClusters)
1094 if (!lpRootPathName)
1097 char* rootPathName = ConvertWCharToUtf8Alloc(lpRootPathName, NULL);
1100 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1103 ret = GetDiskFreeSpaceA(rootPathName, lpSectorsPerCluster, lpBytesPerSector,
1104 lpNumberOfFreeClusters, lpTotalNumberOfClusters);
1117BOOL ValidFileNameComponent(LPCWSTR lpFileName)
1123 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'C' || lpFileName[0] == L
'c')) &&
1124 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'O' || lpFileName[1] == L
'o')) &&
1125 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'N' || lpFileName[2] == L
'n')) &&
1126 (lpFileName[3] == L
'\0'))
1132 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'P' || lpFileName[0] == L
'p')) &&
1133 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'R' || lpFileName[1] == L
'r')) &&
1134 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'N' || lpFileName[2] == L
'n')) &&
1135 (lpFileName[3] == L
'\0'))
1141 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'A' || lpFileName[0] == L
'a')) &&
1142 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'U' || lpFileName[1] == L
'u')) &&
1143 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'X' || lpFileName[2] == L
'x')) &&
1144 (lpFileName[3] == L
'\0'))
1150 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'N' || lpFileName[0] == L
'n')) &&
1151 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'U' || lpFileName[1] == L
'u')) &&
1152 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'L' || lpFileName[2] == L
'l')) &&
1153 (lpFileName[3] == L
'\0'))
1159 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'L' || lpFileName[0] == L
'l')) &&
1160 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'P' || lpFileName[1] == L
'p')) &&
1161 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'T' || lpFileName[2] == L
't')) &&
1162 (lpFileName[3] != L
'\0' && (L
'0' <= lpFileName[3] && lpFileName[3] <= L
'9')) &&
1163 (lpFileName[4] == L
'\0'))
1169 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'C' || lpFileName[0] == L
'c')) &&
1170 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'O' || lpFileName[1] == L
'o')) &&
1171 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'M' || lpFileName[2] == L
'm')) &&
1172 (lpFileName[3] != L
'\0' && (L
'0' <= lpFileName[3] && lpFileName[3] <= L
'9')) &&
1173 (lpFileName[4] == L
'\0'))
1179 for (LPCWSTR c = lpFileName; *c; c++)
1181 if ((*c == L
'<') || (*c == L
'>') || (*c == L
':') || (*c == L
'"') || (*c == L
'/') ||
1182 (*c == L
'\\') || (*c == L
'|') || (*c == L
'?') || (*c == L
'*'))
1193HANDLE CreateFileW(LPCWSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode,
1194 LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition,
1195 DWORD dwFlagsAndAttributes, HANDLE hTemplateFile)
1198 CREATEFILE2_EXTENDED_PARAMETERS params = { 0 };
1200 params.dwSize =
sizeof(CREATEFILE2_EXTENDED_PARAMETERS);
1202 if (dwFlagsAndAttributes & FILE_FLAG_BACKUP_SEMANTICS)
1203 params.dwFileFlags |= FILE_FLAG_BACKUP_SEMANTICS;
1204 if (dwFlagsAndAttributes & FILE_FLAG_DELETE_ON_CLOSE)
1205 params.dwFileFlags |= FILE_FLAG_DELETE_ON_CLOSE;
1206 if (dwFlagsAndAttributes & FILE_FLAG_NO_BUFFERING)
1207 params.dwFileFlags |= FILE_FLAG_NO_BUFFERING;
1208 if (dwFlagsAndAttributes & FILE_FLAG_OPEN_NO_RECALL)
1209 params.dwFileFlags |= FILE_FLAG_OPEN_NO_RECALL;
1210 if (dwFlagsAndAttributes & FILE_FLAG_OPEN_REPARSE_POINT)
1211 params.dwFileFlags |= FILE_FLAG_OPEN_REPARSE_POINT;
1212 if (dwFlagsAndAttributes & FILE_FLAG_OPEN_REQUIRING_OPLOCK)
1213 params.dwFileFlags |= FILE_FLAG_OPEN_REQUIRING_OPLOCK;
1214 if (dwFlagsAndAttributes & FILE_FLAG_OVERLAPPED)
1215 params.dwFileFlags |= FILE_FLAG_OVERLAPPED;
1216 if (dwFlagsAndAttributes & FILE_FLAG_POSIX_SEMANTICS)
1217 params.dwFileFlags |= FILE_FLAG_POSIX_SEMANTICS;
1218 if (dwFlagsAndAttributes & FILE_FLAG_RANDOM_ACCESS)
1219 params.dwFileFlags |= FILE_FLAG_RANDOM_ACCESS;
1220 if (dwFlagsAndAttributes & FILE_FLAG_SESSION_AWARE)
1221 params.dwFileFlags |= FILE_FLAG_SESSION_AWARE;
1222 if (dwFlagsAndAttributes & FILE_FLAG_SEQUENTIAL_SCAN)
1223 params.dwFileFlags |= FILE_FLAG_SEQUENTIAL_SCAN;
1224 if (dwFlagsAndAttributes & FILE_FLAG_WRITE_THROUGH)
1225 params.dwFileFlags |= FILE_FLAG_WRITE_THROUGH;
1227 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_ARCHIVE)
1228 params.dwFileAttributes |= FILE_ATTRIBUTE_ARCHIVE;
1229 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_COMPRESSED)
1230 params.dwFileAttributes |= FILE_ATTRIBUTE_COMPRESSED;
1231 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_DEVICE)
1232 params.dwFileAttributes |= FILE_ATTRIBUTE_DEVICE;
1233 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_DIRECTORY)
1234 params.dwFileAttributes |= FILE_ATTRIBUTE_DIRECTORY;
1235 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_ENCRYPTED)
1236 params.dwFileAttributes |= FILE_ATTRIBUTE_ENCRYPTED;
1237 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_HIDDEN)
1238 params.dwFileAttributes |= FILE_ATTRIBUTE_HIDDEN;
1239 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_INTEGRITY_STREAM)
1240 params.dwFileAttributes |= FILE_ATTRIBUTE_INTEGRITY_STREAM;
1241 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_NORMAL)
1242 params.dwFileAttributes |= FILE_ATTRIBUTE_NORMAL;
1243 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)
1244 params.dwFileAttributes |= FILE_ATTRIBUTE_NOT_CONTENT_INDEXED;
1245 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_NO_SCRUB_DATA)
1246 params.dwFileAttributes |= FILE_ATTRIBUTE_NO_SCRUB_DATA;
1247 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_OFFLINE)
1248 params.dwFileAttributes |= FILE_ATTRIBUTE_OFFLINE;
1249 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_READONLY)
1250 params.dwFileAttributes |= FILE_ATTRIBUTE_READONLY;
1251 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_REPARSE_POINT)
1252 params.dwFileAttributes |= FILE_ATTRIBUTE_REPARSE_POINT;
1253 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_SPARSE_FILE)
1254 params.dwFileAttributes |= FILE_ATTRIBUTE_SPARSE_FILE;
1255 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_SYSTEM)
1256 params.dwFileAttributes |= FILE_ATTRIBUTE_SYSTEM;
1257 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_TEMPORARY)
1258 params.dwFileAttributes |= FILE_ATTRIBUTE_TEMPORARY;
1259 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_VIRTUAL)
1260 params.dwFileAttributes |= FILE_ATTRIBUTE_VIRTUAL;
1262 if (dwFlagsAndAttributes & SECURITY_ANONYMOUS)
1263 params.dwSecurityQosFlags |= SECURITY_ANONYMOUS;
1264 if (dwFlagsAndAttributes & SECURITY_CONTEXT_TRACKING)
1265 params.dwSecurityQosFlags |= SECURITY_CONTEXT_TRACKING;
1266 if (dwFlagsAndAttributes & SECURITY_DELEGATION)
1267 params.dwSecurityQosFlags |= SECURITY_DELEGATION;
1268 if (dwFlagsAndAttributes & SECURITY_EFFECTIVE_ONLY)
1269 params.dwSecurityQosFlags |= SECURITY_EFFECTIVE_ONLY;
1270 if (dwFlagsAndAttributes & SECURITY_IDENTIFICATION)
1271 params.dwSecurityQosFlags |= SECURITY_IDENTIFICATION;
1272 if (dwFlagsAndAttributes & SECURITY_IMPERSONATION)
1273 params.dwSecurityQosFlags |= SECURITY_IMPERSONATION;
1275 params.lpSecurityAttributes = lpSecurityAttributes;
1276 params.hTemplateFile = hTemplateFile;
1278 hFile = CreateFile2(lpFileName, dwDesiredAccess, dwShareMode, dwCreationDisposition, ¶ms);
1283HANDLE CreateFileA(LPCSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode,
1284 LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition,
1285 DWORD dwFlagsAndAttributes, HANDLE hTemplateFile)
1291 WCHAR* lpFileNameW = ConvertUtf8ToWCharAlloc(lpFileName, NULL);
1296 hFile = CreateFileW(lpFileNameW, dwDesiredAccess, dwShareMode, lpSecurityAttributes,
1297 dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile);
1304DWORD WINAPI GetFileSize(HANDLE hFile, LPDWORD lpFileSizeHigh)
1309 if (!lpFileSizeHigh)
1310 return INVALID_FILE_SIZE;
1312 status = GetFileSizeEx(hFile, &fileSize);
1315 return INVALID_FILE_SIZE;
1317 *lpFileSizeHigh = fileSize.HighPart;
1319 return fileSize.LowPart;
1322DWORD SetFilePointer(HANDLE hFile, LONG lDistanceToMove, PLONG lpDistanceToMoveHigh,
1329 liDistanceToMove.LowPart = lDistanceToMove;
1331 status = SetFilePointerEx(hFile, liDistanceToMove, &liNewFilePointer, dwMoveMethod);
1334 return INVALID_SET_FILE_POINTER;
1336 if (lpDistanceToMoveHigh)
1337 *lpDistanceToMoveHigh = liNewFilePointer.HighPart;
1339 return liNewFilePointer.LowPart;
1344 return FindFirstFileExA(lpFileName, FindExInfoStandard, lpFindFileData, FindExSearchNameMatch,
1350 return FindFirstFileExW(lpFileName, FindExInfoStandard, lpFindFileData, FindExSearchNameMatch,
1354DWORD GetFullPathNameA(LPCSTR lpFileName, DWORD nBufferLength, LPSTR lpBuffer, LPSTR* lpFilePart)
1357 WCHAR* lpFileNameW = NULL;
1358 WCHAR* lpBufferW = NULL;
1359 WCHAR* lpFilePartW = NULL;
1360 DWORD nBufferLengthW = nBufferLength *
sizeof(WCHAR);
1362 if (!lpFileName || (nBufferLength < 1))
1365 lpFileNameW = ConvertUtf8ToWCharAlloc(lpFileName, NULL);
1369 lpBufferW = (WCHAR*)malloc(nBufferLengthW);
1374 dwStatus = GetFullPathNameW(lpFileNameW, nBufferLengthW, lpBufferW, &lpFilePartW);
1376 (void)ConvertWCharNToUtf8(lpBufferW, nBufferLengthW /
sizeof(WCHAR), lpBuffer, nBufferLength);
1379 lpFilePart = lpBuffer + (lpFilePartW - lpBufferW);
1384 return dwStatus * 2;
1387BOOL GetDiskFreeSpaceA(LPCSTR lpRootPathName, LPDWORD lpSectorsPerCluster, LPDWORD lpBytesPerSector,
1388 LPDWORD lpNumberOfFreeClusters, LPDWORD lpTotalNumberOfClusters)
1395 status = GetDiskFreeSpaceExA(lpRootPathName, &FreeBytesAvailableToCaller, &TotalNumberOfBytes,
1396 &TotalNumberOfFreeBytes);
1401 *lpBytesPerSector = 1;
1402 *lpSectorsPerCluster = TotalNumberOfBytes.LowPart;
1403 *lpNumberOfFreeClusters = FreeBytesAvailableToCaller.LowPart;
1404 *lpTotalNumberOfClusters = TotalNumberOfFreeBytes.LowPart;
1409BOOL GetDiskFreeSpaceW(LPCWSTR lpRootPathName, LPDWORD lpSectorsPerCluster,
1410 LPDWORD lpBytesPerSector, LPDWORD lpNumberOfFreeClusters,
1411 LPDWORD lpTotalNumberOfClusters)
1418 status = GetDiskFreeSpaceExW(lpRootPathName, &FreeBytesAvailableToCaller, &TotalNumberOfBytes,
1419 &TotalNumberOfFreeBytes);
1424 *lpBytesPerSector = 1;
1425 *lpSectorsPerCluster = TotalNumberOfBytes.LowPart;
1426 *lpNumberOfFreeClusters = FreeBytesAvailableToCaller.LowPart;
1427 *lpTotalNumberOfClusters = TotalNumberOfFreeBytes.LowPart;
1432DWORD GetLogicalDriveStringsA(DWORD nBufferLength, LPSTR lpBuffer)
1434 SetLastError(ERROR_INVALID_FUNCTION);
1438DWORD GetLogicalDriveStringsW(DWORD nBufferLength, LPWSTR lpBuffer)
1440 SetLastError(ERROR_INVALID_FUNCTION);
1444BOOL PathIsDirectoryEmptyA(LPCSTR pszPath)
1462HANDLE GetFileHandleForFileDescriptor(
int fd)
1465 return (HANDLE)_get_osfhandle(fd);
1467 WINPR_FILE* pFile = NULL;
1472 if (fcntl(fd, F_GETFD) == -1 && errno == EBADF)
1473 return INVALID_HANDLE_VALUE;
1475 flags = fcntl(fd, F_GETFL);
1477 return INVALID_HANDLE_VALUE;
1479 if (flags & O_WRONLY)
1480 fp = fdopen(fd,
"wb");
1482 fp = fdopen(fd,
"rb");
1485 return INVALID_HANDLE_VALUE;
1487 (void)setvbuf(fp, NULL, _IONBF, 0);
1490 pFile = FileHandle_New(fp);
1492 return INVALID_HANDLE_VALUE;
1494 return (HANDLE)pFile;
1498FILE* winpr_fopen(
const char* path,
const char* mode)
1501 return fopen(path, mode);
1503 LPWSTR lpPathW = NULL;
1504 LPWSTR lpModeW = NULL;
1505 FILE* result = NULL;
1510 lpPathW = ConvertUtf8ToWCharAlloc(path, NULL);
1514 lpModeW = ConvertUtf8ToWCharAlloc(mode, NULL);
1518 result = _wfopen(lpPathW, lpModeW);