LinkerTest.cpp 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // //
  3. // LinkerTest.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. // //
  9. ///////////////////////////////////////////////////////////////////////////////
  10. #include <memory>
  11. #include <vector>
  12. #include <string>
  13. #include "llvm/ADT/ArrayRef.h"
  14. #include "CompilationResult.h"
  15. #include "HLSLTestData.h"
  16. #include "llvm/Support/ManagedStatic.h"
  17. #include <fstream>
  18. #include "WexTestClass.h"
  19. #include "HlslTestUtils.h"
  20. #include "dxc/dxcapi.h"
  21. #include "DxcTestUtils.h"
  22. using namespace std;
  23. using namespace hlsl;
  24. using namespace llvm;
  25. // The test fixture.
  26. class LinkerTest
  27. {
  28. public:
  29. BEGIN_TEST_CLASS(LinkerTest)
  30. TEST_CLASS_PROPERTY(L"Parallel", L"true")
  31. TEST_METHOD_PROPERTY(L"Priority", L"0")
  32. END_TEST_CLASS()
  33. TEST_CLASS_SETUP(InitSupport);
  34. TEST_METHOD(RunLinkResource);
  35. TEST_METHOD(RunLinkAllProfiles);
  36. TEST_METHOD(RunLinkFailNoDefine);
  37. TEST_METHOD(RunLinkFailReDefine);
  38. TEST_METHOD(RunLinkGlobalInit);
  39. TEST_METHOD(RunLinkNoAlloca);
  40. TEST_METHOD(RunLinkFailReDefineGlobal);
  41. TEST_METHOD(RunLinkFailProfileMismatch);
  42. TEST_METHOD(RunLinkFailEntryNoProps);
  43. dxc::DxcDllSupport m_dllSupport;
  44. VersionSupportInfo m_ver;
  45. void CreateLinker(IDxcLinker **pResultLinker) {
  46. VERIFY_SUCCEEDED(
  47. m_dllSupport.CreateInstance(CLSID_DxcLinker, pResultLinker));
  48. }
  49. void CompileLib(LPCWSTR filename, IDxcBlob **pResultBlob) {
  50. std::wstring fullPath = hlsl_test::GetPathToHlslDataFile(filename);
  51. CComPtr<IDxcBlobEncoding> pSource;
  52. CComPtr<IDxcLibrary> pLibrary;
  53. VERIFY_SUCCEEDED(m_dllSupport.CreateInstance(CLSID_DxcLibrary, &pLibrary));
  54. VERIFY_SUCCEEDED(
  55. pLibrary->CreateBlobFromFile(fullPath.c_str(), nullptr, &pSource));
  56. CComPtr<IDxcCompiler> pCompiler;
  57. CComPtr<IDxcOperationResult> pResult;
  58. CComPtr<IDxcBlob> pProgram;
  59. CA2W shWide("lib_6_1", CP_UTF8);
  60. VERIFY_SUCCEEDED(
  61. m_dllSupport.CreateInstance(CLSID_DxcCompiler, &pCompiler));
  62. VERIFY_SUCCEEDED(pCompiler->Compile(pSource, L"hlsl.hlsl", L"", shWide,
  63. nullptr, 0, nullptr, 0, nullptr,
  64. &pResult));
  65. VERIFY_SUCCEEDED(pResult->GetResult(pResultBlob));
  66. }
  67. void RegisterDxcModule(LPCWSTR pLibName, IDxcBlob *pBlob,
  68. IDxcLinker *pLinker) {
  69. VERIFY_SUCCEEDED(pLinker->RegisterLibrary(pLibName, pBlob));
  70. }
  71. void Link(LPCWSTR pEntryName, LPCWSTR pShaderModel, IDxcLinker *pLinker,
  72. ArrayRef<LPCWSTR> libNames, llvm::ArrayRef<LPCSTR> pCheckMsgs,
  73. llvm::ArrayRef<LPCSTR> pCheckNotMsgs) {
  74. CComPtr<IDxcOperationResult> pResult;
  75. VERIFY_SUCCEEDED(pLinker->Link(pEntryName, pShaderModel, libNames.data(),
  76. libNames.size(), nullptr, 0, &pResult));
  77. CComPtr<IDxcBlob> pProgram;
  78. CheckOperationSucceeded(pResult, &pProgram);
  79. CComPtr<IDxcCompiler> pCompiler;
  80. CComPtr<IDxcBlobEncoding> pDisassembly;
  81. VERIFY_SUCCEEDED(
  82. m_dllSupport.CreateInstance(CLSID_DxcCompiler, &pCompiler));
  83. VERIFY_SUCCEEDED(pCompiler->Disassemble(pProgram, &pDisassembly));
  84. std::string IR = BlobToUtf8(pDisassembly);
  85. CheckMsgs(IR.c_str(), IR.size(), pCheckMsgs.data(), pCheckMsgs.size(), false);
  86. for (auto notMsg : pCheckNotMsgs) {
  87. VERIFY_IS_TRUE(IR.find(notMsg) == std::string::npos);
  88. }
  89. }
  90. void LinkCheckMsg(LPCWSTR pEntryName, LPCWSTR pShaderModel, IDxcLinker *pLinker,
  91. ArrayRef<LPCWSTR> libNames, llvm::ArrayRef<LPCSTR> pErrorMsgs) {
  92. CComPtr<IDxcOperationResult> pResult;
  93. VERIFY_SUCCEEDED(pLinker->Link(pEntryName, pShaderModel, libNames.data(),
  94. libNames.size(), nullptr, 0, &pResult));
  95. CheckOperationResultMsgs(pResult, pErrorMsgs.data(), pErrorMsgs.size(),
  96. false, false);
  97. }
  98. };
  99. bool LinkerTest::InitSupport() {
  100. if (!m_dllSupport.IsEnabled()) {
  101. VERIFY_SUCCEEDED(m_dllSupport.Initialize());
  102. m_ver.Initialize(m_dllSupport);
  103. }
  104. return true;
  105. }
  106. TEST_F(LinkerTest, RunLinkResource) {
  107. CComPtr<IDxcBlob> pResLib;
  108. CompileLib(L"..\\CodeGenHLSL\\lib_resource2.hlsl", &pResLib);
  109. CComPtr<IDxcBlob> pEntryLib;
  110. CompileLib(L"..\\CodeGenHLSL\\lib_cs_entry.hlsl", &pEntryLib);
  111. CComPtr<IDxcLinker> pLinker;
  112. CreateLinker(&pLinker);
  113. LPCWSTR libName = L"entry";
  114. RegisterDxcModule(libName, pEntryLib, pLinker);
  115. LPCWSTR libResName = L"res";
  116. RegisterDxcModule(libResName, pResLib, pLinker);
  117. Link(L"entry", L"cs_6_0", pLinker, {libResName, libName}, {} ,{});
  118. }
  119. TEST_F(LinkerTest, RunLinkAllProfiles) {
  120. CComPtr<IDxcLinker> pLinker;
  121. CreateLinker(&pLinker);
  122. LPCWSTR libName = L"entry";
  123. CComPtr<IDxcBlob> pEntryLib;
  124. CompileLib(L"..\\CodeGenHLSL\\lib_entries2.hlsl", &pEntryLib);
  125. RegisterDxcModule(libName, pEntryLib, pLinker);
  126. Link(L"vs_main", L"vs_6_0", pLinker, {libName}, {},{});
  127. Link(L"hs_main", L"hs_6_0", pLinker, {libName}, {},{});
  128. Link(L"ds_main", L"ds_6_0", pLinker, {libName}, {},{});
  129. Link(L"gs_main", L"gs_6_0", pLinker, {libName}, {},{});
  130. Link(L"ps_main", L"ps_6_0", pLinker, {libName}, {},{});
  131. CComPtr<IDxcBlob> pResLib;
  132. CompileLib(L"..\\CodeGenHLSL\\lib_resource2.hlsl", &pResLib);
  133. LPCWSTR libResName = L"res";
  134. RegisterDxcModule(libResName, pResLib, pLinker);
  135. Link(L"cs_main", L"cs_6_0", pLinker, {libName, libResName}, {},{});
  136. }
  137. TEST_F(LinkerTest, RunLinkFailNoDefine) {
  138. CComPtr<IDxcBlob> pEntryLib;
  139. CompileLib(L"..\\CodeGenHLSL\\lib_cs_entry.hlsl", &pEntryLib);
  140. CComPtr<IDxcLinker> pLinker;
  141. CreateLinker(&pLinker);
  142. LPCWSTR libName = L"entry";
  143. RegisterDxcModule(libName, pEntryLib, pLinker);
  144. LinkCheckMsg(L"entry", L"cs_6_0", pLinker, {libName},
  145. {"Cannot find definition of function"});
  146. }
  147. TEST_F(LinkerTest, RunLinkFailReDefine) {
  148. CComPtr<IDxcBlob> pEntryLib;
  149. CompileLib(L"..\\CodeGenHLSL\\lib_cs_entry.hlsl", &pEntryLib);
  150. CComPtr<IDxcLinker> pLinker;
  151. CreateLinker(&pLinker);
  152. LPCWSTR libName = L"entry";
  153. RegisterDxcModule(libName, pEntryLib, pLinker);
  154. LPCWSTR libName2 = L"entry2";
  155. RegisterDxcModule(libName2, pEntryLib, pLinker);
  156. LinkCheckMsg(L"entry", L"cs_6_0", pLinker, {libName, libName2},
  157. {"Definition already exists for function"});
  158. }
  159. TEST_F(LinkerTest, RunLinkGlobalInit) {
  160. CComPtr<IDxcBlob> pEntryLib;
  161. CompileLib(L"..\\CodeGenHLSL\\lib_global.hlsl", &pEntryLib);
  162. CComPtr<IDxcLinker> pLinker;
  163. CreateLinker(&pLinker);
  164. LPCWSTR libName = L"entry";
  165. RegisterDxcModule(libName, pEntryLib, pLinker);
  166. Link(L"test", L"ps_6_0", pLinker, {libName},
  167. // Make sure cbuffer load is generated.
  168. {"dx.op.cbufferLoad"},{});
  169. }
  170. TEST_F(LinkerTest, RunLinkFailReDefineGlobal) {
  171. CComPtr<IDxcBlob> pEntryLib;
  172. CompileLib(L"..\\CodeGenHLSL\\lib_global2.hlsl", &pEntryLib);
  173. CComPtr<IDxcBlob> pLib0;
  174. CompileLib(L"..\\CodeGenHLSL\\lib_global3.hlsl", &pLib0);
  175. CComPtr<IDxcBlob> pLib1;
  176. CompileLib(L"..\\CodeGenHLSL\\lib_global4.hlsl", &pLib1);
  177. CComPtr<IDxcLinker> pLinker;
  178. CreateLinker(&pLinker);
  179. LPCWSTR libName = L"entry";
  180. RegisterDxcModule(libName, pEntryLib, pLinker);
  181. LPCWSTR libName1 = L"lib0";
  182. RegisterDxcModule(libName1, pLib0, pLinker);
  183. LPCWSTR libName2 = L"lib1";
  184. RegisterDxcModule(libName2, pLib1, pLinker);
  185. LinkCheckMsg(L"entry", L"cs_6_0", pLinker, {libName, libName1, libName2},
  186. {"Definition already exists for global variable", "Resource already exists"});
  187. }
  188. TEST_F(LinkerTest, RunLinkFailProfileMismatch) {
  189. CComPtr<IDxcBlob> pEntryLib;
  190. CompileLib(L"..\\CodeGenHLSL\\lib_global.hlsl", &pEntryLib);
  191. CComPtr<IDxcLinker> pLinker;
  192. CreateLinker(&pLinker);
  193. LPCWSTR libName = L"entry";
  194. RegisterDxcModule(libName, pEntryLib, pLinker);
  195. LinkCheckMsg(L"test", L"cs_6_0", pLinker, {libName},
  196. {"Profile mismatch between entry function and target profile"});
  197. }
  198. TEST_F(LinkerTest, RunLinkFailEntryNoProps) {
  199. CComPtr<IDxcBlob> pEntryLib;
  200. CompileLib(L"..\\CodeGenHLSL\\lib_global.hlsl", &pEntryLib);
  201. CComPtr<IDxcLinker> pLinker;
  202. CreateLinker(&pLinker);
  203. LPCWSTR libName = L"entry";
  204. RegisterDxcModule(libName, pEntryLib, pLinker);
  205. LinkCheckMsg(L"\01?update@@YAXXZ", L"cs_6_0", pLinker, {libName},
  206. {"Cannot find function property for entry function"});
  207. }
  208. TEST_F(LinkerTest, RunLinkNoAlloca) {
  209. CComPtr<IDxcBlob> pEntryLib;
  210. CompileLib(L"..\\CodeGenHLSL\\lib_no_alloca.hlsl", &pEntryLib);
  211. CComPtr<IDxcBlob> pLib;
  212. CompileLib(L"..\\CodeGenHLSL\\lib_no_alloca.h", &pLib);
  213. CComPtr<IDxcLinker> pLinker;
  214. CreateLinker(&pLinker);
  215. LPCWSTR libName = L"ps_main";
  216. RegisterDxcModule(libName, pEntryLib, pLinker);
  217. LPCWSTR libName2 = L"test";
  218. RegisterDxcModule(libName2, pLib, pLinker);
  219. Link(L"ps_main", L"ps_6_0", pLinker, {libName, libName2}, {}, {"alloca"});
  220. }