2015-07-02 03:17:14 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include <atomic>
|
2015-08-09 18:47:27 +00:00
|
|
|
#include <deque>
|
2016-11-20 18:15:37 +00:00
|
|
|
#include <unordered_set>
|
2015-07-02 03:17:14 +00:00
|
|
|
using std::atomic;
|
2015-08-09 18:47:27 +00:00
|
|
|
using std::deque;
|
2016-11-20 18:15:37 +00:00
|
|
|
using std::unordered_set;
|
2015-07-02 03:17:14 +00:00
|
|
|
|
|
|
|
#include "Breakpoint.h"
|
2016-01-11 00:56:40 +00:00
|
|
|
#include "TraceLogger.h"
|
2015-07-02 03:17:14 +00:00
|
|
|
#include "../Utilities/SimpleLock.h"
|
2015-08-17 23:32:10 +00:00
|
|
|
#include "CodeDataLogger.h"
|
2016-11-23 03:38:14 +00:00
|
|
|
#include "DebuggerTypes.h"
|
2015-07-02 03:17:14 +00:00
|
|
|
|
2016-06-18 00:53:05 +00:00
|
|
|
class CPU;
|
2017-08-30 22:31:27 +00:00
|
|
|
class APU;
|
2016-06-18 00:53:05 +00:00
|
|
|
class PPU;
|
2015-07-02 03:17:14 +00:00
|
|
|
class MemoryManager;
|
|
|
|
class Console;
|
2017-03-07 22:51:14 +00:00
|
|
|
class Assembler;
|
2015-07-02 03:17:14 +00:00
|
|
|
class Disassembler;
|
2016-11-23 03:38:14 +00:00
|
|
|
class LabelManager;
|
2016-12-02 00:38:48 +00:00
|
|
|
class MemoryDumper;
|
|
|
|
class MemoryAccessCounter;
|
2016-12-09 00:30:41 +00:00
|
|
|
class Profiler;
|
2017-03-12 02:03:45 +00:00
|
|
|
class CodeRunner;
|
2017-04-01 02:14:16 +00:00
|
|
|
class BaseMapper;
|
2017-08-30 22:31:27 +00:00
|
|
|
class ScriptHost;
|
|
|
|
class DebugHud;
|
2016-06-05 14:26:05 +00:00
|
|
|
|
2015-07-02 03:17:14 +00:00
|
|
|
class Debugger
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
static Debugger* Instance;
|
|
|
|
|
2016-11-19 02:32:07 +00:00
|
|
|
const static int BreakpointTypeCount = 6;
|
|
|
|
|
2017-09-29 01:27:38 +00:00
|
|
|
//Must be static to be thread-safe when switching game
|
|
|
|
static string _disassemblerOutput;
|
|
|
|
|
2017-03-11 20:01:33 +00:00
|
|
|
shared_ptr<Disassembler> _disassembler;
|
2017-03-07 22:51:14 +00:00
|
|
|
shared_ptr<Assembler> _assembler;
|
2016-11-19 04:50:05 +00:00
|
|
|
shared_ptr<MemoryDumper> _memoryDumper;
|
|
|
|
shared_ptr<CodeDataLogger> _codeDataLogger;
|
2016-12-02 00:38:48 +00:00
|
|
|
shared_ptr<MemoryAccessCounter> _memoryAccessCounter;
|
2016-12-09 00:30:41 +00:00
|
|
|
shared_ptr<LabelManager> _labelManager;
|
|
|
|
shared_ptr<TraceLogger> _traceLogger;
|
|
|
|
shared_ptr<Profiler> _profiler;
|
2017-03-12 02:03:45 +00:00
|
|
|
unique_ptr<CodeRunner> _codeRunner;
|
2017-08-30 22:31:27 +00:00
|
|
|
unique_ptr<DebugHud> _debugHud;
|
2016-11-19 04:50:05 +00:00
|
|
|
|
2015-07-02 03:17:14 +00:00
|
|
|
shared_ptr<Console> _console;
|
|
|
|
shared_ptr<CPU> _cpu;
|
|
|
|
shared_ptr<PPU> _ppu;
|
2017-08-30 22:31:27 +00:00
|
|
|
shared_ptr<APU> _apu;
|
2015-07-02 03:17:14 +00:00
|
|
|
shared_ptr<MemoryManager> _memoryManager;
|
|
|
|
shared_ptr<BaseMapper> _mapper;
|
2017-08-30 22:31:27 +00:00
|
|
|
|
|
|
|
bool _hasScript;
|
2017-11-11 00:04:22 +00:00
|
|
|
SimpleLock _scriptLock;
|
2017-08-30 22:31:27 +00:00
|
|
|
int _nextScriptId;
|
|
|
|
vector<shared_ptr<ScriptHost>> _scripts;
|
2016-01-09 18:15:43 +00:00
|
|
|
|
2016-11-19 02:32:07 +00:00
|
|
|
bool _bpUpdateNeeded;
|
|
|
|
SimpleLock _bpUpdateLock;
|
|
|
|
|
2017-10-07 17:01:42 +00:00
|
|
|
atomic<int32_t> _preventResume;
|
2016-06-04 12:55:52 +00:00
|
|
|
atomic<bool> _stopFlag;
|
2016-01-26 21:47:21 +00:00
|
|
|
atomic<bool> _executionStopped;
|
2016-06-04 12:55:52 +00:00
|
|
|
atomic<int32_t> _suspendCount;
|
2016-01-26 21:47:21 +00:00
|
|
|
vector<Breakpoint> _newBreakpoints;
|
2016-11-19 02:32:07 +00:00
|
|
|
vector<Breakpoint> _breakpoints[BreakpointTypeCount];
|
2017-08-05 16:13:53 +00:00
|
|
|
vector<vector<int>> _breakpointRpnList[BreakpointTypeCount];
|
2016-12-14 04:28:28 +00:00
|
|
|
bool _hasBreakpoint[BreakpointTypeCount] = {};
|
2016-01-09 18:15:43 +00:00
|
|
|
|
2017-03-03 01:33:25 +00:00
|
|
|
vector<uint8_t> _frozenAddresses;
|
|
|
|
|
2017-05-01 00:27:02 +00:00
|
|
|
bool _hideTopOfCallstack;
|
2015-08-09 18:47:27 +00:00
|
|
|
deque<uint32_t> _callstackAbsolute;
|
|
|
|
deque<uint32_t> _callstackRelative;
|
2015-07-02 03:17:14 +00:00
|
|
|
|
2016-11-20 18:15:37 +00:00
|
|
|
unordered_set<uint32_t> _functionEntryPoints;
|
|
|
|
|
2017-03-04 02:03:20 +00:00
|
|
|
ExpressionEvaluator _watchExpEval = ExpressionEvaluator(this);
|
|
|
|
ExpressionEvaluator _bpExpEval = ExpressionEvaluator(this);
|
2016-01-26 21:47:21 +00:00
|
|
|
DebugState _debugState;
|
|
|
|
|
2015-07-02 03:17:14 +00:00
|
|
|
SimpleLock _breakLock;
|
|
|
|
|
2016-10-23 14:16:08 +00:00
|
|
|
//Used to alter the executing address via "Set Next Statement"
|
|
|
|
uint16_t *_currentReadAddr;
|
|
|
|
uint8_t *_currentReadValue;
|
2017-03-12 02:03:45 +00:00
|
|
|
int32_t _nextReadAddr;
|
|
|
|
uint16_t _returnToAddress;
|
2015-08-18 01:59:22 +00:00
|
|
|
|
2016-06-05 14:26:05 +00:00
|
|
|
uint32_t _flags;
|
|
|
|
|
2016-06-18 00:53:05 +00:00
|
|
|
string _romName;
|
2015-08-10 00:45:45 +00:00
|
|
|
atomic<int32_t> _stepCount;
|
2016-06-04 19:38:48 +00:00
|
|
|
atomic<int32_t> _ppuStepCount;
|
2015-07-02 03:17:14 +00:00
|
|
|
atomic<int32_t> _stepCycleCount;
|
|
|
|
atomic<uint8_t> _lastInstruction;
|
|
|
|
atomic<bool> _stepOut;
|
|
|
|
atomic<int32_t> _stepOverAddr;
|
|
|
|
|
2016-11-27 01:44:23 +00:00
|
|
|
int32_t _ppuViewerScanline;
|
|
|
|
int32_t _ppuViewerCycle;
|
|
|
|
|
2016-12-02 23:10:37 +00:00
|
|
|
uint16_t _ppuScrollX;
|
|
|
|
uint16_t _ppuScrollY;
|
|
|
|
|
2017-08-02 02:49:50 +00:00
|
|
|
int32_t _prevInstructionCycle;
|
|
|
|
int32_t _curInstructionCycle;
|
|
|
|
int32_t _runToCycle;
|
|
|
|
bool _needRewind;
|
|
|
|
vector<stringstream> _rewindCache;
|
|
|
|
|
2017-08-06 03:26:21 +00:00
|
|
|
uint32_t _inputOverride[4];
|
|
|
|
|
2015-07-02 03:17:14 +00:00
|
|
|
private:
|
2016-01-26 21:47:21 +00:00
|
|
|
void UpdateBreakpoints();
|
|
|
|
|
2016-01-09 18:15:43 +00:00
|
|
|
void PrivateProcessPpuCycle();
|
2017-03-03 01:33:25 +00:00
|
|
|
bool PrivateProcessRamOperation(MemoryOperationType type, uint16_t &addr, uint8_t &value);
|
2017-09-08 21:20:40 +00:00
|
|
|
void PrivateProcessVramReadOperation(MemoryOperationType type, uint16_t addr, uint8_t &value);
|
|
|
|
void PrivateProcessVramWriteOperation(uint16_t addr, uint8_t &value);
|
2017-08-05 16:13:53 +00:00
|
|
|
bool HasMatchingBreakpoint(BreakpointType type, OperationInfo &operationInfo);
|
2016-11-22 05:14:49 +00:00
|
|
|
|
2015-08-17 23:32:10 +00:00
|
|
|
void UpdateCallstack(uint32_t addr);
|
2016-11-22 05:14:49 +00:00
|
|
|
void PrivateProcessInterrupt(uint16_t cpuAddr, uint16_t destCpuAddr, bool forNmi);
|
|
|
|
|
2015-08-17 23:32:10 +00:00
|
|
|
void ProcessStepConditions(uint32_t addr);
|
2015-07-02 03:17:14 +00:00
|
|
|
bool SleepUntilResume();
|
|
|
|
|
2017-08-02 22:43:38 +00:00
|
|
|
void RemoveExcessCallstackEntries();
|
|
|
|
|
2015-07-02 03:17:14 +00:00
|
|
|
public:
|
2017-08-30 22:31:27 +00:00
|
|
|
Debugger(shared_ptr<Console> console, shared_ptr<CPU> cpu, shared_ptr<PPU> ppu, shared_ptr<APU> apu, shared_ptr<MemoryManager> memoryManager, shared_ptr<BaseMapper> mapper);
|
2015-07-02 03:17:14 +00:00
|
|
|
~Debugger();
|
|
|
|
|
2016-06-05 14:26:05 +00:00
|
|
|
void SetFlags(uint32_t flags);
|
|
|
|
bool CheckFlag(DebuggerFlags flag);
|
|
|
|
|
2016-01-09 18:15:43 +00:00
|
|
|
void SetBreakpoints(Breakpoint breakpoints[], uint32_t length);
|
2016-11-23 03:38:14 +00:00
|
|
|
|
|
|
|
shared_ptr<LabelManager> GetLabelManager();
|
2016-11-22 23:28:59 +00:00
|
|
|
|
2017-03-10 04:50:20 +00:00
|
|
|
void GetFunctionEntryPoints(int32_t* entryPoints, int32_t maxCount);
|
|
|
|
int32_t GetFunctionEntryPointCount();
|
|
|
|
|
2015-08-09 18:47:27 +00:00
|
|
|
void GetCallstack(int32_t* callstackAbsolute, int32_t* callstackRelative);
|
2016-11-26 22:48:11 +00:00
|
|
|
|
2017-03-04 02:03:20 +00:00
|
|
|
void GetState(DebugState *state, bool includeMapperInfo = true);
|
2016-11-26 22:48:11 +00:00
|
|
|
void SetState(DebugState state);
|
2015-07-02 03:17:14 +00:00
|
|
|
|
2016-06-04 12:55:52 +00:00
|
|
|
void Suspend();
|
|
|
|
void Resume();
|
|
|
|
|
2016-06-04 19:38:48 +00:00
|
|
|
void PpuStep(uint32_t count = 1);
|
2017-10-07 17:01:42 +00:00
|
|
|
void Step(uint32_t count = 1);
|
2015-07-02 03:17:14 +00:00
|
|
|
void StepCycles(uint32_t cycleCount = 1);
|
|
|
|
void StepOver();
|
|
|
|
void StepOut();
|
2017-08-02 02:49:50 +00:00
|
|
|
void StepBack();
|
2015-07-02 03:17:14 +00:00
|
|
|
void Run();
|
|
|
|
|
2015-08-17 23:32:10 +00:00
|
|
|
bool LoadCdlFile(string cdlFilepath);
|
2017-08-20 02:00:12 +00:00
|
|
|
void ResetCdl();
|
|
|
|
void UpdateCdlCache();
|
2016-11-28 00:43:17 +00:00
|
|
|
bool IsMarkedAsCode(uint16_t relativeAddress);
|
2016-12-06 04:35:07 +00:00
|
|
|
shared_ptr<CodeDataLogger> GetCodeDataLogger();
|
2015-08-17 23:32:10 +00:00
|
|
|
|
2015-08-18 01:59:22 +00:00
|
|
|
void SetNextStatement(uint16_t addr);
|
2016-11-27 01:44:23 +00:00
|
|
|
void SetPpuViewerScanlineCycle(int32_t scanline, int32_t cycle);
|
2015-08-18 01:59:22 +00:00
|
|
|
|
2016-11-26 22:48:11 +00:00
|
|
|
bool IsExecutionStopped();
|
|
|
|
|
2017-10-07 17:01:42 +00:00
|
|
|
void PreventResume();
|
|
|
|
void AllowResume();
|
|
|
|
|
2017-03-07 22:51:14 +00:00
|
|
|
void GenerateCodeOutput();
|
2017-03-10 04:50:20 +00:00
|
|
|
const char* GetCode(uint32_t &length);
|
2016-01-10 18:23:19 +00:00
|
|
|
|
2016-11-23 03:38:14 +00:00
|
|
|
int32_t GetRelativeAddress(uint32_t addr, AddressType type);
|
|
|
|
int32_t GetAbsoluteAddress(uint32_t addr);
|
2017-08-14 00:45:36 +00:00
|
|
|
int32_t GetAbsoluteChrAddress(uint32_t addr);
|
2017-12-31 22:22:54 +00:00
|
|
|
int32_t GetRelativeChrAddress(uint32_t addr);
|
2016-11-23 03:38:14 +00:00
|
|
|
void GetAbsoluteAddressAndType(uint32_t relativeAddr, AddressTypeInfo* info);
|
2016-01-10 18:23:19 +00:00
|
|
|
|
2016-12-09 00:30:41 +00:00
|
|
|
shared_ptr<Profiler> GetProfiler();
|
2017-03-07 22:51:14 +00:00
|
|
|
shared_ptr<Assembler> GetAssembler();
|
2017-03-04 20:18:00 +00:00
|
|
|
shared_ptr<TraceLogger> GetTraceLogger();
|
2016-11-19 04:50:05 +00:00
|
|
|
shared_ptr<MemoryDumper> GetMemoryDumper();
|
2016-12-02 00:38:48 +00:00
|
|
|
shared_ptr<MemoryAccessCounter> GetMemoryAccessCounter();
|
2016-11-19 04:50:05 +00:00
|
|
|
|
2016-01-10 18:23:19 +00:00
|
|
|
int32_t EvaluateExpression(string expression, EvalResultType &resultType);
|
2015-07-02 03:17:14 +00:00
|
|
|
|
2017-03-03 01:33:25 +00:00
|
|
|
static bool ProcessRamOperation(MemoryOperationType type, uint16_t &addr, uint8_t &value);
|
2017-09-08 21:20:40 +00:00
|
|
|
static void ProcessVramReadOperation(MemoryOperationType type, uint16_t addr, uint8_t &value);
|
|
|
|
static void ProcessVramWriteOperation(uint16_t addr, uint8_t &value);
|
2016-01-09 18:15:43 +00:00
|
|
|
static void ProcessPpuCycle();
|
2017-08-30 22:31:27 +00:00
|
|
|
|
2017-09-29 00:05:00 +00:00
|
|
|
static void SetLastFramePpuScroll(uint16_t addr, uint8_t xScroll, bool updateHorizontalScrollOnly);
|
2016-12-02 23:10:37 +00:00
|
|
|
uint32_t GetPpuScroll();
|
|
|
|
|
2016-11-22 05:14:49 +00:00
|
|
|
static void ProcessInterrupt(uint16_t cpuAddr, uint16_t destCpuAddr, bool forNmi);
|
|
|
|
|
2016-06-05 14:26:05 +00:00
|
|
|
static bool IsEnabled();
|
2015-12-30 01:54:55 +00:00
|
|
|
static void BreakIfDebugging();
|
2016-12-02 00:38:48 +00:00
|
|
|
|
2017-03-03 01:33:25 +00:00
|
|
|
void SetFreezeState(uint16_t address, bool frozen);
|
|
|
|
void GetFreezeState(uint16_t startAddress, uint16_t length, bool* freezeState);
|
2017-03-05 03:24:41 +00:00
|
|
|
|
2017-03-12 02:03:45 +00:00
|
|
|
void StartCodeRunner(uint8_t *byteCode, uint32_t codeLength);
|
|
|
|
void StopCodeRunner();
|
|
|
|
|
2017-08-12 20:52:45 +00:00
|
|
|
void GetNesHeader(uint8_t* header);
|
2017-08-19 23:40:02 +00:00
|
|
|
void SaveRomToDisk(string filename, bool saveAsIps, uint8_t* header);
|
2017-08-22 03:11:14 +00:00
|
|
|
void RevertPrgChrChanges();
|
|
|
|
bool HasPrgChrChanges();
|
2017-03-10 01:42:53 +00:00
|
|
|
|
|
|
|
int32_t FindSubEntryPoint(uint16_t relativeAddress);
|
2017-08-06 03:26:21 +00:00
|
|
|
|
|
|
|
static bool HasInputOverride(uint8_t port);
|
|
|
|
static uint32_t GetInputOverride(uint8_t port);
|
|
|
|
void SetInputOverride(uint8_t port, uint32_t state);
|
2017-08-30 22:31:27 +00:00
|
|
|
|
2017-10-07 19:16:14 +00:00
|
|
|
int32_t LoadScript(string name, string content, int32_t scriptId);
|
2017-08-30 22:31:27 +00:00
|
|
|
void RemoveScript(int32_t scriptId);
|
|
|
|
const char* GetScriptLog(int32_t scriptId);
|
2017-10-05 23:44:37 +00:00
|
|
|
|
2017-10-06 00:48:18 +00:00
|
|
|
void ResetCounters();
|
|
|
|
|
2017-10-07 23:48:45 +00:00
|
|
|
void UpdateProgramCounter(uint16_t &addr, uint8_t &value);
|
|
|
|
|
2017-10-05 23:44:37 +00:00
|
|
|
void ProcessScriptSaveState(uint16_t &addr, uint8_t &value);
|
2017-10-07 23:48:45 +00:00
|
|
|
void ProcessCpuOperation(uint16_t &addr, uint8_t &value, MemoryOperationType type);
|
2017-09-08 21:20:40 +00:00
|
|
|
void ProcessPpuOperation(uint16_t addr, uint8_t &value, MemoryOperationType type);
|
2017-08-30 22:31:27 +00:00
|
|
|
void ProcessEvent(EventType type);
|
2015-07-02 03:17:14 +00:00
|
|
|
};
|