mirror of
				https://github.com/winfsp/winfsp.git
				synced 2025-10-30 19:48:38 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			657 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			657 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <winfsp/winfsp.h>
 | |
| #include <tlib/testsuite.h>
 | |
| #include <sddl.h>
 | |
| #include <strsafe.h>
 | |
| #include "memfs.h"
 | |
| 
 | |
| void *memfs_start_ex(ULONG Flags, ULONG FileInfoTimeout);
 | |
| void *memfs_start(ULONG Flags);
 | |
| void memfs_stop(void *data);
 | |
| PWSTR memfs_volumename(void *data);
 | |
| 
 | |
| extern int NtfsTests;
 | |
| extern int WinFspDiskTests;
 | |
| extern int WinFspNetTests;
 | |
| 
 | |
| void getfileinfo_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     HANDLE Handle;
 | |
|     BOOL Success;
 | |
|     WCHAR FilePath[MAX_PATH];
 | |
|     FILE_ATTRIBUTE_TAG_INFO AttributeTagInfo;
 | |
|     FILE_BASIC_INFO BasicInfo;
 | |
|     FILE_STANDARD_INFO StandardInfo;
 | |
|     PUINT8 NameInfoBuf[sizeof(FILE_NAME_INFO) + MAX_PATH];
 | |
|     PFILE_NAME_INFO PNameInfo = (PVOID)NameInfoBuf;
 | |
|     BY_HANDLE_FILE_INFORMATION FileInfo;
 | |
|     FILETIME FileTime;
 | |
|     LONGLONG TimeLo, TimeHi;
 | |
| 
 | |
|     GetSystemTimeAsFileTime(&FileTime);
 | |
|     TimeLo = ((PLARGE_INTEGER)&FileTime)->QuadPart;
 | |
|     TimeHi = TimeLo + 10000 * 10000/* 10 seconds */;
 | |
| 
 | |
