2019-02-12 22:13:09 -05:00
|
|
|
#include "stdafx.h"
|
|
|
|
#include <regex>
|
2019-02-13 13:32:21 -05:00
|
|
|
#include <algorithm>
|
2019-02-12 22:13:09 -05:00
|
|
|
#include "TraceLogger.h"
|
|
|
|
#include "DisassemblyInfo.h"
|
|
|
|
#include "Console.h"
|
2020-02-07 22:55:27 -05:00
|
|
|
#include "EmuSettings.h"
|
2019-02-12 22:13:09 -05:00
|
|
|
#include "Debugger.h"
|
|
|
|
#include "MemoryManager.h"
|
2019-04-28 22:19:52 -04:00
|
|
|
#include "LabelManager.h"
|
2019-07-30 22:34:52 -04:00
|
|
|
#include "DebugUtilities.h"
|
2019-07-14 21:45:12 -04:00
|
|
|
#include "CpuTypes.h"
|
|
|
|
#include "SpcTypes.h"
|
|
|
|
#include "NecDspTypes.h"
|
2019-02-12 22:13:09 -05:00
|
|
|
#include "../Utilities/HexUtilities.h"
|
|
|
|
|
|
|
|
string TraceLogger::_executionTrace = "";
|
|
|
|
|
2019-03-07 20:12:32 -05:00
|
|
|
TraceLogger::TraceLogger(Debugger* debugger, shared_ptr<Console> console)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2019-07-25 22:22:09 -04:00
|
|
|
_console = console.get();
|
2020-02-07 22:55:27 -05:00
|
|
|
_settings = console->GetSettings().get();
|
2019-07-25 22:22:09 -04:00
|
|
|
_labelManager = debugger->GetLabelManager().get();
|
|
|
|
_memoryDumper = debugger->GetMemoryDumper().get();
|
2020-05-18 16:10:53 -04:00
|
|
|
_options = {};
|
2019-02-12 22:13:09 -05:00
|
|
|
_currentPos = 0;
|
|
|
|
_logCount = 0;
|
|
|
|
_logToFile = false;
|
|
|
|
_pendingLog = false;
|
2019-07-18 17:24:32 -04:00
|
|
|
|
|
|
|
_stateCache = new DebugState[TraceLogger::ExecutionLogSize];
|
|
|
|
_stateCacheCopy = new DebugState[TraceLogger::ExecutionLogSize];
|
|
|
|
_disassemblyCache = new DisassemblyInfo[TraceLogger::ExecutionLogSize];
|
|
|
|
_disassemblyCacheCopy = new DisassemblyInfo[TraceLogger::ExecutionLogSize];
|
2019-07-25 22:22:09 -04:00
|
|
|
_logCpuType = new CpuType[TraceLogger::ExecutionLogSize];
|
|
|
|
_logCpuTypeCopy = new CpuType[TraceLogger::ExecutionLogSize];
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
TraceLogger::~TraceLogger()
|
|
|
|
{
|
|
|
|
StopLogging();
|
2020-02-08 15:54:09 -05:00
|
|
|
|
|
|
|
delete[] _stateCache;
|
|
|
|
delete[] _stateCacheCopy;
|
|
|
|
delete[] _disassemblyCache;
|
|
|
|
delete[] _disassemblyCacheCopy;
|
|
|
|
delete[] _logCpuType;
|
|
|
|
delete[] _logCpuTypeCopy;
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
template<typename T>
|
|
|
|
void TraceLogger::WriteValue(string &output, T value, RowPart& rowPart)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2019-02-13 13:32:21 -05:00
|
|
|
string str = rowPart.DisplayInHex ? HexUtilities::ToHex(value) : std::to_string(value);
|
2019-02-12 22:13:09 -05:00
|
|
|
output += str;
|
2021-03-10 11:13:28 -05:00
|
|
|
if(rowPart.MinWidth > (int)str.size()) {
|
2019-02-12 22:13:09 -05:00
|
|
|
output += std::string(rowPart.MinWidth - str.size(), ' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
template<>
|
|
|
|
void TraceLogger::WriteValue(string &output, string value, RowPart& rowPart)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
|
|
|
output += value;
|
2021-03-10 11:13:28 -05:00
|
|
|
if(rowPart.MinWidth > (int)value.size()) {
|
2019-02-12 22:13:09 -05:00
|
|
|
output += std::string(rowPart.MinWidth - value.size(), ' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TraceLogger::SetOptions(TraceLoggerOptions options)
|
|
|
|
{
|
|
|
|
_options = options;
|
2021-03-10 11:13:28 -05:00
|
|
|
|
2019-04-07 16:10:23 -04:00
|
|
|
_logCpu[(int)CpuType::Cpu] = options.LogCpu;
|
|
|
|
_logCpu[(int)CpuType::Spc] = options.LogSpc;
|
2019-07-14 21:45:12 -04:00
|
|
|
_logCpu[(int)CpuType::NecDsp] = options.LogNecDsp;
|
2019-07-25 22:22:09 -04:00
|
|
|
_logCpu[(int)CpuType::Sa1] = options.LogSa1;
|
2019-07-30 22:34:52 -04:00
|
|
|
_logCpu[(int)CpuType::Gsu] = options.LogGsu;
|
2019-08-03 23:43:51 -04:00
|
|
|
_logCpu[(int)CpuType::Cx4] = options.LogCx4;
|
2020-05-18 16:10:53 -04:00
|
|
|
_logCpu[(int)CpuType::Gameboy] = options.LogGameboy;
|
2019-04-07 16:10:23 -04:00
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
string condition = _options.Condition;
|
|
|
|
string format = _options.Format;
|
|
|
|
|
|
|
|
auto lock = _lock.AcquireSafe();
|
|
|
|
/*_conditionData = ExpressionData();
|
|
|
|
if(!condition.empty()) {
|
|
|
|
bool success = false;
|
|
|
|
ExpressionData rpnList = _expEvaluator->GetRpnList(condition, success);
|
|
|
|
if(success) {
|
|
|
|
_conditionData = rpnList;
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
|
2019-04-06 17:38:14 -04:00
|
|
|
ParseFormatString(_rowParts, format);
|
2021-03-10 11:13:28 -05:00
|
|
|
ParseFormatString(_spcRowParts, "[PC,4h] [ByteCode,11h] [Disassembly][EffectiveAddress] [MemoryValue,h][Align,48] A:[A,2h] X:[X,2h] Y:[Y,2h] S:[SP,2h] P:[P,8] H:[Cycle,3] V:[Scanline,3]");
|
|
|
|
ParseFormatString(_dspRowParts, "[PC,4h] [ByteCode,11h] [Disassembly] [Align,65] [A,2h] S:[SP,2h] H:[Cycle,3] V:[Scanline,3]");
|
|
|
|
ParseFormatString(_gsuRowParts, "[PC,6h] [ByteCode,11h] [Disassembly] [Align,50] SRC:[X,2] DST:[Y,2] R0:[A,2h] H:[Cycle,3] V:[Scanline,3]");
|
|
|
|
ParseFormatString(_cx4RowParts, "[PC,6h] [ByteCode,11h] [Disassembly] [Align,45] [A,2h] H:[Cycle,3] V:[Scanline,3]");
|
|
|
|
ParseFormatString(_gbRowParts, "[PC,6h] [ByteCode,11h] [Disassembly] [Align,45] A:[A,2h] B:[B,2h] C:[C,2h] D:[D,2h] E:[E,2h] HL:[H,2h][L,2h] F:[F,2h] SP:[SP,4h] CYC:[Cycle,3] LY:[Scanline,3]");
|
2019-04-06 17:38:14 -04:00
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::ParseFormatString(vector<RowPart> &rowParts, string format)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
|
|
|
rowParts.clear();
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
std::regex formatRegex = std::regex("(\\[\\s*([^[]*?)\\s*(,\\s*([\\d]*)\\s*(h){0,1}){0,1}\\s*\\])|([^[]*)", std::regex_constants::icase);
|
2019-02-12 22:13:09 -05:00
|
|
|
std::sregex_iterator start = std::sregex_iterator(format.cbegin(), format.cend(), formatRegex);
|
|
|
|
std::sregex_iterator end = std::sregex_iterator();
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
for(std::sregex_iterator it = start; it != end; it++) {
|
2019-02-12 22:13:09 -05:00
|
|
|
const std::smatch& match = *it;
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(match.str(1) == "") {
|
2019-02-12 22:13:09 -05:00
|
|
|
RowPart part = {};
|
|
|
|
part.DataType = RowDataType::Text;
|
|
|
|
part.Text = match.str(6);
|
2019-04-06 17:38:14 -04:00
|
|
|
rowParts.push_back(part);
|
2021-03-10 11:13:28 -05:00
|
|
|
} else {
|
2019-02-12 22:13:09 -05:00
|
|
|
RowPart part = {};
|
|
|
|
|
|
|
|
string dataType = match.str(2);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(dataType == "ByteCode") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::ByteCode;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "Disassembly") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::Disassembly;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "EffectiveAddress") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::EffectiveAddress;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "MemoryValue") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::MemoryValue;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "Align") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::Align;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "PC") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::PC;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "A") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::A;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "B") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::B;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "C") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::C;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "D") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::D;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "E") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::E;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "F") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::F;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "H") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::H;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "L") {
|
2020-05-18 16:10:53 -04:00
|
|
|
part.DataType = RowDataType::L;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "X") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::X;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "Y") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::Y;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "D") {
|
2019-02-13 13:32:21 -05:00
|
|
|
part.DataType = RowDataType::D;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "DB") {
|
2019-02-13 13:32:21 -05:00
|
|
|
part.DataType = RowDataType::DB;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "P") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::PS;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "SP") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::SP;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "Cycle") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::Cycle;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "HClock") {
|
2019-06-29 11:34:24 -04:00
|
|
|
part.DataType = RowDataType::HClock;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "Scanline") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::Scanline;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "FrameCount") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::FrameCount;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(dataType == "CycleCount") {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::CycleCount;
|
2021-03-10 11:13:28 -05:00
|
|
|
} else {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.DataType = RowDataType::Text;
|
|
|
|
part.Text = "[Invalid tag]";
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(!match.str(4).empty()) {
|
|
|
|
try {
|
2019-02-12 22:13:09 -05:00
|
|
|
part.MinWidth = std::stoi(match.str(4));
|
2021-03-10 11:13:28 -05:00
|
|
|
} catch(std::exception&) {
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
part.DisplayInHex = match.str(5) == "h";
|
|
|
|
|
2019-04-06 17:38:14 -04:00
|
|
|
rowParts.push_back(part);
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TraceLogger::StartLogging(string filename)
|
|
|
|
{
|
|
|
|
_outputBuffer.clear();
|
|
|
|
_outputFile.open(filename, ios::out | ios::binary);
|
|
|
|
_logToFile = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TraceLogger::StopLogging()
|
|
|
|
{
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_logToFile) {
|
2019-02-12 22:13:09 -05:00
|
|
|
_logToFile = false;
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_outputFile) {
|
|
|
|
if(!_outputBuffer.empty()) {
|
2019-02-12 22:13:09 -05:00
|
|
|
_outputFile << _outputBuffer;
|
|
|
|
}
|
|
|
|
_outputFile.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::LogExtraInfo(const char *log, uint32_t cycleCount)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_logToFile && _options.ShowExtraInfo) {
|
2019-02-12 22:13:09 -05:00
|
|
|
//Flush current buffer
|
|
|
|
_outputFile << _outputBuffer;
|
|
|
|
_outputBuffer.clear();
|
2021-03-10 11:13:28 -05:00
|
|
|
_outputFile << "[" << log << " - Cycle: " << std::to_string(cycleCount) << "]" << (_options.UseWindowsEol ? "\r\n" : "\n");
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
template<CpuType cpuType>
|
|
|
|
void TraceLogger::GetStatusFlag(string &output, uint8_t ps, RowPart& part)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2021-03-10 11:13:28 -05:00
|
|
|
constexpr char cpuActiveStatusLetters[8] = { 'N', 'V', 'M', 'X', 'D', 'I', 'Z', 'C' };
|
|
|
|
constexpr char cpuInactiveStatusLetters[8] = { 'n', 'v', 'm', 'x', 'd', 'i', 'z', 'c' };
|
2019-04-07 16:10:23 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
constexpr char spcActiveStatusLetters[8] = { 'N', 'V', 'P', 'B', 'H', 'I', 'Z', 'C' };
|
|
|
|
constexpr char spcInactiveStatusLetters[8] = { 'n', 'v', 'p', 'b', 'h', 'i', 'z', 'c' };
|
2019-04-07 16:10:23 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
const char *activeStatusLetters = cpuType == CpuType::Cpu ? cpuActiveStatusLetters : spcActiveStatusLetters;
|
|
|
|
const char *inactiveStatusLetters = cpuType == CpuType::Cpu ? cpuInactiveStatusLetters : spcInactiveStatusLetters;
|
2019-04-07 16:10:23 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(part.DisplayInHex) {
|
2019-02-12 22:13:09 -05:00
|
|
|
WriteValue(output, ps, part);
|
2021-03-10 11:13:28 -05:00
|
|
|
} else {
|
2019-02-12 22:13:09 -05:00
|
|
|
string flags;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(int i = 0; i < 8; i++) {
|
|
|
|
if(ps & 0x80) {
|
2019-02-12 22:13:09 -05:00
|
|
|
flags += activeStatusLetters[i];
|
2021-03-10 11:13:28 -05:00
|
|
|
} else if(part.MinWidth >= 8) {
|
2019-02-12 22:13:09 -05:00
|
|
|
flags += inactiveStatusLetters[i];
|
|
|
|
}
|
|
|
|
ps <<= 1;
|
|
|
|
}
|
|
|
|
WriteValue(output, flags, part);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::WriteByteCode(DisassemblyInfo &info, RowPart &rowPart, string &output)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2019-04-06 17:38:14 -04:00
|
|
|
string byteCode;
|
|
|
|
info.GetByteCode(byteCode);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(!rowPart.DisplayInHex) {
|
2019-04-06 17:38:14 -04:00
|
|
|
//Remove $ marks if not in "hex" mode (but still display the bytes as hex)
|
|
|
|
byteCode.erase(std::remove(byteCode.begin(), byteCode.end(), '$'), byteCode.end());
|
|
|
|
}
|
|
|
|
WriteValue(output, byteCode, rowPart);
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::WriteDisassembly(DisassemblyInfo &info, RowPart &rowPart, uint8_t sp, uint32_t pc, string &output)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
|
|
|
int indentLevel = 0;
|
|
|
|
string code;
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_options.IndentCode) {
|
2019-04-06 17:38:14 -04:00
|
|
|
indentLevel = 0xFF - (sp & 0xFF);
|
|
|
|
code = std::string(indentLevel, ' ');
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2019-07-25 22:22:09 -04:00
|
|
|
LabelManager* labelManager = _options.UseLabels ? _labelManager : nullptr;
|
2020-02-07 22:55:27 -05:00
|
|
|
info.GetDisassembly(code, pc, labelManager, _settings);
|
2019-04-06 17:38:14 -04:00
|
|
|
WriteValue(output, code, rowPart);
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::WriteEffectiveAddress(DisassemblyInfo &info, RowPart &rowPart, void *cpuState, string &output, SnesMemoryType cpuMemoryType, CpuType cpuType)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
2020-02-27 19:59:41 -05:00
|
|
|
int32_t effectiveAddress = info.GetEffectiveAddress(_console, cpuState, cpuType);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(effectiveAddress >= 0) {
|
|
|
|
if(_options.UseLabels) {
|
|
|
|
AddressInfo addr { effectiveAddress, cpuMemoryType };
|
2019-04-28 22:19:52 -04:00
|
|
|
string label = _labelManager->GetLabel(addr);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(!label.empty()) {
|
2019-04-28 22:19:52 -04:00
|
|
|
WriteValue(output, " [" + label + "]", rowPart);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2019-04-06 17:38:14 -04:00
|
|
|
WriteValue(output, " [" + HexUtilities::ToHex24(effectiveAddress) + "]", rowPart);
|
|
|
|
}
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::WriteMemoryValue(DisassemblyInfo &info, RowPart &rowPart, void *cpuState, string &output, SnesMemoryType memType, CpuType cpuType)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
2020-02-27 19:59:41 -05:00
|
|
|
int32_t address = info.GetEffectiveAddress(_console, cpuState, cpuType);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(address >= 0) {
|
2019-04-06 17:38:14 -04:00
|
|
|
uint8_t valueSize;
|
2019-07-25 22:22:09 -04:00
|
|
|
uint16_t value = info.GetMemoryValue(address, _memoryDumper, memType, valueSize);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(rowPart.DisplayInHex) {
|
2019-04-06 17:38:14 -04:00
|
|
|
output += "= $";
|
2021-03-10 11:13:28 -05:00
|
|
|
if(valueSize == 2) {
|
2019-04-06 17:38:14 -04:00
|
|
|
WriteValue(output, (uint16_t)value, rowPart);
|
2021-03-10 11:13:28 -05:00
|
|
|
} else {
|
2019-04-06 17:38:14 -04:00
|
|
|
WriteValue(output, (uint8_t)value, rowPart);
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
2021-03-10 11:13:28 -05:00
|
|
|
} else {
|
2019-04-06 17:38:14 -04:00
|
|
|
output += "= ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::WriteAlign(int originalSize, RowPart &rowPart, string &output)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
2021-03-10 11:13:28 -05:00
|
|
|
if((int)output.size() - originalSize < rowPart.MinWidth) {
|
2019-04-06 17:38:14 -04:00
|
|
|
output += std::string(rowPart.MinWidth - (output.size() - originalSize), ' ');
|
|
|
|
}
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string &output, CpuState &cpuState, PpuState &ppuState, DisassemblyInfo &disassemblyInfo, SnesMemoryType memType, CpuType cpuType)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
|
|
|
int originalSize = (int)output.size();
|
|
|
|
uint32_t pcAddress = (cpuState.K << 16) | cpuState.PC;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(RowPart& rowPart : _rowParts) {
|
|
|
|
switch(rowPart.DataType) {
|
|
|
|
case RowDataType::Text: output += rowPart.Text; break;
|
|
|
|
case RowDataType::ByteCode: WriteByteCode(disassemblyInfo, rowPart, output); break;
|
|
|
|
case RowDataType::Disassembly: WriteDisassembly(disassemblyInfo, rowPart, (uint8_t)cpuState.SP, pcAddress, output); break;
|
|
|
|
case RowDataType::EffectiveAddress: WriteEffectiveAddress(disassemblyInfo, rowPart, &cpuState, output, memType, cpuType); break;
|
|
|
|
case RowDataType::MemoryValue: WriteMemoryValue(disassemblyInfo, rowPart, &cpuState, output, memType, cpuType); break;
|
|
|
|
case RowDataType::Align: WriteAlign(originalSize, rowPart, output); break;
|
|
|
|
|
|
|
|
case RowDataType::PC: WriteValue(output, HexUtilities::ToHex24(pcAddress), rowPart); break;
|
|
|
|
case RowDataType::A: WriteValue(output, cpuState.A, rowPart); break;
|
|
|
|
case RowDataType::X: WriteValue(output, cpuState.X, rowPart); break;
|
|
|
|
case RowDataType::Y: WriteValue(output, cpuState.Y, rowPart); break;
|
|
|
|
case RowDataType::D: WriteValue(output, cpuState.D, rowPart); break;
|
|
|
|
case RowDataType::DB: WriteValue(output, cpuState.DBR, rowPart); break;
|
|
|
|
case RowDataType::SP: WriteValue(output, cpuState.SP, rowPart); break;
|
|
|
|
case RowDataType::PS: GetStatusFlag<CpuType::Cpu>(output, cpuState.PS, rowPart); break;
|
|
|
|
case RowDataType::Cycle: WriteValue(output, ppuState.Cycle, rowPart); break;
|
|
|
|
case RowDataType::Scanline: WriteValue(output, ppuState.Scanline, rowPart); break;
|
|
|
|
case RowDataType::HClock: WriteValue(output, ppuState.HClock, rowPart); break;
|
|
|
|
case RowDataType::FrameCount: WriteValue(output, ppuState.FrameCount, rowPart); break;
|
|
|
|
case RowDataType::CycleCount: WriteValue(output, (uint32_t)cpuState.CycleCount, rowPart); break;
|
|
|
|
default: break;
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
output += _options.UseWindowsEol ? "\r\n" : "\n";
|
|
|
|
}
|
2019-04-06 17:38:14 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string &output, SpcState &cpuState, PpuState &ppuState, DisassemblyInfo &disassemblyInfo)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
|
|
|
int originalSize = (int)output.size();
|
|
|
|
uint32_t pcAddress = cpuState.PC;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(RowPart& rowPart : _spcRowParts) {
|
|
|
|
switch(rowPart.DataType) {
|
|
|
|
case RowDataType::Text: output += rowPart.Text; break;
|
|
|
|
case RowDataType::ByteCode: WriteByteCode(disassemblyInfo, rowPart, output); break;
|
|
|
|
case RowDataType::Disassembly: WriteDisassembly(disassemblyInfo, rowPart, cpuState.SP, pcAddress, output); break;
|
|
|
|
case RowDataType::EffectiveAddress: WriteEffectiveAddress(disassemblyInfo, rowPart, &cpuState, output, SnesMemoryType::SpcMemory, CpuType::Spc); break;
|
|
|
|
case RowDataType::MemoryValue: WriteMemoryValue(disassemblyInfo, rowPart, &cpuState, output, SnesMemoryType::SpcMemory, CpuType::Spc); break;
|
|
|
|
case RowDataType::Align: WriteAlign(originalSize, rowPart, output); break;
|
|
|
|
|
|
|
|
case RowDataType::PC: WriteValue(output, HexUtilities::ToHex((uint16_t)pcAddress), rowPart); break;
|
|
|
|
case RowDataType::A: WriteValue(output, cpuState.A, rowPart); break;
|
|
|
|
case RowDataType::X: WriteValue(output, cpuState.X, rowPart); break;
|
|
|
|
case RowDataType::Y: WriteValue(output, cpuState.Y, rowPart); break;
|
|
|
|
case RowDataType::SP: WriteValue(output, cpuState.SP, rowPart); break;
|
|
|
|
case RowDataType::PS: GetStatusFlag<CpuType::Spc>(output, cpuState.PS, rowPart); break;
|
|
|
|
case RowDataType::Cycle: WriteValue(output, ppuState.Cycle, rowPart); break;
|
|
|
|
case RowDataType::Scanline: WriteValue(output, ppuState.Scanline, rowPart); break;
|
|
|
|
case RowDataType::HClock: WriteValue(output, ppuState.HClock, rowPart); break;
|
|
|
|
case RowDataType::FrameCount: WriteValue(output, ppuState.FrameCount, rowPart); break;
|
|
|
|
|
|
|
|
default: break;
|
2019-04-06 17:38:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
output += _options.UseWindowsEol ? "\r\n" : "\n";
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string &output, NecDspState &cpuState, PpuState &ppuState, DisassemblyInfo &disassemblyInfo)
|
2019-07-14 21:45:12 -04:00
|
|
|
{
|
|
|
|
int originalSize = (int)output.size();
|
|
|
|
uint32_t pcAddress = cpuState.PC;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(RowPart& rowPart : _dspRowParts) {
|
|
|
|
switch(rowPart.DataType) {
|
|
|
|
case RowDataType::Text: output += rowPart.Text; break;
|
|
|
|
case RowDataType::ByteCode: WriteByteCode(disassemblyInfo, rowPart, output); break;
|
|
|
|
case RowDataType::Disassembly: WriteDisassembly(disassemblyInfo, rowPart, cpuState.SP, pcAddress, output); break;
|
|
|
|
case RowDataType::Align: WriteAlign(originalSize, rowPart, output); break;
|
|
|
|
|
|
|
|
case RowDataType::PC: WriteValue(output, HexUtilities::ToHex((uint16_t)pcAddress), rowPart); break;
|
|
|
|
case RowDataType::A:
|
|
|
|
output += "A:" + HexUtilities::ToHex(cpuState.A);
|
|
|
|
output += " B:" + HexUtilities::ToHex(cpuState.B);
|
|
|
|
output += " DR:" + HexUtilities::ToHex(cpuState.DR);
|
|
|
|
output += " DP:" + HexUtilities::ToHex(cpuState.DP);
|
|
|
|
output += " SR:" + HexUtilities::ToHex(cpuState.SR);
|
|
|
|
output += " K:" + HexUtilities::ToHex(cpuState.K);
|
|
|
|
output += " L:" + HexUtilities::ToHex(cpuState.L);
|
|
|
|
output += " M:" + HexUtilities::ToHex(cpuState.M);
|
|
|
|
output += " N:" + HexUtilities::ToHex(cpuState.N);
|
|
|
|
output += " RP:" + HexUtilities::ToHex(cpuState.RP);
|
|
|
|
output += " TR:" + HexUtilities::ToHex(cpuState.TR);
|
|
|
|
output += " TRB:" + HexUtilities::ToHex(cpuState.TRB) + " ";
|
|
|
|
//output += "FA=" + HexUtilities::ToHex(cpuState.FlagsA);
|
|
|
|
//output += "FB=" + HexUtilities::ToHex(cpuState.FlagsB);
|
|
|
|
WriteValue(output, cpuState.A, rowPart);
|
|
|
|
break;
|
|
|
|
case RowDataType::SP: WriteValue(output, cpuState.SP, rowPart); break;
|
|
|
|
case RowDataType::Cycle: WriteValue(output, ppuState.Cycle, rowPart); break;
|
|
|
|
case RowDataType::Scanline: WriteValue(output, ppuState.Scanline, rowPart); break;
|
|
|
|
case RowDataType::HClock: WriteValue(output, ppuState.HClock, rowPart); break;
|
|
|
|
case RowDataType::FrameCount: WriteValue(output, ppuState.FrameCount, rowPart); break;
|
|
|
|
default: break;
|
2019-07-14 21:45:12 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
output += _options.UseWindowsEol ? "\r\n" : "\n";
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string &output, GsuState &gsuState, PpuState &ppuState, DisassemblyInfo &disassemblyInfo)
|
2019-07-30 22:34:52 -04:00
|
|
|
{
|
|
|
|
int originalSize = (int)output.size();
|
|
|
|
uint32_t pcAddress = (gsuState.ProgramBank << 16) | gsuState.R[15];
|
2021-03-10 11:13:28 -05:00
|
|
|
for(RowPart& rowPart : _gsuRowParts) {
|
|
|
|
switch(rowPart.DataType) {
|
|
|
|
case RowDataType::Text: output += rowPart.Text; break;
|
|
|
|
case RowDataType::ByteCode: WriteByteCode(disassemblyInfo, rowPart, output); break;
|
|
|
|
case RowDataType::Disassembly: WriteDisassembly(disassemblyInfo, rowPart, 0, pcAddress, output); break;
|
|
|
|
case RowDataType::Align: WriteAlign(originalSize, rowPart, output); break;
|
|
|
|
|
|
|
|
case RowDataType::PC: WriteValue(output, HexUtilities::ToHex24(pcAddress), rowPart); break;
|
|
|
|
case RowDataType::A:
|
|
|
|
WriteValue(output, gsuState.R[0], rowPart);
|
|
|
|
for(int i = 1; i < 16; i++) {
|
|
|
|
output += " R" + std::to_string(i) + ":" + HexUtilities::ToHex(gsuState.R[i]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case RowDataType::X: WriteValue(output, gsuState.SrcReg, rowPart); break;
|
|
|
|
case RowDataType::Y: WriteValue(output, gsuState.DestReg, rowPart); break;
|
|
|
|
|
|
|
|
case RowDataType::Cycle: WriteValue(output, ppuState.Cycle, rowPart); break;
|
|
|
|
case RowDataType::Scanline: WriteValue(output, ppuState.Scanline, rowPart); break;
|
|
|
|
case RowDataType::HClock: WriteValue(output, ppuState.HClock, rowPart); break;
|
|
|
|
case RowDataType::FrameCount: WriteValue(output, ppuState.FrameCount, rowPart); break;
|
|
|
|
default: break;
|
2019-07-30 22:34:52 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
output += _options.UseWindowsEol ? "\r\n" : "\n";
|
|
|
|
}
|
|
|
|
|
2019-08-03 23:43:51 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string &output, Cx4State &cx4State, PpuState &ppuState, DisassemblyInfo &disassemblyInfo)
|
2019-08-03 23:43:51 -04:00
|
|
|
{
|
|
|
|
int originalSize = (int)output.size();
|
|
|
|
uint32_t pcAddress = (cx4State.Cache.Address[cx4State.Cache.Page] + (cx4State.PC * 2)) & 0xFFFFFF;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(RowPart& rowPart : _cx4RowParts) {
|
|
|
|
switch(rowPart.DataType) {
|
|
|
|
case RowDataType::Text: output += rowPart.Text; break;
|
|
|
|
case RowDataType::ByteCode: WriteByteCode(disassemblyInfo, rowPart, output); break;
|
|
|
|
case RowDataType::Disassembly: WriteDisassembly(disassemblyInfo, rowPart, 0, pcAddress, output); break;
|
|
|
|
case RowDataType::Align: WriteAlign(originalSize, rowPart, output); break;
|
|
|
|
|
|
|
|
case RowDataType::PC: WriteValue(output, HexUtilities::ToHex24(pcAddress), rowPart); break;
|
|
|
|
case RowDataType::A:
|
|
|
|
output += " A:" + HexUtilities::ToHex24(cx4State.A);
|
|
|
|
output += string(" ") + (cx4State.Carry ? "C" : "c") + (cx4State.Zero ? "Z" : "z") + (cx4State.Overflow ? "V" : "v") + (cx4State.Negative ? "N" : "n");
|
|
|
|
|
|
|
|
output += " PC:" + HexUtilities::ToHex(cx4State.PC);
|
|
|
|
output += " MAR:" + HexUtilities::ToHex24(cx4State.MemoryAddressReg);
|
|
|
|
output += " MDR:" + HexUtilities::ToHex24(cx4State.MemoryDataReg);
|
|
|
|
output += " DPR:" + HexUtilities::ToHex24(cx4State.DataPointerReg);
|
|
|
|
output += " ML:" + HexUtilities::ToHex24((uint32_t)cx4State.Mult & 0xFFFFFF);
|
|
|
|
output += " MH:" + HexUtilities::ToHex24((uint32_t)(cx4State.Mult >> 24) & 0xFFFFFF);
|
|
|
|
for(int i = 0; i < 16; i++) {
|
|
|
|
output += " R" + std::to_string(i) + ":" + HexUtilities::ToHex24(cx4State.Regs[i]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RowDataType::Cycle: WriteValue(output, ppuState.Cycle, rowPart); break;
|
|
|
|
case RowDataType::Scanline: WriteValue(output, ppuState.Scanline, rowPart); break;
|
|
|
|
case RowDataType::HClock: WriteValue(output, ppuState.HClock, rowPart); break;
|
|
|
|
case RowDataType::FrameCount: WriteValue(output, ppuState.FrameCount, rowPart); break;
|
|
|
|
default: break;
|
2019-08-03 23:43:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
output += _options.UseWindowsEol ? "\r\n" : "\n";
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string& output, GbCpuState& cpuState, GbPpuState& ppuState, DisassemblyInfo& disassemblyInfo)
|
2020-05-18 16:10:53 -04:00
|
|
|
{
|
|
|
|
int originalSize = (int)output.size();
|
|
|
|
uint32_t pcAddress = cpuState.PC;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(RowPart& rowPart : _gbRowParts) {
|
|
|
|
switch(rowPart.DataType) {
|
|
|
|
case RowDataType::Text: output += rowPart.Text; break;
|
|
|
|
case RowDataType::ByteCode: WriteByteCode(disassemblyInfo, rowPart, output); break;
|
|
|
|
case RowDataType::Disassembly: WriteDisassembly(disassemblyInfo, rowPart, (uint8_t)cpuState.SP, pcAddress, output); break;
|
|
|
|
case RowDataType::EffectiveAddress: WriteEffectiveAddress(disassemblyInfo, rowPart, &cpuState, output, SnesMemoryType::GameboyMemory, CpuType::Gameboy); break;
|
|
|
|
case RowDataType::MemoryValue: WriteMemoryValue(disassemblyInfo, rowPart, &cpuState, output, SnesMemoryType::GameboyMemory, CpuType::Gameboy); break;
|
|
|
|
case RowDataType::Align: WriteAlign(originalSize, rowPart, output); break;
|
|
|
|
|
|
|
|
case RowDataType::PC: WriteValue(output, HexUtilities::ToHex((uint16_t)pcAddress), rowPart); break;
|
|
|
|
case RowDataType::A: WriteValue(output, cpuState.A, rowPart); break;
|
|
|
|
case RowDataType::B: WriteValue(output, cpuState.B, rowPart); break;
|
|
|
|
case RowDataType::C: WriteValue(output, cpuState.C, rowPart); break;
|
|
|
|
case RowDataType::D: WriteValue(output, cpuState.D, rowPart); break;
|
|
|
|
case RowDataType::E: WriteValue(output, cpuState.E, rowPart); break;
|
|
|
|
case RowDataType::F: WriteValue(output, cpuState.Flags, rowPart); break;
|
|
|
|
case RowDataType::H: WriteValue(output, cpuState.H, rowPart); break;
|
|
|
|
case RowDataType::L: WriteValue(output, cpuState.L, rowPart); break;
|
|
|
|
case RowDataType::SP: WriteValue(output, cpuState.SP, rowPart); break;
|
|
|
|
case RowDataType::Cycle: WriteValue(output, ppuState.Cycle, rowPart); break;
|
|
|
|
case RowDataType::Scanline: WriteValue(output, ppuState.Scanline, rowPart); break;
|
|
|
|
case RowDataType::FrameCount: WriteValue(output, ppuState.FrameCount, rowPart); break;
|
|
|
|
|
|
|
|
default: break;
|
2020-05-18 16:10:53 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
output += _options.UseWindowsEol ? "\r\n" : "\n";
|
|
|
|
}
|
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
/*
|
|
|
|
bool TraceLogger::ConditionMatches(DebugState &state, DisassemblyInfo &disassemblyInfo, OperationInfo &operationInfo)
|
|
|
|
{
|
|
|
|
if(!_conditionData.RpnQueue.empty()) {
|
|
|
|
EvalResultType type;
|
|
|
|
if(!_expEvaluator->Evaluate(_conditionData, state, type, operationInfo)) {
|
|
|
|
if(operationInfo.OperationType == MemoryOperationType::ExecOpCode) {
|
|
|
|
//Condition did not match, keep state/disassembly info for instruction's subsequent cycles
|
|
|
|
_lastState = state;
|
|
|
|
_lastDisassemblyInfo = disassemblyInfo;
|
|
|
|
_pendingLog = true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
*/
|
2019-04-06 17:38:14 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::GetTraceRow(string &output, CpuType cpuType, DisassemblyInfo &disassemblyInfo, DebugState &state)
|
2019-04-06 17:38:14 -04:00
|
|
|
{
|
2021-03-10 11:13:28 -05:00
|
|
|
switch(cpuType) {
|
|
|
|
case CpuType::Cpu: GetTraceRow(output, state.Cpu, state.Ppu, disassemblyInfo, SnesMemoryType::CpuMemory, cpuType); break;
|
|
|
|
case CpuType::Spc: GetTraceRow(output, state.Spc, state.Ppu, disassemblyInfo); break;
|
|
|
|
case CpuType::NecDsp: GetTraceRow(output, state.NecDsp, state.Ppu, disassemblyInfo); break;
|
|
|
|
case CpuType::Sa1: GetTraceRow(output, state.Sa1.Cpu, state.Ppu, disassemblyInfo, SnesMemoryType::Sa1Memory, cpuType); break;
|
|
|
|
case CpuType::Gsu: GetTraceRow(output, state.Gsu, state.Ppu, disassemblyInfo); break;
|
|
|
|
case CpuType::Cx4: GetTraceRow(output, state.Cx4, state.Ppu, disassemblyInfo); break;
|
|
|
|
case CpuType::Gameboy: GetTraceRow(output, state.Gameboy.Cpu, state.Gameboy.Ppu, disassemblyInfo); break;
|
2019-04-06 17:38:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::AddRow(CpuType cpuType, DisassemblyInfo &disassemblyInfo, DebugState &state)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2019-07-25 22:22:09 -04:00
|
|
|
_logCpuType[_currentPos] = cpuType;
|
2019-02-12 22:13:09 -05:00
|
|
|
_disassemblyCache[_currentPos] = disassemblyInfo;
|
2019-04-06 17:38:14 -04:00
|
|
|
_stateCache[_currentPos] = state;
|
2019-02-12 22:13:09 -05:00
|
|
|
_pendingLog = false;
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_logCount < ExecutionLogSize) {
|
2019-02-12 22:13:09 -05:00
|
|
|
_logCount++;
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_logToFile) {
|
2019-07-25 22:22:09 -04:00
|
|
|
GetTraceRow(_outputBuffer, cpuType, _disassemblyCache[_currentPos], _stateCache[_currentPos]);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_outputBuffer.size() > 32768) {
|
2019-02-28 16:53:04 -05:00
|
|
|
_outputFile << _outputBuffer;
|
|
|
|
_outputBuffer.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
_currentPos = (_currentPos + 1) % ExecutionLogSize;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
void TraceLogger::LogNonExec(OperationInfo& operationInfo)
|
|
|
|
{
|
|
|
|
if(_pendingLog) {
|
|
|
|
auto lock = _lock.AcquireSafe();
|
|
|
|
if(ConditionMatches(_lastState, _lastDisassemblyInfo, operationInfo)) {
|
|
|
|
AddRow(_lastDisassemblyInfo, _lastState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
void TraceLogger::Log(CpuType cpuType, DebugState &state, DisassemblyInfo &disassemblyInfo)
|
2019-02-12 22:13:09 -05:00
|
|
|
{
|
2021-03-10 11:13:28 -05:00
|
|
|
if(_logCpu[(int)cpuType]) {
|
2019-07-14 21:45:12 -04:00
|
|
|
//For the sake of performance, only log data for the CPUs we're actively displaying/logging
|
|
|
|
auto lock = _lock.AcquireSafe();
|
|
|
|
//if(ConditionMatches(state, disassemblyInfo, operationInfo)) {
|
2021-03-10 11:13:28 -05:00
|
|
|
AddRow(cpuType, disassemblyInfo, state);
|
2019-07-14 21:45:12 -04:00
|
|
|
//}
|
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
|
2019-05-20 17:14:50 -04:00
|
|
|
void TraceLogger::Clear()
|
|
|
|
{
|
|
|
|
_logCount = 0;
|
|
|
|
}
|
|
|
|
|
2019-02-12 22:13:09 -05:00
|
|
|
const char* TraceLogger::GetExecutionTrace(uint32_t lineCount)
|
|
|
|
{
|
|
|
|
int startPos;
|
|
|
|
|
|
|
|
_executionTrace.clear();
|
|
|
|
{
|
|
|
|
auto lock = _lock.AcquireSafe();
|
|
|
|
lineCount = std::min(lineCount, _logCount);
|
2019-07-18 17:24:32 -04:00
|
|
|
memcpy(_stateCacheCopy, _stateCache, sizeof(DebugState) * TraceLogger::ExecutionLogSize);
|
|
|
|
memcpy(_disassemblyCacheCopy, _disassemblyCache, sizeof(DisassemblyInfo) * TraceLogger::ExecutionLogSize);
|
2019-07-25 22:22:09 -04:00
|
|
|
memcpy(_logCpuTypeCopy, _logCpuType, sizeof(CpuType) * TraceLogger::ExecutionLogSize);
|
2019-04-20 21:55:52 -04:00
|
|
|
startPos = (_currentPos > 0 ? _currentPos : TraceLogger::ExecutionLogSize) - 1;
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
|
2019-04-07 16:10:23 -04:00
|
|
|
bool enabled = false;
|
2021-03-10 11:13:28 -05:00
|
|
|
for(int i = 0; i <= (int)DebugUtilities::GetLastCpuType(); i++) {
|
2019-04-07 16:10:23 -04:00
|
|
|
enabled |= _logCpu[i];
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if(enabled && lineCount > 0) {
|
|
|
|
for(int i = 0; i < TraceLogger::ExecutionLogSize; i++) {
|
2019-04-20 21:55:52 -04:00
|
|
|
int index = (startPos - i);
|
2021-03-10 11:13:28 -05:00
|
|
|
if(index < 0) {
|
2019-04-20 21:55:52 -04:00
|
|
|
index = TraceLogger::ExecutionLogSize + index;
|
|
|
|
}
|
2019-04-06 17:38:14 -04:00
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
if((i > 0 && startPos == index) || !_disassemblyCacheCopy[index].IsInitialized()) {
|
2019-04-08 11:01:32 -04:00
|
|
|
//If the entire array was checked, or this element is not initialized, stop
|
2019-04-07 16:10:23 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-07-25 22:22:09 -04:00
|
|
|
CpuType cpuType = _logCpuTypeCopy[index];
|
2021-03-10 11:13:28 -05:00
|
|
|
if(!_logCpu[(int)cpuType]) {
|
2019-04-20 21:55:52 -04:00
|
|
|
//This line isn't for a CPU currently being logged
|
2019-04-07 16:10:23 -04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-03-10 11:13:28 -05:00
|
|
|
DebugState &state = _stateCacheCopy[index];
|
|
|
|
switch(cpuType) {
|
|
|
|
case CpuType::Cpu: _executionTrace += "\x2\x1" + HexUtilities::ToHex24((state.Cpu.K << 16) | state.Cpu.PC) + "\x1"; break;
|
|
|
|
case CpuType::Spc: _executionTrace += "\x3\x1" + HexUtilities::ToHex(state.Spc.PC) + "\x1"; break;
|
|
|
|
case CpuType::NecDsp: _executionTrace += "\x4\x1" + HexUtilities::ToHex(state.NecDsp.PC) + "\x1"; break;
|
|
|
|
case CpuType::Sa1: _executionTrace += "\x4\x1" + HexUtilities::ToHex24((state.Sa1.Cpu.K << 16) | state.Sa1.Cpu.PC) + "\x1"; break;
|
|
|
|
case CpuType::Gsu: _executionTrace += "\x4\x1" + HexUtilities::ToHex24((state.Gsu.ProgramBank << 16) | state.Gsu.R[15]) + "\x1"; break;
|
|
|
|
case CpuType::Cx4: _executionTrace += "\x4\x1" + HexUtilities::ToHex24((state.Cx4.Cache.Address[state.Cx4.Cache.Page] + (state.Cx4.PC * 2)) & 0xFFFFFF) + "\x1"; break;
|
|
|
|
case CpuType::Gameboy: _executionTrace += "\x4\x1" + HexUtilities::ToHex(state.Gameboy.Cpu.PC) + "\x1"; break;
|
2019-04-07 16:10:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
string byteCode;
|
|
|
|
_disassemblyCacheCopy[index].GetByteCode(byteCode);
|
|
|
|
_executionTrace += byteCode + "\x1";
|
2019-07-25 22:22:09 -04:00
|
|
|
GetTraceRow(_executionTrace, cpuType, _disassemblyCacheCopy[index], _stateCacheCopy[index]);
|
2019-04-20 21:55:52 -04:00
|
|
|
|
|
|
|
lineCount--;
|
2021-03-10 11:13:28 -05:00
|
|
|
if(lineCount == 0) {
|
2019-04-20 21:55:52 -04:00
|
|
|
break;
|
|
|
|
}
|
2019-04-06 17:38:14 -04:00
|
|
|
}
|
2019-02-12 22:13:09 -05:00
|
|
|
}
|
|
|
|
return _executionTrace.c_str();
|
2021-03-10 11:13:28 -05:00
|
|
|
}
|