ShaderOpTest.cpp 90 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // //
  3. // ShaderOpTest.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 the implementation to run tests based on descriptions. //
  9. // //
  10. ///////////////////////////////////////////////////////////////////////////////
  11. #include <windows.h>
  12. #include <d3d12.h>
  13. #include <dxgi1_4.h>
  14. #include "dxc/Support/d3dx12.h"
  15. #include <d3dcompiler.h>
  16. #include <atlbase.h>
  17. #include <atlenc.h>
  18. #include "ShaderOpTest.h"
  19. #include "dxc/dxcapi.h" // IDxcCompiler
  20. #include "dxc/Support/Global.h" // OutputDebugBytes
  21. #include "dxc/Support/Unicode.h" // IsStarMatchUTF16
  22. #include "dxc/Support/dxcapi.use.h" // DxcDllSupport
  23. #include "dxc/DXIL/DxilConstants.h" // ComponentType
  24. #include "WexTestClass.h" // TAEF
  25. #include "HLSLTestUtils.h" // LogCommentFmt
  26. #include <stdlib.h>
  27. #include <DirectXMath.h>
  28. #include <intsafe.h>
  29. #include <strsafe.h>
  30. #include <xmllite.h>
  31. #pragma comment(lib, "xmllite.lib")
  32. ///////////////////////////////////////////////////////////////////////////////
  33. // Useful helper functions.
  34. static st::OutputStringFn g_OutputStrFn;
  35. static void * g_OutputStrFnCtx;
  36. void st::SetOutputFn(void *pCtx, OutputStringFn F) {
  37. g_OutputStrFnCtx = pCtx;
  38. g_OutputStrFn = F;
  39. }
  40. static void ShaderOpLogFmt(_In_z_ _Printf_format_string_ const wchar_t *fmt, ...) {
  41. va_list args;
  42. va_start(args, fmt);
  43. std::wstring buf(hlsl_test::vFormatToWString(fmt, args));
  44. va_end(args);
  45. if (g_OutputStrFn == nullptr)
  46. WEX::Logging::Log::Comment(buf.data());
  47. else
  48. g_OutputStrFn(g_OutputStrFnCtx, buf.data());
  49. }
  50. // Rely on TAEF Verifier helpers.
  51. #define CHECK_HR(x) { \
  52. if (!g_OutputStrFn) VERIFY_SUCCEEDED(x); else { \
  53. HRESULT _check_hr = (x); \
  54. if (FAILED(_check_hr)) AtlThrow(x); } \
  55. }
  56. // Check the specified HRESULT and return the success value.
  57. static HRESULT CHECK_HR_RET(HRESULT hr) {
  58. CHECK_HR(hr);
  59. return hr;
  60. }
  61. HRESULT LogIfLost(HRESULT hr, ID3D12Device *pDevice) {
  62. if (hr == DXGI_ERROR_DEVICE_REMOVED) {
  63. HRESULT reason = pDevice->GetDeviceRemovedReason();
  64. LPCWSTR reasonText = L"?";
  65. if (reason == DXGI_ERROR_DEVICE_HUNG) reasonText = L"DXGI_ERROR_DEVICE_HUNG";
  66. if (reason == DXGI_ERROR_DEVICE_REMOVED) reasonText = L"DXGI_ERROR_DEVICE_REMOVED";
  67. if (reason == DXGI_ERROR_DEVICE_RESET) reasonText = L"DXGI_ERROR_DEVICE_RESET";
  68. if (reason == DXGI_ERROR_DRIVER_INTERNAL_ERROR) reasonText = L"DXGI_ERROR_DRIVER_INTERNAL_ERROR";
  69. if (reason == DXGI_ERROR_INVALID_CALL) reasonText = L"DXGI_ERROR_INVALID_CALL";
  70. ShaderOpLogFmt(L"Device lost: 0x%08x (%s)", reason, reasonText);
  71. }
  72. return hr;
  73. }
  74. HRESULT LogIfLost(HRESULT hr, ID3D12Resource *pResource) {
  75. if (hr == DXGI_ERROR_DEVICE_REMOVED) {
  76. CComPtr<ID3D12Device> pDevice;
  77. pResource->GetDevice(__uuidof(ID3D12Device), (void**)&pDevice);
  78. LogIfLost(hr, pDevice);
  79. }
  80. return hr;
  81. }
  82. bool UseHardwareDevice(const DXGI_ADAPTER_DESC1 &desc, LPCWSTR AdapterName) {
  83. if (desc.Flags & DXGI_ADAPTER_FLAG_SOFTWARE) {
  84. // Don't select the Basic Render Driver adapter.
  85. return false;
  86. }
  87. if (!AdapterName)
  88. return true;
  89. return Unicode::IsStarMatchUTF16(AdapterName, wcslen(AdapterName),
  90. desc.Description, wcslen(desc.Description));
  91. }
  92. void GetHardwareAdapter(IDXGIFactory2 *pFactory, LPCWSTR AdapterName,
  93. IDXGIAdapter1 **ppAdapter) {
  94. CComPtr<IDXGIAdapter1> adapter;
  95. *ppAdapter = nullptr;
  96. for (UINT adapterIndex = 0;
  97. DXGI_ERROR_NOT_FOUND != pFactory->EnumAdapters1(adapterIndex, &adapter);
  98. ++adapterIndex) {
  99. DXGI_ADAPTER_DESC1 desc;
  100. adapter->GetDesc1(&desc);
  101. if (!UseHardwareDevice(desc, AdapterName)) {
  102. adapter.Release();
  103. continue;
  104. }
  105. // Check to see if the adapter supports Direct3D 12, but don't create the
  106. // actual device yet.
  107. if (SUCCEEDED(D3D12CreateDevice(adapter, D3D_FEATURE_LEVEL_11_0,
  108. _uuidof(ID3D12Device), nullptr))) {
  109. break;
  110. }
  111. adapter.Release();
  112. }
  113. *ppAdapter = adapter.Detach();
  114. }
  115. void RecordTransitionBarrier(ID3D12GraphicsCommandList *pCommandList,
  116. ID3D12Resource *pResource,
  117. D3D12_RESOURCE_STATES before,
  118. D3D12_RESOURCE_STATES after) {
  119. CD3DX12_RESOURCE_BARRIER barrier(
  120. CD3DX12_RESOURCE_BARRIER::Transition(pResource, before, after));
  121. pCommandList->ResourceBarrier(1, &barrier);
  122. }
  123. void ExecuteCommandList(ID3D12CommandQueue *pQueue, ID3D12CommandList *pList) {
  124. ID3D12CommandList *ppCommandLists[] = { pList };
  125. pQueue->ExecuteCommandLists(1, ppCommandLists);
  126. }
  127. HRESULT SetObjectName(ID3D12Object *pObject, LPCSTR pName) {
  128. if (pObject && pName) {
  129. CA2W WideName(pName);
  130. return pObject->SetName(WideName);
  131. }
  132. return S_FALSE;
  133. }
  134. void WaitForSignal(ID3D12CommandQueue *pCQ, ID3D12Fence *pFence,
  135. HANDLE hFence, UINT64 fenceValue) {
  136. // Signal and increment the fence value.
  137. const UINT64 fence = fenceValue;
  138. CHECK_HR(pCQ->Signal(pFence, fence));
  139. if (pFence->GetCompletedValue() < fenceValue) {
  140. CHECK_HR(pFence->SetEventOnCompletion(fenceValue, hFence));
  141. WaitForSingleObject(hFence, INFINITE);
  142. //CHECK_HR(pCQ->Wait(pFence, fenceValue));
  143. }
  144. }
  145. static void SetupComputeValuePattern(std::vector<uint32_t> &values, size_t count) {
  146. values.resize(count); // one element per dispatch group, in bytes
  147. for (size_t i = 0; i < count; ++i) {
  148. values[i] = (uint32_t)i;
  149. }
  150. }
  151. void MappedData::dump() const {
  152. OutputDebugBytes(m_pData, m_size);
  153. }
  154. void MappedData::reset() {
  155. if (m_pResource != nullptr) {
  156. m_pResource->Unmap(0, nullptr);
  157. m_pResource.Release();
  158. }
  159. m_pData = nullptr;
  160. }
  161. ///////////////////////////////////////////////////////////////////////////////
  162. // Helper class for mapped data.
  163. void MappedData::reset(ID3D12Resource *pResource, UINT32 sizeInBytes) {
  164. reset();
  165. D3D12_RANGE r;
  166. r.Begin = 0;
  167. r.End = sizeInBytes;
  168. CHECK_HR(LogIfLost(pResource->Map(0, &r, &m_pData), pResource));
  169. m_pResource = pResource;
  170. m_size = sizeInBytes;
  171. }
  172. ///////////////////////////////////////////////////////////////////////////////
  173. // ShaderOpTest library implementation.
  174. namespace st {
  175. LPCSTR string_table::insert(LPCSTR pValue) {
  176. std::unordered_set<LPCSTR, HashStr, PredStr>::iterator i = m_values.find(pValue);
  177. if (i == m_values.end()) {
  178. size_t bufSize = strlen(pValue) + 1;
  179. std::vector<char> s;
  180. s.resize(bufSize);
  181. strcpy_s(s.data(), bufSize, pValue);
  182. LPCSTR result = s.data();
  183. m_values.insert(result);
  184. m_strings.push_back(std::move(s));
  185. return result;
  186. }
  187. else {
  188. return *i;
  189. }
  190. }
  191. LPCSTR string_table::insert(LPCWSTR pValue) {
  192. CW2A pValueAnsi(pValue);
  193. return insert(pValueAnsi.m_psz);
  194. }
  195. void CommandListRefs::CreateForDevice(ID3D12Device *pDevice, bool compute) {
  196. D3D12_COMMAND_LIST_TYPE T = compute ? D3D12_COMMAND_LIST_TYPE_COMPUTE
  197. : D3D12_COMMAND_LIST_TYPE_DIRECT;
  198. D3D12_COMMAND_QUEUE_DESC queueDesc = {};
  199. queueDesc.Flags = D3D12_COMMAND_QUEUE_FLAG_NONE;
  200. queueDesc.Type = T;
  201. if (Queue == nullptr) {
  202. CHECK_HR(pDevice->CreateCommandQueue(&queueDesc, IID_PPV_ARGS(&Queue)));
  203. }
  204. CHECK_HR(pDevice->CreateCommandAllocator(T, IID_PPV_ARGS(&Allocator)));
  205. CHECK_HR(pDevice->CreateCommandList(0, T, Allocator, nullptr,
  206. IID_PPV_ARGS(&List)));
  207. }
  208. void ShaderOpTest::CopyBackResources() {
  209. CommandListRefs ResCommandList;
  210. ResCommandList.CreateForDevice(m_pDevice, m_pShaderOp->IsCompute());
  211. ID3D12GraphicsCommandList *pList = ResCommandList.List;
  212. pList->SetName(L"ShaderOpTest Resource ReadBack CommandList");
  213. for (ShaderOpResource &R : m_pShaderOp->Resources) {
  214. if (!R.ReadBack)
  215. continue;
  216. ShaderOpResourceData &D = m_ResourceData[R.Name];
  217. RecordTransitionBarrier(pList, D.Resource, D.ResourceState,
  218. D3D12_RESOURCE_STATE_COPY_SOURCE);
  219. D.ResourceState = D3D12_RESOURCE_STATE_COPY_SOURCE;
  220. D3D12_RESOURCE_DESC &Desc = D.ShaderOpRes->Desc;
  221. if (Desc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER) {
  222. pList->CopyResource(D.ReadBack, D.Resource);
  223. }
  224. else {
  225. UINT64 rowPitch = Desc.Width * GetByteSizeForFormat(Desc.Format);
  226. if (rowPitch % D3D12_TEXTURE_DATA_PITCH_ALIGNMENT)
  227. rowPitch += D3D12_TEXTURE_DATA_PITCH_ALIGNMENT - (rowPitch % D3D12_TEXTURE_DATA_PITCH_ALIGNMENT);
  228. D3D12_PLACED_SUBRESOURCE_FOOTPRINT Footprint;
  229. Footprint.Offset = 0;
  230. Footprint.Footprint = CD3DX12_SUBRESOURCE_FOOTPRINT(Desc.Format, (UINT)Desc.Width, Desc.Height, 1, (UINT)rowPitch);
  231. CD3DX12_TEXTURE_COPY_LOCATION DstLoc(D.ReadBack, Footprint);
  232. CD3DX12_TEXTURE_COPY_LOCATION SrcLoc(D.Resource, 0);
  233. pList->CopyTextureRegion(&DstLoc, 0, 0, 0, &SrcLoc, nullptr);
  234. }
  235. }
  236. pList->Close();
  237. ExecuteCommandList(ResCommandList.Queue, pList);
  238. WaitForSignal(ResCommandList.Queue, m_pFence, m_hFence, m_FenceValue++);
  239. }
  240. void ShaderOpTest::CreateCommandList() {
  241. bool priorQueue = m_CommandList.Queue != nullptr;
  242. m_CommandList.CreateForDevice(m_pDevice, m_pShaderOp->IsCompute());
  243. m_CommandList.Allocator->SetName(L"ShaderOpTest Allocator");
  244. m_CommandList.List->SetName(L"ShaderOpTest CommandList");
  245. if (!priorQueue)
  246. m_CommandList.Queue->SetName(L"ShaderOpTest CommandList");
  247. }
  248. void ShaderOpTest::CreateDescriptorHeaps() {
  249. for (ShaderOpDescriptorHeap &H : m_pShaderOp->DescriptorHeaps) {
  250. CComPtr<ID3D12DescriptorHeap> pHeap;
  251. if (H.Desc.NumDescriptors == 0) {
  252. H.Desc.NumDescriptors = (UINT)H.Descriptors.size();
  253. }
  254. CHECK_HR(m_pDevice->CreateDescriptorHeap(&H.Desc, IID_PPV_ARGS(&pHeap)));
  255. m_DescriptorHeaps.push_back(pHeap);
  256. m_DescriptorHeapsByName[H.Name] = pHeap;
  257. SetObjectName(pHeap, H.Name);
  258. const UINT descriptorSize = m_pDevice->GetDescriptorHandleIncrementSize(H.Desc.Type);
  259. CD3DX12_CPU_DESCRIPTOR_HANDLE cpuHandle(pHeap->GetCPUDescriptorHandleForHeapStart());
  260. CD3DX12_GPU_DESCRIPTOR_HANDLE gpuHandle;
  261. if (H.Desc.Type != D3D12_DESCRIPTOR_HEAP_TYPE_RTV)
  262. gpuHandle = CD3DX12_GPU_DESCRIPTOR_HANDLE(pHeap->GetGPUDescriptorHandleForHeapStart());
  263. for (ShaderOpDescriptor &D : H.Descriptors) {
  264. ShaderOpResource *R = m_pShaderOp->GetResourceByName(D.ResName);
  265. if (R == nullptr) {
  266. LPCSTR DescName = D.Name ? D.Name : "[unnamed descriptor]";
  267. ShaderOpLogFmt(L"Descriptor '%S' references missing resource '%S'", DescName, D.ResName);
  268. CHECK_HR(E_INVALIDARG);
  269. }
  270. ShaderOpResourceData &Data = m_ResourceData[D.ResName];
  271. ShaderOpDescriptorData DData;
  272. DData.Descriptor = &D;
  273. DData.ResData = &Data;
  274. if (0 == _stricmp(D.Kind, "UAV")) {
  275. ID3D12Resource *pCounterResource = nullptr;
  276. if (D.CounterName && *D.CounterName) {
  277. ShaderOpResourceData &CounterData = m_ResourceData[D.CounterName];
  278. pCounterResource = CounterData.Resource;
  279. }
  280. m_pDevice->CreateUnorderedAccessView(Data.Resource, pCounterResource,
  281. &D.UavDesc, cpuHandle);
  282. }
  283. else if (0 == _stricmp(D.Kind, "SRV")) {
  284. D3D12_SHADER_RESOURCE_VIEW_DESC *pSrvDesc = nullptr;
  285. if (D.SrvDescPresent) {
  286. pSrvDesc = &D.SrvDesc;
  287. }
  288. m_pDevice->CreateShaderResourceView(Data.Resource, pSrvDesc, cpuHandle);
  289. }
  290. else if (0 == _stricmp(D.Kind, "RTV")) {
  291. m_pDevice->CreateRenderTargetView(Data.Resource, nullptr, cpuHandle);
  292. }
  293. else if (0 == _stricmp(D.Kind, "CBV")) {
  294. D3D12_CONSTANT_BUFFER_VIEW_DESC cbvDesc;
  295. cbvDesc.BufferLocation = Data.Resource->GetGPUVirtualAddress();
  296. cbvDesc.SizeInBytes = (UINT)Data.Resource->GetDesc().Width;
  297. m_pDevice->CreateConstantBufferView(&cbvDesc, cpuHandle);
  298. }
  299. DData.CPUHandle = cpuHandle;
  300. m_DescriptorData[R->Name] = DData;
  301. cpuHandle = cpuHandle.Offset(descriptorSize);
  302. if (H.Desc.Type != D3D12_DESCRIPTOR_HEAP_TYPE_RTV) {
  303. DData.GPUHandle = gpuHandle;
  304. gpuHandle = gpuHandle.Offset(descriptorSize);
  305. }
  306. }
  307. }
  308. // Create query descriptor heap.
  309. D3D12_QUERY_HEAP_DESC queryHeapDesc;
  310. ZeroMemory(&queryHeapDesc, sizeof(queryHeapDesc));
  311. queryHeapDesc.Count = 1;
  312. queryHeapDesc.Type = D3D12_QUERY_HEAP_TYPE_PIPELINE_STATISTICS;
  313. CHECK_HR(m_pDevice->CreateQueryHeap(&queryHeapDesc, IID_PPV_ARGS(&m_pQueryHeap)));
  314. }
  315. void ShaderOpTest::CreateDevice() {
  316. if (m_pDevice == nullptr) {
  317. const D3D_FEATURE_LEVEL FeatureLevelRequired = D3D_FEATURE_LEVEL_11_0;
  318. CComPtr<IDXGIFactory4> factory;
  319. CComPtr<ID3D12Device> pDevice;
  320. CHECK_HR(CreateDXGIFactory1(IID_PPV_ARGS(&factory)));
  321. if (m_pShaderOp->UseWarpDevice) {
  322. CComPtr<IDXGIAdapter> warpAdapter;
  323. CHECK_HR(factory->EnumWarpAdapter(IID_PPV_ARGS(&warpAdapter)));
  324. CHECK_HR(D3D12CreateDevice(warpAdapter, FeatureLevelRequired,
  325. IID_PPV_ARGS(&pDevice)));
  326. } else {
  327. CComPtr<IDXGIAdapter1> hardwareAdapter;
  328. GetHardwareAdapter(factory, m_pShaderOp->AdapterName, &hardwareAdapter);
  329. if (hardwareAdapter == nullptr) {
  330. CHECK_HR(HRESULT_FROM_WIN32(ERROR_NOT_FOUND));
  331. }
  332. CHECK_HR(D3D12CreateDevice(hardwareAdapter, FeatureLevelRequired,
  333. IID_PPV_ARGS(&pDevice)));
  334. }
  335. m_pDevice.Attach(pDevice.Detach());
  336. m_pDevice->SetName(L"ShaderOpTest Device");
  337. }
  338. m_FenceValue = 1;
  339. CHECK_HR(m_pDevice->CreateFence(0, D3D12_FENCE_FLAG_NONE,
  340. __uuidof(ID3D12Fence), (void **)&m_pFence));
  341. m_pFence->SetName(L"ShaderOpTest Fence");
  342. m_hFence = CreateEvent(nullptr, FALSE, FALSE, nullptr);
  343. if (m_hFence == nullptr) {
  344. AtlThrow(HRESULT_FROM_WIN32(GetLastError()));
  345. }
  346. }
  347. static void InitByteCode(D3D12_SHADER_BYTECODE *pBytecode, ID3D10Blob *pBlob) {
  348. if (pBlob == nullptr) {
  349. pBytecode->BytecodeLength = 0;
  350. pBytecode->pShaderBytecode = nullptr;
  351. }
  352. else {
  353. pBytecode->BytecodeLength = pBlob->GetBufferSize();
  354. pBytecode->pShaderBytecode = pBlob->GetBufferPointer();
  355. }
  356. }
  357. template <typename TKey, typename TValue>
  358. TValue map_get_or_null(const std::map<TKey, TValue> &amap, const TKey& key) {
  359. auto it = amap.find(key);
  360. if (it == amap.end()) return nullptr;
  361. return (*it).second;
  362. }
  363. void ShaderOpTest::CreatePipelineState() {
  364. CreateRootSignature();
  365. CreateShaders();
  366. if (m_pShaderOp->IsCompute()) {
  367. CComPtr<ID3D10Blob> pCS;
  368. pCS = m_Shaders[m_pShaderOp->CS];
  369. D3D12_COMPUTE_PIPELINE_STATE_DESC CDesc;
  370. ZeroMemory(&CDesc, sizeof(CDesc));
  371. CDesc.pRootSignature = m_pRootSignature.p;
  372. InitByteCode(&CDesc.CS, pCS);
  373. CHECK_HR(m_pDevice->CreateComputePipelineState(&CDesc, IID_PPV_ARGS(&m_pPSO)));
  374. }
  375. // Wakanda technology, needs vibranium to work
  376. #if defined(NTDDI_WIN10_VB) && WDK_NTDDI_VERSION >= NTDDI_WIN10_VB
  377. else if (m_pShaderOp->MS) {
  378. // A couple types from a future version of d3dx12.h
  379. typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_MS> CD3DX12_PIPELINE_STATE_STREAM_MS;
  380. typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_AS> CD3DX12_PIPELINE_STATE_STREAM_AS;
  381. struct D3DX12_MESH_SHADER_PIPELINE_STATE_DESC
  382. {
  383. CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE RootSignature;
  384. CD3DX12_PIPELINE_STATE_STREAM_AS AS;
  385. CD3DX12_PIPELINE_STATE_STREAM_MS MS;
  386. CD3DX12_PIPELINE_STATE_STREAM_PS PS;
  387. CD3DX12_PIPELINE_STATE_STREAM_PRIMITIVE_TOPOLOGY PrimitiveTopologyType;
  388. CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK SampleMask;
  389. CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS RTVFormats;
  390. } MDesc = {};
  391. CComPtr<ID3D10Blob> pAS, pMS, pPS;
  392. pAS = map_get_or_null(m_Shaders, m_pShaderOp->AS);
  393. pMS = map_get_or_null(m_Shaders, m_pShaderOp->MS);
  394. pPS = map_get_or_null(m_Shaders, m_pShaderOp->PS);
  395. ZeroMemory(&MDesc, sizeof(MDesc));
  396. MDesc.RootSignature = CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE(m_pRootSignature.p);
  397. D3D12_SHADER_BYTECODE BC;
  398. InitByteCode(&BC, pAS);
  399. MDesc.AS = CD3DX12_PIPELINE_STATE_STREAM_AS(BC);
  400. InitByteCode(&BC, pMS);
  401. MDesc.MS = CD3DX12_PIPELINE_STATE_STREAM_MS(BC);
  402. InitByteCode(&BC, pPS);
  403. MDesc.PS = CD3DX12_PIPELINE_STATE_STREAM_PS(BC);
  404. MDesc.PrimitiveTopologyType = CD3DX12_PIPELINE_STATE_STREAM_PRIMITIVE_TOPOLOGY(m_pShaderOp->PrimitiveTopologyType);
  405. MDesc.SampleMask = CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK(m_pShaderOp->SampleMask);
  406. D3D12_RT_FORMAT_ARRAY RtArray;
  407. ZeroMemory(&RtArray, sizeof(RtArray));
  408. RtArray.NumRenderTargets = (UINT)m_pShaderOp->RenderTargets.size();
  409. for (size_t i = 0; i < RtArray.NumRenderTargets; ++i) {
  410. ShaderOpResource *R = m_pShaderOp->GetResourceByName(m_pShaderOp->RenderTargets[i]);
  411. RtArray.RTFormats[i] = R->Desc.Format;
  412. }
  413. MDesc.RTVFormats = CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS(RtArray);
  414. D3D12_PIPELINE_STATE_STREAM_DESC PDesc = {};
  415. PDesc.SizeInBytes = sizeof(MDesc);
  416. PDesc.pPipelineStateSubobjectStream = &MDesc;
  417. ID3D12Device2 *pDevice2;
  418. CHECK_HR(m_pDevice->QueryInterface(&pDevice2));
  419. pDevice2->CreatePipelineState(&PDesc, IID_PPV_ARGS(&m_pPSO));
  420. }
  421. #endif
  422. else {
  423. CComPtr<ID3D10Blob> pVS, pDS, pHS, pGS, pPS;
  424. pPS = map_get_or_null(m_Shaders, m_pShaderOp->PS);
  425. pVS = map_get_or_null(m_Shaders, m_pShaderOp->VS);
  426. pGS = map_get_or_null(m_Shaders, m_pShaderOp->GS);
  427. pHS = map_get_or_null(m_Shaders, m_pShaderOp->HS);
  428. pDS = map_get_or_null(m_Shaders, m_pShaderOp->DS);
  429. D3D12_GRAPHICS_PIPELINE_STATE_DESC GDesc;
  430. ZeroMemory(&GDesc, sizeof(GDesc));
  431. InitByteCode(&GDesc.VS, pVS);
  432. InitByteCode(&GDesc.HS, pHS);
  433. InitByteCode(&GDesc.DS, pDS);
  434. InitByteCode(&GDesc.GS, pGS);
  435. InitByteCode(&GDesc.PS, pPS);
  436. GDesc.InputLayout.NumElements = (UINT)m_pShaderOp->InputElements.size();
  437. GDesc.InputLayout.pInputElementDescs = m_pShaderOp->InputElements.data();
  438. GDesc.PrimitiveTopologyType = m_pShaderOp->PrimitiveTopologyType;
  439. GDesc.NumRenderTargets = (UINT)m_pShaderOp->RenderTargets.size();
  440. GDesc.SampleMask = m_pShaderOp->SampleMask;
  441. for (size_t i = 0; i < m_pShaderOp->RenderTargets.size(); ++i) {
  442. ShaderOpResource *R = m_pShaderOp->GetResourceByName(m_pShaderOp->RenderTargets[i]);
  443. GDesc.RTVFormats[i] = R->Desc.Format;
  444. }
  445. GDesc.SampleDesc.Count = 1; // TODO: read from file, set from shader operation; also apply to count
  446. GDesc.RasterizerState = CD3DX12_RASTERIZER_DESC(D3D12_DEFAULT); // TODO: read from file, set from op
  447. GDesc.BlendState = CD3DX12_BLEND_DESC(D3D12_DEFAULT); // TODO: read from file, set from op
  448. // TODO: pending values to set
  449. #if 0
  450. D3D12_STREAM_OUTPUT_DESC StreamOutput;
  451. D3D12_DEPTH_STENCIL_DESC DepthStencilState;
  452. D3D12_INDEX_BUFFER_STRIP_CUT_VALUE IBStripCutValue;
  453. DXGI_FORMAT DSVFormat;
  454. UINT NodeMask;
  455. D3D12_PIPELINE_STATE_FLAGS Flags;
  456. #endif
  457. GDesc.pRootSignature = m_pRootSignature.p;
  458. CHECK_HR(m_pDevice->CreateGraphicsPipelineState(&GDesc, IID_PPV_ARGS(&m_pPSO)));
  459. }
  460. }
  461. void ShaderOpTest::CreateResources() {
  462. CommandListRefs ResCommandList;
  463. ResCommandList.CreateForDevice(m_pDevice, true);
  464. ResCommandList.Allocator->SetName(L"ShaderOpTest Resource Creation Allocation");
  465. ResCommandList.Queue->SetName(L"ShaderOpTest Resource Creation Queue");
  466. ResCommandList.List->SetName(L"ShaderOpTest Resource Creation CommandList");
  467. ID3D12GraphicsCommandList *pList = ResCommandList.List.p;
  468. std::vector<CComPtr<ID3D12Resource> > intermediates;
  469. for (ShaderOpResource &R : m_pShaderOp->Resources) {
  470. if (m_ResourceData.count(R.Name) > 0) continue;
  471. // Initialize the upload resource early, to allow a by-name initializer
  472. // to set the desired width.
  473. bool initByName = R.Init && 0 == _stricmp("byname", R.Init);
  474. bool initZero = R.Init && 0 == _stricmp("zero", R.Init);
  475. bool initFromBytes = R.Init && 0 == _stricmp("frombytes", R.Init);
  476. bool hasInit = initByName || initZero || initFromBytes;
  477. bool isBuffer = R.Desc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER;
  478. std::vector<BYTE> values;
  479. if (hasInit) {
  480. if (isBuffer) {
  481. values.resize((size_t)R.Desc.Width);
  482. }
  483. else {
  484. // Probably needs more information.
  485. values.resize((size_t)(R.Desc.Width * R.Desc.Height *
  486. GetByteSizeForFormat(R.Desc.Format)));
  487. }
  488. if (initZero) {
  489. memset(values.data(), 0, values.size());
  490. }
  491. else if (initByName) {
  492. m_InitCallbackFn(R.Name, values, m_pShaderOp);
  493. if (isBuffer) {
  494. R.Desc.Width = values.size();
  495. }
  496. }
  497. else if (initFromBytes) {
  498. values = R.InitBytes;
  499. if (R.Desc.Width == 0) {
  500. if (isBuffer) {
  501. R.Desc.Width = values.size();
  502. }
  503. else if (R.Desc.Dimension == D3D12_RESOURCE_DIMENSION_TEXTURE1D) {
  504. R.Desc.Width = values.size() / GetByteSizeForFormat(R.Desc.Format);
  505. }
  506. }
  507. }
  508. }
  509. if (!R.Desc.MipLevels)
  510. R.Desc.MipLevels = 1;
  511. CComPtr<ID3D12Resource> pResource;
  512. CHECK_HR(m_pDevice->CreateCommittedResource(
  513. &R.HeapProperties, R.HeapFlags, &R.Desc, R.InitialResourceState,
  514. nullptr, IID_PPV_ARGS(&pResource)));
  515. ShaderOpResourceData &D = m_ResourceData[R.Name];
  516. D.ShaderOpRes = &R;
  517. D.Resource = pResource;
  518. D.ResourceState = R.InitialResourceState;
  519. SetObjectName(pResource, R.Name);
  520. if (hasInit) {
  521. CComPtr<ID3D12Resource> pIntermediate;
  522. CD3DX12_HEAP_PROPERTIES upload(D3D12_HEAP_TYPE_UPLOAD);
  523. D3D12_RESOURCE_DESC uploadDesc = R.Desc;
  524. // Calculate size required for intermediate buffer
  525. UINT64 totalBytes;
  526. m_pDevice->GetCopyableFootprints(&uploadDesc, 0, R.Desc.MipLevels, 0, nullptr, nullptr, nullptr, &totalBytes);
  527. if (!isBuffer) {
  528. // Assuming a simple linear layout here.
  529. uploadDesc.Dimension = D3D12_RESOURCE_DIMENSION_BUFFER;
  530. uploadDesc.Width = totalBytes;
  531. uploadDesc.Height = 1;
  532. uploadDesc.MipLevels = 1;
  533. uploadDesc.Format = DXGI_FORMAT_UNKNOWN;
  534. uploadDesc.Layout = D3D12_TEXTURE_LAYOUT_ROW_MAJOR;
  535. }
  536. uploadDesc.Flags = D3D12_RESOURCE_FLAG_NONE;
  537. CHECK_HR(m_pDevice->CreateCommittedResource(
  538. &upload, D3D12_HEAP_FLAG_NONE, &uploadDesc,
  539. D3D12_RESOURCE_STATE_GENERIC_READ, nullptr,
  540. IID_PPV_ARGS(&pIntermediate)));
  541. intermediates.push_back(pIntermediate);
  542. char uploadObjectName[128];
  543. if (R.Name && SUCCEEDED(StringCchPrintfA(
  544. uploadObjectName, _countof(uploadObjectName),
  545. "Upload resource for %s", R.Name))) {
  546. SetObjectName(pIntermediate, uploadObjectName);
  547. }
  548. D3D12_SUBRESOURCE_DATA transferData[16];
  549. UINT width = (UINT)R.Desc.Width;
  550. UINT height = R.Desc.Height;
  551. UINT pixelSize = GetByteSizeForFormat(R.Desc.Format);
  552. BYTE *data = values.data();
  553. VERIFY_IS_TRUE(R.Desc.MipLevels <= 16);
  554. for (UINT i = 0; i < R.Desc.MipLevels; i++) {
  555. if(!height) height = 1;
  556. if(!width) width = 1;
  557. transferData[i].pData = data;
  558. transferData[i].RowPitch = width*pixelSize;
  559. transferData[i].SlicePitch = width*height*pixelSize;
  560. data += width*height*pixelSize;
  561. height >>= 1;
  562. width >>= 1;
  563. }
  564. UpdateSubresources<16>(pList, pResource.p, pIntermediate.p, 0, 0, R.Desc.MipLevels,
  565. transferData);
  566. }
  567. if (R.ReadBack) {
  568. CComPtr<ID3D12Resource> pReadbackResource;
  569. CD3DX12_HEAP_PROPERTIES readback(D3D12_HEAP_TYPE_READBACK);
  570. UINT64 width = R.Desc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER
  571. ? R.Desc.Width
  572. : (R.Desc.Height * R.Desc.Width *
  573. GetByteSizeForFormat(R.Desc.Format));
  574. CD3DX12_RESOURCE_DESC readbackDesc(CD3DX12_RESOURCE_DESC::Buffer(width));
  575. readbackDesc.Flags = D3D12_RESOURCE_FLAG_NONE;
  576. CHECK_HR(m_pDevice->CreateCommittedResource(
  577. &readback, D3D12_HEAP_FLAG_NONE, &readbackDesc,
  578. D3D12_RESOURCE_STATE_COPY_DEST, nullptr,
  579. IID_PPV_ARGS(&pReadbackResource)));
  580. D.ReadBack = pReadbackResource;
  581. char readbackObjectName[128];
  582. if (R.Name && SUCCEEDED(StringCchPrintfA(
  583. readbackObjectName, _countof(readbackObjectName),
  584. "Readback resource for %s", R.Name))) {
  585. SetObjectName(pReadbackResource, readbackObjectName);
  586. }
  587. }
  588. if (R.TransitionTo != D.ResourceState) {
  589. RecordTransitionBarrier(pList, D.Resource, D.ResourceState,
  590. R.TransitionTo);
  591. D.ResourceState = R.TransitionTo;
  592. }
  593. }
  594. // Create a buffer to receive query results.
  595. {
  596. CComPtr<ID3D12Resource> pReadbackResource;
  597. CD3DX12_HEAP_PROPERTIES readback(D3D12_HEAP_TYPE_READBACK);
  598. CD3DX12_RESOURCE_DESC readbackDesc(CD3DX12_RESOURCE_DESC::Buffer(sizeof(D3D12_QUERY_DATA_PIPELINE_STATISTICS)));
  599. CHECK_HR(m_pDevice->CreateCommittedResource(
  600. &readback, D3D12_HEAP_FLAG_NONE, &readbackDesc,
  601. D3D12_RESOURCE_STATE_COPY_DEST, nullptr,
  602. IID_PPV_ARGS(&m_pQueryBuffer)));
  603. SetObjectName(m_pQueryBuffer, "Query Pipeline Readback Buffer");
  604. }
  605. CHECK_HR(pList->Close());
  606. ExecuteCommandList(ResCommandList.Queue, pList);
  607. WaitForSignal(ResCommandList.Queue, m_pFence, m_hFence, m_FenceValue++);
  608. }
  609. void ShaderOpTest::CreateRootSignature() {
  610. if (m_pShaderOp->RootSignature == nullptr) {
  611. AtlThrow(E_INVALIDARG);
  612. }
  613. CComPtr<ID3DBlob> pCode;
  614. CComPtr<ID3DBlob> pRootSignatureBlob;
  615. CComPtr<ID3DBlob> pError;
  616. std::string sQuoted;
  617. sQuoted.reserve(2 + strlen(m_pShaderOp->RootSignature) + 1);
  618. sQuoted.append("\"");
  619. sQuoted.append(m_pShaderOp->RootSignature);
  620. sQuoted.append("\"");
  621. char *ch = (char *)sQuoted.data();
  622. while (*ch) {
  623. if (*ch == '\r' || *ch == '\n') *ch = ' ';
  624. ++ch;
  625. }
  626. D3D_SHADER_MACRO M[2] = {
  627. { "RootSigVal", sQuoted.c_str() },
  628. { nullptr, nullptr }
  629. };
  630. HRESULT hr = D3DCompile(nullptr, 0, "RootSigShader", M, nullptr, sQuoted.c_str(),
  631. "rootsig_1_0", 0, 0, &pCode, &pError);
  632. if (FAILED(hr) && pError != nullptr) {
  633. ShaderOpLogFmt(L"Failed to compile root signature:\r\n%*S",
  634. (int)pError->GetBufferSize(),
  635. (LPCSTR)pError->GetBufferPointer());
  636. }
  637. CHECK_HR(hr);
  638. CHECK_HR(D3DGetBlobPart(pCode->GetBufferPointer(), pCode->GetBufferSize(),
  639. D3D_BLOB_ROOT_SIGNATURE, 0, &pRootSignatureBlob));
  640. CHECK_HR(m_pDevice->CreateRootSignature(
  641. 0, pRootSignatureBlob->GetBufferPointer(),
  642. pRootSignatureBlob->GetBufferSize(), IID_PPV_ARGS(&m_pRootSignature)));
  643. }
  644. static bool TargetUsesDxil(LPCSTR pText) {
  645. return (strlen(pText) > 3) && pText[3] >= '6'; // xx_6xx
  646. }
  647. static void splitWStringIntoVectors(LPWSTR str, wchar_t delim, std::vector<LPWSTR> &list) {
  648. if (str) {
  649. LPWSTR cur = str;
  650. list.push_back(cur);
  651. while (*cur != L'\0') {
  652. if (*cur == delim) {
  653. list.push_back(cur+1);
  654. *(cur) = L'\0';
  655. }
  656. cur++;
  657. }
  658. }
  659. }
  660. void ShaderOpTest::CreateShaders() {
  661. for (ShaderOpShader &S : m_pShaderOp->Shaders) {
  662. CComPtr<ID3DBlob> pCode;
  663. HRESULT hr = S_OK;
  664. LPCSTR pText = m_pShaderOp->GetShaderText(&S);
  665. if (S.Compiled) {
  666. int textLen = (int)strlen(pText);
  667. int decodedLen = Base64DecodeGetRequiredLength(textLen);
  668. // Length is an approximation, so we can't creat the final blob yet.
  669. std::vector<BYTE> decoded;
  670. decoded.resize(decodedLen);
  671. if (!Base64Decode(pText, textLen, decoded.data(), &decodedLen)) {
  672. ShaderOpLogFmt(L"Failed to decode compiled shader: %S\r\n", S.Name);
  673. CHECK_HR(E_FAIL);
  674. }
  675. // decodedLen should have the correct size now.
  676. CHECK_HR(D3DCreateBlob(decodedLen, &pCode));
  677. memcpy(pCode->GetBufferPointer(), decoded.data(), decodedLen);
  678. }
  679. else if (TargetUsesDxil(S.Target)) {
  680. CComPtr<IDxcCompiler> pCompiler;
  681. CComPtr<IDxcLibrary> pLibrary;
  682. CComPtr<IDxcBlobEncoding> pTextBlob;
  683. CComPtr<IDxcOperationResult> pResult;
  684. CA2W nameW(S.Name, CP_UTF8);
  685. CA2W entryPointW(S.EntryPoint, CP_UTF8);
  686. CA2W targetW(S.Target, CP_UTF8);
  687. CA2W argumentsW(S.Arguments, CP_UTF8);
  688. std::vector<LPWSTR> argumentsWList;
  689. splitWStringIntoVectors(argumentsW, L' ', argumentsWList);
  690. HRESULT resultCode;
  691. CHECK_HR(m_pDxcSupport->CreateInstance(CLSID_DxcLibrary, &pLibrary));
  692. CHECK_HR(pLibrary->CreateBlobWithEncodingFromPinned(
  693. pText, (UINT32)strlen(pText), CP_UTF8, &pTextBlob));
  694. CHECK_HR(m_pDxcSupport->CreateInstance(CLSID_DxcCompiler, &pCompiler));
  695. CHECK_HR(pCompiler->Compile(pTextBlob, nameW, entryPointW, targetW,
  696. (LPCWSTR *)argumentsWList.data(), (UINT32)argumentsWList.size(),
  697. nullptr, 0,
  698. nullptr, &pResult));
  699. CHECK_HR(pResult->GetStatus(&resultCode));
  700. if (FAILED(resultCode)) {
  701. CComPtr<IDxcBlobEncoding> errors;
  702. CHECK_HR(pResult->GetErrorBuffer(&errors));
  703. ShaderOpLogFmt(L"Failed to compile shader: %*S\r\n",
  704. (int)errors->GetBufferSize(),
  705. errors->GetBufferPointer());
  706. }
  707. CHECK_HR(resultCode);
  708. CHECK_HR(pResult->GetResult((IDxcBlob **)&pCode));
  709. #if 0 // use the following code to test compiled shader
  710. CComPtr<IDxcBlob> pCode;
  711. CHECK_HR(pResult->GetResult(&pCode));
  712. CComPtr<IDxcBlobEncoding> pBlob;
  713. CHECK_HR(pCompiler->Disassemble((IDxcBlob *)pCode, (IDxcBlobEncoding **)&pBlob));
  714. CComPtr<IDxcBlobEncoding> pUtf16Blob;
  715. pLibrary->GetBlobAsUtf16(pBlob, &pUtf16Blob);
  716. hlsl_test::LogCommentFmt(L"%*s", (int)pUtf16Blob->GetBufferSize() / 2, (LPCWSTR)pUtf16Blob->GetBufferPointer());
  717. #endif
  718. } else {
  719. CComPtr<ID3DBlob> pError;
  720. hr = D3DCompile(pText, strlen(pText), S.Name, nullptr, nullptr,
  721. S.EntryPoint, S.Target, 0, 0, &pCode, &pError);
  722. if (FAILED(hr) && pError != nullptr) {
  723. ShaderOpLogFmt(L"%*S\r\n", (int)pError->GetBufferSize(),
  724. ((LPCSTR)pError->GetBufferPointer()));
  725. }
  726. }
  727. CHECK_HR(hr);
  728. m_Shaders[S.Name] = pCode;
  729. }
  730. }
  731. void ShaderOpTest::GetPipelineStats(D3D12_QUERY_DATA_PIPELINE_STATISTICS *pStats) {
  732. MappedData M;
  733. M.reset(m_pQueryBuffer, sizeof(*pStats));
  734. memcpy(pStats, M.data(), sizeof(*pStats));
  735. }
  736. void ShaderOpTest::GetReadBackData(LPCSTR pResourceName, MappedData *pData) {
  737. pResourceName = m_pShaderOp->Strings.insert(pResourceName); // Unique
  738. ShaderOpResourceData &D = m_ResourceData.at(pResourceName);
  739. D3D12_RESOURCE_DESC Desc = D.ReadBack->GetDesc();
  740. UINT32 sizeInBytes = (UINT32)Desc.Width;
  741. pData->reset(D.ReadBack, sizeInBytes);
  742. }
  743. static void SetDescriptorHeaps(ID3D12GraphicsCommandList *pList,
  744. std::vector<ID3D12DescriptorHeap *> &heaps) {
  745. if (heaps.empty())
  746. return;
  747. std::vector<ID3D12DescriptorHeap *> localHeaps;
  748. for (auto &H : heaps) {
  749. if (H->GetDesc().Flags & D3D12_DESCRIPTOR_HEAP_FLAG_SHADER_VISIBLE) {
  750. localHeaps.push_back(H);
  751. }
  752. }
  753. if (!localHeaps.empty())
  754. pList->SetDescriptorHeaps((UINT)localHeaps.size(), localHeaps.data());
  755. }
  756. void ShaderOpTest::RunCommandList() {
  757. ID3D12GraphicsCommandList *pList = m_CommandList.List.p;
  758. if (m_pShaderOp->IsCompute()) {
  759. pList->SetPipelineState(m_pPSO);
  760. pList->SetComputeRootSignature(m_pRootSignature);
  761. SetDescriptorHeaps(pList, m_DescriptorHeaps);
  762. SetRootValues(pList, m_pShaderOp->IsCompute());
  763. pList->Dispatch(m_pShaderOp->DispatchX, m_pShaderOp->DispatchY,
  764. m_pShaderOp->DispatchZ);
  765. } else {
  766. pList->SetPipelineState(m_pPSO);
  767. pList->SetGraphicsRootSignature(m_pRootSignature);
  768. SetDescriptorHeaps(pList, m_DescriptorHeaps);
  769. SetRootValues(pList, m_pShaderOp->IsCompute());
  770. if (!m_pShaderOp->RenderTargets.empty()) {
  771. // Use the first render target to set up the viewport and scissors.
  772. ShaderOpResource *R = m_pShaderOp->GetResourceByName(m_pShaderOp->RenderTargets[0]);
  773. D3D12_VIEWPORT viewport;
  774. D3D12_RECT scissorRect;
  775. memset(&viewport, 0, sizeof(viewport));
  776. viewport.Height = (FLOAT)R->Desc.Height;
  777. viewport.Width = (FLOAT)R->Desc.Width;
  778. viewport.MaxDepth = 1.0f;
  779. memset(&scissorRect, 0, sizeof(scissorRect));
  780. scissorRect.right = (LONG)viewport.Width;
  781. scissorRect.bottom = (LONG)viewport.Height;
  782. pList->RSSetViewports(1, &viewport);
  783. pList->RSSetScissorRects(1, &scissorRect);
  784. }
  785. // Indicate that the buffers will be used as render targets.
  786. D3D12_CPU_DESCRIPTOR_HANDLE rtvHandles[8];
  787. UINT rtvHandleCount = (UINT)m_pShaderOp->RenderTargets.size();
  788. for (size_t i = 0; i < rtvHandleCount; ++i) {
  789. auto &rt = m_pShaderOp->RenderTargets[i];
  790. ShaderOpDescriptorData &DData = m_DescriptorData[rt];
  791. rtvHandles[i] = DData.CPUHandle;
  792. RecordTransitionBarrier(pList, DData.ResData->Resource,
  793. DData.ResData->ResourceState,
  794. D3D12_RESOURCE_STATE_RENDER_TARGET);
  795. DData.ResData->ResourceState = D3D12_RESOURCE_STATE_RENDER_TARGET;
  796. }
  797. pList->OMSetRenderTargets(rtvHandleCount, rtvHandles, FALSE, nullptr);
  798. const float ClearColor[4] = { 0.0f, 0.2f, 0.4f, 1.0f };
  799. pList->ClearRenderTargetView(rtvHandles[0], ClearColor, 0, nullptr);
  800. #if defined(NTDDI_WIN10_VB) && WDK_NTDDI_VERSION >= NTDDI_WIN10_VB
  801. if (m_pShaderOp->MS) {
  802. ID3D12GraphicsCommandList6 *pList6 = m_CommandList.List.p;
  803. pList6->BeginQuery(m_pQueryHeap, D3D12_QUERY_TYPE_PIPELINE_STATISTICS, 0);
  804. pList6->DispatchMesh(1, 1, 1);
  805. pList6->EndQuery(m_pQueryHeap, D3D12_QUERY_TYPE_PIPELINE_STATISTICS, 0);
  806. pList6->ResolveQueryData(m_pQueryHeap, D3D12_QUERY_TYPE_PIPELINE_STATISTICS,
  807. 0, 1, m_pQueryBuffer, 0);
  808. } else
  809. #endif
  810. {
  811. // TODO: set all of this from m_pShaderOp.
  812. ShaderOpResourceData &VBufferData = this->m_ResourceData[m_pShaderOp->Strings.insert("VBuffer")];
  813. D3D_PRIMITIVE_TOPOLOGY topology = D3D_PRIMITIVE_TOPOLOGY_UNDEFINED;
  814. for (ShaderOpResource &resource : m_pShaderOp->Resources) {
  815. if (_strcmpi(resource.Name, "VBuffer") == 0) {
  816. topology = resource.PrimitiveTopology;
  817. break;
  818. }
  819. }
  820. pList->IASetPrimitiveTopology(topology);
  821. // Calculate the stride in bytes from the inputs, assuming linear & contiguous.
  822. UINT strideInBytes = 0;
  823. for (auto && IE : m_pShaderOp->InputElements) {
  824. strideInBytes += GetByteSizeForFormat(IE.Format);
  825. }
  826. D3D12_VERTEX_BUFFER_VIEW vertexBufferView;
  827. vertexBufferView.BufferLocation = VBufferData.Resource->GetGPUVirtualAddress();
  828. vertexBufferView.StrideInBytes = strideInBytes;
  829. vertexBufferView.SizeInBytes = (UINT)VBufferData.ShaderOpRes->Desc.Width;
  830. pList->IASetVertexBuffers(0, 1, &vertexBufferView);
  831. UINT vertexCount = vertexBufferView.SizeInBytes / vertexBufferView.StrideInBytes;
  832. UINT instanceCount = 1;
  833. UINT vertexCountPerInstance = vertexCount / instanceCount;
  834. pList->BeginQuery(m_pQueryHeap, D3D12_QUERY_TYPE_PIPELINE_STATISTICS, 0);
  835. pList->DrawInstanced(vertexCountPerInstance, instanceCount, 0, 0);
  836. pList->EndQuery(m_pQueryHeap, D3D12_QUERY_TYPE_PIPELINE_STATISTICS, 0);
  837. pList->ResolveQueryData(m_pQueryHeap, D3D12_QUERY_TYPE_PIPELINE_STATISTICS,
  838. 0, 1, m_pQueryBuffer, 0);
  839. }
  840. }
  841. CHECK_HR(pList->Close());
  842. ExecuteCommandList(m_CommandList.Queue, pList);
  843. WaitForSignal(m_CommandList.Queue, m_pFence, m_hFence, m_FenceValue++);
  844. }
  845. void ShaderOpTest::RunShaderOp(ShaderOp *pShaderOp) {
  846. m_pShaderOp = pShaderOp;
  847. CreateDevice();
  848. CreateResources();
  849. CreateDescriptorHeaps();
  850. CreatePipelineState();
  851. CreateCommandList();
  852. RunCommandList();
  853. CopyBackResources();
  854. }
  855. void ShaderOpTest::RunShaderOp(std::shared_ptr<ShaderOp> ShaderOp) {
  856. m_OrigShaderOp = ShaderOp;
  857. RunShaderOp(m_OrigShaderOp.get());
  858. }
  859. void ShaderOpTest::SetRootValues(ID3D12GraphicsCommandList *pList,
  860. bool isCompute) {
  861. for (size_t i = 0; i < m_pShaderOp->RootValues.size(); ++i) {
  862. ShaderOpRootValue &V = m_pShaderOp->RootValues[i];
  863. UINT idx = V.Index == 0 ? (UINT)i : V.Index;
  864. if (V.ResName) {
  865. auto r_it = m_ResourceData.find(V.ResName);
  866. if (r_it == m_ResourceData.end()) {
  867. ShaderOpLogFmt(L"Root value #%u refers to missing resource %S", (unsigned)i, V.ResName);
  868. CHECK_HR(E_INVALIDARG);
  869. }
  870. // Issue a warning for trying to bind textures (GPU address will return null)
  871. ShaderOpResourceData &D = r_it->second;
  872. ID3D12Resource *pRes = D.Resource;
  873. if (isCompute) {
  874. switch (D.ShaderOpRes->TransitionTo) {
  875. case D3D12_RESOURCE_STATE_VERTEX_AND_CONSTANT_BUFFER:
  876. pList->SetComputeRootConstantBufferView(idx,
  877. pRes->GetGPUVirtualAddress());
  878. break;
  879. case D3D12_RESOURCE_STATE_UNORDERED_ACCESS:
  880. pList->SetComputeRootUnorderedAccessView(idx,
  881. pRes->GetGPUVirtualAddress());
  882. break;
  883. case D3D12_RESOURCE_STATE_NON_PIXEL_SHADER_RESOURCE:
  884. default:
  885. pList->SetComputeRootShaderResourceView(idx,
  886. pRes->GetGPUVirtualAddress());
  887. break;
  888. }
  889. }
  890. else {
  891. switch (D.ShaderOpRes->TransitionTo) {
  892. case D3D12_RESOURCE_STATE_VERTEX_AND_CONSTANT_BUFFER:
  893. pList->SetGraphicsRootConstantBufferView(idx,
  894. pRes->GetGPUVirtualAddress());
  895. break;
  896. case D3D12_RESOURCE_STATE_UNORDERED_ACCESS:
  897. pList->SetGraphicsRootUnorderedAccessView(idx,
  898. pRes->GetGPUVirtualAddress());
  899. break;
  900. case D3D12_RESOURCE_STATE_NON_PIXEL_SHADER_RESOURCE:
  901. default:
  902. pList->SetGraphicsRootShaderResourceView(idx,
  903. pRes->GetGPUVirtualAddress());
  904. break;
  905. }
  906. }
  907. }
  908. else if (V.HeapName) {
  909. D3D12_GPU_DESCRIPTOR_HANDLE heapBase(m_DescriptorHeapsByName[V.HeapName]->GetGPUDescriptorHandleForHeapStart());
  910. if (isCompute) {
  911. pList->SetComputeRootDescriptorTable(idx, heapBase);
  912. }
  913. else {
  914. pList->SetGraphicsRootDescriptorTable(idx, heapBase);
  915. }
  916. }
  917. }
  918. }
  919. void ShaderOpTest::SetDevice(ID3D12Device *pDevice) {
  920. m_pDevice = pDevice;
  921. }
  922. void ShaderOpTest::SetDxcSupport(dxc::DxcDllSupport *pDxcSupport) {
  923. m_pDxcSupport = pDxcSupport;
  924. }
  925. void ShaderOpTest::SetInitCallback(TInitCallbackFn InitCallbackFn) {
  926. m_InitCallbackFn = InitCallbackFn;
  927. }
  928. void ShaderOpTest::SetupRenderTarget(ShaderOp *pShaderOp, ID3D12Device *pDevice,
  929. ID3D12CommandQueue *pCommandQueue,
  930. ID3D12Resource *pRenderTarget) {
  931. SetDevice(pDevice);
  932. m_CommandList.Queue = pCommandQueue;
  933. // Simplification - add the render target name if missing, set it up 'by hand' if not.
  934. if (pShaderOp->RenderTargets.empty()) {
  935. pShaderOp->RenderTargets.push_back(pShaderOp->Strings.insert("RTarget"));
  936. ShaderOpResource R;
  937. ZeroMemory(&R, sizeof(R));
  938. R.Desc = pRenderTarget->GetDesc();
  939. R.Name = pShaderOp->Strings.insert("RTarget");
  940. R.HeapFlags = D3D12_HEAP_FLAG_NONE;
  941. R.Init = nullptr;
  942. R.InitialResourceState = D3D12_RESOURCE_STATE_PRESENT;
  943. R.ReadBack = FALSE;
  944. pShaderOp->Resources.push_back(R);
  945. ShaderOpResourceData &D = m_ResourceData[R.Name];
  946. D.ShaderOpRes = &pShaderOp->Resources.back();
  947. D.Resource = pRenderTarget;
  948. D.ResourceState = R.InitialResourceState;
  949. }
  950. // Create a render target heap to put this in.
  951. ShaderOpDescriptorHeap *pRtvHeap = pShaderOp->GetDescriptorHeapByName("RtvHeap");
  952. if (pRtvHeap == nullptr) {
  953. ShaderOpDescriptorHeap H;
  954. ZeroMemory(&H, sizeof(H));
  955. H.Name = pShaderOp->Strings.insert("RtvHeap");
  956. H.Desc.NumDescriptors = 1;
  957. H.Desc.Type = D3D12_DESCRIPTOR_HEAP_TYPE_RTV;
  958. pShaderOp->DescriptorHeaps.push_back(H);
  959. pRtvHeap = &pShaderOp->DescriptorHeaps.back();
  960. }
  961. if (pRtvHeap->Descriptors.empty()) {
  962. ShaderOpDescriptor D;
  963. ZeroMemory(&D, sizeof(D));
  964. D.Name = pShaderOp->Strings.insert("RTarget");
  965. D.ResName = D.Name;
  966. D.Kind = pShaderOp->Strings.insert("RTV");
  967. pRtvHeap->Descriptors.push_back(D);
  968. }
  969. }
  970. void ShaderOpTest::PresentRenderTarget(ShaderOp *pShaderOp,
  971. ID3D12CommandQueue *pCommandQueue,
  972. ID3D12Resource *pRenderTarget) {
  973. UNREFERENCED_PARAMETER(pShaderOp);
  974. CommandListRefs ResCommandList;
  975. ResCommandList.Queue = pCommandQueue;
  976. ResCommandList.CreateForDevice(m_pDevice, m_pShaderOp->IsCompute());
  977. ID3D12GraphicsCommandList *pList = ResCommandList.List;
  978. pList->SetName(L"ShaderOpTest Resource Present CommandList");
  979. RecordTransitionBarrier(pList, pRenderTarget,
  980. D3D12_RESOURCE_STATE_RENDER_TARGET,
  981. D3D12_RESOURCE_STATE_PRESENT);
  982. pList->Close();
  983. ExecuteCommandList(ResCommandList.Queue, pList);
  984. WaitForSignal(ResCommandList.Queue, m_pFence, m_hFence, m_FenceValue++);
  985. }
  986. ShaderOp *ShaderOpSet::GetShaderOp(LPCSTR pName) {
  987. for (auto &S : ShaderOps) {
  988. if (S->Name && 0 == _stricmp(pName, S->Name)) {
  989. return S.get();
  990. }
  991. }
  992. return nullptr;
  993. }
  994. ///////////////////////////////////////////////////////////////////////////////
  995. // ShaderOpTest library implementation for deserialization.
  996. #pragma region Parsing support
  997. // Use this class to initialize a ShaderOp object from an XML document.
  998. class ShaderOpParser {
  999. private:
  1000. string_table *m_pStrings;
  1001. bool ReadAtElementName(IXmlReader *pReader, LPCWSTR pName);
  1002. HRESULT ReadAttrStr(IXmlReader *pReader, LPCWSTR pAttrName, LPCSTR *ppValue);
  1003. HRESULT ReadAttrBOOL(IXmlReader *pReader, LPCWSTR pAttrName, BOOL *pValue, BOOL defaultValue = FALSE);
  1004. HRESULT ReadAttrUINT64(IXmlReader *pReader, LPCWSTR pAttrName, UINT64 *pValue, UINT64 defaultValue = 0);
  1005. HRESULT ReadAttrUINT16(IXmlReader *pReader, LPCWSTR pAttrName, UINT16 *pValue, UINT16 defaultValue = 0);
  1006. HRESULT ReadAttrUINT(IXmlReader *pReader, LPCWSTR pAttrName, UINT *pValue, UINT defaultValue = 0);
  1007. void ReadElementContentStr(IXmlReader *pReader, LPCSTR *ppValue);
  1008. void ParseDescriptor(IXmlReader *pReader, ShaderOpDescriptor *pDesc);
  1009. void ParseDescriptorHeap(IXmlReader *pReader, ShaderOpDescriptorHeap *pHeap);
  1010. void ParseInputElement(IXmlReader *pReader, D3D12_INPUT_ELEMENT_DESC *pInputElement);
  1011. void ParseInputElements(IXmlReader *pReader, std::vector<D3D12_INPUT_ELEMENT_DESC> *pInputElements);
  1012. void ParseRenderTargets(IXmlReader *pReader, std::vector<LPCSTR> *pRenderTargets);
  1013. void ParseRootValue(IXmlReader *pReader, ShaderOpRootValue *pRootValue);
  1014. void ParseRootValues(IXmlReader *pReader, std::vector<ShaderOpRootValue> *pRootValues);
  1015. void ParseResource(IXmlReader *pReader, ShaderOpResource *pResource);
  1016. void ParseShader(IXmlReader *pReader, ShaderOpShader *pShader);
  1017. public:
  1018. void ParseShaderOpSet(IStream *pStream, ShaderOpSet *pShaderOpSet);
  1019. void ParseShaderOpSet(IXmlReader *pReader, ShaderOpSet *pShaderOpSet);
  1020. void ParseShaderOp(IXmlReader *pReader, ShaderOp *pShaderOp);
  1021. };
  1022. void ParseShaderOpSetFromStream(IStream *pStream, st::ShaderOpSet *pShaderOpSet) {
  1023. ShaderOpParser parser;
  1024. parser.ParseShaderOpSet(pStream, pShaderOpSet);
  1025. }
  1026. void ParseShaderOpSetFromXml(IXmlReader *pReader, st::ShaderOpSet *pShaderOpSet) {
  1027. ShaderOpParser parser;
  1028. parser.ParseShaderOpSet(pReader, pShaderOpSet);
  1029. }
  1030. enum class ParserEnumKind {
  1031. INPUT_CLASSIFICATION,
  1032. DXGI_FORMAT,
  1033. HEAP_TYPE,
  1034. CPU_PAGE_PROPERTY,
  1035. MEMORY_POOL,
  1036. RESOURCE_DIMENSION,
  1037. TEXTURE_LAYOUT,
  1038. RESOURCE_FLAG,
  1039. HEAP_FLAG,
  1040. RESOURCE_STATE,
  1041. DESCRIPTOR_HEAP_TYPE,
  1042. DESCRIPTOR_HEAP_FLAG,
  1043. SRV_DIMENSION,
  1044. UAV_DIMENSION,
  1045. PRIMITIVE_TOPOLOGY,
  1046. PRIMITIVE_TOPOLOGY_TYPE
  1047. };
  1048. struct ParserEnumValue {
  1049. LPCWSTR Name;
  1050. UINT Value;
  1051. };
  1052. struct ParserEnumTable {
  1053. size_t ValueCount;
  1054. const ParserEnumValue *Values;
  1055. ParserEnumKind Kind;
  1056. };
  1057. static const ParserEnumValue INPUT_CLASSIFICATION_TABLE[] = {
  1058. { L"INSTANCE", D3D12_INPUT_CLASSIFICATION_PER_INSTANCE_DATA },
  1059. { L"VERTEX", D3D12_INPUT_CLASSIFICATION_PER_VERTEX_DATA }
  1060. };
  1061. static const ParserEnumValue DXGI_FORMAT_TABLE[] = {
  1062. { L"UNKNOWN", DXGI_FORMAT_UNKNOWN },
  1063. { L"R32G32B32A32_TYPELESS", DXGI_FORMAT_R32G32B32A32_TYPELESS },
  1064. { L"R32G32B32A32_FLOAT", DXGI_FORMAT_R32G32B32A32_FLOAT },
  1065. { L"R32G32B32A32_UINT", DXGI_FORMAT_R32G32B32A32_UINT },
  1066. { L"R32G32B32A32_SINT", DXGI_FORMAT_R32G32B32A32_SINT },
  1067. { L"R32G32B32_TYPELESS", DXGI_FORMAT_R32G32B32_TYPELESS },
  1068. { L"R32G32B32_FLOAT", DXGI_FORMAT_R32G32B32_FLOAT },
  1069. { L"R32G32B32_UINT", DXGI_FORMAT_R32G32B32_UINT },
  1070. { L"R32G32B32_SINT", DXGI_FORMAT_R32G32B32_SINT },
  1071. { L"R16G16B16A16_TYPELESS", DXGI_FORMAT_R16G16B16A16_TYPELESS },
  1072. { L"R16G16B16A16_FLOAT", DXGI_FORMAT_R16G16B16A16_FLOAT },
  1073. { L"R16G16B16A16_UNORM", DXGI_FORMAT_R16G16B16A16_UNORM },
  1074. { L"R16G16B16A16_UINT", DXGI_FORMAT_R16G16B16A16_UINT },
  1075. { L"R16G16B16A16_SNORM", DXGI_FORMAT_R16G16B16A16_SNORM },
  1076. { L"R16G16B16A16_SINT", DXGI_FORMAT_R16G16B16A16_SINT },
  1077. { L"R32G32_TYPELESS", DXGI_FORMAT_R32G32_TYPELESS },
  1078. { L"R32G32_FLOAT", DXGI_FORMAT_R32G32_FLOAT },
  1079. { L"R32G32_UINT", DXGI_FORMAT_R32G32_UINT },
  1080. { L"R32G32_SINT", DXGI_FORMAT_R32G32_SINT },
  1081. { L"R32G8X24_TYPELESS", DXGI_FORMAT_R32G8X24_TYPELESS },
  1082. { L"D32_FLOAT_S8X24_UINT", DXGI_FORMAT_D32_FLOAT_S8X24_UINT },
  1083. { L"R32_FLOAT_X8X24_TYPELESS", DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS },
  1084. { L"X32_TYPELESS_G8X24_UINT", DXGI_FORMAT_X32_TYPELESS_G8X24_UINT },
  1085. { L"R10G10B10A2_TYPELESS", DXGI_FORMAT_R10G10B10A2_TYPELESS },
  1086. { L"R10G10B10A2_UNORM", DXGI_FORMAT_R10G10B10A2_UNORM },
  1087. { L"R10G10B10A2_UINT", DXGI_FORMAT_R10G10B10A2_UINT },
  1088. { L"R11G11B10_FLOAT", DXGI_FORMAT_R11G11B10_FLOAT },
  1089. { L"R8G8B8A8_TYPELESS", DXGI_FORMAT_R8G8B8A8_TYPELESS },
  1090. { L"R8G8B8A8_UNORM", DXGI_FORMAT_R8G8B8A8_UNORM },
  1091. { L"R8G8B8A8_UNORM_SRGB", DXGI_FORMAT_R8G8B8A8_UNORM_SRGB },
  1092. { L"R8G8B8A8_UINT", DXGI_FORMAT_R8G8B8A8_UINT },
  1093. { L"R8G8B8A8_SNORM", DXGI_FORMAT_R8G8B8A8_SNORM },
  1094. { L"R8G8B8A8_SINT", DXGI_FORMAT_R8G8B8A8_SINT },
  1095. { L"R16G16_TYPELESS", DXGI_FORMAT_R16G16_TYPELESS },
  1096. { L"R16G16_FLOAT", DXGI_FORMAT_R16G16_FLOAT },
  1097. { L"R16G16_UNORM", DXGI_FORMAT_R16G16_UNORM },
  1098. { L"R16G16_UINT", DXGI_FORMAT_R16G16_UINT },
  1099. { L"R16G16_SNORM", DXGI_FORMAT_R16G16_SNORM },
  1100. { L"R16G16_SINT", DXGI_FORMAT_R16G16_SINT },
  1101. { L"R32_TYPELESS", DXGI_FORMAT_R32_TYPELESS },
  1102. { L"D32_FLOAT", DXGI_FORMAT_D32_FLOAT },
  1103. { L"R32_FLOAT", DXGI_FORMAT_R32_FLOAT },
  1104. { L"R32_UINT", DXGI_FORMAT_R32_UINT },
  1105. { L"R32_SINT", DXGI_FORMAT_R32_SINT },
  1106. { L"R24G8_TYPELESS", DXGI_FORMAT_R24G8_TYPELESS },
  1107. { L"D24_UNORM_S8_UINT", DXGI_FORMAT_D24_UNORM_S8_UINT },
  1108. { L"R24_UNORM_X8_TYPELESS", DXGI_FORMAT_R24_UNORM_X8_TYPELESS },
  1109. { L"X24_TYPELESS_G8_UINT", DXGI_FORMAT_X24_TYPELESS_G8_UINT },
  1110. { L"R8G8_TYPELESS", DXGI_FORMAT_R8G8_TYPELESS },
  1111. { L"R8G8_UNORM", DXGI_FORMAT_R8G8_UNORM },
  1112. { L"R8G8_UINT", DXGI_FORMAT_R8G8_UINT },
  1113. { L"R8G8_SNORM", DXGI_FORMAT_R8G8_SNORM },
  1114. { L"R8G8_SINT", DXGI_FORMAT_R8G8_SINT },
  1115. { L"R16_TYPELESS", DXGI_FORMAT_R16_TYPELESS },
  1116. { L"R16_FLOAT", DXGI_FORMAT_R16_FLOAT },
  1117. { L"D16_UNORM", DXGI_FORMAT_D16_UNORM },
  1118. { L"R16_UNORM", DXGI_FORMAT_R16_UNORM },
  1119. { L"R16_UINT", DXGI_FORMAT_R16_UINT },
  1120. { L"R16_SNORM", DXGI_FORMAT_R16_SNORM },
  1121. { L"R16_SINT", DXGI_FORMAT_R16_SINT },
  1122. { L"R8_TYPELESS", DXGI_FORMAT_R8_TYPELESS },
  1123. { L"R8_UNORM", DXGI_FORMAT_R8_UNORM },
  1124. { L"R8_UINT", DXGI_FORMAT_R8_UINT },
  1125. { L"R8_SNORM", DXGI_FORMAT_R8_SNORM },
  1126. { L"R8_SINT", DXGI_FORMAT_R8_SINT },
  1127. { L"A8_UNORM", DXGI_FORMAT_A8_UNORM },
  1128. { L"R1_UNORM", DXGI_FORMAT_R1_UNORM },
  1129. { L"R9G9B9E5_SHAREDEXP", DXGI_FORMAT_R9G9B9E5_SHAREDEXP },
  1130. { L"R8G8_B8G8_UNORM", DXGI_FORMAT_R8G8_B8G8_UNORM },
  1131. { L"G8R8_G8B8_UNORM", DXGI_FORMAT_G8R8_G8B8_UNORM },
  1132. { L"BC1_TYPELESS", DXGI_FORMAT_BC1_TYPELESS },
  1133. { L"BC1_UNORM", DXGI_FORMAT_BC1_UNORM },
  1134. { L"BC1_UNORM_SRGB", DXGI_FORMAT_BC1_UNORM_SRGB },
  1135. { L"BC2_TYPELESS", DXGI_FORMAT_BC2_TYPELESS },
  1136. { L"BC2_UNORM", DXGI_FORMAT_BC2_UNORM },
  1137. { L"BC2_UNORM_SRGB", DXGI_FORMAT_BC2_UNORM_SRGB },
  1138. { L"BC3_TYPELESS", DXGI_FORMAT_BC3_TYPELESS },
  1139. { L"BC3_UNORM", DXGI_FORMAT_BC3_UNORM },
  1140. { L"BC3_UNORM_SRGB", DXGI_FORMAT_BC3_UNORM_SRGB },
  1141. { L"BC4_TYPELESS", DXGI_FORMAT_BC4_TYPELESS },
  1142. { L"BC4_UNORM", DXGI_FORMAT_BC4_UNORM },
  1143. { L"BC4_SNORM", DXGI_FORMAT_BC4_SNORM },
  1144. { L"BC5_TYPELESS", DXGI_FORMAT_BC5_TYPELESS },
  1145. { L"BC5_UNORM", DXGI_FORMAT_BC5_UNORM },
  1146. { L"BC5_SNORM", DXGI_FORMAT_BC5_SNORM },
  1147. { L"B5G6R5_UNORM", DXGI_FORMAT_B5G6R5_UNORM },
  1148. { L"B5G5R5A1_UNORM", DXGI_FORMAT_B5G5R5A1_UNORM },
  1149. { L"B8G8R8A8_UNORM", DXGI_FORMAT_B8G8R8A8_UNORM },
  1150. { L"B8G8R8X8_UNORM", DXGI_FORMAT_B8G8R8X8_UNORM },
  1151. { L"R10G10B10_XR_BIAS_A2_UNORM", DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM },
  1152. { L"B8G8R8A8_TYPELESS", DXGI_FORMAT_B8G8R8A8_TYPELESS },
  1153. { L"B8G8R8A8_UNORM_SRGB", DXGI_FORMAT_B8G8R8A8_UNORM_SRGB },
  1154. { L"B8G8R8X8_TYPELESS", DXGI_FORMAT_B8G8R8X8_TYPELESS },
  1155. { L"B8G8R8X8_UNORM_SRGB", DXGI_FORMAT_B8G8R8X8_UNORM_SRGB },
  1156. { L"BC6H_TYPELESS", DXGI_FORMAT_BC6H_TYPELESS },
  1157. { L"BC6H_UF16", DXGI_FORMAT_BC6H_UF16 },
  1158. { L"BC6H_SF16", DXGI_FORMAT_BC6H_SF16 },
  1159. { L"BC7_TYPELESS", DXGI_FORMAT_BC7_TYPELESS },
  1160. { L"BC7_UNORM", DXGI_FORMAT_BC7_UNORM },
  1161. { L"BC7_UNORM_SRGB", DXGI_FORMAT_BC7_UNORM_SRGB },
  1162. { L"AYUV", DXGI_FORMAT_AYUV },
  1163. { L"Y410", DXGI_FORMAT_Y410 },
  1164. { L"Y416", DXGI_FORMAT_Y416 },
  1165. { L"NV12", DXGI_FORMAT_NV12 },
  1166. { L"P010", DXGI_FORMAT_P010 },
  1167. { L"P016", DXGI_FORMAT_P016 },
  1168. { L"420_OPAQUE", DXGI_FORMAT_420_OPAQUE },
  1169. { L"YUY2", DXGI_FORMAT_YUY2 },
  1170. { L"Y210", DXGI_FORMAT_Y210 },
  1171. { L"Y216", DXGI_FORMAT_Y216 },
  1172. { L"NV11", DXGI_FORMAT_NV11 },
  1173. { L"AI44", DXGI_FORMAT_AI44 },
  1174. { L"IA44", DXGI_FORMAT_IA44 },
  1175. { L"P8", DXGI_FORMAT_P8 },
  1176. { L"A8P8", DXGI_FORMAT_A8P8 },
  1177. { L"B4G4R4A4_UNORM", DXGI_FORMAT_B4G4R4A4_UNORM },
  1178. { L"P208", DXGI_FORMAT_P208 },
  1179. { L"V208", DXGI_FORMAT_V208 },
  1180. { L"V408", DXGI_FORMAT_V408 }
  1181. };
  1182. static const ParserEnumValue HEAP_TYPE_TABLE[] = {
  1183. { L"DEFAULT", D3D12_HEAP_TYPE_DEFAULT },
  1184. { L"UPLOAD", D3D12_HEAP_TYPE_UPLOAD },
  1185. { L"READBACK", D3D12_HEAP_TYPE_READBACK },
  1186. { L"CUSTOM", D3D12_HEAP_TYPE_CUSTOM }
  1187. };
  1188. static const ParserEnumValue CPU_PAGE_PROPERTY_TABLE[] = {
  1189. { L"UNKNOWN", D3D12_CPU_PAGE_PROPERTY_UNKNOWN },
  1190. { L"NOT_AVAILABLE", D3D12_CPU_PAGE_PROPERTY_NOT_AVAILABLE },
  1191. { L"WRITE_COMBINE", D3D12_CPU_PAGE_PROPERTY_WRITE_COMBINE },
  1192. { L"WRITE_BACK", D3D12_CPU_PAGE_PROPERTY_WRITE_BACK }
  1193. };
  1194. static const ParserEnumValue MEMORY_POOL_TABLE[] = {
  1195. { L"UNKNOWN", D3D12_MEMORY_POOL_UNKNOWN },
  1196. { L"L0 ", D3D12_MEMORY_POOL_L0 },
  1197. { L"L1", D3D12_MEMORY_POOL_L1 }
  1198. };
  1199. static const ParserEnumValue RESOURCE_DIMENSION_TABLE[] = {
  1200. { L"UNKNOWN", D3D12_RESOURCE_DIMENSION_UNKNOWN },
  1201. { L"BUFFER", D3D12_RESOURCE_DIMENSION_BUFFER },
  1202. { L"TEXTURE1D", D3D12_RESOURCE_DIMENSION_TEXTURE1D },
  1203. { L"TEXTURE2D", D3D12_RESOURCE_DIMENSION_TEXTURE2D },
  1204. { L"TEXTURE3D", D3D12_RESOURCE_DIMENSION_TEXTURE3D }
  1205. };
  1206. static const ParserEnumValue TEXTURE_LAYOUT_TABLE[] = {
  1207. { L"UNKNOWN", D3D12_TEXTURE_LAYOUT_UNKNOWN },
  1208. { L"ROW_MAJOR", D3D12_TEXTURE_LAYOUT_ROW_MAJOR },
  1209. { L"UNDEFINED_SWIZZLE", D3D12_TEXTURE_LAYOUT_64KB_UNDEFINED_SWIZZLE },
  1210. { L"STANDARD_SWIZZLE", D3D12_TEXTURE_LAYOUT_64KB_STANDARD_SWIZZLE }
  1211. };
  1212. static const ParserEnumValue RESOURCE_FLAG_TABLE[] = {
  1213. { L"NONE", D3D12_RESOURCE_FLAG_NONE },
  1214. { L"ALLOW_RENDER_TARGET", D3D12_RESOURCE_FLAG_ALLOW_RENDER_TARGET },
  1215. { L"ALLOW_DEPTH_STENCIL", D3D12_RESOURCE_FLAG_ALLOW_DEPTH_STENCIL },
  1216. { L"ALLOW_UNORDERED_ACCESS", D3D12_RESOURCE_FLAG_ALLOW_UNORDERED_ACCESS },
  1217. { L"DENY_SHADER_RESOURCE", D3D12_RESOURCE_FLAG_DENY_SHADER_RESOURCE },
  1218. { L"ALLOW_CROSS_ADAPTER", D3D12_RESOURCE_FLAG_ALLOW_CROSS_ADAPTER },
  1219. { L"ALLOW_SIMULTANEOUS_ACCESS", D3D12_RESOURCE_FLAG_ALLOW_SIMULTANEOUS_ACCESS }
  1220. };
  1221. static const ParserEnumValue HEAP_FLAG_TABLE[] = {
  1222. { L"NONE", D3D12_HEAP_FLAG_NONE },
  1223. { L"SHARED", D3D12_HEAP_FLAG_SHARED },
  1224. { L"DENY_BUFFERS", D3D12_HEAP_FLAG_DENY_BUFFERS },
  1225. { L"ALLOW_DISPLAY", D3D12_HEAP_FLAG_ALLOW_DISPLAY },
  1226. { L"SHARED_CROSS_ADAPTER", D3D12_HEAP_FLAG_SHARED_CROSS_ADAPTER },
  1227. { L"DENY_RT_DS_TEXTURES", D3D12_HEAP_FLAG_DENY_RT_DS_TEXTURES },
  1228. { L"DENY_NON_RT_DS_TEXTURES", D3D12_HEAP_FLAG_DENY_NON_RT_DS_TEXTURES },
  1229. { L"ALLOW_ALL_BUFFERS_AND_TEXTURES",D3D12_HEAP_FLAG_ALLOW_ALL_BUFFERS_AND_TEXTURES },
  1230. { L"ALLOW_ONLY_BUFFERS", D3D12_HEAP_FLAG_ALLOW_ONLY_BUFFERS },
  1231. { L"ALLOW_ONLY_NON_RT_DS_TEXTURES", D3D12_HEAP_FLAG_ALLOW_ONLY_NON_RT_DS_TEXTURES },
  1232. { L"ALLOW_ONLY_RT_DS_TEXTURES", D3D12_HEAP_FLAG_ALLOW_ONLY_RT_DS_TEXTURES }
  1233. };
  1234. static const ParserEnumValue RESOURCE_STATE_TABLE[] = {
  1235. { L"COMMON", D3D12_RESOURCE_STATE_COMMON },
  1236. { L"VERTEX_AND_CONSTANT_BUFFER", D3D12_RESOURCE_STATE_VERTEX_AND_CONSTANT_BUFFER },
  1237. { L"INDEX_BUFFER", D3D12_RESOURCE_STATE_INDEX_BUFFER },
  1238. { L"RENDER_TARGET", D3D12_RESOURCE_STATE_RENDER_TARGET },
  1239. { L"UNORDERED_ACCESS", D3D12_RESOURCE_STATE_UNORDERED_ACCESS },
  1240. { L"DEPTH_WRITE", D3D12_RESOURCE_STATE_DEPTH_WRITE },
  1241. { L"DEPTH_READ", D3D12_RESOURCE_STATE_DEPTH_READ },
  1242. { L"NON_PIXEL_SHADER_RESOURCE", D3D12_RESOURCE_STATE_NON_PIXEL_SHADER_RESOURCE },
  1243. { L"PIXEL_SHADER_RESOURCE", D3D12_RESOURCE_STATE_PIXEL_SHADER_RESOURCE },
  1244. { L"STREAM_OUT", D3D12_RESOURCE_STATE_STREAM_OUT },
  1245. { L"INDIRECT_ARGUMENT", D3D12_RESOURCE_STATE_INDIRECT_ARGUMENT },
  1246. { L"COPY_DEST", D3D12_RESOURCE_STATE_COPY_DEST },
  1247. { L"COPY_SOURCE", D3D12_RESOURCE_STATE_COPY_SOURCE },
  1248. { L"RESOLVE_DEST", D3D12_RESOURCE_STATE_RESOLVE_DEST },
  1249. { L"RESOLVE_SOURCE", D3D12_RESOURCE_STATE_RESOLVE_SOURCE },
  1250. { L"GENERIC_READ", D3D12_RESOURCE_STATE_GENERIC_READ },
  1251. { L"PRESENT", D3D12_RESOURCE_STATE_PRESENT },
  1252. { L"PREDICATION", D3D12_RESOURCE_STATE_PREDICATION }
  1253. };
  1254. static const ParserEnumValue DESCRIPTOR_HEAP_TYPE_TABLE[] = {
  1255. { L"CBV_SRV_UAV", D3D12_DESCRIPTOR_HEAP_TYPE_CBV_SRV_UAV },
  1256. { L"SAMPLER", D3D12_DESCRIPTOR_HEAP_TYPE_SAMPLER },
  1257. { L"RTV", D3D12_DESCRIPTOR_HEAP_TYPE_RTV },
  1258. { L"DSV", D3D12_DESCRIPTOR_HEAP_TYPE_DSV }
  1259. };
  1260. static const ParserEnumValue DESCRIPTOR_HEAP_FLAG_TABLE[] = {
  1261. { L"NONE", D3D12_DESCRIPTOR_HEAP_FLAG_NONE },
  1262. { L"SHADER_VISIBLE", D3D12_DESCRIPTOR_HEAP_FLAG_SHADER_VISIBLE }
  1263. };
  1264. static const ParserEnumValue SRV_DIMENSION_TABLE[] = {
  1265. { L"UNKNOWN", D3D12_SRV_DIMENSION_UNKNOWN },
  1266. { L"BUFFER", D3D12_SRV_DIMENSION_BUFFER },
  1267. { L"TEXTURE1D", D3D12_SRV_DIMENSION_TEXTURE1D },
  1268. { L"TEXTURE1DARRAY", D3D12_SRV_DIMENSION_TEXTURE1DARRAY },
  1269. { L"TEXTURE2D", D3D12_SRV_DIMENSION_TEXTURE2D },
  1270. { L"TEXTURE2DARRAY", D3D12_SRV_DIMENSION_TEXTURE2DARRAY },
  1271. { L"TEXTURE2DMS", D3D12_SRV_DIMENSION_TEXTURE2DMS },
  1272. { L"TEXTURE2DMSARRAY", D3D12_SRV_DIMENSION_TEXTURE2DMSARRAY },
  1273. { L"TEXTURE3D", D3D12_SRV_DIMENSION_TEXTURE3D },
  1274. { L"TEXTURECUBE", D3D12_SRV_DIMENSION_TEXTURECUBE },
  1275. { L"TEXTURECUBEARRAY", D3D12_SRV_DIMENSION_TEXTURECUBEARRAY }
  1276. };
  1277. static const ParserEnumValue UAV_DIMENSION_TABLE[] = {
  1278. { L"UNKNOWN", D3D12_UAV_DIMENSION_UNKNOWN },
  1279. { L"BUFFER", D3D12_UAV_DIMENSION_BUFFER },
  1280. { L"TEXTURE1D", D3D12_UAV_DIMENSION_TEXTURE1D },
  1281. { L"TEXTURE1DARRAY", D3D12_UAV_DIMENSION_TEXTURE1DARRAY },
  1282. { L"TEXTURE2D", D3D12_UAV_DIMENSION_TEXTURE2D },
  1283. { L"TEXTURE2DARRAY", D3D12_UAV_DIMENSION_TEXTURE2DARRAY },
  1284. { L"TEXTURE3D", D3D12_UAV_DIMENSION_TEXTURE3D }
  1285. };
  1286. static const ParserEnumValue PRIMITIVE_TOPOLOGY_TABLE[] = {
  1287. { L"UNDEFINED",D3D_PRIMITIVE_TOPOLOGY_UNDEFINED },
  1288. { L"POINTLIST",D3D_PRIMITIVE_TOPOLOGY_POINTLIST },
  1289. { L"LINELIST",D3D_PRIMITIVE_TOPOLOGY_LINELIST },
  1290. { L"LINESTRIP",D3D_PRIMITIVE_TOPOLOGY_LINESTRIP },
  1291. { L"TRIANGLELIST",D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST },
  1292. { L"TRIANGLESTRIP",D3D_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP },
  1293. { L"LINELIST_ADJ",D3D_PRIMITIVE_TOPOLOGY_LINELIST_ADJ },
  1294. { L"LINESTRIP_ADJ",D3D_PRIMITIVE_TOPOLOGY_LINESTRIP_ADJ },
  1295. { L"TRIANGLELIST_ADJ",D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST_ADJ },
  1296. { L"TRIANGLESTRIP_ADJ",D3D_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP_ADJ },
  1297. { L"1_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_1_CONTROL_POINT_PATCHLIST },
  1298. { L"2_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_2_CONTROL_POINT_PATCHLIST },
  1299. { L"3_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_3_CONTROL_POINT_PATCHLIST },
  1300. { L"4_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_4_CONTROL_POINT_PATCHLIST },
  1301. { L"5_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_5_CONTROL_POINT_PATCHLIST },
  1302. { L"6_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_6_CONTROL_POINT_PATCHLIST },
  1303. { L"7_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_7_CONTROL_POINT_PATCHLIST },
  1304. { L"8_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_8_CONTROL_POINT_PATCHLIST },
  1305. { L"9_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_9_CONTROL_POINT_PATCHLIST },
  1306. { L"10_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_10_CONTROL_POINT_PATCHLIST },
  1307. { L"11_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_11_CONTROL_POINT_PATCHLIST },
  1308. { L"12_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_12_CONTROL_POINT_PATCHLIST },
  1309. { L"13_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_13_CONTROL_POINT_PATCHLIST },
  1310. { L"14_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_14_CONTROL_POINT_PATCHLIST },
  1311. { L"15_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_15_CONTROL_POINT_PATCHLIST },
  1312. { L"16_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_16_CONTROL_POINT_PATCHLIST },
  1313. { L"17_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_17_CONTROL_POINT_PATCHLIST },
  1314. { L"18_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_18_CONTROL_POINT_PATCHLIST },
  1315. { L"19_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_19_CONTROL_POINT_PATCHLIST },
  1316. { L"20_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_20_CONTROL_POINT_PATCHLIST },
  1317. { L"21_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_21_CONTROL_POINT_PATCHLIST },
  1318. { L"22_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_22_CONTROL_POINT_PATCHLIST },
  1319. { L"23_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_23_CONTROL_POINT_PATCHLIST },
  1320. { L"24_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_24_CONTROL_POINT_PATCHLIST },
  1321. { L"25_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_25_CONTROL_POINT_PATCHLIST },
  1322. { L"26_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_26_CONTROL_POINT_PATCHLIST },
  1323. { L"27_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_27_CONTROL_POINT_PATCHLIST },
  1324. { L"28_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_28_CONTROL_POINT_PATCHLIST },
  1325. { L"29_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_29_CONTROL_POINT_PATCHLIST },
  1326. { L"30_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_30_CONTROL_POINT_PATCHLIST },
  1327. { L"31_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_31_CONTROL_POINT_PATCHLIST },
  1328. { L"32_CONTROL_POINT_PATCHLIST",D3D_PRIMITIVE_TOPOLOGY_32_CONTROL_POINT_PATCHLIST }
  1329. };
  1330. static const ParserEnumValue PRIMITIVE_TOPOLOGY_TYPE_TABLE[] = {
  1331. { L"UNDEFINED", D3D12_PRIMITIVE_TOPOLOGY_TYPE_UNDEFINED },
  1332. { L"POINT", D3D12_PRIMITIVE_TOPOLOGY_TYPE_POINT },
  1333. { L"LINE", D3D12_PRIMITIVE_TOPOLOGY_TYPE_LINE },
  1334. { L"TRIANGLE", D3D12_PRIMITIVE_TOPOLOGY_TYPE_TRIANGLE },
  1335. { L"PATCH", D3D12_PRIMITIVE_TOPOLOGY_TYPE_PATCH }
  1336. };
  1337. static const ParserEnumTable g_ParserEnumTables[] = {
  1338. { _countof(INPUT_CLASSIFICATION_TABLE), INPUT_CLASSIFICATION_TABLE, ParserEnumKind::INPUT_CLASSIFICATION },
  1339. { _countof(DXGI_FORMAT_TABLE), DXGI_FORMAT_TABLE, ParserEnumKind::DXGI_FORMAT },
  1340. { _countof(HEAP_TYPE_TABLE), HEAP_TYPE_TABLE, ParserEnumKind::HEAP_TYPE },
  1341. { _countof(CPU_PAGE_PROPERTY_TABLE), CPU_PAGE_PROPERTY_TABLE, ParserEnumKind::CPU_PAGE_PROPERTY },
  1342. { _countof(MEMORY_POOL_TABLE), MEMORY_POOL_TABLE, ParserEnumKind::MEMORY_POOL },
  1343. { _countof(RESOURCE_DIMENSION_TABLE), RESOURCE_DIMENSION_TABLE, ParserEnumKind::RESOURCE_DIMENSION },
  1344. { _countof(TEXTURE_LAYOUT_TABLE), TEXTURE_LAYOUT_TABLE, ParserEnumKind::TEXTURE_LAYOUT },
  1345. { _countof(RESOURCE_FLAG_TABLE), RESOURCE_FLAG_TABLE, ParserEnumKind::RESOURCE_FLAG },
  1346. { _countof(HEAP_FLAG_TABLE), HEAP_FLAG_TABLE, ParserEnumKind::HEAP_FLAG },
  1347. { _countof(RESOURCE_STATE_TABLE), RESOURCE_STATE_TABLE, ParserEnumKind::RESOURCE_STATE },
  1348. { _countof(DESCRIPTOR_HEAP_TYPE_TABLE), DESCRIPTOR_HEAP_TYPE_TABLE, ParserEnumKind::DESCRIPTOR_HEAP_TYPE },
  1349. { _countof(DESCRIPTOR_HEAP_FLAG_TABLE), DESCRIPTOR_HEAP_FLAG_TABLE, ParserEnumKind::DESCRIPTOR_HEAP_FLAG },
  1350. { _countof(SRV_DIMENSION_TABLE), SRV_DIMENSION_TABLE, ParserEnumKind::SRV_DIMENSION },
  1351. { _countof(UAV_DIMENSION_TABLE), UAV_DIMENSION_TABLE, ParserEnumKind::UAV_DIMENSION },
  1352. { _countof(PRIMITIVE_TOPOLOGY_TABLE), PRIMITIVE_TOPOLOGY_TABLE, ParserEnumKind::PRIMITIVE_TOPOLOGY },
  1353. { _countof(PRIMITIVE_TOPOLOGY_TYPE_TABLE), PRIMITIVE_TOPOLOGY_TYPE_TABLE, ParserEnumKind::PRIMITIVE_TOPOLOGY_TYPE },
  1354. };
  1355. static HRESULT GetEnumValue(LPCWSTR name, ParserEnumKind K, UINT *pValue) {
  1356. for (size_t i = 0; i < _countof(g_ParserEnumTables); ++i) {
  1357. const ParserEnumTable &T = g_ParserEnumTables[i];
  1358. if (T.Kind != K) {
  1359. continue;
  1360. }
  1361. for (size_t j = 0; j < T.ValueCount; ++j) {
  1362. if (_wcsicmp(name, T.Values[j].Name) == 0) {
  1363. *pValue = T.Values[j].Value;
  1364. return S_OK;
  1365. }
  1366. }
  1367. }
  1368. return E_INVALIDARG;
  1369. }
  1370. template <typename T>
  1371. static HRESULT GetEnumValueT(LPCWSTR name, ParserEnumKind K, T *pValue) {
  1372. UINT u;
  1373. HRESULT hr = GetEnumValue(name, K, &u);
  1374. *pValue = (T)u;
  1375. return hr;
  1376. }
  1377. template <typename T>
  1378. static HRESULT ReadAttrEnumT(IXmlReader *pReader, LPCWSTR pAttrName, ParserEnumKind K, T *pValue, T defaultValue, LPCWSTR pStripPrefix = nullptr) {
  1379. if (S_FALSE == CHECK_HR_RET(pReader->MoveToAttributeByName(pAttrName, nullptr))) {
  1380. *pValue = defaultValue;
  1381. return S_FALSE;
  1382. }
  1383. LPCWSTR pText;
  1384. CHECK_HR(pReader->GetValue(&pText, nullptr));
  1385. if (pStripPrefix && *pStripPrefix && _wcsnicmp(pStripPrefix, pText, wcslen(pStripPrefix)) == 0)
  1386. pText += wcslen(pStripPrefix);
  1387. CHECK_HR(GetEnumValueT(pText, K, pValue));
  1388. CHECK_HR(pReader->MoveToElement());
  1389. return S_OK;
  1390. }
  1391. static HRESULT ReadAttrINPUT_CLASSIFICATION(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_INPUT_CLASSIFICATION *pValue) {
  1392. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::INPUT_CLASSIFICATION, pValue, D3D12_INPUT_CLASSIFICATION_PER_VERTEX_DATA);
  1393. }
  1394. static HRESULT ReadAttrDESCRIPTOR_HEAP_TYPE(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_DESCRIPTOR_HEAP_TYPE *pValue) {
  1395. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::DESCRIPTOR_HEAP_TYPE, pValue, D3D12_DESCRIPTOR_HEAP_TYPE_CBV_SRV_UAV);
  1396. }
  1397. static HRESULT ReadAttrDESCRIPTOR_HEAP_FLAGS(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_DESCRIPTOR_HEAP_FLAGS *pValue) {
  1398. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::DESCRIPTOR_HEAP_FLAG, pValue, D3D12_DESCRIPTOR_HEAP_FLAG_SHADER_VISIBLE);
  1399. }
  1400. static HRESULT ReadAttrDXGI_FORMAT(IXmlReader *pReader, LPCWSTR pAttrName, DXGI_FORMAT *pValue) {
  1401. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::DXGI_FORMAT, pValue, DXGI_FORMAT_UNKNOWN, L"DXGI_FORMAT_");
  1402. }
  1403. static HRESULT ReadAttrHEAP_TYPE(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_HEAP_TYPE *pValue) {
  1404. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::HEAP_TYPE, pValue, D3D12_HEAP_TYPE_DEFAULT);
  1405. }
  1406. static HRESULT ReadAttrCPU_PAGE_PROPERTY(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_CPU_PAGE_PROPERTY *pValue) {
  1407. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::CPU_PAGE_PROPERTY, pValue, D3D12_CPU_PAGE_PROPERTY_UNKNOWN);
  1408. }
  1409. static HRESULT ReadAttrMEMORY_POOL(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_MEMORY_POOL *pValue) {
  1410. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::MEMORY_POOL, pValue, D3D12_MEMORY_POOL_UNKNOWN);
  1411. }
  1412. static HRESULT ReadAttrRESOURCE_DIMENSION(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_RESOURCE_DIMENSION *pValue) {
  1413. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::RESOURCE_DIMENSION, pValue, D3D12_RESOURCE_DIMENSION_BUFFER);
  1414. }
  1415. static HRESULT ReadAttrTEXTURE_LAYOUT(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_TEXTURE_LAYOUT *pValue) {
  1416. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::TEXTURE_LAYOUT, pValue, D3D12_TEXTURE_LAYOUT_UNKNOWN);
  1417. }
  1418. static HRESULT ReadAttrRESOURCE_FLAGS(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_RESOURCE_FLAGS *pValue) {
  1419. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::RESOURCE_FLAG, pValue, D3D12_RESOURCE_FLAG_NONE);
  1420. }
  1421. static HRESULT ReadAttrHEAP_FLAGS(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_HEAP_FLAGS *pValue) {
  1422. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::HEAP_FLAG, pValue, D3D12_HEAP_FLAG_NONE);
  1423. }
  1424. static HRESULT ReadAttrRESOURCE_STATES(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_RESOURCE_STATES *pValue) {
  1425. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::RESOURCE_STATE, pValue, D3D12_RESOURCE_STATE_COMMON);
  1426. }
  1427. static HRESULT ReadAttrSRV_DIMENSION(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_SRV_DIMENSION *pValue) {
  1428. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::SRV_DIMENSION, pValue, D3D12_SRV_DIMENSION_BUFFER);
  1429. }
  1430. static HRESULT ReadAttrUAV_DIMENSION(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_UAV_DIMENSION *pValue) {
  1431. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::UAV_DIMENSION, pValue, D3D12_UAV_DIMENSION_BUFFER);
  1432. }
  1433. static HRESULT ReadAttrPRIMITIVE_TOPOLOGY(IXmlReader *pReader, LPCWSTR pAttrName, D3D_PRIMITIVE_TOPOLOGY *pValue) {
  1434. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::PRIMITIVE_TOPOLOGY, pValue, D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST);
  1435. }
  1436. static HRESULT ReadAttrPRIMITIVE_TOPOLOGY_TYPE(IXmlReader *pReader, LPCWSTR pAttrName, D3D12_PRIMITIVE_TOPOLOGY_TYPE *pValue) {
  1437. return ReadAttrEnumT(pReader, pAttrName, ParserEnumKind::PRIMITIVE_TOPOLOGY_TYPE, pValue, D3D12_PRIMITIVE_TOPOLOGY_TYPE_TRIANGLE);
  1438. }
  1439. HRESULT ShaderOpParser::ReadAttrStr(IXmlReader *pReader, LPCWSTR pAttrName, LPCSTR *ppValue) {
  1440. if (S_FALSE == CHECK_HR_RET(pReader->MoveToAttributeByName(pAttrName, nullptr))) {
  1441. *ppValue = nullptr;
  1442. return S_FALSE;
  1443. }
  1444. LPCWSTR pValue;
  1445. CHECK_HR(pReader->GetValue(&pValue, nullptr));
  1446. *ppValue = m_pStrings->insert(pValue);
  1447. CHECK_HR(pReader->MoveToElement());
  1448. return S_OK;
  1449. }
  1450. HRESULT ShaderOpParser::ReadAttrBOOL(IXmlReader *pReader, LPCWSTR pAttrName, BOOL *pValue, BOOL defaultValue) {
  1451. if (S_FALSE == CHECK_HR_RET(pReader->MoveToAttributeByName(pAttrName, nullptr))) {
  1452. *pValue = defaultValue;
  1453. return S_FALSE;
  1454. }
  1455. LPCWSTR pText;
  1456. CHECK_HR(pReader->GetValue(&pText, nullptr));
  1457. if (_wcsicmp(pText, L"true") == 0) {
  1458. *pValue = TRUE;
  1459. }
  1460. else {
  1461. *pValue = FALSE;
  1462. }
  1463. CHECK_HR(pReader->MoveToElement());
  1464. return S_OK;
  1465. }
  1466. HRESULT ShaderOpParser::ReadAttrUINT64(IXmlReader *pReader, LPCWSTR pAttrName, UINT64 *pValue, UINT64 defaultValue) {
  1467. if (S_FALSE == CHECK_HR_RET(pReader->MoveToAttributeByName(pAttrName, nullptr))) {
  1468. *pValue = defaultValue;
  1469. return S_FALSE;
  1470. }
  1471. LPCWSTR pText;
  1472. CHECK_HR(pReader->GetValue(&pText, nullptr));
  1473. long long ll = _wtol(pText);
  1474. if (errno == ERANGE) CHECK_HR(E_INVALIDARG);
  1475. *pValue = ll;
  1476. CHECK_HR(pReader->MoveToElement());
  1477. return S_OK;
  1478. }
  1479. HRESULT ShaderOpParser::ReadAttrUINT(IXmlReader *pReader, LPCWSTR pAttrName, UINT *pValue, UINT defaultValue) {
  1480. UINT64 u64;
  1481. HRESULT hrRead = CHECK_HR_RET(ReadAttrUINT64(pReader, pAttrName, &u64, defaultValue));
  1482. CHECK_HR(UInt64ToUInt(u64, pValue));
  1483. return hrRead;
  1484. }
  1485. HRESULT ShaderOpParser::ReadAttrUINT16(IXmlReader *pReader, LPCWSTR pAttrName, UINT16 *pValue, UINT16 defaultValue) {
  1486. UINT64 u64;
  1487. HRESULT hrRead = CHECK_HR_RET(ReadAttrUINT64(pReader, pAttrName, &u64, defaultValue));
  1488. CHECK_HR(UInt64ToUInt16(u64, pValue));
  1489. return hrRead;
  1490. }
  1491. void ShaderOpParser::ReadElementContentStr(IXmlReader *pReader, LPCSTR *ppValue) {
  1492. *ppValue = nullptr;
  1493. if (pReader->IsEmptyElement())
  1494. return;
  1495. UINT startDepth;
  1496. XmlNodeType nt;
  1497. CHECK_HR(pReader->GetDepth(&startDepth));
  1498. std::wstring value;
  1499. for (;;) {
  1500. UINT depth;
  1501. CHECK_HR(pReader->Read(&nt));
  1502. CHECK_HR(pReader->GetDepth(&depth));
  1503. if (nt == XmlNodeType_EndElement && depth == startDepth + 1)
  1504. break;
  1505. if (nt == XmlNodeType_CDATA || nt == XmlNodeType_Text || nt == XmlNodeType_Whitespace) {
  1506. LPCWSTR pText;
  1507. CHECK_HR(pReader->GetValue(&pText, nullptr));
  1508. value += pText;
  1509. }
  1510. }
  1511. *ppValue = m_pStrings->insert(value.c_str());
  1512. }
  1513. void ShaderOpParser::ParseDescriptor(IXmlReader *pReader, ShaderOpDescriptor *pDesc) {
  1514. if (!ReadAtElementName(pReader, L"Descriptor"))
  1515. return;
  1516. CHECK_HR(ReadAttrStr(pReader, L"Name", &pDesc->Name));
  1517. CHECK_HR(ReadAttrStr(pReader, L"ResName", &pDesc->ResName));
  1518. CHECK_HR(ReadAttrStr(pReader, L"CounterName", &pDesc->CounterName));
  1519. CHECK_HR(ReadAttrStr(pReader, L"Kind", &pDesc->Kind));
  1520. bool isSRV = pDesc->Kind && 0 == _stricmp(pDesc->Kind, "SRV");
  1521. pDesc->SrvDescPresent = false;
  1522. DXGI_FORMAT *pFormat;
  1523. if (isSRV) {
  1524. // D3D12_SHADER_RESOURCE_VIEW_DESC
  1525. pFormat = &pDesc->SrvDesc.Format;
  1526. }
  1527. else {
  1528. // D3D12_UNORDERED_ACCESS_VIEW_DESC - default for parsing
  1529. pFormat = &pDesc->UavDesc.Format;
  1530. }
  1531. HRESULT hrFormat = ReadAttrDXGI_FORMAT(pReader, L"Format", pFormat);
  1532. CHECK_HR(hrFormat);
  1533. if (isSRV) {
  1534. pDesc->SrvDesc.Shader4ComponentMapping = D3D12_DEFAULT_SHADER_4_COMPONENT_MAPPING;
  1535. pDesc->SrvDescPresent |= S_OK ==
  1536. CHECK_HR_RET(ReadAttrSRV_DIMENSION(pReader, L"Dimension", &pDesc->SrvDesc.ViewDimension));
  1537. switch (pDesc->SrvDesc.ViewDimension) {
  1538. case D3D12_SRV_DIMENSION_BUFFER:
  1539. pDesc->SrvDescPresent |= S_OK ==
  1540. CHECK_HR_RET(ReadAttrUINT64(pReader, L"FirstElement", &pDesc->SrvDesc.Buffer.FirstElement));
  1541. LPCSTR pFlags;
  1542. pDesc->SrvDescPresent |= S_OK ==
  1543. CHECK_HR_RET(ReadAttrStr(pReader, L"Flags", &pFlags));
  1544. if (pFlags && *pFlags && 0 == _stricmp(pFlags, "RAW")) {
  1545. pDesc->SrvDesc.Buffer.Flags = D3D12_BUFFER_SRV_FLAG_RAW;
  1546. }
  1547. else {
  1548. pDesc->SrvDesc.Buffer.Flags = D3D12_BUFFER_SRV_FLAG_NONE;
  1549. }
  1550. pDesc->SrvDescPresent |= S_OK ==
  1551. CHECK_HR_RET(ReadAttrUINT(pReader, L"NumElements", &pDesc->SrvDesc.Buffer.NumElements));
  1552. pDesc->SrvDescPresent |= S_OK ==
  1553. CHECK_HR_RET(ReadAttrUINT(pReader, L"StructureByteStride", &pDesc->SrvDesc.Buffer.StructureByteStride));
  1554. break;
  1555. default:
  1556. CHECK_HR(E_NOTIMPL);
  1557. }
  1558. }
  1559. else {
  1560. CHECK_HR(ReadAttrUAV_DIMENSION(pReader, L"Dimension", &pDesc->UavDesc.ViewDimension));
  1561. switch (pDesc->UavDesc.ViewDimension) {
  1562. case D3D12_UAV_DIMENSION_BUFFER:
  1563. CHECK_HR(ReadAttrUINT64(pReader, L"FirstElement", &pDesc->UavDesc.Buffer.FirstElement));
  1564. CHECK_HR(ReadAttrUINT(pReader, L"NumElements", &pDesc->UavDesc.Buffer.NumElements));
  1565. CHECK_HR(ReadAttrUINT(pReader, L"StructureByteStride", &pDesc->UavDesc.Buffer.StructureByteStride));
  1566. CHECK_HR(ReadAttrUINT64(pReader, L"CounterOffsetInBytes", &pDesc->UavDesc.Buffer.CounterOffsetInBytes));
  1567. LPCSTR pFlags;
  1568. CHECK_HR(ReadAttrStr(pReader, L"Flags", &pFlags));
  1569. if (pFlags && *pFlags && 0 == _stricmp(pFlags, "RAW")) {
  1570. pDesc->UavDesc.Buffer.Flags = D3D12_BUFFER_UAV_FLAG_RAW;
  1571. }
  1572. else {
  1573. pDesc->UavDesc.Buffer.Flags = D3D12_BUFFER_UAV_FLAG_NONE;
  1574. }
  1575. if (hrFormat == S_FALSE && pDesc->UavDesc.Buffer.Flags & D3D12_BUFFER_UAV_FLAG_RAW) {
  1576. pDesc->UavDesc.Format = DXGI_FORMAT_R32_TYPELESS;
  1577. }
  1578. break;
  1579. case D3D12_UAV_DIMENSION_TEXTURE1D:
  1580. CHECK_HR(ReadAttrUINT(pReader, L"MipSlice", &pDesc->UavDesc.Texture1D.MipSlice));
  1581. break;
  1582. case D3D12_UAV_DIMENSION_TEXTURE1DARRAY:
  1583. CHECK_HR(ReadAttrUINT(pReader, L"MipSlice", &pDesc->UavDesc.Texture1DArray.MipSlice));
  1584. CHECK_HR(ReadAttrUINT(pReader, L"FirstArraySlice", &pDesc->UavDesc.Texture1DArray.FirstArraySlice));
  1585. CHECK_HR(ReadAttrUINT(pReader, L"ArraySize", &pDesc->UavDesc.Texture1DArray.ArraySize));
  1586. break;
  1587. case D3D12_UAV_DIMENSION_TEXTURE2D:
  1588. CHECK_HR(ReadAttrUINT(pReader, L"MipSlice", &pDesc->UavDesc.Texture2D.MipSlice));
  1589. CHECK_HR(ReadAttrUINT(pReader, L"PlaneSlice", &pDesc->UavDesc.Texture2D.PlaneSlice));
  1590. break;
  1591. case D3D12_UAV_DIMENSION_TEXTURE2DARRAY:
  1592. CHECK_HR(ReadAttrUINT(pReader, L"MipSlice", &pDesc->UavDesc.Texture2DArray.MipSlice));
  1593. CHECK_HR(ReadAttrUINT(pReader, L"FirstArraySlice", &pDesc->UavDesc.Texture2DArray.FirstArraySlice));
  1594. CHECK_HR(ReadAttrUINT(pReader, L"ArraySize", &pDesc->UavDesc.Texture2DArray.ArraySize));
  1595. CHECK_HR(ReadAttrUINT(pReader, L"PlaneSlice", &pDesc->UavDesc.Texture2DArray.PlaneSlice));
  1596. break;
  1597. case D3D12_UAV_DIMENSION_TEXTURE3D:
  1598. CHECK_HR(ReadAttrUINT(pReader, L"MipSlice", &pDesc->UavDesc.Texture3D.MipSlice));
  1599. CHECK_HR(ReadAttrUINT(pReader, L"FirstWSlice", &pDesc->UavDesc.Texture3D.FirstWSlice));
  1600. CHECK_HR(ReadAttrUINT(pReader, L"WSize", &pDesc->UavDesc.Texture3D.WSize));
  1601. break;
  1602. }
  1603. }
  1604. // If either is missing, set one from the other.
  1605. if (pDesc->Name && !pDesc->ResName) pDesc->ResName = pDesc->Name;
  1606. if (pDesc->ResName && !pDesc->Name) pDesc->Name = pDesc->ResName;
  1607. LPCSTR K = pDesc->Kind;
  1608. if (K == nullptr) {
  1609. ShaderOpLogFmt(L"Descriptor '%S' is missing Kind attribute.", pDesc->Name);
  1610. CHECK_HR(E_INVALIDARG);
  1611. } else if (0 != _stricmp(K, "UAV") && 0 != _stricmp(K, "SRV") &&
  1612. 0 != _stricmp(K, "CBV") && 0 != _stricmp(K, "RTV")) {
  1613. ShaderOpLogFmt(L"Descriptor '%S' references unknown kind '%S'",
  1614. pDesc->Name, K);
  1615. CHECK_HR(E_INVALIDARG);
  1616. }
  1617. }
  1618. void ShaderOpParser::ParseDescriptorHeap(IXmlReader *pReader, ShaderOpDescriptorHeap *pHeap) {
  1619. if (!ReadAtElementName(pReader, L"DescriptorHeap"))
  1620. return;
  1621. CHECK_HR(ReadAttrStr(pReader, L"Name", &pHeap->Name));
  1622. HRESULT hrFlags = ReadAttrDESCRIPTOR_HEAP_FLAGS(pReader, L"Flags", &pHeap->Desc.Flags);
  1623. CHECK_HR(hrFlags);
  1624. CHECK_HR(ReadAttrUINT(pReader, L"NodeMask", &pHeap->Desc.NodeMask));
  1625. CHECK_HR(ReadAttrUINT(pReader, L"NumDescriptors", &pHeap->Desc.NumDescriptors));
  1626. CHECK_HR(ReadAttrDESCRIPTOR_HEAP_TYPE(pReader, L"Type", &pHeap->Desc.Type));
  1627. if (pHeap->Desc.Type == D3D12_DESCRIPTOR_HEAP_TYPE_RTV && hrFlags == S_FALSE)
  1628. pHeap->Desc.Flags = D3D12_DESCRIPTOR_HEAP_FLAG_NONE;
  1629. if (pReader->IsEmptyElement())
  1630. return;
  1631. UINT startDepth;
  1632. XmlNodeType nt;
  1633. CHECK_HR(pReader->GetDepth(&startDepth));
  1634. std::wstring value;
  1635. for (;;) {
  1636. UINT depth;
  1637. CHECK_HR(pReader->Read(&nt));
  1638. CHECK_HR(pReader->GetDepth(&depth));
  1639. if (nt == XmlNodeType_EndElement && depth == startDepth + 1)
  1640. break;
  1641. if (nt == XmlNodeType_Element) {
  1642. LPCWSTR pLocalName;
  1643. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  1644. if (0 == wcscmp(pLocalName, L"Descriptor")) {
  1645. ShaderOpDescriptor D;
  1646. ParseDescriptor(pReader, &D);
  1647. pHeap->Descriptors.push_back(D);
  1648. }
  1649. }
  1650. }
  1651. }
  1652. void ShaderOpParser::ParseInputElement(IXmlReader *pReader, D3D12_INPUT_ELEMENT_DESC *pInputElement) {
  1653. if (!ReadAtElementName(pReader, L"InputElement"))
  1654. return;
  1655. CHECK_HR(ReadAttrStr(pReader, L"SemanticName", &pInputElement->SemanticName));
  1656. CHECK_HR(ReadAttrUINT(pReader, L"SemanticIndex", &pInputElement->SemanticIndex));
  1657. CHECK_HR(ReadAttrDXGI_FORMAT(pReader, L"Format", &pInputElement->Format));
  1658. CHECK_HR(ReadAttrUINT(pReader, L"InputSlot", &pInputElement->InputSlot));
  1659. CHECK_HR(ReadAttrUINT(pReader, L"AlignedByteOffset", &pInputElement->AlignedByteOffset, D3D12_APPEND_ALIGNED_ELEMENT));
  1660. CHECK_HR(ReadAttrINPUT_CLASSIFICATION(pReader, L"InputSlotClass", &pInputElement->InputSlotClass));
  1661. CHECK_HR(ReadAttrUINT(pReader, L"InstanceDataStepRate", &pInputElement->InstanceDataStepRate));
  1662. }
  1663. void ShaderOpParser::ParseInputElements(IXmlReader *pReader, std::vector<D3D12_INPUT_ELEMENT_DESC> *pInputElements) {
  1664. if (!ReadAtElementName(pReader, L"InputElements"))
  1665. return;
  1666. if (pReader->IsEmptyElement()) return;
  1667. UINT startDepth;
  1668. XmlNodeType nt;
  1669. CHECK_HR(pReader->GetDepth(&startDepth));
  1670. for (;;) {
  1671. UINT depth;
  1672. CHECK_HR(pReader->Read(&nt));
  1673. CHECK_HR(pReader->GetDepth(&depth));
  1674. if (nt == XmlNodeType_EndElement && depth == startDepth + 1)
  1675. return;
  1676. if (nt == XmlNodeType_Element) {
  1677. LPCWSTR pLocalName;
  1678. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  1679. if (0 == wcscmp(pLocalName, L"InputElement")) {
  1680. D3D12_INPUT_ELEMENT_DESC desc;
  1681. ParseInputElement(pReader, &desc);
  1682. pInputElements->push_back(desc);
  1683. }
  1684. }
  1685. }
  1686. }
  1687. void ShaderOpParser::ParseRenderTargets(IXmlReader *pReader, std::vector<LPCSTR> *pRenderTargets) {
  1688. if (!ReadAtElementName(pReader, L"RenderTargets"))
  1689. return;
  1690. if (pReader->IsEmptyElement()) return;
  1691. UINT startDepth;
  1692. XmlNodeType nt;
  1693. CHECK_HR(pReader->GetDepth(&startDepth));
  1694. for (;;) {
  1695. UINT depth;
  1696. CHECK_HR(pReader->Read(&nt));
  1697. CHECK_HR(pReader->GetDepth(&depth));
  1698. if (nt == XmlNodeType_EndElement && depth == startDepth + 1)
  1699. return;
  1700. if (nt == XmlNodeType_Element) {
  1701. LPCWSTR pLocalName;
  1702. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  1703. if (0 == wcscmp(pLocalName, L"RenderTarget")) {
  1704. LPCSTR pName;
  1705. CHECK_HR(ReadAttrStr(pReader, L"Name", &pName));
  1706. pRenderTargets->push_back(pName);
  1707. }
  1708. }
  1709. }
  1710. }
  1711. void ShaderOpParser::ParseRootValue(IXmlReader *pReader, ShaderOpRootValue *pRootValue) {
  1712. if (!ReadAtElementName(pReader, L"RootValue"))
  1713. return;
  1714. CHECK_HR(ReadAttrStr(pReader, L"ResName", &pRootValue->ResName));
  1715. CHECK_HR(ReadAttrStr(pReader, L"HeapName", &pRootValue->HeapName));
  1716. CHECK_HR(ReadAttrUINT(pReader, L"Index", &pRootValue->Index));
  1717. }
  1718. void ShaderOpParser::ParseRootValues(IXmlReader *pReader, std::vector<ShaderOpRootValue> *pRootValues) {
  1719. if (!ReadAtElementName(pReader, L"RootValues"))
  1720. return;
  1721. if (pReader->IsEmptyElement()) return;
  1722. UINT startDepth;
  1723. XmlNodeType nt;
  1724. CHECK_HR(pReader->GetDepth(&startDepth));
  1725. for (;;) {
  1726. UINT depth;
  1727. CHECK_HR(pReader->Read(&nt));
  1728. CHECK_HR(pReader->GetDepth(&depth));
  1729. if (nt == XmlNodeType_EndElement && depth == startDepth + 1)
  1730. return;
  1731. if (nt == XmlNodeType_Element) {
  1732. LPCWSTR pLocalName;
  1733. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  1734. if (0 == wcscmp(pLocalName, L"RootValue")) {
  1735. ShaderOpRootValue V;
  1736. ParseRootValue(pReader, &V);
  1737. pRootValues->push_back(V);
  1738. }
  1739. }
  1740. }
  1741. }
  1742. void ShaderOpParser::ParseShaderOpSet(IStream *pStream, ShaderOpSet *pShaderOpSet) {
  1743. CComPtr<IXmlReader> pReader;
  1744. CHECK_HR(CreateXmlReader(__uuidof(IXmlReader), (void **)&pReader, nullptr));
  1745. CHECK_HR(pReader->SetInput(pStream));
  1746. ParseShaderOpSet(pReader, pShaderOpSet);
  1747. }
  1748. void ShaderOpParser::ParseShaderOpSet(IXmlReader *pReader, ShaderOpSet *pShaderOpSet) {
  1749. if (!ReadAtElementName(pReader, L"ShaderOpSet"))
  1750. return;
  1751. UINT startDepth;
  1752. CHECK_HR(pReader->GetDepth(&startDepth));
  1753. XmlNodeType nt = XmlNodeType_Element;
  1754. for (;;) {
  1755. if (nt == XmlNodeType_Element) {
  1756. LPCWSTR pLocalName;
  1757. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  1758. if (0 == wcscmp(pLocalName, L"ShaderOp")) {
  1759. pShaderOpSet->ShaderOps.emplace_back(std::make_unique<ShaderOp>());
  1760. ParseShaderOp(pReader, pShaderOpSet->ShaderOps.back().get());
  1761. }
  1762. }
  1763. else if (nt == XmlNodeType_EndElement) {
  1764. UINT depth;
  1765. CHECK_HR(pReader->GetDepth(&depth));
  1766. if (depth == startDepth + 1)
  1767. return;
  1768. }
  1769. CHECK_HR(pReader->Read(&nt));
  1770. }
  1771. }
  1772. void ShaderOpParser::ParseShaderOp(IXmlReader *pReader, ShaderOp *pShaderOp) {
  1773. m_pStrings = &pShaderOp->Strings;
  1774. // Look for a ShaderOp element.
  1775. if (!ReadAtElementName(pReader, L"ShaderOp"))
  1776. return;
  1777. CHECK_HR(ReadAttrStr(pReader, L"Name", &pShaderOp->Name));
  1778. CHECK_HR(ReadAttrStr(pReader, L"CS", &pShaderOp->CS));
  1779. CHECK_HR(ReadAttrStr(pReader, L"AS", &pShaderOp->AS));
  1780. CHECK_HR(ReadAttrStr(pReader, L"MS", &pShaderOp->MS));
  1781. CHECK_HR(ReadAttrStr(pReader, L"VS", &pShaderOp->VS));
  1782. CHECK_HR(ReadAttrStr(pReader, L"HS", &pShaderOp->HS));
  1783. CHECK_HR(ReadAttrStr(pReader, L"DS", &pShaderOp->DS));
  1784. CHECK_HR(ReadAttrStr(pReader, L"GS", &pShaderOp->GS));
  1785. CHECK_HR(ReadAttrStr(pReader, L"PS", &pShaderOp->PS));
  1786. CHECK_HR(ReadAttrUINT(pReader, L"DispatchX", &pShaderOp->DispatchX, 1));
  1787. CHECK_HR(ReadAttrUINT(pReader, L"DispatchY", &pShaderOp->DispatchY, 1));
  1788. CHECK_HR(ReadAttrUINT(pReader, L"DispatchZ", &pShaderOp->DispatchZ, 1));
  1789. CHECK_HR(ReadAttrPRIMITIVE_TOPOLOGY_TYPE(pReader, L"TopologyType", &pShaderOp->PrimitiveTopologyType));
  1790. UINT startDepth;
  1791. CHECK_HR(pReader->GetDepth(&startDepth));
  1792. XmlNodeType nt = XmlNodeType_Element;
  1793. for (;;) {
  1794. if (nt == XmlNodeType_Element) {
  1795. LPCWSTR pLocalName;
  1796. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  1797. if (0 == wcscmp(pLocalName, L"InputElements")) {
  1798. ParseInputElements(pReader, &pShaderOp->InputElements);
  1799. }
  1800. else if (0 == wcscmp(pLocalName, L"Shader")) {
  1801. ShaderOpShader shader;
  1802. ParseShader(pReader, &shader);
  1803. pShaderOp->Shaders.push_back(shader);
  1804. }
  1805. else if (0 == wcscmp(pLocalName, L"RootSignature")) {
  1806. ReadElementContentStr(pReader, &pShaderOp->RootSignature);
  1807. }
  1808. else if (0 == wcscmp(pLocalName, L"RenderTargets")) {
  1809. ParseRenderTargets(pReader, &pShaderOp->RenderTargets);
  1810. }
  1811. else if (0 == wcscmp(pLocalName, L"Resource")) {
  1812. ShaderOpResource resource;
  1813. ParseResource(pReader, &resource);
  1814. pShaderOp->Resources.push_back(resource);
  1815. }
  1816. else if (0 == wcscmp(pLocalName, L"DescriptorHeap")) {
  1817. ShaderOpDescriptorHeap heap;
  1818. ParseDescriptorHeap(pReader, &heap);
  1819. pShaderOp->DescriptorHeaps.push_back(heap);
  1820. }
  1821. else if (0 == wcscmp(pLocalName, L"RootValues")) {
  1822. ParseRootValues(pReader, &pShaderOp->RootValues);
  1823. }
  1824. }
  1825. else if (nt == XmlNodeType_EndElement) {
  1826. UINT depth;
  1827. CHECK_HR(pReader->GetDepth(&depth));
  1828. if (depth == startDepth + 1)
  1829. return;
  1830. }
  1831. if (S_FALSE == CHECK_HR_RET(pReader->Read(&nt)))
  1832. return;
  1833. }
  1834. }
  1835. LPCWSTR SkipByteInitSeparators(LPCWSTR pText) {
  1836. while (*pText && (*pText == L' ' || *pText == L'\t' ||
  1837. *pText == L'\r' || *pText == L'\n' || *pText == L'{' ||
  1838. *pText == L'}' || *pText == L','))
  1839. ++pText;
  1840. return pText;
  1841. }
  1842. LPCWSTR FindByteInitSeparators(LPCWSTR pText) {
  1843. while (*pText &&
  1844. !(*pText == L' ' || *pText == L'\t' ||
  1845. *pText == L'\r' || *pText == L'\n' || *pText == L'{' ||
  1846. *pText == L'}' || *pText == L','))
  1847. ++pText;
  1848. return pText;
  1849. }
  1850. using namespace hlsl;
  1851. DXIL::ComponentType GetCompType(LPCWSTR pText, LPCWSTR pEnd) {
  1852. // if no prefix shown, use it as a default
  1853. if (pText == pEnd) return DXIL::ComponentType::F32;
  1854. // check if suffix starts with (half)
  1855. if (wcsncmp(pText, L"(half)", 6) == 0) {
  1856. return DXIL::ComponentType::F16;
  1857. }
  1858. switch (*(pEnd - 1)) {
  1859. case L'h':
  1860. case L'H':
  1861. return DXIL::ComponentType::F16;
  1862. case L'l':
  1863. case L'L':
  1864. return DXIL::ComponentType::F64;
  1865. case L'u':
  1866. case L'U':
  1867. return DXIL::ComponentType::U32;
  1868. case L'i':
  1869. case L'I':
  1870. return DXIL::ComponentType::I32;
  1871. case L'f':
  1872. case L'F':
  1873. default:
  1874. return DXIL::ComponentType::F32;
  1875. }
  1876. }
  1877. void ParseDataFromText(LPCWSTR pText, LPCWSTR pEnd, DXIL::ComponentType compType, std::vector<BYTE> &V) {
  1878. BYTE *pB;
  1879. if (compType == DXIL::ComponentType::F16 || compType == DXIL::ComponentType::F32) {
  1880. float fVal;
  1881. size_t wordSize = pEnd - pText;
  1882. if (wordSize >= 3 && 0 == _wcsnicmp(pEnd - 3, L"nan", 3)) {
  1883. fVal = NAN;
  1884. }
  1885. else if (wordSize >= 4 && 0 == _wcsnicmp(pEnd - 4, L"-inf", 4)) {
  1886. fVal = -(INFINITY);
  1887. }
  1888. else if ((wordSize >= 3 && 0 == _wcsnicmp(pEnd - 3, L"inf", 3)) ||
  1889. (wordSize >= 4 && 0 == _wcsnicmp(pEnd - 4, L"+inf", 4))) {
  1890. fVal = INFINITY;
  1891. }
  1892. else if (wordSize >= 7 && 0 == _wcsnicmp(pEnd - 7, L"-denorm", 7)) {
  1893. fVal = -(FLT_MIN / 2);
  1894. }
  1895. else if (wordSize >= 6 && 0 == _wcsnicmp(pEnd - 6, L"denorm", 6)) {
  1896. fVal = (FLT_MIN / 2);
  1897. }
  1898. else {
  1899. fVal = wcstof(pText, nullptr);
  1900. }
  1901. if (compType == DXIL::ComponentType::F16) {
  1902. uint16_t fp16Val = ConvertFloat32ToFloat16(fVal);
  1903. pB = (BYTE *)&fp16Val;
  1904. V.insert(V.end(), pB, pB + sizeof(uint16_t));
  1905. }
  1906. else {
  1907. pB = (BYTE *)&fVal;
  1908. V.insert(V.end(), pB, pB + sizeof(float));
  1909. }
  1910. }
  1911. else if (compType == DXIL::ComponentType::I32) {
  1912. int val = _wtoi(pText);
  1913. pB = (BYTE *)&val;
  1914. V.insert(V.end(), pB, pB + sizeof(int));
  1915. }
  1916. else {
  1917. DXASSERT_ARGS(false, "Unsupported stream component type : %u", compType);
  1918. }
  1919. }
  1920. void ShaderOpParser::ParseResource(IXmlReader *pReader, ShaderOpResource *pResource) {
  1921. if (!ReadAtElementName(pReader, L"Resource"))
  1922. return;
  1923. CHECK_HR(ReadAttrStr(pReader, L"Name", &pResource->Name));
  1924. CHECK_HR(ReadAttrStr(pReader, L"Init", &pResource->Init));
  1925. CHECK_HR(ReadAttrBOOL(pReader, L"ReadBack", &pResource->ReadBack));
  1926. CHECK_HR(ReadAttrHEAP_TYPE(pReader, L"HeapType", &pResource->HeapProperties.Type));
  1927. CHECK_HR(ReadAttrCPU_PAGE_PROPERTY(pReader, L"CPUPageProperty", &pResource->HeapProperties.CPUPageProperty));
  1928. CHECK_HR(ReadAttrMEMORY_POOL(pReader, L"MemoryPoolPreference", &pResource->HeapProperties.MemoryPoolPreference));
  1929. CHECK_HR(ReadAttrUINT(pReader, L"CreationNodeMask", &pResource->HeapProperties.CreationNodeMask));
  1930. CHECK_HR(ReadAttrUINT(pReader, L"VisibleNodeMask", &pResource->HeapProperties.VisibleNodeMask));
  1931. // D3D12_RESOURCE_DESC Desc;
  1932. CHECK_HR(ReadAttrRESOURCE_DIMENSION(pReader, L"Dimension", &pResource->Desc.Dimension));
  1933. CHECK_HR(ReadAttrUINT64(pReader, L"Alignment", &pResource->Desc.Alignment));
  1934. CHECK_HR(ReadAttrUINT64(pReader, L"Width", &pResource->Desc.Width));
  1935. CHECK_HR(ReadAttrUINT(pReader, L"Height", &pResource->Desc.Height));
  1936. CHECK_HR(ReadAttrUINT16(pReader, L"DepthOrArraySize", &pResource->Desc.DepthOrArraySize));
  1937. CHECK_HR(ReadAttrUINT16(pReader, L"MipLevels", &pResource->Desc.MipLevels));
  1938. CHECK_HR(ReadAttrDXGI_FORMAT(pReader, L"Format", &pResource->Desc.Format));
  1939. CHECK_HR(ReadAttrUINT(pReader, L"SampleCount", &pResource->Desc.SampleDesc.Count));
  1940. CHECK_HR(ReadAttrUINT(pReader, L"SampleQual", &pResource->Desc.SampleDesc.Quality));
  1941. CHECK_HR(ReadAttrTEXTURE_LAYOUT(pReader, L"Layout", &pResource->Desc.Layout));
  1942. CHECK_HR(ReadAttrRESOURCE_FLAGS(pReader, L"Flags", &pResource->Desc.Flags));
  1943. CHECK_HR(ReadAttrHEAP_FLAGS(pReader, L"HeapFlags", &pResource->HeapFlags));
  1944. CHECK_HR(ReadAttrRESOURCE_STATES(pReader, L"InitialResourceState", &pResource->InitialResourceState));
  1945. CHECK_HR(ReadAttrRESOURCE_STATES(pReader, L"TransitionTo", &pResource->TransitionTo));
  1946. CHECK_HR(ReadAttrPRIMITIVE_TOPOLOGY(pReader, L"Topology", &pResource->PrimitiveTopology));
  1947. // Set some fixed values.
  1948. if (pResource->Desc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER) {
  1949. pResource->Desc.Height = 1;
  1950. pResource->Desc.DepthOrArraySize = 1;
  1951. pResource->Desc.MipLevels = 1;
  1952. pResource->Desc.Format = DXGI_FORMAT_UNKNOWN;
  1953. pResource->Desc.SampleDesc.Count = 1;
  1954. pResource->Desc.SampleDesc.Quality = 0;
  1955. pResource->Desc.Layout = D3D12_TEXTURE_LAYOUT_ROW_MAJOR;
  1956. }
  1957. if (pResource->Desc.Dimension == D3D12_RESOURCE_DIMENSION_TEXTURE1D) {
  1958. if (pResource->Desc.Height == 0) pResource->Desc.Height = 1;
  1959. if (pResource->Desc.DepthOrArraySize == 0) pResource->Desc.DepthOrArraySize = 1;
  1960. if (pResource->Desc.SampleDesc.Count == 0) pResource->Desc.SampleDesc.Count = 1;
  1961. }
  1962. if (pResource->Desc.Dimension == D3D12_RESOURCE_DIMENSION_TEXTURE2D) {
  1963. if (pResource->Desc.DepthOrArraySize == 0) pResource->Desc.DepthOrArraySize = 1;
  1964. if (pResource->Desc.SampleDesc.Count == 0 ) pResource->Desc.SampleDesc.Count = 1;
  1965. }
  1966. // If the resource has text, that goes into the bytes initialization area.
  1967. if (pReader->IsEmptyElement())
  1968. return;
  1969. std::vector<BYTE> &V = pResource->InitBytes;
  1970. XmlNodeType nt;
  1971. CHECK_HR(pReader->GetNodeType(&nt));
  1972. for (;;) {
  1973. if (nt == XmlNodeType_EndElement) {
  1974. return;
  1975. }
  1976. if (nt == XmlNodeType_Text) {
  1977. // Handle the byte payload. '{', '}', ',', whitespace - these are all
  1978. // separators and are ignored in terms of structure. We simply read
  1979. // literals, figure out their type based on suffix, and write the bytes
  1980. // into the target array.
  1981. LPCWSTR pText;
  1982. pReader->GetValue(&pText, nullptr);
  1983. while (*pText) {
  1984. pText = SkipByteInitSeparators(pText);
  1985. if (!*pText) continue;
  1986. LPCWSTR pEnd = FindByteInitSeparators(pText);
  1987. // Consider looking for prefixes/suffixes to handle bases and types.
  1988. DXIL::ComponentType compType = GetCompType(pText, pEnd);
  1989. ParseDataFromText(pText, pEnd, compType, V);
  1990. pText = pEnd;
  1991. }
  1992. }
  1993. if (S_FALSE == CHECK_HR_RET(pReader->Read(&nt)))
  1994. return;
  1995. }
  1996. }
  1997. void ShaderOpParser::ParseShader(IXmlReader *pReader, ShaderOpShader *pShader) {
  1998. if (!ReadAtElementName(pReader, L"Shader"))
  1999. return;
  2000. CHECK_HR(ReadAttrStr(pReader, L"Name", &pShader->Name));
  2001. CHECK_HR(ReadAttrStr(pReader, L"EntryPoint", &pShader->EntryPoint));
  2002. CHECK_HR(ReadAttrStr(pReader, L"Target", &pShader->Target));
  2003. CHECK_HR(ReadAttrStr(pReader, L"Arguments", &pShader->Arguments));
  2004. CHECK_HR(ReadAttrBOOL(pReader, L"Compiled", &pShader->Compiled))
  2005. ReadElementContentStr(pReader, &pShader->Text);
  2006. bool hasText = pShader->Text && *pShader->Text;
  2007. if (hasText) {
  2008. LPCSTR pCheck;
  2009. CHECK_HR(ReadAttrStr(pReader, L"Text", &pCheck));
  2010. if (pCheck && *pCheck) {
  2011. ShaderOpLogFmt(L"Shader %S has text content and a Text attribute; it "
  2012. L"should only have one",
  2013. pShader->Name);
  2014. CHECK_HR(E_INVALIDARG);
  2015. }
  2016. }
  2017. else {
  2018. CHECK_HR(ReadAttrStr(pReader, L"Text", &pShader->Text));
  2019. }
  2020. if (pShader->EntryPoint == nullptr)
  2021. pShader->EntryPoint = m_pStrings->insert("main");
  2022. }
  2023. bool ShaderOpParser::ReadAtElementName(IXmlReader *pReader, LPCWSTR pName) {
  2024. XmlNodeType nt;
  2025. CHECK_HR(pReader->GetNodeType(&nt));
  2026. for (;;) {
  2027. if (nt == XmlNodeType_Element) {
  2028. LPCWSTR pLocalName;
  2029. CHECK_HR(pReader->GetLocalName(&pLocalName, nullptr));
  2030. if (0 == wcscmp(pLocalName, pName)) {
  2031. return true;
  2032. }
  2033. }
  2034. if (S_FALSE == CHECK_HR_RET(pReader->Read(&nt)))
  2035. return false;
  2036. }
  2037. }
  2038. #pragma endregion Parsing support
  2039. } // namespace st