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 FileFlushFileBuffers(HANDLE hFile)
347 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
351 SetLastError(ERROR_INVALID_HANDLE);
356 if ((pFile->dwOpenMode & GENERIC_WRITE) == 0)
358 SetLastError(ERROR_ACCESS_DENIED);
362 if (fflush(pFile->fp) != 0)
364 SetLastError(map_posix_err(errno));
371static BOOL FileGetFileInformationByHandle(HANDLE hFile,
374 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
377 const char* lastSep = NULL;
381 if (!lpFileInformation)
384 if (fstat(fileno(pFile->fp), &st) == -1)
386 char ebuffer[256] = { 0 };
387 WLog_ERR(TAG,
"fstat failed with %s [%#08X]", errno,
388 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)));
392 lpFileInformation->dwFileAttributes = 0;
394 if (S_ISDIR(st.st_mode))
395 lpFileInformation->dwFileAttributes |= FILE_ATTRIBUTE_DIRECTORY;
397 if (lpFileInformation->dwFileAttributes == 0)
398 lpFileInformation->dwFileAttributes = FILE_ATTRIBUTE_ARCHIVE;
400 lastSep = strrchr(pFile->lpFileName,
'/');
404 const char* name = lastSep + 1;
405 const size_t namelen = strlen(name);
407 if ((namelen > 1) && (name[0] ==
'.') && (name[1] !=
'.'))
408 lpFileInformation->dwFileAttributes |= FILE_ATTRIBUTE_HIDDEN;
411 if (!(st.st_mode & S_IWUSR))
412 lpFileInformation->dwFileAttributes |= FILE_ATTRIBUTE_READONLY;
414#ifdef _DARWIN_FEATURE_64_BIT_INODE
415 ft = STAT_TIME_TO_FILETIME(st.st_birthtime);
417 ft = STAT_TIME_TO_FILETIME(st.st_ctime);
419 lpFileInformation->ftCreationTime.dwHighDateTime = (ft) >> 32ULL;
420 lpFileInformation->ftCreationTime.dwLowDateTime = ft & 0xFFFFFFFF;
421 ft = STAT_TIME_TO_FILETIME(st.st_mtime);
422 lpFileInformation->ftLastWriteTime.dwHighDateTime = (ft) >> 32ULL;
423 lpFileInformation->ftLastWriteTime.dwLowDateTime = ft & 0xFFFFFFFF;
424 ft = STAT_TIME_TO_FILETIME(st.st_atime);
425 lpFileInformation->ftLastAccessTime.dwHighDateTime = (ft) >> 32ULL;
426 lpFileInformation->ftLastAccessTime.dwLowDateTime = ft & 0xFFFFFFFF;
427 lpFileInformation->nFileSizeHigh = ((UINT64)st.st_size) >> 32ULL;
428 lpFileInformation->nFileSizeLow = st.st_size & 0xFFFFFFFF;
429 lpFileInformation->dwVolumeSerialNumber = (UINT32)st.st_dev;
430 lpFileInformation->nNumberOfLinks = (UINT32)st.st_nlink;
431 lpFileInformation->nFileIndexHigh = (st.st_ino >> 4) & 0xFFFFFFFF;
432 lpFileInformation->nFileIndexLow = st.st_ino & 0xFFFFFFFF;
436static BOOL FileLockFileEx(HANDLE hFile, DWORD dwFlags, WINPR_ATTR_UNUSED DWORD dwReserved,
437 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToLockLow,
438 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToLockHigh,
447 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
451 WLog_ERR(TAG,
"WinPR does not support the lpOverlapped parameter");
452 SetLastError(ERROR_NOT_SUPPORTED);
461 WLog_ERR(TAG,
"File %s already locked!", pFile->lpFileName);
468 lock.l_whence = SEEK_SET;
470 if (dwFlags & LOCKFILE_EXCLUSIVE_LOCK)
471 lock.l_type = F_WRLCK;
473 lock.l_type = F_WRLCK;
475 if (dwFlags & LOCKFILE_FAIL_IMMEDIATELY)
480 if (fcntl(fileno(pFile->fp), lckcmd, &lock) == -1)
482 char ebuffer[256] = { 0 };
483 WLog_ERR(TAG,
"F_SETLK failed with %s [0x%08X]",
484 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
488 if (dwFlags & LOCKFILE_EXCLUSIVE_LOCK)
493 if (dwFlags & LOCKFILE_FAIL_IMMEDIATELY)
496 if (flock(fileno(pFile->fp), lock) < 0)
498 char ebuffer[256] = { 0 };
499 WLog_ERR(TAG,
"flock failed with %s [0x%08X]",
500 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
505 pFile->bLocked = TRUE;
510static BOOL FileUnlockFile(HANDLE hFile, WINPR_ATTR_UNUSED DWORD dwFileOffsetLow,
511 WINPR_ATTR_UNUSED DWORD dwFileOffsetHigh,
512 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockLow,
513 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockHigh)
515 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
525 WLog_ERR(TAG,
"File %s is not locked!", pFile->lpFileName);
532 lock.l_whence = SEEK_SET;
533 lock.l_type = F_UNLCK;
534 if (fcntl(fileno(pFile->fp), F_GETLK, &lock) == -1)
536 char ebuffer[256] = { 0 };
537 WLog_ERR(TAG,
"F_UNLCK on %s failed with %s [0x%08X]", pFile->lpFileName,
538 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
543 if (flock(fileno(pFile->fp), LOCK_UN) < 0)
545 char ebuffer[256] = { 0 };
546 WLog_ERR(TAG,
"flock(LOCK_UN) %s failed with %s [0x%08X]", pFile->lpFileName,
547 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
555static BOOL FileUnlockFileEx(HANDLE hFile, WINPR_ATTR_UNUSED DWORD dwReserved,
556 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockLow,
557 WINPR_ATTR_UNUSED DWORD nNumberOfBytesToUnlockHigh,
560 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
567 WLog_ERR(TAG,
"WinPR does not support the lpOverlapped parameter");
568 SetLastError(ERROR_NOT_SUPPORTED);
577 WLog_ERR(TAG,
"File %s is not locked!", pFile->lpFileName);
584 lock.l_whence = SEEK_SET;
585 lock.l_type = F_UNLCK;
586 if (fcntl(fileno(pFile->fp), F_GETLK, &lock) == -1)
588 char ebuffer[256] = { 0 };
589 WLog_ERR(TAG,
"F_UNLCK on %s failed with %s [0x%08X]", pFile->lpFileName,
590 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
594 if (flock(fileno(pFile->fp), LOCK_UN) < 0)
596 char ebuffer[256] = { 0 };
597 WLog_ERR(TAG,
"flock(LOCK_UN) %s failed with %s [0x%08X]", pFile->lpFileName,
598 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
606static INT64 FileTimeToUS(
const FILETIME* ft)
608 const INT64 EPOCH_DIFF_US = EPOCH_DIFF * 1000000LL;
609 INT64 tmp = ((INT64)ft->dwHighDateTime) << 32 | ft->dwLowDateTime;
611 tmp -= EPOCH_DIFF_US;
615#if defined(_POSIX_C_SOURCE) && (_POSIX_C_SOURCE >= 200809L)
616static struct timespec filetimeToTimespec(const
FILETIME* ftime)
619 INT64 tmp = FileTimeToUS(ftime);
620 struct timespec ts = { 0 };
621 ts.tv_sec = tmp / 1000000LL;
622 ts.tv_nsec = (tmp % 1000000LL) * 1000LL;
626static BOOL FileSetFileTime(HANDLE hFile, WINPR_ATTR_UNUSED
const FILETIME* lpCreationTime,
629 struct timespec times[2] = { { UTIME_OMIT, UTIME_OMIT },
630 { UTIME_OMIT, UTIME_OMIT } };
631 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
636 if (lpLastAccessTime)
637 times[0] = filetimeToTimespec(lpLastAccessTime);
640 times[1] = filetimeToTimespec(lpLastWriteTime);
643 const int rc = futimens(fileno(pFile->fp), times);
649#elif defined(__APPLE__) || defined(ANDROID) || defined(__FreeBSD__) || defined(KFREEBSD)
650static struct timeval filetimeToTimeval(const
FILETIME* ftime)
653 UINT64 tmp = FileTimeToUS(ftime);
654 struct timeval tv = { 0 };
655 tv.tv_sec = tmp / 1000000ULL;
656 tv.tv_usec = tmp % 1000000ULL;
660static struct timeval statToTimeval(const struct stat* sval)
663 struct timeval tv = { 0 };
664#if defined(__FreeBSD__) || defined(__APPLE__) || defined(KFREEBSD)
665 tv.tv_sec = sval->st_atime;
667 TIMESPEC_TO_TIMEVAL(&tv, &sval->st_atim);
669 TIMESPEC_TO_TIMEVAL(&tv, &sval->st_atimespec);
671#elif defined(ANDROID)
672 tv.tv_sec = sval->st_atime;
673 tv.tv_usec = sval->st_atimensec / 1000UL;
678static BOOL FileSetFileTime(HANDLE hFile,
const FILETIME* lpCreationTime,
681 struct stat buf = { 0 };
683 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
688 const int rc = fstat(fileno(pFile->fp), &buf);
692 struct timeval timevals[2] = { statToTimeval(&buf), statToTimeval(&buf) };
693 if (lpLastAccessTime)
694 timevals[0] = filetimeToTimeval(lpLastAccessTime);
697 timevals[1] = filetimeToTimeval(lpLastWriteTime);
701 const int res = utimes(pFile->lpFileName, timevals);
709static BOOL FileSetFileTime(HANDLE hFile,
const FILETIME* lpCreationTime,
712 WINPR_FILE* pFile = (WINPR_FILE*)hFile;
717 WLog_WARN(TAG,
"TODO: Creation, Access and Write time can not be handled!");
719 "TODO: Define _POSIX_C_SOURCE >= 200809L or implement a platform specific handler!");
736 FileFlushFileBuffers,
739 FileSetFilePointerEx,
745 FileGetFileInformationByHandle,
769 FileGetFileInformationByHandle,
772static const char* FileGetMode(DWORD dwDesiredAccess, DWORD dwCreationDisposition, BOOL* create)
774 BOOL writeable = (dwDesiredAccess & (GENERIC_WRITE | FILE_WRITE_DATA | FILE_APPEND_DATA)) != 0;
776 switch (dwCreationDisposition)
780 return (writeable) ?
"wb+" :
"rwb";
789 return (writeable) ?
"rb+" :
"rb";
790 case TRUNCATE_EXISTING:
799UINT32 map_posix_err(
int fs_errno)
815 rc = ERROR_FILE_NOT_FOUND;
821 rc = ERROR_ACCESS_DENIED;
825 rc = ERROR_FILE_NOT_FOUND;
829 rc = ERROR_BUSY_DRIVE;
833 rc = ERROR_FILE_EXISTS;
837 rc = STATUS_FILE_IS_A_DIRECTORY;
841 rc = STATUS_DIRECTORY_NOT_EMPTY;
845 rc = STATUS_TOO_MANY_OPENED_FILES;
850 char ebuffer[256] = { 0 };
851 WLog_ERR(TAG,
"Missing ERRNO mapping %s [%d]",
852 winpr_strerror(fs_errno, ebuffer,
sizeof(ebuffer)), fs_errno);
853 rc = STATUS_UNSUCCESSFUL;
861static HANDLE FileCreateFileA(LPCSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode,
862 LPSECURITY_ATTRIBUTES lpSecurityAttributes,
863 DWORD dwCreationDisposition, DWORD dwFlagsAndAttributes,
864 HANDLE hTemplateFile)
866 WINPR_FILE* pFile = NULL;
868 const char* mode = FileGetMode(dwDesiredAccess, dwCreationDisposition, &create);
877 if (dwFlagsAndAttributes & FILE_FLAG_OVERLAPPED)
879 WLog_ERR(TAG,
"WinPR does not support the FILE_FLAG_OVERLAPPED flag");
880 SetLastError(ERROR_NOT_SUPPORTED);
881 return INVALID_HANDLE_VALUE;
884 pFile = (WINPR_FILE*)calloc(1,
sizeof(WINPR_FILE));
887 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
888 return INVALID_HANDLE_VALUE;
891 WINPR_HANDLE_SET_TYPE_AND_MODE(pFile, HANDLE_TYPE_FILE, WINPR_FD_READ);
892 pFile->common.ops = &fileOps;
894 pFile->lpFileName = _strdup(lpFileName);
895 if (!pFile->lpFileName)
897 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
899 return INVALID_HANDLE_VALUE;
902 pFile->dwOpenMode = dwDesiredAccess;
903 pFile->dwShareMode = dwShareMode;
904 pFile->dwFlagsAndAttributes = dwFlagsAndAttributes;
905 pFile->lpSecurityAttributes = lpSecurityAttributes;
906 pFile->dwCreationDisposition = dwCreationDisposition;
907 pFile->hTemplateFile = hTemplateFile;
911 if (dwCreationDisposition == CREATE_NEW)
913 if (stat(pFile->lpFileName, &st) == 0)
915 SetLastError(ERROR_FILE_EXISTS);
916 free(pFile->lpFileName);
918 return INVALID_HANDLE_VALUE;
922 fp = winpr_fopen(pFile->lpFileName,
"ab");
925 SetLastError(map_posix_err(errno));
926 free(pFile->lpFileName);
928 return INVALID_HANDLE_VALUE;
931 fp = freopen(pFile->lpFileName, mode, fp);
935 if (stat(pFile->lpFileName, &st) != 0)
937 SetLastError(map_posix_err(errno));
938 free(pFile->lpFileName);
940 return INVALID_HANDLE_VALUE;
946 if (S_ISFIFO(st.st_mode))
948 SetLastError(ERROR_FILE_NOT_FOUND);
949 free(pFile->lpFileName);
951 return INVALID_HANDLE_VALUE;
956 fp = winpr_fopen(pFile->lpFileName, mode);
963 SetLastError(map_posix_err(errno));
964 free(pFile->lpFileName);
966 return INVALID_HANDLE_VALUE;
969 (void)setvbuf(fp, NULL, _IONBF, 0);
974 lock.l_whence = SEEK_SET;
976 if (dwShareMode & FILE_SHARE_READ)
977 lock.l_type = F_RDLCK;
978 if (dwShareMode & FILE_SHARE_WRITE)
979 lock.l_type = F_RDLCK;
981 if (dwShareMode & FILE_SHARE_READ)
983 if (dwShareMode & FILE_SHARE_WRITE)
987 if (dwShareMode & (FILE_SHARE_READ | FILE_SHARE_WRITE))
990 if (fcntl(fileno(pFile->fp), F_SETLKW, &lock) == -1)
992 if (flock(fileno(pFile->fp), lock) < 0)
995 char ebuffer[256] = { 0 };
997 WLog_ERR(TAG,
"F_SETLKW failed with %s [0x%08X]",
998 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
1000 WLog_ERR(TAG,
"flock failed with %s [0x%08X]",
1001 winpr_strerror(errno, ebuffer,
sizeof(ebuffer)), errno);
1004 SetLastError(map_posix_err(errno));
1005 FileCloseHandle(pFile);
1006 return INVALID_HANDLE_VALUE;
1009 pFile->bLocked = TRUE;
1012 if (fstat(fileno(pFile->fp), &st) == 0 && dwFlagsAndAttributes & FILE_ATTRIBUTE_READONLY)
1014 st.st_mode &= WINPR_ASSERTING_INT_CAST(mode_t, (mode_t)(~(S_IWUSR | S_IWGRP | S_IWOTH)));
1015 fchmod(fileno(pFile->fp), st.st_mode);
1018 SetLastError(STATUS_SUCCESS);
1022static BOOL IsFileDevice(WINPR_ATTR_UNUSED LPCTSTR lpDeviceName)
1027static const HANDLE_CREATOR FileHandleCreator = { IsFileDevice, FileCreateFileA };
1031 return &FileHandleCreator;
1034static WINPR_FILE* FileHandle_New(FILE* fp)
1036 WINPR_FILE* pFile = NULL;
1037 char name[MAX_PATH] = { 0 };
1039 (void)_snprintf(name,
sizeof(name),
"device_%d", fileno(fp));
1040 pFile = (WINPR_FILE*)calloc(1,
sizeof(WINPR_FILE));
1043 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1047 pFile->common.ops = &shmOps;
1048 pFile->lpFileName = _strdup(name);
1050 WINPR_HANDLE_SET_TYPE_AND_MODE(pFile, HANDLE_TYPE_FILE, WINPR_FD_READ);
1054void GetStdHandle_Uninit(
void)
1056 FileCloseHandleInt(pStdHandleFile, TRUE);
1059HANDLE GetStdHandle(DWORD nStdHandle)
1065 case STD_INPUT_HANDLE:
1068 case STD_OUTPUT_HANDLE:
1071 case STD_ERROR_HANDLE:
1075 return INVALID_HANDLE_VALUE;
1077 if (!pStdHandleFile)
1078 pStdHandleFile = FileHandle_New(fp);
1080 if (!pStdHandleFile)
1081 return INVALID_HANDLE_VALUE;
1083 return (HANDLE)pStdHandleFile;
1086BOOL SetStdHandle(WINPR_ATTR_UNUSED DWORD nStdHandle, WINPR_ATTR_UNUSED HANDLE hHandle)
1091BOOL SetStdHandleEx(WINPR_ATTR_UNUSED DWORD dwStdHandle, WINPR_ATTR_UNUSED HANDLE hNewHandle,
1092 WINPR_ATTR_UNUSED HANDLE* phOldHandle)
1097BOOL GetDiskFreeSpaceA(LPCSTR lpRootPathName, LPDWORD lpSectorsPerCluster, LPDWORD lpBytesPerSector,
1098 LPDWORD lpNumberOfFreeClusters, LPDWORD lpTotalNumberOfClusters)
1101#define STATVFS statfs
1103#define STATVFS statvfs
1106 struct STATVFS svfst = { 0 };
1107 STATVFS(lpRootPathName, &svfst);
1108 *lpSectorsPerCluster = (UINT32)MIN(svfst.f_frsize, UINT32_MAX);
1109 *lpBytesPerSector = 1;
1110 *lpNumberOfFreeClusters = (UINT32)MIN(svfst.f_bavail, UINT32_MAX);
1111 *lpTotalNumberOfClusters = (UINT32)MIN(svfst.f_blocks, UINT32_MAX);
1115BOOL GetDiskFreeSpaceW(LPCWSTR lpRootPathName, LPDWORD lpSectorsPerCluster,
1116 LPDWORD lpBytesPerSector, LPDWORD lpNumberOfFreeClusters,
1117 LPDWORD lpTotalNumberOfClusters)
1120 if (!lpRootPathName)
1123 char* rootPathName = ConvertWCharToUtf8Alloc(lpRootPathName, NULL);
1126 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1129 ret = GetDiskFreeSpaceA(rootPathName, lpSectorsPerCluster, lpBytesPerSector,
1130 lpNumberOfFreeClusters, lpTotalNumberOfClusters);
1143BOOL ValidFileNameComponent(LPCWSTR lpFileName)
1149 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'C' || lpFileName[0] == L
'c')) &&
1150 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'O' || lpFileName[1] == L
'o')) &&
1151 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'N' || lpFileName[2] == L
'n')) &&
1152 (lpFileName[3] == L
'\0'))
1158 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'P' || lpFileName[0] == L
'p')) &&
1159 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'R' || lpFileName[1] == L
'r')) &&
1160 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'N' || lpFileName[2] == L
'n')) &&
1161 (lpFileName[3] == L
'\0'))
1167 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'A' || lpFileName[0] == L
'a')) &&
1168 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'U' || lpFileName[1] == L
'u')) &&
1169 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'X' || lpFileName[2] == L
'x')) &&
1170 (lpFileName[3] == L
'\0'))
1176 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'N' || lpFileName[0] == L
'n')) &&
1177 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'U' || lpFileName[1] == L
'u')) &&
1178 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'L' || lpFileName[2] == L
'l')) &&
1179 (lpFileName[3] == L
'\0'))
1185 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'L' || lpFileName[0] == L
'l')) &&
1186 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'P' || lpFileName[1] == L
'p')) &&
1187 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'T' || lpFileName[2] == L
't')) &&
1188 (lpFileName[3] != L
'\0' && (L
'0' <= lpFileName[3] && lpFileName[3] <= L
'9')) &&
1189 (lpFileName[4] == L
'\0'))
1195 if ((lpFileName[0] != L
'\0' && (lpFileName[0] == L
'C' || lpFileName[0] == L
'c')) &&
1196 (lpFileName[1] != L
'\0' && (lpFileName[1] == L
'O' || lpFileName[1] == L
'o')) &&
1197 (lpFileName[2] != L
'\0' && (lpFileName[2] == L
'M' || lpFileName[2] == L
'm')) &&
1198 (lpFileName[3] != L
'\0' && (L
'0' <= lpFileName[3] && lpFileName[3] <= L
'9')) &&
1199 (lpFileName[4] == L
'\0'))
1205 for (LPCWSTR c = lpFileName; *c; c++)
1207 if ((*c == L
'<') || (*c == L
'>') || (*c == L
':') || (*c == L
'"') || (*c == L
'/') ||
1208 (*c == L
'\\') || (*c == L
'|') || (*c == L
'?') || (*c == L
'*'))
1219HANDLE CreateFileW(LPCWSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode,
1220 LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition,
1221 DWORD dwFlagsAndAttributes, HANDLE hTemplateFile)
1224 CREATEFILE2_EXTENDED_PARAMETERS params = { 0 };
1226 params.dwSize =
sizeof(CREATEFILE2_EXTENDED_PARAMETERS);
1228 if (dwFlagsAndAttributes & FILE_FLAG_BACKUP_SEMANTICS)
1229 params.dwFileFlags |= FILE_FLAG_BACKUP_SEMANTICS;
1230 if (dwFlagsAndAttributes & FILE_FLAG_DELETE_ON_CLOSE)
1231 params.dwFileFlags |= FILE_FLAG_DELETE_ON_CLOSE;
1232 if (dwFlagsAndAttributes & FILE_FLAG_NO_BUFFERING)
1233 params.dwFileFlags |= FILE_FLAG_NO_BUFFERING;
1234 if (dwFlagsAndAttributes & FILE_FLAG_OPEN_NO_RECALL)
1235 params.dwFileFlags |= FILE_FLAG_OPEN_NO_RECALL;
1236 if (dwFlagsAndAttributes & FILE_FLAG_OPEN_REPARSE_POINT)
1237 params.dwFileFlags |= FILE_FLAG_OPEN_REPARSE_POINT;
1238 if (dwFlagsAndAttributes & FILE_FLAG_OPEN_REQUIRING_OPLOCK)
1239 params.dwFileFlags |= FILE_FLAG_OPEN_REQUIRING_OPLOCK;
1240 if (dwFlagsAndAttributes & FILE_FLAG_OVERLAPPED)
1241 params.dwFileFlags |= FILE_FLAG_OVERLAPPED;
1242 if (dwFlagsAndAttributes & FILE_FLAG_POSIX_SEMANTICS)
1243 params.dwFileFlags |= FILE_FLAG_POSIX_SEMANTICS;
1244 if (dwFlagsAndAttributes & FILE_FLAG_RANDOM_ACCESS)
1245 params.dwFileFlags |= FILE_FLAG_RANDOM_ACCESS;
1246 if (dwFlagsAndAttributes & FILE_FLAG_SESSION_AWARE)
1247 params.dwFileFlags |= FILE_FLAG_SESSION_AWARE;
1248 if (dwFlagsAndAttributes & FILE_FLAG_SEQUENTIAL_SCAN)
1249 params.dwFileFlags |= FILE_FLAG_SEQUENTIAL_SCAN;
1250 if (dwFlagsAndAttributes & FILE_FLAG_WRITE_THROUGH)
1251 params.dwFileFlags |= FILE_FLAG_WRITE_THROUGH;
1253 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_ARCHIVE)
1254 params.dwFileAttributes |= FILE_ATTRIBUTE_ARCHIVE;
1255 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_COMPRESSED)
1256 params.dwFileAttributes |= FILE_ATTRIBUTE_COMPRESSED;
1257 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_DEVICE)
1258 params.dwFileAttributes |= FILE_ATTRIBUTE_DEVICE;
1259 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_DIRECTORY)
1260 params.dwFileAttributes |= FILE_ATTRIBUTE_DIRECTORY;
1261 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_ENCRYPTED)
1262 params.dwFileAttributes |= FILE_ATTRIBUTE_ENCRYPTED;
1263 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_HIDDEN)
1264 params.dwFileAttributes |= FILE_ATTRIBUTE_HIDDEN;
1265 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_INTEGRITY_STREAM)
1266 params.dwFileAttributes |= FILE_ATTRIBUTE_INTEGRITY_STREAM;
1267 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_NORMAL)
1268 params.dwFileAttributes |= FILE_ATTRIBUTE_NORMAL;
1269 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)
1270 params.dwFileAttributes |= FILE_ATTRIBUTE_NOT_CONTENT_INDEXED;
1271 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_NO_SCRUB_DATA)
1272 params.dwFileAttributes |= FILE_ATTRIBUTE_NO_SCRUB_DATA;
1273 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_OFFLINE)
1274 params.dwFileAttributes |= FILE_ATTRIBUTE_OFFLINE;
1275 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_READONLY)
1276 params.dwFileAttributes |= FILE_ATTRIBUTE_READONLY;
1277 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_REPARSE_POINT)
1278 params.dwFileAttributes |= FILE_ATTRIBUTE_REPARSE_POINT;
1279 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_SPARSE_FILE)
1280 params.dwFileAttributes |= FILE_ATTRIBUTE_SPARSE_FILE;
1281 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_SYSTEM)
1282 params.dwFileAttributes |= FILE_ATTRIBUTE_SYSTEM;
1283 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_TEMPORARY)
1284 params.dwFileAttributes |= FILE_ATTRIBUTE_TEMPORARY;
1285 if (dwFlagsAndAttributes & FILE_ATTRIBUTE_VIRTUAL)
1286 params.dwFileAttributes |= FILE_ATTRIBUTE_VIRTUAL;
1288 if (dwFlagsAndAttributes & SECURITY_ANONYMOUS)
1289 params.dwSecurityQosFlags |= SECURITY_ANONYMOUS;
1290 if (dwFlagsAndAttributes & SECURITY_CONTEXT_TRACKING)
1291 params.dwSecurityQosFlags |= SECURITY_CONTEXT_TRACKING;
1292 if (dwFlagsAndAttributes & SECURITY_DELEGATION)
1293 params.dwSecurityQosFlags |= SECURITY_DELEGATION;
1294 if (dwFlagsAndAttributes & SECURITY_EFFECTIVE_ONLY)
1295 params.dwSecurityQosFlags |= SECURITY_EFFECTIVE_ONLY;
1296 if (dwFlagsAndAttributes & SECURITY_IDENTIFICATION)
1297 params.dwSecurityQosFlags |= SECURITY_IDENTIFICATION;
1298 if (dwFlagsAndAttributes & SECURITY_IMPERSONATION)
1299 params.dwSecurityQosFlags |= SECURITY_IMPERSONATION;
1301 params.lpSecurityAttributes = lpSecurityAttributes;
1302 params.hTemplateFile = hTemplateFile;
1304 hFile = CreateFile2(lpFileName, dwDesiredAccess, dwShareMode, dwCreationDisposition, ¶ms);
1309HANDLE CreateFileA(LPCSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode,
1310 LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition,
1311 DWORD dwFlagsAndAttributes, HANDLE hTemplateFile)
1317 WCHAR* lpFileNameW = ConvertUtf8ToWCharAlloc(lpFileName, NULL);
1322 hFile = CreateFileW(lpFileNameW, dwDesiredAccess, dwShareMode, lpSecurityAttributes,
1323 dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile);
1330DWORD WINAPI GetFileSize(HANDLE hFile, LPDWORD lpFileSizeHigh)
1335 if (!lpFileSizeHigh)
1336 return INVALID_FILE_SIZE;
1338 status = GetFileSizeEx(hFile, &fileSize);
1341 return INVALID_FILE_SIZE;
1343 *lpFileSizeHigh = fileSize.HighPart;
1345 return fileSize.LowPart;
1348DWORD SetFilePointer(HANDLE hFile, LONG lDistanceToMove, PLONG lpDistanceToMoveHigh,
1355 liDistanceToMove.LowPart = lDistanceToMove;
1357 status = SetFilePointerEx(hFile, liDistanceToMove, &liNewFilePointer, dwMoveMethod);
1360 return INVALID_SET_FILE_POINTER;
1362 if (lpDistanceToMoveHigh)
1363 *lpDistanceToMoveHigh = liNewFilePointer.HighPart;
1365 return liNewFilePointer.LowPart;
1370 return FindFirstFileExA(lpFileName, FindExInfoStandard, lpFindFileData, FindExSearchNameMatch,
1376 return FindFirstFileExW(lpFileName, FindExInfoStandard, lpFindFileData, FindExSearchNameMatch,
1380DWORD GetFullPathNameA(LPCSTR lpFileName, DWORD nBufferLength, LPSTR lpBuffer, LPSTR* lpFilePart)
1383 WCHAR* lpFileNameW = NULL;
1384 WCHAR* lpBufferW = NULL;
1385 WCHAR* lpFilePartW = NULL;
1386 DWORD nBufferLengthW = nBufferLength *
sizeof(WCHAR);
1388 if (!lpFileName || (nBufferLength < 1))
1391 lpFileNameW = ConvertUtf8ToWCharAlloc(lpFileName, NULL);
1395 lpBufferW = (WCHAR*)malloc(nBufferLengthW);
1400 dwStatus = GetFullPathNameW(lpFileNameW, nBufferLengthW, lpBufferW, &lpFilePartW);
1402 (void)ConvertWCharNToUtf8(lpBufferW, nBufferLengthW /
sizeof(WCHAR), lpBuffer, nBufferLength);
1405 lpFilePart = lpBuffer + (lpFilePartW - lpBufferW);
1410 return dwStatus * 2;
1413BOOL GetDiskFreeSpaceA(LPCSTR lpRootPathName, LPDWORD lpSectorsPerCluster, LPDWORD lpBytesPerSector,
1414 LPDWORD lpNumberOfFreeClusters, LPDWORD lpTotalNumberOfClusters)
1421 status = GetDiskFreeSpaceExA(lpRootPathName, &FreeBytesAvailableToCaller, &TotalNumberOfBytes,
1422 &TotalNumberOfFreeBytes);
1427 *lpBytesPerSector = 1;
1428 *lpSectorsPerCluster = TotalNumberOfBytes.LowPart;
1429 *lpNumberOfFreeClusters = FreeBytesAvailableToCaller.LowPart;
1430 *lpTotalNumberOfClusters = TotalNumberOfFreeBytes.LowPart;
1435BOOL GetDiskFreeSpaceW(LPCWSTR lpRootPathName, LPDWORD lpSectorsPerCluster,
1436 LPDWORD lpBytesPerSector, LPDWORD lpNumberOfFreeClusters,
1437 LPDWORD lpTotalNumberOfClusters)
1444 status = GetDiskFreeSpaceExW(lpRootPathName, &FreeBytesAvailableToCaller, &TotalNumberOfBytes,
1445 &TotalNumberOfFreeBytes);
1450 *lpBytesPerSector = 1;
1451 *lpSectorsPerCluster = TotalNumberOfBytes.LowPart;
1452 *lpNumberOfFreeClusters = FreeBytesAvailableToCaller.LowPart;
1453 *lpTotalNumberOfClusters = TotalNumberOfFreeBytes.LowPart;
1458DWORD GetLogicalDriveStringsA(DWORD nBufferLength, LPSTR lpBuffer)
1460 SetLastError(ERROR_INVALID_FUNCTION);
1464DWORD GetLogicalDriveStringsW(DWORD nBufferLength, LPWSTR lpBuffer)
1466 SetLastError(ERROR_INVALID_FUNCTION);
1470BOOL PathIsDirectoryEmptyA(LPCSTR pszPath)
1488HANDLE GetFileHandleForFileDescriptor(
int fd)
1491 return (HANDLE)_get_osfhandle(fd);
1493 WINPR_FILE* pFile = NULL;
1498 if (fcntl(fd, F_GETFD) == -1 && errno == EBADF)
1499 return INVALID_HANDLE_VALUE;
1501 flags = fcntl(fd, F_GETFL);
1503 return INVALID_HANDLE_VALUE;
1505 if (flags & O_WRONLY)
1506 fp = fdopen(fd,
"wb");
1508 fp = fdopen(fd,
"rb");
1511 return INVALID_HANDLE_VALUE;
1513 (void)setvbuf(fp, NULL, _IONBF, 0);
1516 pFile = FileHandle_New(fp);
1518 return INVALID_HANDLE_VALUE;
1520 return (HANDLE)pFile;
1524FILE* winpr_fopen(
const char* path,
const char* mode)
1527 return fopen(path, mode);
1529 LPWSTR lpPathW = NULL;
1530 LPWSTR lpModeW = NULL;
1531 FILE* result = NULL;
1536 lpPathW = ConvertUtf8ToWCharAlloc(path, NULL);
1540 lpModeW = ConvertUtf8ToWCharAlloc(mode, NULL);
1544 result = _wfopen(lpPathW, lpModeW);