BizHawk/BizHawk.Emulation.Common/Base Implementations/CallbackBasedTraceBuffer.cs

99 lines
2.5 KiB
C#
Raw Normal View History

2017-04-15 19:53:02 +00:00
using System;
using System.Collections.Generic;
using BizHawk.Emulation.Common.IEmulatorExtensions;
namespace BizHawk.Emulation.Common
{
/// <summary>
/// An implementation of ITraceable that is implementation using only methods
/// from IDebuggable, IMemoryDomains, and IDisassemblable
/// Useful for ported cores that have these hooks but no trace logging hook,
/// This allows for a traceable implementation without the need for additional API
/// Note that this technique will always be significantly slower than a direct implementation
/// </summary>
/// <seealso cref="ITraceable"/>
/// <seealso cref="IDebuggable"/>
/// <seealso cref="IMemoryDomains"/>
/// <seealso cref="IDisassemblable"/>
2017-04-15 19:53:02 +00:00
public abstract class CallbackBasedTraceBuffer : ITraceable
{
public CallbackBasedTraceBuffer(IDebuggable debuggableCore, IMemoryDomains memoryDomains, IDisassemblable disassembler)
{
if (!debuggableCore.MemoryCallbacksAvailable())
{
throw new InvalidOperationException("Memory callbacks are required");
}
try
{
debuggableCore.GetCpuFlagsAndRegisters();
}
catch (NotImplementedException)
{
throw new InvalidOperationException("GetCpuFlagsAndRegisters is required");
}
Header = "Instructions";
DebuggableCore = debuggableCore;
MemoryDomains = memoryDomains;
Disassembler = disassembler;
}
protected readonly IMemoryDomains MemoryDomains;
protected readonly IDisassemblable Disassembler;
protected readonly IDebuggable DebuggableCore;
protected readonly List<TraceInfo> Buffer = new List<TraceInfo>();
public abstract void TraceFromCallback();
2016-12-14 20:30:43 +00:00
private ITraceSink _sink;
2017-04-15 19:53:02 +00:00
public bool Enabled => Sink != null;
public void Put(TraceInfo info)
2016-12-14 20:30:43 +00:00
{
Sink.Put(info);
}
public ITraceSink Sink
{
get
{
return _sink;
}
set
{
_sink = value;
DebuggableCore.MemoryCallbacks.Remove(TraceFromCallback);
if (_sink != null)
{
DebuggableCore.MemoryCallbacks.Add(new TracingMemoryCallback(TraceFromCallback));
}
}
2017-04-15 19:53:02 +00:00
}
public string Header { get; set; }
public class TracingMemoryCallback : IMemoryCallback
{
public TracingMemoryCallback(Action callback)
{
Callback = callback;
}
public MemoryCallbackType Type => MemoryCallbackType.Execute;
public string Name => "Trace Logging";
public Action Callback { get; }
public uint? Address => null;
public uint? AddressMask => null;
}
}
}