mirror of
https://github.com/reactos/reactosdbg.git
synced 2024-11-23 11:49:53 +00:00
6030ecd415
locals and stack trace at the place where reactos is stopped. svn path=/trunk/tools/reactosdbg/; revision=759
264 lines
8.8 KiB
C#
264 lines
8.8 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Net;
|
|
using System.Net.Sockets;
|
|
using AbstractPipe;
|
|
using DebugProtocol;
|
|
using KDBGProtocol;
|
|
|
|
namespace RosDBG
|
|
{
|
|
public class DebugConnectedEventArgs : EventArgs
|
|
{
|
|
public DebugConnectedEventArgs()
|
|
{
|
|
}
|
|
}
|
|
|
|
public delegate void DebugConnectedEventHandler(object sender, DebugConnectedEventArgs args);
|
|
|
|
public class DebugConnectionModeChangedEventArgs : EventArgs
|
|
{
|
|
public readonly DebugConnection.Mode Mode;
|
|
public DebugConnectionModeChangedEventArgs(DebugConnection.Mode mode)
|
|
{
|
|
Mode = mode;
|
|
}
|
|
}
|
|
public delegate void DebugConnectionModeChangedEventHandler(object sender, DebugConnectionModeChangedEventArgs args);
|
|
|
|
public class DebugRegisterChangeEventArgs : EventArgs
|
|
{
|
|
public readonly Registers Registers;
|
|
public DebugRegisterChangeEventArgs(Registers regs)
|
|
{
|
|
Registers = regs;
|
|
}
|
|
}
|
|
public delegate void DebugRegisterChangeEventHandler(object sender, DebugRegisterChangeEventArgs args);
|
|
|
|
public class DebugRunningChangeEventArgs : EventArgs
|
|
{
|
|
public readonly bool Running;
|
|
public DebugRunningChangeEventArgs(bool running) { Running = running; }
|
|
}
|
|
public delegate void DebugRunningChangeEventHandler(object sender, DebugRunningChangeEventArgs args);
|
|
|
|
public class DebugModuleChangedEventArgs : EventArgs
|
|
{
|
|
public readonly uint ModuleAddr;
|
|
public readonly string ModuleName;
|
|
public DebugModuleChangedEventArgs(uint moduleAddr, string moduleName)
|
|
{
|
|
ModuleAddr = moduleAddr;
|
|
ModuleName = moduleName;
|
|
}
|
|
}
|
|
public delegate void DebugModuleChangedEventHandler(object sender, DebugModuleChangedEventArgs args);
|
|
|
|
public class DebugRawTrafficEventArgs : EventArgs
|
|
{
|
|
public readonly string Data;
|
|
public DebugRawTrafficEventArgs(string data) { Data = data; }
|
|
}
|
|
public delegate void DebugRawTrafficEventHandler(object sender, DebugRawTrafficEventArgs args);
|
|
|
|
public class DebugConnection
|
|
{
|
|
#region Primary State
|
|
public enum Mode { ClosedMode, SocketMode }
|
|
public Mode ConnectionMode
|
|
{
|
|
get { return mConnectionMode; }
|
|
set
|
|
{
|
|
mConnectionMode = value;
|
|
if (DebugConnectionModeChangedEvent != null)
|
|
DebugConnectionModeChangedEvent(this, new DebugConnectionModeChangedEventArgs(value));
|
|
}
|
|
}
|
|
public bool mRunning = true;
|
|
public bool Running
|
|
{
|
|
get { return mRunning; }
|
|
set
|
|
{
|
|
mRunning = value;
|
|
if (DebugRunningChangeEvent != null)
|
|
DebugRunningChangeEvent(this, new DebugRunningChangeEventArgs(value));
|
|
}
|
|
}
|
|
KDBG mKdb;
|
|
Registers mRegisters = new Registers();
|
|
public IDebugProtocol Debugger { get { return mKdb; } }
|
|
Pipe mMedium;
|
|
Mode mConnectionMode;
|
|
List<WeakReference> mMemoryReaders = new List<WeakReference>();
|
|
#endregion
|
|
|
|
#region Socket Mode State
|
|
int mRemotePort;
|
|
Socket mSocket;
|
|
SocketAsyncEventArgs mAsyncConnect;
|
|
AsyncCallback mDnsLookup;
|
|
IAsyncResult mDnsAsyncResult;
|
|
#endregion
|
|
|
|
public event DebugRegisterChangeEventHandler DebugRegisterChangeEvent;
|
|
public event DebugConnectedEventHandler DebugConnectionConnectedEvent;
|
|
public event DebugConnectionModeChangedEventHandler DebugConnectionModeChangedEvent;
|
|
public event DebugRunningChangeEventHandler DebugRunningChangeEvent;
|
|
public event DebugModuleChangedEventHandler DebugModuleChangedEvent;
|
|
public event DebugRawTrafficEventHandler DebugRawTrafficEvent;
|
|
|
|
public DebugConnection()
|
|
{
|
|
}
|
|
|
|
void DnsLookupResult(IAsyncResult result)
|
|
{
|
|
if (result.IsCompleted)
|
|
{
|
|
IPHostEntry entry = Dns.EndGetHostEntry(mDnsAsyncResult);
|
|
if (entry.AddressList.Length > 0)
|
|
{
|
|
int i;
|
|
// Check for an ipv4 target
|
|
for (i = 0; i < entry.AddressList.Length; i++)
|
|
if (entry.AddressList[i].AddressFamily == AddressFamily.InterNetwork)
|
|
break;
|
|
// Otherwise just fall back to the first one
|
|
if (i == entry.AddressList.Length) i = 0;
|
|
mAsyncConnect.RemoteEndPoint = new IPEndPoint(entry.AddressList[i], mRemotePort);
|
|
mSocket.ConnectAsync(mAsyncConnect);
|
|
}
|
|
else
|
|
{
|
|
Close();
|
|
}
|
|
}
|
|
}
|
|
|
|
public void Start(string host, int port)
|
|
{
|
|
Close();
|
|
mRemotePort = port;
|
|
ConnectionMode = Mode.SocketMode;
|
|
mSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
|
mAsyncConnect = new SocketAsyncEventArgs();
|
|
mAsyncConnect.UserToken = this;
|
|
mAsyncConnect.Completed += SocketConnectCompleted;
|
|
mDnsLookup = new AsyncCallback(DnsLookupResult);
|
|
mDnsAsyncResult = Dns.BeginGetHostEntry(host, mDnsLookup, this);
|
|
}
|
|
|
|
public void Close()
|
|
{
|
|
switch (ConnectionMode)
|
|
{
|
|
case Mode.SocketMode:
|
|
mSocket.Close();
|
|
mSocket = null;
|
|
break;
|
|
}
|
|
|
|
mMedium = null;
|
|
if (mKdb != null)
|
|
mKdb.Close();
|
|
mKdb = null;
|
|
ConnectionMode = Mode.ClosedMode;
|
|
}
|
|
|
|
void MediumError(object sender, PipeErrorEventArgs args)
|
|
{
|
|
Close();
|
|
}
|
|
|
|
void SocketConnectCompleted(object sender, SocketAsyncEventArgs e)
|
|
{
|
|
if (mAsyncConnect.SocketError == SocketError.Success)
|
|
{
|
|
mMedium = new SocketPipe(mSocket);
|
|
mMedium.PipeErrorEvent += MediumError;
|
|
mMedium.PipeReceiveEvent += PipeReceiveEvent;
|
|
mKdb = new KDBG(mMedium);
|
|
mKdb.RegisterChangeEvent += RegisterChangeEvent;
|
|
mKdb.ModuleListEvent += ModuleListEvent;
|
|
mKdb.MemoryUpdateEvent += MemoryUpdateEvent;
|
|
Running = true;
|
|
if (DebugConnectionConnectedEvent != null)
|
|
DebugConnectionConnectedEvent(this, new DebugConnectedEventArgs());
|
|
}
|
|
else
|
|
Close();
|
|
}
|
|
|
|
void MemoryUpdateEvent(object sender, MemoryUpdateEventArgs args)
|
|
{
|
|
List<WeakReference> deadStreams = new List<WeakReference>();
|
|
|
|
foreach (WeakReference memStreamRef in mMemoryReaders)
|
|
{
|
|
DebugMemoryStream memStream = memStreamRef.Target as DebugMemoryStream;
|
|
|
|
if (memStream == null)
|
|
{
|
|
deadStreams.Add(memStreamRef);
|
|
continue;
|
|
}
|
|
|
|
memStream.Update(args.Address, args.Memory);
|
|
}
|
|
}
|
|
|
|
void PipeReceiveEvent(object sender, PipeReceiveEventArgs args)
|
|
{
|
|
if (DebugRawTrafficEvent != null)
|
|
DebugRawTrafficEvent(this, new DebugRawTrafficEventArgs(args.Received));
|
|
}
|
|
|
|
void ModuleListEvent(object sender, ModuleListEventArgs args)
|
|
{
|
|
if (DebugModuleChangedEvent != null)
|
|
DebugModuleChangedEvent(this, new DebugModuleChangedEventArgs((uint)args.Address, args.Module));
|
|
}
|
|
|
|
void RegisterChangeEvent(object sender, RegisterChangeEventArgs args)
|
|
{
|
|
args.Registers.CopyTo(mRegisters.RegisterSet);
|
|
Running = false;
|
|
if (DebugRegisterChangeEvent != null)
|
|
DebugRegisterChangeEvent(this, new DebugRegisterChangeEventArgs(mRegisters));
|
|
}
|
|
|
|
public void Break()
|
|
{
|
|
if (mKdb != null)
|
|
mKdb.Break();
|
|
}
|
|
|
|
public void Step()
|
|
{
|
|
Running = true;
|
|
if (mKdb != null) mKdb.Step();
|
|
Running = false;
|
|
}
|
|
|
|
public void Go()
|
|
{
|
|
if (mKdb != null)
|
|
{
|
|
mKdb.Go(0);
|
|
Running = true;
|
|
}
|
|
}
|
|
|
|
public void NewMemoryStream(ulong baseAddr, int len)
|
|
{
|
|
mKdb.GetMemoryUpdate(baseAddr, len);
|
|
}
|
|
}
|
|
}
|