project64/Source/Project64/UserInterface/Debugger/Symbols.cpp

502 lines
12 KiB
C++
Raw Normal View History

2017-08-18 05:08:22 +00:00
#include "stdafx.h"
2022-09-21 05:16:07 +00:00
2017-08-18 05:08:22 +00:00
#include "Symbols.h"
2022-09-21 05:16:07 +00:00
CSymbolTable::CSymbolTable(CDebuggerUI * debugger) :
2019-12-25 00:41:20 +00:00
m_Debugger(debugger),
m_NextSymbolId(0),
2021-04-12 11:35:39 +00:00
m_SymFileBuffer(nullptr),
2019-12-25 00:41:20 +00:00
m_SymFileSize(0),
2021-04-12 11:35:39 +00:00
m_ParserToken(nullptr),
2019-12-25 00:41:20 +00:00
m_ParserTokenLength(0),
m_ParserDelimeter(0),
2021-04-12 11:35:39 +00:00
m_SymFileParseBuffer(nullptr),
2019-12-25 00:41:20 +00:00
m_bHaveFirstToken(false),
2021-04-12 11:35:39 +00:00
m_TokPos(nullptr)
2017-08-18 05:08:22 +00:00
{
}
2019-12-25 00:41:20 +00:00
CSymbolTable::~CSymbolTable()
2017-08-18 05:08:22 +00:00
{
2019-12-27 18:48:55 +00:00
delete[] m_SymFileBuffer;
delete[] m_SymFileParseBuffer;
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
symbol_type_info_t CSymbolTable::m_SymbolTypes[] = {
2022-09-21 05:16:07 +00:00
{SYM_CODE, "code", 1},
{SYM_DATA, "data", 1},
{SYM_U8, "u8", 1},
{SYM_U16, "u16", 2},
{SYM_U32, "u32", 4},
{SYM_U64, "u64", 8},
{SYM_S8, "s8", 1},
{SYM_S16, "s16", 2},
{SYM_S32, "s32", 4},
{SYM_S64, "s64", 8},
{SYM_FLOAT, "float", 4},
{SYM_DOUBLE, "double", 8},
{SYM_VECTOR2, "v2", 8},
{SYM_VECTOR3, "v3", 12},
{SYM_VECTOR4, "v4", 16},
2022-09-26 02:31:54 +00:00
{SYM_INVALID, nullptr, 0},
2019-12-25 00:41:20 +00:00
};
2022-09-21 05:16:07 +00:00
symbol_type_id_t CSymbolTable::GetTypeId(char * typeName)
2019-12-25 00:41:20 +00:00
{
2022-09-21 05:16:07 +00:00
const char * name;
2021-04-12 11:35:39 +00:00
for (int i = 0; (name = m_SymbolTypes[i].name) != nullptr; i++)
2019-12-25 00:41:20 +00:00
{
if (strcmp(typeName, name) == 0)
{
return (symbol_type_id_t)i;
}
}
return SYM_INVALID;
}
2017-08-18 05:08:22 +00:00
2022-09-21 05:16:07 +00:00
const char * CSymbolTable::GetTypeName(int typeId)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
if (typeId >= NUM_SYM_TYPES)
{
2021-04-12 11:35:39 +00:00
return nullptr;
2019-12-25 00:41:20 +00:00
}
return m_SymbolTypes[typeId].name;
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
int CSymbolTable::GetTypeSize(int typeId)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
if (typeId >= NUM_SYM_TYPES)
{
return NULL;
}
return m_SymbolTypes[typeId].size;
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
// Open symbols file for game and parse into list
CPath CSymbolTable::GetSymFilePath()
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
stdstr symFileName;
symFileName.Format("%s.sym", g_Settings->LoadStringVal(Game_GameName).c_str());
CPath symFilePath(g_Settings->LoadStringVal(Directory_NativeSave).c_str(), symFileName.c_str());
if (g_Settings->LoadBool(Setting_UniqueSaveDir))
{
symFilePath.AppendDirectory(g_Settings->LoadStringVal(Game_UniqueSaveDir).c_str());
}
symFilePath.NormalizePath(CPath(CPath::MODULE_DIRECTORY));
if (!symFilePath.DirectoryExists())
{
symFilePath.DirectoryCreate();
}
return symFilePath;
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
void CSymbolTable::ParserFetchToken(const char * delim)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
if (!m_bHaveFirstToken)
{
2021-04-12 11:35:39 +00:00
m_TokPos = nullptr;
2019-12-25 00:41:20 +00:00
m_ParserToken = strtok_s(m_SymFileParseBuffer, delim, &m_TokPos);
m_bHaveFirstToken = true;
}
else
{
2021-04-12 11:35:39 +00:00
m_ParserToken = strtok_s(nullptr, delim, &m_TokPos);
2019-12-25 00:41:20 +00:00
}
2022-09-21 05:16:07 +00:00
2021-04-12 11:35:39 +00:00
if (m_ParserToken != nullptr)
2019-12-25 00:41:20 +00:00
{
m_ParserTokenLength = strlen(m_ParserToken);
m_ParserDelimeter = m_SymFileBuffer[m_ParserToken - m_SymFileParseBuffer + m_ParserTokenLength];
}
else
{
m_ParserTokenLength = 0;
m_ParserDelimeter = '\0';
}
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
void CSymbolTable::Load()
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
m_AddressMap.clear();
2019-12-25 00:41:20 +00:00
m_NextSymbolId = 0;
m_Symbols.clear();
if (g_Settings->LoadStringVal(Game_GameName).length() == 0)
{
2021-04-12 11:35:39 +00:00
MessageBox(nullptr, L"Game must be loaded", L"Symbols", MB_ICONWARNING | MB_OK);
2019-12-25 00:41:20 +00:00
return;
}
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
CPath symFilePath = GetSymFilePath();
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
bool bOpened = m_SymFileHandle.Open(symFilePath, CFileBase::modeRead);
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
if (!bOpened)
{
return;
}
2022-09-21 05:16:07 +00:00
2021-04-12 11:35:39 +00:00
if (m_SymFileBuffer != nullptr)
2019-12-27 18:48:55 +00:00
{
delete[] m_SymFileBuffer;
}
2021-04-12 11:35:39 +00:00
if (m_SymFileParseBuffer != nullptr)
2019-12-27 18:48:55 +00:00
{
delete[] m_SymFileParseBuffer;
}
2019-12-25 00:41:20 +00:00
m_SymFileSize = m_SymFileHandle.GetLength();
2019-12-27 18:48:55 +00:00
m_SymFileBuffer = new char[m_SymFileSize + 1];
m_SymFileParseBuffer = new char[m_SymFileSize + 1];
2023-02-27 23:39:08 +00:00
m_SymFileHandle.Read(m_SymFileBuffer, (uint32_t)((UINT_PTR)m_SymFileSize));
2019-12-25 00:41:20 +00:00
m_SymFileHandle.Close();
m_SymFileBuffer[m_SymFileSize] = '\0';
2022-09-21 05:16:07 +00:00
2019-12-27 18:48:55 +00:00
strcpy(m_SymFileParseBuffer, m_SymFileBuffer);
m_bHaveFirstToken = false;
2019-12-25 00:41:20 +00:00
symbol_parse_error_t errorCode = ERR_SUCCESS;
int lineNumber = 1;
while (true)
{
uint32_t address = 0;
int type = 0;
2022-09-21 05:16:07 +00:00
char * name = nullptr;
char * description = nullptr;
2019-12-25 00:41:20 +00:00
// Address
ParserFetchToken(",\n\0");
2021-04-12 11:35:39 +00:00
if (m_ParserToken == nullptr || m_ParserTokenLength == 0)
2019-12-25 00:41:20 +00:00
{
// Empty line at the EOF
2019-12-25 00:41:20 +00:00
errorCode = ERR_SUCCESS;
break;
}
2022-09-21 05:16:07 +00:00
char * endptr;
2019-12-25 00:41:20 +00:00
address = (uint32_t)strtoull(m_ParserToken, &endptr, 16);
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
if (endptr == m_ParserToken)
{
errorCode = ERR_INVALID_ADDR;
break;
}
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
// Type
if (m_ParserDelimeter != ',')
{
errorCode = ERR_MISSING_FIELDS;
break;
}
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
ParserFetchToken(",\n\0");
type = GetTypeId(m_ParserToken);
if (type == -1)
{
errorCode = ERR_INVALID_TYPE;
break;
}
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
// Name
if (m_ParserDelimeter != ',')
{
errorCode = ERR_MISSING_FIELDS;
break;
}
ParserFetchToken(",\n\0");
name = m_ParserToken;
// Optional description
if (m_ParserDelimeter == ',')
{
ParserFetchToken("\n\0");
description = m_ParserToken;
}
2022-09-21 05:16:07 +00:00
2019-12-25 00:41:20 +00:00
// Add symbol object to the vector
AddSymbol(type, address, name, description, false);
2019-12-25 00:41:20 +00:00
if (m_ParserDelimeter == '\0')
{
errorCode = ERR_SUCCESS;
break;
}
lineNumber++;
}
sort(m_Symbols.begin(), m_Symbols.end(), CmpSymbolAddresses);
UpdateAddressMap();
2022-09-21 05:16:07 +00:00
2019-12-27 18:48:55 +00:00
delete[] m_SymFileParseBuffer;
2021-04-12 11:35:39 +00:00
m_SymFileParseBuffer = nullptr;
2019-12-27 18:48:55 +00:00
delete[] m_SymFileBuffer;
2021-04-12 11:35:39 +00:00
m_SymFileBuffer = nullptr;
2019-12-25 00:41:20 +00:00
switch (errorCode)
{
case ERR_SUCCESS:
break;
case ERR_INVALID_ADDR:
ParseErrorAlert("Invalid address", lineNumber);
break;
case ERR_INVALID_TYPE:
ParseErrorAlert("Invalid type", lineNumber);
break;
case ERR_INVALID_NAME:
ParseErrorAlert("Invalid name", lineNumber);
break;
case ERR_MISSING_FIELDS:
ParseErrorAlert("Missing required field(s)", lineNumber);
break;
}
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
void CSymbolTable::Save()
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
m_SymFileHandle.Open(GetSymFilePath(), CFileBase::modeCreate | CFileBase::modeReadWrite);
m_SymFileHandle.SeekToBegin();
for (size_t i = 0; i < m_Symbols.size(); i++)
{
2022-09-21 05:16:07 +00:00
CSymbol & symbol = m_Symbols[i];
2019-12-25 00:41:20 +00:00
stdstr strLine = stdstr_f("%08X,%s,%s", symbol.m_Address, symbol.TypeName(), symbol.m_Name);
2022-09-21 05:16:07 +00:00
2021-04-12 11:35:39 +00:00
if (symbol.m_Description != nullptr)
2019-12-25 00:41:20 +00:00
{
strLine += stdstr_f(",%s", symbol.m_Description);
}
strLine += "\n";
2023-02-27 23:39:08 +00:00
m_SymFileHandle.Write(strLine.c_str(), (uint32_t)((UINT_PTR)strLine.length()));
2019-12-25 00:41:20 +00:00
}
m_SymFileHandle.SetEndOfFile();
m_SymFileHandle.Close();
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
void CSymbolTable::GetValueString(char * dst, CSymbol * symbol)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
union
{
2022-09-21 05:16:07 +00:00
uint8_t u8;
int8_t s8;
2019-12-25 00:41:20 +00:00
uint16_t u16;
2022-09-21 05:16:07 +00:00
int16_t s16;
2019-12-25 00:41:20 +00:00
uint32_t u32;
2022-09-21 05:16:07 +00:00
int32_t s32;
2019-12-25 00:41:20 +00:00
uint64_t u64;
2022-09-21 05:16:07 +00:00
int64_t s64;
float f32;
double f64;
2019-12-25 00:41:20 +00:00
} value;
uint32_t address = symbol->m_Address;
float xyzw[4];
2019-12-25 00:41:20 +00:00
switch (symbol->m_Type)
{
case SYM_CODE:
case SYM_DATA:
sprintf(dst, "");
break;
case SYM_U8:
m_Debugger->DebugLoad_VAddr(address, value.u8);
sprintf(dst, "%u", value.u8);
break;
case SYM_U16:
m_Debugger->DebugLoad_VAddr(address, value.u16);
sprintf(dst, "%u", value.u16);
break;
case SYM_U32:
m_Debugger->DebugLoad_VAddr(address, value.u32);
sprintf(dst, "%u", value.u32);
break;
case SYM_U64:
m_Debugger->DebugLoad_VAddr(address, value.u64);
2023-11-15 22:33:32 +00:00
sprintf(dst, "%lld", value.u64);
2019-12-25 00:41:20 +00:00
break;
case SYM_S8:
m_Debugger->DebugLoad_VAddr(address, value.s8);
sprintf(dst, "%ihh", value.s8);
break;
case SYM_S16:
m_Debugger->DebugLoad_VAddr(address, value.s16);
sprintf(dst, "%i", value.s16);
break;
case SYM_S32:
m_Debugger->DebugLoad_VAddr(address, value.s32);
sprintf(dst, "%i", value.s32);
break;
case SYM_S64:
m_Debugger->DebugLoad_VAddr(address, value.s64);
sprintf(dst, "%I64i", value.s64);
break;
case SYM_FLOAT:
m_Debugger->DebugLoad_VAddr(address, value.f32);
sprintf(dst, "%f", value.f32);
break;
case SYM_DOUBLE:
m_Debugger->DebugLoad_VAddr(address, value.f64);
sprintf(dst, "%f", value.f64);
break;
case SYM_VECTOR2:
2022-09-21 05:16:07 +00:00
for (int i = 0; i < 2; i++)
{
m_Debugger->DebugLoad_VAddr(address + (i * sizeof(float)), value.f32);
xyzw[i] = value.f32;
}
sprintf(dst, "%f, %f", xyzw[0], xyzw[1]);
break;
case SYM_VECTOR3:
2022-09-21 05:16:07 +00:00
for (int i = 0; i < 3; i++)
{
m_Debugger->DebugLoad_VAddr(address + (i * sizeof(float)), value.f32);
xyzw[i] = value.f32;
}
sprintf(dst, "%f, %f, %f", xyzw[0], xyzw[1], xyzw[2]);
break;
case SYM_VECTOR4:
2022-09-21 05:16:07 +00:00
for (int i = 0; i < 4; i++)
{
m_Debugger->DebugLoad_VAddr(address + (i * sizeof(float)), value.f32);
xyzw[i] = value.f32;
}
sprintf(dst, "%f, %f, %f, %f", xyzw[0], xyzw[1], xyzw[2], xyzw[3]);
break;
2019-12-25 00:41:20 +00:00
default:
g_Notify->BreakPoint(__FILE__, __LINE__);
break;
}
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
void CSymbolTable::ParseErrorAlert(char * message, int lineNumber)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
stdstr messageFormatted = stdstr_f("%s\nLine %d", message, lineNumber);
MessageBox(nullptr, messageFormatted.ToUTF16().c_str(), L"Symbol parse error", MB_OK | MB_ICONWARNING);
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
void CSymbolTable::Reset()
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
m_Symbols.clear();
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
bool CSymbolTable::CmpSymbolAddresses(CSymbol & a, CSymbol & b)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
return (a.m_Address < b.m_Address);
}
2017-08-18 05:08:22 +00:00
2022-09-21 05:16:07 +00:00
void CSymbolTable::AddSymbol(int type, uint32_t address, const char * name, const char * description, bool bSortAfter)
2019-12-25 00:41:20 +00:00
{
CGuard guard(m_CS);
2017-08-18 05:08:22 +00:00
2021-04-12 11:35:39 +00:00
if (name == nullptr || strlen(name) == 0)
2019-12-25 00:41:20 +00:00
{
return;
}
2017-08-18 05:08:22 +00:00
2021-04-12 11:35:39 +00:00
if (description == nullptr || strlen(description) == 0)
2019-12-25 00:41:20 +00:00
{
2021-04-12 11:35:39 +00:00
description = nullptr;
2019-12-25 00:41:20 +00:00
}
2017-08-18 05:08:22 +00:00
2019-12-25 00:41:20 +00:00
int id = m_NextSymbolId++;
2017-08-18 05:08:22 +00:00
2019-12-25 00:41:20 +00:00
CSymbol symbol = CSymbol(id, type, address, name, description);
m_AddressMap[address] = m_Symbols.size();
2019-12-25 00:41:20 +00:00
m_Symbols.push_back(symbol);
2017-08-18 05:08:22 +00:00
if (bSortAfter)
{
sort(m_Symbols.begin(), m_Symbols.end(), CmpSymbolAddresses);
UpdateAddressMap();
}
}
void CSymbolTable::UpdateAddressMap()
{
m_AddressMap.clear();
for (size_t i = 0; i < m_Symbols.size(); i++)
{
m_AddressMap[m_Symbols[i].m_Address] = i;
}
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
int CSymbolTable::GetCount()
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
2023-02-27 23:39:08 +00:00
return (int)((UINT_PTR)m_Symbols.size());
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
bool CSymbolTable::GetSymbolByIndex(size_t index, CSymbol * symbol)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
if (index < 0 || index >= m_Symbols.size())
{
return false;
}
*symbol = m_Symbols[index];
return true;
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
bool CSymbolTable::GetSymbolByAddress(uint32_t address, CSymbol * symbol)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
2017-08-18 05:08:22 +00:00
if (m_AddressMap.count(address) == 0)
2019-12-25 00:41:20 +00:00
{
return false;
2019-12-25 00:41:20 +00:00
}
size_t index = m_AddressMap[address];
*symbol = m_Symbols[index];
return true;
2017-08-18 05:08:22 +00:00
}
2022-09-21 05:16:07 +00:00
bool CSymbolTable::GetSymbolById(int id, CSymbol * symbol)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
for (size_t i = 0; i < m_Symbols.size(); i++)
{
if (m_Symbols[i].m_Id == id)
{
*symbol = m_Symbols[i];
return true;
}
}
return false;
2017-08-18 05:08:22 +00:00
}
2019-12-25 00:41:20 +00:00
bool CSymbolTable::RemoveSymbolById(int id)
2017-08-18 05:08:22 +00:00
{
2019-12-25 00:41:20 +00:00
CGuard guard(m_CS);
for (size_t i = 0; i < m_Symbols.size(); i++)
{
if (m_Symbols[i].m_Id == id)
{
m_Symbols.erase(m_Symbols.begin() + i);
UpdateAddressMap();
2019-12-25 00:41:20 +00:00
return true;
}
}
2019-12-25 00:41:20 +00:00
return false;
2017-08-18 05:08:22 +00:00
}