ShaderSourceMap.cs 2.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. using System;
  2. using System.Collections.Generic;
  3. using UnityEditor.Graphing;
  4. namespace UnityEditor.ShaderGraph
  5. {
  6. class ShaderSourceMap
  7. {
  8. // Indicates where a new node begins
  9. List<int> m_LineStarts;
  10. List<AbstractMaterialNode> m_Nodes;
  11. int m_LineCount;
  12. internal ShaderSourceMap(string source, List<ShaderStringMapping> mappings)
  13. {
  14. m_LineStarts = new List<int>();
  15. m_Nodes = new List<AbstractMaterialNode>();
  16. // File line numbers are 1-based
  17. var line = 1;
  18. var currentIndex = 0;
  19. foreach (var mapping in mappings)
  20. {
  21. var stopIndex = mapping.startIndex + mapping.count;
  22. if (currentIndex >= stopIndex)
  23. continue;
  24. m_LineStarts.Add(line);
  25. m_Nodes.Add(mapping.node);
  26. while (currentIndex < stopIndex && currentIndex != -1)
  27. {
  28. currentIndex = source.IndexOf('\n', currentIndex + 1);
  29. line++;
  30. }
  31. if (currentIndex == -1)
  32. break;
  33. }
  34. m_LineCount = line-1;
  35. }
  36. // Binary search that behaves like C++'s std::lower_bound()
  37. public AbstractMaterialNode FindNode(int line)
  38. {
  39. // line is 1-based throughout this function
  40. if (line > m_LineCount || line <= 0)
  41. return null;
  42. var l = 0;
  43. var r = m_LineStarts.Count - 1;
  44. while (l <= r)
  45. {
  46. var m = (l + r) / 2;
  47. var lineStart = m_LineStarts[m];
  48. var lineStop = m == m_LineStarts.Count-1 ? m_LineCount+1 : m_LineStarts[m + 1];
  49. if (line >= lineStop)
  50. l = m + 1;
  51. else if (line < lineStart)
  52. r = m - 1;
  53. else
  54. return m_Nodes[m];
  55. }
  56. throw new Exception("Something went wrong in binary search");
  57. }
  58. }
  59. }