DxilMetadataHelper.cpp 104 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // //
  3. // DxilMetadataHelper.cpp //
  4. // Copyright (C) Microsoft Corporation. All rights reserved. //
  5. // This file is distributed under the University of Illinois Open Source //
  6. // License. See LICENSE.TXT for details. //
  7. // //
  8. ///////////////////////////////////////////////////////////////////////////////
  9. #include "dxc/Support/Global.h"
  10. #include "dxc/DXIL/DxilMetadataHelper.h"
  11. #include "dxc/DXIL/DxilShaderModel.h"
  12. #include "dxc/DXIL/DxilCBuffer.h"
  13. #include "dxc/DXIL/DxilResource.h"
  14. #include "dxc/DXIL/DxilSampler.h"
  15. #include "dxc/DXIL/DxilSignatureElement.h"
  16. #include "dxc/DXIL/DxilSignature.h"
  17. #include "dxc/DXIL/DxilTypeSystem.h"
  18. #include "dxc/DXIL/DxilFunctionProps.h"
  19. #include "dxc/DXIL/DxilShaderFlags.h"
  20. #include "dxc/DXIL/DxilSubobject.h"
  21. #include "llvm/IR/Constants.h"
  22. #include "llvm/IR/Function.h"
  23. #include "llvm/IR/IntrinsicInst.h"
  24. #include "llvm/IR/LLVMContext.h"
  25. #include "llvm/IR/Metadata.h"
  26. #include "llvm/IR/Module.h"
  27. #include "llvm/Support/raw_ostream.h"
  28. #include "llvm/ADT/DenseSet.h"
  29. #include <array>
  30. #include <algorithm>
  31. #include "dxc/Support/WinIncludes.h"
  32. #include "dxc/Support/WinFunctions.h"
  33. using namespace llvm;
  34. using std::string;
  35. using std::vector;
  36. using std::unique_ptr;
  37. namespace hlsl {
  38. const char DxilMDHelper::kDxilVersionMDName[] = "dx.version";
  39. const char DxilMDHelper::kDxilShaderModelMDName[] = "dx.shaderModel";
  40. const char DxilMDHelper::kDxilEntryPointsMDName[] = "dx.entryPoints";
  41. const char DxilMDHelper::kDxilResourcesMDName[] = "dx.resources";
  42. const char DxilMDHelper::kDxilTypeSystemMDName[] = "dx.typeAnnotations";
  43. const char DxilMDHelper::kDxilTypeSystemHelperVariablePrefix[] = "dx.typevar.";
  44. const char DxilMDHelper::kDxilControlFlowHintMDName[] = "dx.controlflow.hints";
  45. const char DxilMDHelper::kDxilPreciseAttributeMDName[] = "dx.precise";
  46. const char DxilMDHelper::kDxilVariableDebugLayoutMDName[] = "dx.dbg.varlayout";
  47. const char DxilMDHelper::kDxilTempAllocaMDName[] = "dx.temp";
  48. const char DxilMDHelper::kDxilNonUniformAttributeMDName[] = "dx.nonuniform";
  49. const char DxilMDHelper::kHLDxilResourceAttributeMDName[] = "dx.hl.resource.attribute";
  50. const char DxilMDHelper::kDxilValidatorVersionMDName[] = "dx.valver";
  51. // This named metadata is not valid in final module (should be moved to DxilContainer)
  52. const char DxilMDHelper::kDxilRootSignatureMDName[] = "dx.rootSignature";
  53. const char DxilMDHelper::kDxilIntermediateOptionsMDName[] = "dx.intermediateOptions";
  54. const char DxilMDHelper::kDxilViewIdStateMDName[] = "dx.viewIdState";
  55. const char DxilMDHelper::kDxilSubobjectsMDName[] = "dx.subobjects";
  56. const char DxilMDHelper::kDxilSourceContentsMDName[] = "dx.source.contents";
  57. const char DxilMDHelper::kDxilSourceDefinesMDName[] = "dx.source.defines";
  58. const char DxilMDHelper::kDxilSourceMainFileNameMDName[] = "dx.source.mainFileName";
  59. const char DxilMDHelper::kDxilSourceArgsMDName[] = "dx.source.args";
  60. static std::array<const char *, 7> DxilMDNames = { {
  61. DxilMDHelper::kDxilVersionMDName,
  62. DxilMDHelper::kDxilShaderModelMDName,
  63. DxilMDHelper::kDxilEntryPointsMDName,
  64. DxilMDHelper::kDxilResourcesMDName,
  65. DxilMDHelper::kDxilTypeSystemMDName,
  66. DxilMDHelper::kDxilValidatorVersionMDName,
  67. DxilMDHelper::kDxilViewIdStateMDName,
  68. }};
  69. DxilMDHelper::DxilMDHelper(Module *pModule, std::unique_ptr<ExtraPropertyHelper> EPH)
  70. : m_Ctx(pModule->getContext())
  71. , m_pModule(pModule)
  72. , m_pSM(nullptr)
  73. , m_ExtraPropertyHelper(std::move(EPH))
  74. , m_ValMajor(1)
  75. , m_ValMinor(0)
  76. , m_MinValMajor(1)
  77. , m_MinValMinor(0)
  78. , m_bExtraMetadata(false)
  79. {
  80. }
  81. DxilMDHelper::~DxilMDHelper() {
  82. }
  83. void DxilMDHelper::SetShaderModel(const ShaderModel *pSM) {
  84. m_pSM = pSM;
  85. m_pSM->GetMinValidatorVersion(m_MinValMajor, m_MinValMinor);
  86. if (DXIL::CompareVersions(m_ValMajor, m_ValMinor, m_MinValMajor, m_MinValMinor) < 0) {
  87. m_ValMajor = m_MinValMajor;
  88. m_ValMinor = m_MinValMinor;
  89. }
  90. // Validator version 0.0 is not meant for validation or retail driver consumption,
  91. // and is used for separate reflection.
  92. // MinVal version drives metadata decisions for compatilbility, so snap this to
  93. // latest for reflection/no validation case.
  94. if (DXIL::CompareVersions(m_ValMajor, m_ValMinor, 0, 0) == 0) {
  95. m_MinValMajor = 0;
  96. m_MinValMinor = 0;
  97. }
  98. if (m_ExtraPropertyHelper) {
  99. m_ExtraPropertyHelper->m_ValMajor = m_ValMajor;
  100. m_ExtraPropertyHelper->m_ValMinor = m_ValMinor;
  101. m_ExtraPropertyHelper->m_MinValMajor = m_MinValMajor;
  102. m_ExtraPropertyHelper->m_MinValMinor = m_MinValMinor;
  103. }
  104. }
  105. const ShaderModel *DxilMDHelper::GetShaderModel() const {
  106. return m_pSM;
  107. }
  108. //
  109. // DXIL version.
  110. //
  111. void DxilMDHelper::EmitDxilVersion(unsigned Major, unsigned Minor) {
  112. NamedMDNode *pDxilVersionMD = m_pModule->getNamedMetadata(kDxilVersionMDName);
  113. IFTBOOL(pDxilVersionMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  114. pDxilVersionMD = m_pModule->getOrInsertNamedMetadata(kDxilVersionMDName);
  115. Metadata *MDVals[kDxilVersionNumFields];
  116. MDVals[kDxilVersionMajorIdx] = Uint32ToConstMD(Major);
  117. MDVals[kDxilVersionMinorIdx] = Uint32ToConstMD(Minor);
  118. pDxilVersionMD->addOperand(MDNode::get(m_Ctx, MDVals));
  119. }
  120. void DxilMDHelper::LoadDxilVersion(unsigned &Major, unsigned &Minor) {
  121. NamedMDNode *pDxilVersionMD = m_pModule->getNamedMetadata(kDxilVersionMDName);
  122. IFTBOOL(pDxilVersionMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  123. IFTBOOL(pDxilVersionMD->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  124. MDNode *pVersionMD = pDxilVersionMD->getOperand(0);
  125. IFTBOOL(pVersionMD->getNumOperands() == kDxilVersionNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  126. Major = ConstMDToUint32(pVersionMD->getOperand(kDxilVersionMajorIdx));
  127. Minor = ConstMDToUint32(pVersionMD->getOperand(kDxilVersionMinorIdx));
  128. }
  129. //
  130. // Validator version.
  131. //
  132. void DxilMDHelper::EmitValidatorVersion(unsigned Major, unsigned Minor) {
  133. NamedMDNode *pDxilValidatorVersionMD = m_pModule->getNamedMetadata(kDxilValidatorVersionMDName);
  134. // Allow re-writing the validator version, since this can be changed at later points.
  135. if (pDxilValidatorVersionMD)
  136. m_pModule->eraseNamedMetadata(pDxilValidatorVersionMD);
  137. pDxilValidatorVersionMD = m_pModule->getOrInsertNamedMetadata(kDxilValidatorVersionMDName);
  138. Metadata *MDVals[kDxilVersionNumFields];
  139. MDVals[kDxilVersionMajorIdx] = Uint32ToConstMD(Major);
  140. MDVals[kDxilVersionMinorIdx] = Uint32ToConstMD(Minor);
  141. pDxilValidatorVersionMD->addOperand(MDNode::get(m_Ctx, MDVals));
  142. m_ValMajor = Major; m_ValMinor = Minor; // Keep these for later use
  143. }
  144. void DxilMDHelper::LoadValidatorVersion(unsigned &Major, unsigned &Minor) {
  145. NamedMDNode *pDxilValidatorVersionMD = m_pModule->getNamedMetadata(kDxilValidatorVersionMDName);
  146. if (pDxilValidatorVersionMD == nullptr) {
  147. // If no validator version metadata, assume 1.0
  148. Major = 1;
  149. Minor = 0;
  150. m_ValMajor = Major; m_ValMinor = Minor; // Keep these for later use
  151. return;
  152. }
  153. IFTBOOL(pDxilValidatorVersionMD->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  154. MDNode *pVersionMD = pDxilValidatorVersionMD->getOperand(0);
  155. IFTBOOL(pVersionMD->getNumOperands() == kDxilVersionNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  156. Major = ConstMDToUint32(pVersionMD->getOperand(kDxilVersionMajorIdx));
  157. Minor = ConstMDToUint32(pVersionMD->getOperand(kDxilVersionMinorIdx));
  158. m_ValMajor = Major; m_ValMinor = Minor; // Keep these for later use
  159. }
  160. //
  161. // DXIL shader model.
  162. //
  163. void DxilMDHelper::EmitDxilShaderModel(const ShaderModel *pSM) {
  164. NamedMDNode *pShaderModelNamedMD = m_pModule->getNamedMetadata(kDxilShaderModelMDName);
  165. IFTBOOL(pShaderModelNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  166. pShaderModelNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilShaderModelMDName);
  167. Metadata *MDVals[kDxilShaderModelNumFields];
  168. MDVals[kDxilShaderModelTypeIdx ] = MDString::get(m_Ctx, pSM->GetKindName());
  169. MDVals[kDxilShaderModelMajorIdx] = Uint32ToConstMD(pSM->GetMajor());
  170. MDVals[kDxilShaderModelMinorIdx] = Uint32ToConstMD(pSM->GetMinor());
  171. pShaderModelNamedMD->addOperand(MDNode::get(m_Ctx, MDVals));
  172. SetShaderModel(pSM);
  173. }
  174. void DxilMDHelper::LoadDxilShaderModel(const ShaderModel *&pSM) {
  175. NamedMDNode *pShaderModelNamedMD = m_pModule->getNamedMetadata(kDxilShaderModelMDName);
  176. IFTBOOL(pShaderModelNamedMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  177. IFTBOOL(pShaderModelNamedMD->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  178. MDNode *pShaderModelMD = pShaderModelNamedMD->getOperand(0);
  179. IFTBOOL(pShaderModelMD->getNumOperands() == kDxilShaderModelNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  180. MDString *pShaderTypeMD = dyn_cast<MDString>(pShaderModelMD->getOperand(kDxilShaderModelTypeIdx));
  181. IFTBOOL(pShaderTypeMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  182. unsigned Major = ConstMDToUint32(pShaderModelMD->getOperand(kDxilShaderModelMajorIdx));
  183. unsigned Minor = ConstMDToUint32(pShaderModelMD->getOperand(kDxilShaderModelMinorIdx));
  184. string ShaderModelName = pShaderTypeMD->getString();
  185. ShaderModelName += "_" + std::to_string(Major) + "_" +
  186. (Minor == ShaderModel::kOfflineMinor ? "x" : std::to_string(Minor));
  187. pSM = ShaderModel::GetByName(ShaderModelName.c_str());
  188. if (!pSM->IsValidForDxil()) {
  189. char ErrorMsgTxt[40];
  190. StringCchPrintfA(ErrorMsgTxt, _countof(ErrorMsgTxt),
  191. "Unknown shader model '%s'", ShaderModelName.c_str());
  192. string ErrorMsg(ErrorMsgTxt);
  193. throw hlsl::Exception(DXC_E_INCORRECT_DXIL_METADATA, ErrorMsg);
  194. }
  195. SetShaderModel(pSM);
  196. }
  197. //
  198. // intermediate options.
  199. //
  200. void DxilMDHelper::EmitDxilIntermediateOptions(uint32_t flags) {
  201. if (flags == 0) return;
  202. NamedMDNode *pIntermediateOptionsNamedMD = m_pModule->getNamedMetadata(kDxilIntermediateOptionsMDName);
  203. IFTBOOL(pIntermediateOptionsNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  204. pIntermediateOptionsNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilIntermediateOptionsMDName);
  205. pIntermediateOptionsNamedMD->addOperand(
  206. MDNode::get(m_Ctx, { Uint32ToConstMD(kDxilIntermediateOptionsFlags), Uint32ToConstMD(flags) }));
  207. }
  208. void DxilMDHelper::LoadDxilIntermediateOptions(uint32_t &flags) {
  209. flags = 0;
  210. NamedMDNode *pIntermediateOptionsNamedMD = m_pModule->getNamedMetadata(kDxilIntermediateOptionsMDName);
  211. if (pIntermediateOptionsNamedMD == nullptr) return;
  212. for (unsigned i = 0; i < pIntermediateOptionsNamedMD->getNumOperands(); i++) {
  213. MDTuple *pEntry = dyn_cast<MDTuple>(pIntermediateOptionsNamedMD->getOperand(i));
  214. IFTBOOL(pEntry != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  215. IFTBOOL(pEntry->getNumOperands() >= 1, DXC_E_INCORRECT_DXIL_METADATA);
  216. uint32_t id = ConstMDToUint32(pEntry->getOperand(0));
  217. switch (id) {
  218. case kDxilIntermediateOptionsFlags:
  219. IFTBOOL(pEntry->getNumOperands() == 2, DXC_E_INCORRECT_DXIL_METADATA);
  220. flags = ConstMDToUint32(pEntry->getOperand(1));
  221. break;
  222. default: throw hlsl::Exception(DXC_E_INCORRECT_DXIL_METADATA, "Unrecognized intermediate options metadata");
  223. }
  224. }
  225. }
  226. //
  227. // Entry points.
  228. //
  229. void DxilMDHelper::EmitDxilEntryPoints(vector<MDNode *> &MDEntries) {
  230. DXASSERT(MDEntries.size() == 1 || GetShaderModel()->IsLib(),
  231. "only one entry point is supported for now");
  232. NamedMDNode *pEntryPointsNamedMD = m_pModule->getNamedMetadata(kDxilEntryPointsMDName);
  233. IFTBOOL(pEntryPointsNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  234. pEntryPointsNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilEntryPointsMDName);
  235. for (size_t i = 0; i < MDEntries.size(); i++) {
  236. pEntryPointsNamedMD->addOperand(MDEntries[i]);
  237. }
  238. }
  239. void DxilMDHelper::UpdateDxilEntryPoints(vector<MDNode *> &MDEntries) {
  240. DXASSERT(MDEntries.size() == 1, "only one entry point is supported for now");
  241. NamedMDNode *pEntryPointsNamedMD =
  242. m_pModule->getNamedMetadata(kDxilEntryPointsMDName);
  243. IFTBOOL(pEntryPointsNamedMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  244. for (size_t i = 0; i < MDEntries.size(); i++) {
  245. pEntryPointsNamedMD->setOperand(i, MDEntries[i]);
  246. }
  247. }
  248. const NamedMDNode *DxilMDHelper::GetDxilEntryPoints() {
  249. NamedMDNode *pEntryPointsNamedMD = m_pModule->getNamedMetadata(kDxilEntryPointsMDName);
  250. IFTBOOL(pEntryPointsNamedMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  251. return pEntryPointsNamedMD;
  252. }
  253. MDTuple *DxilMDHelper::EmitDxilEntryPointTuple(Function *pFunc, const string &Name,
  254. MDTuple *pSignatures, MDTuple *pResources,
  255. MDTuple *pProperties) {
  256. Metadata *MDVals[kDxilEntryPointNumFields];
  257. MDVals[kDxilEntryPointFunction ] = pFunc ? ValueAsMetadata::get(pFunc) : nullptr;
  258. MDVals[kDxilEntryPointName ] = MDString::get(m_Ctx, Name.c_str());
  259. MDVals[kDxilEntryPointSignatures] = pSignatures;
  260. MDVals[kDxilEntryPointResources ] = pResources;
  261. MDVals[kDxilEntryPointProperties] = pProperties;
  262. return MDNode::get(m_Ctx, MDVals);
  263. }
  264. void DxilMDHelper::GetDxilEntryPoint(const MDNode *MDO, Function *&pFunc, string &Name,
  265. const MDOperand *&pSignatures, const MDOperand *&pResources,
  266. const MDOperand *&pProperties) {
  267. IFTBOOL(MDO != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  268. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO);
  269. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  270. IFTBOOL(pTupleMD->getNumOperands() == kDxilEntryPointNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  271. // Retrieve entry function symbol.
  272. const MDOperand &MDOFunc = pTupleMD->getOperand(kDxilEntryPointFunction);
  273. if (MDOFunc.get() != nullptr) {
  274. ValueAsMetadata *pValueFunc = dyn_cast<ValueAsMetadata>(MDOFunc.get());
  275. IFTBOOL(pValueFunc != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  276. pFunc = dyn_cast<Function>(pValueFunc->getValue());
  277. IFTBOOL(pFunc != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  278. } else {
  279. pFunc = nullptr; // pass-through CP.
  280. }
  281. // Retrieve entry function name.
  282. const MDOperand &MDOName = pTupleMD->getOperand(kDxilEntryPointName);
  283. IFTBOOL(MDOName.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  284. MDString *pMDName = dyn_cast<MDString>(MDOName);
  285. IFTBOOL(pMDName != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  286. Name = pMDName->getString();
  287. pSignatures = &pTupleMD->getOperand(kDxilEntryPointSignatures);
  288. pResources = &pTupleMD->getOperand(kDxilEntryPointResources );
  289. pProperties = &pTupleMD->getOperand(kDxilEntryPointProperties);
  290. }
  291. //
  292. // Signatures.
  293. //
  294. MDTuple *DxilMDHelper::EmitDxilSignatures(const DxilEntrySignature &EntrySig) {
  295. MDTuple *pSignatureTupleMD = nullptr;
  296. const DxilSignature &InputSig = EntrySig.InputSignature;
  297. const DxilSignature &OutputSig = EntrySig.OutputSignature;
  298. const DxilSignature &PCPSig = EntrySig.PatchConstOrPrimSignature;
  299. if (!InputSig.GetElements().empty() || !OutputSig.GetElements().empty() || !PCPSig.GetElements().empty()) {
  300. Metadata *MDVals[kDxilNumSignatureFields];
  301. MDVals[kDxilInputSignature] = EmitSignatureMetadata(InputSig);
  302. MDVals[kDxilOutputSignature] = EmitSignatureMetadata(OutputSig);
  303. MDVals[kDxilPatchConstantSignature] = EmitSignatureMetadata(PCPSig);
  304. pSignatureTupleMD = MDNode::get(m_Ctx, MDVals);
  305. }
  306. return pSignatureTupleMD;
  307. }
  308. void DxilMDHelper::EmitRootSignature(
  309. std::vector<uint8_t> &SerializedRootSignature) {
  310. if (SerializedRootSignature.empty()) {
  311. return;
  312. }
  313. Constant *V = llvm::ConstantDataArray::get(
  314. m_Ctx, llvm::ArrayRef<uint8_t>(SerializedRootSignature.data(),
  315. SerializedRootSignature.size()));
  316. NamedMDNode *pRootSignatureNamedMD = m_pModule->getNamedMetadata(kDxilRootSignatureMDName);
  317. IFTBOOL(pRootSignatureNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  318. pRootSignatureNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilRootSignatureMDName);
  319. pRootSignatureNamedMD->addOperand(MDNode::get(m_Ctx, {ConstantAsMetadata::get(V)}));
  320. return ;
  321. }
  322. void DxilMDHelper::LoadDxilSignatures(const MDOperand &MDO, DxilEntrySignature &EntrySig) {
  323. if (MDO.get() == nullptr)
  324. return;
  325. DxilSignature &InputSig = EntrySig.InputSignature;
  326. DxilSignature &OutputSig = EntrySig.OutputSignature;
  327. DxilSignature &PCPSig = EntrySig.PatchConstOrPrimSignature;
  328. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  329. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  330. IFTBOOL(pTupleMD->getNumOperands() == kDxilNumSignatureFields, DXC_E_INCORRECT_DXIL_METADATA);
  331. LoadSignatureMetadata(pTupleMD->getOperand(kDxilInputSignature), InputSig);
  332. LoadSignatureMetadata(pTupleMD->getOperand(kDxilOutputSignature), OutputSig);
  333. LoadSignatureMetadata(pTupleMD->getOperand(kDxilPatchConstantSignature), PCPSig);
  334. }
  335. MDTuple *DxilMDHelper::EmitSignatureMetadata(const DxilSignature &Sig) {
  336. auto &Elements = Sig.GetElements();
  337. if (Elements.empty())
  338. return nullptr;
  339. vector<Metadata *> MDVals;
  340. for (size_t i = 0; i < Elements.size(); i++) {
  341. MDVals.emplace_back(EmitSignatureElement(*Elements[i]));
  342. }
  343. return MDNode::get(m_Ctx, MDVals);
  344. }
  345. void DxilMDHelper::LoadSignatureMetadata(const MDOperand &MDO, DxilSignature &Sig) {
  346. if (MDO.get() == nullptr)
  347. return;
  348. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  349. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  350. for (unsigned i = 0; i < pTupleMD->getNumOperands(); i++) {
  351. unique_ptr<DxilSignatureElement> pSE(Sig.CreateElement());
  352. LoadSignatureElement(pTupleMD->getOperand(i), *pSE.get());
  353. Sig.AppendElement(std::move(pSE));
  354. }
  355. }
  356. void DxilMDHelper::LoadRootSignature(std::vector<uint8_t> &SerializedRootSignature) {
  357. NamedMDNode *pRootSignatureNamedMD = m_pModule->getNamedMetadata(kDxilRootSignatureMDName);
  358. if(!pRootSignatureNamedMD)
  359. return;
  360. IFTBOOL(pRootSignatureNamedMD->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  361. MDNode *pNode = pRootSignatureNamedMD->getOperand(0);
  362. IFTBOOL(pNode->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  363. const MDOperand &MDO = pNode->getOperand(0);
  364. const ConstantAsMetadata *pMetaData = dyn_cast<ConstantAsMetadata>(MDO.get());
  365. IFTBOOL(pMetaData != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  366. const ConstantDataArray *pData =
  367. dyn_cast<ConstantDataArray>(pMetaData->getValue());
  368. IFTBOOL(pData != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  369. IFTBOOL(pData->getElementType() == Type::getInt8Ty(m_Ctx),
  370. DXC_E_INCORRECT_DXIL_METADATA);
  371. SerializedRootSignature.clear();
  372. unsigned size = pData->getRawDataValues().size();
  373. SerializedRootSignature.resize(size);
  374. memcpy(SerializedRootSignature.data(),
  375. (const uint8_t *)pData->getRawDataValues().begin(), size);
  376. }
  377. static const MDTuple *CastToTupleOrNull(const MDOperand &MDO) {
  378. if (MDO.get() == nullptr)
  379. return nullptr;
  380. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  381. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  382. return pTupleMD;
  383. }
  384. MDTuple *DxilMDHelper::EmitSignatureElement(const DxilSignatureElement &SE) {
  385. Metadata *MDVals[kDxilSignatureElementNumFields];
  386. MDVals[kDxilSignatureElementID ] = Uint32ToConstMD(SE.GetID());
  387. MDVals[kDxilSignatureElementName ] = MDString::get(m_Ctx, SE.GetName());
  388. MDVals[kDxilSignatureElementType ] = Uint8ToConstMD((uint8_t)SE.GetCompType().GetKind());
  389. MDVals[kDxilSignatureElementSystemValue ] = Uint8ToConstMD((uint8_t)SE.GetKind());
  390. MDVals[kDxilSignatureElementIndexVector ] = Uint32VectorToConstMDTuple(SE.GetSemanticIndexVec());
  391. MDVals[kDxilSignatureElementInterpMode ] = Uint8ToConstMD((uint8_t)SE.GetInterpolationMode()->GetKind());
  392. MDVals[kDxilSignatureElementRows ] = Uint32ToConstMD(SE.GetRows());
  393. MDVals[kDxilSignatureElementCols ] = Uint8ToConstMD((uint8_t)SE.GetCols());
  394. MDVals[kDxilSignatureElementStartRow ] = Int32ToConstMD(SE.GetStartRow());
  395. MDVals[kDxilSignatureElementStartCol ] = Int8ToConstMD((int8_t)SE.GetStartCol());
  396. // Name-value list of extended properties.
  397. MDVals[kDxilSignatureElementNameValueList] = nullptr;
  398. vector<Metadata *> MDExtraVals;
  399. m_ExtraPropertyHelper->EmitSignatureElementProperties(SE, MDExtraVals);
  400. if (!MDExtraVals.empty()) {
  401. MDVals[kDxilSignatureElementNameValueList] = MDNode::get(m_Ctx, MDExtraVals);
  402. }
  403. // NOTE: when extra properties for signature elements are needed, extend ExtraPropertyHelper.
  404. return MDNode::get(m_Ctx, MDVals);
  405. }
  406. void DxilMDHelper::LoadSignatureElement(const MDOperand &MDO, DxilSignatureElement &SE) {
  407. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  408. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  409. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  410. IFTBOOL(pTupleMD->getNumOperands() == kDxilSignatureElementNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  411. unsigned ID = ConstMDToUint32( pTupleMD->getOperand(kDxilSignatureElementID));
  412. MDString *pName = dyn_cast<MDString>( pTupleMD->getOperand(kDxilSignatureElementName));
  413. CompType CT = CompType(ConstMDToUint8( pTupleMD->getOperand(kDxilSignatureElementType)));
  414. DXIL::SemanticKind SemKind =
  415. (DXIL::SemanticKind)ConstMDToUint8( pTupleMD->getOperand(kDxilSignatureElementSystemValue));
  416. MDTuple *pSemanticIndexVectorMD = dyn_cast<MDTuple>(pTupleMD->getOperand(kDxilSignatureElementIndexVector));
  417. InterpolationMode IM(ConstMDToUint8( pTupleMD->getOperand(kDxilSignatureElementInterpMode)));
  418. unsigned NumRows = ConstMDToUint32( pTupleMD->getOperand(kDxilSignatureElementRows));
  419. uint8_t NumCols = ConstMDToUint8( pTupleMD->getOperand(kDxilSignatureElementCols));
  420. int32_t StartRow = ConstMDToInt32( pTupleMD->getOperand(kDxilSignatureElementStartRow));
  421. int8_t StartCol = ConstMDToInt8( pTupleMD->getOperand(kDxilSignatureElementStartCol));
  422. IFTBOOL(pName != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  423. IFTBOOL(pSemanticIndexVectorMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  424. vector<unsigned> SemanticIndexVector;
  425. ConstMDTupleToUint32Vector(pSemanticIndexVectorMD, SemanticIndexVector);
  426. SE.Initialize(pName->getString(), CT, IM, NumRows, NumCols, StartRow, StartCol, ID, SemanticIndexVector);
  427. SE.SetKind(SemKind);
  428. // For case a system semantic don't have index, add 0 for it.
  429. if (SemanticIndexVector.empty() && !SE.IsArbitrary()) {
  430. SE.SetSemanticIndexVec({0});
  431. }
  432. // Name-value list of extended properties.
  433. m_ExtraPropertyHelper->m_bExtraMetadata = false;
  434. m_ExtraPropertyHelper->LoadSignatureElementProperties(pTupleMD->getOperand(kDxilSignatureElementNameValueList), SE);
  435. m_bExtraMetadata |= m_ExtraPropertyHelper->m_bExtraMetadata;
  436. }
  437. //
  438. // Resources.
  439. //
  440. MDTuple *DxilMDHelper::EmitDxilResourceTuple(MDTuple *pSRVs, MDTuple *pUAVs,
  441. MDTuple *pCBuffers, MDTuple *pSamplers) {
  442. DXASSERT(pSRVs != nullptr || pUAVs != nullptr || pCBuffers != nullptr || pSamplers != nullptr, "resource tuple should not be emitted if there are no resources");
  443. Metadata *MDVals[kDxilNumResourceFields];
  444. MDVals[kDxilResourceSRVs ] = pSRVs;
  445. MDVals[kDxilResourceUAVs ] = pUAVs;
  446. MDVals[kDxilResourceCBuffers] = pCBuffers;
  447. MDVals[kDxilResourceSamplers] = pSamplers;
  448. MDTuple *pTupleMD = MDNode::get(m_Ctx, MDVals);
  449. return pTupleMD;
  450. }
  451. void DxilMDHelper::EmitDxilResources(llvm::MDTuple *pDxilResourceTuple) {
  452. NamedMDNode *pResourcesNamedMD = m_pModule->getNamedMetadata(kDxilResourcesMDName);
  453. IFTBOOL(pResourcesNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  454. pResourcesNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilResourcesMDName);
  455. pResourcesNamedMD->addOperand(pDxilResourceTuple);
  456. }
  457. void DxilMDHelper::UpdateDxilResources(llvm::MDTuple *pDxilResourceTuple) {
  458. NamedMDNode *pResourcesNamedMD =
  459. m_pModule->getNamedMetadata(kDxilResourcesMDName);
  460. if (!pResourcesNamedMD) {
  461. pResourcesNamedMD =
  462. m_pModule->getOrInsertNamedMetadata(kDxilResourcesMDName);
  463. }
  464. if (pDxilResourceTuple) {
  465. if (pResourcesNamedMD->getNumOperands() != 0) {
  466. pResourcesNamedMD->setOperand(0, pDxilResourceTuple);
  467. }
  468. else {
  469. pResourcesNamedMD->addOperand(pDxilResourceTuple);
  470. }
  471. } else {
  472. m_pModule->eraseNamedMetadata(pResourcesNamedMD);
  473. }
  474. }
  475. void DxilMDHelper::GetDxilResources(const MDOperand &MDO, const MDTuple *&pSRVs,
  476. const MDTuple *&pUAVs, const MDTuple *&pCBuffers,
  477. const MDTuple *&pSamplers) {
  478. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  479. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  480. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  481. IFTBOOL(pTupleMD->getNumOperands() == kDxilNumResourceFields, DXC_E_INCORRECT_DXIL_METADATA);
  482. pSRVs = CastToTupleOrNull(pTupleMD->getOperand(kDxilResourceSRVs ));
  483. pUAVs = CastToTupleOrNull(pTupleMD->getOperand(kDxilResourceUAVs ));
  484. pCBuffers = CastToTupleOrNull(pTupleMD->getOperand(kDxilResourceCBuffers));
  485. pSamplers = CastToTupleOrNull(pTupleMD->getOperand(kDxilResourceSamplers));
  486. }
  487. void DxilMDHelper::EmitDxilResourceBase(const DxilResourceBase &R, Metadata *ppMDVals[]) {
  488. ppMDVals[kDxilResourceBaseID ] = Uint32ToConstMD(R.GetID());
  489. ppMDVals[kDxilResourceBaseVariable ] = ValueAsMetadata::get(R.GetGlobalSymbol());
  490. ppMDVals[kDxilResourceBaseName ] = MDString::get(m_Ctx, R.GetGlobalName());
  491. ppMDVals[kDxilResourceBaseSpaceID ] = Uint32ToConstMD(R.GetSpaceID());
  492. ppMDVals[kDxilResourceBaseLowerBound] = Uint32ToConstMD(R.GetLowerBound());
  493. ppMDVals[kDxilResourceBaseRangeSize ] = Uint32ToConstMD(R.GetRangeSize());
  494. }
  495. void DxilMDHelper::LoadDxilResourceBase(const MDOperand &MDO, DxilResourceBase &R) {
  496. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  497. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  498. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  499. IFTBOOL(pTupleMD->getNumOperands() >= kDxilResourceBaseNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  500. R.SetID(ConstMDToUint32(pTupleMD->getOperand(kDxilResourceBaseID)));
  501. R.SetGlobalSymbol(dyn_cast<Constant>(ValueMDToValue(pTupleMD->getOperand(kDxilResourceBaseVariable))));
  502. R.SetGlobalName(StringMDToString(pTupleMD->getOperand(kDxilResourceBaseName)));
  503. R.SetSpaceID(ConstMDToUint32(pTupleMD->getOperand(kDxilResourceBaseSpaceID)));
  504. R.SetLowerBound(ConstMDToUint32(pTupleMD->getOperand(kDxilResourceBaseLowerBound)));
  505. R.SetRangeSize(ConstMDToUint32(pTupleMD->getOperand(kDxilResourceBaseRangeSize)));
  506. }
  507. MDTuple *DxilMDHelper::EmitDxilSRV(const DxilResource &SRV) {
  508. Metadata *MDVals[kDxilSRVNumFields];
  509. EmitDxilResourceBase(SRV, &MDVals[0]);
  510. // SRV-specific fields.
  511. MDVals[kDxilSRVShape ] = Uint32ToConstMD((unsigned)SRV.GetKind());
  512. MDVals[kDxilSRVSampleCount ] = Uint32ToConstMD(SRV.GetSampleCount());
  513. // Name-value list of extended properties.
  514. MDVals[kDxilSRVNameValueList] = nullptr;
  515. vector<Metadata *> MDExtraVals;
  516. m_ExtraPropertyHelper->EmitSRVProperties(SRV, MDExtraVals);
  517. if (!MDExtraVals.empty()) {
  518. MDVals[kDxilSRVNameValueList] = MDNode::get(m_Ctx, MDExtraVals);
  519. }
  520. return MDNode::get(m_Ctx, MDVals);
  521. }
  522. void DxilMDHelper::LoadDxilSRV(const MDOperand &MDO, DxilResource &SRV) {
  523. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  524. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  525. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  526. IFTBOOL(pTupleMD->getNumOperands() == kDxilSRVNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  527. SRV.SetRW(false);
  528. LoadDxilResourceBase(MDO, SRV);
  529. // SRV-specific fields.
  530. SRV.SetKind((DxilResource::Kind)ConstMDToUint32(pTupleMD->getOperand(kDxilSRVShape)));
  531. SRV.SetSampleCount(ConstMDToUint32(pTupleMD->getOperand(kDxilSRVSampleCount)));
  532. // Name-value list of extended properties.
  533. m_ExtraPropertyHelper->m_bExtraMetadata = false;
  534. m_ExtraPropertyHelper->LoadSRVProperties(pTupleMD->getOperand(kDxilSRVNameValueList), SRV);
  535. m_bExtraMetadata |= m_ExtraPropertyHelper->m_bExtraMetadata;
  536. }
  537. MDTuple *DxilMDHelper::EmitDxilUAV(const DxilResource &UAV) {
  538. Metadata *MDVals[kDxilUAVNumFields];
  539. EmitDxilResourceBase(UAV, &MDVals[0]);
  540. // UAV-specific fields.
  541. MDVals[kDxilUAVShape ] = Uint32ToConstMD((unsigned)UAV.GetKind());
  542. MDVals[kDxilUAVGloballyCoherent ] = BoolToConstMD(UAV.IsGloballyCoherent());
  543. MDVals[kDxilUAVCounter ] = BoolToConstMD(UAV.HasCounter());
  544. MDVals[kDxilUAVRasterizerOrderedView] = BoolToConstMD(UAV.IsROV());
  545. // Name-value list of extended properties.
  546. MDVals[kDxilUAVNameValueList ] = nullptr;
  547. vector<Metadata *> MDExtraVals;
  548. m_ExtraPropertyHelper->EmitUAVProperties(UAV, MDExtraVals);
  549. if (!MDExtraVals.empty()) {
  550. MDVals[kDxilUAVNameValueList] = MDNode::get(m_Ctx, MDExtraVals);
  551. }
  552. return MDNode::get(m_Ctx, MDVals);
  553. }
  554. void DxilMDHelper::LoadDxilUAV(const MDOperand &MDO, DxilResource &UAV) {
  555. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  556. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  557. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  558. IFTBOOL(pTupleMD->getNumOperands() == kDxilUAVNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  559. UAV.SetRW(true);
  560. LoadDxilResourceBase(MDO, UAV);
  561. // UAV-specific fields.
  562. UAV.SetKind((DxilResource::Kind)ConstMDToUint32(pTupleMD->getOperand(kDxilUAVShape)));
  563. UAV.SetGloballyCoherent(ConstMDToBool(pTupleMD->getOperand(kDxilUAVGloballyCoherent)));
  564. UAV.SetHasCounter(ConstMDToBool(pTupleMD->getOperand(kDxilUAVCounter)));
  565. UAV.SetROV(ConstMDToBool(pTupleMD->getOperand(kDxilUAVRasterizerOrderedView)));
  566. // Name-value list of extended properties.
  567. m_ExtraPropertyHelper->m_bExtraMetadata = false;
  568. m_ExtraPropertyHelper->LoadUAVProperties(pTupleMD->getOperand(kDxilUAVNameValueList), UAV);
  569. m_bExtraMetadata |= m_ExtraPropertyHelper->m_bExtraMetadata;
  570. }
  571. MDTuple *DxilMDHelper::EmitDxilCBuffer(const DxilCBuffer &CB) {
  572. Metadata *MDVals[kDxilCBufferNumFields];
  573. EmitDxilResourceBase(CB, &MDVals[0]);
  574. // CBuffer-specific fields.
  575. // CBuffer size in bytes.
  576. MDVals[kDxilCBufferSizeInBytes ] = Uint32ToConstMD(CB.GetSize());
  577. // Name-value list of extended properties.
  578. MDVals[kDxilCBufferNameValueList] = nullptr;
  579. vector<Metadata *> MDExtraVals;
  580. m_ExtraPropertyHelper->EmitCBufferProperties(CB, MDExtraVals);
  581. if (!MDExtraVals.empty()) {
  582. MDVals[kDxilCBufferNameValueList] = MDNode::get(m_Ctx, MDExtraVals);
  583. }
  584. return MDNode::get(m_Ctx, MDVals);
  585. }
  586. void DxilMDHelper::LoadDxilCBuffer(const MDOperand &MDO, DxilCBuffer &CB) {
  587. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  588. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  589. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  590. IFTBOOL(pTupleMD->getNumOperands() == kDxilCBufferNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  591. LoadDxilResourceBase(MDO, CB);
  592. // CBuffer-specific fields.
  593. CB.SetSize(ConstMDToUint32(pTupleMD->getOperand(kDxilCBufferSizeInBytes)));
  594. // Name-value list of extended properties.
  595. m_ExtraPropertyHelper->m_bExtraMetadata = false;
  596. m_ExtraPropertyHelper->LoadCBufferProperties(pTupleMD->getOperand(kDxilCBufferNameValueList), CB);
  597. m_bExtraMetadata |= m_ExtraPropertyHelper->m_bExtraMetadata;
  598. }
  599. void DxilMDHelper::EmitDxilTypeSystem(DxilTypeSystem &TypeSystem, vector<GlobalVariable*> &LLVMUsed) {
  600. auto &TypeMap = TypeSystem.GetStructAnnotationMap();
  601. vector<Metadata *> MDVals;
  602. MDVals.emplace_back(Uint32ToConstMD(kDxilTypeSystemStructTag)); // Tag
  603. unsigned GVIdx = 0;
  604. for (auto it = TypeMap.begin(); it != TypeMap.end(); ++it, GVIdx++) {
  605. StructType *pStructType = const_cast<StructType *>(it->first);
  606. DxilStructAnnotation *pA = it->second.get();
  607. // Don't emit type annotation for empty struct.
  608. if (pA->IsEmptyStruct())
  609. continue;
  610. // Emit struct type field annotations.
  611. Metadata *pMD = EmitDxilStructAnnotation(*pA);
  612. MDVals.push_back(ValueAsMetadata::get(UndefValue::get(pStructType)));
  613. MDVals.push_back(pMD);
  614. }
  615. auto &FuncMap = TypeSystem.GetFunctionAnnotationMap();
  616. vector<Metadata *> MDFuncVals;
  617. MDFuncVals.emplace_back(Uint32ToConstMD(kDxilTypeSystemFunctionTag)); // Tag
  618. for (auto it = FuncMap.begin(); it != FuncMap.end(); ++it) {
  619. DxilFunctionAnnotation *pA = it->second.get();
  620. MDFuncVals.push_back(ValueAsMetadata::get(const_cast<Function*>(pA->GetFunction())));
  621. // Emit function annotations.
  622. Metadata *pMD;
  623. pMD = EmitDxilFunctionAnnotation(*pA);
  624. MDFuncVals.push_back(pMD);
  625. }
  626. NamedMDNode *pDxilTypeAnnotationsMD = m_pModule->getNamedMetadata(kDxilTypeSystemMDName);
  627. if (pDxilTypeAnnotationsMD != nullptr) {
  628. m_pModule->eraseNamedMetadata(pDxilTypeAnnotationsMD);
  629. }
  630. if (MDVals.size() > 1) {
  631. pDxilTypeAnnotationsMD = m_pModule->getOrInsertNamedMetadata(kDxilTypeSystemMDName);
  632. pDxilTypeAnnotationsMD->addOperand(MDNode::get(m_Ctx, MDVals));
  633. }
  634. if (MDFuncVals.size() > 1) {
  635. NamedMDNode *pDxilTypeAnnotationsMD = m_pModule->getNamedMetadata(kDxilTypeSystemMDName);
  636. if (pDxilTypeAnnotationsMD == nullptr)
  637. pDxilTypeAnnotationsMD = m_pModule->getOrInsertNamedMetadata(kDxilTypeSystemMDName);
  638. pDxilTypeAnnotationsMD->addOperand(MDNode::get(m_Ctx, MDFuncVals));
  639. }
  640. }
  641. void DxilMDHelper::LoadDxilTypeSystemNode(const llvm::MDTuple &MDT,
  642. DxilTypeSystem &TypeSystem) {
  643. unsigned Tag = ConstMDToUint32(MDT.getOperand(0));
  644. if (Tag == kDxilTypeSystemStructTag) {
  645. IFTBOOL((MDT.getNumOperands() & 0x1) == 1, DXC_E_INCORRECT_DXIL_METADATA);
  646. for (unsigned i = 1; i < MDT.getNumOperands(); i += 2) {
  647. Constant *pGV =
  648. dyn_cast<Constant>(ValueMDToValue(MDT.getOperand(i)));
  649. IFTBOOL(pGV != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  650. StructType *pGVType =
  651. dyn_cast<StructType>(pGV->getType());
  652. IFTBOOL(pGVType != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  653. DxilStructAnnotation *pSA = TypeSystem.AddStructAnnotation(pGVType);
  654. LoadDxilStructAnnotation(MDT.getOperand(i + 1), *pSA);
  655. }
  656. } else {
  657. IFTBOOL((MDT.getNumOperands() & 0x1) == 1, DXC_E_INCORRECT_DXIL_METADATA);
  658. for (unsigned i = 1; i < MDT.getNumOperands(); i += 2) {
  659. Function *F = dyn_cast<Function>(ValueMDToValue(MDT.getOperand(i)));
  660. DxilFunctionAnnotation *pFA = TypeSystem.AddFunctionAnnotation(F);
  661. LoadDxilFunctionAnnotation(MDT.getOperand(i + 1), *pFA);
  662. }
  663. }
  664. }
  665. void DxilMDHelper::LoadDxilTypeSystem(DxilTypeSystem &TypeSystem) {
  666. NamedMDNode *pDxilTypeAnnotationsMD = m_pModule->getNamedMetadata(kDxilTypeSystemMDName);
  667. if (pDxilTypeAnnotationsMD == nullptr)
  668. return;
  669. IFTBOOL(pDxilTypeAnnotationsMD->getNumOperands() <= 2, DXC_E_INCORRECT_DXIL_METADATA);
  670. for (unsigned i = 0; i < pDxilTypeAnnotationsMD->getNumOperands(); i++) {
  671. const MDTuple *pTupleMD = dyn_cast<MDTuple>(pDxilTypeAnnotationsMD->getOperand(i));
  672. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  673. LoadDxilTypeSystemNode(*pTupleMD, TypeSystem);
  674. }
  675. }
  676. Metadata *DxilMDHelper::EmitDxilTemplateArgAnnotation(const DxilTemplateArgAnnotation &annotation) {
  677. SmallVector<Metadata *, 2> MDVals;
  678. if (annotation.IsType()) {
  679. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilTemplateArgTypeTag));
  680. MDVals.emplace_back(ValueAsMetadata::get(UndefValue::get(const_cast<Type*>(annotation.GetType()))));
  681. } else if (annotation.IsIntegral()) {
  682. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilTemplateArgIntegralTag));
  683. MDVals.emplace_back(Uint64ToConstMD((uint64_t)annotation.GetIntegral()));
  684. }
  685. return MDNode::get(m_Ctx, MDVals);
  686. }
  687. void DxilMDHelper::LoadDxilTemplateArgAnnotation(const llvm::MDOperand &MDO, DxilTemplateArgAnnotation &annotation) {
  688. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  689. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  690. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  691. IFTBOOL(pTupleMD->getNumOperands() >= 1, DXC_E_INCORRECT_DXIL_METADATA);
  692. unsigned Tag = ConstMDToUint32(pTupleMD->getOperand(0));
  693. switch (Tag) {
  694. case kDxilTemplateArgTypeTag: {
  695. IFTBOOL(pTupleMD->getNumOperands() == 2, DXC_E_INCORRECT_DXIL_METADATA);
  696. Constant *C = dyn_cast<Constant>(ValueMDToValue(pTupleMD->getOperand(kDxilTemplateArgValue)));
  697. IFTBOOL(C != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  698. annotation.SetType(C->getType());
  699. } break;
  700. case kDxilTemplateArgIntegralTag:
  701. IFTBOOL(pTupleMD->getNumOperands() == 2, DXC_E_INCORRECT_DXIL_METADATA);
  702. annotation.SetIntegral((int64_t)ConstMDToUint64(pTupleMD->getOperand(kDxilTemplateArgValue)));
  703. break;
  704. default:
  705. DXASSERT(false, "Unknown template argument type tag.");
  706. m_bExtraMetadata = true;
  707. break;
  708. }
  709. }
  710. Metadata *DxilMDHelper::EmitDxilStructAnnotation(const DxilStructAnnotation &SA) {
  711. bool bSupportExtended = DXIL::CompareVersions(m_MinValMajor, m_MinValMinor, 1, 5) >= 0;
  712. vector<Metadata *> MDVals;
  713. MDVals.reserve(SA.GetNumFields() + 2); // In case of extended 1.5 property list
  714. MDVals.resize(SA.GetNumFields() + 1);
  715. MDVals[0] = Uint32ToConstMD(SA.GetCBufferSize());
  716. for (unsigned i = 0; i < SA.GetNumFields(); i++) {
  717. MDVals[i+1] = EmitDxilFieldAnnotation(SA.GetFieldAnnotation(i));
  718. }
  719. // Only add template args if shader target requires validator version that supports them.
  720. if (bSupportExtended && SA.GetNumTemplateArgs()) {
  721. vector<Metadata *> MDTemplateArgs(SA.GetNumTemplateArgs());
  722. for (unsigned i = 0; i < SA.GetNumTemplateArgs(); ++i) {
  723. MDTemplateArgs[i] = EmitDxilTemplateArgAnnotation(SA.GetTemplateArgAnnotation(i));
  724. }
  725. SmallVector<Metadata *, 2> MDExtraVals;
  726. MDExtraVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilTemplateArgumentsTag));
  727. MDExtraVals.emplace_back(MDNode::get(m_Ctx, MDTemplateArgs));
  728. MDVals.emplace_back(MDNode::get(m_Ctx, MDExtraVals));
  729. }
  730. return MDNode::get(m_Ctx, MDVals);
  731. }
  732. void DxilMDHelper::LoadDxilStructAnnotation(const MDOperand &MDO, DxilStructAnnotation &SA) {
  733. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  734. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  735. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  736. if (pTupleMD->getNumOperands() == 1) {
  737. SA.MarkEmptyStruct();
  738. }
  739. if (pTupleMD->getNumOperands() == SA.GetNumFields()+2) {
  740. DXASSERT(DXIL::CompareVersions(m_MinValMajor, m_MinValMinor, 1, 5) >= 0,
  741. "otherwise, template annotation emitted for dxil version < 1.5");
  742. // Load template args from extended operand
  743. const MDOperand &MDOExtra = pTupleMD->getOperand(SA.GetNumFields()+1);
  744. const MDTuple *pTupleMDExtra = dyn_cast_or_null<MDTuple>(MDOExtra.get());
  745. if(pTupleMDExtra) {
  746. for (unsigned i = 0; i < pTupleMDExtra->getNumOperands(); i += 2) {
  747. unsigned Tag = ConstMDToUint32(pTupleMDExtra->getOperand(i));
  748. const MDOperand &MDO = pTupleMDExtra->getOperand(i + 1);
  749. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  750. switch (Tag) {
  751. case kDxilTemplateArgumentsTag: {
  752. const MDTuple *pTupleTemplateArgs = dyn_cast_or_null<MDTuple>(pTupleMDExtra->getOperand(1).get());
  753. IFTBOOL(pTupleTemplateArgs, DXC_E_INCORRECT_DXIL_METADATA);
  754. SA.SetNumTemplateArgs(pTupleTemplateArgs->getNumOperands());
  755. for (unsigned i = 0; i < pTupleTemplateArgs->getNumOperands(); ++i) {
  756. LoadDxilTemplateArgAnnotation(pTupleTemplateArgs->getOperand(i), SA.GetTemplateArgAnnotation(i));
  757. }
  758. } break;
  759. default:
  760. DXASSERT(false, "unknown extended tag for struct annotation.");
  761. m_bExtraMetadata = true;
  762. break;
  763. }
  764. }
  765. }
  766. } else {
  767. IFTBOOL(pTupleMD->getNumOperands() == SA.GetNumFields()+1, DXC_E_INCORRECT_DXIL_METADATA);
  768. }
  769. SA.SetCBufferSize(ConstMDToUint32(pTupleMD->getOperand(0)));
  770. for (unsigned i = 0; i < SA.GetNumFields(); i++) {
  771. const MDOperand &MDO = pTupleMD->getOperand(i+1);
  772. DxilFieldAnnotation &FA = SA.GetFieldAnnotation(i);
  773. LoadDxilFieldAnnotation(MDO, FA);
  774. }
  775. }
  776. Metadata *
  777. DxilMDHelper::EmitDxilFunctionAnnotation(const DxilFunctionAnnotation &FA) {
  778. return EmitDxilParamAnnotations(FA);
  779. }
  780. void DxilMDHelper::LoadDxilFunctionAnnotation(const MDOperand &MDO,
  781. DxilFunctionAnnotation &FA) {
  782. LoadDxilParamAnnotations(MDO, FA);
  783. }
  784. llvm::Metadata *
  785. DxilMDHelper::EmitDxilParamAnnotations(const DxilFunctionAnnotation &FA) {
  786. vector<Metadata *> MDParamAnnotations(FA.GetNumParameters() + 1);
  787. MDParamAnnotations[0] = EmitDxilParamAnnotation(FA.GetRetTypeAnnotation());
  788. for (unsigned i = 0; i < FA.GetNumParameters(); i++) {
  789. MDParamAnnotations[i + 1] =
  790. EmitDxilParamAnnotation(FA.GetParameterAnnotation(i));
  791. }
  792. return MDNode::get(m_Ctx, MDParamAnnotations);
  793. }
  794. void DxilMDHelper::LoadDxilParamAnnotations(const llvm::MDOperand &MDO,
  795. DxilFunctionAnnotation &FA) {
  796. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  797. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  798. IFTBOOL(pTupleMD->getNumOperands() == FA.GetNumParameters() + 1,
  799. DXC_E_INCORRECT_DXIL_METADATA);
  800. DxilParameterAnnotation &retTyAnnotation = FA.GetRetTypeAnnotation();
  801. LoadDxilParamAnnotation(pTupleMD->getOperand(0), retTyAnnotation);
  802. for (unsigned i = 0; i < FA.GetNumParameters(); i++) {
  803. const MDOperand &MDO = pTupleMD->getOperand(i + 1);
  804. DxilParameterAnnotation &PA = FA.GetParameterAnnotation(i);
  805. LoadDxilParamAnnotation(MDO, PA);
  806. }
  807. }
  808. Metadata *
  809. DxilMDHelper::EmitDxilParamAnnotation(const DxilParameterAnnotation &PA) {
  810. vector<Metadata *> MDVals(3);
  811. MDVals[0] = Uint32ToConstMD(static_cast<unsigned>(PA.GetParamInputQual()));
  812. MDVals[1] = EmitDxilFieldAnnotation(PA);
  813. MDVals[2] = Uint32VectorToConstMDTuple(PA.GetSemanticIndexVec());
  814. return MDNode::get(m_Ctx, MDVals);
  815. }
  816. void DxilMDHelper::LoadDxilParamAnnotation(const MDOperand &MDO,
  817. DxilParameterAnnotation &PA) {
  818. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  819. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  820. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  821. IFTBOOL(pTupleMD->getNumOperands() == 3, DXC_E_INCORRECT_DXIL_METADATA);
  822. PA.SetParamInputQual(static_cast<DxilParamInputQual>(
  823. ConstMDToUint32(pTupleMD->getOperand(0))));
  824. LoadDxilFieldAnnotation(pTupleMD->getOperand(1), PA);
  825. MDTuple *pSemanticIndexVectorMD = dyn_cast<MDTuple>(pTupleMD->getOperand(2));
  826. vector<unsigned> SemanticIndexVector;
  827. ConstMDTupleToUint32Vector(pSemanticIndexVectorMD, SemanticIndexVector);
  828. PA.SetSemanticIndexVec(SemanticIndexVector);
  829. }
  830. Metadata *DxilMDHelper::EmitDxilFieldAnnotation(const DxilFieldAnnotation &FA) {
  831. vector<Metadata *> MDVals; // Tag-Value list.
  832. if (FA.HasFieldName()) {
  833. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationFieldNameTag));
  834. MDVals.emplace_back(MDString::get(m_Ctx, FA.GetFieldName()));
  835. }
  836. if (FA.IsPrecise()) {
  837. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationPreciseTag)); // Tag
  838. MDVals.emplace_back(BoolToConstMD(true)); // Value
  839. }
  840. if (FA.HasMatrixAnnotation()) {
  841. const DxilMatrixAnnotation &MA = FA.GetMatrixAnnotation();
  842. Metadata *MatrixMD[3];
  843. MatrixMD[0] = Uint32ToConstMD(MA.Rows);
  844. MatrixMD[1] = Uint32ToConstMD(MA.Cols);
  845. MatrixMD[2] = Uint32ToConstMD((unsigned)MA.Orientation);
  846. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationMatrixTag));
  847. MDVals.emplace_back(MDNode::get(m_Ctx, MatrixMD));
  848. }
  849. if (FA.HasCBufferOffset()) {
  850. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationCBufferOffsetTag));
  851. MDVals.emplace_back(Uint32ToConstMD(FA.GetCBufferOffset()));
  852. }
  853. if (FA.HasSemanticString()) {
  854. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationSemanticStringTag));
  855. MDVals.emplace_back(MDString::get(m_Ctx, FA.GetSemanticString()));
  856. }
  857. if (FA.HasInterpolationMode()) {
  858. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationInterpolationModeTag));
  859. MDVals.emplace_back(Uint32ToConstMD((unsigned)FA.GetInterpolationMode().GetKind()));
  860. }
  861. if (FA.HasCompType()) {
  862. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationCompTypeTag));
  863. MDVals.emplace_back(Uint32ToConstMD((unsigned)FA.GetCompType().GetKind()));
  864. }
  865. if (FA.IsCBVarUsed() &&
  866. DXIL::CompareVersions(m_MinValMajor, m_MinValMinor, 1, 5) >= 0) {
  867. MDVals.emplace_back(Uint32ToConstMD(kDxilFieldAnnotationCBUsedTag));
  868. MDVals.emplace_back(BoolToConstMD(true));
  869. }
  870. return MDNode::get(m_Ctx, MDVals);
  871. }
  872. void DxilMDHelper::LoadDxilFieldAnnotation(const MDOperand &MDO, DxilFieldAnnotation &FA) {
  873. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  874. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  875. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  876. IFTBOOL((pTupleMD->getNumOperands() & 0x1) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  877. for (unsigned i = 0; i < pTupleMD->getNumOperands(); i += 2) {
  878. unsigned Tag = ConstMDToUint32(pTupleMD->getOperand(i));
  879. const MDOperand &MDO = pTupleMD->getOperand(i + 1);
  880. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  881. switch (Tag) {
  882. case kDxilFieldAnnotationPreciseTag:
  883. FA.SetPrecise(ConstMDToBool(MDO));
  884. break;
  885. case kDxilFieldAnnotationMatrixTag: {
  886. DxilMatrixAnnotation MA;
  887. const MDTuple *pMATupleMD = dyn_cast<MDTuple>(MDO.get());
  888. IFTBOOL(pMATupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  889. IFTBOOL(pMATupleMD->getNumOperands() == 3, DXC_E_INCORRECT_DXIL_METADATA);
  890. MA.Rows = ConstMDToUint32(pMATupleMD->getOperand(0));
  891. MA.Cols = ConstMDToUint32(pMATupleMD->getOperand(1));
  892. MA.Orientation = (MatrixOrientation)ConstMDToUint32(pMATupleMD->getOperand(2));
  893. FA.SetMatrixAnnotation(MA);
  894. } break;
  895. case kDxilFieldAnnotationCBufferOffsetTag:
  896. FA.SetCBufferOffset(ConstMDToUint32(MDO));
  897. break;
  898. case kDxilFieldAnnotationSemanticStringTag:
  899. FA.SetSemanticString(StringMDToString(MDO));
  900. break;
  901. case kDxilFieldAnnotationInterpolationModeTag:
  902. FA.SetInterpolationMode(InterpolationMode((InterpolationMode::Kind)ConstMDToUint32(MDO)));
  903. break;
  904. case kDxilFieldAnnotationFieldNameTag:
  905. FA.SetFieldName(StringMDToString(MDO));
  906. break;
  907. case kDxilFieldAnnotationCompTypeTag:
  908. FA.SetCompType((CompType::Kind)ConstMDToUint32(MDO));
  909. break;
  910. case kDxilFieldAnnotationCBUsedTag:
  911. FA.SetCBVarUsed(ConstMDToBool(MDO));
  912. break;
  913. default:
  914. DXASSERT(false, "Unknown extended shader properties tag");
  915. m_bExtraMetadata = true;
  916. break;
  917. }
  918. }
  919. }
  920. const Function *DxilMDHelper::LoadDxilFunctionProps(const MDTuple *pProps,
  921. hlsl::DxilFunctionProps *props) {
  922. unsigned idx = 0;
  923. const Function *F = dyn_cast<Function>(
  924. dyn_cast<ValueAsMetadata>(pProps->getOperand(idx++))->getValue());
  925. DXIL::ShaderKind shaderKind =
  926. static_cast<DXIL::ShaderKind>(ConstMDToUint32(pProps->getOperand(idx++)));
  927. bool bRayAttributes = false;
  928. props->shaderKind = shaderKind;
  929. switch (shaderKind) {
  930. case DXIL::ShaderKind::Compute:
  931. props->ShaderProps.CS.numThreads[0] =
  932. ConstMDToUint32(pProps->getOperand(idx++));
  933. props->ShaderProps.CS.numThreads[1] =
  934. ConstMDToUint32(pProps->getOperand(idx++));
  935. props->ShaderProps.CS.numThreads[2] =
  936. ConstMDToUint32(pProps->getOperand(idx++));
  937. break;
  938. case DXIL::ShaderKind::Geometry:
  939. props->ShaderProps.GS.inputPrimitive =
  940. (DXIL::InputPrimitive)ConstMDToUint32(pProps->getOperand(idx++));
  941. props->ShaderProps.GS.maxVertexCount =
  942. ConstMDToUint32(pProps->getOperand(idx++));
  943. props->ShaderProps.GS.instanceCount =
  944. ConstMDToUint32(pProps->getOperand(idx++));
  945. for (size_t i = 0;
  946. i < _countof(props->ShaderProps.GS.streamPrimitiveTopologies); ++i)
  947. props->ShaderProps.GS.streamPrimitiveTopologies[i] =
  948. (DXIL::PrimitiveTopology)ConstMDToUint32(pProps->getOperand(idx++));
  949. break;
  950. case DXIL::ShaderKind::Hull:
  951. props->ShaderProps.HS.patchConstantFunc = dyn_cast<Function>(
  952. dyn_cast<ValueAsMetadata>(pProps->getOperand(idx++))->getValue());
  953. props->ShaderProps.HS.domain =
  954. (DXIL::TessellatorDomain)ConstMDToUint32(pProps->getOperand(idx++));
  955. props->ShaderProps.HS.partition =
  956. (DXIL::TessellatorPartitioning)ConstMDToUint32(
  957. pProps->getOperand(idx++));
  958. props->ShaderProps.HS.outputPrimitive =
  959. (DXIL::TessellatorOutputPrimitive)ConstMDToUint32(
  960. pProps->getOperand(idx++));
  961. props->ShaderProps.HS.inputControlPoints =
  962. ConstMDToUint32(pProps->getOperand(idx++));
  963. props->ShaderProps.HS.outputControlPoints =
  964. ConstMDToUint32(pProps->getOperand(idx++));
  965. props->ShaderProps.HS.maxTessFactor =
  966. ConstMDToFloat(pProps->getOperand(idx++));
  967. break;
  968. case DXIL::ShaderKind::Domain:
  969. props->ShaderProps.DS.domain =
  970. (DXIL::TessellatorDomain)ConstMDToUint32(pProps->getOperand(idx++));
  971. props->ShaderProps.DS.inputControlPoints =
  972. ConstMDToUint32(pProps->getOperand(idx++));
  973. break;
  974. case DXIL::ShaderKind::Pixel:
  975. props->ShaderProps.PS.EarlyDepthStencil =
  976. ConstMDToUint32(pProps->getOperand(idx++));
  977. break;
  978. case DXIL::ShaderKind::AnyHit:
  979. case DXIL::ShaderKind::ClosestHit:
  980. bRayAttributes = true;
  981. case DXIL::ShaderKind::Miss:
  982. case DXIL::ShaderKind::Callable:
  983. // payload/params unioned and first:
  984. props->ShaderProps.Ray.payloadSizeInBytes =
  985. ConstMDToUint32(pProps->getOperand(idx++));
  986. if (bRayAttributes)
  987. props->ShaderProps.Ray.attributeSizeInBytes =
  988. ConstMDToUint32(pProps->getOperand(idx++));
  989. break;
  990. case DXIL::ShaderKind::Mesh:
  991. props->ShaderProps.MS.numThreads[0] =
  992. ConstMDToUint32(pProps->getOperand(idx++));
  993. props->ShaderProps.MS.numThreads[1] =
  994. ConstMDToUint32(pProps->getOperand(idx++));
  995. props->ShaderProps.MS.numThreads[2] =
  996. ConstMDToUint32(pProps->getOperand(idx++));
  997. props->ShaderProps.MS.maxVertexCount =
  998. ConstMDToUint32(pProps->getOperand(idx++));
  999. props->ShaderProps.MS.maxPrimitiveCount =
  1000. ConstMDToUint32(pProps->getOperand(idx++));
  1001. props->ShaderProps.MS.outputTopology =
  1002. (DXIL::MeshOutputTopology)ConstMDToUint32(pProps->getOperand(idx++));
  1003. props->ShaderProps.MS.payloadSizeInBytes =
  1004. ConstMDToUint32(pProps->getOperand(idx++));
  1005. break;
  1006. case DXIL::ShaderKind::Amplification:
  1007. props->ShaderProps.AS.numThreads[0] =
  1008. ConstMDToUint32(pProps->getOperand(idx++));
  1009. props->ShaderProps.AS.numThreads[1] =
  1010. ConstMDToUint32(pProps->getOperand(idx++));
  1011. props->ShaderProps.AS.numThreads[2] =
  1012. ConstMDToUint32(pProps->getOperand(idx++));
  1013. props->ShaderProps.AS.payloadSizeInBytes =
  1014. ConstMDToUint32(pProps->getOperand(idx++));
  1015. break;
  1016. default:
  1017. break;
  1018. }
  1019. return F;
  1020. }
  1021. MDTuple *DxilMDHelper::EmitDxilEntryProperties(uint64_t rawShaderFlag,
  1022. const DxilFunctionProps &props,
  1023. unsigned autoBindingSpace) {
  1024. vector<Metadata *> MDVals;
  1025. // DXIL shader flags.
  1026. if (props.IsPS()) {
  1027. if (props.ShaderProps.PS.EarlyDepthStencil) {
  1028. ShaderFlags flags;
  1029. flags.SetShaderFlagsRaw(rawShaderFlag);
  1030. flags.SetForceEarlyDepthStencil(true);
  1031. rawShaderFlag = flags.GetShaderFlagsRaw();
  1032. }
  1033. }
  1034. if (rawShaderFlag != 0) {
  1035. MDVals.emplace_back(Uint32ToConstMD(kDxilShaderFlagsTag));
  1036. MDVals.emplace_back(Uint64ToConstMD(rawShaderFlag));
  1037. }
  1038. // Add shader kind for lib entrys.
  1039. if (m_pSM->IsLib() && props.shaderKind != DXIL::ShaderKind::Library) {
  1040. MDVals.emplace_back(Uint32ToConstMD(kDxilShaderKindTag));
  1041. MDVals.emplace_back(
  1042. Uint32ToConstMD(static_cast<unsigned>(props.shaderKind)));
  1043. }
  1044. switch (props.shaderKind) {
  1045. // Compute shader.
  1046. case DXIL::ShaderKind::Compute: {
  1047. auto &CS = props.ShaderProps.CS;
  1048. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilNumThreadsTag));
  1049. vector<Metadata *> NumThreadVals;
  1050. NumThreadVals.emplace_back(Uint32ToConstMD(CS.numThreads[0]));
  1051. NumThreadVals.emplace_back(Uint32ToConstMD(CS.numThreads[1]));
  1052. NumThreadVals.emplace_back(Uint32ToConstMD(CS.numThreads[2]));
  1053. MDVals.emplace_back(MDNode::get(m_Ctx, NumThreadVals));
  1054. } break;
  1055. // Geometry shader.
  1056. case DXIL::ShaderKind::Geometry: {
  1057. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilGSStateTag));
  1058. DXIL::PrimitiveTopology topo = DXIL::PrimitiveTopology::Undefined;
  1059. unsigned activeStreamMask = 0;
  1060. for (size_t i = 0;
  1061. i < _countof(props.ShaderProps.GS.streamPrimitiveTopologies); ++i) {
  1062. if (props.ShaderProps.GS.streamPrimitiveTopologies[i] !=
  1063. DXIL::PrimitiveTopology::Undefined) {
  1064. activeStreamMask |= 1 << i;
  1065. DXASSERT_NOMSG(topo == DXIL::PrimitiveTopology::Undefined ||
  1066. topo ==
  1067. props.ShaderProps.GS.streamPrimitiveTopologies[i]);
  1068. topo = props.ShaderProps.GS.streamPrimitiveTopologies[i];
  1069. }
  1070. }
  1071. MDTuple *pMDTuple =
  1072. EmitDxilGSState(props.ShaderProps.GS.inputPrimitive,
  1073. props.ShaderProps.GS.maxVertexCount, activeStreamMask,
  1074. topo, props.ShaderProps.GS.instanceCount);
  1075. MDVals.emplace_back(pMDTuple);
  1076. } break;
  1077. // Domain shader.
  1078. case DXIL::ShaderKind::Domain: {
  1079. auto &DS = props.ShaderProps.DS;
  1080. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilDSStateTag));
  1081. MDTuple *pMDTuple = EmitDxilDSState(DS.domain, DS.inputControlPoints);
  1082. MDVals.emplace_back(pMDTuple);
  1083. } break;
  1084. // Hull shader.
  1085. case DXIL::ShaderKind::Hull: {
  1086. auto &HS = props.ShaderProps.HS;
  1087. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilHSStateTag));
  1088. MDTuple *pMDTuple = EmitDxilHSState(
  1089. HS.patchConstantFunc, HS.inputControlPoints, HS.outputControlPoints,
  1090. HS.domain, HS.partition, HS.outputPrimitive, HS.maxTessFactor);
  1091. MDVals.emplace_back(pMDTuple);
  1092. } break;
  1093. // Raytracing.
  1094. case DXIL::ShaderKind::AnyHit:
  1095. case DXIL::ShaderKind::ClosestHit: {
  1096. MDVals.emplace_back(Uint32ToConstMD(kDxilRayPayloadSizeTag));
  1097. MDVals.emplace_back(
  1098. Uint32ToConstMD(props.ShaderProps.Ray.payloadSizeInBytes));
  1099. MDVals.emplace_back(Uint32ToConstMD(kDxilRayAttribSizeTag));
  1100. MDVals.emplace_back(
  1101. Uint32ToConstMD(props.ShaderProps.Ray.attributeSizeInBytes));
  1102. } break;
  1103. case DXIL::ShaderKind::Miss:
  1104. case DXIL::ShaderKind::Callable: {
  1105. MDVals.emplace_back(Uint32ToConstMD(kDxilRayPayloadSizeTag));
  1106. MDVals.emplace_back(
  1107. Uint32ToConstMD(props.ShaderProps.Ray.payloadSizeInBytes));
  1108. } break;
  1109. case DXIL::ShaderKind::Mesh: {
  1110. auto &MS = props.ShaderProps.MS;
  1111. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilMSStateTag));
  1112. MDTuple *pMDTuple = EmitDxilMSState(MS.numThreads,
  1113. MS.maxVertexCount,
  1114. MS.maxPrimitiveCount,
  1115. MS.outputTopology,
  1116. MS.payloadSizeInBytes);
  1117. MDVals.emplace_back(pMDTuple);
  1118. } break;
  1119. case DXIL::ShaderKind::Amplification: {
  1120. auto &AS = props.ShaderProps.AS;
  1121. MDVals.emplace_back(Uint32ToConstMD(DxilMDHelper::kDxilASStateTag));
  1122. MDTuple *pMDTuple = EmitDxilASState(AS.numThreads, AS.payloadSizeInBytes);
  1123. MDVals.emplace_back(pMDTuple);
  1124. } break;
  1125. default:
  1126. break;
  1127. }
  1128. if (autoBindingSpace != UINT_MAX && m_pSM->IsSMAtLeast(6, 3)) {
  1129. MDVals.emplace_back(Uint32ToConstMD(kDxilAutoBindingSpaceTag));
  1130. MDVals.emplace_back(
  1131. MDNode::get(m_Ctx, {Uint32ToConstMD(autoBindingSpace)}));
  1132. }
  1133. if (!MDVals.empty())
  1134. return MDNode::get(m_Ctx, MDVals);
  1135. else
  1136. return nullptr;
  1137. }
  1138. void DxilMDHelper::LoadDxilEntryProperties(const MDOperand &MDO,
  1139. uint64_t &rawShaderFlag,
  1140. DxilFunctionProps &props,
  1141. uint32_t &autoBindingSpace) {
  1142. if (MDO.get() == nullptr)
  1143. return;
  1144. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1145. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1146. IFTBOOL((pTupleMD->getNumOperands() & 0x1) == 0,
  1147. DXC_E_INCORRECT_DXIL_METADATA);
  1148. bool bEarlyDepth = false;
  1149. if (!m_pSM->IsLib()) {
  1150. props.shaderKind = m_pSM->GetKind();
  1151. } else {
  1152. props.shaderKind = DXIL::ShaderKind::Library;
  1153. }
  1154. for (unsigned iNode = 0; iNode < pTupleMD->getNumOperands(); iNode += 2) {
  1155. unsigned Tag = DxilMDHelper::ConstMDToUint32(pTupleMD->getOperand(iNode));
  1156. const MDOperand &MDO = pTupleMD->getOperand(iNode + 1);
  1157. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1158. switch (Tag) {
  1159. case DxilMDHelper::kDxilShaderFlagsTag: {
  1160. rawShaderFlag = ConstMDToUint64(MDO);
  1161. ShaderFlags flags;
  1162. flags.SetShaderFlagsRaw(rawShaderFlag);
  1163. bEarlyDepth = flags.GetForceEarlyDepthStencil();
  1164. } break;
  1165. case DxilMDHelper::kDxilNumThreadsTag: {
  1166. DXASSERT(props.IsCS(), "else invalid shader kind");
  1167. auto &CS = props.ShaderProps.CS;
  1168. MDNode *pNode = cast<MDNode>(MDO.get());
  1169. CS.numThreads[0] = ConstMDToUint32(pNode->getOperand(0));
  1170. CS.numThreads[1] = ConstMDToUint32(pNode->getOperand(1));
  1171. CS.numThreads[2] = ConstMDToUint32(pNode->getOperand(2));
  1172. } break;
  1173. case DxilMDHelper::kDxilGSStateTag: {
  1174. DXASSERT(props.IsGS(), "else invalid shader kind");
  1175. auto &GS = props.ShaderProps.GS;
  1176. DXIL::PrimitiveTopology topo = DXIL::PrimitiveTopology::Undefined;
  1177. unsigned activeStreamMask;
  1178. LoadDxilGSState(MDO, GS.inputPrimitive, GS.maxVertexCount,
  1179. activeStreamMask, topo, GS.instanceCount);
  1180. if (topo != DXIL::PrimitiveTopology::Undefined) {
  1181. for (size_t i = 0; i < _countof(GS.streamPrimitiveTopologies); ++i) {
  1182. unsigned mask = 1 << i;
  1183. if (activeStreamMask & mask) {
  1184. GS.streamPrimitiveTopologies[i] = topo;
  1185. } else {
  1186. GS.streamPrimitiveTopologies[i] =
  1187. DXIL::PrimitiveTopology::Undefined;
  1188. }
  1189. }
  1190. }
  1191. } break;
  1192. case DxilMDHelper::kDxilDSStateTag: {
  1193. DXASSERT(props.IsDS(), "else invalid shader kind");
  1194. auto &DS = props.ShaderProps.DS;
  1195. LoadDxilDSState(MDO, DS.domain, DS.inputControlPoints);
  1196. } break;
  1197. case DxilMDHelper::kDxilHSStateTag: {
  1198. DXASSERT(props.IsHS(), "else invalid shader kind");
  1199. auto &HS = props.ShaderProps.HS;
  1200. LoadDxilHSState(MDO, HS.patchConstantFunc, HS.inputControlPoints,
  1201. HS.outputControlPoints, HS.domain, HS.partition,
  1202. HS.outputPrimitive, HS.maxTessFactor);
  1203. } break;
  1204. case DxilMDHelper::kDxilAutoBindingSpaceTag: {
  1205. MDNode *pNode = cast<MDNode>(MDO.get());
  1206. autoBindingSpace = ConstMDToUint32(pNode->getOperand(0));
  1207. break;
  1208. }
  1209. case DxilMDHelper::kDxilRayPayloadSizeTag: {
  1210. DXASSERT(props.IsAnyHit() || props.IsClosestHit() || props.IsMiss() ||
  1211. props.IsCallable(),
  1212. "else invalid shader kind");
  1213. props.ShaderProps.Ray.payloadSizeInBytes =
  1214. ConstMDToUint32(MDO);
  1215. } break;
  1216. case DxilMDHelper::kDxilRayAttribSizeTag: {
  1217. DXASSERT(props.IsAnyHit() || props.IsClosestHit(),
  1218. "else invalid shader kind");
  1219. props.ShaderProps.Ray.attributeSizeInBytes =
  1220. ConstMDToUint32(MDO);
  1221. } break;
  1222. case DxilMDHelper::kDxilShaderKindTag: {
  1223. DXIL::ShaderKind kind =
  1224. static_cast<DXIL::ShaderKind>(ConstMDToUint32(MDO));
  1225. DXASSERT(props.shaderKind == DXIL::ShaderKind::Library,
  1226. "else invalid shader kind");
  1227. props.shaderKind = kind;
  1228. } break;
  1229. case DxilMDHelper::kDxilMSStateTag: {
  1230. DXASSERT(props.IsMS(), "else invalid shader kind");
  1231. auto &MS = props.ShaderProps.MS;
  1232. LoadDxilMSState(MDO, MS.numThreads, MS.maxVertexCount,
  1233. MS.maxPrimitiveCount, MS.outputTopology,
  1234. MS.payloadSizeInBytes);
  1235. } break;
  1236. case DxilMDHelper::kDxilASStateTag: {
  1237. DXASSERT(props.IsAS(), "else invalid shader kind");
  1238. auto &AS = props.ShaderProps.AS;
  1239. LoadDxilASState(MDO, AS.numThreads, AS.payloadSizeInBytes);
  1240. } break;
  1241. default:
  1242. DXASSERT(false, "Unknown extended shader properties tag");
  1243. m_bExtraMetadata = true;
  1244. break;
  1245. }
  1246. }
  1247. if (bEarlyDepth) {
  1248. DXASSERT(props.IsPS(), "else invalid shader kind");
  1249. props.ShaderProps.PS.EarlyDepthStencil = true;
  1250. }
  1251. }
  1252. MDTuple *
  1253. DxilMDHelper::EmitDxilFunctionProps(const hlsl::DxilFunctionProps *props,
  1254. const Function *F) {
  1255. bool bRayAttributes = false;
  1256. Metadata *MDVals[30];
  1257. std::fill(MDVals, MDVals + _countof(MDVals), nullptr);
  1258. unsigned valIdx = 0;
  1259. MDVals[valIdx++] = ValueAsMetadata::get(const_cast<Function*>(F));
  1260. MDVals[valIdx++] = Uint32ToConstMD(static_cast<unsigned>(props->shaderKind));
  1261. switch (props->shaderKind) {
  1262. case DXIL::ShaderKind::Compute:
  1263. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.CS.numThreads[0]);
  1264. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.CS.numThreads[1]);
  1265. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.CS.numThreads[2]);
  1266. break;
  1267. case DXIL::ShaderKind::Geometry:
  1268. MDVals[valIdx++] =
  1269. Uint8ToConstMD((uint8_t)props->ShaderProps.GS.inputPrimitive);
  1270. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.GS.maxVertexCount);
  1271. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.GS.instanceCount);
  1272. for (size_t i = 0;
  1273. i < _countof(props->ShaderProps.GS.streamPrimitiveTopologies); ++i)
  1274. MDVals[valIdx++] = Uint8ToConstMD(
  1275. (uint8_t)props->ShaderProps.GS.streamPrimitiveTopologies[i]);
  1276. break;
  1277. case DXIL::ShaderKind::Hull:
  1278. MDVals[valIdx++] =
  1279. ValueAsMetadata::get(props->ShaderProps.HS.patchConstantFunc);
  1280. MDVals[valIdx++] = Uint8ToConstMD((uint8_t)props->ShaderProps.HS.domain);
  1281. MDVals[valIdx++] = Uint8ToConstMD((uint8_t)props->ShaderProps.HS.partition);
  1282. MDVals[valIdx++] =
  1283. Uint8ToConstMD((uint8_t)props->ShaderProps.HS.outputPrimitive);
  1284. MDVals[valIdx++] =
  1285. Uint32ToConstMD(props->ShaderProps.HS.inputControlPoints);
  1286. MDVals[valIdx++] =
  1287. Uint32ToConstMD(props->ShaderProps.HS.outputControlPoints);
  1288. MDVals[valIdx++] = FloatToConstMD(props->ShaderProps.HS.maxTessFactor);
  1289. break;
  1290. case DXIL::ShaderKind::Domain:
  1291. MDVals[valIdx++] = Uint8ToConstMD((uint8_t)props->ShaderProps.DS.domain);
  1292. MDVals[valIdx++] =
  1293. Uint32ToConstMD(props->ShaderProps.DS.inputControlPoints);
  1294. break;
  1295. case DXIL::ShaderKind::Pixel:
  1296. MDVals[valIdx++] = BoolToConstMD(props->ShaderProps.PS.EarlyDepthStencil);
  1297. break;
  1298. case DXIL::ShaderKind::AnyHit:
  1299. case DXIL::ShaderKind::ClosestHit:
  1300. bRayAttributes = true;
  1301. case DXIL::ShaderKind::Miss:
  1302. case DXIL::ShaderKind::Callable:
  1303. // payload/params unioned and first:
  1304. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.Ray.payloadSizeInBytes);
  1305. if (bRayAttributes)
  1306. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.Ray.attributeSizeInBytes);
  1307. break;
  1308. case DXIL::ShaderKind::Mesh:
  1309. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.MS.numThreads[0]);
  1310. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.MS.numThreads[1]);
  1311. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.MS.numThreads[2]);
  1312. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.MS.maxVertexCount);
  1313. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.MS.maxPrimitiveCount);
  1314. MDVals[valIdx++] = Uint8ToConstMD((uint8_t)props->ShaderProps.MS.outputTopology);
  1315. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.MS.payloadSizeInBytes);
  1316. break;
  1317. case DXIL::ShaderKind::Amplification:
  1318. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.AS.numThreads[0]);
  1319. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.AS.numThreads[1]);
  1320. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.AS.numThreads[2]);
  1321. MDVals[valIdx++] = Uint32ToConstMD(props->ShaderProps.AS.payloadSizeInBytes);
  1322. break;
  1323. default:
  1324. break;
  1325. }
  1326. return MDTuple::get(m_Ctx, ArrayRef<llvm::Metadata *>(MDVals, valIdx));
  1327. }
  1328. void DxilMDHelper::EmitDxilViewIdState(std::vector<unsigned> &SerializedState) {
  1329. const vector<unsigned> &Data = SerializedState;
  1330. // If all UINTs are zero, do not emit ViewIdState.
  1331. if (!std::any_of(Data.begin(), Data.end(), [](unsigned e){return e!=0;}))
  1332. return;
  1333. Constant *V = ConstantDataArray::get(m_Ctx, ArrayRef<uint32_t>(Data));
  1334. NamedMDNode *pViewIdNamedMD = m_pModule->getNamedMetadata(kDxilViewIdStateMDName);
  1335. IFTBOOL(pViewIdNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1336. pViewIdNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilViewIdStateMDName);
  1337. pViewIdNamedMD->addOperand(MDNode::get(m_Ctx, {ConstantAsMetadata::get(V)}));
  1338. }
  1339. void DxilMDHelper::LoadDxilViewIdState(std::vector<unsigned> &SerializedState) {
  1340. NamedMDNode *pViewIdStateNamedMD = m_pModule->getNamedMetadata(kDxilViewIdStateMDName);
  1341. if(!pViewIdStateNamedMD)
  1342. return;
  1343. IFTBOOL(pViewIdStateNamedMD->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  1344. MDNode *pNode = pViewIdStateNamedMD->getOperand(0);
  1345. IFTBOOL(pNode->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  1346. const MDOperand &MDO = pNode->getOperand(0);
  1347. const ConstantAsMetadata *pMetaData = dyn_cast<ConstantAsMetadata>(MDO.get());
  1348. IFTBOOL(pMetaData != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1349. if (isa<ConstantAggregateZero>(pMetaData->getValue()))
  1350. return;
  1351. const ConstantDataArray *pData = dyn_cast<ConstantDataArray>(pMetaData->getValue());
  1352. IFTBOOL(pData != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1353. IFTBOOL(pData->getElementType() == Type::getInt32Ty(m_Ctx), DXC_E_INCORRECT_DXIL_METADATA);
  1354. IFTBOOL(pData->getRawDataValues().size() < UINT_MAX &&
  1355. (pData->getRawDataValues().size() & 3) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  1356. SerializedState.clear();
  1357. unsigned size = (unsigned)pData->getRawDataValues().size() / 4;
  1358. SerializedState.resize(size);
  1359. const unsigned *Ptr = (const unsigned *)pData->getRawDataValues().begin();
  1360. memcpy(SerializedState.data(), Ptr, size * sizeof(unsigned));
  1361. }
  1362. MDNode *DxilMDHelper::EmitControlFlowHints(llvm::LLVMContext &Ctx, std::vector<DXIL::ControlFlowHint> &hints) {
  1363. SmallVector<Metadata *, 4> Args;
  1364. // Reserve operand 0 for self reference.
  1365. auto TempNode = MDNode::getTemporary(Ctx, None);
  1366. Args.emplace_back(TempNode.get());
  1367. Args.emplace_back(MDString::get(Ctx, kDxilControlFlowHintMDName));
  1368. for (DXIL::ControlFlowHint &hint : hints)
  1369. Args.emplace_back(Uint32ToConstMD(static_cast<unsigned>(hint), Ctx));
  1370. MDNode *hintsNode = MDNode::get(Ctx, Args);
  1371. // Set the first operand to itself.
  1372. hintsNode->replaceOperandWith(0, hintsNode);
  1373. return hintsNode;
  1374. }
  1375. unsigned DxilMDHelper::GetControlFlowHintMask(const Instruction *I) {
  1376. // Check that there are control hint to use
  1377. // branch.
  1378. MDNode *MD = I->getMetadata(hlsl::DxilMDHelper::kDxilControlFlowHintMDName);
  1379. if (!MD)
  1380. return 0;
  1381. if (MD->getNumOperands() < 3)
  1382. return 0;
  1383. unsigned mask = 0;
  1384. for (unsigned i = 2; i < MD->getNumOperands(); i++) {
  1385. Metadata *Op = MD->getOperand(2).get();
  1386. auto ConstOp = cast<ConstantAsMetadata>(Op);
  1387. unsigned hint = ConstOp->getValue()->getUniqueInteger().getLimitedValue();
  1388. mask |= 1 << hint;
  1389. }
  1390. return mask;
  1391. }
  1392. bool DxilMDHelper::HasControlFlowHintToPreventFlatten(
  1393. const llvm::Instruction *I) {
  1394. unsigned mask = GetControlFlowHintMask(I);
  1395. const unsigned BranchMask =
  1396. 1 << (unsigned)(DXIL::ControlFlowHint::Branch) |
  1397. 1 << (unsigned)(DXIL::ControlFlowHint::Call) |
  1398. 1 << (unsigned)(DXIL::ControlFlowHint::ForceCase);
  1399. return mask & BranchMask;
  1400. }
  1401. void DxilMDHelper::EmitSubobjects(const DxilSubobjects &Subobjects) {
  1402. NamedMDNode *pSubobjectsNamedMD = m_pModule->getNamedMetadata(kDxilSubobjectsMDName);
  1403. IFTBOOL(pSubobjectsNamedMD == nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1404. pSubobjectsNamedMD = m_pModule->getOrInsertNamedMetadata(kDxilSubobjectsMDName);
  1405. const auto &objMap = Subobjects.GetSubobjects();
  1406. for (auto &it : objMap)
  1407. pSubobjectsNamedMD->addOperand(cast<MDNode>(EmitSubobject(*it.second)));
  1408. }
  1409. void DxilMDHelper::LoadSubobjects(DxilSubobjects &Subobjects) {
  1410. NamedMDNode *pSubobjectsNamedMD = m_pModule->getNamedMetadata(kDxilSubobjectsMDName);
  1411. if (!pSubobjectsNamedMD)
  1412. return;
  1413. for (unsigned i = 0; i < pSubobjectsNamedMD->getNumOperands(); ++i)
  1414. LoadSubobject(*pSubobjectsNamedMD->getOperand(i), Subobjects);
  1415. }
  1416. Metadata *DxilMDHelper::EmitSubobject(const DxilSubobject &obj) {
  1417. SmallVector<Metadata *, 6> Args;
  1418. Args.emplace_back(MDString::get(m_Ctx, obj.GetName()));
  1419. Args.emplace_back(Uint32ToConstMD((unsigned)obj.GetKind()));
  1420. bool bLocalRS = false;
  1421. IFTBOOL(DXIL::IsValidSubobjectKind(obj.GetKind()), DXC_E_INCORRECT_DXIL_METADATA);
  1422. switch (obj.GetKind()) {
  1423. case DXIL::SubobjectKind::StateObjectConfig: {
  1424. uint32_t Flags;
  1425. IFTBOOL(obj.GetStateObjectConfig(Flags),
  1426. DXC_E_INCORRECT_DXIL_METADATA);
  1427. Args.emplace_back(Uint32ToConstMD((unsigned)Flags));
  1428. break;
  1429. }
  1430. case DXIL::SubobjectKind::LocalRootSignature:
  1431. bLocalRS = true;
  1432. __fallthrough;
  1433. case DXIL::SubobjectKind::GlobalRootSignature: {
  1434. const char * Text;
  1435. const void * Data;
  1436. uint32_t Size;
  1437. IFTBOOL(obj.GetRootSignature(bLocalRS, Data, Size, &Text),
  1438. DXC_E_INCORRECT_DXIL_METADATA);
  1439. Constant *V = ConstantDataArray::get(m_Ctx,
  1440. ArrayRef<uint8_t>((const uint8_t *)Data, Size));
  1441. Args.emplace_back(MDNode::get(m_Ctx, { ConstantAsMetadata::get(V) }));
  1442. Args.emplace_back(MDString::get(m_Ctx, Text));
  1443. break;
  1444. }
  1445. case DXIL::SubobjectKind::SubobjectToExportsAssociation: {
  1446. StringRef Subobj;
  1447. const char * const * Exports;
  1448. uint32_t NumExports;
  1449. IFTBOOL(obj.GetSubobjectToExportsAssociation(Subobj, Exports, NumExports),
  1450. DXC_E_INCORRECT_DXIL_METADATA);
  1451. SmallVector<Metadata *, 4> strArgs;
  1452. for (unsigned i = 0; i < NumExports; ++i) {
  1453. strArgs.emplace_back(MDString::get(m_Ctx, Exports[i]));
  1454. }
  1455. Args.emplace_back(MDString::get(m_Ctx, Subobj));
  1456. Args.emplace_back(MDNode::get(m_Ctx, strArgs));
  1457. break;
  1458. }
  1459. case DXIL::SubobjectKind::RaytracingShaderConfig: {
  1460. uint32_t MaxPayloadSizeInBytes;
  1461. uint32_t MaxAttributeSizeInBytes;
  1462. IFTBOOL(obj.GetRaytracingShaderConfig(MaxPayloadSizeInBytes,
  1463. MaxAttributeSizeInBytes),
  1464. DXC_E_INCORRECT_DXIL_METADATA);
  1465. Args.emplace_back(Uint32ToConstMD(MaxPayloadSizeInBytes));
  1466. Args.emplace_back(Uint32ToConstMD(MaxAttributeSizeInBytes));
  1467. break;
  1468. }
  1469. case DXIL::SubobjectKind::RaytracingPipelineConfig: {
  1470. uint32_t MaxTraceRecursionDepth;
  1471. IFTBOOL(obj.GetRaytracingPipelineConfig(MaxTraceRecursionDepth),
  1472. DXC_E_INCORRECT_DXIL_METADATA);
  1473. Args.emplace_back(Uint32ToConstMD(MaxTraceRecursionDepth));
  1474. break;
  1475. }
  1476. case DXIL::SubobjectKind::HitGroup: {
  1477. llvm::StringRef Intersection, AnyHit, ClosestHit;
  1478. DXIL::HitGroupType hgType;
  1479. IFTBOOL(obj.GetHitGroup(hgType, Intersection, AnyHit, ClosestHit),
  1480. DXC_E_INCORRECT_DXIL_METADATA);
  1481. Args.emplace_back(Uint32ToConstMD((uint32_t)hgType));
  1482. Args.emplace_back(MDString::get(m_Ctx, Intersection));
  1483. Args.emplace_back(MDString::get(m_Ctx, AnyHit));
  1484. Args.emplace_back(MDString::get(m_Ctx, ClosestHit));
  1485. break;
  1486. }
  1487. case DXIL::SubobjectKind::RaytracingPipelineConfig1: {
  1488. uint32_t MaxTraceRecursionDepth;
  1489. uint32_t Flags;
  1490. IFTBOOL(obj.GetRaytracingPipelineConfig1(MaxTraceRecursionDepth, Flags),
  1491. DXC_E_INCORRECT_DXIL_METADATA);
  1492. Args.emplace_back(Uint32ToConstMD(MaxTraceRecursionDepth));
  1493. Args.emplace_back(Uint32ToConstMD(Flags));
  1494. break;
  1495. }
  1496. default:
  1497. DXASSERT(false, "otherwise, we didn't handle a valid subobject kind");
  1498. m_bExtraMetadata = true;
  1499. break;
  1500. }
  1501. return MDNode::get(m_Ctx, Args);
  1502. }
  1503. void DxilMDHelper::LoadSubobject(const llvm::MDNode &MD, DxilSubobjects &Subobjects) {
  1504. IFTBOOL(MD.getNumOperands() >= 2, DXC_E_INCORRECT_DXIL_METADATA);
  1505. unsigned i = 0;
  1506. StringRef name(StringMDToStringRef(MD.getOperand(i++)));
  1507. DXIL::SubobjectKind kind = (DXIL::SubobjectKind)ConstMDToUint32(MD.getOperand(i++));
  1508. IFTBOOL(DXIL::IsValidSubobjectKind(kind), DXC_E_INCORRECT_DXIL_METADATA);
  1509. bool bLocalRS = false;
  1510. switch (kind) {
  1511. case DXIL::SubobjectKind::StateObjectConfig: {
  1512. uint32_t Flags = ConstMDToUint32(MD.getOperand(i++));
  1513. IFTBOOL(0 == ((~(uint32_t)DXIL::StateObjectFlags::ValidMask) & Flags),
  1514. DXC_E_INCORRECT_DXIL_METADATA);
  1515. Subobjects.CreateStateObjectConfig(name, Flags);
  1516. break;
  1517. }
  1518. case DXIL::SubobjectKind::LocalRootSignature:
  1519. bLocalRS = true;
  1520. __fallthrough;
  1521. case DXIL::SubobjectKind::GlobalRootSignature: {
  1522. const MDNode *pDataMDWrapper = dyn_cast<MDNode>(MD.getOperand(i++));
  1523. IFTBOOL(pDataMDWrapper != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1524. IFTBOOL(pDataMDWrapper->getNumOperands() == 1, DXC_E_INCORRECT_DXIL_METADATA);
  1525. const ConstantAsMetadata *pDataMD = dyn_cast<ConstantAsMetadata>(pDataMDWrapper->getOperand(0));
  1526. const ConstantDataArray *pData = dyn_cast<ConstantDataArray>(pDataMD->getValue());
  1527. IFTBOOL(pData != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1528. IFTBOOL(pData->getElementType() == Type::getInt8Ty(m_Ctx), DXC_E_INCORRECT_DXIL_METADATA);
  1529. IFTBOOL(pData->getRawDataValues().size() < UINT_MAX &&
  1530. (pData->getRawDataValues().size() & 3) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  1531. const void *Data = pData->getRawDataValues().begin();
  1532. uint32_t Size = pData->getRawDataValues().size();
  1533. StringRef Text(StringMDToStringRef(MD.getOperand(i++)));
  1534. Subobjects.CreateRootSignature(name, bLocalRS, Data, Size, Text.size() ? &Text : nullptr);
  1535. break;
  1536. }
  1537. case DXIL::SubobjectKind::SubobjectToExportsAssociation: {
  1538. StringRef Subobj(StringMDToStringRef(MD.getOperand(i++)));
  1539. const MDNode *exportMD = dyn_cast<MDNode>(MD.getOperand(i++));
  1540. SmallVector<StringRef, 4> Exports;
  1541. for (unsigned iExport = 0; iExport < exportMD->getNumOperands(); iExport++) {
  1542. Exports.push_back(StringMDToStringRef(exportMD->getOperand(iExport)));
  1543. }
  1544. Subobjects.CreateSubobjectToExportsAssociation(name, Subobj, Exports.data(), Exports.size());
  1545. break;
  1546. }
  1547. case DXIL::SubobjectKind::RaytracingShaderConfig: {
  1548. uint32_t MaxPayloadSizeInBytes = ConstMDToUint32(MD.getOperand(i++));;
  1549. uint32_t MaxAttributeSizeInBytes = ConstMDToUint32(MD.getOperand(i++));;
  1550. Subobjects.CreateRaytracingShaderConfig(name, MaxPayloadSizeInBytes, MaxAttributeSizeInBytes);
  1551. break;
  1552. }
  1553. case DXIL::SubobjectKind::RaytracingPipelineConfig: {
  1554. uint32_t MaxTraceRecursionDepth = ConstMDToUint32(MD.getOperand(i++));;
  1555. Subobjects.CreateRaytracingPipelineConfig(name, MaxTraceRecursionDepth);
  1556. break;
  1557. }
  1558. case DXIL::SubobjectKind::HitGroup: {
  1559. uint32_t hgType = ConstMDToUint32(MD.getOperand(i++));
  1560. StringRef Intersection(StringMDToStringRef(MD.getOperand(i++)));
  1561. StringRef AnyHit(StringMDToStringRef(MD.getOperand(i++)));
  1562. StringRef ClosestHit(StringMDToStringRef(MD.getOperand(i++)));
  1563. Subobjects.CreateHitGroup(name, (DXIL::HitGroupType)hgType, AnyHit, ClosestHit, Intersection);
  1564. break;
  1565. }
  1566. case DXIL::SubobjectKind::RaytracingPipelineConfig1: {
  1567. uint32_t MaxTraceRecursionDepth = ConstMDToUint32(MD.getOperand(i++));
  1568. uint32_t Flags = ConstMDToUint32(MD.getOperand(i++));
  1569. IFTBOOL(0 ==
  1570. ((~(uint32_t)DXIL::RaytracingPipelineFlags::ValidMask) & Flags),
  1571. DXC_E_INCORRECT_DXIL_METADATA);
  1572. Subobjects.CreateRaytracingPipelineConfig1(name, MaxTraceRecursionDepth,
  1573. Flags);
  1574. break;
  1575. }
  1576. default:
  1577. DXASSERT(false, "otherwise, we didn't handle a valid subobject kind");
  1578. m_bExtraMetadata = true;
  1579. break;
  1580. }
  1581. }
  1582. MDTuple *DxilMDHelper::EmitDxilSampler(const DxilSampler &S) {
  1583. Metadata *MDVals[kDxilSamplerNumFields];
  1584. EmitDxilResourceBase(S, &MDVals[0]);
  1585. // Sampler-specific fields.
  1586. MDVals[kDxilSamplerType ] = Uint32ToConstMD((unsigned)S.GetSamplerKind());
  1587. // Name-value list of extended properties.
  1588. MDVals[kDxilSamplerNameValueList] = nullptr;
  1589. vector<Metadata *> MDExtraVals;
  1590. m_ExtraPropertyHelper->EmitSamplerProperties(S, MDExtraVals);
  1591. if (!MDExtraVals.empty()) {
  1592. MDVals[kDxilSamplerNameValueList] = MDNode::get(m_Ctx, MDExtraVals);
  1593. }
  1594. return MDNode::get(m_Ctx, MDVals);
  1595. }
  1596. void DxilMDHelper::LoadDxilSampler(const MDOperand &MDO, DxilSampler &S) {
  1597. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1598. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1599. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1600. IFTBOOL(pTupleMD->getNumOperands() == kDxilSamplerNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  1601. LoadDxilResourceBase(MDO, S);
  1602. // Sampler-specific fields.
  1603. S.SetSamplerKind((DxilSampler::SamplerKind)ConstMDToUint32(pTupleMD->getOperand(kDxilSamplerType)));
  1604. // Name-value list of extended properties.
  1605. m_ExtraPropertyHelper->m_bExtraMetadata = false;
  1606. m_ExtraPropertyHelper->LoadSamplerProperties(pTupleMD->getOperand(kDxilSamplerNameValueList), S);
  1607. m_bExtraMetadata |= m_ExtraPropertyHelper->m_bExtraMetadata;
  1608. }
  1609. const MDOperand &DxilMDHelper::GetResourceClass(llvm::MDNode *MD,
  1610. DXIL::ResourceClass &RC) {
  1611. IFTBOOL(MD->getNumOperands() >=
  1612. DxilMDHelper::kHLDxilResourceAttributeNumFields,
  1613. DXC_E_INCORRECT_DXIL_METADATA);
  1614. RC = static_cast<DxilResource::Class>(ConstMDToUint32(
  1615. MD->getOperand(DxilMDHelper::kHLDxilResourceAttributeClass)));
  1616. return MD->getOperand(DxilMDHelper::kHLDxilResourceAttributeMeta);
  1617. }
  1618. void DxilMDHelper::LoadDxilResourceBaseFromMDNode(llvm::MDNode *MD,
  1619. DxilResourceBase &R) {
  1620. DxilResource::Class RC = DxilResource::Class::Invalid;
  1621. const MDOperand &Meta = GetResourceClass(MD, RC);
  1622. switch (RC) {
  1623. case DxilResource::Class::CBuffer: {
  1624. DxilCBuffer CB;
  1625. LoadDxilCBuffer(Meta, CB);
  1626. R = CB;
  1627. } break;
  1628. case DxilResource::Class::Sampler: {
  1629. DxilSampler S;
  1630. LoadDxilSampler(Meta, S);
  1631. R = S;
  1632. } break;
  1633. case DxilResource::Class::SRV: {
  1634. DxilResource Res;
  1635. LoadDxilSRV(Meta, Res);
  1636. R = Res;
  1637. } break;
  1638. case DxilResource::Class::UAV: {
  1639. DxilResource Res;
  1640. LoadDxilUAV(Meta, Res);
  1641. R = Res;
  1642. } break;
  1643. default:
  1644. DXASSERT(0, "Invalid metadata");
  1645. }
  1646. }
  1647. void DxilMDHelper::LoadDxilResourceFromMDNode(llvm::MDNode *MD,
  1648. DxilResource &R) {
  1649. DxilResource::Class RC = DxilResource::Class::Invalid;
  1650. const MDOperand &Meta = GetResourceClass(MD, RC);
  1651. switch (RC) {
  1652. case DxilResource::Class::SRV: {
  1653. LoadDxilSRV(Meta, R);
  1654. } break;
  1655. case DxilResource::Class::UAV: {
  1656. LoadDxilUAV(Meta, R);
  1657. } break;
  1658. default:
  1659. DXASSERT(0, "Invalid metadata");
  1660. }
  1661. }
  1662. void DxilMDHelper::LoadDxilSamplerFromMDNode(llvm::MDNode *MD, DxilSampler &S) {
  1663. DxilResource::Class RC = DxilResource::Class::Invalid;
  1664. const MDOperand &Meta = GetResourceClass(MD, RC);
  1665. switch (RC) {
  1666. case DxilResource::Class::Sampler: {
  1667. LoadDxilSampler(Meta, S);
  1668. } break;
  1669. default:
  1670. DXASSERT(0, "Invalid metadata");
  1671. }
  1672. }
  1673. //
  1674. // DxilExtraPropertyHelper shader-specific methods.
  1675. //
  1676. MDTuple *DxilMDHelper::EmitDxilGSState(DXIL::InputPrimitive Primitive,
  1677. unsigned MaxVertexCount,
  1678. unsigned ActiveStreamMask,
  1679. DXIL::PrimitiveTopology StreamPrimitiveTopology,
  1680. unsigned GSInstanceCount) {
  1681. Metadata *MDVals[kDxilGSStateNumFields];
  1682. MDVals[kDxilGSStateInputPrimitive ] = Uint32ToConstMD((unsigned)Primitive);
  1683. MDVals[kDxilGSStateMaxVertexCount ] = Uint32ToConstMD(MaxVertexCount);
  1684. MDVals[kDxilGSStateActiveStreamMask ] = Uint32ToConstMD(ActiveStreamMask);
  1685. MDVals[kDxilGSStateOutputStreamTopology] = Uint32ToConstMD((unsigned)StreamPrimitiveTopology);
  1686. MDVals[kDxilGSStateGSInstanceCount ] = Uint32ToConstMD(GSInstanceCount);
  1687. return MDNode::get(m_Ctx, MDVals);
  1688. }
  1689. void DxilMDHelper::LoadDxilGSState(const MDOperand &MDO,
  1690. DXIL::InputPrimitive &Primitive,
  1691. unsigned &MaxVertexCount,
  1692. unsigned &ActiveStreamMask,
  1693. DXIL::PrimitiveTopology &StreamPrimitiveTopology,
  1694. unsigned &GSInstanceCount) {
  1695. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1696. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1697. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1698. IFTBOOL(pTupleMD->getNumOperands() == kDxilGSStateNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  1699. Primitive = (DXIL::InputPrimitive)ConstMDToUint32(pTupleMD->getOperand(kDxilGSStateInputPrimitive));
  1700. MaxVertexCount = ConstMDToUint32(pTupleMD->getOperand(kDxilGSStateMaxVertexCount));
  1701. ActiveStreamMask = ConstMDToUint32(pTupleMD->getOperand(kDxilGSStateActiveStreamMask));
  1702. StreamPrimitiveTopology = (DXIL::PrimitiveTopology)ConstMDToUint32(pTupleMD->getOperand(kDxilGSStateOutputStreamTopology));
  1703. GSInstanceCount = ConstMDToUint32(pTupleMD->getOperand(kDxilGSStateGSInstanceCount));
  1704. }
  1705. MDTuple *DxilMDHelper::EmitDxilDSState(DXIL::TessellatorDomain Domain, unsigned InputControlPointCount) {
  1706. Metadata *MDVals[kDxilDSStateNumFields];
  1707. MDVals[kDxilDSStateTessellatorDomain ] = Uint32ToConstMD((unsigned)Domain);
  1708. MDVals[kDxilDSStateInputControlPointCount] = Uint32ToConstMD(InputControlPointCount);
  1709. return MDNode::get(m_Ctx, MDVals);
  1710. }
  1711. void DxilMDHelper::LoadDxilDSState(const MDOperand &MDO,
  1712. DXIL::TessellatorDomain &Domain,
  1713. unsigned &InputControlPointCount) {
  1714. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1715. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1716. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1717. IFTBOOL(pTupleMD->getNumOperands() == kDxilDSStateNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  1718. Domain = (DXIL::TessellatorDomain)ConstMDToUint32(pTupleMD->getOperand(kDxilDSStateTessellatorDomain));
  1719. InputControlPointCount = ConstMDToUint32(pTupleMD->getOperand(kDxilDSStateInputControlPointCount));
  1720. }
  1721. MDTuple *DxilMDHelper::EmitDxilHSState(Function *pPatchConstantFunction,
  1722. unsigned InputControlPointCount,
  1723. unsigned OutputControlPointCount,
  1724. DXIL::TessellatorDomain TessDomain,
  1725. DXIL::TessellatorPartitioning TessPartitioning,
  1726. DXIL::TessellatorOutputPrimitive TessOutputPrimitive,
  1727. float MaxTessFactor) {
  1728. Metadata *MDVals[kDxilHSStateNumFields];
  1729. MDVals[kDxilHSStatePatchConstantFunction ] = ValueAsMetadata::get(pPatchConstantFunction);
  1730. MDVals[kDxilHSStateInputControlPointCount ] = Uint32ToConstMD(InputControlPointCount);
  1731. MDVals[kDxilHSStateOutputControlPointCount ] = Uint32ToConstMD(OutputControlPointCount);
  1732. MDVals[kDxilHSStateTessellatorDomain ] = Uint32ToConstMD((unsigned)TessDomain);
  1733. MDVals[kDxilHSStateTessellatorPartitioning ] = Uint32ToConstMD((unsigned)TessPartitioning);
  1734. MDVals[kDxilHSStateTessellatorOutputPrimitive] = Uint32ToConstMD((unsigned)TessOutputPrimitive);
  1735. MDVals[kDxilHSStateMaxTessellationFactor ] = FloatToConstMD(MaxTessFactor);
  1736. return MDNode::get(m_Ctx, MDVals);
  1737. }
  1738. void DxilMDHelper::LoadDxilHSState(const MDOperand &MDO,
  1739. Function *&pPatchConstantFunction,
  1740. unsigned &InputControlPointCount,
  1741. unsigned &OutputControlPointCount,
  1742. DXIL::TessellatorDomain &TessDomain,
  1743. DXIL::TessellatorPartitioning &TessPartitioning,
  1744. DXIL::TessellatorOutputPrimitive &TessOutputPrimitive,
  1745. float &MaxTessFactor) {
  1746. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1747. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1748. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1749. IFTBOOL(pTupleMD->getNumOperands() == kDxilHSStateNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  1750. pPatchConstantFunction = dyn_cast<Function>(ValueMDToValue(pTupleMD->getOperand(kDxilHSStatePatchConstantFunction)));
  1751. InputControlPointCount = ConstMDToUint32(pTupleMD->getOperand(kDxilHSStateInputControlPointCount));
  1752. OutputControlPointCount = ConstMDToUint32(pTupleMD->getOperand(kDxilHSStateOutputControlPointCount));
  1753. TessDomain = (DXIL::TessellatorDomain)ConstMDToUint32(pTupleMD->getOperand(kDxilHSStateTessellatorDomain));
  1754. TessPartitioning = (DXIL::TessellatorPartitioning)ConstMDToUint32(pTupleMD->getOperand(kDxilHSStateTessellatorPartitioning));
  1755. TessOutputPrimitive = (DXIL::TessellatorOutputPrimitive)ConstMDToUint32(pTupleMD->getOperand(kDxilHSStateTessellatorOutputPrimitive));
  1756. MaxTessFactor = ConstMDToFloat(pTupleMD->getOperand(kDxilHSStateMaxTessellationFactor));
  1757. }
  1758. MDTuple *DxilMDHelper::EmitDxilMSState(const unsigned *NumThreads,
  1759. unsigned MaxVertexCount,
  1760. unsigned MaxPrimitiveCount,
  1761. DXIL::MeshOutputTopology OutputTopology,
  1762. unsigned payloadSizeInBytes) {
  1763. Metadata *MDVals[kDxilMSStateNumFields];
  1764. vector<Metadata *> NumThreadVals;
  1765. NumThreadVals.emplace_back(Uint32ToConstMD(NumThreads[0]));
  1766. NumThreadVals.emplace_back(Uint32ToConstMD(NumThreads[1]));
  1767. NumThreadVals.emplace_back(Uint32ToConstMD(NumThreads[2]));
  1768. MDVals[kDxilMSStateNumThreads] = MDNode::get(m_Ctx, NumThreadVals);
  1769. MDVals[kDxilMSStateMaxVertexCount] = Uint32ToConstMD(MaxVertexCount);
  1770. MDVals[kDxilMSStateMaxPrimitiveCount] = Uint32ToConstMD(MaxPrimitiveCount);
  1771. MDVals[kDxilMSStateOutputTopology] = Uint32ToConstMD((unsigned)OutputTopology);
  1772. MDVals[kDxilMSStatePayloadSizeInBytes] = Uint32ToConstMD(payloadSizeInBytes);
  1773. return MDNode::get(m_Ctx, MDVals);
  1774. }
  1775. void DxilMDHelper::LoadDxilMSState(const MDOperand &MDO,
  1776. unsigned *NumThreads,
  1777. unsigned &MaxVertexCount,
  1778. unsigned &MaxPrimitiveCount,
  1779. DXIL::MeshOutputTopology &OutputTopology,
  1780. unsigned &payloadSizeInBytes) {
  1781. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1782. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1783. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1784. IFTBOOL(pTupleMD->getNumOperands() == kDxilMSStateNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  1785. MDNode *pNode = cast<MDNode>(pTupleMD->getOperand(kDxilMSStateNumThreads));
  1786. NumThreads[0] = ConstMDToUint32(pNode->getOperand(0));
  1787. NumThreads[1] = ConstMDToUint32(pNode->getOperand(1));
  1788. NumThreads[2] = ConstMDToUint32(pNode->getOperand(2));
  1789. MaxVertexCount = ConstMDToUint32(pTupleMD->getOperand(kDxilMSStateMaxVertexCount));
  1790. MaxPrimitiveCount = ConstMDToUint32(pTupleMD->getOperand(kDxilMSStateMaxPrimitiveCount));
  1791. OutputTopology = (DXIL::MeshOutputTopology)ConstMDToUint32(pTupleMD->getOperand(kDxilMSStateOutputTopology));
  1792. payloadSizeInBytes = ConstMDToUint32(pTupleMD->getOperand(kDxilMSStatePayloadSizeInBytes));
  1793. }
  1794. MDTuple *DxilMDHelper::EmitDxilASState(const unsigned *NumThreads, unsigned payloadSizeInBytes) {
  1795. Metadata *MDVals[kDxilASStateNumFields];
  1796. vector<Metadata *> NumThreadVals;
  1797. NumThreadVals.emplace_back(Uint32ToConstMD(NumThreads[0]));
  1798. NumThreadVals.emplace_back(Uint32ToConstMD(NumThreads[1]));
  1799. NumThreadVals.emplace_back(Uint32ToConstMD(NumThreads[2]));
  1800. MDVals[kDxilASStateNumThreads] = MDNode::get(m_Ctx, NumThreadVals);
  1801. MDVals[kDxilASStatePayloadSizeInBytes] = Uint32ToConstMD(payloadSizeInBytes);
  1802. return MDNode::get(m_Ctx, MDVals);
  1803. }
  1804. void DxilMDHelper::LoadDxilASState(const MDOperand &MDO, unsigned *NumThreads, unsigned &payloadSizeInBytes) {
  1805. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1806. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1807. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1808. IFTBOOL(pTupleMD->getNumOperands() == kDxilASStateNumFields, DXC_E_INCORRECT_DXIL_METADATA);
  1809. MDNode *pNode = cast<MDNode>(pTupleMD->getOperand(kDxilASStateNumThreads));
  1810. NumThreads[0] = ConstMDToUint32(pNode->getOperand(0));
  1811. NumThreads[1] = ConstMDToUint32(pNode->getOperand(1));
  1812. NumThreads[2] = ConstMDToUint32(pNode->getOperand(2));
  1813. payloadSizeInBytes = ConstMDToUint32(pTupleMD->getOperand(kDxilASStatePayloadSizeInBytes));
  1814. }
  1815. //
  1816. // DxilExtraPropertyHelper methods.
  1817. //
  1818. DxilMDHelper::ExtraPropertyHelper::ExtraPropertyHelper(Module *pModule)
  1819. : m_Ctx(pModule->getContext())
  1820. , m_pModule(pModule)
  1821. , m_bExtraMetadata(false) {
  1822. }
  1823. DxilExtraPropertyHelper::DxilExtraPropertyHelper(Module *pModule)
  1824. : ExtraPropertyHelper(pModule) {
  1825. }
  1826. void DxilExtraPropertyHelper::EmitSRVProperties(const DxilResource &SRV, std::vector<Metadata *> &MDVals) {
  1827. // Element type for typed resource.
  1828. if (!SRV.IsStructuredBuffer() && !SRV.IsRawBuffer()) {
  1829. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilTypedBufferElementTypeTag, m_Ctx));
  1830. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD((unsigned)SRV.GetCompType().GetKind(), m_Ctx));
  1831. }
  1832. // Element stride for structured buffer.
  1833. if (SRV.IsStructuredBuffer()) {
  1834. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilStructuredBufferElementStrideTag, m_Ctx));
  1835. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(SRV.GetElementStride(), m_Ctx));
  1836. }
  1837. }
  1838. void DxilExtraPropertyHelper::LoadSRVProperties(const MDOperand &MDO, DxilResource &SRV) {
  1839. SRV.SetElementStride(SRV.IsRawBuffer() ? 1 : 4);
  1840. SRV.SetCompType(CompType());
  1841. if (MDO.get() == nullptr) {
  1842. return;
  1843. }
  1844. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1845. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1846. IFTBOOL((pTupleMD->getNumOperands() & 0x1) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  1847. for (unsigned i = 0; i < pTupleMD->getNumOperands(); i += 2) {
  1848. unsigned Tag = DxilMDHelper::ConstMDToUint32(pTupleMD->getOperand(i));
  1849. const MDOperand &MDO = pTupleMD->getOperand(i + 1);
  1850. switch (Tag) {
  1851. case DxilMDHelper::kDxilTypedBufferElementTypeTag:
  1852. DXASSERT_NOMSG(!SRV.IsStructuredBuffer() && !SRV.IsRawBuffer());
  1853. SRV.SetCompType(CompType(DxilMDHelper::ConstMDToUint32(MDO)));
  1854. break;
  1855. case DxilMDHelper::kDxilStructuredBufferElementStrideTag:
  1856. DXASSERT_NOMSG(SRV.IsStructuredBuffer());
  1857. SRV.SetElementStride(DxilMDHelper::ConstMDToUint32(MDO));
  1858. break;
  1859. default:
  1860. DXASSERT(false, "Unknown resource record tag");
  1861. m_bExtraMetadata = true;
  1862. break;
  1863. }
  1864. }
  1865. }
  1866. void DxilExtraPropertyHelper::EmitUAVProperties(const DxilResource &UAV, std::vector<Metadata *> &MDVals) {
  1867. // Element type for typed RW resource.
  1868. if (!UAV.IsStructuredBuffer() && !UAV.IsRawBuffer() && !UAV.GetCompType().IsInvalid()) {
  1869. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilTypedBufferElementTypeTag, m_Ctx));
  1870. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD((unsigned)UAV.GetCompType().GetKind(), m_Ctx));
  1871. }
  1872. // Element stride for structured RW buffer.
  1873. if (UAV.IsStructuredBuffer()) {
  1874. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilStructuredBufferElementStrideTag, m_Ctx));
  1875. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(UAV.GetElementStride(), m_Ctx));
  1876. }
  1877. // Sampler feedback kind
  1878. if (UAV.IsFeedbackTexture()) {
  1879. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilSamplerFeedbackKindTag, m_Ctx));
  1880. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD((unsigned)UAV.GetSamplerFeedbackType(), m_Ctx));
  1881. }
  1882. }
  1883. void DxilExtraPropertyHelper::LoadUAVProperties(const MDOperand &MDO, DxilResource &UAV) {
  1884. UAV.SetElementStride(UAV.IsRawBuffer() ? 1 : 4);
  1885. UAV.SetCompType(CompType());
  1886. if (MDO.get() == nullptr) {
  1887. return;
  1888. }
  1889. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1890. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1891. IFTBOOL((pTupleMD->getNumOperands() & 0x1) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  1892. for (unsigned i = 0; i < pTupleMD->getNumOperands(); i += 2) {
  1893. unsigned Tag = DxilMDHelper::ConstMDToUint32(pTupleMD->getOperand(i));
  1894. const MDOperand &MDO = pTupleMD->getOperand(i + 1);
  1895. switch (Tag) {
  1896. case DxilMDHelper::kDxilTypedBufferElementTypeTag:
  1897. DXASSERT_NOMSG(!UAV.IsStructuredBuffer() && !UAV.IsRawBuffer());
  1898. UAV.SetCompType(CompType(DxilMDHelper::ConstMDToUint32(MDO)));
  1899. break;
  1900. case DxilMDHelper::kDxilStructuredBufferElementStrideTag:
  1901. DXASSERT_NOMSG(UAV.IsStructuredBuffer());
  1902. UAV.SetElementStride(DxilMDHelper::ConstMDToUint32(MDO));
  1903. break;
  1904. case DxilMDHelper::kDxilSamplerFeedbackKindTag:
  1905. DXASSERT_NOMSG(UAV.IsFeedbackTexture());
  1906. UAV.SetSamplerFeedbackType((DXIL::SamplerFeedbackType)DxilMDHelper::ConstMDToUint32(MDO));
  1907. break;
  1908. default:
  1909. DXASSERT(false, "Unknown resource record tag");
  1910. m_bExtraMetadata = true;
  1911. break;
  1912. }
  1913. }
  1914. }
  1915. void DxilExtraPropertyHelper::EmitCBufferProperties(const DxilCBuffer &CB, vector<Metadata *> &MDVals) {
  1916. // Emit property to preserve tbuffer kind
  1917. if (CB.GetKind() == DXIL::ResourceKind::TBuffer) {
  1918. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kHLCBufferIsTBufferTag, m_Ctx));
  1919. MDVals.emplace_back(DxilMDHelper::BoolToConstMD(true, m_Ctx));
  1920. }
  1921. }
  1922. void DxilExtraPropertyHelper::LoadCBufferProperties(const MDOperand &MDO, DxilCBuffer &CB) {
  1923. if (MDO.get() == nullptr)
  1924. return;
  1925. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1926. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1927. IFTBOOL((pTupleMD->getNumOperands() & 0x1) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  1928. // Override kind for tbuffer that has not yet been converted to SRV.
  1929. CB.SetKind(DXIL::ResourceKind::CBuffer);
  1930. for (unsigned i = 0; i < pTupleMD->getNumOperands(); i += 2) {
  1931. unsigned Tag = DxilMDHelper::ConstMDToUint32(pTupleMD->getOperand(i));
  1932. const MDOperand &MDO = pTupleMD->getOperand(i + 1);
  1933. switch (Tag) {
  1934. case DxilMDHelper::kHLCBufferIsTBufferTag:
  1935. if (DxilMDHelper::ConstMDToBool(MDO)) {
  1936. CB.SetKind(DXIL::ResourceKind::TBuffer);
  1937. }
  1938. break;
  1939. default:
  1940. DXASSERT(false, "Unknown cbuffer tag");
  1941. m_bExtraMetadata = true;
  1942. break;
  1943. }
  1944. }
  1945. }
  1946. void DxilExtraPropertyHelper::EmitSamplerProperties(const DxilSampler &S, std::vector<Metadata *> &MDVals) {
  1947. // Nothing yet.
  1948. }
  1949. void DxilExtraPropertyHelper::LoadSamplerProperties(const MDOperand &MDO, DxilSampler &S) {
  1950. // Nothing yet.
  1951. }
  1952. void DxilExtraPropertyHelper::EmitSignatureElementProperties(const DxilSignatureElement &SE,
  1953. vector<Metadata *> &MDVals) {
  1954. // Output stream, if non-zero.
  1955. if (SE.GetOutputStream() != 0) {
  1956. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilSignatureElementOutputStreamTag, m_Ctx));
  1957. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(SE.GetOutputStream(), m_Ctx));
  1958. }
  1959. // Mask of Dynamically indexed components.
  1960. if (SE.GetDynIdxCompMask() != 0) {
  1961. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilSignatureElementDynIdxCompMaskTag, m_Ctx));
  1962. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(SE.GetDynIdxCompMask(), m_Ctx));
  1963. }
  1964. if (SE.GetUsageMask() != 0 &&
  1965. DXIL::CompareVersions(m_ValMajor, m_ValMinor, 1, 5) >= 0) {
  1966. // Emitting this will not hurt old reatil loader (only asserts),
  1967. // and is required for signatures to match in validation.
  1968. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(DxilMDHelper::kDxilSignatureElementUsageCompMaskTag, m_Ctx));
  1969. MDVals.emplace_back(DxilMDHelper::Uint32ToConstMD(SE.GetUsageMask(), m_Ctx));
  1970. }
  1971. }
  1972. void DxilExtraPropertyHelper::LoadSignatureElementProperties(const MDOperand &MDO, DxilSignatureElement &SE) {
  1973. if (MDO.get() == nullptr)
  1974. return;
  1975. const MDTuple *pTupleMD = dyn_cast<MDTuple>(MDO.get());
  1976. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  1977. IFTBOOL((pTupleMD->getNumOperands() & 0x1) == 0, DXC_E_INCORRECT_DXIL_METADATA);
  1978. // Stream.
  1979. for (unsigned i = 0; i < pTupleMD->getNumOperands(); i += 2) {
  1980. unsigned Tag = DxilMDHelper::ConstMDToUint32(pTupleMD->getOperand(i));
  1981. const MDOperand &MDO = pTupleMD->getOperand(i + 1);
  1982. switch (Tag) {
  1983. case DxilMDHelper::kDxilSignatureElementOutputStreamTag:
  1984. SE.SetOutputStream(DxilMDHelper::ConstMDToUint32(MDO));
  1985. break;
  1986. case DxilMDHelper::kHLSignatureElementGlobalSymbolTag:
  1987. break;
  1988. case DxilMDHelper::kDxilSignatureElementDynIdxCompMaskTag:
  1989. SE.SetDynIdxCompMask(DxilMDHelper::ConstMDToUint32(MDO));
  1990. break;
  1991. case DxilMDHelper::kDxilSignatureElementUsageCompMaskTag:
  1992. SE.SetUsageMask(DxilMDHelper::ConstMDToUint32(MDO));
  1993. break;
  1994. default:
  1995. DXASSERT(false, "Unknown signature element tag");
  1996. m_bExtraMetadata = true;
  1997. break;
  1998. }
  1999. }
  2000. }
  2001. //
  2002. // Utilities.
  2003. //
  2004. bool DxilMDHelper::IsKnownNamedMetaData(const llvm::NamedMDNode &Node) {
  2005. StringRef name = Node.getName();
  2006. for (unsigned i = 0; i < DxilMDNames.size(); i++) {
  2007. if (name == DxilMDNames[i]) {
  2008. return true;
  2009. }
  2010. }
  2011. return false;
  2012. }
  2013. void DxilMDHelper::combineDxilMetadata(llvm::Instruction *K,
  2014. const llvm::Instruction *J) {
  2015. if (IsMarkedNonUniform(J))
  2016. MarkNonUniform(K);
  2017. if (IsMarkedPrecise(J))
  2018. MarkPrecise(K);
  2019. }
  2020. ConstantAsMetadata *DxilMDHelper::Int32ToConstMD(int32_t v, LLVMContext &Ctx) {
  2021. return ConstantAsMetadata::get(Constant::getIntegerValue(IntegerType::get(Ctx, 32), APInt(32, v)));
  2022. }
  2023. ConstantAsMetadata *DxilMDHelper::Int32ToConstMD(int32_t v) {
  2024. return DxilMDHelper::Int32ToConstMD(v, m_Ctx);
  2025. }
  2026. ConstantAsMetadata *DxilMDHelper::Uint32ToConstMD(unsigned v, LLVMContext &Ctx) {
  2027. return ConstantAsMetadata::get(Constant::getIntegerValue(IntegerType::get(Ctx, 32), APInt(32, v)));
  2028. }
  2029. ConstantAsMetadata *DxilMDHelper::Uint32ToConstMD(unsigned v) {
  2030. return DxilMDHelper::Uint32ToConstMD(v, m_Ctx);
  2031. }
  2032. ConstantAsMetadata *DxilMDHelper::Uint64ToConstMD(uint64_t v, LLVMContext &Ctx) {
  2033. return ConstantAsMetadata::get(Constant::getIntegerValue(IntegerType::get(Ctx, 64), APInt(64, v)));
  2034. }
  2035. ConstantAsMetadata *DxilMDHelper::Uint64ToConstMD(uint64_t v) {
  2036. return DxilMDHelper::Uint64ToConstMD(v, m_Ctx);
  2037. }
  2038. ConstantAsMetadata *DxilMDHelper::Int8ToConstMD(int8_t v) {
  2039. return ConstantAsMetadata::get(Constant::getIntegerValue(IntegerType::get(m_Ctx, 8), APInt(8, v)));
  2040. }
  2041. ConstantAsMetadata *DxilMDHelper::Uint8ToConstMD(uint8_t v) {
  2042. return ConstantAsMetadata::get(Constant::getIntegerValue(IntegerType::get(m_Ctx, 8), APInt(8, v)));
  2043. }
  2044. ConstantAsMetadata *DxilMDHelper::BoolToConstMD(bool v, LLVMContext &Ctx) {
  2045. return ConstantAsMetadata::get(Constant::getIntegerValue(IntegerType::get(Ctx, 1), APInt(1, v ? 1 : 0)));
  2046. }
  2047. ConstantAsMetadata *DxilMDHelper::BoolToConstMD(bool v) {
  2048. return DxilMDHelper::BoolToConstMD(v, m_Ctx);
  2049. }
  2050. ConstantAsMetadata *DxilMDHelper::FloatToConstMD(float v) {
  2051. return ConstantAsMetadata::get(ConstantFP::get(m_Ctx, APFloat(v)));
  2052. }
  2053. int32_t DxilMDHelper::ConstMDToInt32(const MDOperand &MDO) {
  2054. ConstantInt *pConst = mdconst::extract<ConstantInt>(MDO);
  2055. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2056. return (int32_t)pConst->getZExtValue();
  2057. }
  2058. unsigned DxilMDHelper::ConstMDToUint32(const MDOperand &MDO) {
  2059. ConstantInt *pConst = mdconst::extract<ConstantInt>(MDO);
  2060. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2061. return (unsigned)pConst->getZExtValue();
  2062. }
  2063. uint64_t DxilMDHelper::ConstMDToUint64(const MDOperand &MDO) {
  2064. ConstantInt *pConst = mdconst::extract<ConstantInt>(MDO);
  2065. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2066. return pConst->getZExtValue();
  2067. }
  2068. int8_t DxilMDHelper::ConstMDToInt8(const MDOperand &MDO) {
  2069. ConstantInt *pConst = mdconst::extract<ConstantInt>(MDO);
  2070. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2071. return (int8_t)pConst->getZExtValue();
  2072. }
  2073. uint8_t DxilMDHelper::ConstMDToUint8(const MDOperand &MDO) {
  2074. ConstantInt *pConst = mdconst::extract<ConstantInt>(MDO);
  2075. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2076. return (uint8_t)pConst->getZExtValue();
  2077. }
  2078. bool DxilMDHelper::ConstMDToBool(const MDOperand &MDO) {
  2079. ConstantInt *pConst = mdconst::extract<ConstantInt>(MDO);
  2080. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2081. return pConst->getZExtValue() != 0;
  2082. }
  2083. float DxilMDHelper::ConstMDToFloat(const MDOperand &MDO) {
  2084. ConstantFP *pConst = mdconst::extract<ConstantFP>(MDO);
  2085. IFTBOOL(pConst != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2086. return pConst->getValueAPF().convertToFloat();
  2087. }
  2088. string DxilMDHelper::StringMDToString(const MDOperand &MDO) {
  2089. MDString *pMDString = dyn_cast<MDString>(MDO.get());
  2090. IFTBOOL(pMDString != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2091. return pMDString->getString();
  2092. }
  2093. StringRef DxilMDHelper::StringMDToStringRef(const MDOperand &MDO) {
  2094. MDString *pMDString = dyn_cast<MDString>(MDO.get());
  2095. IFTBOOL(pMDString != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2096. return pMDString->getString();
  2097. }
  2098. Value *DxilMDHelper::ValueMDToValue(const MDOperand &MDO) {
  2099. IFTBOOL(MDO.get() != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2100. ValueAsMetadata *pValAsMD = dyn_cast<ValueAsMetadata>(MDO.get());
  2101. IFTBOOL(pValAsMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2102. Value *pValue = pValAsMD->getValue();
  2103. IFTBOOL(pValue != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2104. return pValue;
  2105. }
  2106. MDTuple *DxilMDHelper::Uint32VectorToConstMDTuple(const std::vector<unsigned> &Vec) {
  2107. vector<Metadata *> MDVals;
  2108. MDVals.resize(Vec.size());
  2109. for (size_t i = 0; i < Vec.size(); i++) {
  2110. MDVals[i] = Uint32ToConstMD(Vec[i]);
  2111. }
  2112. return MDNode::get(m_Ctx, MDVals);
  2113. }
  2114. void DxilMDHelper::ConstMDTupleToUint32Vector(MDTuple *pTupleMD, std::vector<unsigned> &Vec) {
  2115. IFTBOOL(pTupleMD != nullptr, DXC_E_INCORRECT_DXIL_METADATA);
  2116. Vec.resize(pTupleMD->getNumOperands());
  2117. for (size_t i = 0; i < pTupleMD->getNumOperands(); i++) {
  2118. Vec[i] = ConstMDToUint32(pTupleMD->getOperand(i));
  2119. }
  2120. }
  2121. bool DxilMDHelper::IsMarkedPrecise(const Instruction *inst) {
  2122. int32_t val = 0;
  2123. if (MDNode *precise = inst->getMetadata(kDxilPreciseAttributeMDName)) {
  2124. assert(precise->getNumOperands() == 1);
  2125. val = ConstMDToInt32(precise->getOperand(0));
  2126. }
  2127. return val;
  2128. }
  2129. void DxilMDHelper::MarkPrecise(Instruction *I) {
  2130. LLVMContext &Ctx = I->getContext();
  2131. MDNode *preciseNode = MDNode::get(
  2132. Ctx,
  2133. { ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Ctx), 1)) });
  2134. I->setMetadata(DxilMDHelper::kDxilPreciseAttributeMDName, preciseNode);
  2135. }
  2136. bool DxilMDHelper::IsMarkedNonUniform(const Instruction *inst) {
  2137. int32_t val = 0;
  2138. if (MDNode *precise = inst->getMetadata(kDxilNonUniformAttributeMDName)) {
  2139. assert(precise->getNumOperands() == 1);
  2140. val = ConstMDToInt32(precise->getOperand(0));
  2141. }
  2142. return val;
  2143. }
  2144. void DxilMDHelper::MarkNonUniform(Instruction *I) {
  2145. LLVMContext &Ctx = I->getContext();
  2146. MDNode *preciseNode = MDNode::get(
  2147. Ctx,
  2148. { ConstantAsMetadata::get(ConstantInt::get(Type::getInt32Ty(Ctx), 1)) });
  2149. I->setMetadata(DxilMDHelper::kDxilNonUniformAttributeMDName, preciseNode);
  2150. }
  2151. bool DxilMDHelper::GetVariableDebugLayout(llvm::DbgDeclareInst *inst,
  2152. unsigned &StartOffsetInBits, std::vector<DxilDIArrayDim> &ArrayDims) {
  2153. llvm::MDTuple *Tuple = dyn_cast_or_null<MDTuple>(inst->getMetadata(DxilMDHelper::kDxilVariableDebugLayoutMDName));
  2154. if (Tuple == nullptr) return false;
  2155. IFTBOOL(Tuple->getNumOperands() % 2 == 1, DXC_E_INCORRECT_DXIL_METADATA);
  2156. StartOffsetInBits = ConstMDToUint32(Tuple->getOperand(0));
  2157. for (unsigned Idx = 1; Idx < Tuple->getNumOperands(); Idx += 2) {
  2158. DxilDIArrayDim ArrayDim = {};
  2159. ArrayDim.StrideInBits = ConstMDToUint32(Tuple->getOperand(Idx + 0));
  2160. ArrayDim.NumElements = ConstMDToUint32(Tuple->getOperand(Idx + 1));
  2161. ArrayDims.emplace_back(ArrayDim);
  2162. }
  2163. return true;
  2164. }
  2165. void DxilMDHelper::SetVariableDebugLayout(llvm::DbgDeclareInst *inst,
  2166. unsigned StartOffsetInBits, const std::vector<DxilDIArrayDim> &ArrayDims) {
  2167. LLVMContext &Ctx = inst->getContext();
  2168. std::vector<Metadata*> MDVals;
  2169. MDVals.reserve(ArrayDims.size() + 1);
  2170. MDVals.emplace_back(Uint32ToConstMD(StartOffsetInBits, Ctx));
  2171. for (const DxilDIArrayDim &ArrayDim : ArrayDims) {
  2172. MDVals.emplace_back(Uint32ToConstMD(ArrayDim.StrideInBits, Ctx));
  2173. MDVals.emplace_back(Uint32ToConstMD(ArrayDim.NumElements, Ctx));
  2174. }
  2175. inst->setMetadata(DxilMDHelper::kDxilVariableDebugLayoutMDName, MDNode::get(Ctx, MDVals));
  2176. }
  2177. void DxilMDHelper::CopyMetadata(Instruction &I, Instruction &SrcInst, ArrayRef<unsigned> WL) {
  2178. if (!SrcInst.hasMetadata())
  2179. return;
  2180. DenseSet<unsigned> WLS;
  2181. for (unsigned M : WL)
  2182. WLS.insert(M);
  2183. // Otherwise, enumerate and copy over metadata from the old instruction to the
  2184. // new one.
  2185. SmallVector<std::pair<unsigned, MDNode *>, 4> TheMDs;
  2186. SrcInst.getAllMetadataOtherThanDebugLoc(TheMDs);
  2187. for (const auto &MD : TheMDs) {
  2188. if (WL.empty() || WLS.count(MD.first))
  2189. I.setMetadata(MD.first, MD.second);
  2190. }
  2191. if (WL.empty() || WLS.count(LLVMContext::MD_dbg))
  2192. I.setDebugLoc(SrcInst.getDebugLoc());
  2193. }
  2194. } // namespace hlsl