|     StringCbPrintfW(FilePath, sizeof FilePath, L"%s%s\\file0",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Handle = CreateFileW(FilePath,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
| 
 | |
|     Success = GetFileInformationByHandleEx(Handle, FileAttributeTagInfo, &AttributeTagInfo, sizeof AttributeTagInfo);
 | |
|     ASSERT(Success);
 | |
|     //ASSERT(FILE_ATTRIBUTE_ARCHIVE == AttributeTagInfo.FileAttributes);
 | |
|     ASSERT(0 == AttributeTagInfo.ReparseTag);
 | |
| 
 | |
|     Success = GetFileInformationByHandleEx(Handle, FileBasicInfo, &BasicInfo, sizeof BasicInfo);
 | |
|     ASSERT(Success);
 | |
|     //ASSERT(FILE_ATTRIBUTE_ARCHIVE == BasicInfo.FileAttributes);
 | |
|     if (-1 != Flags)
 | |
|         ASSERT(
 | |
|             TimeLo <= BasicInfo.CreationTime.QuadPart &&
 | |
|             TimeHi >  BasicInfo.CreationTime.QuadPart);
 | |
|     ASSERT(
 | |
|         TimeLo <= BasicInfo.LastAccessTime.QuadPart &&
 | |
|         TimeHi >  BasicInfo.LastAccessTime.QuadPart);
 | |
|     ASSERT(
 | |
|         TimeLo <= BasicInfo.LastWriteTime.QuadPart &&
 | |
|         TimeHi >  BasicInfo.LastWriteTime.QuadPart);
 | |
|     ASSERT(
 | |
|         TimeLo <= BasicInfo.ChangeTime.QuadPart &&
 | |
|         TimeHi >  BasicInfo.ChangeTime.QuadPart);
 | |
| 
 | |
|     Success = GetFileInformationByHandleEx(Handle, FileStandardInfo, &StandardInfo, sizeof StandardInfo);
 | |
|     ASSERT(Success);
 | |
|     ASSERT(0 == StandardInfo.AllocationSize.QuadPart);
 | |
|     ASSERT(0 == StandardInfo.EndOfFile.QuadPart);
 | |
|     ASSERT(1 == StandardInfo.NumberOfLinks);
 | |
|     ASSERT(!StandardInfo.DeletePending);
 | |
|     ASSERT(!StandardInfo.Directory);
 | |
| 
 | |
|     Success = GetFileInformationByHandleEx(Handle, FileNameInfo, PNameInfo, sizeof *PNameInfo);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_MORE_DATA == GetLastError());
 | |
|     if (-1 == Flags)
 | |
|         ASSERT(PNameInfo->FileNameLength == wcslen(FilePath + 6) * sizeof(WCHAR));
 | |
|     else if (0 == Prefix)
 | |
|         ASSERT(PNameInfo->FileNameLength == wcslen(L"\\file0") * sizeof(WCHAR));
 | |
|     else
 | |
|         ASSERT(PNameInfo->FileNameLength == wcslen(FilePath + 1) * sizeof(WCHAR));
 | |
|     ASSERT(L'\\' == PNameInfo->FileName[0]);
 | |
| 
 | |
|     Success = GetFileInformationByHandleEx(Handle, FileNameInfo, PNameInfo, sizeof NameInfoBuf);
 | |
|     ASSERT(Success);
 | |
|     if (-1 == Flags)
 | |
|         ASSERT(PNameInfo->FileNameLength == wcslen(FilePath + 6) * sizeof(WCHAR));
 | |
|     else if (0 == Prefix)
 | |
|         ASSERT(PNameInfo->FileNameLength == wcslen(L"\\file0") * sizeof(WCHAR));
 | |
|     else
 | |
|         ASSERT(PNameInfo->FileNameLength == wcslen(FilePath + 1) * sizeof(WCHAR));
 | |
|     if (-1 == Flags)
 | |
|         ASSERT(0 == memcmp(FilePath + 6, PNameInfo->FileName, PNameInfo->FileNameLength));
 | |
|     else if (0 == Prefix)
 | |
|         ASSERT(0 == memcmp(L"\\file0", PNameInfo->FileName, PNameInfo->FileNameLength));
 | |
|     else
 | |
|         ASSERT(0 == memcmp(FilePath + 1, PNameInfo->FileName, PNameInfo->FileNameLength));
 | |
| 
 | |
|     Success = GetFileInformationByHandle(Handle, &FileInfo);
 | |
|     ASSERT(Success);
 | |
|     //ASSERT(FILE_ATTRIBUTE_ARCHIVE == FileInfo.dwFileAttributes);
 | |
|     if (-1 != Flags)
 | |
|         ASSERT(
 | |
|             TimeLo <= ((PLARGE_INTEGER)&FileInfo.ftCreationTime)->QuadPart &&
 | |
|             TimeHi >  ((PLARGE_INTEGER)&FileInfo.ftCreationTime)->QuadPart);
 | |
|     ASSERT(
 | |
|         TimeLo <= ((PLARGE_INTEGER)&FileInfo.ftLastAccessTime)->QuadPart &&
 | |
|         TimeHi >  ((PLARGE_INTEGER)&FileInfo.ftLastAccessTime)->QuadPart);
 | |
|     ASSERT(
 | |
|         TimeLo <= ((PLARGE_INTEGER)&FileInfo.ftLastWriteTime)->QuadPart &&
 | |
|         TimeHi >  ((PLARGE_INTEGER)&FileInfo.ftLastWriteTime)->QuadPart);
 | |
|     ASSERT(0 == FileInfo.nFileSizeLow && 0 == FileInfo.nFileSizeHigh);
 | |
|     ASSERT(1 == FileInfo.nNumberOfLinks);
 | |
| 
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void getfileinfo_test(void)
 | |
| {
 | |
|     if (NtfsTests)
 | |
|     {
 | |
|         WCHAR DirBuf[MAX_PATH] = L"\\\\?\\";
 | |
|         GetCurrentDirectoryW(MAX_PATH - 4, DirBuf + 4);
 | |
|         getfileinfo_dotest(-1, DirBuf, 0);
 | |
|     }
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         getfileinfo_dotest(MemfsDisk, 0, 0);
 | |
|         getfileinfo_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         getfileinfo_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         getfileinfo_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void setfileinfo_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     HANDLE Handle;
 | |
|     BOOL Success;
 | |
|     WCHAR FilePath[MAX_PATH];
 | |
|     BY_HANDLE_FILE_INFORMATION FileInfo0, FileInfo;
 | |
|     FILETIME FileTime;
 | |
|     DWORD Offset;
 | |
| 
 | |
|     StringCbPrintfW(FilePath, sizeof FilePath, L"%s%s\\file0",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Handle = CreateFileW(FilePath,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
| 
 | |
|     Success = GetFileInformationByHandle(Handle, &FileInfo0);
 | |
|     ASSERT(Success);
 | |
|     //ASSERT(FILE_ATTRIBUTE_ARCHIVE == FileInfo0.dwFileAttributes);
 | |
| 
 | |
|     Success = SetFileAttributesW(FilePath, FILE_ATTRIBUTE_DIRECTORY | FILE_ATTRIBUTE_HIDDEN);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = GetFileInformationByHandle(Handle, &FileInfo);
 | |
|     ASSERT(Success);
 | |
|     ASSERT(FILE_ATTRIBUTE_HIDDEN == FileInfo.dwFileAttributes);
 | |
| 
 | |
|     *(PUINT64)&FileTime = 0x4200000042ULL;
 | |
|     Success = SetFileTime(Handle, 0, &FileTime, &FileTime);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = GetFileInformationByHandle(Handle, &FileInfo);
 | |
|     ASSERT(Success);
 | |
|     ASSERT(*(PUINT64)&FileInfo0.ftCreationTime == *(PUINT64)&FileInfo.ftCreationTime);
 | |
|     ASSERT(0x4200000042ULL == *(PUINT64)&FileInfo.ftLastAccessTime);
 | |
|     ASSERT(0x4200000042ULL == *(PUINT64)&FileInfo.ftLastWriteTime);
 | |
| 
 | |
|     Success = SetFileTime(Handle, &FileTime, 0, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = GetFileInformationByHandle(Handle, &FileInfo);
 | |
|     ASSERT(Success);
 | |
|     ASSERT(0x4200000042ULL == *(PUINT64)&FileInfo.ftCreationTime);
 | |
| 
 | |
|     Offset = SetFilePointer(Handle, 42, 0, 0);
 | |
|     ASSERT(42 == Offset);
 | |
| 
 | |
|     Success = SetEndOfFile(Handle);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = GetFileInformationByHandle(Handle, &FileInfo);
 | |
|     ASSERT(Success);
 | |
|     ASSERT(42 == FileInfo.nFileSizeLow);
 | |
|     ASSERT(0 == FileInfo.nFileSizeHigh);
 | |
| 
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void setfileinfo_test(void)
 | |
| {
 | |
|     if (NtfsTests)
 | |
|     {
 | |
|         WCHAR DirBuf[MAX_PATH] = L"\\\\?\\";
 | |
|         GetCurrentDirectoryW(MAX_PATH - 4, DirBuf + 4);
 | |
|         setfileinfo_dotest(-1, DirBuf, 0);
 | |
|     }
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         setfileinfo_dotest(MemfsDisk, 0, 0);
 | |
|         setfileinfo_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         setfileinfo_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         setfileinfo_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void delete_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     HANDLE Handle;
 | |
|     BOOL Success;
 | |
|     WCHAR Dir1Path[MAX_PATH];
 | |
|     WCHAR FilePath[MAX_PATH];
 | |
| 
 | |
|     StringCbPrintfW(Dir1Path, sizeof Dir1Path, L"%s%s\\dir1",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     StringCbPrintfW(FilePath, sizeof FilePath, L"%s%s\\dir1\\file0",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Success = CreateDirectoryW(Dir1Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Handle = CreateFileW(FilePath,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     Success = RemoveDirectoryW(Dir1Path);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_DIR_NOT_EMPTY == GetLastError());
 | |
| 
 | |
|     Success = DeleteFileW(FilePath);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = DeleteFileW(FilePath);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_FILE_NOT_FOUND == GetLastError());
 | |
| 
 | |
|     Success = RemoveDirectoryW(Dir1Path);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = RemoveDirectoryW(Dir1Path);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_FILE_NOT_FOUND == GetLastError());
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void delete_test(void)
 | |
| {
 | |
|     if (NtfsTests)
 | |
|     {
 | |
|         WCHAR DirBuf[MAX_PATH] = L"\\\\?\\";
 | |
|         GetCurrentDirectoryW(MAX_PATH - 4, DirBuf + 4);
 | |
|         delete_dotest(-1, DirBuf, 0);
 | |
|     }
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         delete_dotest(MemfsDisk, 0, 0);
 | |
|         delete_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         delete_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         delete_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void delete_access_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     HANDLE Handle;
 | |
|     BOOL Success;
 | |
|     WCHAR FilePath[MAX_PATH];
 | |
| 
 | |
|     StringCbPrintfW(FilePath, sizeof FilePath, L"%s%s\\file0",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Handle = CreateFileW(FilePath,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_READONLY, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     Success = DeleteFileW(FilePath);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_ACCESS_DENIED == GetLastError());
 | |
| 
 | |
|     Success = SetFileAttributesW(FilePath, FILE_ATTRIBUTE_NORMAL);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = DeleteFileW(FilePath);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     /* enable this test when we have proper FILE_DELETE_CHILD support on the parent directory! */
 | |
| #if 0
 | |
|     static PWSTR Sddl0 = L"D:P(D;;GA;;;SY)(D;;GA;;;BA)(D;;GA;;;WD)";
 | |
|     static PWSTR Sddl1 = L"D:P(A;;GA;;;SY)(A;;GA;;;BA)(A;;GA;;;WD)";
 | |
|     PSECURITY_DESCRIPTOR SecurityDescriptor;
 | |
|     SECURITY_ATTRIBUTES SecurityAttributes = { 0 };
 | |
| 
 | |
|     Success = ConvertStringSecurityDescriptorToSecurityDescriptorW(Sddl0, SDDL_REVISION_1, &SecurityDescriptor, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     SecurityAttributes.nLength = sizeof SecurityAttributes;
 | |
|     SecurityAttributes.lpSecurityDescriptor = SecurityDescriptor;
 | |
| 
 | |
|     Handle = CreateFileW(FilePath,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, &SecurityAttributes,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     LocalFree(SecurityDescriptor);
 | |
| 
 | |
|     Success = DeleteFileW(FilePath);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_ACCESS_DENIED == GetLastError());
 | |
| 
 | |
|     Success = ConvertStringSecurityDescriptorToSecurityDescriptorW(Sddl1, SDDL_REVISION_1, &SecurityDescriptor, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Handle = CreateFileW(FilePath,
 | |
|         GENERIC_READ | GENERIC_WRITE | WRITE_DAC, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         OPEN_EXISTING, 0, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
|     Success = SetKernelObjectSecurity(Handle, DACL_SECURITY_INFORMATION, SecurityDescriptor);
 | |
|     ASSERT(Success);
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     LocalFree(SecurityDescriptor);
 | |
| 
 | |
|     Success = DeleteFileW(FilePath);
 | |
|     ASSERT(Success);
 | |
| #endif
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void delete_access_test(void)
 | |
| {
 | |
|     if (NtfsTests)
 | |
|     {
 | |
|         WCHAR DirBuf[MAX_PATH] = L"\\\\?\\";
 | |
|         GetCurrentDirectoryW(MAX_PATH - 4, DirBuf + 4);
 | |
|         delete_access_dotest(-1, DirBuf, 0);
 | |
|     }
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         delete_access_dotest(MemfsDisk, 0, 0);
 | |
|         delete_access_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         delete_access_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         delete_access_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void rename_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     HANDLE Handle;
 | |
|     BOOL Success;
 | |
|     WCHAR Dir1Path[MAX_PATH];
 | |
|     WCHAR Dir2Path[MAX_PATH];
 | |
|     WCHAR File0Path[MAX_PATH];
 | |
|     WCHAR File1Path[MAX_PATH];
 | |
|     WCHAR File2Path[MAX_PATH];
 | |
| 
 | |
|     StringCbPrintfW(Dir1Path, sizeof Dir1Path, L"%s%s\\dir1",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     StringCbPrintfW(Dir2Path, sizeof Dir2Path, L"%s%s\\dir2",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     StringCbPrintfW(File0Path, sizeof File0Path, L"%s%s\\dir1\\file0",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     StringCbPrintfW(File1Path, sizeof File1Path, L"%s%s\\dir1\\file1",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     StringCbPrintfW(File2Path, sizeof File2Path, L"%s%s\\dir2\\file2",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Success = CreateDirectoryW(Dir1Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = CreateDirectoryW(Dir2Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Handle = CreateFileW(File0Path,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     Success = MoveFileExW(File0Path, File1Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Handle = CreateFileW(File0Path,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     Success = MoveFileExW(File0Path, File1Path, 0);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_ALREADY_EXISTS == GetLastError());
 | |
| 
 | |
|     Success = MoveFileExW(File0Path, File1Path, MOVEFILE_REPLACE_EXISTING);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = MoveFileExW(File1Path, File2Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     /* cannot replace existing directory regardless of MOVEFILE_REPLACE_EXISTING */
 | |
|     Success = MoveFileExW(Dir2Path, Dir1Path, MOVEFILE_REPLACE_EXISTING);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_ACCESS_DENIED == GetLastError());
 | |
| 
 | |
|     Success = RemoveDirectoryW(Dir1Path);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = MoveFileExW(Dir2Path, Dir1Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = MoveFileExW(Dir1Path, Dir2Path, 0);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Handle = CreateFileW(File2Path,
 | |
|         GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0,
 | |
|         OPEN_EXISTING, 0, 0);
 | |
|     ASSERT(INVALID_HANDLE_VALUE != Handle);
 | |
| 
 | |
|     Success = MoveFileExW(Dir2Path, Dir1Path, 0);
 | |
|     ASSERT(!Success);
 | |
|     ASSERT(ERROR_ACCESS_DENIED == GetLastError());
 | |
| 
 | |
|     CloseHandle(Handle);
 | |
| 
 | |
|     Success = DeleteFileW(File2Path);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = RemoveDirectoryW(Dir2Path);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void rename_test(void)
 | |
| {
 | |
|     if (NtfsTests)
 | |
|     {
 | |
|         WCHAR DirBuf[MAX_PATH] = L"\\\\?\\";
 | |
|         GetCurrentDirectoryW(MAX_PATH - 4, DirBuf + 4);
 | |
|         rename_dotest(-1, DirBuf, 0);
 | |
|     }
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         rename_dotest(MemfsDisk, 0, 0);
 | |
|         rename_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         rename_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         rename_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void getvolinfo_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     BOOL Success;
 | |
|     WCHAR FilePath[MAX_PATH];
 | |
|     WCHAR VolumeLabelBuf[MAX_PATH];
 | |
|     DWORD VolumeSerialNumber;
 | |
|     DWORD MaxComponentLength;
 | |
|     DWORD FileSystemFlags;
 | |
|     WCHAR FileSystemNameBuf[MAX_PATH];
 | |
|     DWORD SectorsPerCluster;
 | |
|     DWORD BytesPerSector;
 | |
|     DWORD FreeClusters;
 | |
|     DWORD TotalClusters;
 | |
|     ULARGE_INTEGER CallerFreeBytes;
 | |
|     ULARGE_INTEGER TotalBytes;
 | |
|     ULARGE_INTEGER FreeBytes;
 | |
| 
 | |
|     StringCbPrintfW(FilePath, sizeof FilePath, L"%s%s\\",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Success = GetVolumeInformationW(FilePath,
 | |
|         VolumeLabelBuf, sizeof VolumeLabelBuf,
 | |
|         &VolumeSerialNumber, &MaxComponentLength, &FileSystemFlags,
 | |
|         FileSystemNameBuf, sizeof FileSystemNameBuf);
 | |
|     ASSERT(Success);
 | |
|     if (-1 != Flags)
 | |
|     {
 | |
|         ASSERT(0 == wcscmp(VolumeLabelBuf, L"MEMFS"));
 | |
|         ASSERT(255 == MaxComponentLength);
 | |
|         ASSERT(0 != (FileSystemFlags &
 | |
|             (FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES | FILE_UNICODE_ON_DISK | FILE_PERSISTENT_ACLS)));
 | |
|         ASSERT(0 == wcscmp(FileSystemNameBuf, L"WinFsp"));
 | |
|     }
 | |
| 
 | |
|     Success = GetDiskFreeSpaceW(FilePath, &SectorsPerCluster, &BytesPerSector, &FreeClusters, &TotalClusters);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     Success = GetDiskFreeSpaceExW(FilePath, &CallerFreeBytes, &TotalBytes, &FreeBytes);
 | |
|     ASSERT(Success);
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void getvolinfo_test(void)
 | |
| {
 | |
|     if (NtfsTests)
 | |
|         getvolinfo_dotest(-1, L"C:", 0);
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         getvolinfo_dotest(MemfsDisk, 0, 0);
 | |
|         getvolinfo_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         getvolinfo_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         getvolinfo_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void setvolinfo_dotest(ULONG Flags, PWSTR Prefix, ULONG FileInfoTimeout)
 | |
| {
 | |
|     if (-1 == Flags)
 | |
|         return;/* avoid accidentally changing the volume label on our NTFS disk */
 | |
|     if (0 != Prefix)
 | |
|         return;/* cannot do SetVolumeLabel on a network share! */
 | |
| 
 | |
|     void *memfs = memfs_start_ex(Flags, FileInfoTimeout);
 | |
| 
 | |
|     BOOL Success;
 | |
|     WCHAR FilePath[MAX_PATH];
 | |
|     WCHAR VolumeLabelBuf[MAX_PATH];
 | |
|     DWORD VolumeSerialNumber;
 | |
|     DWORD MaxComponentLength;
 | |
|     DWORD FileSystemFlags;
 | |
|     WCHAR FileSystemNameBuf[MAX_PATH];
 | |
| 
 | |
|     StringCbPrintfW(FilePath, sizeof FilePath, L"%s%s\\",
 | |
|         Prefix ? L"" : L"\\\\?\\GLOBALROOT", Prefix ? Prefix : memfs_volumename(memfs));
 | |
| 
 | |
|     Success = SetVolumeLabelW(FilePath, L"12345678901234567890123456789012");
 | |
|     if (Success)
 | |
|     {
 | |
|         Success = GetVolumeInformationW(FilePath,
 | |
|             VolumeLabelBuf, sizeof VolumeLabelBuf,
 | |
|             &VolumeSerialNumber, &MaxComponentLength, &FileSystemFlags,
 | |
|             FileSystemNameBuf, sizeof FileSystemNameBuf);
 | |
|         ASSERT(Success);
 | |
|         if (-1 != Flags)
 | |
|         {
 | |
|             ASSERT(0 == wcscmp(VolumeLabelBuf, L"12345678901234567890123456789012"));
 | |
|             ASSERT(255 == MaxComponentLength);
 | |
|             ASSERT(0 != (FileSystemFlags &
 | |
|                 (FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES | FILE_UNICODE_ON_DISK | FILE_PERSISTENT_ACLS)));
 | |
|             ASSERT(0 == wcscmp(FileSystemNameBuf, L"WinFsp"));
 | |
|         }
 | |
| 
 | |
|         Success = SetVolumeLabelW(FilePath, L"TestLabel");
 | |
|         ASSERT(Success);
 | |
| 
 | |
|         Success = GetVolumeInformationW(FilePath,
 | |
|             VolumeLabelBuf, sizeof VolumeLabelBuf,
 | |
|             &VolumeSerialNumber, &MaxComponentLength, &FileSystemFlags,
 | |
|             FileSystemNameBuf, sizeof FileSystemNameBuf);
 | |
|         ASSERT(Success);
 | |
|         if (-1 != Flags)
 | |
|         {
 | |
|             ASSERT(0 == wcscmp(VolumeLabelBuf, L"TestLabel"));
 | |
|             ASSERT(255 == MaxComponentLength);
 | |
|             ASSERT(0 != (FileSystemFlags &
 | |
|                 (FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES | FILE_UNICODE_ON_DISK | FILE_PERSISTENT_ACLS)));
 | |
|             ASSERT(0 == wcscmp(FileSystemNameBuf, L"WinFsp"));
 | |
|         }
 | |
| 
 | |
|         Success = SetVolumeLabelW(FilePath, L"123456789012345678901234567890123");
 | |
|         ASSERT(Success);
 | |
| 
 | |
|         Success = GetVolumeInformationW(FilePath,
 | |
|             VolumeLabelBuf, sizeof VolumeLabelBuf,
 | |
|             &VolumeSerialNumber, &MaxComponentLength, &FileSystemFlags,
 | |
|             FileSystemNameBuf, sizeof FileSystemNameBuf);
 | |
|         ASSERT(Success);
 | |
|         if (-1 != Flags)
 | |
|         {
 | |
|             ASSERT(0 == wcscmp(VolumeLabelBuf, L"12345678901234567890123456789012"));
 | |
|             ASSERT(255 == MaxComponentLength);
 | |
|             ASSERT(0 != (FileSystemFlags &
 | |
|                 (FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES | FILE_UNICODE_ON_DISK | FILE_PERSISTENT_ACLS)));
 | |
|             ASSERT(0 == wcscmp(FileSystemNameBuf, L"WinFsp"));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     memfs_stop(memfs);
 | |
| }
 | |
| 
 | |
| void setvolinfo_test(void)
 | |
| {
 | |
| #if 0
 | |
|     if (NtfsTests)
 | |
|         setvolinfo_dotest(-1, L"C:", 0);
 | |
| #endif
 | |
|     if (WinFspDiskTests)
 | |
|     {
 | |
|         setvolinfo_dotest(MemfsDisk, 0, 0);
 | |
|         setvolinfo_dotest(MemfsDisk, 0, 1000);
 | |
|     }
 | |
|     if (WinFspNetTests)
 | |
|     {
 | |
|         setvolinfo_dotest(MemfsNet, L"\\\\memfs\\share", 0);
 | |
|         setvolinfo_dotest(MemfsNet, L"\\\\memfs\\share", 1000);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void info_tests(void)
 | |
| {
 | |
|     TEST(getfileinfo_test);
 | |
|     TEST(setfileinfo_test);
 | |
|     TEST(delete_test);
 | |
|     TEST(delete_access_test);
 | |
|     TEST(rename_test);
 | |
|     TEST(getvolinfo_test);
 | |
|     TEST(setvolinfo_test);
 | |
| }
 |