KeywordUtil.cs 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using UnityEditor.ShaderGraph.Drawing;
  6. using UnityEngine;
  7. using UnityEditor.ShaderGraph.Internal;
  8. namespace UnityEditor.ShaderGraph
  9. {
  10. static class BuiltinKeyword
  11. {
  12. [BuiltinKeyword]
  13. public static KeywordDescriptor QualityKeyword()
  14. {
  15. return new KeywordDescriptor()
  16. {
  17. displayName = "Material Quality",
  18. referenceName = "MATERIAL_QUALITY",
  19. type = KeywordType.Enum,
  20. definition = KeywordDefinition.MultiCompile,
  21. scope = KeywordScope.Global,
  22. value = 0,
  23. entries = new KeywordEntry[]
  24. {
  25. new KeywordEntry("High", "HIGH"),
  26. new KeywordEntry("Medium", "MEDIUM"),
  27. new KeywordEntry("Low", "LOW"),
  28. },
  29. };
  30. }
  31. }
  32. static class KeywordUtil
  33. {
  34. public static IEnumerable<KeywordDescriptor> GetBuiltinKeywordDescriptors() =>
  35. TypeCache.GetMethodsWithAttribute<BuiltinKeywordAttribute>()
  36. .Where(method => method.IsStatic && method.ReturnType == typeof(KeywordDescriptor))
  37. .Select(method =>
  38. (KeywordDescriptor) method.Invoke(null, new object[0] { }));
  39. public static ConcreteSlotValueType ToConcreteSlotValueType(this KeywordType keywordType)
  40. {
  41. switch(keywordType)
  42. {
  43. case KeywordType.Boolean:
  44. return ConcreteSlotValueType.Boolean;
  45. case KeywordType.Enum:
  46. return ConcreteSlotValueType.Vector1;
  47. default:
  48. throw new ArgumentOutOfRangeException();
  49. }
  50. }
  51. public static string ToDeclarationString(this KeywordDefinition keywordDefinition)
  52. {
  53. switch(keywordDefinition)
  54. {
  55. case KeywordDefinition.MultiCompile:
  56. return "multi_compile";
  57. case KeywordDefinition.ShaderFeature:
  58. return "shader_feature";
  59. default:
  60. return string.Empty;
  61. }
  62. }
  63. public static string ToDeclarationString(this KeywordDescriptor keyword)
  64. {
  65. // Get definition type using scope
  66. string scopeString = keyword.scope == KeywordScope.Local ? "_local" : string.Empty;
  67. string definitionString = $"{keyword.definition.ToDeclarationString()}{scopeString}";
  68. switch(keyword.type)
  69. {
  70. case KeywordType.Boolean:
  71. return $"#pragma {definitionString} _ {keyword.referenceName}";
  72. case KeywordType.Enum:
  73. var enumEntryDefinitions = keyword.entries.Select(x => $"{keyword.referenceName}_{x.referenceName}");
  74. string enumEntriesString = string.Join(" ", enumEntryDefinitions);
  75. return $"#pragma {definitionString} {enumEntriesString}";
  76. default:
  77. throw new ArgumentOutOfRangeException();
  78. }
  79. }
  80. public static int GetKeywordPermutationCount(this GraphData graph)
  81. {
  82. // Gather all unique keywords from the Graph including Sub Graphs
  83. IEnumerable<ShaderKeyword> allKeywords = graph.keywords;
  84. var subGraphNodes = graph.GetNodes<SubGraphNode>();
  85. foreach(SubGraphNode subGraphNode in subGraphNodes)
  86. {
  87. if (subGraphNode.asset == null)
  88. {
  89. continue;
  90. }
  91. allKeywords = allKeywords.Union(subGraphNode.asset.keywords);
  92. }
  93. allKeywords = allKeywords.Distinct();
  94. // Get permutation count for all Keywords
  95. int permutationCount = 1;
  96. foreach(ShaderKeyword keyword in allKeywords)
  97. {
  98. if(keyword.keywordType == KeywordType.Boolean)
  99. permutationCount *= 2;
  100. else
  101. permutationCount *= keyword.entries.Count;
  102. }
  103. return permutationCount;
  104. }
  105. public static string GetKeywordPermutationSetConditional(List<int> permutationSet)
  106. {
  107. StringBuilder sb = new StringBuilder();
  108. sb.Append("#if ");
  109. for(int i = 0; i < permutationSet.Count; i++)
  110. {
  111. // Subsequent permutation predicates require ||
  112. if(i != 0)
  113. sb.Append(" || ");
  114. // Append permutation
  115. sb.Append($"defined(KEYWORD_PERMUTATION_{permutationSet[i]})");
  116. }
  117. return sb.ToString();
  118. }
  119. public static void GetKeywordPermutationDeclarations(ShaderStringBuilder sb, List<List<KeyValuePair<ShaderKeyword, int>>> permutations)
  120. {
  121. if (permutations.Count == 0)
  122. return;
  123. for(int p = 0; p < permutations.Count; p++)
  124. {
  125. // ShaderStringBuilder.Append doesnt apply indentation
  126. sb.AppendIndentation();
  127. // Append correct if
  128. bool isLast = false;
  129. if(p == 0)
  130. {
  131. sb.Append("#if ");
  132. }
  133. else if(p == permutations.Count - 1)
  134. {
  135. sb.Append("#else");
  136. isLast = true;
  137. }
  138. else
  139. {
  140. sb.Append("#elif ");
  141. }
  142. // Last permutation is always #else
  143. if(!isLast)
  144. {
  145. // Track whether && is required
  146. bool appendAnd = false;
  147. // Iterate all keywords that are part of the permutation
  148. for(int i = 0; i < permutations[p].Count; i++)
  149. {
  150. // When previous keyword was inserted subsequent requires &&
  151. string and = appendAnd ? " && " : string.Empty;
  152. switch(permutations[p][i].Key.keywordType)
  153. {
  154. case KeywordType.Enum:
  155. {
  156. sb.Append($"{and}defined({permutations[p][i].Key.referenceName}_{permutations[p][i].Key.entries[permutations[p][i].Value].referenceName})");
  157. appendAnd = true;
  158. break;
  159. }
  160. case KeywordType.Boolean:
  161. {
  162. // HLSL does not support a !value predicate
  163. if(permutations[p][i].Value != 0)
  164. {
  165. continue;
  166. }
  167. sb.Append($"{and}defined({permutations[p][i].Key.referenceName})");
  168. appendAnd = true;
  169. break;
  170. }
  171. default:
  172. throw new ArgumentOutOfRangeException();
  173. }
  174. }
  175. }
  176. sb.AppendNewLine();
  177. // Define the matching permutation keyword
  178. sb.IncreaseIndent();
  179. sb.AppendLine($"#define KEYWORD_PERMUTATION_{p}");
  180. sb.DecreaseIndent();
  181. }
  182. // End statement
  183. sb.AppendLine("#endif");
  184. }
  185. }
  186. }