|
|
|
@ -21,6 +21,7 @@
|
|
|
|
|
#include "Thread.h"
|
|
|
|
|
#include "FifoPlayer/FifoPlayer.h"
|
|
|
|
|
#include "FifoPlayer/FifoRecorder.h"
|
|
|
|
|
#include "OpcodeDecoding.h"
|
|
|
|
|
#include <wx/spinctrl.h>
|
|
|
|
|
|
|
|
|
|
DECLARE_EVENT_TYPE(RECORDING_FINISHED_EVENT, -1)
|
|
|
|
@ -36,7 +37,7 @@ wxEvtHandler *volatile FifoPlayerDlg::m_EvtHandler = NULL;
|
|
|
|
|
|
|
|
|
|
FifoPlayerDlg::FifoPlayerDlg(wxWindow * const parent) :
|
|
|
|
|
wxDialog(parent, wxID_ANY, _("FIFO Player"), wxDefaultPosition, wxDefaultSize),
|
|
|
|
|
m_FramesToRecord(1)
|
|
|
|
|
m_search_result_idx(0), m_FramesToRecord(1)
|
|
|
|
|
{
|
|
|
|
|
CreateGUIControls();
|
|
|
|
|
|
|
|
|
@ -65,6 +66,10 @@ FifoPlayerDlg::~FifoPlayerDlg()
|
|
|
|
|
m_FramesToRecordCtrl->Disconnect(wxEVT_COMMAND_SPINCTRL_UPDATED, wxSpinEventHandler(FifoPlayerDlg::OnNumFramesToRecord), NULL, this);
|
|
|
|
|
m_Close->Disconnect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnCloseClick), NULL, this);
|
|
|
|
|
|
|
|
|
|
m_framesList->Disconnect(wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler(FifoPlayerDlg::OnFrameListSelectionChanged), NULL, this);
|
|
|
|
|
m_objectsList->Disconnect(wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler(FifoPlayerDlg::OnObjectListSelectionChanged), NULL, this);
|
|
|
|
|
m_objectCmdList->Disconnect(wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler(FifoPlayerDlg::OnObjectCmdListSelectionChanged), NULL, this);
|
|
|
|
|
|
|
|
|
|
FifoPlayer::GetInstance().SetFrameWrittenCallback(NULL);
|
|
|
|
|
|
|
|
|
|
sMutex.lock();
|
|
|
|
@ -74,12 +79,12 @@ FifoPlayerDlg::~FifoPlayerDlg()
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::CreateGUIControls()
|
|
|
|
|
{
|
|
|
|
|
SetSizeHints(wxDefaultSize, wxDefaultSize);
|
|
|
|
|
|
|
|
|
|
wxBoxSizer* sMain;
|
|
|
|
|
sMain = new wxBoxSizer(wxVERTICAL);
|
|
|
|
|
|
|
|
|
|
m_Notebook = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0);
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
m_PlayPage = new wxPanel(m_Notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
|
|
|
|
|
wxBoxSizer* sPlayPage;
|
|
|
|
|
sPlayPage = new wxBoxSizer(wxVERTICAL);
|
|
|
|
@ -146,11 +151,15 @@ void FifoPlayerDlg::CreateGUIControls()
|
|
|
|
|
sPlayOptions->Add(m_EarlyMemoryUpdates, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
sPlayPage->Add(sPlayOptions, 0, wxEXPAND, 5);
|
|
|
|
|
|
|
|
|
|
sPlayPage->AddStretchSpacer();
|
|
|
|
|
|
|
|
|
|
m_PlayPage->SetSizer(sPlayPage);
|
|
|
|
|
m_PlayPage->Layout();
|
|
|
|
|
sPlayPage->Fit(m_PlayPage);
|
|
|
|
|
m_Notebook->AddPage(m_PlayPage, _("Play"), true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
m_RecordPage = new wxPanel(m_Notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
|
|
|
|
|
wxBoxSizer* sRecordPage;
|
|
|
|
|
sRecordPage = new wxBoxSizer(wxVERTICAL);
|
|
|
|
@ -194,33 +203,103 @@ void FifoPlayerDlg::CreateGUIControls()
|
|
|
|
|
sRecordingOptions->Add(m_FramesToRecordCtrl, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
sRecordPage->Add(sRecordingOptions, 0, wxEXPAND, 5);
|
|
|
|
|
|
|
|
|
|
sRecordPage->AddStretchSpacer();
|
|
|
|
|
|
|
|
|
|
m_RecordPage->SetSizer(sRecordPage);
|
|
|
|
|
m_RecordPage->Layout();
|
|
|
|
|
sRecordPage->Fit(m_RecordPage);
|
|
|
|
|
m_Notebook->AddPage(m_RecordPage, _("Record"), false);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Analyze page
|
|
|
|
|
{
|
|
|
|
|
m_AnalyzePage = new wxPanel(m_Notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
|
|
|
|
|
wxBoxSizer* sAnalyzePage;
|
|
|
|
|
sAnalyzePage = new wxBoxSizer(wxVERTICAL);
|
|
|
|
|
|
|
|
|
|
wxStaticBoxSizer* sFrameInfoSizer;
|
|
|
|
|
sFrameInfoSizer = new wxStaticBoxSizer(new wxStaticBox(m_AnalyzePage, wxID_ANY, _("Frame Info")), wxVERTICAL);
|
|
|
|
|
|
|
|
|
|
wxBoxSizer* sListsSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
|
|
|
|
|
m_framesList = new wxListBox(m_AnalyzePage, wxID_ANY);
|
|
|
|
|
m_framesList->SetMinSize(wxSize(100, 250));
|
|
|
|
|
sListsSizer->Add(m_framesList, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
m_objectsList = new wxListBox(m_AnalyzePage, wxID_ANY);
|
|
|
|
|
m_objectsList->SetMinSize(wxSize(110, 250));
|
|
|
|
|
sListsSizer->Add(m_objectsList, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
m_objectCmdList = new wxListBox(m_AnalyzePage, wxID_ANY);
|
|
|
|
|
m_objectCmdList->SetMinSize(wxSize(175, 250));
|
|
|
|
|
sListsSizer->Add(m_objectCmdList, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
sFrameInfoSizer->Add(sListsSizer, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
m_objectCmdInfo = new wxStaticText(m_AnalyzePage, wxID_ANY, wxString());
|
|
|
|
|
sFrameInfoSizer->Add(m_objectCmdInfo, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
sAnalyzePage->Add(sFrameInfoSizer, 0, wxEXPAND, 5);
|
|
|
|
|
|
|
|
|
|
wxStaticBoxSizer* sSearchSizer = new wxStaticBoxSizer(new wxStaticBox(m_AnalyzePage, wxID_ANY, _("Search current Object")), wxVERTICAL);
|
|
|
|
|
|
|
|
|
|
wxBoxSizer* sSearchField = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
|
|
|
|
|
sSearchField->Add(new wxStaticText(m_AnalyzePage, wxID_ANY, _("Search for hex Value:")), 0, wxALIGN_CENTER_VERTICAL, 5);
|
|
|
|
|
// TODO: ugh, wxValidator sucks - but we should use it anyway.
|
|
|
|
|
m_searchField = new wxTextCtrl(m_AnalyzePage, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER);
|
|
|
|
|
m_numResultsText = new wxStaticText(m_AnalyzePage, wxID_ANY, wxEmptyString);
|
|
|
|
|
|
|
|
|
|
sSearchField->Add(m_searchField, 0, wxALL, 5);
|
|
|
|
|
sSearchField->Add(m_numResultsText, 0, wxALIGN_CENTER_VERTICAL, 5);
|
|
|
|
|
|
|
|
|
|
wxBoxSizer* sSearchButtons = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
|
|
|
|
|
m_beginSearch = new wxButton(m_AnalyzePage, wxID_ANY, _("Search"));
|
|
|
|
|
m_findNext = new wxButton(m_AnalyzePage, wxID_ANY, _("Find next"));
|
|
|
|
|
m_findPrevious = new wxButton(m_AnalyzePage, wxID_ANY, _("Find previous"));
|
|
|
|
|
|
|
|
|
|
m_beginSearch->Disable();
|
|
|
|
|
m_findNext->Disable();
|
|
|
|
|
m_findPrevious->Disable();
|
|
|
|
|
|
|
|
|
|
sSearchButtons->Add(m_beginSearch, 0, wxALL, 5);
|
|
|
|
|
sSearchButtons->Add(m_findNext, 0, wxALL, 5);
|
|
|
|
|
sSearchButtons->Add(m_findPrevious, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
sSearchSizer->Add(sSearchField, 0, wxEXPAND, 5);
|
|
|
|
|
sSearchSizer->Add(sSearchButtons, 0, wxEXPAND, 5);
|
|
|
|
|
|
|
|
|
|
sAnalyzePage->Add(sSearchSizer, 0, wxEXPAND, 5);
|
|
|
|
|
sAnalyzePage->AddStretchSpacer();
|
|
|
|
|
|
|
|
|
|
m_AnalyzePage->SetSizer(sAnalyzePage);
|
|
|
|
|
m_AnalyzePage->Layout();
|
|
|
|
|
sAnalyzePage->Fit(m_AnalyzePage);
|
|
|
|
|
m_Notebook->AddPage(m_AnalyzePage, _("Analyze"), false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sMain->Add(m_Notebook, 1, wxEXPAND | wxALL, 5);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
wxBoxSizer* sButtons;
|
|
|
|
|
sButtons = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
wxBoxSizer* sCloseButtonExpander;
|
|
|
|
|
sCloseButtonExpander = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sButtons->Add(sCloseButtonExpander, 1, wxEXPAND, 5);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
m_Close = new wxButton(this, wxID_ANY, _("Close"), wxDefaultPosition, wxDefaultSize, 0);
|
|
|
|
|
sButtons->Add(m_Close, 0, wxALL, 5);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sMain->Add(sButtons, 0, wxEXPAND, 5);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
SetSizer(sMain);
|
|
|
|
|
Layout();
|
|
|
|
|
sMain->Fit(this);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Center(wxBOTH);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Connect Events
|
|
|
|
|
Connect(wxEVT_PAINT, wxPaintEventHandler(FifoPlayerDlg::OnPaint));
|
|
|
|
|
m_FrameFromCtrl->Connect(wxEVT_COMMAND_SPINCTRL_UPDATED, wxSpinEventHandler(FifoPlayerDlg::OnFrameFrom), NULL, this);
|
|
|
|
@ -231,7 +310,18 @@ void FifoPlayerDlg::CreateGUIControls()
|
|
|
|
|
m_RecordStop->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnRecordStop), NULL, this);
|
|
|
|
|
m_Save->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnSaveFile), NULL, this);
|
|
|
|
|
m_FramesToRecordCtrl->Connect(wxEVT_COMMAND_SPINCTRL_UPDATED, wxSpinEventHandler(FifoPlayerDlg::OnNumFramesToRecord), NULL, this);
|
|
|
|
|
m_Close->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnCloseClick), NULL, this);
|
|
|
|
|
Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnCloseClick), NULL, this);
|
|
|
|
|
|
|
|
|
|
m_framesList->Connect(wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler(FifoPlayerDlg::OnFrameListSelectionChanged), NULL, this);
|
|
|
|
|
m_objectsList->Connect(wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler(FifoPlayerDlg::OnObjectListSelectionChanged), NULL, this);
|
|
|
|
|
m_objectCmdList->Connect(wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEventHandler(FifoPlayerDlg::OnObjectCmdListSelectionChanged), NULL, this);
|
|
|
|
|
|
|
|
|
|
m_beginSearch->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnBeginSearch), NULL, this);
|
|
|
|
|
m_findNext->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnFindNextClick), NULL, this);
|
|
|
|
|
m_findPrevious->Connect(wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler(FifoPlayerDlg::OnFindPreviousClick), NULL, this);
|
|
|
|
|
|
|
|
|
|
m_searchField->Connect(wxEVT_COMMAND_TEXT_ENTER, wxCommandEventHandler(FifoPlayerDlg::OnBeginSearch), NULL, this);
|
|
|
|
|
m_searchField->Connect(wxEVT_COMMAND_TEXT_UPDATED, wxCommandEventHandler(FifoPlayerDlg::OnSearchFieldTextChanged), NULL, this);
|
|
|
|
|
|
|
|
|
|
Connect(RECORDING_FINISHED_EVENT, wxCommandEventHandler(FifoPlayerDlg::OnRecordingFinished), NULL, this);
|
|
|
|
|
Connect(FRAME_WRITTEN_EVENT, wxCommandEventHandler(FifoPlayerDlg::OnFrameWritten), NULL, this);
|
|
|
|
@ -243,6 +333,7 @@ void FifoPlayerDlg::OnPaint(wxPaintEvent& event)
|
|
|
|
|
{
|
|
|
|
|
UpdatePlayGui();
|
|
|
|
|
UpdateRecorderGui();
|
|
|
|
|
UpdateAnalyzerGui();
|
|
|
|
|
|
|
|
|
|
event.Skip();
|
|
|
|
|
}
|
|
|
|
@ -250,10 +341,11 @@ void FifoPlayerDlg::OnPaint(wxPaintEvent& event)
|
|
|
|
|
void FifoPlayerDlg::OnFrameFrom(wxSpinEvent& event)
|
|
|
|
|
{
|
|
|
|
|
FifoPlayer &player = FifoPlayer::GetInstance();
|
|
|
|
|
|
|
|
|
|
player.SetFrameRangeStart(event.GetPosition());
|
|
|
|
|
|
|
|
|
|
m_FrameFromCtrl->SetValue(player.GetFrameRangeStart());
|
|
|
|
|
m_FrameToCtrl->SetValue(player.GetFrameRangeEnd());
|
|
|
|
|
m_FrameToCtrl->SetValue(player.GetFrameRangeEnd());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnFrameTo(wxSpinEvent& event)
|
|
|
|
@ -326,6 +418,351 @@ void FifoPlayerDlg::OnNumFramesToRecord(wxSpinEvent& event)
|
|
|
|
|
m_FramesToRecord = -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnBeginSearch(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
wxString str_search_val = m_searchField->GetValue();
|
|
|
|
|
|
|
|
|
|
if (m_framesList->GetSelection() == -1)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// TODO: Limited to even str lengths...
|
|
|
|
|
if (str_search_val.Length() && str_search_val.Length() % 2)
|
|
|
|
|
{
|
|
|
|
|
m_numResultsText->SetLabel(_("Invalid search string (only even string lengths supported)"));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int val_length = str_search_val.Length() / 2;
|
|
|
|
|
u8* search_val = new u8[val_length];
|
|
|
|
|
for (unsigned int i = 0; i < val_length; ++i)
|
|
|
|
|
{
|
|
|
|
|
wxString char_str = str_search_val.Mid(2*i, 2);
|
|
|
|
|
unsigned long val;
|
|
|
|
|
if (!char_str.ToULong(&val, 16))
|
|
|
|
|
{
|
|
|
|
|
m_numResultsText->SetLabel(_("Invalid search string (couldn't convert to number)"));
|
|
|
|
|
delete[] search_val;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
search_val[i] = (u8)val;
|
|
|
|
|
}
|
|
|
|
|
search_results.clear();
|
|
|
|
|
|
|
|
|
|
u8* start_ptr;
|
|
|
|
|
u8* end_ptr;
|
|
|
|
|
|
|
|
|
|
int frame_idx = m_framesList->GetSelection();
|
|
|
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
|
|
|
const AnalyzedFrameInfo& frame = player.GetAnalyzedFrameInfo(frame_idx);
|
|
|
|
|
const FifoFrameInfo& fifo_frame = player.GetFile()->GetFrame(frame_idx);
|
|
|
|
|
|
|
|
|
|
// TODO: Support searching through the last object... How do we know were the cmd data ends?
|
|
|
|
|
// TODO: Support searching for bit patterns
|
|
|
|
|
int obj_idx = m_objectsList->GetSelection();
|
|
|
|
|
if (obj_idx == -1)
|
|
|
|
|
{
|
|
|
|
|
m_numResultsText->SetLabel(_("Invalid search parameters (no object selected)"));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
start_ptr = &fifo_frame.fifoData[frame.objectStarts[obj_idx]];
|
|
|
|
|
end_ptr = &fifo_frame.fifoData[frame.objectStarts[obj_idx+1]];
|
|
|
|
|
|
|
|
|
|
for (u8* ptr = start_ptr; ptr < end_ptr-val_length+1; ++ptr)
|
|
|
|
|
{
|
|
|
|
|
if (memcmp(ptr, search_val, val_length) == 0)
|
|
|
|
|
{
|
|
|
|
|
SearchResult result;
|
|
|
|
|
result.frame_idx = frame_idx;
|
|
|
|
|
|
|
|
|
|
int obj_idx = m_objectsList->GetSelection();
|
|
|
|
|
result.obj_idx = obj_idx;
|
|
|
|
|
result.cmd_idx = 0;
|
|
|
|
|
for (unsigned int cmd_idx = 1; cmd_idx < m_objectCmdOffsets.size(); ++cmd_idx)
|
|
|
|
|
{
|
|
|
|
|
if (ptr < start_ptr + m_objectCmdOffsets[cmd_idx])
|
|
|
|
|
{
|
|
|
|
|
result.cmd_idx = cmd_idx-1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
search_results.push_back(result);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
delete[] search_val;
|
|
|
|
|
|
|
|
|
|
ChangeSearchResult(0);
|
|
|
|
|
m_beginSearch->Disable();
|
|
|
|
|
m_findNext->Enable();
|
|
|
|
|
m_findPrevious->Enable();
|
|
|
|
|
m_numResultsText->SetLabel(wxString::Format(_("Found %d results for \'"), search_results.size()) + m_searchField->GetValue() + _("\'"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnSearchFieldTextChanged(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
m_beginSearch->Enable(m_searchField->GetLineLength(0) > 0);
|
|
|
|
|
m_findNext->Disable();
|
|
|
|
|
m_findPrevious->Disable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnFindNextClick(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
int cur_cmd_index = m_objectCmdList->GetSelection();
|
|
|
|
|
if (cur_cmd_index == -1)
|
|
|
|
|
{
|
|
|
|
|
ChangeSearchResult(0);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (std::vector<SearchResult>::iterator it = search_results.begin(); it != search_results.end(); ++it)
|
|
|
|
|
{
|
|
|
|
|
if (it->cmd_idx > cur_cmd_index)
|
|
|
|
|
{
|
|
|
|
|
ChangeSearchResult(it - search_results.begin());
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnFindPreviousClick(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
int cur_cmd_index = m_objectCmdList->GetSelection();
|
|
|
|
|
if (cur_cmd_index == -1)
|
|
|
|
|
{
|
|
|
|
|
ChangeSearchResult(search_results.size() - 1);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (std::vector<SearchResult>::reverse_iterator it = search_results.rbegin(); it != search_results.rend(); ++it)
|
|
|
|
|
{
|
|
|
|
|
if (it->cmd_idx < cur_cmd_index)
|
|
|
|
|
{
|
|
|
|
|
ChangeSearchResult(search_results.size()-1 - (it - search_results.rbegin()));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::ChangeSearchResult(unsigned int result_idx)
|
|
|
|
|
{
|
|
|
|
|
if (search_results.size() > result_idx)
|
|
|
|
|
{
|
|
|
|
|
m_search_result_idx = result_idx;
|
|
|
|
|
int prev_frame = m_framesList->GetSelection();
|
|
|
|
|
int prev_obj = m_objectsList->GetSelection();
|
|
|
|
|
int prev_cmd = m_objectCmdList->GetSelection();
|
|
|
|
|
m_framesList->SetSelection(search_results[result_idx].frame_idx);
|
|
|
|
|
m_objectsList->SetSelection(search_results[result_idx].obj_idx);
|
|
|
|
|
m_objectCmdList->SetSelection(search_results[result_idx].cmd_idx);
|
|
|
|
|
|
|
|
|
|
wxCommandEvent ev(wxEVT_COMMAND_LISTBOX_SELECTED);
|
|
|
|
|
if (prev_frame != m_framesList->GetSelection())
|
|
|
|
|
{
|
|
|
|
|
ev.SetInt(m_framesList->GetSelection());
|
|
|
|
|
OnFrameListSelectionChanged(ev);
|
|
|
|
|
}
|
|
|
|
|
if (prev_obj != m_objectsList->GetSelection())
|
|
|
|
|
{
|
|
|
|
|
ev.SetInt(m_objectsList->GetSelection());
|
|
|
|
|
OnObjectListSelectionChanged(ev);
|
|
|
|
|
}
|
|
|
|
|
if (prev_cmd != m_objectCmdList->GetSelection())
|
|
|
|
|
{
|
|
|
|
|
ev.SetInt(m_objectCmdList->GetSelection());
|
|
|
|
|
OnObjectCmdListSelectionChanged(ev);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (search_results.size())
|
|
|
|
|
{
|
|
|
|
|
ChangeSearchResult(search_results.size() - 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnFrameListSelectionChanged(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
|
|
|
|
|
|
|
|
m_objectsList->Clear();
|
|
|
|
|
if (event.GetInt() != -1)
|
|
|
|
|
{
|
|
|
|
|
int num_objects = player.GetAnalyzedFrameInfo(event.GetInt()).objectStarts.size();
|
|
|
|
|
for (int i = 0; i < num_objects; ++i)
|
|
|
|
|
m_objectsList->Append(wxString::Format(wxT("Object %i"), i));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update object list
|
|
|
|
|
wxCommandEvent ev = wxCommandEvent(wxEVT_COMMAND_LISTBOX_SELECTED);
|
|
|
|
|
ev.SetInt(-1);
|
|
|
|
|
OnObjectListSelectionChanged(ev);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnObjectListSelectionChanged(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
|
|
|
|
|
|
|
|
int frame_idx = m_framesList->GetSelection();
|
|
|
|
|
int object_idx = event.GetInt();
|
|
|
|
|
|
|
|
|
|
m_objectCmdList->Clear();
|
|
|
|
|
m_objectCmdOffsets.clear();
|
|
|
|
|
if (frame_idx != -1 && object_idx != -1)
|
|
|
|
|
{
|
|
|
|
|
const AnalyzedFrameInfo& frame = player.GetAnalyzedFrameInfo(frame_idx);
|
|
|
|
|
const FifoFrameInfo& fifo_frame = player.GetFile()->GetFrame(frame_idx);
|
|
|
|
|
const u8* objectdata_start = &fifo_frame.fifoData[frame.objectStarts[object_idx]];
|
|
|
|
|
const u8* objectdata_end = &fifo_frame.fifoData[frame.objectEnds[object_idx]];
|
|
|
|
|
u8* objectdata = (u8*)objectdata_start;
|
|
|
|
|
const int obj_offset = objectdata_start - &fifo_frame.fifoData[frame.objectStarts[0]];
|
|
|
|
|
|
|
|
|
|
int cmd = *objectdata++;
|
|
|
|
|
int stream_size = Common::swap16(objectdata);
|
|
|
|
|
objectdata += 2;
|
|
|
|
|
int vertex_size = (objectdata_end - objectdata) / stream_size;
|
|
|
|
|
wxString newLabel = wxString::Format(wxT("%08X: %02X %04X "), obj_offset, cmd, stream_size);
|
|
|
|
|
if ((objectdata_end - objectdata) % stream_size) newLabel += _("NOTE: Stream size doesn't match actual data length\n");
|
|
|
|
|
while (objectdata < objectdata_end)
|
|
|
|
|
{
|
|
|
|
|
newLabel += wxString::Format(wxT("%02X"), *objectdata++);
|
|
|
|
|
}
|
|
|
|
|
m_objectCmdList->Append(newLabel);
|
|
|
|
|
m_objectCmdOffsets.push_back(0);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Between objectdata_end and next_objdata_start, there are register setting commands
|
|
|
|
|
if (object_idx + 1 < (int)frame.objectStarts.size())
|
|
|
|
|
{
|
|
|
|
|
const u8* next_objdata_start = &fifo_frame.fifoData[frame.objectStarts[object_idx+1]];
|
|
|
|
|
while (objectdata < next_objdata_start)
|
|
|
|
|
{
|
|
|
|
|
m_objectCmdOffsets.push_back(objectdata - objectdata_start);
|
|
|
|
|
int new_offset = objectdata - &fifo_frame.fifoData[frame.objectStarts[0]];
|
|
|
|
|
int cmd = *objectdata++;
|
|
|
|
|
switch (cmd)
|
|
|
|
|
{
|
|
|
|
|
case GX_NOP:
|
|
|
|
|
newLabel = _("NOP");
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 0x44:
|
|
|
|
|
newLabel = _("0x44");
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GX_CMD_INVL_VC:
|
|
|
|
|
newLabel = _("GX_CMD_INVL_VC");
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GX_LOAD_CP_REG:
|
|
|
|
|
{
|
|
|
|
|
u32 cmd2 = *objectdata++;
|
|
|
|
|
u32 value = Common::swap32(objectdata);
|
|
|
|
|
objectdata += 4;
|
|
|
|
|
|
|
|
|
|
newLabel = wxString::Format(wxT("CP %02X %08X"), cmd2, value);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GX_LOAD_XF_REG:
|
|
|
|
|
{
|
|
|
|
|
u32 cmd2 = Common::swap32(objectdata);
|
|
|
|
|
objectdata += 4;
|
|
|
|
|
|
|
|
|
|
u8 streamSize = ((cmd2 >> 16) & 15) + 1;
|
|
|
|
|
|
|
|
|
|
const u8* stream_start = objectdata;
|
|
|
|
|
const u8* stream_end = stream_start + streamSize * 4;
|
|
|
|
|
|
|
|
|
|
newLabel = wxString::Format(wxT("XF %08X "), cmd2);
|
|
|
|
|
while (objectdata < stream_end)
|
|
|
|
|
{
|
|
|
|
|
newLabel += wxString::Format(wxT("%02X"), *objectdata++);
|
|
|
|
|
if (((objectdata - stream_start) % 4) == 0) newLabel += wxT(" ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GX_LOAD_INDX_A:
|
|
|
|
|
case GX_LOAD_INDX_B:
|
|
|
|
|
case GX_LOAD_INDX_C:
|
|
|
|
|
case GX_LOAD_INDX_D:
|
|
|
|
|
objectdata += 4;
|
|
|
|
|
newLabel = wxString::Format(wxT("LOAD INDX %s"), (cmd == GX_LOAD_INDX_A) ? _("A") :
|
|
|
|
|
(cmd == GX_LOAD_INDX_B) ? _("B") :
|
|
|
|
|
(cmd == GX_LOAD_INDX_C) ? _("C") : _("D"));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GX_CMD_CALL_DL:
|
|
|
|
|
// The recorder should have expanded display lists into the fifo stream and skipped the call to start them
|
|
|
|
|
// That is done to make it easier to track where memory is updated
|
|
|
|
|
_assert_(false);
|
|
|
|
|
objectdata += 8;
|
|
|
|
|
newLabel = wxString::Format(wxT("CALL DL"));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GX_LOAD_BP_REG:
|
|
|
|
|
{
|
|
|
|
|
u32 cmd2 = Common::swap32(objectdata);
|
|
|
|
|
objectdata += 4;
|
|
|
|
|
newLabel = wxString::Format(wxT("BP %02X %06X"), cmd2 >> 24, cmd2 & 0xFFFFFF);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
newLabel = _("Unexpected 0x80 call? Aborting...");
|
|
|
|
|
objectdata = (u8*)next_objdata_start;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
newLabel = wxString::Format(_("%08X: "), new_offset) + newLabel;
|
|
|
|
|
m_objectCmdList->Append(newLabel);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Update command list
|
|
|
|
|
wxCommandEvent ev = wxCommandEvent(wxEVT_COMMAND_LISTBOX_SELECTED);
|
|
|
|
|
ev.SetInt(-1);
|
|
|
|
|
OnObjectCmdListSelectionChanged(ev);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnObjectCmdListSelectionChanged(wxCommandEvent& event)
|
|
|
|
|
{
|
|
|
|
|
const int frame_idx = m_framesList->GetSelection();
|
|
|
|
|
const int object_idx = m_objectsList->GetSelection();
|
|
|
|
|
|
|
|
|
|
if (event.GetInt() == -1 || frame_idx == -1 || object_idx == -1)
|
|
|
|
|
{
|
|
|
|
|
m_objectCmdInfo->SetLabel(wxEmptyString);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
|
|
|
const AnalyzedFrameInfo& frame = player.GetAnalyzedFrameInfo(frame_idx);
|
|
|
|
|
const FifoFrameInfo& fifo_frame = player.GetFile()->GetFrame(frame_idx);
|
|
|
|
|
const u8* cmddata = &fifo_frame.fifoData[frame.objectStarts[object_idx]] + m_objectCmdOffsets[event.GetInt()];
|
|
|
|
|
|
|
|
|
|
// TODO: Not sure whether we should bother translating the descriptions
|
|
|
|
|
wxString newLabel;
|
|
|
|
|
if (*cmddata == GX_LOAD_BP_REG)
|
|
|
|
|
{
|
|
|
|
|
char name[64]="\0", desc[512]="\0";
|
|
|
|
|
GetBPRegInfo(cmddata+1, name, sizeof(name), desc, sizeof(desc));
|
|
|
|
|
newLabel = _("BP register ");
|
|
|
|
|
newLabel += (name[0] != '\0') ? wxString(name, *wxConvCurrent) : wxString::Format(_("UNKNOWN_%02X"), *(cmddata+1));
|
|
|
|
|
newLabel += wxT(":\n");
|
|
|
|
|
if (desc[0] != '\0')
|
|
|
|
|
newLabel += wxString(desc, *wxConvCurrent);
|
|
|
|
|
else
|
|
|
|
|
newLabel += _("No description available");
|
|
|
|
|
}
|
|
|
|
|
else if (*cmddata == GX_LOAD_CP_REG)
|
|
|
|
|
newLabel = _("CP reg");
|
|
|
|
|
else if (*cmddata == GX_LOAD_XF_REG)
|
|
|
|
|
newLabel = _("XF reg");
|
|
|
|
|
else
|
|
|
|
|
newLabel = _("No description available");
|
|
|
|
|
|
|
|
|
|
m_objectCmdInfo->SetLabel(newLabel);
|
|
|
|
|
Layout();
|
|
|
|
|
Fit();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::OnCloseClick(wxCommandEvent& WXUNUSED(event))
|
|
|
|
|
{
|
|
|
|
|
Hide();
|
|
|
|
@ -375,6 +812,25 @@ void FifoPlayerDlg::UpdateRecorderGui()
|
|
|
|
|
m_Save->Enable(GetSaveButtonEnabled());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FifoPlayerDlg::UpdateAnalyzerGui()
|
|
|
|
|
{
|
|
|
|
|
FifoPlayer &player = FifoPlayer::GetInstance();
|
|
|
|
|
FifoDataFile* file = player.GetFile();
|
|
|
|
|
|
|
|
|
|
int num_frames = (file) ? player.GetFile()->GetFrameCount() : 0;
|
|
|
|
|
if ((int)m_framesList->GetCount() != num_frames)
|
|
|
|
|
{
|
|
|
|
|
m_framesList->Clear();
|
|
|
|
|
for (int i = 0; i < num_frames; ++i)
|
|
|
|
|
{
|
|
|
|
|
m_framesList->Append(wxString::Format(wxT("Frame %i"), i));
|
|
|
|
|
}
|
|
|
|
|
wxCommandEvent ev = wxCommandEvent(wxEVT_COMMAND_LISTBOX_SELECTED);
|
|
|
|
|
ev.SetInt(-1);
|
|
|
|
|
OnFrameListSelectionChanged(ev);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wxString FifoPlayerDlg::CreateFileFrameCountLabel() const
|
|
|
|
|
{
|
|
|
|
|
FifoDataFile *file = FifoPlayer::GetInstance().GetFile();
|
|
|
|
|