SubShaderGenerator.cs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using UnityEditor.Graphing;
  5. using UnityEditor.ShaderGraph.Internal;
  6. using Data.Util;
  7. namespace UnityEditor.ShaderGraph
  8. {
  9. static class SubShaderGenerator
  10. {
  11. public static void GeneratePropertiesBlock(ShaderStringBuilder sb, PropertyCollector propertyCollector, KeywordCollector keywordCollector, GenerationMode mode)
  12. {
  13. sb.AppendLine("Properties");
  14. using (sb.BlockScope())
  15. {
  16. foreach (var prop in propertyCollector.properties.Where(x => x.generatePropertyBlock))
  17. {
  18. sb.AppendLine(prop.GetPropertyBlockString());
  19. }
  20. // Keywords use hardcoded state in preview
  21. // Do not add them to the Property Block
  22. if(mode == GenerationMode.Preview)
  23. return;
  24. foreach (var key in keywordCollector.keywords.Where(x => x.generatePropertyBlock))
  25. {
  26. sb.AppendLine(key.GetPropertyBlockString());
  27. }
  28. }
  29. }
  30. public static void GenerateApplicationVertexInputs(ShaderGraphRequirements graphRequirements, ShaderStringBuilder vertexInputs)
  31. {
  32. vertexInputs.AppendLine("struct GraphVertexInput");
  33. using (vertexInputs.BlockSemicolonScope())
  34. {
  35. vertexInputs.AppendLine("float4 vertex : POSITION;");
  36. if(graphRequirements.requiresNormal != NeededCoordinateSpace.None || graphRequirements.requiresBitangent != NeededCoordinateSpace.None)
  37. vertexInputs.AppendLine("float3 normal : NORMAL;");
  38. if(graphRequirements.requiresTangent != NeededCoordinateSpace.None || graphRequirements.requiresBitangent != NeededCoordinateSpace.None)
  39. vertexInputs.AppendLine("float4 tangent : TANGENT;");
  40. if (graphRequirements.requiresVertexColor)
  41. {
  42. vertexInputs.AppendLine("float4 color : COLOR;");
  43. }
  44. foreach (var channel in graphRequirements.requiresMeshUVs.Distinct())
  45. vertexInputs.AppendLine("float4 texcoord{0} : TEXCOORD{0};", (int)channel);
  46. vertexInputs.AppendLine("UNITY_VERTEX_INPUT_INSTANCE_ID");
  47. }
  48. }
  49. public static GenerationResults GetShader(this GraphData graph, AbstractMaterialNode node, GenerationMode mode, string name)
  50. {
  51. // ----------------------------------------------------- //
  52. // SETUP //
  53. // ----------------------------------------------------- //
  54. // -------------------------------------
  55. // String builders
  56. var finalShader = new ShaderStringBuilder();
  57. var results = new GenerationResults();
  58. var shaderProperties = new PropertyCollector();
  59. var shaderKeywords = new KeywordCollector();
  60. var shaderPropertyUniforms = new ShaderStringBuilder();
  61. var shaderKeywordDeclarations = new ShaderStringBuilder();
  62. var shaderKeywordPermutations = new ShaderStringBuilder(1);
  63. var functionBuilder = new ShaderStringBuilder();
  64. var functionRegistry = new FunctionRegistry(functionBuilder);
  65. var vertexDescriptionFunction = new ShaderStringBuilder(0);
  66. var surfaceDescriptionInputStruct = new ShaderStringBuilder(0);
  67. var surfaceDescriptionStruct = new ShaderStringBuilder(0);
  68. var surfaceDescriptionFunction = new ShaderStringBuilder(0);
  69. var vertexInputs = new ShaderStringBuilder(0);
  70. graph.CollectShaderKeywords(shaderKeywords, mode);
  71. if(graph.GetKeywordPermutationCount() > ShaderGraphPreferences.variantLimit)
  72. {
  73. graph.AddValidationError(node.tempId, ShaderKeyword.kVariantLimitWarning, Rendering.ShaderCompilerMessageSeverity.Error);
  74. results.configuredTextures = shaderProperties.GetConfiguredTexutres();
  75. results.shader = string.Empty;
  76. return results;
  77. }
  78. // -------------------------------------
  79. // Get Slot and Node lists
  80. var activeNodeList = ListPool<AbstractMaterialNode>.Get();
  81. NodeUtils.DepthFirstCollectNodesFromNode(activeNodeList, node);
  82. var slots = new List<MaterialSlot>();
  83. if (node is IMasterNode || node is SubGraphOutputNode)
  84. slots.AddRange(node.GetInputSlots<MaterialSlot>());
  85. else
  86. {
  87. var outputSlots = node.GetOutputSlots<MaterialSlot>().ToList();
  88. if (outputSlots.Count > 0)
  89. slots.Add(outputSlots[0]);
  90. }
  91. // -------------------------------------
  92. // Get Requirements
  93. var requirements = ShaderGraphRequirements.FromNodes(activeNodeList, ShaderStageCapability.Fragment);
  94. // ----------------------------------------------------- //
  95. // KEYWORDS //
  96. // ----------------------------------------------------- //
  97. // -------------------------------------
  98. // Get keyword permutations
  99. graph.CollectShaderKeywords(shaderKeywords, mode);
  100. // Track permutation indicies for all nodes and requirements
  101. List<int>[] keywordPermutationsPerNode = new List<int>[activeNodeList.Count];
  102. // -------------------------------------
  103. // Evaluate all permutations
  104. for(int i = 0; i < shaderKeywords.permutations.Count; i++)
  105. {
  106. // Get active nodes for this permutation
  107. var localNodes = ListPool<AbstractMaterialNode>.Get();
  108. NodeUtils.DepthFirstCollectNodesFromNode(localNodes, node, keywordPermutation: shaderKeywords.permutations[i]);
  109. // Track each pixel node in this permutation
  110. foreach(AbstractMaterialNode pixelNode in localNodes)
  111. {
  112. int nodeIndex = activeNodeList.IndexOf(pixelNode);
  113. if(keywordPermutationsPerNode[nodeIndex] == null)
  114. keywordPermutationsPerNode[nodeIndex] = new List<int>();
  115. keywordPermutationsPerNode[nodeIndex].Add(i);
  116. }
  117. // Get active requirements for this permutation
  118. var localSurfaceRequirements = ShaderGraphRequirements.FromNodes(localNodes, ShaderStageCapability.Fragment, false);
  119. var localPixelRequirements = ShaderGraphRequirements.FromNodes(localNodes, ShaderStageCapability.Fragment);
  120. }
  121. // ----------------------------------------------------- //
  122. // START VERTEX DESCRIPTION //
  123. // ----------------------------------------------------- //
  124. // -------------------------------------
  125. // Generate Vertex Description function
  126. vertexDescriptionFunction.AppendLine("GraphVertexInput PopulateVertexData(GraphVertexInput v)");
  127. using (vertexDescriptionFunction.BlockScope())
  128. {
  129. vertexDescriptionFunction.AppendLine("return v;");
  130. }
  131. // ----------------------------------------------------- //
  132. // START SURFACE DESCRIPTION //
  133. // ----------------------------------------------------- //
  134. // -------------------------------------
  135. // Generate Input structure for Surface Description function
  136. // Surface Description Input requirements are needed to exclude intermediate translation spaces
  137. GenerateSurfaceInputStruct(surfaceDescriptionInputStruct, requirements, "SurfaceDescriptionInputs");
  138. results.previewMode = PreviewMode.Preview2D;
  139. foreach (var pNode in activeNodeList)
  140. {
  141. if (pNode.previewMode == PreviewMode.Preview3D)
  142. {
  143. results.previewMode = PreviewMode.Preview3D;
  144. break;
  145. }
  146. }
  147. // -------------------------------------
  148. // Generate Output structure for Surface Description function
  149. GenerateSurfaceDescriptionStruct(surfaceDescriptionStruct, slots, useIdsInNames: !(node is IMasterNode));
  150. // -------------------------------------
  151. // Generate Surface Description function
  152. GenerateSurfaceDescriptionFunction(
  153. activeNodeList,
  154. keywordPermutationsPerNode,
  155. node,
  156. graph,
  157. surfaceDescriptionFunction,
  158. functionRegistry,
  159. shaderProperties,
  160. shaderKeywords,
  161. mode,
  162. outputIdProperty: results.outputIdProperty);
  163. // ----------------------------------------------------- //
  164. // GENERATE VERTEX > PIXEL PIPELINE //
  165. // ----------------------------------------------------- //
  166. // -------------------------------------
  167. // Keyword declarations
  168. shaderKeywords.GetKeywordsDeclaration(shaderKeywordDeclarations, mode);
  169. // -------------------------------------
  170. // Property uniforms
  171. shaderProperties.GetPropertiesDeclaration(shaderPropertyUniforms, mode, graph.concretePrecision);
  172. // -------------------------------------
  173. // Generate Input structure for Vertex shader
  174. GenerateApplicationVertexInputs(requirements, vertexInputs);
  175. // ----------------------------------------------------- //
  176. // FINALIZE //
  177. // ----------------------------------------------------- //
  178. // -------------------------------------
  179. // Build final shader
  180. finalShader.AppendLine(@"Shader ""{0}""", name);
  181. using (finalShader.BlockScope())
  182. {
  183. SubShaderGenerator.GeneratePropertiesBlock(finalShader, shaderProperties, shaderKeywords, mode);
  184. finalShader.AppendNewLine();
  185. finalShader.AppendLine(@"HLSLINCLUDE");
  186. finalShader.AppendLine(@"#include ""Packages/com.unity.render-pipelines.core/ShaderLibrary/Common.hlsl""");
  187. finalShader.AppendLine(@"#include ""Packages/com.unity.render-pipelines.core/ShaderLibrary/Packing.hlsl""");
  188. finalShader.AppendLine(@"#include ""Packages/com.unity.render-pipelines.core/ShaderLibrary/NormalSurfaceGradient.hlsl""");
  189. finalShader.AppendLine(@"#include ""Packages/com.unity.render-pipelines.core/ShaderLibrary/Color.hlsl""");
  190. finalShader.AppendLine(@"#include ""Packages/com.unity.render-pipelines.core/ShaderLibrary/UnityInstancing.hlsl""");
  191. finalShader.AppendLine(@"#include ""Packages/com.unity.render-pipelines.core/ShaderLibrary/EntityLighting.hlsl""");
  192. finalShader.AppendLine(@"#include ""Packages/com.unity.shadergraph/ShaderGraphLibrary/ShaderVariables.hlsl""");
  193. finalShader.AppendLine(@"#include ""Packages/com.unity.shadergraph/ShaderGraphLibrary/ShaderVariablesFunctions.hlsl""");
  194. finalShader.AppendLine(@"#include ""Packages/com.unity.shadergraph/ShaderGraphLibrary/Functions.hlsl""");
  195. finalShader.AppendLines(shaderKeywordDeclarations.ToString());
  196. finalShader.AppendLine(@"#define SHADERGRAPH_PREVIEW 1");
  197. finalShader.AppendNewLine();
  198. finalShader.AppendLines(shaderKeywordPermutations.ToString());
  199. finalShader.AppendLines(shaderPropertyUniforms.ToString());
  200. finalShader.AppendNewLine();
  201. finalShader.AppendLines(surfaceDescriptionInputStruct.ToString());
  202. finalShader.AppendNewLine();
  203. finalShader.Concat(functionBuilder);
  204. finalShader.AppendNewLine();
  205. finalShader.AppendLines(surfaceDescriptionStruct.ToString());
  206. finalShader.AppendNewLine();
  207. finalShader.AppendLines(surfaceDescriptionFunction.ToString());
  208. finalShader.AppendNewLine();
  209. finalShader.AppendLines(vertexInputs.ToString());
  210. finalShader.AppendNewLine();
  211. finalShader.AppendLines(vertexDescriptionFunction.ToString());
  212. finalShader.AppendNewLine();
  213. finalShader.AppendLine(@"ENDHLSL");
  214. finalShader.AppendLines(ShaderGenerator.GetPreviewSubShader(node, requirements));
  215. ListPool<AbstractMaterialNode>.Release(activeNodeList);
  216. }
  217. // -------------------------------------
  218. // Finalize
  219. results.configuredTextures = shaderProperties.GetConfiguredTexutres();
  220. ShaderSourceMap sourceMap;
  221. results.shader = finalShader.ToString(out sourceMap);
  222. results.sourceMap = sourceMap;
  223. return results;
  224. }
  225. public static void GenerateSurfaceInputStruct(ShaderStringBuilder sb, ShaderGraphRequirements requirements, string structName)
  226. {
  227. sb.AppendLine($"struct {structName}");
  228. using (sb.BlockSemicolonScope())
  229. {
  230. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresNormal, InterpolatorType.Normal, sb);
  231. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresTangent, InterpolatorType.Tangent, sb);
  232. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresBitangent, InterpolatorType.BiTangent, sb);
  233. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresViewDir, InterpolatorType.ViewDirection, sb);
  234. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresPosition, InterpolatorType.Position, sb);
  235. if (requirements.requiresVertexColor)
  236. sb.AppendLine("float4 {0};", ShaderGeneratorNames.VertexColor);
  237. if (requirements.requiresScreenPosition)
  238. sb.AppendLine("float4 {0};", ShaderGeneratorNames.ScreenPosition);
  239. if (requirements.requiresFaceSign)
  240. sb.AppendLine("float {0};", ShaderGeneratorNames.FaceSign);
  241. foreach (var channel in requirements.requiresMeshUVs.Distinct())
  242. sb.AppendLine("half4 {0};", channel.GetUVName());
  243. if (requirements.requiresTime)
  244. {
  245. sb.AppendLine("float3 {0};", ShaderGeneratorNames.TimeParameters);
  246. }
  247. }
  248. }
  249. public static void GenerateSurfaceInputTransferCode(ShaderStringBuilder sb, ShaderGraphRequirements requirements, string structName, string variableName)
  250. {
  251. sb.AppendLine($"{structName} {variableName};");
  252. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresNormal, InterpolatorType.Normal, sb, $"{variableName}.{{0}} = IN.{{0}};");
  253. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresTangent, InterpolatorType.Tangent, sb, $"{variableName}.{{0}} = IN.{{0}};");
  254. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresBitangent, InterpolatorType.BiTangent, sb, $"{variableName}.{{0}} = IN.{{0}};");
  255. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresViewDir, InterpolatorType.ViewDirection, sb, $"{variableName}.{{0}} = IN.{{0}};");
  256. ShaderGenerator.GenerateSpaceTranslationSurfaceInputs(requirements.requiresPosition, InterpolatorType.Position, sb, $"{variableName}.{{0}} = IN.{{0}};");
  257. if (requirements.requiresVertexColor)
  258. sb.AppendLine($"{variableName}.{ShaderGeneratorNames.VertexColor} = IN.{ShaderGeneratorNames.VertexColor};");
  259. if (requirements.requiresScreenPosition)
  260. sb.AppendLine($"{variableName}.{ShaderGeneratorNames.ScreenPosition} = IN.{ShaderGeneratorNames.ScreenPosition};");
  261. if (requirements.requiresFaceSign)
  262. sb.AppendLine($"{variableName}.{ShaderGeneratorNames.FaceSign} = IN.{ShaderGeneratorNames.FaceSign};");
  263. foreach (var channel in requirements.requiresMeshUVs.Distinct())
  264. sb.AppendLine($"{variableName}.{channel.GetUVName()} = IN.{channel.GetUVName()};");
  265. if (requirements.requiresTime)
  266. {
  267. sb.AppendLine($"{variableName}.{ShaderGeneratorNames.TimeParameters} = IN.{ShaderGeneratorNames.TimeParameters};");
  268. }
  269. }
  270. public static void GenerateSurfaceDescriptionStruct(ShaderStringBuilder surfaceDescriptionStruct, List<MaterialSlot> slots, string structName = "SurfaceDescription", IActiveFieldsSet activeFields = null, bool useIdsInNames = false)
  271. {
  272. surfaceDescriptionStruct.AppendLine("struct {0}", structName);
  273. using (surfaceDescriptionStruct.BlockSemicolonScope())
  274. {
  275. foreach (var slot in slots)
  276. {
  277. string hlslName = NodeUtils.GetHLSLSafeName(slot.shaderOutputName);
  278. if (useIdsInNames)
  279. {
  280. hlslName = $"{hlslName}_{slot.id}";
  281. }
  282. surfaceDescriptionStruct.AppendLine("{0} {1};", slot.concreteValueType.ToShaderString(slot.owner.concretePrecision), hlslName);
  283. if (activeFields != null)
  284. {
  285. activeFields.AddAll(structName + "." + hlslName);
  286. }
  287. }
  288. }
  289. }
  290. public static void GenerateSurfaceDescriptionFunction(
  291. List<AbstractMaterialNode> nodes,
  292. List<int>[] keywordPermutationsPerNode,
  293. AbstractMaterialNode rootNode,
  294. GraphData graph,
  295. ShaderStringBuilder surfaceDescriptionFunction,
  296. FunctionRegistry functionRegistry,
  297. PropertyCollector shaderProperties,
  298. KeywordCollector shaderKeywords,
  299. GenerationMode mode,
  300. string functionName = "PopulateSurfaceData",
  301. string surfaceDescriptionName = "SurfaceDescription",
  302. Vector1ShaderProperty outputIdProperty = null,
  303. IEnumerable<MaterialSlot> slots = null,
  304. string graphInputStructName = "SurfaceDescriptionInputs")
  305. {
  306. if (graph == null)
  307. return;
  308. graph.CollectShaderProperties(shaderProperties, mode);
  309. surfaceDescriptionFunction.AppendLine(String.Format("{0} {1}(SurfaceDescriptionInputs IN)", surfaceDescriptionName, functionName), false);
  310. using (surfaceDescriptionFunction.BlockScope())
  311. {
  312. surfaceDescriptionFunction.AppendLine("{0} surface = ({0})0;", surfaceDescriptionName);
  313. for(int i = 0; i < nodes.Count; i++)
  314. {
  315. GenerateDescriptionForNode(nodes[i], keywordPermutationsPerNode[i], functionRegistry, surfaceDescriptionFunction,
  316. shaderProperties, shaderKeywords,
  317. graph, mode);
  318. }
  319. functionRegistry.builder.currentNode = null;
  320. surfaceDescriptionFunction.currentNode = null;
  321. GenerateSurfaceDescriptionRemap(graph, rootNode, slots,
  322. surfaceDescriptionFunction, mode);
  323. surfaceDescriptionFunction.AppendLine("return surface;");
  324. }
  325. }
  326. static void GenerateDescriptionForNode(
  327. AbstractMaterialNode activeNode,
  328. List<int> keywordPermutations,
  329. FunctionRegistry functionRegistry,
  330. ShaderStringBuilder descriptionFunction,
  331. PropertyCollector shaderProperties,
  332. KeywordCollector shaderKeywords,
  333. GraphData graph,
  334. GenerationMode mode)
  335. {
  336. if (activeNode is IGeneratesFunction functionNode)
  337. {
  338. functionRegistry.builder.currentNode = activeNode;
  339. functionNode.GenerateNodeFunction(functionRegistry, mode);
  340. functionRegistry.builder.ReplaceInCurrentMapping(PrecisionUtil.Token, activeNode.concretePrecision.ToShaderString());
  341. }
  342. if (activeNode is IGeneratesBodyCode bodyNode)
  343. {
  344. if(keywordPermutations != null)
  345. descriptionFunction.AppendLine(KeywordUtil.GetKeywordPermutationSetConditional(keywordPermutations));
  346. descriptionFunction.currentNode = activeNode;
  347. bodyNode.GenerateNodeCode(descriptionFunction, mode);
  348. descriptionFunction.ReplaceInCurrentMapping(PrecisionUtil.Token, activeNode.concretePrecision.ToShaderString());
  349. if(keywordPermutations != null)
  350. descriptionFunction.AppendLine("#endif");
  351. }
  352. activeNode.CollectShaderProperties(shaderProperties, mode);
  353. if (activeNode is SubGraphNode subGraphNode)
  354. {
  355. subGraphNode.CollectShaderKeywords(shaderKeywords, mode);
  356. }
  357. }
  358. static void GenerateSurfaceDescriptionRemap(
  359. GraphData graph,
  360. AbstractMaterialNode rootNode,
  361. IEnumerable<MaterialSlot> slots,
  362. ShaderStringBuilder surfaceDescriptionFunction,
  363. GenerationMode mode)
  364. {
  365. if (rootNode is IMasterNode || rootNode is SubGraphOutputNode)
  366. {
  367. var usedSlots = slots ?? rootNode.GetInputSlots<MaterialSlot>();
  368. foreach (var input in usedSlots)
  369. {
  370. if (input != null)
  371. {
  372. var foundEdges = graph.GetEdges(input.slotReference).ToArray();
  373. var hlslName = NodeUtils.GetHLSLSafeName(input.shaderOutputName);
  374. if (rootNode is SubGraphOutputNode)
  375. {
  376. hlslName = $"{hlslName}_{input.id}";
  377. }
  378. if (foundEdges.Any())
  379. {
  380. surfaceDescriptionFunction.AppendLine("surface.{0} = {1};",
  381. hlslName,
  382. rootNode.GetSlotValue(input.id, mode, rootNode.concretePrecision));
  383. }
  384. else
  385. {
  386. surfaceDescriptionFunction.AppendLine("surface.{0} = {1};",
  387. hlslName, input.GetDefaultValue(mode, rootNode.concretePrecision));
  388. }
  389. }
  390. }
  391. }
  392. else if (rootNode.hasPreview)
  393. {
  394. var slot = rootNode.GetOutputSlots<MaterialSlot>().FirstOrDefault();
  395. if (slot != null)
  396. {
  397. var hlslSafeName = $"{NodeUtils.GetHLSLSafeName(slot.shaderOutputName)}_{slot.id}";
  398. surfaceDescriptionFunction.AppendLine("surface.{0} = {1};",
  399. hlslSafeName, rootNode.GetSlotValue(slot.id, mode, rootNode.concretePrecision));
  400. }
  401. }
  402. }
  403. const string k_VertexDescriptionStructName = "VertexDescription";
  404. public static void GenerateVertexDescriptionStruct(ShaderStringBuilder builder, List<MaterialSlot> slots, string structName = k_VertexDescriptionStructName, IActiveFieldsSet activeFields = null)
  405. {
  406. builder.AppendLine("struct {0}", structName);
  407. using (builder.BlockSemicolonScope())
  408. {
  409. foreach (var slot in slots)
  410. {
  411. string hlslName = NodeUtils.GetHLSLSafeName(slot.shaderOutputName);
  412. builder.AppendLine("{0} {1};", slot.concreteValueType.ToShaderString(slot.owner.concretePrecision), hlslName);
  413. if (activeFields != null)
  414. {
  415. activeFields.AddAll(structName + "." + hlslName);
  416. }
  417. }
  418. }
  419. }
  420. public static void GenerateVertexDescriptionFunction(
  421. GraphData graph,
  422. ShaderStringBuilder builder,
  423. FunctionRegistry functionRegistry,
  424. PropertyCollector shaderProperties,
  425. KeywordCollector shaderKeywords,
  426. GenerationMode mode,
  427. AbstractMaterialNode rootNode,
  428. List<AbstractMaterialNode> nodes,
  429. List<int>[] keywordPermutationsPerNode,
  430. List<MaterialSlot> slots,
  431. string graphInputStructName = "VertexDescriptionInputs",
  432. string functionName = "PopulateVertexData",
  433. string graphOutputStructName = k_VertexDescriptionStructName)
  434. {
  435. if (graph == null)
  436. return;
  437. graph.CollectShaderProperties(shaderProperties, mode);
  438. builder.AppendLine("{0} {1}({2} IN)", graphOutputStructName, functionName, graphInputStructName);
  439. using (builder.BlockScope())
  440. {
  441. builder.AppendLine("{0} description = ({0})0;", graphOutputStructName);
  442. for(int i = 0; i < nodes.Count; i++)
  443. {
  444. GenerateDescriptionForNode(nodes[i], keywordPermutationsPerNode[i], functionRegistry, builder,
  445. shaderProperties, shaderKeywords,
  446. graph, mode);
  447. }
  448. functionRegistry.builder.currentNode = null;
  449. builder.currentNode = null;
  450. if(slots.Count != 0)
  451. {
  452. foreach (var slot in slots)
  453. {
  454. var isSlotConnected = slot.owner.owner.GetEdges(slot.slotReference).Any();
  455. var slotName = NodeUtils.GetHLSLSafeName(slot.shaderOutputName);
  456. var slotValue = isSlotConnected ?
  457. ((AbstractMaterialNode)slot.owner).GetSlotValue(slot.id, mode, slot.owner.concretePrecision) : slot.GetDefaultValue(mode, slot.owner.concretePrecision);
  458. builder.AppendLine("description.{0} = {1};", slotName, slotValue);
  459. }
  460. }
  461. builder.AppendLine("return description;");
  462. }
  463. }
  464. public static GenerationResults GetPreviewShader(this GraphData graph, AbstractMaterialNode node)
  465. {
  466. return graph.GetShader(node, GenerationMode.Preview, String.Format("hidden/preview/{0}", node.GetVariableNameForNode()));
  467. }
  468. }
  469. }