FilesystemWindows.cpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. // Copyright (C) 2009-present, 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. static constexpr U kMaxPathLen = MAX_PATH - 1;
  11. Bool fileExists(const CString& filename)
  12. {
  13. DWORD dwAttrib = GetFileAttributesA(&filename[0]);
  14. return dwAttrib != INVALID_FILE_ATTRIBUTES && !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY);
  15. }
  16. Bool directoryExists(const CString& filename)
  17. {
  18. DWORD dwAttrib = GetFileAttributesA(filename.cstr());
  19. return dwAttrib != INVALID_FILE_ATTRIBUTES && (dwAttrib & FILE_ATTRIBUTE_DIRECTORY);
  20. }
  21. Error removeDirectory(const CString& dirname)
  22. {
  23. // For some reason dirname should be double null terminated
  24. if(dirname.getLength() > MAX_PATH - 2)
  25. {
  26. ANKI_UTIL_LOGE("Path too big");
  27. return Error::kFunctionFailed;
  28. }
  29. Array<char, MAX_PATH> dirname2;
  30. memcpy(&dirname2[0], &dirname[0], dirname.getLength() + 1);
  31. dirname2[dirname.getLength() + 1] = '\0';
  32. Error err = Error::kNone;
  33. SHFILEOPSTRUCTA fileOperation = {NULL, FO_DELETE, &dirname2[0], "", FOF_NOCONFIRMATION | FOF_NOERRORUI | FOF_SILENT, false, 0, ""};
  34. I result = SHFileOperationA(&fileOperation);
  35. if(result != 0)
  36. {
  37. ANKI_UTIL_LOGE("Could not delete directory %s", &dirname[0]);
  38. err = Error::kFunctionFailed;
  39. }
  40. return err;
  41. }
  42. Error createDirectory(const CString& dir)
  43. {
  44. Error err = Error::kNone;
  45. if(CreateDirectoryA(dir.cstr(), NULL) == 0)
  46. {
  47. ANKI_UTIL_LOGE("Failed to create directory %s", dir.cstr());
  48. err = Error::kFunctionFailed;
  49. }
  50. return err;
  51. }
  52. Error getHomeDirectory(String& out)
  53. {
  54. char path[MAX_PATH];
  55. if(SHGetFolderPathA(NULL, CSIDL_PROFILE, nullptr, 0, path) != S_OK)
  56. {
  57. ANKI_UTIL_LOGE("SHGetFolderPath() failed");
  58. return Error::kFunctionFailed;
  59. }
  60. out = path;
  61. return Error::kNone;
  62. }
  63. Error getTempDirectory(String& out)
  64. {
  65. char path[MAX_PATH + 1];
  66. const DWORD len = GetTempPathA(sizeof(path), path);
  67. if(len == 0)
  68. {
  69. ANKI_UTIL_LOGE("GetTempPathA() failed");
  70. return Error::kFunctionFailed;
  71. }
  72. out = path;
  73. return Error::kNone;
  74. }
  75. static Error walkDirectoryTreeRecursive(CString dir,
  76. const Function<Error(WalkDirectoryArgs&), SingletonMemoryPoolWrapper<DefaultMemoryPool>>& callback,
  77. U baseDirLen)
  78. {
  79. // Append something to the path
  80. if(dir.getLength() > kMaxPathLen - 2)
  81. {
  82. ANKI_UTIL_LOGE("Path too long");
  83. return Error::kFunctionFailed;
  84. }
  85. Array<char, MAX_PATH> dir2;
  86. memcpy(&dir2[0], &dir[0], dir.getLength());
  87. if(dir[dir.getLength() - 1] != '/')
  88. {
  89. dir2[dir.getLength() + 0] = '/';
  90. dir2[dir.getLength() + 1] = '*';
  91. dir2[dir.getLength() + 2] = '\0';
  92. }
  93. else
  94. {
  95. dir2[dir.getLength() + 0] = '*';
  96. dir2[dir.getLength() + 1] = '\0';
  97. }
  98. // Find files
  99. HANDLE handle = INVALID_HANDLE_VALUE;
  100. WIN32_FIND_DATAA find;
  101. handle = FindFirstFileA(&dir2[0], &find);
  102. if(handle == INVALID_HANDLE_VALUE)
  103. {
  104. ANKI_UTIL_LOGE("FindFirstFile() failed");
  105. return Error::kFunctionFailed;
  106. }
  107. // Remove '*' from dir2
  108. dir2[strlen(&dir2[0]) - 1] = '\0';
  109. do
  110. {
  111. CString filename(find.cFileName);
  112. if(filename != "." && filename != "..")
  113. {
  114. Bool isDir = find.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY;
  115. // Compute new path
  116. const PtrSize oldLen = strlen(&dir2[0]);
  117. if(oldLen + filename.getLength() > kMaxPathLen)
  118. {
  119. ANKI_UTIL_LOGE("Path too long");
  120. return Error::kFunctionFailed;
  121. }
  122. strcat(&dir2[0], &filename[0]);
  123. WalkDirectoryArgs args;
  124. args.m_path = &dir2[0] + baseDirLen;
  125. args.m_isDirectory = isDir;
  126. args.m_stopSearch = false;
  127. const Error err = callback(args);
  128. if(err || args.m_stopSearch)
  129. {
  130. FindClose(handle);
  131. return err;
  132. }
  133. // Move to next dir
  134. if(isDir)
  135. {
  136. const Error err = walkDirectoryTreeRecursive(&dir2[0], callback, baseDirLen);
  137. if(err)
  138. {
  139. FindClose(handle);
  140. return err;
  141. }
  142. }
  143. dir2[oldLen] = '\0';
  144. }
  145. } while(FindNextFileA(handle, &find) != 0);
  146. if(GetLastError() != ERROR_NO_MORE_FILES)
  147. {
  148. ANKI_UTIL_LOGE("Unknown error");
  149. FindClose(handle);
  150. return Error::kFunctionFailed;
  151. }
  152. FindClose(handle);
  153. return Error::kNone;
  154. }
  155. Error walkDirectoryTreeInternal(CString dir, const Function<Error(WalkDirectoryArgs&)>& callback)
  156. {
  157. U baseDirLen = 0;
  158. const U len = dir.getLength();
  159. if(dir[len - 1] == '/')
  160. {
  161. baseDirLen = len;
  162. }
  163. else
  164. {
  165. baseDirLen = len + 1;
  166. }
  167. return walkDirectoryTreeRecursive(dir, callback, baseDirLen);
  168. }
  169. Error getApplicationPath(String& out)
  170. {
  171. DynamicArray<Char, SingletonMemoryPoolWrapper<DefaultMemoryPool>> buff;
  172. buff.resize(1024);
  173. const DWORD result = GetModuleFileNameA(nullptr, &buff[0], buff.getSize());
  174. DWORD lastError = GetLastError();
  175. if(result == 0 || (result == buff.getSize() && (lastError == ERROR_INSUFFICIENT_BUFFER || lastError == ERROR_SUCCESS)))
  176. {
  177. ANKI_UTIL_LOGE("GetModuleFileNameA() failed");
  178. return Error::kFunctionFailed;
  179. }
  180. out = String('0', result);
  181. memcpy(&out[0], &buff[0], result);
  182. return Error::kNone;
  183. }
  184. } // end namespace anki