WuhuIslandTesting/Library/PackageCache/com.unity.burst@1.8.4/Editor/BurstInspectorGUI.cs
2025-01-07 02:06:59 +01:00

1571 lines
64 KiB
C#
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Text;
using System.Threading.Tasks;
using Unity.Burst.LowLevel;
using UnityEditor;
using System.Text.RegularExpressions;
using UnityEditor.IMGUI.Controls;
using UnityEngine;
[assembly: InternalsVisibleTo("Unity.Burst.Editor.Tests")]
[assembly: InternalsVisibleTo("Unity.Burst.Tester.Editor.Tests")]
namespace Unity.Burst.Editor
{
internal class BurstInspectorGUI : EditorWindow
{
private static bool Initialized;
private static void EnsureInitialized()
{
if (Initialized)
{
return;
}
Initialized = true;
BurstLoader.OnBurstShutdown += () =>
{
if (EditorWindow.HasOpenInstances<BurstInspectorGUI>())
{
var window = EditorWindow.GetWindow<BurstInspectorGUI>("Burst Inspector");
window.Close();
}
};
}
private const string FontSizeIndexPref = "BurstInspectorFontSizeIndex";
private static readonly string[] DisassemblyKindNames =
{
"Assembly",
".NET IL",
"LLVM IR (Unoptimized)",
"LLVM IR (Optimized)",
"LLVM IR Optimisation Diagnostics"
};
internal enum AssemblyOptions
{
PlainWithoutDebugInformation = 0,
PlainWithDebugInformation = 1,
EnhancedWithMinimalDebugInformation = 2,
EnhancedWithFullDebugInformation = 3,
ColouredWithMinimalDebugInformation = 4,
ColouredWithFullDebugInformation = 5
}
internal AssemblyOptions? _assemblyKind = null;
private AssemblyOptions? _assemblyKindPrior = null;
private AssemblyOptions _oldAssemblyKind;
private bool SupportsEnhancedRendering => _disasmKind == DisassemblyKind.Asm || _disasmKind == DisassemblyKind.OptimizedIR || _disasmKind == DisassemblyKind.UnoptimizedIR;
private static string[] DisasmOptions;
internal static string[] GetDisasmOptions()
{
if (DisasmOptions == null)
{
// We can't initialize this in BurstInspectorGUI.cctor because BurstCompilerOptions may not yet
// have been initialized by BurstLoader. So we initialize on-demand here. This method doesn't need to
// be thread-safe because it's only called from the UI thread.
DisasmOptions = new[]
{
"\n" + BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDump, NativeDumpFlags.Asm),
"\n" + BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDump, NativeDumpFlags.IL),
"\n" + BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDump, NativeDumpFlags.IR),
"\n" + BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDump, NativeDumpFlags.IROptimized),
"\n" + BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDump, NativeDumpFlags.IRPassAnalysis)
};
}
return DisasmOptions;
}
private static readonly SplitterState TreeViewSplitterState = new SplitterState(new float[] { 30, 70 }, new int[] { 128, 128 }, null);
private static readonly string[] TargetCpuNames = Enum.GetNames(typeof(BurstTargetCpu));
private static readonly string[] SIMDSmellTest = { "False", "True" };
private static readonly int[] FontSizes =
{
8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 20
};
private static string[] _fontSizesText;
internal const int _scrollbarThickness = 14;
internal float _buttonOverlapInspectorView = 0;
/// <remarks>Used because it's not legal to change layout of GUI in a frame without the users input.</remarks>
private float _buttonBarWidth = -1;
[NonSerialized]
internal readonly BurstDisassembler _burstDisassembler;
private const string BurstSettingText = "Inspector Settings/";
[SerializeField] private BurstTargetCpu _targetCpu = BurstTargetCpu.Auto;
[SerializeField] private DisassemblyKind _disasmKind = DisassemblyKind.Asm;
[SerializeField] private DisassemblyKind _oldDisasmKind = DisassemblyKind.Asm;
[NonSerialized]
internal GUIStyle fixedFontStyle;
[NonSerialized]
internal int fontSizeIndex = -1;
[SerializeField] private int _previousTargetIndex = -1;
[SerializeField] private bool _safetyChecks = false;
[SerializeField] private bool _showBranchMarkers = true;
[SerializeField] private bool _enhancedDisassembly = true;
[SerializeField] private string _searchFilterJobs;
[SerializeField] private bool _showUnityNamespaceJobs = false;
[SerializeField] private bool _showDOTSGeneratedJobs = false;
[SerializeField] private bool _focusTargetJob = true;
[SerializeField] private string _searchFilterAssembly = String.Empty;
[SerializeField] private bool _sameTargetButDifferentAssemblyKind = false;
[SerializeField] internal Vector2 _scrollPos;
internal SearchField _searchFieldJobs;
internal SearchField _searchFieldAssembly;
private bool saveSearchFieldFromEvent = false;
[SerializeField] private bool _searchBarVisible = true;
[SerializeField] private string _selectedItem;
[NonSerialized]
private BurstCompileTarget _target;
[NonSerialized]
private List<BurstCompileTarget> _targets;
// Used as a serialized representation of _targets:
[SerializeField] private List<string> targetNames;
[NonSerialized]
internal LongTextArea _textArea;
internal Rect _inspectorView;
[NonSerialized]
internal Font _font;
[NonSerialized]
internal BurstMethodTreeView _treeView;
// Serialized representation of _treeView:
[SerializeField] private TreeViewState treeViewState;
[NonSerialized]
internal bool _initialized;
[NonSerialized]
private bool _requiresRepaint;
private int FontSize => FontSizes[fontSizeIndex];
private static readonly Regex _rx = new Regex(@"^.*\(\d+,\d+\):\sBurst\serror");
private bool _leftClicked = false;
[SerializeField] private bool _isCompileError = false;
[SerializeField] private bool _prevWasCompileError;
[SerializeField] private bool _smellTest = false;
// Caching GUIContent and style options for button bar
private readonly GUIContent _contentShowUnityNamespaceJobs = new GUIContent("Show Unity Namespace");
private readonly GUIContent _contentShowDOTSGeneratedJobs = new GUIContent("Show \".Generated\"");
private readonly GUIContent _contentDisasm = new GUIContent("Enhanced With Minimal Debug Information");
private readonly GUIContent _contentCollapseToCode = new GUIContent("Focus on Code");
private readonly GUIContent _contentExpandAll = new GUIContent("Expand All");
private readonly GUIContent _contentBranchLines = new GUIContent("Show Branch Flow");
private readonly GUIContent[] _contentsTarget =
Enum.GetNames(typeof(BurstTargetCpu)).Select(str => new GUIContent($"Target ({str})")).ToArray();
private readonly GUIContent[] _contentsFontSize =
FontSizes.Select(v => new GUIContent($"Font Size ({v})")).ToArray();
private readonly GUIContent[] _contentsSmellTest =
{
new GUIContent("Highlight SIMD Scalar vs Packed (False)"),
new GUIContent("Highlight SIMD Scalar vs Packed (True)")
};
// content for button search bar
private readonly GUIContent _ignoreCase = new GUIContent("Match Case");
private readonly GUIContent _matchWord = new GUIContent("Whole words");
private readonly GUIContent _regexSearch = new GUIContent("Regex");
private readonly GUILayoutOption[] _toolbarStyleOptions = { GUILayout.ExpandWidth(true), GUILayout.MinWidth(5 * 10) };
private readonly string[] _branchMarkerOptions = { "Hide Branch Flow", "Show Branch Flow" };
private readonly string[] _safetyCheckOptions = { "Safety Check On", "Safety Check Off" };
private enum KeyboardOperation
{
SelectAll,
Copy,
MoveLeft,
MoveRight,
MoveUp,
MoveDown,
Search,
Escape,
Enter,
}
private Dictionary<Event, KeyboardOperation> _keyboardEvents;
private void FillKeyboardEvent()
{
if (_keyboardEvents != null)
{
return;
}
_keyboardEvents = new Dictionary<Event, KeyboardOperation>();
_keyboardEvents.Add(Event.KeyboardEvent("#left"), KeyboardOperation.MoveLeft);
_keyboardEvents.Add(Event.KeyboardEvent("#right"), KeyboardOperation.MoveRight);
_keyboardEvents.Add(Event.KeyboardEvent("#down"), KeyboardOperation.MoveDown);
_keyboardEvents.Add(Event.KeyboardEvent("#up"), KeyboardOperation.MoveUp);
_keyboardEvents.Add(Event.KeyboardEvent("escape"), KeyboardOperation.Escape);
_keyboardEvents.Add(Event.KeyboardEvent("return"), KeyboardOperation.Enter);
_keyboardEvents.Add(Event.KeyboardEvent("#return"), KeyboardOperation.Enter);
_keyboardEvents.Add(Event.KeyboardEvent("left"), KeyboardOperation.MoveLeft);
_keyboardEvents.Add(Event.KeyboardEvent("right"), KeyboardOperation.MoveRight);
_keyboardEvents.Add(Event.KeyboardEvent("up"), KeyboardOperation.MoveUp);
_keyboardEvents.Add(Event.KeyboardEvent("down"), KeyboardOperation.MoveDown);
if (SystemInfo.operatingSystemFamily == OperatingSystemFamily.MacOSX)
{
_keyboardEvents.Add(Event.KeyboardEvent("%a"), KeyboardOperation.SelectAll);
_keyboardEvents.Add(Event.KeyboardEvent("%c"), KeyboardOperation.Copy);
_keyboardEvents.Add(Event.KeyboardEvent("%f"), KeyboardOperation.Search);
}
else
{
// windows or linux bindings.
_keyboardEvents.Add(Event.KeyboardEvent("^a"), KeyboardOperation.SelectAll);
_keyboardEvents.Add(Event.KeyboardEvent("^c"), KeyboardOperation.Copy);
_keyboardEvents.Add(Event.KeyboardEvent("^f"), KeyboardOperation.Search);
}
}
public BurstInspectorGUI()
{
_burstDisassembler = new BurstDisassembler();
}
private bool DisplayAssemblyKind(Enum assemblyKind)
{
var assemblyOption = (AssemblyOptions)assemblyKind;
if (_disasmKind != DisassemblyKind.Asm || _isCompileError)
{
return assemblyOption == AssemblyOptions.PlainWithoutDebugInformation;
}
return true;
}
public void OnEnable()
{
EnsureInitialized();
var newTreeState = false;
if (treeViewState is null)
{
treeViewState = new TreeViewState();
newTreeState = true;
}
_treeView ??= _treeView = new BurstMethodTreeView
(
treeViewState,
() => _searchFilterJobs,
() => (_showUnityNamespaceJobs, _showDOTSGeneratedJobs)
);
if (_keyboardEvents == null) FillKeyboardEvent();
var assemblyList = BurstReflection.EditorAssembliesThatCanPossiblyContainJobs;
Task.Run(
() =>
{
// Do this stuff asynchronously.
var result = BurstReflection.FindExecuteMethods(assemblyList, BurstReflectionAssemblyOptions.None);
_targets = result.CompileTargets;
_targets.Sort((left, right) => string.Compare(left.GetDisplayName(), right.GetDisplayName(), StringComparison.Ordinal));
return result;
})
.ContinueWith(t =>
{
// Do this stuff on the main (UI) thread.
if (t.Status == TaskStatus.RanToCompletion)
{
foreach (var logMessage in t.Result.LogMessages)
{
switch (logMessage.LogType)
{
case BurstReflection.LogType.Warning:
Debug.LogWarning(logMessage.Message);
break;
case BurstReflection.LogType.Exception:
Debug.LogException(logMessage.Exception);
break;
default:
throw new InvalidOperationException();
}
}
List<string> newNames = _targets.Select(x => x.GetDisplayName()).ToList();
bool identical = !newTreeState && newNames.SequenceEqual(targetNames);
targetNames = newNames;
_treeView.Initialize(_targets, identical);
if (_selectedItem == null || !_treeView.TrySelectByDisplayName(_selectedItem))
{
_previousTargetIndex = -1;
_scrollPos = Vector2.zero;
}
_requiresRepaint = true;
_initialized = true;
}
else if (t.Exception != null)
{
Debug.LogError($"Could not load Inspector: {t.Exception}");
}
});
}
#if !UNITY_2023_1_OR_NEWER
private void CleanupFont()
{
if (_font != null)
{
DestroyImmediate(_font, true);
_font = null;
}
}
public void OnDisable()
{
CleanupFont();
}
#endif
public void Update()
{
// Need to do this because if we call Repaint from anywhere else,
// it doesn't do anything if this window is not currently focused.
if (_requiresRepaint)
{
Repaint();
_requiresRepaint = false;
}
// Need this because pressing new target, and then not invoking new events,
// will leave the assembly unrendered.
// This is not included in above, to minimize needed calls.
if (_target != null && _target.JustLoaded)
{
Repaint();
}
}
/// <summary>
/// Checks if there is space for given content withs style, and starts new horizontalgroup
/// if there is no space on this line.
/// </summary>
private void FlowToNewLine(ref float remainingWidth, float width, Vector2 size)
{
float sizeX = size.x + _scrollbarThickness / 2;
if (sizeX >= remainingWidth)
{
_buttonOverlapInspectorView += size.y + 2;
remainingWidth = width - sizeX;
GUILayout.EndHorizontal();
GUILayout.BeginHorizontal();
}
else
{
remainingWidth -= sizeX;
}
}
private bool IsRaw(AssemblyOptions kind)
{
return kind == AssemblyOptions.PlainWithoutDebugInformation || kind == AssemblyOptions.PlainWithDebugInformation;
}
private bool IsEnhanced(AssemblyOptions kind)
{
return !IsRaw(kind);
}
private bool IsColoured(AssemblyOptions kind)
{
return kind == AssemblyOptions.ColouredWithMinimalDebugInformation || kind == AssemblyOptions.ColouredWithFullDebugInformation;
}
/// <summary>
/// Renders buttons bar, and handles saving/loading of _assemblyKind options when changing in inspector settings
/// that disable/enables some options for _assemblyKind.
/// </summary>
private void HandleButtonBars(BurstCompileTarget target, bool targetChanged, out int fontIndex, out bool collapse, out bool focusCode)
{
// We can only make an educated guess for the correct width.
if (_buttonBarWidth == -1)
{
_buttonBarWidth = (position.width * 2) / 3 - _scrollbarThickness;
}
RenderButtonBars(_buttonBarWidth, target, out fontIndex, out collapse, out focusCode);
var disasmKindChanged = _oldDisasmKind != _disasmKind;
// Handles saving and loading _assemblyKind option when going between settings, that disable/enable some options for it
if ((disasmKindChanged && _oldDisasmKind == DisassemblyKind.Asm && !_isCompileError)
|| (targetChanged && !_prevWasCompileError && _isCompileError && _disasmKind == DisassemblyKind.Asm))
{
// save when _disasmKind changed from Asm WHEN we are not looking at a burst compile error,
// or when target changed from non compile error to compile error and current _disasmKind is Asm.
_oldAssemblyKind = (AssemblyOptions)_assemblyKind;
}
else if ((disasmKindChanged && _disasmKind == DisassemblyKind.Asm && !_isCompileError) ||
(targetChanged && _prevWasCompileError && _disasmKind == DisassemblyKind.Asm))
{
// load when _diasmKind changed to Asm and we are not at burst compile error,
// or when target changed from a burst compile error while _disasmKind is Asm.
_assemblyKind = _oldAssemblyKind;
}
// if _assemblyKind is something that is not available, force it up to PlainWithoutDebugInformation.
if ((_disasmKind != DisassemblyKind.Asm && _assemblyKind != AssemblyOptions.PlainWithoutDebugInformation)
|| _isCompileError)
{
_assemblyKind = AssemblyOptions.PlainWithoutDebugInformation;
}
}
private void RenderButtonBars(float width, BurstCompileTarget target, out int fontIndex, out bool collapse, out bool focus)
{
var remainingWidth = width;
GUILayout.BeginHorizontal();
// First button should never call beginHorizontal().
remainingWidth -= (EditorStyles.popup.CalcSize(_contentDisasm).x + _scrollbarThickness / 2f);
EditorGUI.BeginDisabledGroup(target.DisassemblyKind == DisassemblyKind.IRPassAnalysis);
_assemblyKind = (AssemblyOptions)EditorGUILayout.EnumPopup(GUIContent.none, _assemblyKind, DisplayAssemblyKind, true);
EditorGUI.EndDisabledGroup();
FlowToNewLine(ref remainingWidth, width, EditorStyles.popup.CalcSize(_contentBranchLines));
// Reversed "logic" to match the array of options, which has "positive" case on idx 0.
_safetyChecks = EditorGUILayout.Popup(_safetyChecks ? 0 : 1, _safetyCheckOptions) == 0;
EditorGUI.BeginDisabledGroup(!target.HasRequiredBurstCompileAttributes);
GUIContent targetContent = _contentsTarget[(int)_targetCpu];
FlowToNewLine(ref remainingWidth, width, EditorStyles.popup.CalcSize(targetContent));
_targetCpu = (BurstTargetCpu)LabeledPopup.Popup((int)_targetCpu, targetContent, TargetCpuNames);
GUIContent fontSizeContent = _contentsFontSize[fontSizeIndex];
FlowToNewLine(ref remainingWidth, width, EditorStyles.popup.CalcSize(fontSizeContent));
fontIndex = LabeledPopup.Popup(fontSizeIndex, fontSizeContent, _fontSizesText);
EditorGUI.EndDisabledGroup();
EditorGUI.BeginDisabledGroup(!IsEnhanced((AssemblyOptions)_assemblyKind) || !SupportsEnhancedRendering || _isCompileError);
FlowToNewLine(ref remainingWidth, width, EditorStyles.miniButton.CalcSize(_contentCollapseToCode));
focus = GUILayout.Button(_contentCollapseToCode, EditorStyles.miniButton);
FlowToNewLine(ref remainingWidth, width, EditorStyles.miniButton.CalcSize(_contentExpandAll));
collapse = GUILayout.Button(_contentExpandAll, EditorStyles.miniButton);
FlowToNewLine(ref remainingWidth, width, EditorStyles.popup.CalcSize(_contentBranchLines));
_showBranchMarkers = EditorGUILayout.Popup(Convert.ToInt32(_showBranchMarkers), _branchMarkerOptions) == 1;
int smellTestIdx = Convert.ToInt32(_smellTest);
GUIContent smellTestContent = _contentsSmellTest[smellTestIdx];
FlowToNewLine(ref remainingWidth, width, EditorStyles.popup.CalcSize(smellTestContent));
_smellTest = LabeledPopup.Popup(smellTestIdx, smellTestContent, SIMDSmellTest) == 1;
EditorGUI.EndDisabledGroup();
GUILayout.EndHorizontal();
_oldDisasmKind = _disasmKind;
_disasmKind = (DisassemblyKind)GUILayout.Toolbar((int)_disasmKind, DisassemblyKindNames, _toolbarStyleOptions);
}
/// <summary>
/// Handles mouse events for selecting text.
/// </summary>
/// <remarks>
/// Must be called after Render(...), as it uses the mouse events, and Render(...)
/// need mouse events for buttons etc.
/// </remarks>
private void HandleMouseEventForSelection(Rect workingArea, int controlID, bool showBranchMarkers)
{
var evt = Event.current;
var mousePos = evt.mousePosition;
if (_textArea.MouseOutsideView(workingArea, mousePos, controlID))
{
return;
}
switch (evt.type)
{
case EventType.MouseDown:
// button 0 is left and 1 is right
if (evt.button == 0)
{
_textArea.MouseClicked(showBranchMarkers, evt.shift, mousePos, controlID);
}
else
{
_leftClicked = true;
}
evt.Use();
break;
case EventType.MouseDrag:
_textArea.DragMouse(mousePos, showBranchMarkers);
evt.Use();
break;
case EventType.MouseUp:
_textArea.MouseReleased();
evt.Use();
break;
case EventType.ScrollWheel:
_textArea.DoScroll(workingArea, evt.delta.y);
// we cannot Use() (consume) scrollWheel events, as they are still needed in EndScrollView.
break;
}
}
private bool AssemblyFocused() => !((_treeView != null && _treeView.HasFocus()) || (_searchFieldAssembly != null && _searchFieldAssembly.HasFocus()));
private void HandleKeyboardEventAssemblyView(Rect workingArea, KeyboardOperation op, Event evt, bool showBranchMarkers)
{
switch (op)
{
case KeyboardOperation.SelectAll:
_textArea.SelectAll();
evt.Use();
break;
case KeyboardOperation.Copy:
_textArea.DoSelectionCopy();
evt.Use();
break;
case KeyboardOperation.MoveLeft:
if (evt.shift)
{
if (_textArea.HasSelection) _textArea.MoveSelectionLeft(workingArea, showBranchMarkers);
}
else
{
_textArea.MoveView(LongTextArea.Direction.Left, workingArea);
}
evt.Use();
break;
case KeyboardOperation.MoveRight:
if (evt.shift)
{
if (_textArea.HasSelection) _textArea.MoveSelectionRight(workingArea, showBranchMarkers);
}
else
{
_textArea.MoveView(LongTextArea.Direction.Right, workingArea);
}
evt.Use();
break;
case KeyboardOperation.MoveUp:
if (evt.shift)
{
if (_textArea.HasSelection) _textArea.MoveSelectionUp(workingArea, showBranchMarkers);
}
else
{
_textArea.MoveView(LongTextArea.Direction.Up, workingArea);
}
evt.Use();
break;
case KeyboardOperation.MoveDown:
if (evt.shift)
{
if (_textArea.HasSelection) _textArea.MoveSelectionDown(workingArea, showBranchMarkers);
}
else
{
_textArea.MoveView(LongTextArea.Direction.Down, workingArea);
}
evt.Use();
break;
case KeyboardOperation.Search:
_searchBarVisible = true;
_searchFieldAssembly?.SetFocus();
evt.Use();
break;
}
}
/// <remarks>
/// Must be called after Render(...) because of depenency on LongTextArea.finalAreaSize.
/// </remarks>
private void HandleKeyboardEventForSelection(Rect workingArea, bool showBranchMarkers)
{
var evt = Event.current;
if (!_keyboardEvents.TryGetValue(evt, out var op))
{
return;
}
if (AssemblyFocused())
{
// Do input handling for assembly view.
HandleKeyboardEventAssemblyView(workingArea, op, evt, showBranchMarkers);
}
else
{
// This amounts to logic for all else.
switch (op)
{
case KeyboardOperation.Escape:
if (_searchFieldAssembly != null && _searchFieldAssembly.HasFocus() && _searchFilterAssembly == "")
{
_searchBarVisible = false;
evt.Use();
}
break;
case KeyboardOperation.Enter:
if (_searchFieldAssembly != null && _searchFieldAssembly.HasFocus())
{
_textArea.NextSearchHit(evt.shift, workingArea);
saveSearchFieldFromEvent = true;
evt.Use();
}
break;
}
}
}
private void RenderCompileTargetsFilters(float width)
{
GUILayout.BeginHorizontal();
// Handle and render filtering toggles:
var newShowUnityTests = GUILayout.Toggle(_showUnityNamespaceJobs, _contentShowUnityNamespaceJobs);
FlowToNewLine(ref width, width, EditorStyles.toggle.CalcSize(_contentShowDOTSGeneratedJobs));
var newShowDOTSGeneratedJobs = GUILayout.Toggle(_showDOTSGeneratedJobs, _contentShowDOTSGeneratedJobs);
GUILayout.EndHorizontal();
if (newShowUnityTests != _showUnityNamespaceJobs || newShowDOTSGeneratedJobs != _showDOTSGeneratedJobs)
{
_showDOTSGeneratedJobs = newShowDOTSGeneratedJobs;
_showUnityNamespaceJobs = newShowUnityTests;
_treeView.Reload();
}
// Handle and render search filter:
var newFilter = _searchFieldJobs.OnGUI(_searchFilterJobs);
if (newFilter != _searchFilterJobs)
{
_searchFilterJobs = newFilter;
_treeView.Reload();
}
}
private void CompileNewTarget(BurstCompileTarget target, BurstCompilerOptions targetOptions)
{
if (target.IsLoading) { return; }
target.IsLoading = true;
target.JustLoaded = false;
// Setup target and it's compilation options.
// This is done here as EditorGUIUtility.isProSkin must be on main thread.
target.TargetCpu = _targetCpu;
target.DisassemblyKind = _disasmKind;
targetOptions.EnableBurstSafetyChecks = _safetyChecks;
target.IsDarkMode = EditorGUIUtility.isProSkin;
targetOptions.EnableBurstCompileSynchronously = true;
// Don't set debug mode, because it disables optimizations.
// Instead we set debug level (None, Full, LineOnly) below.
targetOptions.EnableBurstDebug = false;
Task.Run(() =>
{
var options = new StringBuilder();
if (targetOptions.TryGetOptions(target.IsStaticMethod ? (MemberInfo)target.Method : target.JobType, true, out var defaultOptions))
{
options.AppendLine(defaultOptions);
// Disables the 2 current warnings generated from code (since they clutter up the inspector display)
// BC1370 - throw inside code not guarded with ConditionalSafetyCheck attribute
// BC1322 - loop intrinsic on loop that has been optimized away
options.AppendLine($"{BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDisableWarnings, "BC1370;BC1322")}");
options.AppendLine($"{BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionTarget, TargetCpuNames[(int)_targetCpu])}");
// For IRPassAnalysis, we always want full debug information.
if (_disasmKind != DisassemblyKind.IRPassAnalysis)
{
switch (_assemblyKind)
{
case AssemblyOptions.EnhancedWithMinimalDebugInformation:
case AssemblyOptions.ColouredWithMinimalDebugInformation:
options.AppendLine($"{BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDebug, "2")}");
break;
case AssemblyOptions.ColouredWithFullDebugInformation:
case AssemblyOptions.EnhancedWithFullDebugInformation:
case AssemblyOptions.PlainWithDebugInformation:
options.AppendLine($"{BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDebug, "1")}");
break;
default:
case AssemblyOptions.PlainWithoutDebugInformation:
options.AppendLine($"{BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDebug, "0")}");
break;
}
}
else
{
options.AppendLine($"{BurstCompilerOptions.GetOption(BurstCompilerOptions.OptionDebug, "1")}");
}
var baseOptions = options.ToString();
target.RawDisassembly = GetDisassembly(target.Method, baseOptions + GetDisasmOptions()[(int)_disasmKind]);
target.FormattedDisassembly = null;
target.IsBurstError = IsBurstError(target.RawDisassembly);
}
target.IsLoading = false;
target.JustLoaded = true;
});
}
private void RenderBurstJobMenu()
{
float width = position.width / 3;
GUILayout.BeginVertical(GUILayout.Width(width));
// Render Treeview showing burst targets:
GUILayout.Label("Compile Targets", EditorStyles.boldLabel);
RenderCompileTargetsFilters(width);
// Does not give proper rect during layout event.
_inspectorView = GUILayoutUtility.GetRect(GUIContent.none, GUIStyle.none, GUILayout.ExpandHeight(true), GUILayout.ExpandWidth(true));
_treeView.OnGUI(_inspectorView);
GUILayout.EndVertical();
}
private void HandleHorizontalFocus(float workingWidth, bool shouldSetupText, bool isTextFormatted)
{
if (!shouldSetupText || !isTextFormatted || !_burstDisassembler.IsInitialized) { return; }
var branchFiller = _textArea.MaxLineDepth * 10;
if (branchFiller < workingWidth / 2f) { return; }
// Do horizontal padding:
_scrollPos.x = _textArea.MaxLineDepth * 10;
}
private static void RenderLoading()
{
GUILayout.BeginHorizontal();
GUILayout.FlexibleSpace();
GUILayout.BeginVertical();
GUILayout.FlexibleSpace();
GUILayout.Label("Loading...");
GUILayout.FlexibleSpace();
GUILayout.EndVertical();
GUILayout.FlexibleSpace();
GUILayout.EndHorizontal();
}
public void OnGUI()
{
if (!_initialized)
{
RenderLoading();
return;
}
// used to give hot control to inspector when a mouseDown event has happened.
// This way we can register a mouseUp happening outside inspector.
int controlID = GUIUtility.GetControlID(FocusType.Passive);
// Make sure that editor options are synchronized
BurstEditorOptions.EnsureSynchronized();
if (_fontSizesText == null)
{
_fontSizesText = new string[FontSizes.Length];
for (var i = 0; i < FontSizes.Length; ++i) _fontSizesText[i] = FontSizes[i].ToString();
}
if (fontSizeIndex == -1)
{
fontSizeIndex = EditorPrefs.GetInt(FontSizeIndexPref, 5);
fontSizeIndex = Math.Max(0, fontSizeIndex);
fontSizeIndex = Math.Min(fontSizeIndex, FontSizes.Length - 1);
}
if (fixedFontStyle == null || fixedFontStyle.font == null) // also check .font as it's reset somewhere when going out of play mode.
{
fixedFontStyle = new GUIStyle(GUI.skin.label);
#if UNITY_2023_1_OR_NEWER
_font = EditorGUIUtility.Load("Fonts/RobotoMono/RobotoMono-Regular.ttf") as Font;
#else
string fontName;
if (Application.platform == RuntimePlatform.WindowsEditor)
{
fontName = "Consolas";
}
else
{
fontName = "Courier";
}
CleanupFont();
_font = Font.CreateDynamicFontFromOSFont(fontName, FontSize);
#endif
fixedFontStyle.font = _font;
fixedFontStyle.fontSize = FontSize;
}
if (_searchFieldJobs == null) _searchFieldJobs = new SearchField();
if (_textArea == null) _textArea = new LongTextArea();
GUILayout.BeginHorizontal();
// SplitterGUILayout.BeginHorizontalSplit is internal in Unity but we don't have much choice
SplitterGUILayout.BeginHorizontalSplit(TreeViewSplitterState);
RenderBurstJobMenu();
GUILayout.BeginVertical();
var selection = _treeView.GetSelection();
if (selection.Count == 1)
{
var targetIndex = selection[0];
_target = _targets[targetIndex - 1];
var targetOptions = _target.Options;
var targetChanged = _previousTargetIndex != targetIndex;
_previousTargetIndex = targetIndex;
// Stash selected item name to handle domain reloads more gracefully
_selectedItem = _target.GetDisplayName();
if (_assemblyKind == null)
{
if (_enhancedDisassembly)
{
_assemblyKind = AssemblyOptions.ColouredWithMinimalDebugInformation;
}
else
{
_assemblyKind = AssemblyOptions.PlainWithoutDebugInformation;
}
_oldAssemblyKind = (AssemblyOptions)_assemblyKind;
}
// We are currently formatting only Asm output
var isTextFormatted = IsEnhanced((AssemblyOptions)_assemblyKind) && SupportsEnhancedRendering;
// Depending if we are formatted or not, we don't render the same text
var textToRender = _target.RawDisassembly?.TrimStart('\n');
// Only refresh if we are switching to a new selection that hasn't been disassembled yet
// Or we are changing disassembly settings (safety checks / enhanced disassembly)
var targetRefresh = textToRender == null
|| _target.DisassemblyKind != _disasmKind
|| targetOptions.EnableBurstSafetyChecks != _safetyChecks
|| _target.TargetCpu != _targetCpu
|| _target.IsDarkMode != EditorGUIUtility.isProSkin;
if (_assemblyKindPrior != _assemblyKind)
{
targetRefresh = true;
_assemblyKindPrior = _assemblyKind; // Needs to be refreshed, as we need to change disassembly options
// If the target did not changed but our assembly kind did, we need to remember this.
if (!targetChanged)
{
_sameTargetButDifferentAssemblyKind = true;
}
}
// If the previous target changed the assembly kind and we have a target change, we need to
// refresh the assembly because we'll have cached the previous assembly kinds output rather
// than the one requested.
if (_sameTargetButDifferentAssemblyKind && targetChanged)
{
targetRefresh = true;
_sameTargetButDifferentAssemblyKind = false;
}
if (targetRefresh)
{
CompileNewTarget(_target, targetOptions);
}
_prevWasCompileError = _isCompileError;
_isCompileError = _target.IsBurstError;
_buttonOverlapInspectorView = 0;
var oldSimdSmellTest = _smellTest;
HandleButtonBars(_target, targetChanged, out var fontSize, out var expandAllBlocks, out var focusCode);
var simdSmellTestChanged = oldSimdSmellTest != _smellTest;
// Guard against _textArea being used, as the assembly isn't ready yet.
// Have to test against event so it cannot finish between a Layout event and Repaint event;
// this is necessary as we cannot alter GUI between these events.
if (_target.HasRequiredBurstCompileAttributes && (_target.IsLoading || (_target.JustLoaded && Event.current.type != EventType.Layout)))
{
RenderLoading();
// Need to close the splits we opened.
GUILayout.EndVertical();
SplitterGUILayout.EndHorizontalSplit();
GUILayout.EndHorizontal();
return;
}
var justLoaded = _target.JustLoaded;
_target.JustLoaded = false;
// If ´CompileNewTarget´ finishes before we enter loading screen above `textToRender` might not be set.
textToRender ??= _target.RawDisassembly?.TrimStart('\n');
if (!string.IsNullOrEmpty(textToRender))
{
// we should only call SetDisassembler(...) the first time assemblyKind is changed with same target.
// Otherwise it will kep re-initializing fields such as _folded, meaning we can no longer fold/unfold.
var shouldSetupText = !_textArea.IsTextSet(_selectedItem)
|| justLoaded
|| simdSmellTestChanged;
if (shouldSetupText)
{
_textArea.SetText(
_selectedItem,
textToRender,
_target.IsDarkMode,
_burstDisassembler,
isTextFormatted && _burstDisassembler.Initialize(
textToRender,
FetchAsmKind(_targetCpu, _disasmKind),
_target.IsDarkMode,
IsColoured((AssemblyOptions)_assemblyKind),
_smellTest));
}
if (justLoaded)
{
_scrollPos = Vector2.zero;
}
HandleHorizontalFocus(
_inspectorView.width == 1f ? _buttonBarWidth : _inspectorView.width,
shouldSetupText,
isTextFormatted
);
// Fixing lastRectSize to actually be size of scroll view
_inspectorView.position = _scrollPos;
_inspectorView.width = position.width - (_inspectorView.width + _scrollbarThickness);
_inspectorView.height -= (_buttonOverlapInspectorView + 4); //+4 for alignment.
if (_searchBarVisible) _inspectorView.height -= EditorStyles.searchField.CalcHeight(GUIContent.none, 2); // 2 is just arbitrary, as the width does not alter height
// repaint indicate end of frame, so we can alter width for menu items to new correct.
if (Event.current.type == EventType.Repaint)
{
_buttonBarWidth = _inspectorView.width - _scrollbarThickness;
}
// Do search if we did not try and find assembly and we were actually going to do a search.
if (_focusTargetJob && TryFocusJobInAssemblyView(ref _inspectorView, shouldSetupText, _target))
{
_scrollPos.y = _inspectorView.y - _textArea.fontHeight*10;
}
_scrollPos = GUILayout.BeginScrollView(_scrollPos, true, true);
if (Event.current.type != EventType.Layout) // we always want mouse position feedback
{
_textArea.Interact(_inspectorView, Event.current.type);
}
// Set up search information if it is happening.
Regex regx = default;
SearchCriteria sc = default;
var doSearch = _searchBarVisible && _searchFilterAssembly != "";
var wrongRegx = false;
if (doSearch)
{
sc = new SearchCriteria(_searchFilterAssembly, _doIgnoreCase, _doWholeWordMatch, _doRegex);
if (_doRegex)
{
try
{
var opt = RegexOptions.Compiled | RegexOptions.CultureInvariant;
if (!_doIgnoreCase) opt |= RegexOptions.IgnoreCase;
var filter = _searchFilterAssembly;
if (_doWholeWordMatch) filter = @"\b" + filter + @"\b";
regx = new Regex(filter, opt);
}
catch (Exception)
{
// Regex was invalid
wrongRegx = true;
doSearch = false;
}
}
}
var doRepaint = _textArea.Render(fixedFontStyle, _inspectorView, _showBranchMarkers, doSearch, sc, regx);
// A change in the underlying textArea has happened, that requires the GUI to be repainted during this frame.
if (doRepaint) Repaint();
if (Event.current.type != EventType.Layout)
{
HandleMouseEventForSelection(_inspectorView, controlID, _showBranchMarkers);
HandleKeyboardEventForSelection(_inspectorView, _showBranchMarkers);
}
if (_leftClicked)
{
GenericMenu menu = new GenericMenu();
menu.AddItem(EditorGUIUtility.TrTextContent("Copy Selection"), false, _textArea.DoSelectionCopy);
menu.AddItem(EditorGUIUtility.TrTextContent("Copy Color Tags"), _textArea.CopyColorTags, _textArea.ChangeCopyMode);
menu.AddItem(EditorGUIUtility.TrTextContent("Select All"), false, _textArea.SelectAll);
menu.AddItem(EditorGUIUtility.TrTextContent($"Find in {DisassemblyKindNames[(int)_disasmKind]}"), _searchBarVisible, EnableDisableSearchBar);
menu.ShowAsContext();
_leftClicked = false;
}
GUILayout.EndScrollView();
if (_searchBarVisible)
{
if (_searchFieldAssembly == null)
{
_searchFieldAssembly = new SearchField();
_searchFieldAssembly.autoSetFocusOnFindCommand = false;
}
int hitnumbers = _textArea.NrSearchHits > 0 ? _textArea.ActiveSearchNr + 1 : 0;
var hitNumberContent = new GUIContent(" " + hitnumbers + " of " + _textArea.NrSearchHits + " hits");
GUILayout.BeginHorizontal();
// Makes sure that on "enter" keyboard event, the focus is not taken away from searchField.
if (saveSearchFieldFromEvent) GUI.FocusControl("BurstInspectorGUI");
string newFilterAssembly;
if (wrongRegx)
{
var colb = GUI.contentColor;
GUI.contentColor = Color.red;
newFilterAssembly = _searchFieldAssembly.OnGUI(_searchFilterAssembly);
GUI.contentColor = colb;
}
else
{
newFilterAssembly = _searchFieldAssembly.OnGUI(_searchFilterAssembly);
}
// Give back focus to the searchField, if we took it away.
if (saveSearchFieldFromEvent)
{
_searchFieldAssembly.SetFocus();
saveSearchFieldFromEvent = false;
}
if (newFilterAssembly != _searchFilterAssembly)
{
_searchFilterAssembly = newFilterAssembly;
_textArea.StopSearching();
}
_doIgnoreCase = GUILayout.Toggle(_doIgnoreCase, _ignoreCase);
_doWholeWordMatch = GUILayout.Toggle(_doWholeWordMatch, _matchWord);
_doRegex = GUILayout.Toggle(_doRegex, _regexSearch);
GUILayout.Label(hitNumberContent);
if (GUILayout.Button(GUIContent.none, EditorStyles.searchFieldCancelButton))
{
_searchBarVisible = false;
_textArea.StopSearching();
}
GUILayout.EndHorizontal();
}
}
if (fontSize != fontSizeIndex)
{
_textArea.Invalidate();
fontSizeIndex = fontSize;
EditorPrefs.SetInt(FontSizeIndexPref, fontSize);
fixedFontStyle = null;
}
if (expandAllBlocks)
{
_textArea.ExpandAllBlocks();
}
if (focusCode)
{
_textArea.FocusCodeBlocks();
}
}
GUILayout.EndVertical();
SplitterGUILayout.EndHorizontalSplit();
GUILayout.EndHorizontal();
}
public static bool IsBurstError(string disassembly)
{
return _rx.IsMatch(disassembly ?? "");
}
/// <summary>
/// Focuses the view on the active function if a jump is doable.
/// </summary>
/// <param name="workingArea">Current assembly view.</param>
/// <param name="wasTextSetup">Whether text was set in <see cref="_textArea"/>.</param>
/// <param name="target">Target job to find function in.</param>
/// <returns>Whether a focus was attempted or not.</returns>
private bool TryFocusJobInAssemblyView(ref Rect workingArea, bool wasTextSetup, BurstCompileTarget target)
{
bool TryFindByLabel(ref Rect workingArea)
{
var regx = default(Regex);
var sb = new StringBuilder();
if (target.IsStaticMethod)
{
// Search for fullname as label
// Null reference not a danger, because of target being a static method
sb.Append(target.Method.DeclaringType.ToString().Replace("+", "."));
// Generic labels will be sorounded by "", while standard static methods won't
var genericArguments = target.JobType.GenericTypeArguments;
if (genericArguments.Length > 0)
{
// Need to alter the generic arguments from [] to <> form
// Removing [] form
var idx = sb.ToString().LastIndexOf('[');
sb.Remove(idx, sb.Length - idx);
// Adding <> form
sb.Append('<').Append(BurstCompileTarget.Pretty(genericArguments[0]));
for (var i = 1; i < genericArguments.Length; i++)
{
sb.Append(",").Append(BurstCompileTarget.Pretty(genericArguments[i]));
}
sb.Append('>').Append('.').Append(target.Method.Name);
}
else
{
sb.Append('.').Append(target.Method.Name);
}
const RegexOptions opt = RegexOptions.Compiled | RegexOptions.CultureInvariant;
regx = new Regex(@$"{Regex.Escape(sb.ToString())}[^"":]+"":", opt);
}
else
{
// Append full method name. Using display name for simpler access
var targetName = target.GetDisplayName();
// Remove part that tells about used interface
var idx = 0;
// If generic the argument part must also be removed, as they won't match
if ((idx = targetName.IndexOf('[')) == -1) idx = targetName.IndexOf('-') - 1;
targetName = targetName.Remove(idx);
sb.Append($@""".*<{targetName}.*"":");
const RegexOptions opt = RegexOptions.Compiled | RegexOptions.CultureInvariant;
regx = new Regex(sb.ToString(), opt);
}
var sc = new SearchCriteria(sb.ToString(), false, false, true);
return _textArea.SearchText(sc, regx, ref workingArea, true, true);
}
var foundTarget = false;
// _isTextSetLastEvent used so we call this at the first scroll-able event after text was set.
// We cannot scroll during used or layout events, and the order of events are:
// 1. Used event: text is set in textArea
// 2. Layout event: Cannot do the jump yet
// 3. Repaint event: Now jump is doable
// Hence _isTextSetLastEvent is only set on layout events (during phase 2)
if (wasTextSetup)
{
// Need to call Layout to setup fontsize before searching
_textArea.Layout(fixedFontStyle, _textArea.horizontalPad);
foundTarget = TryFindByLabel(ref workingArea);
_textArea.StopSearching(); // Clear the internals of _textArea from this search; to avoid highlighting
// Clear other possible search, so it won't interfere with this.
_searchFilterAssembly = string.Empty;
// We need to do a Repaint() in order for the view to actually update immediately.
if (foundTarget) Repaint();
}
return foundTarget;
}
private void EnableDisableSearchBar()
{
_searchBarVisible = !_searchBarVisible;
if (_searchBarVisible && _searchFieldAssembly != null)
{
_searchFieldAssembly.SetFocus();
}
else if (!_searchBarVisible)
{
_textArea.StopSearching();
}
}
private bool _doIgnoreCase = false;
private bool _doWholeWordMatch = false;
private bool _doRegex = false;
internal static string GetDisassembly(MethodInfo method, string options)
{
try
{
var result = BurstCompilerService.GetDisassembly(method, options);
if (result.IndexOf('\t') >= 0)
{
result = result.Replace("\t", " ");
}
// Workaround to remove timings
if (result.Contains("Burst timings"))
{
var index = result.IndexOf("While compiling", StringComparison.Ordinal);
if (index > 0)
{
result = result.Substring(index);
}
}
return result;
}
catch (Exception e)
{
return "Failed to compile:\n" + e.Message;
}
}
private static BurstDisassembler.AsmKind FetchAsmKind(BurstTargetCpu cpu, DisassemblyKind kind)
{
if (kind == DisassemblyKind.Asm)
{
switch (cpu)
{
case BurstTargetCpu.ARMV7A_NEON32:
case BurstTargetCpu.ARMV8A_AARCH64:
case BurstTargetCpu.ARMV8A_AARCH64_HALFFP:
case BurstTargetCpu.THUMB2_NEON32:
return BurstDisassembler.AsmKind.ARM;
case BurstTargetCpu.WASM32:
return BurstDisassembler.AsmKind.Wasm;
}
return BurstDisassembler.AsmKind.Intel;
}
else
{
return BurstDisassembler.AsmKind.LLVMIR;
}
}
}
/// <summary>
/// Important: id for namespaces are negative, and ids for jobs are positive.
/// This lets us use the id for a job as an index directy into <see cref="_targets"/>.
/// Hence before going from <see cref="TreeViewItem"/> to <see cref="_targets"/> index,
/// One should check whether current item has any children (Only jobs are leafs).
/// </summary>
internal class BurstMethodTreeView : TreeView
{
private readonly Func<string> _getFilter;
private readonly Func<(bool,bool)> _getJobListFilterToggles;
private List<BurstCompileTarget> _targets;
public BurstMethodTreeView(TreeViewState state, Func<string> getFilter, Func<(bool,bool)> getJobListFilterToggles) : base(state)
{
_getFilter = getFilter;
_getJobListFilterToggles = getJobListFilterToggles;
showBorder = true;
}
public void Initialize(List<BurstCompileTarget> targets, bool identicalTargets)
{
_targets = targets;
Reload();
if (!identicalTargets) { ExpandAll(); }
}
/// <remarks>
/// Assumes that <see cref="str"/> is derived from <see cref="Type"/>.<see cref="Type.FullName"/>
/// i.e. types are separated by '+'.
/// </remarks>
/// <param name="str">Given type name string.</param>
/// <returns>(List of namespaces/types, index of method name in <see cref="str"/>)</returns>
internal static (List<StringSlice> ns, int nsEndIdx) ExtractNameSpaces(in string str)
{
if (str is null) { throw new ArgumentNullException(nameof(str)); }
var nameSpaces = new List<StringSlice>();
int len = str.Length;
int scope = 0;
int previdx = 0;
for (int i = 0; i < len; i++)
{
bool stop = false;
char c = str[i];
switch (c)
{
case '(':
// Jump out as we just found argument list!!!
stop = true;
break;
// We keep looking, as classes might have these in name:
case '{':
case '<':
case '[':
scope++;
break;
case '}':
case '>':
case ']':
scope--;
break;
case '+' when scope == 0:
nameSpaces.Add(new StringSlice(str, previdx, i - previdx));
previdx = i + 1;
break;
}
if (stop) { break; }
}
return (nameSpaces, previdx);
}
internal static (int idN, List<TreeViewItem> added, List<StringSlice> nameSpace)
ProcessNewItem(int idN, int idJ, BurstCompileTarget newTarget, List<StringSlice> oldNameSpace)
{
// Find all namespaces used for new target:
string fns = newTarget.JobType.FullName;
string dn = newTarget.GetDisplayName();
(List<StringSlice> newNameSpaces, int nameSpaceEndIdx) = ExtractNameSpaces(fns);
int methodNameIdx = nameSpaceEndIdx;
if (newTarget.IsStaticMethod)
{
// Static method does not have the function name in fns, so fix methodNameIdx.
methodNameIdx = dn.IndexOf(newTarget.Method.Name, StringComparison.InvariantCulture);
// Add the last namespace:
newNameSpaces.Add(new StringSlice(dn, nameSpaceEndIdx, methodNameIdx-1 - nameSpaceEndIdx));
}
string methodName = dn.Substring(methodNameIdx);
int iNewNs = 0;
int lNewNs = newNameSpaces.Count;
int iOldNs = 0;
int lOldNs = oldNameSpace.Count;
var added = new List<TreeViewItem>(lNewNs);
int depth = 0;
// Skip all namespaces shared by previous but increase depth accordingly:
for (; iNewNs < lNewNs && iOldNs < lOldNs && newNameSpaces[iNewNs] == oldNameSpace[iOldNs];
depth++, iNewNs++, iOldNs++) {}
// Handle all new namespaces:
for (; iNewNs < lNewNs;
depth++, iNewNs++)
{
added.Add(new TreeViewItem { id = --idN, depth = depth, displayName = newNameSpaces[iNewNs].ToString()});
}
// Add the function name:
added.Add(new TreeViewItem { id = idJ, depth = depth, displayName = methodName });
return (idN, added, newNameSpaces);
}
protected override TreeViewItem BuildRoot()
{
var root = new TreeViewItem {id = 0, depth = -1, displayName = "Root"};
var allItems = new List<TreeViewItem>();
if (_targets != null)
{
var filter = _getFilter();
var (showUnityNamespaceJobs, showDOTSGeneratedJobs) = _getJobListFilterToggles();
// Have two separate ids so "jobs ids == jobs index".
int idJ = 0;
int idN = 0;
var oldNameSpace = new List<StringSlice>();
foreach (BurstCompileTarget target in _targets)
{
// idJ used as index into _targets, which means it should also take hidden targets into account!
idJ++;
string displayName = target.GetDisplayName();
bool filtered =
(!string.IsNullOrEmpty(filter) &&
displayName.IndexOf(filter, 0, displayName.Length,
StringComparison.InvariantCultureIgnoreCase) < 0)
|| (!showUnityNamespaceJobs &&
displayName.StartsWith("Unity.", StringComparison.InvariantCultureIgnoreCase))
|| (!showDOTSGeneratedJobs &&
displayName.Contains(".Generated"));
if (filtered) { continue; }
try
{
var (newIdN, added, nameSpace) =
ProcessNewItem(idN, idJ, target, oldNameSpace);
allItems.AddRange(added);
idN = newIdN;
oldNameSpace = nameSpace;
}
catch (Exception ex)
{
Debug.Log($"Internal error: Could not add {displayName}\n Because: {ex.Message}");
}
}
}
SetupParentsAndChildrenFromDepths(root, allItems.ToList());
return root;
}
public new IList<int> GetSelection()
{
IList<int> selection = base.GetSelection();
// selection == non-leaf node => no job selected
if (selection.Any() && selection[0] < 0) { return new List<int>(); }
return selection;
}
internal bool TrySelectByDisplayName(string name)
{
var id = 1;
foreach (var t in _targets)
{
if (t.GetDisplayName() == name)
{
try
{
SetSelection(new[] { id });
FrameItem(id);
return true;
}
catch (ArgumentException)
{
// When a search is made in the job list, such that the job we search for is filtered away
// FrameItem(id) will throw a dictionary error. So we catch this, and tell the caller that
// it cannot be selected.
return false;
}
}
else
{
++id;
}
}
return false;
}
protected override void RowGUI(RowGUIArgs args)
{
if (!args.item.hasChildren)
{
var target = _targets[args.item.id - 1];
var wasEnabled = GUI.enabled;
GUI.enabled = target.HasRequiredBurstCompileAttributes;
base.RowGUI(args);
GUI.enabled = wasEnabled;
}
else
{
// Label GUI:
base.RowGUI(args);
}
}
protected override void SingleClickedItem(int id)
{
// If labeled click try and fold/expand:
if (id < 0)
{
SetExpanded(id, !IsExpanded(id));
SetSelection(new List<int>());
}
}
protected override bool CanMultiSelect(TreeViewItem item)
{
return false;
}
}
}