MSFileSysTest.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // //
  3. // MSFileSysTest.cpp //
  4. // Copyright (C) Microsoft Corporation. All rights reserved. //
  5. // This file is distributed under the University of Illinois Open Source //
  6. // License. See LICENSE.TXT for details. //
  7. // //
  8. // Provides tests for the file system abstraction API. //
  9. // //
  10. ///////////////////////////////////////////////////////////////////////////////
  11. #include <stdint.h>
  12. #include <dxc/Support/WinIncludes.h>
  13. #include "WexTestClass.h"
  14. #include "HlslTestUtils.h"
  15. #include "llvm/Support/MSFileSystem.h"
  16. #include <D3Dcommon.h>
  17. #include "dxc/dxcapi.internal.h"
  18. #include <algorithm>
  19. #include <vector>
  20. #include <memory>
  21. using namespace llvm;
  22. using namespace llvm::sys;
  23. using namespace llvm::sys::fs;
  24. const GUID DECLSPEC_SELECTANY GUID_NULL = { 0, 0, 0, { 0, 0, 0, 0, 0, 0, 0, 0 } };
  25. class MSFileSysTest
  26. {
  27. public:
  28. BEGIN_TEST_CLASS(MSFileSysTest)
  29. TEST_CLASS_PROPERTY(L"Parallel", L"true")
  30. TEST_METHOD_PROPERTY(L"Priority", L"0")
  31. END_TEST_CLASS()
  32. TEST_METHOD(CreationWhenInvokedThenNonNull);
  33. TEST_METHOD(FindFirstWhenInvokedThenHasFile);
  34. TEST_METHOD(FindFirstWhenInvokedThenFailsIfNoMatch);
  35. TEST_METHOD(FindNextWhenLastThenNoMatch);
  36. TEST_METHOD(FindNextWhenExistsThenMatch);
  37. TEST_METHOD(OpenWhenNewThenZeroSize);
  38. };
  39. static
  40. LPWSTR CoTaskMemDup(LPCWSTR text)
  41. {
  42. if (text == nullptr) return nullptr;
  43. size_t len = wcslen(text) + 1;
  44. LPWSTR result = (LPWSTR)CoTaskMemAlloc(sizeof(wchar_t) * len);
  45. StringCchCopyW(result, len, text);
  46. return result;
  47. }
  48. class FixedEnumSTATSTG : public IEnumSTATSTG
  49. {
  50. SIMPLE_IUNKNOWN_IMPL1(IEnumSTATSTG)
  51. private:
  52. std::vector<STATSTG> m_items;
  53. unsigned m_index;
  54. public:
  55. FixedEnumSTATSTG(_In_count_(itemCount) const STATSTG* items, unsigned itemCount)
  56. {
  57. m_dwRef = 0;
  58. m_index = 0;
  59. m_items.reserve(itemCount);
  60. for (unsigned i = 0; i < itemCount; ++i)
  61. {
  62. m_items.push_back(items[i]);
  63. m_items[i].pwcsName = CoTaskMemDup(m_items[i].pwcsName);
  64. }
  65. }
  66. ~FixedEnumSTATSTG()
  67. {
  68. for (auto& item : m_items) CoTaskMemFree(item.pwcsName);
  69. }
  70. virtual HRESULT STDMETHODCALLTYPE Next(ULONG celt, _Out_writes_to_(celt, *pceltFetched) STATSTG *rgelt, _Out_opt_ ULONG *pceltFetched)
  71. {
  72. if (celt != 1 || pceltFetched == nullptr) return E_NOTIMPL;
  73. if (m_index >= m_items.size())
  74. {
  75. *pceltFetched = 0;
  76. return S_FALSE;
  77. }
  78. *pceltFetched = 1;
  79. *rgelt = m_items[m_index];
  80. (*rgelt).pwcsName = CoTaskMemDup((*rgelt).pwcsName);
  81. ++m_index;
  82. return S_OK;
  83. }
  84. virtual HRESULT STDMETHODCALLTYPE Skip(ULONG celt) { return E_NOTIMPL; }
  85. virtual HRESULT STDMETHODCALLTYPE Reset(void) { return E_NOTIMPL; }
  86. virtual HRESULT STDMETHODCALLTYPE Clone(IEnumSTATSTG **) { return E_NOTIMPL; }
  87. };
  88. class MockDxcSystemAccess : public IDxcSystemAccess
  89. {
  90. SIMPLE_IUNKNOWN_IMPL1(IDxcSystemAccess)
  91. private:
  92. LPCSTR m_fileName;
  93. LPCSTR m_contents;
  94. unsigned m_length;
  95. public:
  96. unsigned findCount;
  97. MockDxcSystemAccess() : findCount(1), m_dwRef(0)
  98. {
  99. }
  100. static HRESULT Create(MockDxcSystemAccess** pResult)
  101. {
  102. *pResult = new (std::nothrow) MockDxcSystemAccess();
  103. if (*pResult == nullptr) return E_OUTOFMEMORY;
  104. (*pResult)->AddRef();
  105. return S_OK;
  106. }
  107. virtual HRESULT STDMETHODCALLTYPE EnumFiles(LPCWSTR fileName, IEnumSTATSTG** pResult) override {
  108. STATSTG items[] =
  109. {
  110. { L"filename.hlsl", STGTY_STREAM, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, 0, 0, GUID_NULL, 0, 0 },
  111. { L"filename2.fx", STGTY_STREAM, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, 0, 0, GUID_NULL, 0, 0 }
  112. };
  113. unsigned testCount = (unsigned)_countof(items);
  114. FixedEnumSTATSTG* resultEnum = new (std::nothrow) FixedEnumSTATSTG(items, std::min(testCount, findCount));
  115. if (resultEnum == nullptr) {
  116. *pResult = nullptr;
  117. return E_OUTOFMEMORY;
  118. }
  119. resultEnum->AddRef();
  120. *pResult = resultEnum;
  121. return S_OK;
  122. }
  123. virtual HRESULT STDMETHODCALLTYPE OpenStorage(
  124. _In_ LPCWSTR lpFileName,
  125. _In_ DWORD dwDesiredAccess,
  126. _In_ DWORD dwShareMode,
  127. _In_ DWORD dwCreationDisposition,
  128. _In_ DWORD dwFlagsAndAttributes,
  129. IUnknown** pResult) override {
  130. *pResult = SHCreateMemStream(nullptr, 0);
  131. return (*pResult == nullptr) ? E_OUTOFMEMORY : S_OK;
  132. }
  133. virtual HRESULT STDMETHODCALLTYPE SetStorageTime(_In_ IUnknown* storage,
  134. _In_opt_ const FILETIME *lpCreationTime,
  135. _In_opt_ const FILETIME *lpLastAccessTime,
  136. _In_opt_ const FILETIME *lpLastWriteTime) override {
  137. return E_NOTIMPL;
  138. }
  139. virtual HRESULT STDMETHODCALLTYPE GetFileInformationForStorage(_In_ IUnknown* storage, _Out_ LPBY_HANDLE_FILE_INFORMATION lpFileInformation) override {
  140. return E_NOTIMPL;
  141. }
  142. virtual HRESULT STDMETHODCALLTYPE GetFileTypeForStorage(_In_ IUnknown* storage, _Out_ DWORD* fileType) override {
  143. return E_NOTIMPL;
  144. }
  145. virtual HRESULT STDMETHODCALLTYPE CreateHardLinkInStorage(_In_ LPCWSTR lpFileName, _In_ LPCWSTR lpExistingFileName) override {
  146. return E_NOTIMPL;
  147. }
  148. virtual HRESULT STDMETHODCALLTYPE MoveStorage(_In_ LPCWSTR lpExistingFileName, _In_opt_ LPCWSTR lpNewFileName, _In_ DWORD dwFlags) override {
  149. return E_NOTIMPL;
  150. }
  151. virtual HRESULT STDMETHODCALLTYPE GetFileAttributesForStorage(_In_ LPCWSTR lpFileName, _Out_ DWORD* pResult) override {
  152. return E_NOTIMPL;
  153. }
  154. virtual HRESULT STDMETHODCALLTYPE DeleteStorage(_In_ LPCWSTR lpFileName) override {
  155. return E_NOTIMPL;
  156. }
  157. virtual HRESULT STDMETHODCALLTYPE RemoveDirectoryStorage(LPCWSTR lpFileName) override {
  158. return E_NOTIMPL;
  159. }
  160. virtual HRESULT STDMETHODCALLTYPE CreateDirectoryStorage(_In_ LPCWSTR lpPathName) override {
  161. return E_NOTIMPL;
  162. }
  163. virtual HRESULT STDMETHODCALLTYPE GetCurrentDirectoryForStorage(DWORD nBufferLength, _Out_writes_(nBufferLength) LPWSTR lpBuffer, _Out_ DWORD* len) override {
  164. return E_NOTIMPL;
  165. }
  166. virtual HRESULT STDMETHODCALLTYPE GetMainModuleFileNameW(DWORD nBufferLength, _Out_writes_(nBufferLength) LPWSTR lpBuffer, _Out_ DWORD* len) override {
  167. return E_NOTIMPL;
  168. }
  169. virtual HRESULT STDMETHODCALLTYPE GetTempStoragePath(DWORD nBufferLength, _Out_writes_(nBufferLength) LPWSTR lpBuffer, _Out_ DWORD* len) override {
  170. return E_NOTIMPL;
  171. }
  172. virtual HRESULT STDMETHODCALLTYPE SupportsCreateSymbolicLink(_Out_ BOOL* pResult) override {
  173. return E_NOTIMPL;
  174. }
  175. virtual HRESULT STDMETHODCALLTYPE CreateSymbolicLinkInStorage(_In_ LPCWSTR lpSymlinkFileName, _In_ LPCWSTR lpTargetFileName, DWORD dwFlags) override {
  176. return E_NOTIMPL;
  177. }
  178. virtual HRESULT STDMETHODCALLTYPE CreateStorageMapping(
  179. _In_ IUnknown* hFile,
  180. _In_ DWORD flProtect,
  181. _In_ DWORD dwMaximumSizeHigh,
  182. _In_ DWORD dwMaximumSizeLow,
  183. _Outptr_ IUnknown** pResult) override {
  184. return E_NOTIMPL;
  185. }
  186. virtual HRESULT MapViewOfFile(
  187. _In_ IUnknown* hFileMappingObject,
  188. _In_ DWORD dwDesiredAccess,
  189. _In_ DWORD dwFileOffsetHigh,
  190. _In_ DWORD dwFileOffsetLow,
  191. _In_ SIZE_T dwNumberOfBytesToMap,
  192. _Outptr_ ID3D10Blob** pResult) override {
  193. return E_NOTIMPL;
  194. }
  195. virtual HRESULT STDMETHODCALLTYPE OpenStdStorage(int standardFD, _Outptr_ IUnknown** pResult) override {
  196. return E_NOTIMPL;
  197. }
  198. virtual HRESULT STDMETHODCALLTYPE GetStreamDisplay(_COM_Outptr_result_maybenull_ ITextFont** textFont, _Out_ unsigned* columnCount) override {
  199. return E_NOTIMPL;
  200. }
  201. };
  202. void MSFileSysTest::CreationWhenInvokedThenNonNull()
  203. {
  204. CComPtr<MockDxcSystemAccess> access;
  205. VERIFY_SUCCEEDED(MockDxcSystemAccess::Create(&access));
  206. MSFileSystem* fileSystem;
  207. VERIFY_SUCCEEDED(CreateMSFileSystemForIface(access, &fileSystem));
  208. VERIFY_IS_NOT_NULL(fileSystem);
  209. delete fileSystem;
  210. }
  211. void MSFileSysTest::FindFirstWhenInvokedThenHasFile()
  212. {
  213. CComPtr<MockDxcSystemAccess> access;
  214. MockDxcSystemAccess::Create(&access);
  215. MSFileSystem* fileSystem;
  216. CreateMSFileSystemForIface(access, &fileSystem);
  217. WIN32_FIND_DATAW findData;
  218. HANDLE h = fileSystem->FindFirstFileW(L"foobar", &findData);
  219. VERIFY_ARE_EQUAL_WSTR(L"filename.hlsl", findData.cFileName);
  220. VERIFY_ARE_NOT_EQUAL(INVALID_HANDLE_VALUE, h);
  221. fileSystem->FindClose(h);
  222. delete fileSystem;
  223. }
  224. void MSFileSysTest::FindFirstWhenInvokedThenFailsIfNoMatch()
  225. {
  226. CComPtr<MockDxcSystemAccess> access;
  227. MockDxcSystemAccess::Create(&access);
  228. access->findCount = 0;
  229. MSFileSystem* fileSystem;
  230. CreateMSFileSystemForIface(access, &fileSystem);
  231. WIN32_FIND_DATAW findData;
  232. HANDLE h = fileSystem->FindFirstFileW(L"foobar", &findData);
  233. VERIFY_ARE_EQUAL(ERROR_FILE_NOT_FOUND, GetLastError());
  234. VERIFY_ARE_EQUAL(INVALID_HANDLE_VALUE, h);
  235. VERIFY_ARE_EQUAL_WSTR(L"", findData.cFileName);
  236. delete fileSystem;
  237. }
  238. void MSFileSysTest::FindNextWhenLastThenNoMatch()
  239. {
  240. CComPtr<MockDxcSystemAccess> access;
  241. MockDxcSystemAccess::Create(&access);
  242. MSFileSystem* fileSystem;
  243. CreateMSFileSystemForIface(access, &fileSystem);
  244. WIN32_FIND_DATAW findData;
  245. HANDLE h = fileSystem->FindFirstFileW(L"foobar", &findData);
  246. VERIFY_ARE_NOT_EQUAL(INVALID_HANDLE_VALUE, h);
  247. BOOL findNext = fileSystem->FindNextFileW(h, &findData);
  248. VERIFY_IS_FALSE(findNext);
  249. VERIFY_ARE_EQUAL(ERROR_FILE_NOT_FOUND, GetLastError());
  250. fileSystem->FindClose(h);
  251. delete fileSystem;
  252. }
  253. void MSFileSysTest::FindNextWhenExistsThenMatch()
  254. {
  255. CComPtr<MockDxcSystemAccess> access;
  256. MockDxcSystemAccess::Create(&access);
  257. access->findCount = 2;
  258. MSFileSystem* fileSystem;
  259. CreateMSFileSystemForIface(access, &fileSystem);
  260. WIN32_FIND_DATAW findData;
  261. HANDLE h = fileSystem->FindFirstFileW(L"foobar", &findData);
  262. VERIFY_ARE_NOT_EQUAL(INVALID_HANDLE_VALUE, h);
  263. BOOL findNext = fileSystem->FindNextFileW(h, &findData);
  264. VERIFY_IS_TRUE(findNext);
  265. VERIFY_ARE_EQUAL_WSTR(L"filename2.fx", findData.cFileName);
  266. VERIFY_IS_FALSE(fileSystem->FindNextFileW(h, &findData));
  267. fileSystem->FindClose(h);
  268. delete fileSystem;
  269. }
  270. void MSFileSysTest::OpenWhenNewThenZeroSize()
  271. {
  272. CComPtr<MockDxcSystemAccess> access;
  273. MockDxcSystemAccess::Create(&access);
  274. MSFileSystem* fileSystem;
  275. CreateMSFileSystemForIface(access, &fileSystem);
  276. HANDLE h = fileSystem->CreateFileW(L"new.hlsl", 0, 0, 0, 0);
  277. VERIFY_ARE_NOT_EQUAL(INVALID_HANDLE_VALUE, h);
  278. char buf[4];
  279. DWORD bytesRead;
  280. VERIFY_IS_TRUE(fileSystem->ReadFile(h, buf, _countof(buf), &bytesRead));
  281. VERIFY_ARE_EQUAL(0, bytesRead);
  282. fileSystem->CloseHandle(h);
  283. delete fileSystem;
  284. }