FilesystemWindows.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. // Copyright (C) 2009-2020, Panagiotis Christopoulos Charitos and contributors.
  2. // All rights reserved.
  3. // Code licensed under the BSD License.
  4. // http://www.anki3d.org/LICENSE
  5. #include <anki/util/Filesystem.h>
  6. #include <anki/util/Assert.h>
  7. #include <anki/util/Logger.h>
  8. #include <anki/util/Win32Minimal.h>
  9. namespace anki
  10. {
  11. static const U MAX_PATH_LEN = MAX_PATH - 1;
  12. Bool fileExists(const CString& filename)
  13. {
  14. DWORD dwAttrib = GetFileAttributesA(&filename[0]);
  15. return dwAttrib != INVALID_FILE_ATTRIBUTES && !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY);
  16. }
  17. Bool directoryExists(const CString& filename)
  18. {
  19. DWORD dwAttrib = GetFileAttributesA(filename.cstr());
  20. return dwAttrib != INVALID_FILE_ATTRIBUTES && (dwAttrib & FILE_ATTRIBUTE_DIRECTORY);
  21. }
  22. Error removeDirectory(const CString& dirname, GenericMemoryPoolAllocator<U8> alloc)
  23. {
  24. // For some reason dirname should be double null terminated
  25. if(dirname.getLength() > MAX_PATH - 2)
  26. {
  27. ANKI_UTIL_LOGE("Path too big");
  28. return Error::FUNCTION_FAILED;
  29. }
  30. Array<char, MAX_PATH> dirname2;
  31. memcpy(&dirname2[0], &dirname[0], dirname.getLength() + 1);
  32. dirname2[dirname.getLength() + 1] = '\0';
  33. Error err = Error::NONE;
  34. SHFILEOPSTRUCTA fileOperation = {
  35. NULL, FO_DELETE, &dirname2[0], "", FOF_NOCONFIRMATION | FOF_NOERRORUI | FOF_SILENT, false, 0, ""};
  36. I result = SHFileOperationA(&fileOperation);
  37. if(result != 0)
  38. {
  39. ANKI_UTIL_LOGE("Could not delete directory %s", &dirname[0]);
  40. err = Error::FUNCTION_FAILED;
  41. }
  42. return err;
  43. }
  44. Error createDirectory(const CString& dir)
  45. {
  46. Error err = Error::NONE;
  47. if(CreateDirectoryA(dir.cstr(), NULL) == 0)
  48. {
  49. ANKI_UTIL_LOGE("Failed to create directory %s", dir.cstr());
  50. err = Error::FUNCTION_FAILED;
  51. }
  52. return err;
  53. }
  54. Error getHomeDirectory(StringAuto& out)
  55. {
  56. char path[MAX_PATH];
  57. if(SHGetFolderPathA(NULL, CSIDL_PROFILE, nullptr, 0, path) != S_OK)
  58. {
  59. ANKI_UTIL_LOGE("SHGetFolderPath() failed");
  60. return Error::FUNCTION_FAILED;
  61. }
  62. out.create(path);
  63. return Error::NONE;
  64. }
  65. static Error walkDirectoryTreeInternal(const CString& dir, void* userData, WalkDirectoryTreeCallback callback,
  66. U baseDirLen)
  67. {
  68. // Append something to the path
  69. if(dir.getLength() > MAX_PATH_LEN - 2)
  70. {
  71. ANKI_UTIL_LOGE("Path too long");
  72. return Error::FUNCTION_FAILED;
  73. }
  74. Array<char, MAX_PATH> dir2;
  75. memcpy(&dir2[0], &dir[0], dir.getLength());
  76. if(dir[dir.getLength() - 1] != '/')
  77. {
  78. dir2[dir.getLength() + 0] = '/';
  79. dir2[dir.getLength() + 1] = '*';
  80. dir2[dir.getLength() + 2] = '\0';
  81. }
  82. else
  83. {
  84. dir2[dir.getLength() + 0] = '*';
  85. dir2[dir.getLength() + 1] = '\0';
  86. }
  87. // Find files
  88. HANDLE handle = INVALID_HANDLE_VALUE;
  89. WIN32_FIND_DATAA find;
  90. handle = FindFirstFileA(&dir2[0], &find);
  91. if(handle == INVALID_HANDLE_VALUE)
  92. {
  93. ANKI_UTIL_LOGE("FindFirstFile() failed");
  94. return Error::FUNCTION_FAILED;
  95. }
  96. // Remove '*' from dir2
  97. dir2[strlen(&dir2[0]) - 1] = '\0';
  98. do
  99. {
  100. CString filename(find.cFileName);
  101. if(filename != "." && filename != "..")
  102. {
  103. Bool isDir = find.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY;
  104. // Compute new path
  105. U oldLen = strlen(&dir2[0]);
  106. if(oldLen + filename.getLength() > MAX_PATH_LEN)
  107. {
  108. ANKI_UTIL_LOGE("Path too long");
  109. return Error::FUNCTION_FAILED;
  110. }
  111. strcat(&dir2[0], &filename[0]);
  112. Error err = callback(&dir2[0] + baseDirLen, userData, isDir);
  113. if(err)
  114. {
  115. FindClose(handle);
  116. return err;
  117. }
  118. // Move to next dir
  119. if(isDir)
  120. {
  121. Error err = walkDirectoryTreeInternal(&dir2[0], userData, callback, baseDirLen);
  122. if(err)
  123. {
  124. FindClose(handle);
  125. return err;
  126. }
  127. }
  128. dir2[oldLen] = '\0';
  129. }
  130. } while(FindNextFileA(handle, &find) != 0);
  131. if(GetLastError() != ERROR_NO_MORE_FILES)
  132. {
  133. ANKI_UTIL_LOGE("Unknown error");
  134. FindClose(handle);
  135. return Error::FUNCTION_FAILED;
  136. }
  137. FindClose(handle);
  138. return Error::NONE;
  139. }
  140. Error walkDirectoryTree(const CString& dir, void* userData, WalkDirectoryTreeCallback callback)
  141. {
  142. U baseDirLen = 0;
  143. U len = dir.getLength();
  144. if(dir[len - 1] == '/')
  145. {
  146. baseDirLen = len;
  147. }
  148. else
  149. {
  150. baseDirLen = len + 1;
  151. }
  152. return walkDirectoryTreeInternal(dir, userData, callback, baseDirLen);
  153. }
  154. } // end namespace anki