ShaderSpliceUtil.cs 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Reflection;
  6. using Data.Util;
  7. namespace UnityEditor.ShaderGraph
  8. {
  9. static class ShaderSpliceUtil
  10. {
  11. enum BaseFieldType
  12. {
  13. Invalid,
  14. Float,
  15. Uint,
  16. };
  17. private static BaseFieldType GetBaseFieldType(string typeName)
  18. {
  19. if (typeName.StartsWith("Vector") || typeName.Equals("Single"))
  20. {
  21. return BaseFieldType.Float;
  22. }
  23. if (typeName.StartsWith("UInt32")) // We don't have proper support for uint (Uint, Uint2, Uint3, Uint4). Need these types, for now just supporting instancing via a single uint.
  24. {
  25. return BaseFieldType.Uint;
  26. }
  27. return BaseFieldType.Invalid;
  28. }
  29. private static int GetComponentCount(string typeName)
  30. {
  31. switch (GetBaseFieldType(typeName))
  32. {
  33. case BaseFieldType.Float:
  34. return GetFloatVectorCount(typeName);
  35. case BaseFieldType.Uint:
  36. return GetUintCount(typeName);
  37. default:
  38. return 0;
  39. }
  40. }
  41. private static int GetFloatVectorCount(string typeName)
  42. {
  43. if (typeName.Equals("Vector4"))
  44. {
  45. return 4;
  46. }
  47. else if (typeName.Equals("Vector3"))
  48. {
  49. return 3;
  50. }
  51. else if (typeName.Equals("Vector2"))
  52. {
  53. return 2;
  54. }
  55. else if (typeName.Equals("Single"))
  56. {
  57. return 1;
  58. }
  59. else
  60. {
  61. return 0;
  62. }
  63. }
  64. // Need uint types
  65. private static int GetUintCount(string typeName)
  66. {
  67. if (typeName.Equals("UInt32"))
  68. {
  69. return 1;
  70. }
  71. if(typeName.Equals("UInt32_4"))
  72. {
  73. return 4;
  74. }
  75. else
  76. {
  77. return 0;
  78. }
  79. }
  80. private static string[] vectorTypeNames =
  81. {
  82. "unknown",
  83. "float",
  84. "float2",
  85. "float3",
  86. "float4"
  87. };
  88. private static string[] uintTypeNames =
  89. {
  90. "unknown",
  91. "uint",
  92. "uint2",
  93. "uint3",
  94. "uint4",
  95. };
  96. private static char[] channelNames =
  97. { 'x', 'y', 'z', 'w' };
  98. private static string GetChannelSwizzle(int firstChannel, int channelCount)
  99. {
  100. System.Text.StringBuilder result = new System.Text.StringBuilder();
  101. int lastChannel = System.Math.Min(firstChannel + channelCount - 1, 4);
  102. for (int index = firstChannel; index <= lastChannel; index++)
  103. {
  104. result.Append(channelNames[index]);
  105. }
  106. return result.ToString();
  107. }
  108. private static bool ShouldSpliceField(System.Type parentType, FieldInfo field, IActiveFields activeFields, out bool isOptional)
  109. {
  110. bool fieldActive = true;
  111. isOptional = field.IsDefined(typeof(Optional), false);
  112. if (isOptional)
  113. {
  114. string fullName = parentType.Name + "." + field.Name;
  115. if (!activeFields.Contains(fullName))
  116. {
  117. // not active, skip the optional field
  118. fieldActive = false;
  119. }
  120. }
  121. return fieldActive;
  122. }
  123. private static string GetFieldSemantic(FieldInfo field)
  124. {
  125. string semanticString = null;
  126. object[] semantics = field.GetCustomAttributes(typeof(Semantic), false);
  127. if (semantics.Length > 0)
  128. {
  129. Semantic firstSemantic = (Semantic)semantics[0];
  130. semanticString = " : " + firstSemantic.semantic;
  131. }
  132. return semanticString;
  133. }
  134. private static string GetFieldType(FieldInfo field, out int componentCount)
  135. {
  136. string fieldType;
  137. object[] overrideType = field.GetCustomAttributes(typeof(OverrideType), false);
  138. if (overrideType.Length > 0)
  139. {
  140. OverrideType first = (OverrideType)overrideType[0];
  141. fieldType = first.typeName;
  142. componentCount = 0;
  143. }
  144. else
  145. {
  146. // TODO: handle non-float types
  147. componentCount = GetComponentCount(field.FieldType.Name);
  148. switch (GetBaseFieldType(field.FieldType.Name))
  149. {
  150. case BaseFieldType.Float:
  151. fieldType = vectorTypeNames[componentCount];
  152. break;
  153. case BaseFieldType.Uint:
  154. fieldType = uintTypeNames[componentCount];
  155. break;
  156. default:
  157. fieldType = "unknown";
  158. break;
  159. }
  160. }
  161. return fieldType;
  162. }
  163. private static bool IsFloatVectorType(string type)
  164. {
  165. return GetFloatVectorCount(type) != 0;
  166. }
  167. private static string GetFieldConditional(FieldInfo field)
  168. {
  169. string conditional = null;
  170. object[] overrideType = field.GetCustomAttributes(typeof(PreprocessorIf), false);
  171. if (overrideType.Length > 0)
  172. {
  173. PreprocessorIf first = (PreprocessorIf)overrideType[0];
  174. conditional = first.conditional;
  175. }
  176. return conditional;
  177. }
  178. public static void BuildType(System.Type t, ActiveFields activeFields, ShaderGenerator result, bool isDebug)
  179. {
  180. result.AddShaderChunk("struct " + t.Name);
  181. result.AddShaderChunk("{");
  182. result.Indent();
  183. foreach (FieldInfo field in t.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public))
  184. {
  185. if (field.MemberType == MemberTypes.Field)
  186. {
  187. bool isOptional = false;
  188. var fieldIsActive = false;
  189. var keywordIfdefs = string.Empty;
  190. if (activeFields.permutationCount > 0)
  191. {
  192. // Evaluate all activeFields instance
  193. var instances = activeFields
  194. .allPermutations.instances
  195. .Where(i => ShouldSpliceField(t, field, i, out isOptional))
  196. .ToList();
  197. fieldIsActive = instances.Count > 0;
  198. if (fieldIsActive)
  199. keywordIfdefs = KeywordUtil.GetKeywordPermutationSetConditional(instances
  200. .Select(i => i.permutationIndex).ToList());
  201. }
  202. else
  203. {
  204. fieldIsActive = ShouldSpliceField(t, field, activeFields.baseInstance, out isOptional);
  205. }
  206. if (fieldIsActive)
  207. {
  208. // The field is used, so generate it
  209. var semanticString = GetFieldSemantic(field);
  210. int componentCount;
  211. var fieldType = GetFieldType(field, out componentCount);
  212. var conditional = GetFieldConditional(field);
  213. if (conditional != null)
  214. result.AddShaderChunk("#if " + conditional);
  215. if (!string.IsNullOrEmpty(keywordIfdefs))
  216. result.AddShaderChunk(keywordIfdefs);
  217. var fieldDecl = fieldType + " " + field.Name + semanticString + ";" + (isOptional & isDebug ? " // optional" : string.Empty);
  218. result.AddShaderChunk(fieldDecl);
  219. if (!string.IsNullOrEmpty(keywordIfdefs))
  220. result.AddShaderChunk("#endif" + (isDebug ? " // Shader Graph Keywords" : string.Empty));
  221. if (conditional != null)
  222. result.AddShaderChunk("#endif" + (isDebug ? $" // {conditional}" : string.Empty));
  223. }
  224. }
  225. }
  226. result.Deindent();
  227. result.AddShaderChunk("};");
  228. object[] packAttributes = t.GetCustomAttributes(typeof(InterpolatorPack), false);
  229. if (packAttributes.Length > 0)
  230. {
  231. result.AddNewLine();
  232. if (activeFields.permutationCount > 0)
  233. {
  234. var generatedPackedTypes = new Dictionary<string, (ShaderGenerator, List<int>)>();
  235. foreach (var instance in activeFields.allPermutations.instances)
  236. {
  237. var instanceGenerator = new ShaderGenerator();
  238. BuildPackedType(t, instance, instanceGenerator, isDebug);
  239. var key = instanceGenerator.GetShaderString(0);
  240. if (generatedPackedTypes.TryGetValue(key, out var value))
  241. value.Item2.Add(instance.permutationIndex);
  242. else
  243. generatedPackedTypes.Add(key, (instanceGenerator, new List<int> { instance.permutationIndex }));
  244. }
  245. var isFirst = true;
  246. foreach (var generated in generatedPackedTypes)
  247. {
  248. if (isFirst)
  249. {
  250. isFirst = false;
  251. result.AddShaderChunk(KeywordUtil.GetKeywordPermutationSetConditional(generated.Value.Item2));
  252. }
  253. else
  254. result.AddShaderChunk(KeywordUtil.GetKeywordPermutationSetConditional(generated.Value.Item2).Replace("#if", "#elif"));
  255. result.AddGenerator(generated.Value.Item1);
  256. }
  257. if (generatedPackedTypes.Count > 0)
  258. result.AddShaderChunk("#endif");
  259. }
  260. else
  261. {
  262. BuildPackedType(t, activeFields.baseInstance, result, isDebug);
  263. }
  264. }
  265. }
  266. public static void BuildPackedType(System.Type unpacked, IActiveFields activeFields, ShaderGenerator result, bool isDebug)
  267. {
  268. // for each interpolator, the number of components used (up to 4 for a float4 interpolator)
  269. List<int> packedCounts = new List<int>();
  270. ShaderGenerator packingStruct = new ShaderGenerator();
  271. ShaderGenerator packer = new ShaderGenerator();
  272. ShaderGenerator unpacker = new ShaderGenerator();
  273. ShaderGenerator systemGenerated = new ShaderGenerator();
  274. string unpackedStruct = unpacked.Name.ToString();
  275. string packedStruct = "Packed" + unpacked.Name;
  276. string packerFunction = "Pack" + unpacked.Name;
  277. string unpackerFunction = "Unpack" + unpacked.Name;
  278. // declare struct header:
  279. // struct packedStruct {
  280. packingStruct.AddShaderChunk("// Generated Type: Packed" + unpacked.Name);
  281. packingStruct.AddShaderChunk("struct " + packedStruct);
  282. packingStruct.AddShaderChunk("{");
  283. packingStruct.Indent();
  284. // declare function headers:
  285. // packedStruct packerFunction(unpackedStruct input)
  286. // {
  287. // packedStruct output;
  288. packer.AddShaderChunk("// Packed Type: " + unpacked.Name);
  289. packer.AddShaderChunk(packedStruct + " " + packerFunction + "(" + unpackedStruct + " input)");
  290. packer.AddShaderChunk("{");
  291. packer.Indent();
  292. packer.AddShaderChunk(packedStruct + " output = (" + packedStruct + ")0;");
  293. // unpackedStruct unpackerFunction(packedStruct input)
  294. // {
  295. // unpackedStruct output;
  296. unpacker.AddShaderChunk("// Unpacked Type: " + unpacked.Name);
  297. unpacker.AddShaderChunk(unpackedStruct + " " + unpackerFunction + "(" + packedStruct + " input)");
  298. unpacker.AddShaderChunk("{");
  299. unpacker.Indent();
  300. unpacker.AddShaderChunk(unpackedStruct + " output = (" + unpackedStruct + ")0;");
  301. // TODO: this could do a better job packing
  302. // especially if we allowed breaking up fields across multiple interpolators (to pack them into remaining space...)
  303. // though we would want to only do this if it improves final interpolator count, and is worth it on the target machine
  304. foreach (FieldInfo field in unpacked.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public))
  305. {
  306. if (field.MemberType == MemberTypes.Field)
  307. {
  308. bool isOptional;
  309. if (ShouldSpliceField(unpacked, field, activeFields, out isOptional))
  310. {
  311. string semanticString = GetFieldSemantic(field);
  312. int floatVectorCount;
  313. string fieldType = GetFieldType(field, out floatVectorCount);
  314. string conditional = GetFieldConditional(field);
  315. // System generated fields must appear last in struct definitions
  316. bool isSystemGenerated = field.IsDefined(typeof(SystemGenerated), false);
  317. if (conditional != null)
  318. {
  319. if(isSystemGenerated)
  320. {
  321. systemGenerated.AddShaderChunk("#if " + conditional);
  322. }
  323. else
  324. {
  325. packingStruct.AddShaderChunk("#if " + conditional);
  326. }
  327. packer.AddShaderChunk("#if " + conditional);
  328. unpacker.AddShaderChunk("#if " + conditional);
  329. }
  330. if ((semanticString != null) || (floatVectorCount == 0))
  331. {
  332. if(isSystemGenerated)
  333. {
  334. systemGenerated.AddShaderChunk(fieldType + " " + field.Name + semanticString + ";" + (isDebug ? " // unpacked" : string.Empty));
  335. }
  336. else
  337. {
  338. packingStruct.AddShaderChunk(fieldType + " " + field.Name + semanticString + ";" + (isDebug ? " // unpacked" : string.Empty));
  339. }
  340. packer.AddShaderChunk("output." + field.Name + " = input." + field.Name + ";");
  341. unpacker.AddShaderChunk("output." + field.Name + " = input." + field.Name + ";");
  342. }
  343. else
  344. {
  345. // pack float field
  346. // super simple packing: use the first interpolator that has room for the whole value
  347. int interpIndex = packedCounts.FindIndex(x => (x + floatVectorCount <= 4));
  348. int firstChannel;
  349. if (interpIndex < 0)
  350. {
  351. // allocate a new interpolator
  352. interpIndex = packedCounts.Count;
  353. firstChannel = 0;
  354. packedCounts.Add(floatVectorCount);
  355. }
  356. else
  357. {
  358. // pack into existing interpolator
  359. firstChannel = packedCounts[interpIndex];
  360. packedCounts[interpIndex] += floatVectorCount;
  361. }
  362. // add code to packer and unpacker -- packed data declaration is handled later
  363. string packedChannels = GetChannelSwizzle(firstChannel, floatVectorCount);
  364. packer.AddShaderChunk(string.Format("output.interp{0:00}.{1} = input.{2};", interpIndex, packedChannels, field.Name));
  365. unpacker.AddShaderChunk(string.Format("output.{0} = input.interp{1:00}.{2};", field.Name, interpIndex, packedChannels));
  366. }
  367. if (conditional != null)
  368. {
  369. if(isSystemGenerated)
  370. {
  371. systemGenerated.AddShaderChunk("#endif" + (isDebug ? $" // conditional" : string.Empty));
  372. }
  373. else
  374. {
  375. packingStruct.AddShaderChunk("#endif" + (isDebug ? $" // conditional" : string.Empty));
  376. }
  377. packer.AddShaderChunk("#endif" + (isDebug ? $" // conditional" : string.Empty));
  378. unpacker.AddShaderChunk("#endif" + (isDebug ? $" // conditional" : string.Empty));
  379. }
  380. }
  381. }
  382. }
  383. // add packed data declarations to struct, using the packedCounts
  384. for (int index = 0; index < packedCounts.Count; index++)
  385. {
  386. int count = packedCounts[index];
  387. packingStruct.AddShaderChunk(string.Format("{0} interp{1:00} : TEXCOORD{1};" + (isDebug ? " // auto-packed" : string.Empty), vectorTypeNames[count], index));
  388. }
  389. // Add system generated fields at the end
  390. if(systemGenerated.numberOfChunks > 0)
  391. {
  392. packingStruct.AddGenerator(systemGenerated);
  393. }
  394. // close declarations
  395. packingStruct.Deindent();
  396. packingStruct.AddShaderChunk("};");
  397. packingStruct.AddNewLine();
  398. packer.AddShaderChunk("return output;");
  399. packer.Deindent();
  400. packer.AddShaderChunk("}");
  401. packer.AddNewLine();
  402. unpacker.AddShaderChunk("return output;");
  403. unpacker.Deindent();
  404. unpacker.AddShaderChunk("}");
  405. // combine all of the code into the result
  406. result.AddGenerator(packingStruct);
  407. result.AddGenerator(packer);
  408. result.AddGenerator(unpacker);
  409. }
  410. // returns the offset of the first non-whitespace character, in the range [start, end] inclusive ... will return end if none found
  411. private static int SkipWhitespace(string str, int start, int end)
  412. {
  413. int index = start;
  414. while (index < end)
  415. {
  416. char c = str[index];
  417. if (!Char.IsWhiteSpace(c))
  418. {
  419. break;
  420. }
  421. index++;
  422. }
  423. return index;
  424. }
  425. public class TemplatePreprocessor
  426. {
  427. // inputs
  428. ActiveFields activeFields;
  429. Dictionary<string, string> namedFragments;
  430. string templatePath;
  431. bool isDebug;
  432. string assemblyName;
  433. string resourceClassName;
  434. // intermediates
  435. HashSet<string> includedFiles;
  436. // outputs
  437. ShaderStringBuilder result;
  438. List<string> sourceAssetDependencyPaths;
  439. public TemplatePreprocessor(ActiveFields activeFields, Dictionary<string, string> namedFragments, bool isDebug, string templatePath, List<string> sourceAssetDependencyPaths, string assemblyName, string resourceClassName, ShaderStringBuilder outShaderCodeResult = null)
  440. {
  441. this.activeFields = activeFields;
  442. this.namedFragments = namedFragments;
  443. this.isDebug = isDebug;
  444. this.templatePath = templatePath;
  445. this.sourceAssetDependencyPaths = sourceAssetDependencyPaths;
  446. this.assemblyName = assemblyName;
  447. this.resourceClassName = resourceClassName;
  448. this.result = outShaderCodeResult ?? new ShaderStringBuilder();
  449. includedFiles = new HashSet<string>();
  450. }
  451. public ShaderStringBuilder GetShaderCode()
  452. {
  453. return result;
  454. }
  455. public void ProcessTemplateFile(string filePath)
  456. {
  457. if (File.Exists(filePath) &&
  458. !includedFiles.Contains(filePath))
  459. {
  460. includedFiles.Add(filePath);
  461. if (sourceAssetDependencyPaths != null)
  462. sourceAssetDependencyPaths.Add(filePath);
  463. string[] templateLines = File.ReadAllLines(filePath);
  464. foreach (string line in templateLines)
  465. {
  466. ProcessTemplateLine(line, 0, line.Length);
  467. }
  468. }
  469. }
  470. private struct Token
  471. {
  472. public string s;
  473. public int start;
  474. public int end;
  475. public Token(string s, int start, int end)
  476. {
  477. this.s = s;
  478. this.start = start;
  479. this.end = end;
  480. }
  481. public static Token Invalid()
  482. {
  483. return new Token(null, 0, 0);
  484. }
  485. public bool IsValid()
  486. {
  487. return (s != null);
  488. }
  489. public bool Is(string other)
  490. {
  491. int len = end - start;
  492. return (other.Length == len) && (0 == string.Compare(s, start, other, 0, len));
  493. }
  494. public string GetString()
  495. {
  496. int len = end - start;
  497. if (len > 0)
  498. {
  499. return s.Substring(start, end - start);
  500. }
  501. return null;
  502. }
  503. }
  504. public void ProcessTemplateLine(string line, int start, int end)
  505. {
  506. bool appendEndln = true;
  507. int cur = start;
  508. while (cur < end)
  509. {
  510. // find an escape code '$'
  511. int dollar = line.IndexOf('$', cur, end - cur);
  512. if (dollar < 0)
  513. {
  514. // no escape code found in the remaining code -- just append the rest verbatim
  515. AppendSubstring(line, cur, true, end, false);
  516. break;
  517. }
  518. else
  519. {
  520. // found $ escape sequence
  521. Token command = ParseIdentifier(line, dollar+1, end);
  522. if (!command.IsValid())
  523. {
  524. Error("ERROR: $ must be followed by a command string (if, splice, or include)", line, dollar+1);
  525. break;
  526. }
  527. else
  528. {
  529. if (command.Is("include"))
  530. {
  531. ProcessIncludeCommand(command, end);
  532. appendEndln = false;
  533. break; // include command always ignores the rest of the line, error or not
  534. }
  535. else if (command.Is("splice"))
  536. {
  537. if (!ProcessSpliceCommand(command, end, ref cur))
  538. {
  539. // error, skip the rest of the line
  540. break;
  541. }
  542. }
  543. else if (command.Is("buildType"))
  544. {
  545. ProcessBuildTypeCommand(command, end);
  546. appendEndln = false;
  547. break; // buildType command always ignores the rest of the line, error or not
  548. }
  549. else
  550. {
  551. // let's see if it is a predicate
  552. Token predicate = ParseUntil(line, dollar + 1, end, ':');
  553. if (!predicate.IsValid())
  554. {
  555. Error("ERROR: unrecognized command: " + command.GetString(), line, command.start);
  556. break;
  557. }
  558. else
  559. {
  560. if (!ProcessPredicate(predicate, end, ref cur, ref appendEndln))
  561. {
  562. break; // skip the rest of the line
  563. }
  564. }
  565. }
  566. }
  567. }
  568. }
  569. if (appendEndln)
  570. {
  571. result.AppendNewLine();
  572. }
  573. }
  574. private void ProcessIncludeCommand(Token includeCommand, int lineEnd)
  575. {
  576. if (Expect(includeCommand.s, includeCommand.end, '('))
  577. {
  578. Token param = ParseString(includeCommand.s, includeCommand.end + 1, lineEnd);
  579. if (!param.IsValid())
  580. {
  581. Error("ERROR: $include expected a string file path parameter", includeCommand.s, includeCommand.end + 1);
  582. }
  583. else
  584. {
  585. var includeLocation = Path.Combine(templatePath, param.GetString());
  586. if (!File.Exists(includeLocation))
  587. {
  588. Error("ERROR: $include cannot find file : " + includeLocation, includeCommand.s, param.start);
  589. }
  590. else
  591. {
  592. int endIndex = result.length;
  593. using(var temp = new ShaderStringBuilder())
  594. {
  595. // Wrap in debug mode
  596. if(isDebug)
  597. {
  598. result.AppendLine("//-------------------------------------------------------------------------------------");
  599. result.AppendLine("// TEMPLATE INCLUDE : " + param.GetString());
  600. result.AppendLine("//-------------------------------------------------------------------------------------");
  601. result.AppendNewLine();
  602. }
  603. // Recursively process templates
  604. ProcessTemplateFile(includeLocation);
  605. // Wrap in debug mode
  606. if(isDebug)
  607. {
  608. result.AppendNewLine();
  609. result.AppendLine("//-------------------------------------------------------------------------------------");
  610. result.AppendLine("// END TEMPLATE INCLUDE : " + param.GetString());
  611. result.AppendLine("//-------------------------------------------------------------------------------------");
  612. }
  613. result.AppendNewLine();
  614. // Required to enforce indentation rules
  615. // Append lines from this include into temporary StringBuilder
  616. // Reduce result length to remove this include
  617. temp.AppendLines(result.ToString(endIndex, result.length - endIndex));
  618. result.length = endIndex;
  619. result.AppendLines(temp.ToCodeBlack());
  620. }
  621. }
  622. }
  623. }
  624. }
  625. private bool ProcessSpliceCommand(Token spliceCommand, int lineEnd, ref int cur)
  626. {
  627. if (!Expect(spliceCommand.s, spliceCommand.end, '('))
  628. {
  629. return false;
  630. }
  631. else
  632. {
  633. Token param = ParseUntil(spliceCommand.s, spliceCommand.end + 1, lineEnd, ')');
  634. if (!param.IsValid())
  635. {
  636. Error("ERROR: splice command is missing a ')'", spliceCommand.s, spliceCommand.start);
  637. return false;
  638. }
  639. else
  640. {
  641. // append everything before the beginning of the escape sequence
  642. AppendSubstring(spliceCommand.s, cur, true, spliceCommand.start-1, false);
  643. // find the named fragment
  644. string name = param.GetString(); // unfortunately this allocates a new string
  645. string fragment;
  646. if ((namedFragments != null) && namedFragments.TryGetValue(name, out fragment))
  647. {
  648. // splice the fragment
  649. result.Append(fragment);
  650. }
  651. else
  652. {
  653. // no named fragment found
  654. result.Append("/* WARNING: $splice Could not find named fragment '{0}' */", name);
  655. }
  656. // advance to just after the ')' and continue parsing
  657. cur = param.end + 1;
  658. }
  659. }
  660. return true;
  661. }
  662. private void ProcessBuildTypeCommand(Token command, int endLine)
  663. {
  664. if (Expect(command.s, command.end, '('))
  665. {
  666. Token param = ParseUntil(command.s, command.end + 1, endLine, ')');
  667. if (!param.IsValid())
  668. {
  669. Error("ERROR: buildType command is missing a ')'", command.s, command.start);
  670. }
  671. else
  672. {
  673. string typeName = param.GetString();
  674. Type type = GenerationUtils.GetTypeForStruct(typeName, resourceClassName, assemblyName);
  675. if (type == null)
  676. {
  677. Error("ERROR: buildType could not find type : " + typeName, command.s, param.start);
  678. }
  679. else
  680. {
  681. ShaderGenerator temp = new ShaderGenerator();
  682. temp.Indent();
  683. temp.AddShaderChunk("// Generated Type: " + typeName);
  684. BuildType(type, activeFields, temp, isDebug);
  685. result.AppendLine(temp.GetShaderString(0, false));
  686. }
  687. }
  688. }
  689. }
  690. private bool ProcessPredicate(Token predicate, int endLine, ref int cur, ref bool appendEndln)
  691. {
  692. // eval if(param)
  693. var fieldName = predicate.GetString();
  694. var nonwhitespace = SkipWhitespace(predicate.s, predicate.end + 1, endLine);
  695. if (!fieldName.StartsWith("features") && activeFields.permutationCount > 0)
  696. {
  697. var passedPermutations = activeFields.allPermutations.instances
  698. .Where(i => i.Contains(fieldName))
  699. .ToList();
  700. if (passedPermutations.Count > 0)
  701. {
  702. var ifdefs = KeywordUtil.GetKeywordPermutationSetConditional(
  703. passedPermutations.Select(i => i.permutationIndex).ToList()
  704. );
  705. result.AppendLine(ifdefs);
  706. // Append the rest of the line
  707. AppendSubstring(predicate.s, nonwhitespace, true, endLine, false);
  708. result.AppendLine("");
  709. result.AppendLine("#endif");
  710. return false;
  711. }
  712. return false;
  713. }
  714. else
  715. {
  716. // eval if(param)
  717. if (activeFields.baseInstance.Contains(fieldName))
  718. {
  719. // predicate is active
  720. // append everything before the beginning of the escape sequence
  721. AppendSubstring(predicate.s, cur, true, predicate.start-1, false);
  722. // continue parsing the rest of the line, starting with the first nonwhitespace character
  723. cur = nonwhitespace;
  724. return true;
  725. }
  726. else
  727. {
  728. // predicate is not active
  729. if (isDebug)
  730. {
  731. // append everything before the beginning of the escape sequence
  732. AppendSubstring(predicate.s, cur, true, predicate.start-1, false);
  733. // append the rest of the line, commented out
  734. result.Append("// ");
  735. AppendSubstring(predicate.s, nonwhitespace, true, endLine, false);
  736. }
  737. else
  738. {
  739. // don't append anything
  740. appendEndln = false;
  741. }
  742. return false;
  743. }
  744. }
  745. }
  746. private Token ParseIdentifier(string code, int start, int end)
  747. {
  748. if (start < end)
  749. {
  750. char c = code[start];
  751. if (Char.IsLetter(c) || (c == '_'))
  752. {
  753. int cur = start + 1;
  754. while (cur < end)
  755. {
  756. c = code[cur];
  757. if (!(Char.IsLetterOrDigit(c) || (c == '_')))
  758. break;
  759. cur++;
  760. }
  761. return new Token(code, start, cur);
  762. }
  763. }
  764. return Token.Invalid();
  765. }
  766. private Token ParseString(string line, int start, int end)
  767. {
  768. if (Expect(line, start, '"'))
  769. {
  770. return ParseUntil(line, start + 1, end, '"');
  771. }
  772. return Token.Invalid();
  773. }
  774. private Token ParseUntil(string line, int start, int end, char endChar)
  775. {
  776. int cur = start;
  777. while (cur < end)
  778. {
  779. if (line[cur] == endChar)
  780. {
  781. return new Token(line, start, cur);
  782. }
  783. cur++;
  784. }
  785. return Token.Invalid();
  786. }
  787. private bool Expect(string line, int location, char expected)
  788. {
  789. if ((location < line.Length) && (line[location] == expected))
  790. {
  791. return true;
  792. }
  793. Error("Expected '" + expected + "'", line, location);
  794. return false;
  795. }
  796. private void Error(string error, string line, int location)
  797. {
  798. // append the line for context
  799. result.Append("\n");
  800. result.Append("// ");
  801. AppendSubstring(line, 0, true, line.Length, false);
  802. result.Append("\n");
  803. // append the location marker, and error description
  804. result.Append("// ");
  805. result.AppendSpaces(location);
  806. result.Append("^ ");
  807. result.Append(error);
  808. result.Append("\n");
  809. }
  810. // an easier to use version of substring Append() -- explicit inclusion on each end, and checks for positive length
  811. private void AppendSubstring(string str, int start, bool includeStart, int end, bool includeEnd)
  812. {
  813. if (!includeStart)
  814. {
  815. start++;
  816. }
  817. if (!includeEnd)
  818. {
  819. end--;
  820. }
  821. int count = end - start + 1;
  822. if (count > 0)
  823. {
  824. result.Append(str, start, count);
  825. }
  826. }
  827. }
  828. public static void ApplyDependencies(IActiveFields activeFields, List<Dependency[]> dependsList)
  829. {
  830. // add active fields to queue
  831. Queue<string> fieldsToPropagate = new Queue<string>();
  832. foreach (var f in activeFields.fields)
  833. {
  834. fieldsToPropagate.Enqueue(f);
  835. }
  836. // foreach field in queue:
  837. while (fieldsToPropagate.Count > 0)
  838. {
  839. string field = fieldsToPropagate.Dequeue();
  840. if (activeFields.Contains(field)) // this should always be true
  841. {
  842. // find all dependencies of field that are not already active
  843. foreach (Dependency[] dependArray in dependsList)
  844. {
  845. foreach (Dependency d in dependArray.Where(d => (d.name == field) && !activeFields.Contains(d.dependsOn)))
  846. {
  847. // activate them and add them to the queue
  848. activeFields.Add(d.dependsOn);
  849. fieldsToPropagate.Enqueue(d.dependsOn);
  850. }
  851. }
  852. }
  853. }
  854. }
  855. }
  856. }