MSFileSysTest.cpp 11 KB

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