Adding modules/functions to the debugger.
This commit is contained in:
parent
9d7d6df476
commit
573f190a43
|
@ -11,6 +11,8 @@ using WeifenLuo.WinFormsUI.Docking;
|
||||||
|
|
||||||
namespace Xenia.Debug.UI.Controls {
|
namespace Xenia.Debug.UI.Controls {
|
||||||
public partial class BaseDocument : DockContent {
|
public partial class BaseDocument : DockContent {
|
||||||
|
public static MainWindow MainWindow;
|
||||||
|
|
||||||
public BaseDocument() {
|
public BaseDocument() {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,8 @@ using WeifenLuo.WinFormsUI.Docking;
|
||||||
|
|
||||||
namespace Xenia.Debug.UI.Controls {
|
namespace Xenia.Debug.UI.Controls {
|
||||||
public partial class BasePanel : DockContent {
|
public partial class BasePanel : DockContent {
|
||||||
|
public static MainWindow MainWindow;
|
||||||
|
|
||||||
public BasePanel() {
|
public BasePanel() {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Windows.Forms;
|
using System.Windows.Forms;
|
||||||
using WeifenLuo.WinFormsUI.Docking;
|
using WeifenLuo.WinFormsUI.Docking;
|
||||||
|
using Xenia.Debug.UI.Controls;
|
||||||
using Xenia.Debug.UI.Views;
|
using Xenia.Debug.UI.Views;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
|
@ -36,6 +37,9 @@ namespace Xenia.Debug.UI {
|
||||||
public MainWindow() {
|
public MainWindow() {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
|
|
||||||
|
BasePanel.MainWindow = this;
|
||||||
|
BaseDocument.MainWindow = this;
|
||||||
|
|
||||||
dockPanel = new DockPanel();
|
dockPanel = new DockPanel();
|
||||||
dockPanel.Dock = System.Windows.Forms.DockStyle.Fill;
|
dockPanel.Dock = System.Windows.Forms.DockStyle.Fill;
|
||||||
dockPanel.DockBackColor = System.Drawing.SystemColors.AppWorkspace;
|
dockPanel.DockBackColor = System.Drawing.SystemColors.AppWorkspace;
|
||||||
|
@ -76,7 +80,7 @@ namespace Xenia.Debug.UI {
|
||||||
Debugger.StateChanged += Debugger_StateChanged;
|
Debugger.StateChanged += Debugger_StateChanged;
|
||||||
Debugger_StateChanged(this, Debugger.CurrentState);
|
Debugger_StateChanged(this, Debugger.CurrentState);
|
||||||
Debugger.CurrentContext.Changed += CurrentContext_Changed;
|
Debugger.CurrentContext.Changed += CurrentContext_Changed;
|
||||||
CurrentContext_Changed();
|
CurrentContext_Changed(Debugger.CurrentContext);
|
||||||
|
|
||||||
Debugger.Attach();
|
Debugger.Attach();
|
||||||
}
|
}
|
||||||
|
@ -102,7 +106,7 @@ namespace Xenia.Debug.UI {
|
||||||
controlToolStrip.Enabled = enabled;
|
controlToolStrip.Enabled = enabled;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void CurrentContext_Changed() {
|
private void CurrentContext_Changed(Context sender) {
|
||||||
bool enabled = false;
|
bool enabled = false;
|
||||||
switch (Debugger.CurrentContext.RunState) {
|
switch (Debugger.CurrentContext.RunState) {
|
||||||
case RunState.Updating:
|
case RunState.Updating:
|
||||||
|
@ -157,6 +161,11 @@ namespace Xenia.Debug.UI {
|
||||||
dockPanel.ResumeLayout(true, true);
|
dockPanel.ResumeLayout(true, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void OpenFunction(Function function) {
|
||||||
|
var document = codeDocuments[0];
|
||||||
|
document.Show(function);
|
||||||
|
}
|
||||||
|
|
||||||
protected override bool ProcessCmdKey(ref Message msg, Keys keyData) {
|
protected override bool ProcessCmdKey(ref Message msg, Keys keyData) {
|
||||||
//if (keyData == (Keys.Control | Keys.F)) {
|
//if (keyData == (Keys.Control | Keys.F)) {
|
||||||
// MessageBox.Show("What the Ctrl+F?");
|
// MessageBox.Show("What the Ctrl+F?");
|
||||||
|
|
|
@ -19,10 +19,10 @@ namespace Xenia.Debug.UI.Views {
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
|
|
||||||
debugger.BreakpointList.Changed += UpdateBreakpointsList;
|
debugger.BreakpointList.Changed += UpdateBreakpointsList;
|
||||||
UpdateBreakpointsList();
|
UpdateBreakpointsList(debugger.BreakpointList);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void UpdateBreakpointsList() {
|
private void UpdateBreakpointsList(BreakpointList sender) {
|
||||||
breakpointsListView.BeginUpdate();
|
breakpointsListView.BeginUpdate();
|
||||||
breakpointsListView.Items.Clear();
|
breakpointsListView.Items.Clear();
|
||||||
foreach (Breakpoint breakpoint in debugger.BreakpointList) {
|
foreach (Breakpoint breakpoint in debugger.BreakpointList) {
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
namespace Xenia.Debug.UI.Views {
|
using System;
|
||||||
|
|
||||||
|
namespace Xenia.Debug.UI.Views {
|
||||||
partial class CodeDocument {
|
partial class CodeDocument {
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Required designer variable.
|
/// Required designer variable.
|
||||||
|
@ -24,6 +26,12 @@
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private void InitializeComponent() {
|
private void InitializeComponent() {
|
||||||
this.sourceTextBox = new System.Windows.Forms.TextBox();
|
this.sourceTextBox = new System.Windows.Forms.TextBox();
|
||||||
|
this.panel1 = new System.Windows.Forms.Panel();
|
||||||
|
this.disasmMachineCodeCheckBox = new System.Windows.Forms.CheckBox();
|
||||||
|
this.disasmOptimizedHirCheckBox = new System.Windows.Forms.CheckBox();
|
||||||
|
this.disasmUnoptimizedHirCheckBox = new System.Windows.Forms.CheckBox();
|
||||||
|
this.disasmPpcCheckBox = new System.Windows.Forms.CheckBox();
|
||||||
|
this.panel1.SuspendLayout();
|
||||||
this.SuspendLayout();
|
this.SuspendLayout();
|
||||||
//
|
//
|
||||||
// sourceTextBox
|
// sourceTextBox
|
||||||
|
@ -31,20 +39,93 @@
|
||||||
this.sourceTextBox.Anchor = ((System.Windows.Forms.AnchorStyles)((((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom)
|
this.sourceTextBox.Anchor = ((System.Windows.Forms.AnchorStyles)((((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom)
|
||||||
| System.Windows.Forms.AnchorStyles.Left)
|
| System.Windows.Forms.AnchorStyles.Left)
|
||||||
| System.Windows.Forms.AnchorStyles.Right)));
|
| System.Windows.Forms.AnchorStyles.Right)));
|
||||||
|
this.sourceTextBox.Font = new System.Drawing.Font("Consolas", 8.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
|
||||||
|
this.sourceTextBox.HideSelection = false;
|
||||||
this.sourceTextBox.Location = new System.Drawing.Point(12, 75);
|
this.sourceTextBox.Location = new System.Drawing.Point(12, 75);
|
||||||
|
this.sourceTextBox.MaxLength = 99999;
|
||||||
this.sourceTextBox.Multiline = true;
|
this.sourceTextBox.Multiline = true;
|
||||||
this.sourceTextBox.Name = "sourceTextBox";
|
this.sourceTextBox.Name = "sourceTextBox";
|
||||||
this.sourceTextBox.Size = new System.Drawing.Size(740, 586);
|
this.sourceTextBox.ScrollBars = System.Windows.Forms.ScrollBars.Both;
|
||||||
|
this.sourceTextBox.Size = new System.Drawing.Size(740, 563);
|
||||||
this.sourceTextBox.TabIndex = 0;
|
this.sourceTextBox.TabIndex = 0;
|
||||||
//
|
//
|
||||||
|
// panel1
|
||||||
|
//
|
||||||
|
this.panel1.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Bottom | System.Windows.Forms.AnchorStyles.Left)
|
||||||
|
| System.Windows.Forms.AnchorStyles.Right)));
|
||||||
|
this.panel1.Controls.Add(this.disasmMachineCodeCheckBox);
|
||||||
|
this.panel1.Controls.Add(this.disasmOptimizedHirCheckBox);
|
||||||
|
this.panel1.Controls.Add(this.disasmUnoptimizedHirCheckBox);
|
||||||
|
this.panel1.Controls.Add(this.disasmPpcCheckBox);
|
||||||
|
this.panel1.Location = new System.Drawing.Point(12, 644);
|
||||||
|
this.panel1.Name = "panel1";
|
||||||
|
this.panel1.Size = new System.Drawing.Size(740, 17);
|
||||||
|
this.panel1.TabIndex = 1;
|
||||||
|
//
|
||||||
|
// disasmMachineCodeCheckBox
|
||||||
|
//
|
||||||
|
this.disasmMachineCodeCheckBox.AutoSize = true;
|
||||||
|
this.disasmMachineCodeCheckBox.Checked = true;
|
||||||
|
this.disasmMachineCodeCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
|
||||||
|
this.disasmMachineCodeCheckBox.Location = new System.Drawing.Point(302, 0);
|
||||||
|
this.disasmMachineCodeCheckBox.Name = "disasmMachineCodeCheckBox";
|
||||||
|
this.disasmMachineCodeCheckBox.Size = new System.Drawing.Size(95, 17);
|
||||||
|
this.disasmMachineCodeCheckBox.TabIndex = 3;
|
||||||
|
this.disasmMachineCodeCheckBox.Text = "Machine Code";
|
||||||
|
this.disasmMachineCodeCheckBox.UseVisualStyleBackColor = true;
|
||||||
|
this.disasmMachineCodeCheckBox.CheckedChanged += new System.EventHandler(this.DisasmCheckBoxChanged);
|
||||||
|
//
|
||||||
|
// disasmOptimizedHirCheckBox
|
||||||
|
//
|
||||||
|
this.disasmOptimizedHirCheckBox.AutoSize = true;
|
||||||
|
this.disasmOptimizedHirCheckBox.Checked = true;
|
||||||
|
this.disasmOptimizedHirCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
|
||||||
|
this.disasmOptimizedHirCheckBox.Location = new System.Drawing.Point(202, 0);
|
||||||
|
this.disasmOptimizedHirCheckBox.Name = "disasmOptimizedHirCheckBox";
|
||||||
|
this.disasmOptimizedHirCheckBox.Size = new System.Drawing.Size(94, 17);
|
||||||
|
this.disasmOptimizedHirCheckBox.TabIndex = 2;
|
||||||
|
this.disasmOptimizedHirCheckBox.Text = "Optimized HIR";
|
||||||
|
this.disasmOptimizedHirCheckBox.UseVisualStyleBackColor = true;
|
||||||
|
this.disasmOptimizedHirCheckBox.CheckedChanged += new System.EventHandler(this.DisasmCheckBoxChanged);
|
||||||
|
//
|
||||||
|
// disasmUnoptimizedHirCheckBox
|
||||||
|
//
|
||||||
|
this.disasmUnoptimizedHirCheckBox.AutoSize = true;
|
||||||
|
this.disasmUnoptimizedHirCheckBox.Checked = true;
|
||||||
|
this.disasmUnoptimizedHirCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
|
||||||
|
this.disasmUnoptimizedHirCheckBox.Location = new System.Drawing.Point(90, 0);
|
||||||
|
this.disasmUnoptimizedHirCheckBox.Name = "disasmUnoptimizedHirCheckBox";
|
||||||
|
this.disasmUnoptimizedHirCheckBox.Size = new System.Drawing.Size(106, 17);
|
||||||
|
this.disasmUnoptimizedHirCheckBox.TabIndex = 1;
|
||||||
|
this.disasmUnoptimizedHirCheckBox.Text = "Unoptimized HIR";
|
||||||
|
this.disasmUnoptimizedHirCheckBox.UseVisualStyleBackColor = true;
|
||||||
|
this.disasmUnoptimizedHirCheckBox.CheckedChanged += new System.EventHandler(this.DisasmCheckBoxChanged);
|
||||||
|
//
|
||||||
|
// disasmPpcCheckBox
|
||||||
|
//
|
||||||
|
this.disasmPpcCheckBox.AutoSize = true;
|
||||||
|
this.disasmPpcCheckBox.Checked = true;
|
||||||
|
this.disasmPpcCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
|
||||||
|
this.disasmPpcCheckBox.Location = new System.Drawing.Point(0, 0);
|
||||||
|
this.disasmPpcCheckBox.Name = "disasmPpcCheckBox";
|
||||||
|
this.disasmPpcCheckBox.Size = new System.Drawing.Size(84, 17);
|
||||||
|
this.disasmPpcCheckBox.TabIndex = 0;
|
||||||
|
this.disasmPpcCheckBox.Text = "Source PPC";
|
||||||
|
this.disasmPpcCheckBox.UseVisualStyleBackColor = true;
|
||||||
|
this.disasmPpcCheckBox.CheckedChanged += new System.EventHandler(this.DisasmCheckBoxChanged);
|
||||||
|
//
|
||||||
// CodeDocument
|
// CodeDocument
|
||||||
//
|
//
|
||||||
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
|
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
|
||||||
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
|
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
|
||||||
this.ClientSize = new System.Drawing.Size(764, 673);
|
this.ClientSize = new System.Drawing.Size(764, 673);
|
||||||
|
this.Controls.Add(this.panel1);
|
||||||
this.Controls.Add(this.sourceTextBox);
|
this.Controls.Add(this.sourceTextBox);
|
||||||
|
this.Font = new System.Drawing.Font("Microsoft Sans Serif", 8.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
|
||||||
this.Name = "CodeDocument";
|
this.Name = "CodeDocument";
|
||||||
this.Text = "Code";
|
this.Text = "Code";
|
||||||
|
this.panel1.ResumeLayout(false);
|
||||||
|
this.panel1.PerformLayout();
|
||||||
this.ResumeLayout(false);
|
this.ResumeLayout(false);
|
||||||
this.PerformLayout();
|
this.PerformLayout();
|
||||||
|
|
||||||
|
@ -53,5 +134,10 @@
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
private System.Windows.Forms.TextBox sourceTextBox;
|
private System.Windows.Forms.TextBox sourceTextBox;
|
||||||
|
private System.Windows.Forms.Panel panel1;
|
||||||
|
private System.Windows.Forms.CheckBox disasmMachineCodeCheckBox;
|
||||||
|
private System.Windows.Forms.CheckBox disasmOptimizedHirCheckBox;
|
||||||
|
private System.Windows.Forms.CheckBox disasmUnoptimizedHirCheckBox;
|
||||||
|
private System.Windows.Forms.CheckBox disasmPpcCheckBox;
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -14,9 +14,67 @@ namespace Xenia.Debug.UI.Views {
|
||||||
public partial class CodeDocument : BaseDocument {
|
public partial class CodeDocument : BaseDocument {
|
||||||
private readonly Debugger debugger;
|
private readonly Debugger debugger;
|
||||||
|
|
||||||
|
private Function function;
|
||||||
|
|
||||||
public CodeDocument(Debugger debugger) {
|
public CodeDocument(Debugger debugger) {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async void Show(Function function) {
|
||||||
|
this.function = function;
|
||||||
|
|
||||||
|
this.sourceTextBox.Text = "Loading...";
|
||||||
|
await function.Invalidate();
|
||||||
|
|
||||||
|
UpdateSourceTextBox();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UpdateSourceTextBox() {
|
||||||
|
string combinedString = "";
|
||||||
|
|
||||||
|
if (disasmPpcCheckBox.Checked) {
|
||||||
|
combinedString += "Source PPC" + Environment.NewLine;
|
||||||
|
if (function.DisasmPpc != null) {
|
||||||
|
combinedString += function.DisasmPpc;
|
||||||
|
} else {
|
||||||
|
combinedString += "(unavailable)";
|
||||||
|
}
|
||||||
|
combinedString += Environment.NewLine + Environment.NewLine;
|
||||||
|
}
|
||||||
|
if (disasmUnoptimizedHirCheckBox.Checked) {
|
||||||
|
combinedString += "Unoptimized HIR" + Environment.NewLine;
|
||||||
|
if (function.DisasmHirUnoptimized != null) {
|
||||||
|
combinedString += function.DisasmHirUnoptimized;
|
||||||
|
} else {
|
||||||
|
combinedString += "(unavailable)";
|
||||||
|
}
|
||||||
|
combinedString += Environment.NewLine + Environment.NewLine;
|
||||||
|
}
|
||||||
|
if (disasmOptimizedHirCheckBox.Checked) {
|
||||||
|
combinedString += "Optimized HIR" + Environment.NewLine;
|
||||||
|
if (function.DisasmHirOptimized != null) {
|
||||||
|
combinedString += function.DisasmHirOptimized;
|
||||||
|
} else {
|
||||||
|
combinedString += "(unavailable)";
|
||||||
|
}
|
||||||
|
combinedString += Environment.NewLine + Environment.NewLine;
|
||||||
|
}
|
||||||
|
if (disasmMachineCodeCheckBox.Checked) {
|
||||||
|
combinedString += "Machine Code" + Environment.NewLine;
|
||||||
|
if (function.DisasmMachineCode != null) {
|
||||||
|
combinedString += function.DisasmMachineCode;
|
||||||
|
} else {
|
||||||
|
combinedString += "(unavailable)";
|
||||||
|
}
|
||||||
|
combinedString += Environment.NewLine + Environment.NewLine;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.sourceTextBox.Text = combinedString;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DisasmCheckBoxChanged(object sender, EventArgs e) {
|
||||||
|
UpdateSourceTextBox();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,15 +28,17 @@
|
||||||
this.functionsListBox = new System.Windows.Forms.ListBox();
|
this.functionsListBox = new System.Windows.Forms.ListBox();
|
||||||
this.SuspendLayout();
|
this.SuspendLayout();
|
||||||
//
|
//
|
||||||
// moduleComboBox
|
// modulesComboBox
|
||||||
//
|
//
|
||||||
this.modulesComboBox.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left)
|
this.modulesComboBox.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left)
|
||||||
| System.Windows.Forms.AnchorStyles.Right)));
|
| System.Windows.Forms.AnchorStyles.Right)));
|
||||||
|
this.modulesComboBox.DropDownStyle = System.Windows.Forms.ComboBoxStyle.DropDownList;
|
||||||
this.modulesComboBox.FormattingEnabled = true;
|
this.modulesComboBox.FormattingEnabled = true;
|
||||||
this.modulesComboBox.Location = new System.Drawing.Point(12, 12);
|
this.modulesComboBox.Location = new System.Drawing.Point(12, 12);
|
||||||
this.modulesComboBox.Name = "moduleComboBox";
|
this.modulesComboBox.Name = "modulesComboBox";
|
||||||
this.modulesComboBox.Size = new System.Drawing.Size(234, 21);
|
this.modulesComboBox.Size = new System.Drawing.Size(234, 21);
|
||||||
this.modulesComboBox.TabIndex = 0;
|
this.modulesComboBox.TabIndex = 0;
|
||||||
|
this.modulesComboBox.SelectedIndexChanged += new System.EventHandler(this.modulesComboBox_SelectedIndexChanged);
|
||||||
//
|
//
|
||||||
// filterTextBox
|
// filterTextBox
|
||||||
//
|
//
|
||||||
|
@ -48,6 +50,7 @@
|
||||||
this.filterTextBox.Name = "filterTextBox";
|
this.filterTextBox.Name = "filterTextBox";
|
||||||
this.filterTextBox.Size = new System.Drawing.Size(234, 20);
|
this.filterTextBox.Size = new System.Drawing.Size(234, 20);
|
||||||
this.filterTextBox.TabIndex = 1;
|
this.filterTextBox.TabIndex = 1;
|
||||||
|
this.filterTextBox.TextChanged += new System.EventHandler(this.filterTextBox_TextChanged);
|
||||||
//
|
//
|
||||||
// functionsListBox
|
// functionsListBox
|
||||||
//
|
//
|
||||||
|
@ -59,6 +62,7 @@
|
||||||
this.functionsListBox.Name = "functionsListBox";
|
this.functionsListBox.Name = "functionsListBox";
|
||||||
this.functionsListBox.Size = new System.Drawing.Size(234, 524);
|
this.functionsListBox.Size = new System.Drawing.Size(234, 524);
|
||||||
this.functionsListBox.TabIndex = 2;
|
this.functionsListBox.TabIndex = 2;
|
||||||
|
this.functionsListBox.SelectedIndexChanged += new System.EventHandler(this.functionsListBox_SelectedIndexChanged);
|
||||||
//
|
//
|
||||||
// FunctionsPanel
|
// FunctionsPanel
|
||||||
//
|
//
|
||||||
|
@ -68,6 +72,7 @@
|
||||||
this.Controls.Add(this.functionsListBox);
|
this.Controls.Add(this.functionsListBox);
|
||||||
this.Controls.Add(this.filterTextBox);
|
this.Controls.Add(this.filterTextBox);
|
||||||
this.Controls.Add(this.modulesComboBox);
|
this.Controls.Add(this.modulesComboBox);
|
||||||
|
this.Font = new System.Drawing.Font("Microsoft Sans Serif", 8.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
|
||||||
this.Name = "FunctionsPanel";
|
this.Name = "FunctionsPanel";
|
||||||
this.Text = "Functions";
|
this.Text = "Functions";
|
||||||
this.ResumeLayout(false);
|
this.ResumeLayout(false);
|
||||||
|
|
|
@ -18,17 +18,84 @@ namespace Xenia.Debug.UI.Views {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
|
|
||||||
|
RefreshFunctionList();
|
||||||
|
|
||||||
debugger.ModuleList.Changed += UpdateModulesList;
|
debugger.ModuleList.Changed += UpdateModulesList;
|
||||||
UpdateModulesList();
|
UpdateModulesList(debugger.ModuleList);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void UpdateModulesList() {
|
private void UpdateModulesList(ModuleList sender) {
|
||||||
modulesComboBox.BeginUpdate();
|
modulesComboBox.BeginUpdate();
|
||||||
modulesComboBox.Items.Clear();
|
modulesComboBox.Items.Clear();
|
||||||
foreach (Module module in debugger.ModuleList) {
|
foreach (Module module in debugger.ModuleList) {
|
||||||
modulesComboBox.Items.Add("Module A");
|
modulesComboBox.Items.Add(module);
|
||||||
|
module.Changed += Module_Changed;
|
||||||
}
|
}
|
||||||
modulesComboBox.EndUpdate();
|
modulesComboBox.EndUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void Module_Changed(KernelObject sender) {
|
||||||
|
if (modulesComboBox.SelectedItem != sender) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
RefreshFunctionList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void modulesComboBox_SelectedIndexChanged(object sender, EventArgs e) {
|
||||||
|
if (modulesComboBox.SelectedItem == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var module = (Module)modulesComboBox.SelectedItem;
|
||||||
|
|
||||||
|
RefreshFunctionList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RefreshFunctionList() {
|
||||||
|
if (modulesComboBox.SelectedItem == null) {
|
||||||
|
functionsListBox.Items.Clear();
|
||||||
|
functionsListBox.Enabled = false;
|
||||||
|
filterTextBox.Enabled = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
functionsListBox.Enabled = true;
|
||||||
|
filterTextBox.Enabled = true;
|
||||||
|
var module = (Module)modulesComboBox.SelectedItem;
|
||||||
|
|
||||||
|
functionsListBox.BeginUpdate();
|
||||||
|
functionsListBox.Items.Clear();
|
||||||
|
|
||||||
|
foreach (Function function in module) {
|
||||||
|
functionsListBox.Items.Add(function);
|
||||||
|
}
|
||||||
|
|
||||||
|
functionsListBox.EndUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void filterTextBox_TextChanged(object sender, EventArgs e) {
|
||||||
|
var module = (Module)modulesComboBox.SelectedItem;
|
||||||
|
|
||||||
|
var filter = filterTextBox.Text.ToLowerInvariant();
|
||||||
|
|
||||||
|
functionsListBox.BeginUpdate();
|
||||||
|
functionsListBox.Items.Clear();
|
||||||
|
|
||||||
|
foreach (Function function in module) {
|
||||||
|
if (filter.Length == 0 || function.LowerName.Contains(filter)) {
|
||||||
|
functionsListBox.Items.Add(function);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
functionsListBox.EndUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void functionsListBox_SelectedIndexChanged(object sender, EventArgs e) {
|
||||||
|
var function = (Function)functionsListBox.SelectedItem;
|
||||||
|
if (function == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
MainWindow.OpenFunction(function);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,10 +23,15 @@
|
||||||
/// the contents of this method with the code editor.
|
/// the contents of this method with the code editor.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private void InitializeComponent() {
|
private void InitializeComponent() {
|
||||||
|
System.Windows.Forms.ColumnHeader columnHeader1;
|
||||||
|
System.Windows.Forms.ColumnHeader columnHeader2;
|
||||||
|
System.Windows.Forms.ColumnHeader columnHeader3;
|
||||||
|
System.Windows.Forms.ColumnHeader columnHeader4;
|
||||||
this.modulesListView = new System.Windows.Forms.ListView();
|
this.modulesListView = new System.Windows.Forms.ListView();
|
||||||
this.columnHeader1 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
columnHeader1 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
||||||
this.columnHeader2 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
columnHeader2 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
||||||
this.columnHeader3 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
columnHeader3 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
||||||
|
columnHeader4 = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
|
||||||
this.SuspendLayout();
|
this.SuspendLayout();
|
||||||
//
|
//
|
||||||
// modulesListView
|
// modulesListView
|
||||||
|
@ -35,9 +40,10 @@
|
||||||
| System.Windows.Forms.AnchorStyles.Left)
|
| System.Windows.Forms.AnchorStyles.Left)
|
||||||
| System.Windows.Forms.AnchorStyles.Right)));
|
| System.Windows.Forms.AnchorStyles.Right)));
|
||||||
this.modulesListView.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] {
|
this.modulesListView.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] {
|
||||||
this.columnHeader1,
|
columnHeader1,
|
||||||
this.columnHeader2,
|
columnHeader2,
|
||||||
this.columnHeader3});
|
columnHeader3,
|
||||||
|
columnHeader4});
|
||||||
this.modulesListView.Location = new System.Drawing.Point(12, 12);
|
this.modulesListView.Location = new System.Drawing.Point(12, 12);
|
||||||
this.modulesListView.Name = "modulesListView";
|
this.modulesListView.Name = "modulesListView";
|
||||||
this.modulesListView.Size = new System.Drawing.Size(748, 206);
|
this.modulesListView.Size = new System.Drawing.Size(748, 206);
|
||||||
|
@ -45,12 +51,32 @@
|
||||||
this.modulesListView.UseCompatibleStateImageBehavior = false;
|
this.modulesListView.UseCompatibleStateImageBehavior = false;
|
||||||
this.modulesListView.View = System.Windows.Forms.View.Details;
|
this.modulesListView.View = System.Windows.Forms.View.Details;
|
||||||
//
|
//
|
||||||
|
// columnHeader1
|
||||||
|
//
|
||||||
|
columnHeader1.Text = "Handle";
|
||||||
|
//
|
||||||
|
// columnHeader2
|
||||||
|
//
|
||||||
|
columnHeader2.Text = "Type";
|
||||||
|
columnHeader2.Width = 80;
|
||||||
|
//
|
||||||
|
// columnHeader3
|
||||||
|
//
|
||||||
|
columnHeader3.Text = "Name";
|
||||||
|
columnHeader3.Width = 120;
|
||||||
|
//
|
||||||
|
// columnHeader4
|
||||||
|
//
|
||||||
|
columnHeader4.Text = "Path";
|
||||||
|
columnHeader4.Width = 300;
|
||||||
|
//
|
||||||
// ModulesPanel
|
// ModulesPanel
|
||||||
//
|
//
|
||||||
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
|
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
|
||||||
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
|
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
|
||||||
this.ClientSize = new System.Drawing.Size(772, 230);
|
this.ClientSize = new System.Drawing.Size(772, 230);
|
||||||
this.Controls.Add(this.modulesListView);
|
this.Controls.Add(this.modulesListView);
|
||||||
|
this.Font = new System.Drawing.Font("Microsoft Sans Serif", 8.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0)));
|
||||||
this.Name = "ModulesPanel";
|
this.Name = "ModulesPanel";
|
||||||
this.Text = "Modules";
|
this.Text = "Modules";
|
||||||
this.ResumeLayout(false);
|
this.ResumeLayout(false);
|
||||||
|
@ -60,8 +86,5 @@
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
private System.Windows.Forms.ListView modulesListView;
|
private System.Windows.Forms.ListView modulesListView;
|
||||||
private System.Windows.Forms.ColumnHeader columnHeader1;
|
|
||||||
private System.Windows.Forms.ColumnHeader columnHeader2;
|
|
||||||
private System.Windows.Forms.ColumnHeader columnHeader3;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -19,14 +19,21 @@ namespace Xenia.Debug.UI.Views {
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
|
|
||||||
debugger.ModuleList.Changed += UpdateModulesList;
|
debugger.ModuleList.Changed += UpdateModulesList;
|
||||||
UpdateModulesList();
|
UpdateModulesList(debugger.ModuleList);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void UpdateModulesList() {
|
private void UpdateModulesList(ModuleList sender) {
|
||||||
modulesListView.BeginUpdate();
|
modulesListView.BeginUpdate();
|
||||||
modulesListView.Items.Clear();
|
modulesListView.Items.Clear();
|
||||||
foreach (Module module in debugger.ModuleList) {
|
foreach (Module module in debugger.ModuleList) {
|
||||||
modulesListView.Items.Add("Module A");
|
var item = new ListViewItem(new string[]{
|
||||||
|
module.Handle.ToString("X4"),
|
||||||
|
module.ModuleType == xe.debug.proto.ModuleType.Kernel ? "Kernel"
|
||||||
|
: "User",
|
||||||
|
module.Name,
|
||||||
|
module.Path,
|
||||||
|
});
|
||||||
|
modulesListView.Items.Add(item);
|
||||||
}
|
}
|
||||||
modulesListView.EndUpdate();
|
modulesListView.EndUpdate();
|
||||||
}
|
}
|
||||||
|
|
|
@ -117,4 +117,16 @@
|
||||||
<resheader name="writer">
|
<resheader name="writer">
|
||||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||||
</resheader>
|
</resheader>
|
||||||
|
<metadata name="columnHeader1.GenerateMember" type="System.Boolean, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
|
||||||
|
<value>False</value>
|
||||||
|
</metadata>
|
||||||
|
<metadata name="columnHeader2.GenerateMember" type="System.Boolean, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
|
||||||
|
<value>False</value>
|
||||||
|
</metadata>
|
||||||
|
<metadata name="columnHeader3.GenerateMember" type="System.Boolean, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
|
||||||
|
<value>False</value>
|
||||||
|
</metadata>
|
||||||
|
<metadata name="columnHeader4.GenerateMember" type="System.Boolean, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
|
||||||
|
<value>False</value>
|
||||||
|
</metadata>
|
||||||
</root>
|
</root>
|
|
@ -19,10 +19,10 @@ namespace Xenia.Debug.UI.Views {
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
|
|
||||||
debugger.ThreadList.Changed += UpdateThreadList;
|
debugger.ThreadList.Changed += UpdateThreadList;
|
||||||
UpdateThreadList();
|
UpdateThreadList(debugger.ThreadList);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void UpdateThreadList() {
|
private void UpdateThreadList(ThreadList sender) {
|
||||||
threadsListView.BeginUpdate();
|
threadsListView.BeginUpdate();
|
||||||
threadsListView.Items.Clear();
|
threadsListView.Items.Clear();
|
||||||
foreach (Thread thread in debugger.ThreadList) {
|
foreach (Thread thread in debugger.ThreadList) {
|
||||||
|
|
|
@ -6,10 +6,14 @@ using System.Threading.Tasks;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class Breakpoint : Changeable {
|
public class Breakpoint : Changeable<Breakpoint> {
|
||||||
// type code/data/kernel
|
// type code/data/kernel
|
||||||
// address+[end address]
|
// address+[end address]
|
||||||
// conditions? script?
|
// conditions? script?
|
||||||
// action (suspend, trace, etc)
|
// action (suspend, trace, etc)
|
||||||
|
|
||||||
|
public Breakpoint() {
|
||||||
|
this.self = this;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,11 +7,12 @@ using System.Threading.Tasks;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class BreakpointList : Changeable, IReadOnlyCollection<Breakpoint> {
|
public class BreakpointList : Changeable<BreakpointList>, IReadOnlyCollection<Breakpoint> {
|
||||||
private readonly Debugger debugger;
|
private readonly Debugger debugger;
|
||||||
private readonly List<Breakpoint> breakpoints = new List<Breakpoint>();
|
private readonly List<Breakpoint> breakpoints = new List<Breakpoint>();
|
||||||
|
|
||||||
public BreakpointList(Debugger debugger) {
|
public BreakpointList(Debugger debugger) {
|
||||||
|
this.self = this;
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,10 +12,11 @@ namespace Xenia.Debug {
|
||||||
Paused,
|
Paused,
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Context : Changeable {
|
public class Context : Changeable<Context> {
|
||||||
private readonly Debugger debugger;
|
private readonly Debugger debugger;
|
||||||
|
|
||||||
public Context(Debugger debugger) {
|
public Context(Debugger debugger) {
|
||||||
|
this.self = this;
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -193,15 +193,20 @@ namespace Xenia.Debug {
|
||||||
|
|
||||||
// Read body.
|
// Read body.
|
||||||
var bodyBuffer = new byte[length];
|
var bodyBuffer = new byte[length];
|
||||||
|
int bodyOffset = 0;
|
||||||
|
while (bodyOffset != bodyBuffer.Length) {
|
||||||
receiveLength = await Task.Factory.FromAsync(
|
receiveLength = await Task.Factory.FromAsync(
|
||||||
(callback, state) => socket.BeginReceive(bodyBuffer, 0, bodyBuffer.Length, SocketFlags.None, callback, state),
|
(callback, state) => socket.BeginReceive(
|
||||||
asyncResult => socket.EndReceive(asyncResult),
|
bodyBuffer, bodyOffset, bodyBuffer.Length - bodyOffset,
|
||||||
null);
|
SocketFlags.None, callback, state),
|
||||||
if (receiveLength == 0 || receiveLength != bodyBuffer.Length) {
|
asyncResult => socket.EndReceive(asyncResult), null);
|
||||||
|
if (receiveLength == 0) {
|
||||||
// Failed?
|
// Failed?
|
||||||
ReceivePump();
|
ReceivePump();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
bodyOffset += receiveLength;
|
||||||
|
}
|
||||||
|
|
||||||
// Emit message.
|
// Emit message.
|
||||||
OnMessageReceived(bodyBuffer);
|
OnMessageReceived(bodyBuffer);
|
||||||
|
|
|
@ -3,18 +3,21 @@ using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using xe.debug.proto;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class Function : Changeable {
|
public class Function : Changeable<Function> {
|
||||||
// status: declared, defined, failed
|
// status: declared, defined, failed
|
||||||
// module
|
|
||||||
// name
|
|
||||||
// startAddress
|
|
||||||
// endAddress
|
|
||||||
// behavior: default, prolog, epilog, epilog_return, extern
|
// behavior: default, prolog, epilog, epilog_return, extern
|
||||||
// extern info?
|
// extern info?
|
||||||
|
|
||||||
|
public readonly Debugger Debugger;
|
||||||
|
public readonly Module Module;
|
||||||
|
public readonly ulong Identifier;
|
||||||
|
public readonly uint AddressStart;
|
||||||
|
public readonly uint AddressEnd;
|
||||||
|
|
||||||
// source map
|
// source map
|
||||||
|
|
||||||
// disasm:
|
// disasm:
|
||||||
|
@ -29,5 +32,79 @@ namespace Xenia.Debug {
|
||||||
// call count
|
// call count
|
||||||
// caller history
|
// caller history
|
||||||
// instruction execution counts
|
// instruction execution counts
|
||||||
|
|
||||||
|
public string DisasmPpc {
|
||||||
|
get; private set;
|
||||||
|
}
|
||||||
|
public string DisasmHirUnoptimized {
|
||||||
|
get; private set;
|
||||||
|
}
|
||||||
|
public string DisasmHirOptimized {
|
||||||
|
get; private set;
|
||||||
|
}
|
||||||
|
public string DisasmMachineCode {
|
||||||
|
get; private set;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Function(Debugger debugger, Module module, xe.debug.proto.FunctionEntry functionEntry) {
|
||||||
|
this.self = this;
|
||||||
|
this.Debugger = debugger;
|
||||||
|
this.Module = module;
|
||||||
|
Identifier = functionEntry.Identifier;
|
||||||
|
AddressStart = functionEntry.AddressStart;
|
||||||
|
AddressEnd = functionEntry.AddressEnd;
|
||||||
|
Name = functionEntry.Name;
|
||||||
|
}
|
||||||
|
|
||||||
|
private string name;
|
||||||
|
public string Name {
|
||||||
|
get {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
set {
|
||||||
|
name = value;
|
||||||
|
if (value == null) {
|
||||||
|
name = "sub_" + AddressStart.ToString("X8");
|
||||||
|
}
|
||||||
|
LowerName = name.ToLowerInvariant();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public string LowerName {
|
||||||
|
get; set;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString() {
|
||||||
|
return Name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task Invalidate() {
|
||||||
|
var fbb = Debugger.BeginRequest();
|
||||||
|
int requestDataOffset = GetFunctionRequest.CreateGetFunctionRequest(fbb, Identifier);
|
||||||
|
var response = await Debugger.CommitRequest(
|
||||||
|
fbb, RequestData.GetFunctionRequest, requestDataOffset);
|
||||||
|
|
||||||
|
var responseData = new GetFunctionResponse();
|
||||||
|
response.GetResponseData(responseData);
|
||||||
|
var functionData = responseData.Function;
|
||||||
|
|
||||||
|
this.DisasmPpc = functionData.DisasmPpc;
|
||||||
|
this.DisasmHirUnoptimized = functionData.DisasmHirRaw;
|
||||||
|
this.DisasmHirOptimized = functionData.DisasmHirOpt;
|
||||||
|
this.DisasmMachineCode = functionData.DisasmMachineCode;
|
||||||
|
if (DisasmPpc != null) {
|
||||||
|
DisasmPpc = DisasmPpc.Replace("\n", "\r\n");
|
||||||
|
}
|
||||||
|
if (DisasmHirUnoptimized != null) {
|
||||||
|
DisasmHirUnoptimized = DisasmHirUnoptimized.Replace("\n", "\r\n");
|
||||||
|
}
|
||||||
|
if (DisasmHirOptimized != null) {
|
||||||
|
DisasmHirOptimized = DisasmHirOptimized.Replace("\n", "\r\n");
|
||||||
|
}
|
||||||
|
if (DisasmMachineCode != null) {
|
||||||
|
DisasmMachineCode = DisasmMachineCode.Replace("\n", "\r\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
OnChanged();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,11 +6,12 @@ using System.Threading.Tasks;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class KernelObject : Changeable {
|
public class KernelObject : Changeable<KernelObject> {
|
||||||
public readonly Debugger Debugger;
|
public readonly Debugger Debugger;
|
||||||
public readonly uint Handle;
|
public readonly uint Handle;
|
||||||
|
|
||||||
public KernelObject(Debugger debugger, uint handle) {
|
public KernelObject(Debugger debugger, uint handle) {
|
||||||
|
this.self = this;
|
||||||
this.Debugger = debugger;
|
this.Debugger = debugger;
|
||||||
this.Handle = handle;
|
this.Handle = handle;
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ using System.Threading.Tasks;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class Memory : Changeable, IDisposable {
|
public class Memory : Changeable<Memory>, IDisposable {
|
||||||
private readonly Debugger debugger;
|
private readonly Debugger debugger;
|
||||||
|
|
||||||
private class MapInfo {
|
private class MapInfo {
|
||||||
|
@ -41,6 +41,7 @@ namespace Xenia.Debug {
|
||||||
public UIntPtr PhysicalMembase;
|
public UIntPtr PhysicalMembase;
|
||||||
|
|
||||||
public Memory(Debugger debugger) {
|
public Memory(Debugger debugger) {
|
||||||
|
this.self = this;
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,10 +6,11 @@ using System.Threading.Tasks;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class MemoryView : Changeable {
|
public class MemoryView : Changeable<MemoryView> {
|
||||||
private readonly Memory memory;
|
private readonly Memory memory;
|
||||||
|
|
||||||
public MemoryView(Memory memory) {
|
public MemoryView(Memory memory) {
|
||||||
|
this.self = this;
|
||||||
this.memory = memory;
|
this.memory = memory;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,26 +9,95 @@ using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class Module : KernelObject, IReadOnlyCollection<Function> {
|
public class Module : KernelObject, IReadOnlyCollection<Function> {
|
||||||
|
private bool hasFetched = false;
|
||||||
|
private uint functionCount = 0;
|
||||||
private readonly List<Function> functions = new List<Function>();
|
private readonly List<Function> functions = new List<Function>();
|
||||||
|
|
||||||
// xobject: handle
|
|
||||||
// path
|
|
||||||
// type: user, kernel
|
|
||||||
// if user:
|
|
||||||
// xex header?
|
|
||||||
|
|
||||||
public Module(Debugger debugger, uint moduleHandle) : base(debugger, moduleHandle) {
|
public Module(Debugger debugger, uint moduleHandle) : base(debugger, moduleHandle) {
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task Invalidate() {
|
public async Task Invalidate(uint newFunctionCount) {
|
||||||
|
if (hasFetched && newFunctionCount == functionCount) {
|
||||||
|
// No-op.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var pendingTasks = new List<Task>();
|
||||||
|
|
||||||
|
if (!hasFetched) {
|
||||||
|
pendingTasks.Add(InvalidateModule());
|
||||||
|
hasFetched = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newFunctionCount != functionCount) {
|
||||||
|
uint functionIndexStart = functionCount;
|
||||||
|
uint functionIndexEnd = newFunctionCount - 1;
|
||||||
|
functionCount = newFunctionCount;
|
||||||
|
pendingTasks.Add(InvalidateFunctions(functionIndexStart, functionIndexEnd));
|
||||||
|
}
|
||||||
|
|
||||||
|
await Task.WhenAll(pendingTasks);
|
||||||
|
|
||||||
|
OnChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task InvalidateModule() {
|
||||||
var fbb = Debugger.BeginRequest();
|
var fbb = Debugger.BeginRequest();
|
||||||
int requestDataOffset = GetModuleRequest.CreateGetModuleRequest(fbb, Handle);
|
int requestDataOffset = GetModuleRequest.CreateGetModuleRequest(fbb, Handle);
|
||||||
var response = await Debugger.CommitRequest(
|
var response = await Debugger.CommitRequest(
|
||||||
fbb, RequestData.GetModuleRequest, requestDataOffset);
|
fbb, RequestData.GetModuleRequest, requestDataOffset);
|
||||||
GetModuleResponse responseData = new GetModuleResponse();
|
|
||||||
|
var responseData = new GetModuleResponse();
|
||||||
response.GetResponseData(responseData);
|
response.GetResponseData(responseData);
|
||||||
|
|
||||||
//
|
ModuleType = responseData.Module.Type;
|
||||||
|
Name = responseData.Module.Name;
|
||||||
|
Path = responseData.Module.Path;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task InvalidateFunctions(uint functionIndexStart, uint functionIndexEnd) {
|
||||||
|
var fbb = Debugger.BeginRequest();
|
||||||
|
int requestDataOffset = ListFunctionsRequest.CreateListFunctionsRequest(
|
||||||
|
fbb, Handle, functionIndexStart, functionIndexEnd);
|
||||||
|
var response = await Debugger.CommitRequest(
|
||||||
|
fbb, RequestData.ListFunctionsRequest, requestDataOffset);
|
||||||
|
|
||||||
|
var responseData = new ListFunctionsResponse();
|
||||||
|
response.GetResponseData(responseData);
|
||||||
|
|
||||||
|
var functionEntry = new xe.debug.proto.FunctionEntry();
|
||||||
|
for (int i = 0; i < responseData.EntryLength; ++i) {
|
||||||
|
responseData.GetEntry(functionEntry, i);
|
||||||
|
var function = new Function(Debugger, this, functionEntry);
|
||||||
|
functions.Add(function);
|
||||||
|
}
|
||||||
|
|
||||||
|
functions.Sort((Function a, Function b) => {
|
||||||
|
return (int)((long)a.AddressStart - (long)b.AddressStart);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public ModuleType ModuleType {
|
||||||
|
get;
|
||||||
|
private set;
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name {
|
||||||
|
get;
|
||||||
|
private set;
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Path {
|
||||||
|
get;
|
||||||
|
private set;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString() {
|
||||||
|
return ToShortString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public string ToShortString() {
|
||||||
|
return string.Format("[{0:X4}] {1}", Handle, Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Count {
|
public int Count {
|
||||||
|
|
|
@ -8,11 +8,12 @@ using xe.debug.proto;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class ModuleList : Changeable, IReadOnlyCollection<Module> {
|
public class ModuleList : Changeable<ModuleList>, IReadOnlyCollection<Module> {
|
||||||
private readonly Debugger debugger;
|
private readonly Debugger debugger;
|
||||||
private readonly List<Module> modules = new List<Module>();
|
private readonly List<Module> modules = new List<Module>();
|
||||||
|
|
||||||
public ModuleList(Debugger debugger) {
|
public ModuleList(Debugger debugger) {
|
||||||
|
this.self = this;
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,20 +27,23 @@ namespace Xenia.Debug {
|
||||||
response.GetResponseData(responseData);
|
response.GetResponseData(responseData);
|
||||||
|
|
||||||
var pendingTasks = new List<Task>();
|
var pendingTasks = new List<Task>();
|
||||||
for (int i = 0; i < responseData.ModuleIdsLength; ++i) {
|
for (int i = 0; i < responseData.EntryLength; ++i) {
|
||||||
uint moduleHandle = responseData.GetModuleIds(i);
|
var moduleEntry = responseData.GetEntry(i);
|
||||||
var module = modules.Find((m) => m.Handle == moduleHandle);
|
var module = modules.Find((m) => m.Handle == moduleEntry.Handle);
|
||||||
if (module == null) {
|
if (module == null) {
|
||||||
// Module not found.
|
// Module not found.
|
||||||
module = new Module(debugger, moduleHandle);
|
module = new Module(debugger, moduleEntry.Handle);
|
||||||
pendingTasks.Add(module.Invalidate());
|
pendingTasks.Add(module.Invalidate(moduleEntry.FunctionCount));
|
||||||
|
modules.Add(module);
|
||||||
} else {
|
} else {
|
||||||
// Module already present.
|
// Module already present.
|
||||||
// Modules are immutable, so ignore?
|
pendingTasks.Add(module.Invalidate(moduleEntry.FunctionCount));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await Task.WhenAll(pendingTasks);
|
await Task.WhenAll(pendingTasks);
|
||||||
|
|
||||||
|
OnChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Count {
|
public int Count {
|
||||||
|
|
|
@ -0,0 +1,59 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class Function : Table {
|
||||||
|
public static Function GetRootAsFunction(ByteBuffer _bb) { return GetRootAsFunction(_bb, new Function()); }
|
||||||
|
public static Function GetRootAsFunction(ByteBuffer _bb, Function obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
|
public Function __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public ulong Identifier { get { int o = __offset(4); return o != 0 ? bb.GetUlong(o + bb_pos) : (ulong)0; } }
|
||||||
|
public uint AddressStart { get { int o = __offset(6); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
public uint AddressEnd { get { int o = __offset(8); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
public string Name { get { int o = __offset(10); return o != 0 ? __string(o + bb_pos) : null; } }
|
||||||
|
public string DisasmPpc { get { int o = __offset(12); return o != 0 ? __string(o + bb_pos) : null; } }
|
||||||
|
public string DisasmHirRaw { get { int o = __offset(14); return o != 0 ? __string(o + bb_pos) : null; } }
|
||||||
|
public string DisasmHirOpt { get { int o = __offset(16); return o != 0 ? __string(o + bb_pos) : null; } }
|
||||||
|
public string DisasmMachineCode { get { int o = __offset(18); return o != 0 ? __string(o + bb_pos) : null; } }
|
||||||
|
|
||||||
|
public static int CreateFunction(FlatBufferBuilder builder,
|
||||||
|
ulong identifier = 0,
|
||||||
|
uint address_start = 0,
|
||||||
|
uint address_end = 0,
|
||||||
|
int name = 0,
|
||||||
|
int disasm_ppc = 0,
|
||||||
|
int disasm_hir_raw = 0,
|
||||||
|
int disasm_hir_opt = 0,
|
||||||
|
int disasm_machine_code = 0) {
|
||||||
|
builder.StartObject(8);
|
||||||
|
Function.AddIdentifier(builder, identifier);
|
||||||
|
Function.AddDisasmMachineCode(builder, disasm_machine_code);
|
||||||
|
Function.AddDisasmHirOpt(builder, disasm_hir_opt);
|
||||||
|
Function.AddDisasmHirRaw(builder, disasm_hir_raw);
|
||||||
|
Function.AddDisasmPpc(builder, disasm_ppc);
|
||||||
|
Function.AddName(builder, name);
|
||||||
|
Function.AddAddressEnd(builder, address_end);
|
||||||
|
Function.AddAddressStart(builder, address_start);
|
||||||
|
return Function.EndFunction(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void StartFunction(FlatBufferBuilder builder) { builder.StartObject(8); }
|
||||||
|
public static void AddIdentifier(FlatBufferBuilder builder, ulong identifier) { builder.AddUlong(0, identifier, 0); }
|
||||||
|
public static void AddAddressStart(FlatBufferBuilder builder, uint addressStart) { builder.AddUint(1, addressStart, 0); }
|
||||||
|
public static void AddAddressEnd(FlatBufferBuilder builder, uint addressEnd) { builder.AddUint(2, addressEnd, 0); }
|
||||||
|
public static void AddName(FlatBufferBuilder builder, int nameOffset) { builder.AddOffset(3, nameOffset, 0); }
|
||||||
|
public static void AddDisasmPpc(FlatBufferBuilder builder, int disasmPpcOffset) { builder.AddOffset(4, disasmPpcOffset, 0); }
|
||||||
|
public static void AddDisasmHirRaw(FlatBufferBuilder builder, int disasmHirRawOffset) { builder.AddOffset(5, disasmHirRawOffset, 0); }
|
||||||
|
public static void AddDisasmHirOpt(FlatBufferBuilder builder, int disasmHirOptOffset) { builder.AddOffset(6, disasmHirOptOffset, 0); }
|
||||||
|
public static void AddDisasmMachineCode(FlatBufferBuilder builder, int disasmMachineCodeOffset) { builder.AddOffset(7, disasmMachineCodeOffset, 0); }
|
||||||
|
public static int EndFunction(FlatBufferBuilder builder) {
|
||||||
|
int o = builder.EndObject();
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,43 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class FunctionEntry : Table {
|
||||||
|
public static FunctionEntry GetRootAsFunctionEntry(ByteBuffer _bb) { return GetRootAsFunctionEntry(_bb, new FunctionEntry()); }
|
||||||
|
public static FunctionEntry GetRootAsFunctionEntry(ByteBuffer _bb, FunctionEntry obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
|
public FunctionEntry __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public ulong Identifier { get { int o = __offset(4); return o != 0 ? bb.GetUlong(o + bb_pos) : (ulong)0; } }
|
||||||
|
public uint AddressStart { get { int o = __offset(6); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
public uint AddressEnd { get { int o = __offset(8); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
public string Name { get { int o = __offset(10); return o != 0 ? __string(o + bb_pos) : null; } }
|
||||||
|
|
||||||
|
public static int CreateFunctionEntry(FlatBufferBuilder builder,
|
||||||
|
ulong identifier = 0,
|
||||||
|
uint address_start = 0,
|
||||||
|
uint address_end = 0,
|
||||||
|
int name = 0) {
|
||||||
|
builder.StartObject(4);
|
||||||
|
FunctionEntry.AddIdentifier(builder, identifier);
|
||||||
|
FunctionEntry.AddName(builder, name);
|
||||||
|
FunctionEntry.AddAddressEnd(builder, address_end);
|
||||||
|
FunctionEntry.AddAddressStart(builder, address_start);
|
||||||
|
return FunctionEntry.EndFunctionEntry(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void StartFunctionEntry(FlatBufferBuilder builder) { builder.StartObject(4); }
|
||||||
|
public static void AddIdentifier(FlatBufferBuilder builder, ulong identifier) { builder.AddUlong(0, identifier, 0); }
|
||||||
|
public static void AddAddressStart(FlatBufferBuilder builder, uint addressStart) { builder.AddUint(1, addressStart, 0); }
|
||||||
|
public static void AddAddressEnd(FlatBufferBuilder builder, uint addressEnd) { builder.AddUint(2, addressEnd, 0); }
|
||||||
|
public static void AddName(FlatBufferBuilder builder, int nameOffset) { builder.AddOffset(3, nameOffset, 0); }
|
||||||
|
public static int EndFunctionEntry(FlatBufferBuilder builder) {
|
||||||
|
int o = builder.EndObject();
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class GetFunctionRequest : Table {
|
||||||
|
public static GetFunctionRequest GetRootAsGetFunctionRequest(ByteBuffer _bb) { return GetRootAsGetFunctionRequest(_bb, new GetFunctionRequest()); }
|
||||||
|
public static GetFunctionRequest GetRootAsGetFunctionRequest(ByteBuffer _bb, GetFunctionRequest obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
|
public GetFunctionRequest __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public ulong Identifier { get { int o = __offset(4); return o != 0 ? bb.GetUlong(o + bb_pos) : (ulong)0; } }
|
||||||
|
|
||||||
|
public static int CreateGetFunctionRequest(FlatBufferBuilder builder,
|
||||||
|
ulong identifier = 0) {
|
||||||
|
builder.StartObject(1);
|
||||||
|
GetFunctionRequest.AddIdentifier(builder, identifier);
|
||||||
|
return GetFunctionRequest.EndGetFunctionRequest(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void StartGetFunctionRequest(FlatBufferBuilder builder) { builder.StartObject(1); }
|
||||||
|
public static void AddIdentifier(FlatBufferBuilder builder, ulong identifier) { builder.AddUlong(0, identifier, 0); }
|
||||||
|
public static int EndGetFunctionRequest(FlatBufferBuilder builder) {
|
||||||
|
int o = builder.EndObject();
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,32 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class GetFunctionResponse : Table {
|
||||||
|
public static GetFunctionResponse GetRootAsGetFunctionResponse(ByteBuffer _bb) { return GetRootAsGetFunctionResponse(_bb, new GetFunctionResponse()); }
|
||||||
|
public static GetFunctionResponse GetRootAsGetFunctionResponse(ByteBuffer _bb, GetFunctionResponse obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
|
public GetFunctionResponse __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public Function Function { get { return GetFunction(new Function()); } }
|
||||||
|
public Function GetFunction(Function obj) { int o = __offset(4); return o != 0 ? obj.__init(__indirect(o + bb_pos), bb) : null; }
|
||||||
|
|
||||||
|
public static int CreateGetFunctionResponse(FlatBufferBuilder builder,
|
||||||
|
int function = 0) {
|
||||||
|
builder.StartObject(1);
|
||||||
|
GetFunctionResponse.AddFunction(builder, function);
|
||||||
|
return GetFunctionResponse.EndGetFunctionResponse(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void StartGetFunctionResponse(FlatBufferBuilder builder) { builder.StartObject(1); }
|
||||||
|
public static void AddFunction(FlatBufferBuilder builder, int functionOffset) { builder.AddOffset(0, functionOffset, 0); }
|
||||||
|
public static int EndGetFunctionResponse(FlatBufferBuilder builder) {
|
||||||
|
int o = builder.EndObject();
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,39 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class ListFunctionsRequest : Table {
|
||||||
|
public static ListFunctionsRequest GetRootAsListFunctionsRequest(ByteBuffer _bb) { return GetRootAsListFunctionsRequest(_bb, new ListFunctionsRequest()); }
|
||||||
|
public static ListFunctionsRequest GetRootAsListFunctionsRequest(ByteBuffer _bb, ListFunctionsRequest obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
|
public ListFunctionsRequest __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public uint ModuleId { get { int o = __offset(4); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
public uint FunctionIndexStart { get { int o = __offset(6); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
public uint FunctionIndexEnd { get { int o = __offset(8); return o != 0 ? bb.GetUint(o + bb_pos) : (uint)0; } }
|
||||||
|
|
||||||
|
public static int CreateListFunctionsRequest(FlatBufferBuilder builder,
|
||||||
|
uint module_id = 0,
|
||||||
|
uint function_index_start = 0,
|
||||||
|
uint function_index_end = 0) {
|
||||||
|
builder.StartObject(3);
|
||||||
|
ListFunctionsRequest.AddFunctionIndexEnd(builder, function_index_end);
|
||||||
|
ListFunctionsRequest.AddFunctionIndexStart(builder, function_index_start);
|
||||||
|
ListFunctionsRequest.AddModuleId(builder, module_id);
|
||||||
|
return ListFunctionsRequest.EndListFunctionsRequest(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void StartListFunctionsRequest(FlatBufferBuilder builder) { builder.StartObject(3); }
|
||||||
|
public static void AddModuleId(FlatBufferBuilder builder, uint moduleId) { builder.AddUint(0, moduleId, 0); }
|
||||||
|
public static void AddFunctionIndexStart(FlatBufferBuilder builder, uint functionIndexStart) { builder.AddUint(1, functionIndexStart, 0); }
|
||||||
|
public static void AddFunctionIndexEnd(FlatBufferBuilder builder, uint functionIndexEnd) { builder.AddUint(2, functionIndexEnd, 0); }
|
||||||
|
public static int EndListFunctionsRequest(FlatBufferBuilder builder) {
|
||||||
|
int o = builder.EndObject();
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,35 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class ListFunctionsResponse : Table {
|
||||||
|
public static ListFunctionsResponse GetRootAsListFunctionsResponse(ByteBuffer _bb) { return GetRootAsListFunctionsResponse(_bb, new ListFunctionsResponse()); }
|
||||||
|
public static ListFunctionsResponse GetRootAsListFunctionsResponse(ByteBuffer _bb, ListFunctionsResponse obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
|
public ListFunctionsResponse __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public FunctionEntry GetEntry(int j) { return GetEntry(new FunctionEntry(), j); }
|
||||||
|
public FunctionEntry GetEntry(FunctionEntry obj, int j) { int o = __offset(4); return o != 0 ? obj.__init(__indirect(__vector(o) + j * 4), bb) : null; }
|
||||||
|
public int EntryLength { get { int o = __offset(4); return o != 0 ? __vector_len(o) : 0; } }
|
||||||
|
|
||||||
|
public static int CreateListFunctionsResponse(FlatBufferBuilder builder,
|
||||||
|
int entry = 0) {
|
||||||
|
builder.StartObject(1);
|
||||||
|
ListFunctionsResponse.AddEntry(builder, entry);
|
||||||
|
return ListFunctionsResponse.EndListFunctionsResponse(builder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void StartListFunctionsResponse(FlatBufferBuilder builder) { builder.StartObject(1); }
|
||||||
|
public static void AddEntry(FlatBufferBuilder builder, int entryOffset) { builder.AddOffset(0, entryOffset, 0); }
|
||||||
|
public static int CreateEntryVector(FlatBufferBuilder builder, int[] data) { builder.StartVector(4, data.Length, 4); for (int i = data.Length - 1; i >= 0; i--) builder.AddOffset(data[i]); return builder.EndVector(); }
|
||||||
|
public static void StartEntryVector(FlatBufferBuilder builder, int numElems) { builder.StartVector(4, numElems, 4); }
|
||||||
|
public static int EndListFunctionsResponse(FlatBufferBuilder builder) {
|
||||||
|
int o = builder.EndObject();
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,23 @@
|
||||||
|
// automatically generated, do not modify
|
||||||
|
|
||||||
|
namespace xe.debug.proto
|
||||||
|
{
|
||||||
|
|
||||||
|
using FlatBuffers;
|
||||||
|
|
||||||
|
public sealed class ListModuleEntry : Struct {
|
||||||
|
public ListModuleEntry __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
|
public uint Handle { get { return bb.GetUint(bb_pos + 0); } }
|
||||||
|
public uint FunctionCount { get { return bb.GetUint(bb_pos + 4); } }
|
||||||
|
|
||||||
|
public static int CreateListModuleEntry(FlatBufferBuilder builder, uint Handle, uint FunctionCount) {
|
||||||
|
builder.Prep(4, 8);
|
||||||
|
builder.PutUint(FunctionCount);
|
||||||
|
builder.PutUint(Handle);
|
||||||
|
return builder.Offset;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -10,20 +10,20 @@ public sealed class ListModulesResponse : Table {
|
||||||
public static ListModulesResponse GetRootAsListModulesResponse(ByteBuffer _bb, ListModulesResponse obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
public static ListModulesResponse GetRootAsListModulesResponse(ByteBuffer _bb, ListModulesResponse obj) { return (obj.__init(_bb.GetInt(_bb.Position) + _bb.Position, _bb)); }
|
||||||
public ListModulesResponse __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
public ListModulesResponse __init(int _i, ByteBuffer _bb) { bb_pos = _i; bb = _bb; return this; }
|
||||||
|
|
||||||
public uint GetModuleIds(int j) { int o = __offset(4); return o != 0 ? bb.GetUint(__vector(o) + j * 4) : (uint)0; }
|
public ListModuleEntry GetEntry(int j) { return GetEntry(new ListModuleEntry(), j); }
|
||||||
public int ModuleIdsLength { get { int o = __offset(4); return o != 0 ? __vector_len(o) : 0; } }
|
public ListModuleEntry GetEntry(ListModuleEntry obj, int j) { int o = __offset(4); return o != 0 ? obj.__init(__vector(o) + j * 8, bb) : null; }
|
||||||
|
public int EntryLength { get { int o = __offset(4); return o != 0 ? __vector_len(o) : 0; } }
|
||||||
|
|
||||||
public static int CreateListModulesResponse(FlatBufferBuilder builder,
|
public static int CreateListModulesResponse(FlatBufferBuilder builder,
|
||||||
int module_ids = 0) {
|
int entry = 0) {
|
||||||
builder.StartObject(1);
|
builder.StartObject(1);
|
||||||
ListModulesResponse.AddModuleIds(builder, module_ids);
|
ListModulesResponse.AddEntry(builder, entry);
|
||||||
return ListModulesResponse.EndListModulesResponse(builder);
|
return ListModulesResponse.EndListModulesResponse(builder);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void StartListModulesResponse(FlatBufferBuilder builder) { builder.StartObject(1); }
|
public static void StartListModulesResponse(FlatBufferBuilder builder) { builder.StartObject(1); }
|
||||||
public static void AddModuleIds(FlatBufferBuilder builder, int moduleIdsOffset) { builder.AddOffset(0, moduleIdsOffset, 0); }
|
public static void AddEntry(FlatBufferBuilder builder, int entryOffset) { builder.AddOffset(0, entryOffset, 0); }
|
||||||
public static int CreateModuleIdsVector(FlatBufferBuilder builder, uint[] data) { builder.StartVector(4, data.Length, 4); for (int i = data.Length - 1; i >= 0; i--) builder.AddUint(data[i]); return builder.EndVector(); }
|
public static void StartEntryVector(FlatBufferBuilder builder, int numElems) { builder.StartVector(8, numElems, 4); }
|
||||||
public static void StartModuleIdsVector(FlatBufferBuilder builder, int numElems) { builder.StartVector(4, numElems, 4); }
|
|
||||||
public static int EndListModulesResponse(FlatBufferBuilder builder) {
|
public static int EndListModulesResponse(FlatBufferBuilder builder) {
|
||||||
int o = builder.EndObject();
|
int o = builder.EndObject();
|
||||||
return o;
|
return o;
|
||||||
|
|
|
@ -13,10 +13,12 @@ public enum RequestData : byte
|
||||||
RemoveBreakpointsRequest = 5,
|
RemoveBreakpointsRequest = 5,
|
||||||
ListModulesRequest = 6,
|
ListModulesRequest = 6,
|
||||||
GetModuleRequest = 7,
|
GetModuleRequest = 7,
|
||||||
StopRequest = 8,
|
ListFunctionsRequest = 8,
|
||||||
BreakRequest = 9,
|
GetFunctionRequest = 9,
|
||||||
ContinueRequest = 10,
|
StopRequest = 10,
|
||||||
StepRequest = 11,
|
BreakRequest = 11,
|
||||||
|
ContinueRequest = 12,
|
||||||
|
StepRequest = 13,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -13,12 +13,14 @@ public enum ResponseData : byte
|
||||||
RemoveBreakpointsResponse = 5,
|
RemoveBreakpointsResponse = 5,
|
||||||
ListModulesResponse = 6,
|
ListModulesResponse = 6,
|
||||||
GetModuleResponse = 7,
|
GetModuleResponse = 7,
|
||||||
StopResponse = 8,
|
ListFunctionsResponse = 8,
|
||||||
BreakResponse = 9,
|
GetFunctionResponse = 9,
|
||||||
ContinueResponse = 10,
|
StopResponse = 10,
|
||||||
StepResponse = 11,
|
BreakResponse = 11,
|
||||||
BreakpointEvent = 12,
|
ContinueResponse = 12,
|
||||||
AccessViolationEvent = 13,
|
StepResponse = 13,
|
||||||
|
BreakpointEvent = 14,
|
||||||
|
AccessViolationEvent = 15,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -7,11 +7,12 @@ using System.Threading.Tasks;
|
||||||
using Xenia.Debug.Utilities;
|
using Xenia.Debug.Utilities;
|
||||||
|
|
||||||
namespace Xenia.Debug {
|
namespace Xenia.Debug {
|
||||||
public class ThreadList : Changeable, IReadOnlyCollection<Thread> {
|
public class ThreadList : Changeable<ThreadList>, IReadOnlyCollection<Thread> {
|
||||||
private readonly Debugger debugger;
|
private readonly Debugger debugger;
|
||||||
private readonly List<Thread> threads = new List<Thread>();
|
private readonly List<Thread> threads = new List<Thread>();
|
||||||
|
|
||||||
public ThreadList(Debugger debugger) {
|
public ThreadList(Debugger debugger) {
|
||||||
|
this.self = this;
|
||||||
this.debugger = debugger;
|
this.debugger = debugger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,11 +5,13 @@ using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
namespace Xenia.Debug.Utilities {
|
namespace Xenia.Debug.Utilities {
|
||||||
public delegate void ChangedEventHandler();
|
public delegate void ChangedEventHandler<T>(T sender);
|
||||||
|
|
||||||
public class Changeable {
|
public class Changeable<T> {
|
||||||
|
protected T self;
|
||||||
private int changeDepth;
|
private int changeDepth;
|
||||||
public event ChangedEventHandler Changed;
|
|
||||||
|
public event ChangedEventHandler<T> Changed;
|
||||||
|
|
||||||
protected void BeginChanging() {
|
protected void BeginChanging() {
|
||||||
++changeDepth;
|
++changeDepth;
|
||||||
|
@ -24,7 +26,7 @@ namespace Xenia.Debug.Utilities {
|
||||||
protected void OnChanged() {
|
protected void OnChanged() {
|
||||||
System.Diagnostics.Debug.Assert(changeDepth == 0);
|
System.Diagnostics.Debug.Assert(changeDepth == 0);
|
||||||
if (Changed != null) {
|
if (Changed != null) {
|
||||||
Changed();
|
Changed(self);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -84,10 +84,17 @@
|
||||||
<Compile Include="Proto\xe\debug\proto\ContinueAction.cs" />
|
<Compile Include="Proto\xe\debug\proto\ContinueAction.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ContinueRequest.cs" />
|
<Compile Include="Proto\xe\debug\proto\ContinueRequest.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ContinueResponse.cs" />
|
<Compile Include="Proto\xe\debug\proto\ContinueResponse.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\Function.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\FunctionEntry.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\GetFunctionRequest.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\GetFunctionResponse.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\GetModuleRequest.cs" />
|
<Compile Include="Proto\xe\debug\proto\GetModuleRequest.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\GetModuleResponse.cs" />
|
<Compile Include="Proto\xe\debug\proto\GetModuleResponse.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ListBreakpointsRequest.cs" />
|
<Compile Include="Proto\xe\debug\proto\ListBreakpointsRequest.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ListBreakpointsResponse.cs" />
|
<Compile Include="Proto\xe\debug\proto\ListBreakpointsResponse.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\ListFunctionsRequest.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\ListFunctionsResponse.cs" />
|
||||||
|
<Compile Include="Proto\xe\debug\proto\ListModuleEntry.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ListModulesRequest.cs" />
|
<Compile Include="Proto\xe\debug\proto\ListModulesRequest.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ListModulesResponse.cs" />
|
<Compile Include="Proto\xe\debug\proto\ListModulesResponse.cs" />
|
||||||
<Compile Include="Proto\xe\debug\proto\ListThreadsRequest.cs" />
|
<Compile Include="Proto\xe\debug\proto\ListThreadsRequest.cs" />
|
||||||
|
|
|
@ -201,7 +201,7 @@ void AudioSystem::WorkerThreadMain() {
|
||||||
void AudioSystem::DecoderThreadMain() {
|
void AudioSystem::DecoderThreadMain() {
|
||||||
while (decoder_running_) {
|
while (decoder_running_) {
|
||||||
// Wait for a kick from WriteRegister.
|
// Wait for a kick from WriteRegister.
|
||||||
decoder_fence_.Wait();
|
//decoder_fence_.Wait();
|
||||||
|
|
||||||
// Check to see if we're supposed to exit
|
// Check to see if we're supposed to exit
|
||||||
if (!decoder_running_) {
|
if (!decoder_running_) {
|
||||||
|
|
|
@ -149,7 +149,6 @@ SymbolStatus Module::DefineVariable(VariableInfo* symbol_info) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void Module::ForEachFunction(std::function<void(FunctionInfo*)> callback) {
|
void Module::ForEachFunction(std::function<void(FunctionInfo*)> callback) {
|
||||||
SCOPE_profile_cpu_f("cpu");
|
|
||||||
std::lock_guard<xe::mutex> guard(lock_);
|
std::lock_guard<xe::mutex> guard(lock_);
|
||||||
for (auto& symbol_info : list_) {
|
for (auto& symbol_info : list_) {
|
||||||
if (symbol_info->type() == SymbolType::kFunction) {
|
if (symbol_info->type() == SymbolType::kFunction) {
|
||||||
|
@ -159,19 +158,20 @@ void Module::ForEachFunction(std::function<void(FunctionInfo*)> callback) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Module::ForEachFunction(size_t since, size_t& version,
|
void Module::ForEachSymbol(size_t start_index, size_t end_index,
|
||||||
std::function<void(FunctionInfo*)> callback) {
|
std::function<void(SymbolInfo*)> callback) {
|
||||||
SCOPE_profile_cpu_f("cpu");
|
|
||||||
std::lock_guard<xe::mutex> guard(lock_);
|
std::lock_guard<xe::mutex> guard(lock_);
|
||||||
size_t count = list_.size();
|
start_index = std::min(start_index, list_.size());
|
||||||
version = count;
|
end_index = std::min(end_index, list_.size());
|
||||||
for (size_t n = since; n < count; n++) {
|
for (size_t i = start_index; i <= end_index; ++i) {
|
||||||
auto& symbol_info = list_[n];
|
auto& symbol_info = list_[i];
|
||||||
if (symbol_info->type() == SymbolType::kFunction) {
|
callback(symbol_info.get());
|
||||||
FunctionInfo* info = static_cast<FunctionInfo*>(symbol_info.get());
|
|
||||||
callback(info);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t Module::QuerySymbolCount() {
|
||||||
|
std::lock_guard<xe::mutex> guard(lock_);
|
||||||
|
return list_.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Module::ReadMap(const char* file_name) {
|
bool Module::ReadMap(const char* file_name) {
|
||||||
|
|
|
@ -47,8 +47,9 @@ class Module {
|
||||||
SymbolStatus DefineVariable(VariableInfo* symbol_info);
|
SymbolStatus DefineVariable(VariableInfo* symbol_info);
|
||||||
|
|
||||||
void ForEachFunction(std::function<void(FunctionInfo*)> callback);
|
void ForEachFunction(std::function<void(FunctionInfo*)> callback);
|
||||||
void ForEachFunction(size_t since, size_t& version,
|
void ForEachSymbol(size_t start_index, size_t end_index,
|
||||||
std::function<void(FunctionInfo*)> callback);
|
std::function<void(SymbolInfo*)> callback);
|
||||||
|
size_t QuerySymbolCount();
|
||||||
|
|
||||||
bool ReadMap(const char* file_name);
|
bool ReadMap(const char* file_name);
|
||||||
|
|
||||||
|
|
|
@ -22,6 +22,9 @@
|
||||||
#include "xenia/cpu/function.h"
|
#include "xenia/cpu/function.h"
|
||||||
#include "xenia/cpu/processor.h"
|
#include "xenia/cpu/processor.h"
|
||||||
#include "xenia/emulator.h"
|
#include "xenia/emulator.h"
|
||||||
|
#include "xenia/kernel/objects/xkernel_module.h"
|
||||||
|
#include "xenia/kernel/objects/xmodule.h"
|
||||||
|
#include "xenia/kernel/objects/xuser_module.h"
|
||||||
|
|
||||||
// Autogenerated Flatbuffers files:
|
// Autogenerated Flatbuffers files:
|
||||||
#include "xenia/debug/proto/breakpoints_generated.h"
|
#include "xenia/debug/proto/breakpoints_generated.h"
|
||||||
|
@ -39,6 +42,8 @@ DEFINE_bool(exit_with_debugger, true, "Exit whe the debugger disconnects.");
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace debug {
|
namespace debug {
|
||||||
|
|
||||||
|
using namespace xe::kernel;
|
||||||
|
|
||||||
using xe::cpu::ThreadState;
|
using xe::cpu::ThreadState;
|
||||||
|
|
||||||
Breakpoint::Breakpoint(Type type, uint32_t address)
|
Breakpoint::Breakpoint(Type type, uint32_t address)
|
||||||
|
@ -235,14 +240,149 @@ void Debugger::OnMessage(std::vector<uint8_t> buffer) {
|
||||||
|
|
||||||
case proto::RequestData_ListModulesRequest: {
|
case proto::RequestData_ListModulesRequest: {
|
||||||
response_data_type = proto::ResponseData_ListModulesResponse;
|
response_data_type = proto::ResponseData_ListModulesResponse;
|
||||||
|
auto modules =
|
||||||
|
emulator()->kernel_state()->object_table()->GetObjectsByType<XModule>(
|
||||||
|
XObject::kTypeModule);
|
||||||
|
std::vector<proto::ListModuleEntry> module_entries;
|
||||||
|
for (size_t i = 0; i < modules.size(); ++i) {
|
||||||
|
auto& module = modules[i];
|
||||||
|
auto processor_module = module->processor_module();
|
||||||
|
module_entries.emplace_back(
|
||||||
|
module->handle(),
|
||||||
|
processor_module ? uint32_t(processor_module->QuerySymbolCount())
|
||||||
|
: 0);
|
||||||
|
}
|
||||||
|
auto module_entries_offset =
|
||||||
|
fbb.CreateVectorOfStructs<proto::ListModuleEntry>(module_entries);
|
||||||
auto response_data = proto::ListModulesResponseBuilder(fbb);
|
auto response_data = proto::ListModulesResponseBuilder(fbb);
|
||||||
//
|
response_data.add_entry(module_entries_offset);
|
||||||
response_data_offset = response_data.Finish().Union();
|
response_data_offset = response_data.Finish().Union();
|
||||||
} break;
|
} break;
|
||||||
case proto::RequestData_GetModuleRequest: {
|
case proto::RequestData_GetModuleRequest: {
|
||||||
|
auto request_data = reinterpret_cast<const proto::GetModuleRequest*>(
|
||||||
|
request->request_data());
|
||||||
|
auto module =
|
||||||
|
emulator()->kernel_state()->object_table()->LookupObject<XModule>(
|
||||||
|
request_data->module_id());
|
||||||
|
flatbuffers::Offset<proto::Module> module_offset;
|
||||||
|
if (module) {
|
||||||
|
proto::XObject xobject_data(module->handle());
|
||||||
|
auto module_name_offset = fbb.CreateString(module->name());
|
||||||
|
auto module_path_offset = fbb.CreateString(module->path());
|
||||||
|
proto::ModuleBuilder module_builder(fbb);
|
||||||
|
module_builder.add_object(&xobject_data);
|
||||||
|
module_builder.add_type(
|
||||||
|
static_cast<proto::ModuleType>(module->module_type()));
|
||||||
|
module_builder.add_name(module_name_offset);
|
||||||
|
module_builder.add_path(module_path_offset);
|
||||||
|
switch (module->module_type()) {
|
||||||
|
case XModule::ModuleType::kKernelModule: {
|
||||||
|
auto kernel_module = reinterpret_cast<XKernelModule*>(module.get());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case XModule::ModuleType::kUserModule: {
|
||||||
|
auto user_module = reinterpret_cast<XUserModule*>(module.get());
|
||||||
|
// user_module->xex?
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
assert_unhandled_case(module->module_type());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
module_offset = module_builder.Finish();
|
||||||
|
}
|
||||||
response_data_type = proto::ResponseData_GetModuleResponse;
|
response_data_type = proto::ResponseData_GetModuleResponse;
|
||||||
auto response_data = proto::GetModuleResponseBuilder(fbb);
|
auto response_data = proto::GetModuleResponseBuilder(fbb);
|
||||||
//
|
response_data.add_module(module_offset);
|
||||||
|
response_data_offset = response_data.Finish().Union();
|
||||||
|
} break;
|
||||||
|
case proto::RequestData_ListFunctionsRequest: {
|
||||||
|
auto request_data = reinterpret_cast<const proto::ListFunctionsRequest*>(
|
||||||
|
request->request_data());
|
||||||
|
auto module =
|
||||||
|
emulator()->kernel_state()->object_table()->LookupObject<XModule>(
|
||||||
|
request_data->module_id());
|
||||||
|
if (!module || !module->processor_module()) {
|
||||||
|
response_data_type = proto::ResponseData_ListFunctionsResponse;
|
||||||
|
auto response_data = proto::ListFunctionsResponseBuilder(fbb);
|
||||||
|
response_data_offset = response_data.Finish().Union();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
auto processor_module = module->processor_module();
|
||||||
|
size_t max_function_count = request_data->function_index_end() -
|
||||||
|
request_data->function_index_start() + 1;
|
||||||
|
std::vector<flatbuffers::Offset<proto::FunctionEntry>> function_list;
|
||||||
|
function_list.reserve(max_function_count);
|
||||||
|
processor_module->ForEachSymbol(request_data->function_index_start(),
|
||||||
|
request_data->function_index_end(),
|
||||||
|
[&](xe::cpu::SymbolInfo* symbol_info) {
|
||||||
|
if (symbol_info->type() != xe::cpu::SymbolType::kFunction) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto function_info =
|
||||||
|
reinterpret_cast<xe::cpu::FunctionInfo*>(symbol_info);
|
||||||
|
flatbuffers::Offset<flatbuffers::String> name_offset;
|
||||||
|
if (!function_info->name().empty()) {
|
||||||
|
name_offset = fbb.CreateString(function_info->name());
|
||||||
|
}
|
||||||
|
auto function_entry = proto::FunctionEntryBuilder(fbb);
|
||||||
|
function_entry.add_identifier(
|
||||||
|
reinterpret_cast<uintptr_t>(function_info));
|
||||||
|
function_entry.add_address_start(function_info->address());
|
||||||
|
function_entry.add_address_end(function_info->end_address());
|
||||||
|
function_entry.add_name(name_offset);
|
||||||
|
function_list.push_back(function_entry.Finish());
|
||||||
|
});
|
||||||
|
auto function_list_data = fbb.CreateVector(function_list);
|
||||||
|
response_data_type = proto::ResponseData_ListFunctionsResponse;
|
||||||
|
auto response_data = proto::ListFunctionsResponseBuilder(fbb);
|
||||||
|
response_data.add_entry(function_list_data);
|
||||||
|
response_data_offset = response_data.Finish().Union();
|
||||||
|
} break;
|
||||||
|
case proto::RequestData_GetFunctionRequest: {
|
||||||
|
auto request_data = reinterpret_cast<const proto::GetFunctionRequest*>(
|
||||||
|
request->request_data());
|
||||||
|
auto function_info =
|
||||||
|
reinterpret_cast<xe::cpu::FunctionInfo*>(request_data->identifier());
|
||||||
|
auto function = function_info->function();
|
||||||
|
flatbuffers::Offset<flatbuffers::String> name_offset;
|
||||||
|
if (!function_info->name().empty()) {
|
||||||
|
name_offset = fbb.CreateString(function_info->name());
|
||||||
|
}
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_ppc_offset;
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_hir_raw_offset;
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_hir_opt_offset;
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_machine_code_offset;
|
||||||
|
if (function && function->debug_info()) {
|
||||||
|
auto debug_info = function->debug_info();
|
||||||
|
if (debug_info->source_disasm()) {
|
||||||
|
disasm_ppc_offset = fbb.CreateString(debug_info->source_disasm());
|
||||||
|
}
|
||||||
|
if (debug_info->raw_hir_disasm()) {
|
||||||
|
disasm_hir_raw_offset =
|
||||||
|
fbb.CreateString(debug_info->raw_hir_disasm());
|
||||||
|
}
|
||||||
|
if (debug_info->hir_disasm()) {
|
||||||
|
disasm_hir_opt_offset = fbb.CreateString(debug_info->hir_disasm());
|
||||||
|
}
|
||||||
|
if (debug_info->machine_code_disasm()) {
|
||||||
|
disasm_machine_code_offset =
|
||||||
|
fbb.CreateString(debug_info->machine_code_disasm());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
auto function_data = proto::FunctionBuilder(fbb);
|
||||||
|
function_data.add_identifier(request_data->identifier());
|
||||||
|
function_data.add_address_start(function_info->address());
|
||||||
|
function_data.add_address_end(function_info->end_address());
|
||||||
|
function_data.add_name(name_offset);
|
||||||
|
function_data.add_disasm_ppc(disasm_ppc_offset);
|
||||||
|
function_data.add_disasm_hir_raw(disasm_hir_raw_offset);
|
||||||
|
function_data.add_disasm_hir_opt(disasm_hir_opt_offset);
|
||||||
|
function_data.add_disasm_machine_code(disasm_machine_code_offset);
|
||||||
|
auto function_offset = function_data.Finish();
|
||||||
|
response_data_type = proto::ResponseData_GetFunctionResponse;
|
||||||
|
auto response_data = proto::GetFunctionResponseBuilder(fbb);
|
||||||
|
response_data.add_function(function_offset);
|
||||||
response_data_offset = response_data.Finish().Union();
|
response_data_offset = response_data.Finish().Union();
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,8 @@ union RequestData {
|
||||||
|
|
||||||
ListModulesRequest,
|
ListModulesRequest,
|
||||||
GetModuleRequest,
|
GetModuleRequest,
|
||||||
|
ListFunctionsRequest,
|
||||||
|
GetFunctionRequest,
|
||||||
|
|
||||||
StopRequest,
|
StopRequest,
|
||||||
BreakRequest,
|
BreakRequest,
|
||||||
|
@ -47,6 +49,8 @@ union ResponseData {
|
||||||
|
|
||||||
ListModulesResponse,
|
ListModulesResponse,
|
||||||
GetModuleResponse,
|
GetModuleResponse,
|
||||||
|
ListFunctionsResponse,
|
||||||
|
GetFunctionResponse,
|
||||||
|
|
||||||
StopResponse,
|
StopResponse,
|
||||||
BreakResponse,
|
BreakResponse,
|
||||||
|
|
|
@ -48,9 +48,16 @@ namespace debug {
|
||||||
namespace proto {
|
namespace proto {
|
||||||
struct Module;
|
struct Module;
|
||||||
struct ListModulesRequest;
|
struct ListModulesRequest;
|
||||||
|
struct ListModuleEntry;
|
||||||
struct ListModulesResponse;
|
struct ListModulesResponse;
|
||||||
struct GetModuleRequest;
|
struct GetModuleRequest;
|
||||||
struct GetModuleResponse;
|
struct GetModuleResponse;
|
||||||
|
struct FunctionEntry;
|
||||||
|
struct Function;
|
||||||
|
struct ListFunctionsRequest;
|
||||||
|
struct ListFunctionsResponse;
|
||||||
|
struct GetFunctionRequest;
|
||||||
|
struct GetFunctionResponse;
|
||||||
} // namespace proto
|
} // namespace proto
|
||||||
} // namespace debug
|
} // namespace debug
|
||||||
} // namespace xe
|
} // namespace xe
|
||||||
|
@ -82,14 +89,24 @@ enum RequestData {
|
||||||
RequestData_RemoveBreakpointsRequest = 5,
|
RequestData_RemoveBreakpointsRequest = 5,
|
||||||
RequestData_ListModulesRequest = 6,
|
RequestData_ListModulesRequest = 6,
|
||||||
RequestData_GetModuleRequest = 7,
|
RequestData_GetModuleRequest = 7,
|
||||||
RequestData_StopRequest = 8,
|
RequestData_ListFunctionsRequest = 8,
|
||||||
RequestData_BreakRequest = 9,
|
RequestData_GetFunctionRequest = 9,
|
||||||
RequestData_ContinueRequest = 10,
|
RequestData_StopRequest = 10,
|
||||||
RequestData_StepRequest = 11
|
RequestData_BreakRequest = 11,
|
||||||
|
RequestData_ContinueRequest = 12,
|
||||||
|
RequestData_StepRequest = 13
|
||||||
};
|
};
|
||||||
|
|
||||||
inline const char **EnumNamesRequestData() {
|
inline const char **EnumNamesRequestData() {
|
||||||
static const char *names[] = { "NONE", "AttachRequest", "ListBreakpointsRequest", "AddBreakpointsRequest", "UpdateBreakpointsRequest", "RemoveBreakpointsRequest", "ListModulesRequest", "GetModuleRequest", "StopRequest", "BreakRequest", "ContinueRequest", "StepRequest", nullptr };
|
static const char *names[] = {
|
||||||
|
"NONE", "AttachRequest",
|
||||||
|
"ListBreakpointsRequest", "AddBreakpointsRequest",
|
||||||
|
"UpdateBreakpointsRequest", "RemoveBreakpointsRequest",
|
||||||
|
"ListModulesRequest", "GetModuleRequest",
|
||||||
|
"ListFunctionsRequest", "GetFunctionRequest",
|
||||||
|
"StopRequest", "BreakRequest",
|
||||||
|
"ContinueRequest", "StepRequest",
|
||||||
|
nullptr};
|
||||||
return names;
|
return names;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -106,16 +123,27 @@ enum ResponseData {
|
||||||
ResponseData_RemoveBreakpointsResponse = 5,
|
ResponseData_RemoveBreakpointsResponse = 5,
|
||||||
ResponseData_ListModulesResponse = 6,
|
ResponseData_ListModulesResponse = 6,
|
||||||
ResponseData_GetModuleResponse = 7,
|
ResponseData_GetModuleResponse = 7,
|
||||||
ResponseData_StopResponse = 8,
|
ResponseData_ListFunctionsResponse = 8,
|
||||||
ResponseData_BreakResponse = 9,
|
ResponseData_GetFunctionResponse = 9,
|
||||||
ResponseData_ContinueResponse = 10,
|
ResponseData_StopResponse = 10,
|
||||||
ResponseData_StepResponse = 11,
|
ResponseData_BreakResponse = 11,
|
||||||
ResponseData_BreakpointEvent = 12,
|
ResponseData_ContinueResponse = 12,
|
||||||
ResponseData_AccessViolationEvent = 13
|
ResponseData_StepResponse = 13,
|
||||||
|
ResponseData_BreakpointEvent = 14,
|
||||||
|
ResponseData_AccessViolationEvent = 15
|
||||||
};
|
};
|
||||||
|
|
||||||
inline const char **EnumNamesResponseData() {
|
inline const char **EnumNamesResponseData() {
|
||||||
static const char *names[] = { "NONE", "AttachResponse", "ListBreakpointsResponse", "AddBreakpointsResponse", "UpdateBreakpointsResponse", "RemoveBreakpointsResponse", "ListModulesResponse", "GetModuleResponse", "StopResponse", "BreakResponse", "ContinueResponse", "StepResponse", "BreakpointEvent", "AccessViolationEvent", nullptr };
|
static const char *names[] = {
|
||||||
|
"NONE", "AttachResponse",
|
||||||
|
"ListBreakpointsResponse", "AddBreakpointsResponse",
|
||||||
|
"UpdateBreakpointsResponse", "RemoveBreakpointsResponse",
|
||||||
|
"ListModulesResponse", "GetModuleResponse",
|
||||||
|
"ListFunctionsResponse", "GetFunctionResponse",
|
||||||
|
"StopResponse", "BreakResponse",
|
||||||
|
"ContinueResponse", "StepResponse",
|
||||||
|
"BreakpointEvent", "AccessViolationEvent",
|
||||||
|
nullptr};
|
||||||
return names;
|
return names;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -275,6 +303,14 @@ inline bool VerifyRequestData(flatbuffers::Verifier &verifier, const void *union
|
||||||
case RequestData_RemoveBreakpointsRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::RemoveBreakpointsRequest *>(union_obj));
|
case RequestData_RemoveBreakpointsRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::RemoveBreakpointsRequest *>(union_obj));
|
||||||
case RequestData_ListModulesRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ListModulesRequest *>(union_obj));
|
case RequestData_ListModulesRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ListModulesRequest *>(union_obj));
|
||||||
case RequestData_GetModuleRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::GetModuleRequest *>(union_obj));
|
case RequestData_GetModuleRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::GetModuleRequest *>(union_obj));
|
||||||
|
case RequestData_ListFunctionsRequest:
|
||||||
|
return verifier.VerifyTable(
|
||||||
|
reinterpret_cast<const xe::debug::proto::ListFunctionsRequest *>(
|
||||||
|
union_obj));
|
||||||
|
case RequestData_GetFunctionRequest:
|
||||||
|
return verifier.VerifyTable(
|
||||||
|
reinterpret_cast<const xe::debug::proto::GetFunctionRequest *>(
|
||||||
|
union_obj));
|
||||||
case RequestData_StopRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::StopRequest *>(union_obj));
|
case RequestData_StopRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::StopRequest *>(union_obj));
|
||||||
case RequestData_BreakRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::BreakRequest *>(union_obj));
|
case RequestData_BreakRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::BreakRequest *>(union_obj));
|
||||||
case RequestData_ContinueRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ContinueRequest *>(union_obj));
|
case RequestData_ContinueRequest: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ContinueRequest *>(union_obj));
|
||||||
|
@ -293,6 +329,14 @@ inline bool VerifyResponseData(flatbuffers::Verifier &verifier, const void *unio
|
||||||
case ResponseData_RemoveBreakpointsResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::RemoveBreakpointsResponse *>(union_obj));
|
case ResponseData_RemoveBreakpointsResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::RemoveBreakpointsResponse *>(union_obj));
|
||||||
case ResponseData_ListModulesResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ListModulesResponse *>(union_obj));
|
case ResponseData_ListModulesResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ListModulesResponse *>(union_obj));
|
||||||
case ResponseData_GetModuleResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::GetModuleResponse *>(union_obj));
|
case ResponseData_GetModuleResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::GetModuleResponse *>(union_obj));
|
||||||
|
case ResponseData_ListFunctionsResponse:
|
||||||
|
return verifier.VerifyTable(
|
||||||
|
reinterpret_cast<const xe::debug::proto::ListFunctionsResponse *>(
|
||||||
|
union_obj));
|
||||||
|
case ResponseData_GetFunctionResponse:
|
||||||
|
return verifier.VerifyTable(
|
||||||
|
reinterpret_cast<const xe::debug::proto::GetFunctionResponse *>(
|
||||||
|
union_obj));
|
||||||
case ResponseData_StopResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::StopResponse *>(union_obj));
|
case ResponseData_StopResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::StopResponse *>(union_obj));
|
||||||
case ResponseData_BreakResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::BreakResponse *>(union_obj));
|
case ResponseData_BreakResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::BreakResponse *>(union_obj));
|
||||||
case ResponseData_ContinueResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ContinueResponse *>(union_obj));
|
case ResponseData_ContinueResponse: return verifier.VerifyTable(reinterpret_cast<const xe::debug::proto::ContinueResponse *>(union_obj));
|
||||||
|
|
|
@ -3,8 +3,8 @@ include "common.fbs";
|
||||||
namespace xe.debug.proto;
|
namespace xe.debug.proto;
|
||||||
|
|
||||||
enum ModuleType:byte {
|
enum ModuleType:byte {
|
||||||
Kernel,
|
Kernel = 0,
|
||||||
User,
|
User = 1,
|
||||||
}
|
}
|
||||||
|
|
||||||
table Module {
|
table Module {
|
||||||
|
@ -18,8 +18,12 @@ table Module {
|
||||||
|
|
||||||
table ListModulesRequest {
|
table ListModulesRequest {
|
||||||
}
|
}
|
||||||
|
struct ListModuleEntry {
|
||||||
|
handle:uint;
|
||||||
|
function_count:uint;
|
||||||
|
}
|
||||||
table ListModulesResponse {
|
table ListModulesResponse {
|
||||||
module_ids:[uint];
|
entry:[ListModuleEntry];
|
||||||
}
|
}
|
||||||
|
|
||||||
table GetModuleRequest {
|
table GetModuleRequest {
|
||||||
|
@ -29,3 +33,36 @@ table GetModuleResponse {
|
||||||
module:Module;
|
module:Module;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
table FunctionEntry {
|
||||||
|
identifier:ulong;
|
||||||
|
address_start:uint;
|
||||||
|
address_end:uint;
|
||||||
|
name:string;
|
||||||
|
}
|
||||||
|
table Function {
|
||||||
|
identifier:ulong;
|
||||||
|
address_start:uint;
|
||||||
|
address_end:uint;
|
||||||
|
name:string;
|
||||||
|
|
||||||
|
disasm_ppc:string;
|
||||||
|
disasm_hir_raw:string;
|
||||||
|
disasm_hir_opt:string;
|
||||||
|
disasm_machine_code:string;
|
||||||
|
}
|
||||||
|
|
||||||
|
table ListFunctionsRequest {
|
||||||
|
module_id:uint;
|
||||||
|
function_index_start:uint;
|
||||||
|
function_index_end:uint;
|
||||||
|
}
|
||||||
|
table ListFunctionsResponse {
|
||||||
|
entry:[FunctionEntry];
|
||||||
|
}
|
||||||
|
|
||||||
|
table GetFunctionRequest {
|
||||||
|
identifier:ulong;
|
||||||
|
}
|
||||||
|
table GetFunctionResponse {
|
||||||
|
function:Function;
|
||||||
|
}
|
||||||
|
|
|
@ -19,9 +19,16 @@ namespace proto {
|
||||||
|
|
||||||
struct Module;
|
struct Module;
|
||||||
struct ListModulesRequest;
|
struct ListModulesRequest;
|
||||||
|
struct ListModuleEntry;
|
||||||
struct ListModulesResponse;
|
struct ListModulesResponse;
|
||||||
struct GetModuleRequest;
|
struct GetModuleRequest;
|
||||||
struct GetModuleResponse;
|
struct GetModuleResponse;
|
||||||
|
struct FunctionEntry;
|
||||||
|
struct Function;
|
||||||
|
struct ListFunctionsRequest;
|
||||||
|
struct ListFunctionsResponse;
|
||||||
|
struct GetFunctionRequest;
|
||||||
|
struct GetFunctionResponse;
|
||||||
|
|
||||||
enum ModuleType {
|
enum ModuleType {
|
||||||
ModuleType_Kernel = 0,
|
ModuleType_Kernel = 0,
|
||||||
|
@ -35,6 +42,23 @@ inline const char **EnumNamesModuleType() {
|
||||||
|
|
||||||
inline const char *EnumNameModuleType(ModuleType e) { return EnumNamesModuleType()[e]; }
|
inline const char *EnumNameModuleType(ModuleType e) { return EnumNamesModuleType()[e]; }
|
||||||
|
|
||||||
|
MANUALLY_ALIGNED_STRUCT(4) ListModuleEntry FLATBUFFERS_FINAL_CLASS {
|
||||||
|
private:
|
||||||
|
uint32_t handle_;
|
||||||
|
uint32_t function_count_;
|
||||||
|
|
||||||
|
public:
|
||||||
|
ListModuleEntry(uint32_t handle, uint32_t function_count)
|
||||||
|
: handle_(flatbuffers::EndianScalar(handle)),
|
||||||
|
function_count_(flatbuffers::EndianScalar(function_count)) {}
|
||||||
|
|
||||||
|
uint32_t handle() const { return flatbuffers::EndianScalar(handle_); }
|
||||||
|
uint32_t function_count() const {
|
||||||
|
return flatbuffers::EndianScalar(function_count_);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
STRUCT_END(ListModuleEntry, 8);
|
||||||
|
|
||||||
struct Module FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
struct Module FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||||
const xe::debug::proto::XObject *object() const { return GetStruct<const xe::debug::proto::XObject *>(4); }
|
const xe::debug::proto::XObject *object() const { return GetStruct<const xe::debug::proto::XObject *>(4); }
|
||||||
ModuleType type() const { return static_cast<ModuleType>(GetField<int8_t>(6, 0)); }
|
ModuleType type() const { return static_cast<ModuleType>(GetField<int8_t>(6, 0)); }
|
||||||
|
@ -104,19 +128,23 @@ inline flatbuffers::Offset<ListModulesRequest> CreateListModulesRequest(flatbuff
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ListModulesResponse FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
struct ListModulesResponse FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||||
const flatbuffers::Vector<uint32_t> *module_ids() const { return GetPointer<const flatbuffers::Vector<uint32_t> *>(4); }
|
const flatbuffers::Vector<const ListModuleEntry *> *entry() const {
|
||||||
|
return GetPointer<const flatbuffers::Vector<const ListModuleEntry *> *>(4);
|
||||||
|
}
|
||||||
bool Verify(flatbuffers::Verifier &verifier) const {
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
return VerifyTableStart(verifier) &&
|
return VerifyTableStart(verifier) &&
|
||||||
VerifyField<flatbuffers::uoffset_t>(verifier, 4 /* module_ids */) &&
|
VerifyField<flatbuffers::uoffset_t>(verifier, 4 /* entry */) &&
|
||||||
verifier.Verify(module_ids()) &&
|
verifier.Verify(entry()) && verifier.EndTable();
|
||||||
verifier.EndTable();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ListModulesResponseBuilder {
|
struct ListModulesResponseBuilder {
|
||||||
flatbuffers::FlatBufferBuilder &fbb_;
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
flatbuffers::uoffset_t start_;
|
flatbuffers::uoffset_t start_;
|
||||||
void add_module_ids(flatbuffers::Offset<flatbuffers::Vector<uint32_t>> module_ids) { fbb_.AddOffset(4, module_ids); }
|
void add_entry(
|
||||||
|
flatbuffers::Offset<flatbuffers::Vector<const ListModuleEntry *>> entry) {
|
||||||
|
fbb_.AddOffset(4, entry);
|
||||||
|
}
|
||||||
ListModulesResponseBuilder(flatbuffers::FlatBufferBuilder &_fbb) : fbb_(_fbb) { start_ = fbb_.StartTable(); }
|
ListModulesResponseBuilder(flatbuffers::FlatBufferBuilder &_fbb) : fbb_(_fbb) { start_ = fbb_.StartTable(); }
|
||||||
ListModulesResponseBuilder &operator=(const ListModulesResponseBuilder &);
|
ListModulesResponseBuilder &operator=(const ListModulesResponseBuilder &);
|
||||||
flatbuffers::Offset<ListModulesResponse> Finish() {
|
flatbuffers::Offset<ListModulesResponse> Finish() {
|
||||||
|
@ -125,10 +153,12 @@ struct ListModulesResponseBuilder {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
inline flatbuffers::Offset<ListModulesResponse> CreateListModulesResponse(flatbuffers::FlatBufferBuilder &_fbb,
|
inline flatbuffers::Offset<ListModulesResponse> CreateListModulesResponse(
|
||||||
flatbuffers::Offset<flatbuffers::Vector<uint32_t>> module_ids = 0) {
|
flatbuffers::FlatBufferBuilder &_fbb,
|
||||||
|
flatbuffers::Offset<flatbuffers::Vector<const ListModuleEntry *>> entry =
|
||||||
|
0) {
|
||||||
ListModulesResponseBuilder builder_(_fbb);
|
ListModulesResponseBuilder builder_(_fbb);
|
||||||
builder_.add_module_ids(module_ids);
|
builder_.add_entry(entry);
|
||||||
return builder_.Finish();
|
return builder_.Finish();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -189,6 +219,317 @@ inline flatbuffers::Offset<GetModuleResponse> CreateGetModuleResponse(flatbuffer
|
||||||
return builder_.Finish();
|
return builder_.Finish();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct FunctionEntry FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||||
|
uint64_t identifier() const { return GetField<uint64_t>(4, 0); }
|
||||||
|
uint32_t address_start() const { return GetField<uint32_t>(6, 0); }
|
||||||
|
uint32_t address_end() const { return GetField<uint32_t>(8, 0); }
|
||||||
|
const flatbuffers::String *name() const {
|
||||||
|
return GetPointer<const flatbuffers::String *>(10);
|
||||||
|
}
|
||||||
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
|
return VerifyTableStart(verifier) &&
|
||||||
|
VerifyField<uint64_t>(verifier, 4 /* identifier */) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 6 /* address_start */) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 8 /* address_end */) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier, 10 /* name */) &&
|
||||||
|
verifier.Verify(name()) && verifier.EndTable();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct FunctionEntryBuilder {
|
||||||
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
|
flatbuffers::uoffset_t start_;
|
||||||
|
void add_identifier(uint64_t identifier) {
|
||||||
|
fbb_.AddElement<uint64_t>(4, identifier, 0);
|
||||||
|
}
|
||||||
|
void add_address_start(uint32_t address_start) {
|
||||||
|
fbb_.AddElement<uint32_t>(6, address_start, 0);
|
||||||
|
}
|
||||||
|
void add_address_end(uint32_t address_end) {
|
||||||
|
fbb_.AddElement<uint32_t>(8, address_end, 0);
|
||||||
|
}
|
||||||
|
void add_name(flatbuffers::Offset<flatbuffers::String> name) {
|
||||||
|
fbb_.AddOffset(10, name);
|
||||||
|
}
|
||||||
|
FunctionEntryBuilder(flatbuffers::FlatBufferBuilder &_fbb) : fbb_(_fbb) {
|
||||||
|
start_ = fbb_.StartTable();
|
||||||
|
}
|
||||||
|
FunctionEntryBuilder &operator=(const FunctionEntryBuilder &);
|
||||||
|
flatbuffers::Offset<FunctionEntry> Finish() {
|
||||||
|
auto o = flatbuffers::Offset<FunctionEntry>(fbb_.EndTable(start_, 4));
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
inline flatbuffers::Offset<FunctionEntry> CreateFunctionEntry(
|
||||||
|
flatbuffers::FlatBufferBuilder &_fbb, uint64_t identifier = 0,
|
||||||
|
uint32_t address_start = 0, uint32_t address_end = 0,
|
||||||
|
flatbuffers::Offset<flatbuffers::String> name = 0) {
|
||||||
|
FunctionEntryBuilder builder_(_fbb);
|
||||||
|
builder_.add_identifier(identifier);
|
||||||
|
builder_.add_name(name);
|
||||||
|
builder_.add_address_end(address_end);
|
||||||
|
builder_.add_address_start(address_start);
|
||||||
|
return builder_.Finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Function FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||||
|
uint64_t identifier() const { return GetField<uint64_t>(4, 0); }
|
||||||
|
uint32_t address_start() const { return GetField<uint32_t>(6, 0); }
|
||||||
|
uint32_t address_end() const { return GetField<uint32_t>(8, 0); }
|
||||||
|
const flatbuffers::String *name() const {
|
||||||
|
return GetPointer<const flatbuffers::String *>(10);
|
||||||
|
}
|
||||||
|
const flatbuffers::String *disasm_ppc() const {
|
||||||
|
return GetPointer<const flatbuffers::String *>(12);
|
||||||
|
}
|
||||||
|
const flatbuffers::String *disasm_hir_raw() const {
|
||||||
|
return GetPointer<const flatbuffers::String *>(14);
|
||||||
|
}
|
||||||
|
const flatbuffers::String *disasm_hir_opt() const {
|
||||||
|
return GetPointer<const flatbuffers::String *>(16);
|
||||||
|
}
|
||||||
|
const flatbuffers::String *disasm_machine_code() const {
|
||||||
|
return GetPointer<const flatbuffers::String *>(18);
|
||||||
|
}
|
||||||
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
|
return VerifyTableStart(verifier) &&
|
||||||
|
VerifyField<uint64_t>(verifier, 4 /* identifier */) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 6 /* address_start */) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 8 /* address_end */) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier, 10 /* name */) &&
|
||||||
|
verifier.Verify(name()) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier, 12 /* disasm_ppc */) &&
|
||||||
|
verifier.Verify(disasm_ppc()) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier,
|
||||||
|
14 /* disasm_hir_raw */) &&
|
||||||
|
verifier.Verify(disasm_hir_raw()) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier,
|
||||||
|
16 /* disasm_hir_opt */) &&
|
||||||
|
verifier.Verify(disasm_hir_opt()) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier,
|
||||||
|
18 /* disasm_machine_code */) &&
|
||||||
|
verifier.Verify(disasm_machine_code()) && verifier.EndTable();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct FunctionBuilder {
|
||||||
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
|
flatbuffers::uoffset_t start_;
|
||||||
|
void add_identifier(uint64_t identifier) {
|
||||||
|
fbb_.AddElement<uint64_t>(4, identifier, 0);
|
||||||
|
}
|
||||||
|
void add_address_start(uint32_t address_start) {
|
||||||
|
fbb_.AddElement<uint32_t>(6, address_start, 0);
|
||||||
|
}
|
||||||
|
void add_address_end(uint32_t address_end) {
|
||||||
|
fbb_.AddElement<uint32_t>(8, address_end, 0);
|
||||||
|
}
|
||||||
|
void add_name(flatbuffers::Offset<flatbuffers::String> name) {
|
||||||
|
fbb_.AddOffset(10, name);
|
||||||
|
}
|
||||||
|
void add_disasm_ppc(flatbuffers::Offset<flatbuffers::String> disasm_ppc) {
|
||||||
|
fbb_.AddOffset(12, disasm_ppc);
|
||||||
|
}
|
||||||
|
void add_disasm_hir_raw(
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_hir_raw) {
|
||||||
|
fbb_.AddOffset(14, disasm_hir_raw);
|
||||||
|
}
|
||||||
|
void add_disasm_hir_opt(
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_hir_opt) {
|
||||||
|
fbb_.AddOffset(16, disasm_hir_opt);
|
||||||
|
}
|
||||||
|
void add_disasm_machine_code(
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_machine_code) {
|
||||||
|
fbb_.AddOffset(18, disasm_machine_code);
|
||||||
|
}
|
||||||
|
FunctionBuilder(flatbuffers::FlatBufferBuilder &_fbb) : fbb_(_fbb) {
|
||||||
|
start_ = fbb_.StartTable();
|
||||||
|
}
|
||||||
|
FunctionBuilder &operator=(const FunctionBuilder &);
|
||||||
|
flatbuffers::Offset<Function> Finish() {
|
||||||
|
auto o = flatbuffers::Offset<Function>(fbb_.EndTable(start_, 8));
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
inline flatbuffers::Offset<Function> CreateFunction(
|
||||||
|
flatbuffers::FlatBufferBuilder &_fbb, uint64_t identifier = 0,
|
||||||
|
uint32_t address_start = 0, uint32_t address_end = 0,
|
||||||
|
flatbuffers::Offset<flatbuffers::String> name = 0,
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_ppc = 0,
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_hir_raw = 0,
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_hir_opt = 0,
|
||||||
|
flatbuffers::Offset<flatbuffers::String> disasm_machine_code = 0) {
|
||||||
|
FunctionBuilder builder_(_fbb);
|
||||||
|
builder_.add_identifier(identifier);
|
||||||
|
builder_.add_disasm_machine_code(disasm_machine_code);
|
||||||
|
builder_.add_disasm_hir_opt(disasm_hir_opt);
|
||||||
|
builder_.add_disasm_hir_raw(disasm_hir_raw);
|
||||||
|
builder_.add_disasm_ppc(disasm_ppc);
|
||||||
|
builder_.add_name(name);
|
||||||
|
builder_.add_address_end(address_end);
|
||||||
|
builder_.add_address_start(address_start);
|
||||||
|
return builder_.Finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ListFunctionsRequest FLATBUFFERS_FINAL_CLASS
|
||||||
|
: private flatbuffers::Table {
|
||||||
|
uint32_t module_id() const { return GetField<uint32_t>(4, 0); }
|
||||||
|
uint32_t function_index_start() const { return GetField<uint32_t>(6, 0); }
|
||||||
|
uint32_t function_index_end() const { return GetField<uint32_t>(8, 0); }
|
||||||
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
|
return VerifyTableStart(verifier) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 4 /* module_id */) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 6 /* function_index_start */) &&
|
||||||
|
VerifyField<uint32_t>(verifier, 8 /* function_index_end */) &&
|
||||||
|
verifier.EndTable();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ListFunctionsRequestBuilder {
|
||||||
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
|
flatbuffers::uoffset_t start_;
|
||||||
|
void add_module_id(uint32_t module_id) {
|
||||||
|
fbb_.AddElement<uint32_t>(4, module_id, 0);
|
||||||
|
}
|
||||||
|
void add_function_index_start(uint32_t function_index_start) {
|
||||||
|
fbb_.AddElement<uint32_t>(6, function_index_start, 0);
|
||||||
|
}
|
||||||
|
void add_function_index_end(uint32_t function_index_end) {
|
||||||
|
fbb_.AddElement<uint32_t>(8, function_index_end, 0);
|
||||||
|
}
|
||||||
|
ListFunctionsRequestBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||||
|
: fbb_(_fbb) {
|
||||||
|
start_ = fbb_.StartTable();
|
||||||
|
}
|
||||||
|
ListFunctionsRequestBuilder &operator=(const ListFunctionsRequestBuilder &);
|
||||||
|
flatbuffers::Offset<ListFunctionsRequest> Finish() {
|
||||||
|
auto o =
|
||||||
|
flatbuffers::Offset<ListFunctionsRequest>(fbb_.EndTable(start_, 3));
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
inline flatbuffers::Offset<ListFunctionsRequest> CreateListFunctionsRequest(
|
||||||
|
flatbuffers::FlatBufferBuilder &_fbb, uint32_t module_id = 0,
|
||||||
|
uint32_t function_index_start = 0, uint32_t function_index_end = 0) {
|
||||||
|
ListFunctionsRequestBuilder builder_(_fbb);
|
||||||
|
builder_.add_function_index_end(function_index_end);
|
||||||
|
builder_.add_function_index_start(function_index_start);
|
||||||
|
builder_.add_module_id(module_id);
|
||||||
|
return builder_.Finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ListFunctionsResponse FLATBUFFERS_FINAL_CLASS
|
||||||
|
: private flatbuffers::Table {
|
||||||
|
const flatbuffers::Vector<flatbuffers::Offset<FunctionEntry>> *entry() const {
|
||||||
|
return GetPointer<
|
||||||
|
const flatbuffers::Vector<flatbuffers::Offset<FunctionEntry>> *>(4);
|
||||||
|
}
|
||||||
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
|
return VerifyTableStart(verifier) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier, 4 /* entry */) &&
|
||||||
|
verifier.Verify(entry()) && verifier.VerifyVectorOfTables(entry()) &&
|
||||||
|
verifier.EndTable();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ListFunctionsResponseBuilder {
|
||||||
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
|
flatbuffers::uoffset_t start_;
|
||||||
|
void add_entry(flatbuffers::Offset<
|
||||||
|
flatbuffers::Vector<flatbuffers::Offset<FunctionEntry>>> entry) {
|
||||||
|
fbb_.AddOffset(4, entry);
|
||||||
|
}
|
||||||
|
ListFunctionsResponseBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||||
|
: fbb_(_fbb) {
|
||||||
|
start_ = fbb_.StartTable();
|
||||||
|
}
|
||||||
|
ListFunctionsResponseBuilder &operator=(const ListFunctionsResponseBuilder &);
|
||||||
|
flatbuffers::Offset<ListFunctionsResponse> Finish() {
|
||||||
|
auto o =
|
||||||
|
flatbuffers::Offset<ListFunctionsResponse>(fbb_.EndTable(start_, 1));
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
inline flatbuffers::Offset<ListFunctionsResponse> CreateListFunctionsResponse(
|
||||||
|
flatbuffers::FlatBufferBuilder &_fbb,
|
||||||
|
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<FunctionEntry>>>
|
||||||
|
entry = 0) {
|
||||||
|
ListFunctionsResponseBuilder builder_(_fbb);
|
||||||
|
builder_.add_entry(entry);
|
||||||
|
return builder_.Finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct GetFunctionRequest FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||||
|
uint64_t identifier() const { return GetField<uint64_t>(4, 0); }
|
||||||
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
|
return VerifyTableStart(verifier) &&
|
||||||
|
VerifyField<uint64_t>(verifier, 4 /* identifier */) &&
|
||||||
|
verifier.EndTable();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct GetFunctionRequestBuilder {
|
||||||
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
|
flatbuffers::uoffset_t start_;
|
||||||
|
void add_identifier(uint64_t identifier) {
|
||||||
|
fbb_.AddElement<uint64_t>(4, identifier, 0);
|
||||||
|
}
|
||||||
|
GetFunctionRequestBuilder(flatbuffers::FlatBufferBuilder &_fbb) : fbb_(_fbb) {
|
||||||
|
start_ = fbb_.StartTable();
|
||||||
|
}
|
||||||
|
GetFunctionRequestBuilder &operator=(const GetFunctionRequestBuilder &);
|
||||||
|
flatbuffers::Offset<GetFunctionRequest> Finish() {
|
||||||
|
auto o = flatbuffers::Offset<GetFunctionRequest>(fbb_.EndTable(start_, 1));
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
inline flatbuffers::Offset<GetFunctionRequest> CreateGetFunctionRequest(
|
||||||
|
flatbuffers::FlatBufferBuilder &_fbb, uint64_t identifier = 0) {
|
||||||
|
GetFunctionRequestBuilder builder_(_fbb);
|
||||||
|
builder_.add_identifier(identifier);
|
||||||
|
return builder_.Finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct GetFunctionResponse FLATBUFFERS_FINAL_CLASS
|
||||||
|
: private flatbuffers::Table {
|
||||||
|
const Function *function() const { return GetPointer<const Function *>(4); }
|
||||||
|
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||||
|
return VerifyTableStart(verifier) &&
|
||||||
|
VerifyField<flatbuffers::uoffset_t>(verifier, 4 /* function */) &&
|
||||||
|
verifier.VerifyTable(function()) && verifier.EndTable();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct GetFunctionResponseBuilder {
|
||||||
|
flatbuffers::FlatBufferBuilder &fbb_;
|
||||||
|
flatbuffers::uoffset_t start_;
|
||||||
|
void add_function(flatbuffers::Offset<Function> function) {
|
||||||
|
fbb_.AddOffset(4, function);
|
||||||
|
}
|
||||||
|
GetFunctionResponseBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||||
|
: fbb_(_fbb) {
|
||||||
|
start_ = fbb_.StartTable();
|
||||||
|
}
|
||||||
|
GetFunctionResponseBuilder &operator=(const GetFunctionResponseBuilder &);
|
||||||
|
flatbuffers::Offset<GetFunctionResponse> Finish() {
|
||||||
|
auto o = flatbuffers::Offset<GetFunctionResponse>(fbb_.EndTable(start_, 1));
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
inline flatbuffers::Offset<GetFunctionResponse> CreateGetFunctionResponse(
|
||||||
|
flatbuffers::FlatBufferBuilder &_fbb,
|
||||||
|
flatbuffers::Offset<Function> function = 0) {
|
||||||
|
GetFunctionResponseBuilder builder_(_fbb);
|
||||||
|
builder_.add_function(function);
|
||||||
|
return builder_.Finish();
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace proto
|
} // namespace proto
|
||||||
} // namespace debug
|
} // namespace debug
|
||||||
} // namespace xe
|
} // namespace xe
|
||||||
|
|
|
@ -18,7 +18,7 @@ namespace xe {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
|
|
||||||
XKernelModule::XKernelModule(KernelState* kernel_state, const char* path)
|
XKernelModule::XKernelModule(KernelState* kernel_state, const char* path)
|
||||||
: XModule(kernel_state, path) {
|
: XModule(kernel_state, ModuleType::kKernelModule, path) {
|
||||||
emulator_ = kernel_state->emulator();
|
emulator_ = kernel_state->emulator();
|
||||||
memory_ = emulator_->memory();
|
memory_ = emulator_->memory();
|
||||||
export_resolver_ = kernel_state->emulator()->export_resolver();
|
export_resolver_ = kernel_state->emulator()->export_resolver();
|
||||||
|
|
|
@ -15,8 +15,12 @@
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
|
|
||||||
XModule::XModule(KernelState* kernel_state, const std::string& path)
|
XModule::XModule(KernelState* kernel_state, ModuleType module_type,
|
||||||
: XObject(kernel_state, kTypeModule), path_(path) {
|
const std::string& path)
|
||||||
|
: XObject(kernel_state, kTypeModule),
|
||||||
|
module_type_(module_type),
|
||||||
|
path_(path),
|
||||||
|
processor_module_(nullptr) {
|
||||||
auto last_slash = path.find_last_of('/');
|
auto last_slash = path.find_last_of('/');
|
||||||
if (last_slash == path.npos) {
|
if (last_slash == path.npos) {
|
||||||
last_slash = path.find_last_of('\\');
|
last_slash = path.find_last_of('\\');
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
|
#include "xenia/cpu/module.h"
|
||||||
#include "xenia/kernel/xobject.h"
|
#include "xenia/kernel/xobject.h"
|
||||||
#include "xenia/xbox.h"
|
#include "xenia/xbox.h"
|
||||||
|
|
||||||
|
@ -20,13 +21,23 @@ namespace kernel {
|
||||||
|
|
||||||
class XModule : public XObject {
|
class XModule : public XObject {
|
||||||
public:
|
public:
|
||||||
XModule(KernelState* kernel_state, const std::string& path);
|
enum class ModuleType {
|
||||||
|
// Matches debugger Module type.
|
||||||
|
kKernelModule = 0,
|
||||||
|
kUserModule = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
XModule(KernelState* kernel_state, ModuleType module_type,
|
||||||
|
const std::string& path);
|
||||||
virtual ~XModule();
|
virtual ~XModule();
|
||||||
|
|
||||||
|
ModuleType module_type() const { return module_type_; }
|
||||||
const std::string& path() const { return path_; }
|
const std::string& path() const { return path_; }
|
||||||
const std::string& name() const { return name_; }
|
const std::string& name() const { return name_; }
|
||||||
bool Matches(const std::string& name) const;
|
bool Matches(const std::string& name) const;
|
||||||
|
|
||||||
|
xe::cpu::Module* processor_module() const { return processor_module_; }
|
||||||
|
|
||||||
virtual uint32_t GetProcAddressByOrdinal(uint16_t ordinal) = 0;
|
virtual uint32_t GetProcAddressByOrdinal(uint16_t ordinal) = 0;
|
||||||
virtual uint32_t GetProcAddressByName(const char* name) = 0;
|
virtual uint32_t GetProcAddressByName(const char* name) = 0;
|
||||||
virtual X_STATUS GetSection(const char* name, uint32_t* out_section_data,
|
virtual X_STATUS GetSection(const char* name, uint32_t* out_section_data,
|
||||||
|
@ -35,8 +46,11 @@ class XModule : public XObject {
|
||||||
protected:
|
protected:
|
||||||
void OnLoad();
|
void OnLoad();
|
||||||
|
|
||||||
|
ModuleType module_type_;
|
||||||
std::string name_;
|
std::string name_;
|
||||||
std::string path_;
|
std::string path_;
|
||||||
|
|
||||||
|
xe::cpu::Module* processor_module_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
|
|
|
@ -21,7 +21,9 @@ namespace kernel {
|
||||||
using namespace xe::cpu;
|
using namespace xe::cpu;
|
||||||
|
|
||||||
XUserModule::XUserModule(KernelState* kernel_state, const char* path)
|
XUserModule::XUserModule(KernelState* kernel_state, const char* path)
|
||||||
: XModule(kernel_state, path), xex_(nullptr), execution_info_ptr_(0) {}
|
: XModule(kernel_state, ModuleType::kUserModule, path),
|
||||||
|
xex_(nullptr),
|
||||||
|
execution_info_ptr_(0) {}
|
||||||
|
|
||||||
XUserModule::~XUserModule() {
|
XUserModule::~XUserModule() {
|
||||||
kernel_state()->memory()->SystemHeapFree(execution_info_ptr_);
|
kernel_state()->memory()->SystemHeapFree(execution_info_ptr_);
|
||||||
|
@ -119,6 +121,7 @@ X_STATUS XUserModule::LoadFromMemory(const void* addr, const size_t length) {
|
||||||
if (!xex_module->Load(name_, path_, xex_)) {
|
if (!xex_module->Load(name_, path_, xex_)) {
|
||||||
return X_STATUS_UNSUCCESSFUL;
|
return X_STATUS_UNSUCCESSFUL;
|
||||||
}
|
}
|
||||||
|
processor_module_ = xex_module.get();
|
||||||
if (!processor->AddModule(std::move(xex_module))) {
|
if (!processor->AddModule(std::move(xex_module))) {
|
||||||
return X_STATUS_UNSUCCESSFUL;
|
return X_STATUS_UNSUCCESSFUL;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue