988 lines
32 KiB
C++
988 lines
32 KiB
C++
// Copyright 2011 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include "DolphinWX/FifoPlayerDlg.h"
|
|
|
|
#include <algorithm>
|
|
#include <cstddef>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include <wx/button.h>
|
|
#include <wx/checkbox.h>
|
|
#include <wx/clipbrd.h>
|
|
#include <wx/dataobj.h>
|
|
#include <wx/dialog.h>
|
|
#include <wx/filedlg.h>
|
|
#include <wx/listbox.h>
|
|
#include <wx/msgdlg.h>
|
|
#include <wx/notebook.h>
|
|
#include <wx/panel.h>
|
|
#include <wx/sizer.h>
|
|
#include <wx/spinbutt.h>
|
|
#include <wx/spinctrl.h>
|
|
#include <wx/statbox.h>
|
|
#include <wx/stattext.h>
|
|
#include <wx/textctrl.h>
|
|
|
|
#include "Common/Assert.h"
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/Swap.h"
|
|
#include "Core/FifoPlayer/FifoDataFile.h"
|
|
#include "Core/FifoPlayer/FifoPlaybackAnalyzer.h"
|
|
#include "Core/FifoPlayer/FifoPlayer.h"
|
|
#include "Core/FifoPlayer/FifoRecorder.h"
|
|
#include "DolphinWX/WxUtils.h"
|
|
#include "VideoCommon/BPMemory.h"
|
|
#include "VideoCommon/OpcodeDecoding.h"
|
|
|
|
wxDEFINE_EVENT(RECORDING_FINISHED_EVENT, wxCommandEvent);
|
|
wxDEFINE_EVENT(FRAME_WRITTEN_EVENT, wxCommandEvent);
|
|
|
|
static std::recursive_mutex sMutex;
|
|
wxEvtHandler* volatile FifoPlayerDlg::m_EvtHandler = nullptr;
|
|
|
|
FifoPlayerDlg::FifoPlayerDlg(wxWindow* const parent)
|
|
: wxDialog(parent, wxID_ANY, _("FIFO Player")), m_search_result_idx(0), m_FramesToRecord(1)
|
|
{
|
|
CreateGUIControls();
|
|
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock{sMutex};
|
|
m_EvtHandler = GetEventHandler();
|
|
}
|
|
|
|
FifoPlayer::GetInstance().SetFileLoadedCallback(FileLoaded);
|
|
FifoPlayer::GetInstance().SetFrameWrittenCallback(FrameWritten);
|
|
}
|
|
|
|
FifoPlayerDlg::~FifoPlayerDlg()
|
|
{
|
|
FifoPlayer::GetInstance().SetFrameWrittenCallback(nullptr);
|
|
|
|
std::lock_guard<std::recursive_mutex> lock{sMutex};
|
|
m_EvtHandler = nullptr;
|
|
}
|
|
|
|
void FifoPlayerDlg::CreateGUIControls()
|
|
{
|
|
const int space5 = FromDIP(5);
|
|
|
|
m_Notebook = new wxNotebook(this, wxID_ANY);
|
|
|
|
{
|
|
m_PlayPage =
|
|
new wxPanel(m_Notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
|
|
|
|
// File Info
|
|
m_NumFramesLabel = new wxStaticText(m_PlayPage, wxID_ANY, wxEmptyString);
|
|
m_CurrentFrameLabel = new wxStaticText(m_PlayPage, wxID_ANY, wxEmptyString);
|
|
m_NumObjectsLabel = new wxStaticText(m_PlayPage, wxID_ANY, wxEmptyString);
|
|
|
|
// Frame Range
|
|
m_FrameFromLabel = new wxStaticText(m_PlayPage, wxID_ANY, _("From"));
|
|
m_FrameFromCtrl = new wxSpinCtrl(m_PlayPage, wxID_ANY, wxEmptyString, wxDefaultPosition,
|
|
wxDefaultSize, wxSP_ARROW_KEYS, 0, 10, 0);
|
|
m_FrameToLabel = new wxStaticText(m_PlayPage, wxID_ANY, _("To"));
|
|
m_FrameToCtrl = new wxSpinCtrl(m_PlayPage, wxID_ANY, wxEmptyString, wxDefaultPosition,
|
|
wxDefaultSize, wxSP_ARROW_KEYS, 0, 10, 0);
|
|
|
|
// Object Range
|
|
m_ObjectFromLabel = new wxStaticText(m_PlayPage, wxID_ANY, _("From"));
|
|
m_ObjectFromCtrl = new wxSpinCtrl(m_PlayPage, wxID_ANY, wxEmptyString, wxDefaultPosition,
|
|
wxDefaultSize, wxSP_ARROW_KEYS, 0, 10000, 0);
|
|
m_ObjectToLabel = new wxStaticText(m_PlayPage, wxID_ANY, _("To"));
|
|
m_ObjectToCtrl = new wxSpinCtrl(m_PlayPage, wxID_ANY, wxEmptyString, wxDefaultPosition,
|
|
wxDefaultSize, wxSP_ARROW_KEYS, 0, 10000, 0);
|
|
|
|
// Playback Options
|
|
m_EarlyMemoryUpdates = new wxCheckBox(m_PlayPage, wxID_ANY, _("Early Memory Updates"));
|
|
|
|
wxStaticBoxSizer* sPlayInfo = new wxStaticBoxSizer(wxVERTICAL, m_PlayPage, _("File Info"));
|
|
sPlayInfo->AddSpacer(space5);
|
|
sPlayInfo->Add(m_NumFramesLabel, 0, wxLEFT | wxRIGHT, space5);
|
|
sPlayInfo->AddSpacer(space5);
|
|
sPlayInfo->Add(m_CurrentFrameLabel, 0, wxLEFT | wxRIGHT, space5);
|
|
sPlayInfo->AddSpacer(space5);
|
|
sPlayInfo->Add(m_NumObjectsLabel, 0, wxLEFT | wxRIGHT, space5);
|
|
sPlayInfo->AddSpacer(space5);
|
|
|
|
wxStaticBoxSizer* sFrameRange =
|
|
new wxStaticBoxSizer(wxHORIZONTAL, m_PlayPage, _("Frame Range"));
|
|
sFrameRange->AddSpacer(space5);
|
|
sFrameRange->Add(m_FrameFromLabel, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sFrameRange->AddSpacer(space5);
|
|
sFrameRange->Add(m_FrameFromCtrl, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sFrameRange->AddSpacer(space5);
|
|
sFrameRange->Add(m_FrameToLabel, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sFrameRange->AddSpacer(space5);
|
|
sFrameRange->Add(m_FrameToCtrl, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sFrameRange->AddSpacer(space5);
|
|
|
|
wxStaticBoxSizer* sObjectRange =
|
|
new wxStaticBoxSizer(wxHORIZONTAL, m_PlayPage, _("Object Range"));
|
|
sObjectRange->AddSpacer(space5);
|
|
sObjectRange->Add(m_ObjectFromLabel, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sObjectRange->AddSpacer(space5);
|
|
sObjectRange->Add(m_ObjectFromCtrl, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sObjectRange->AddSpacer(space5);
|
|
sObjectRange->Add(m_ObjectToLabel, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sObjectRange->AddSpacer(space5);
|
|
sObjectRange->Add(m_ObjectToCtrl, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM, space5);
|
|
sObjectRange->AddSpacer(space5);
|
|
|
|
wxStaticBoxSizer* sPlayOptions =
|
|
new wxStaticBoxSizer(wxVERTICAL, m_PlayPage, _("Playback Options"));
|
|
sPlayOptions->AddSpacer(space5);
|
|
sPlayOptions->Add(m_EarlyMemoryUpdates, 0, wxLEFT | wxRIGHT, space5);
|
|
sPlayOptions->AddSpacer(space5);
|
|
|
|
wxBoxSizer* sPlayPage = new wxBoxSizer(wxVERTICAL);
|
|
sPlayPage->Add(sPlayInfo, 1, wxEXPAND);
|
|
sPlayPage->Add(sFrameRange, 0, wxEXPAND);
|
|
sPlayPage->Add(sObjectRange, 0, wxEXPAND);
|
|
sPlayPage->Add(sPlayOptions, 0, wxEXPAND);
|
|
sPlayPage->AddStretchSpacer();
|
|
|
|
m_PlayPage->SetSizer(sPlayPage);
|
|
m_Notebook->AddPage(m_PlayPage, _("Play"), true);
|
|
}
|
|
|
|
{
|
|
m_RecordPage =
|
|
new wxPanel(m_Notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
|
|
|
|
// Recording Info
|
|
m_RecordingFifoSizeLabel = new wxStaticText(m_RecordPage, wxID_ANY, wxEmptyString);
|
|
m_RecordingMemSizeLabel = new wxStaticText(m_RecordPage, wxID_ANY, wxEmptyString);
|
|
m_RecordingFramesLabel = new wxStaticText(m_RecordPage, wxID_ANY, wxEmptyString);
|
|
|
|
// Recording Buttons
|
|
m_RecordStop = new wxButton(m_RecordPage, wxID_ANY, _("Record"));
|
|
m_Save = new wxButton(m_RecordPage, wxID_ANY, _("Save"));
|
|
|
|
// Recording Options
|
|
m_FramesToRecordLabel = new wxStaticText(m_RecordPage, wxID_ANY, _("Frames to Record"));
|
|
m_FramesToRecordCtrl =
|
|
new wxSpinCtrl(m_RecordPage, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize,
|
|
wxSP_ARROW_KEYS, 0, 10000, m_FramesToRecord);
|
|
|
|
wxStaticBoxSizer* sRecordInfo =
|
|
new wxStaticBoxSizer(wxVERTICAL, m_RecordPage, _("Recording Info"));
|
|
sRecordInfo->AddSpacer(space5);
|
|
sRecordInfo->Add(m_RecordingFifoSizeLabel, 0, wxLEFT | wxRIGHT, space5);
|
|
sRecordInfo->AddSpacer(space5);
|
|
sRecordInfo->Add(m_RecordingMemSizeLabel, 0, wxLEFT | wxRIGHT, space5);
|
|
sRecordInfo->AddSpacer(space5);
|
|
sRecordInfo->Add(m_RecordingFramesLabel, 0, wxLEFT | wxRIGHT, space5);
|
|
sRecordInfo->AddSpacer(space5);
|
|
|
|
wxBoxSizer* sRecordButtons = new wxBoxSizer(wxHORIZONTAL);
|
|
sRecordButtons->Add(m_RecordStop);
|
|
sRecordButtons->Add(m_Save, 0, wxLEFT, space5);
|
|
|
|
wxStaticBoxSizer* sRecordingOptions =
|
|
new wxStaticBoxSizer(wxHORIZONTAL, m_RecordPage, _("Recording Options"));
|
|
sRecordingOptions->AddSpacer(space5);
|
|
sRecordingOptions->Add(m_FramesToRecordLabel, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM,
|
|
space5);
|
|
sRecordingOptions->AddSpacer(space5);
|
|
sRecordingOptions->Add(m_FramesToRecordCtrl, 0, wxALIGN_CENTER_VERTICAL | wxTOP | wxBOTTOM,
|
|
space5);
|
|
sRecordingOptions->AddSpacer(space5);
|
|
|
|
wxBoxSizer* sRecordPage = new wxBoxSizer(wxVERTICAL);
|
|
sRecordPage->Add(sRecordInfo, 0, wxEXPAND);
|
|
sRecordPage->AddSpacer(space5);
|
|
sRecordPage->Add(sRecordButtons, 0, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sRecordPage->AddSpacer(space5);
|
|
sRecordPage->Add(sRecordingOptions, 0, wxEXPAND);
|
|
|
|
m_RecordPage->SetSizer(sRecordPage);
|
|
m_Notebook->AddPage(m_RecordPage, _("Record"), false);
|
|
}
|
|
|
|
// Analyze page
|
|
{
|
|
m_AnalyzePage =
|
|
new wxPanel(m_Notebook, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
|
|
|
|
// FIFO Content Lists
|
|
m_framesList = new wxListBox(m_AnalyzePage, wxID_ANY, wxDefaultPosition,
|
|
wxDLG_UNIT(this, wxSize(72, 185)));
|
|
m_objectsList = new wxListBox(m_AnalyzePage, wxID_ANY, wxDefaultPosition,
|
|
wxDLG_UNIT(this, wxSize(72, 185)));
|
|
m_objectCmdList =
|
|
new wxListBox(m_AnalyzePage, wxID_ANY, wxDefaultPosition,
|
|
wxDLG_UNIT(this, wxSize(144, 185)), wxArrayString(), wxLB_HSCROLL);
|
|
|
|
// Selected command breakdown
|
|
m_objectCmdInfo = new wxStaticText(m_AnalyzePage, wxID_ANY, wxEmptyString);
|
|
|
|
// Search box
|
|
wxStaticText* search_label =
|
|
new wxStaticText(m_AnalyzePage, wxID_ANY, _("Search for Hex Value:"));
|
|
// 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);
|
|
|
|
// Search buttons
|
|
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"));
|
|
|
|
ResetSearch();
|
|
|
|
wxBoxSizer* sListsSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
sListsSizer->Add(m_framesList);
|
|
sListsSizer->Add(m_objectsList, 0, wxLEFT, space5);
|
|
sListsSizer->Add(m_objectCmdList, 1, wxLEFT, space5);
|
|
|
|
wxStaticBoxSizer* sFrameInfoSizer =
|
|
new wxStaticBoxSizer(wxVERTICAL, m_AnalyzePage, _("Frame Info"));
|
|
sFrameInfoSizer->AddSpacer(space5);
|
|
sFrameInfoSizer->Add(sListsSizer, 0, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sFrameInfoSizer->AddSpacer(space5);
|
|
sFrameInfoSizer->Add(m_objectCmdInfo, 0, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sFrameInfoSizer->AddSpacer(space5);
|
|
|
|
wxBoxSizer* sSearchField = new wxBoxSizer(wxHORIZONTAL);
|
|
sSearchField->Add(search_label, 0, wxALIGN_CENTER_VERTICAL);
|
|
sSearchField->Add(m_searchField, 0, wxLEFT, space5);
|
|
sSearchField->Add(m_numResultsText, 0, wxALIGN_CENTER_VERTICAL | wxLEFT, space5);
|
|
|
|
wxBoxSizer* sSearchButtons = new wxBoxSizer(wxHORIZONTAL);
|
|
sSearchButtons->Add(m_beginSearch);
|
|
sSearchButtons->Add(m_findNext, 0, wxLEFT, space5);
|
|
sSearchButtons->Add(m_findPrevious, 0, wxLEFT, space5);
|
|
|
|
wxStaticBoxSizer* sSearchSizer =
|
|
new wxStaticBoxSizer(wxVERTICAL, m_AnalyzePage, _("Search Current Object"));
|
|
sSearchSizer->Add(sSearchField, 0, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sSearchSizer->AddSpacer(space5);
|
|
sSearchSizer->Add(sSearchButtons, 0, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sSearchSizer->AddSpacer(space5);
|
|
|
|
wxBoxSizer* sAnalyzePage = new wxBoxSizer(wxVERTICAL);
|
|
sAnalyzePage->Add(sFrameInfoSizer, 0, wxEXPAND);
|
|
sAnalyzePage->Add(sSearchSizer, 0, wxEXPAND);
|
|
|
|
m_AnalyzePage->SetSizer(sAnalyzePage);
|
|
m_Notebook->AddPage(m_AnalyzePage, _("Analyze"), false);
|
|
}
|
|
|
|
wxStdDialogButtonSizer* close_btn_sizer = CreateStdDialogButtonSizer(wxCLOSE);
|
|
close_btn_sizer->GetCancelButton()->SetLabel(_("Close"));
|
|
|
|
wxBoxSizer* sMain = new wxBoxSizer(wxVERTICAL);
|
|
sMain->AddSpacer(space5);
|
|
sMain->Add(m_Notebook, 1, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sMain->AddSpacer(space5);
|
|
sMain->Add(close_btn_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT, space5);
|
|
sMain->AddSpacer(space5);
|
|
|
|
SetLayoutAdaptationMode(wxDIALOG_ADAPTATION_MODE_ENABLED);
|
|
SetLayoutAdaptationLevel(wxDIALOG_ADAPTATION_STANDARD_SIZER);
|
|
SetSizerAndFit(sMain);
|
|
Center();
|
|
|
|
// Connect Events
|
|
Bind(wxEVT_PAINT, &FifoPlayerDlg::OnPaint, this);
|
|
m_FrameFromCtrl->Bind(wxEVT_SPINCTRL, &FifoPlayerDlg::OnFrameFrom, this);
|
|
m_FrameToCtrl->Bind(wxEVT_SPINCTRL, &FifoPlayerDlg::OnFrameTo, this);
|
|
m_ObjectFromCtrl->Bind(wxEVT_SPINCTRL, &FifoPlayerDlg::OnObjectFrom, this);
|
|
m_ObjectToCtrl->Bind(wxEVT_SPINCTRL, &FifoPlayerDlg::OnObjectTo, this);
|
|
m_EarlyMemoryUpdates->Bind(wxEVT_CHECKBOX, &FifoPlayerDlg::OnCheckEarlyMemoryUpdates, this);
|
|
m_RecordStop->Bind(wxEVT_BUTTON, &FifoPlayerDlg::OnRecordStop, this);
|
|
m_Save->Bind(wxEVT_BUTTON, &FifoPlayerDlg::OnSaveFile, this);
|
|
m_FramesToRecordCtrl->Bind(wxEVT_SPINCTRL, &FifoPlayerDlg::OnNumFramesToRecord, this);
|
|
|
|
m_framesList->Bind(wxEVT_LISTBOX, &FifoPlayerDlg::OnFrameListSelectionChanged, this);
|
|
m_objectsList->Bind(wxEVT_LISTBOX, &FifoPlayerDlg::OnObjectListSelectionChanged, this);
|
|
m_objectCmdList->Bind(wxEVT_LISTBOX, &FifoPlayerDlg::OnObjectCmdListSelectionChanged, this);
|
|
|
|
m_beginSearch->Bind(wxEVT_BUTTON, &FifoPlayerDlg::OnBeginSearch, this);
|
|
m_findNext->Bind(wxEVT_BUTTON, &FifoPlayerDlg::OnFindNextClick, this);
|
|
m_findPrevious->Bind(wxEVT_BUTTON, &FifoPlayerDlg::OnFindPreviousClick, this);
|
|
|
|
m_searchField->Bind(wxEVT_TEXT_ENTER, &FifoPlayerDlg::OnBeginSearch, this);
|
|
m_searchField->Bind(wxEVT_TEXT, &FifoPlayerDlg::OnSearchFieldTextChanged, this);
|
|
|
|
// Setup command copying
|
|
wxAcceleratorEntry entry;
|
|
entry.Set(wxACCEL_CTRL, (int)'C', wxID_COPY);
|
|
wxAcceleratorTable accel(1, &entry);
|
|
m_objectCmdList->SetAcceleratorTable(accel);
|
|
m_objectCmdList->Bind(wxEVT_MENU, &FifoPlayerDlg::OnObjectCmdListSelectionCopy, this, wxID_COPY);
|
|
|
|
Bind(RECORDING_FINISHED_EVENT, &FifoPlayerDlg::OnRecordingFinished, this);
|
|
Bind(FRAME_WRITTEN_EVENT, &FifoPlayerDlg::OnFrameWritten, this);
|
|
|
|
Show();
|
|
}
|
|
|
|
void FifoPlayerDlg::OnPaint(wxPaintEvent& event)
|
|
{
|
|
UpdatePlayGui();
|
|
UpdateRecorderGui();
|
|
UpdateAnalyzerGui();
|
|
|
|
event.Skip();
|
|
}
|
|
|
|
void FifoPlayerDlg::OnFrameFrom(wxSpinEvent& event)
|
|
{
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
|
|
player.SetFrameRangeStart(event.GetPosition());
|
|
|
|
m_FrameFromCtrl->SetValue(player.GetFrameRangeStart());
|
|
m_FrameToCtrl->SetValue(player.GetFrameRangeEnd());
|
|
}
|
|
|
|
void FifoPlayerDlg::OnFrameTo(wxSpinEvent& event)
|
|
{
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
player.SetFrameRangeEnd(event.GetPosition());
|
|
|
|
m_FrameFromCtrl->SetValue(player.GetFrameRangeStart());
|
|
m_FrameToCtrl->SetValue(player.GetFrameRangeEnd());
|
|
}
|
|
|
|
void FifoPlayerDlg::OnObjectFrom(wxSpinEvent& event)
|
|
{
|
|
FifoPlayer::GetInstance().SetObjectRangeStart(event.GetPosition());
|
|
}
|
|
|
|
void FifoPlayerDlg::OnObjectTo(wxSpinEvent& event)
|
|
{
|
|
FifoPlayer::GetInstance().SetObjectRangeEnd(event.GetPosition());
|
|
}
|
|
|
|
void FifoPlayerDlg::OnCheckEarlyMemoryUpdates(wxCommandEvent& event)
|
|
{
|
|
FifoPlayer::GetInstance().SetEarlyMemoryUpdates(event.IsChecked());
|
|
}
|
|
|
|
void FifoPlayerDlg::OnSaveFile(wxCommandEvent& WXUNUSED(event))
|
|
{
|
|
// Pointer to the file data that was created as a result of recording.
|
|
FifoDataFile* file = FifoRecorder::GetInstance().GetRecordedFile();
|
|
|
|
if (file)
|
|
{
|
|
// Bring up a save file dialog. The location the user chooses will be assigned to this variable.
|
|
wxString path = wxSaveFileSelector(_("Dolphin FIFO"), "dff", wxEmptyString, this);
|
|
|
|
// Has a valid file path
|
|
if (!path.empty())
|
|
{
|
|
// Attempt to save the file to the path the user chose
|
|
wxBeginBusyCursor();
|
|
bool result = file->Save(WxStrToStr(path));
|
|
wxEndBusyCursor();
|
|
|
|
// Wasn't able to save the file, shit's whack, yo.
|
|
if (!result)
|
|
WxUtils::ShowErrorDialog(_("Error saving file."));
|
|
}
|
|
}
|
|
}
|
|
|
|
void FifoPlayerDlg::OnRecordStop(wxCommandEvent& WXUNUSED(event))
|
|
{
|
|
FifoRecorder& recorder = FifoRecorder::GetInstance();
|
|
|
|
// Recorder is still recording
|
|
if (recorder.IsRecording())
|
|
{
|
|
// Then stop recording
|
|
recorder.StopRecording();
|
|
|
|
// and change the button label accordingly.
|
|
m_RecordStop->SetLabel(_("Record"));
|
|
}
|
|
else // Recorder is actually about to start recording
|
|
{
|
|
// So start recording
|
|
recorder.StartRecording(m_FramesToRecord, RecordingFinished);
|
|
|
|
// and change the button label accordingly.
|
|
m_RecordStop->SetLabel(_("Stop"));
|
|
}
|
|
}
|
|
|
|
void FifoPlayerDlg::OnNumFramesToRecord(wxSpinEvent& event)
|
|
{
|
|
m_FramesToRecord = event.GetPosition();
|
|
|
|
// Entering 0 frames in the control indicates infinite frames to record
|
|
// The fifo recorder takes any value < 0 to be infinite frames
|
|
if (m_FramesToRecord < 1)
|
|
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.empty() && str_search_val.length() % 2)
|
|
{
|
|
m_numResultsText->SetLabel(_("Invalid search string (only even string lengths supported)"));
|
|
return;
|
|
}
|
|
|
|
unsigned int const val_length = str_search_val.length() / 2;
|
|
std::vector<u8> search_val(val_length);
|
|
for (unsigned int i = 0; i < val_length; ++i)
|
|
{
|
|
wxString char_str = str_search_val.Mid(2 * i, 2);
|
|
unsigned long val = 0;
|
|
if (!char_str.ToULong(&val, 16))
|
|
{
|
|
m_numResultsText->SetLabel(_("Invalid search string (couldn't convert to number)"));
|
|
return;
|
|
}
|
|
search_val[i] = (u8)val;
|
|
}
|
|
search_results.clear();
|
|
|
|
int const 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;
|
|
}
|
|
|
|
const u8* const start_ptr = &fifo_frame.fifoData[frame.objectStarts[obj_idx]];
|
|
const u8* const end_ptr = &fifo_frame.fifoData[frame.objectStarts[obj_idx + 1]];
|
|
|
|
for (const u8* ptr = start_ptr; ptr < end_ptr - val_length + 1; ++ptr)
|
|
{
|
|
if (std::equal(search_val.begin(), search_val.end(), ptr))
|
|
{
|
|
SearchResult result;
|
|
result.frame_idx = frame_idx;
|
|
|
|
result.obj_idx = m_objectsList->GetSelection();
|
|
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);
|
|
}
|
|
}
|
|
|
|
ChangeSearchResult(0);
|
|
m_beginSearch->Disable();
|
|
m_numResultsText->SetLabel(
|
|
wxString::Format(_("Found %u results for \'"), (u32)search_results.size()) +
|
|
m_searchField->GetValue() + "\'");
|
|
}
|
|
|
|
void FifoPlayerDlg::OnSearchFieldTextChanged(wxCommandEvent& event)
|
|
{
|
|
ResetSearch();
|
|
}
|
|
|
|
void FifoPlayerDlg::OnFindNextClick(wxCommandEvent& event)
|
|
{
|
|
int cur_cmd_index = m_objectCmdList->GetSelection();
|
|
if (cur_cmd_index == -1)
|
|
{
|
|
ChangeSearchResult(0);
|
|
return;
|
|
}
|
|
|
|
for (auto 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 (auto 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 (result_idx < search_results.size()) // if index is valid
|
|
{
|
|
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_LISTBOX);
|
|
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);
|
|
}
|
|
|
|
m_findNext->Enable(result_idx + 1 < search_results.size());
|
|
m_findPrevious->Enable(result_idx != 0);
|
|
}
|
|
else if (search_results.size())
|
|
{
|
|
ChangeSearchResult(search_results.size() - 1);
|
|
}
|
|
}
|
|
|
|
void FifoPlayerDlg::ResetSearch()
|
|
{
|
|
m_beginSearch->Enable(m_searchField->GetLineLength(0) > 0);
|
|
m_findNext->Disable();
|
|
m_findPrevious->Disable();
|
|
|
|
search_results.clear();
|
|
}
|
|
|
|
void FifoPlayerDlg::OnFrameListSelectionChanged(wxCommandEvent& event)
|
|
{
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
|
|
m_objectsList->Clear();
|
|
if (event.GetInt() != -1)
|
|
{
|
|
size_t num_objects = player.GetAnalyzedFrameInfo(event.GetInt()).objectStarts.size();
|
|
for (size_t i = 0; i < num_objects; ++i)
|
|
m_objectsList->Append(wxString::Format(_("Object %zu"), i));
|
|
}
|
|
|
|
// Update object list
|
|
wxCommandEvent ev = wxCommandEvent(wxEVT_LISTBOX);
|
|
ev.SetInt(-1);
|
|
OnObjectListSelectionChanged(ev);
|
|
|
|
ResetSearch();
|
|
}
|
|
|
|
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;
|
|
wxString newLabel = wxString::Format("%08X: %02X %04X ", obj_offset, cmd, stream_size);
|
|
if (stream_size && ((objectdata_end - objectdata) % stream_size))
|
|
newLabel += _("NOTE: Stream size doesn't match actual data length\n");
|
|
|
|
while (objectdata < objectdata_end)
|
|
{
|
|
newLabel += wxString::Format("%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 command = *objectdata++;
|
|
switch (command)
|
|
{
|
|
case OpcodeDecoder::GX_NOP:
|
|
newLabel = "NOP";
|
|
break;
|
|
|
|
case 0x44:
|
|
newLabel = "0x44";
|
|
break;
|
|
|
|
case OpcodeDecoder::GX_CMD_INVL_VC:
|
|
newLabel = "GX_CMD_INVL_VC";
|
|
break;
|
|
|
|
case OpcodeDecoder::GX_LOAD_CP_REG:
|
|
{
|
|
u32 cmd2 = *objectdata++;
|
|
u32 value = Common::swap32(objectdata);
|
|
objectdata += 4;
|
|
|
|
newLabel = wxString::Format("CP %02X %08X", cmd2, value);
|
|
}
|
|
break;
|
|
|
|
case OpcodeDecoder::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("XF %08X ", cmd2);
|
|
while (objectdata < stream_end)
|
|
{
|
|
newLabel += wxString::Format("%02X", *objectdata++);
|
|
|
|
if (((objectdata - stream_start) % 4) == 0)
|
|
newLabel += " ";
|
|
}
|
|
}
|
|
break;
|
|
|
|
case OpcodeDecoder::GX_LOAD_INDX_A:
|
|
case OpcodeDecoder::GX_LOAD_INDX_B:
|
|
case OpcodeDecoder::GX_LOAD_INDX_C:
|
|
case OpcodeDecoder::GX_LOAD_INDX_D:
|
|
{
|
|
objectdata += 4;
|
|
newLabel = wxString::Format("LOAD INDX %s",
|
|
(command == OpcodeDecoder::GX_LOAD_INDX_A) ?
|
|
"A" :
|
|
(command == OpcodeDecoder::GX_LOAD_INDX_B) ?
|
|
"B" :
|
|
(command == OpcodeDecoder::GX_LOAD_INDX_C) ? "C" : "D");
|
|
}
|
|
break;
|
|
|
|
case OpcodeDecoder::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("CALL DL");
|
|
break;
|
|
|
|
case OpcodeDecoder::GX_LOAD_BP_REG:
|
|
{
|
|
u32 cmd2 = Common::swap32(objectdata);
|
|
objectdata += 4;
|
|
newLabel = wxString::Format("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_LISTBOX);
|
|
ev.SetInt(-1);
|
|
OnObjectCmdListSelectionChanged(ev);
|
|
|
|
ResetSearch();
|
|
}
|
|
|
|
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 == OpcodeDecoder::GX_LOAD_BP_REG)
|
|
{
|
|
std::string name;
|
|
std::string desc;
|
|
GetBPRegInfo(cmddata + 1, &name, &desc);
|
|
|
|
newLabel = _("BP register ");
|
|
newLabel +=
|
|
(name.empty()) ? wxString::Format(_("UNKNOWN_%02X"), *(cmddata + 1)) : StrToWxStr(name);
|
|
newLabel += ":\n";
|
|
|
|
if (desc.empty())
|
|
newLabel += _("No description available");
|
|
else
|
|
newLabel += StrToWxStr(desc);
|
|
}
|
|
else if (*cmddata == OpcodeDecoder::GX_LOAD_CP_REG)
|
|
{
|
|
newLabel = _("CP register ");
|
|
}
|
|
else if (*cmddata == OpcodeDecoder::GX_LOAD_XF_REG)
|
|
{
|
|
newLabel = _("XF register ");
|
|
}
|
|
else
|
|
{
|
|
newLabel = _("No description available");
|
|
}
|
|
|
|
m_objectCmdInfo->SetLabel(newLabel);
|
|
Layout();
|
|
Fit();
|
|
}
|
|
|
|
void FifoPlayerDlg::OnObjectCmdListSelectionCopy(wxCommandEvent& WXUNUSED(event))
|
|
{
|
|
if (wxTheClipboard->Open())
|
|
{
|
|
wxTheClipboard->SetData(new wxTextDataObject(m_objectCmdList->GetStringSelection()));
|
|
wxTheClipboard->Close();
|
|
}
|
|
}
|
|
|
|
void FifoPlayerDlg::OnRecordingFinished(wxEvent&)
|
|
{
|
|
m_RecordStop->SetLabel(_("Record"));
|
|
m_RecordStop->Enable();
|
|
|
|
UpdateRecorderGui();
|
|
}
|
|
|
|
void FifoPlayerDlg::OnFrameWritten(wxEvent&)
|
|
{
|
|
m_CurrentFrameLabel->SetLabel(CreateCurrentFrameLabel());
|
|
m_NumObjectsLabel->SetLabel(CreateFileObjectCountLabel());
|
|
}
|
|
|
|
void FifoPlayerDlg::UpdatePlayGui()
|
|
{
|
|
m_NumFramesLabel->SetLabel(CreateFileFrameCountLabel());
|
|
m_CurrentFrameLabel->SetLabel(CreateCurrentFrameLabel());
|
|
m_NumObjectsLabel->SetLabel(CreateFileObjectCountLabel());
|
|
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
FifoDataFile* file = player.GetFile();
|
|
u32 frameCount = 0;
|
|
if (file)
|
|
frameCount = file->GetFrameCount();
|
|
|
|
m_FrameFromCtrl->SetRange(0, frameCount);
|
|
m_FrameFromCtrl->SetValue(player.GetFrameRangeStart());
|
|
|
|
m_FrameToCtrl->SetRange(0, frameCount);
|
|
m_FrameToCtrl->SetValue(player.GetFrameRangeEnd());
|
|
|
|
m_ObjectFromCtrl->SetValue(player.GetObjectRangeStart());
|
|
m_ObjectToCtrl->SetValue(player.GetObjectRangeEnd());
|
|
}
|
|
|
|
void FifoPlayerDlg::UpdateRecorderGui()
|
|
{
|
|
m_RecordingFifoSizeLabel->SetLabel(CreateRecordingFifoSizeLabel());
|
|
m_RecordingMemSizeLabel->SetLabel(CreateRecordingMemSizeLabel());
|
|
m_RecordingFramesLabel->SetLabel(CreateRecordingFrameCountLabel());
|
|
m_Save->Enable(GetSaveButtonEnabled());
|
|
}
|
|
|
|
void FifoPlayerDlg::UpdateAnalyzerGui()
|
|
{
|
|
FifoPlayer& player = FifoPlayer::GetInstance();
|
|
FifoDataFile* file = player.GetFile();
|
|
|
|
size_t num_frames = (file) ? player.GetFile()->GetFrameCount() : 0U;
|
|
if (m_framesList->GetCount() != num_frames)
|
|
{
|
|
m_framesList->Clear();
|
|
|
|
for (size_t i = 0; i < num_frames; ++i)
|
|
{
|
|
m_framesList->Append(wxString::Format(_("Frame %zu"), i));
|
|
}
|
|
|
|
wxCommandEvent ev = wxCommandEvent(wxEVT_LISTBOX);
|
|
ev.SetInt(-1);
|
|
OnFrameListSelectionChanged(ev);
|
|
}
|
|
}
|
|
|
|
wxString FifoPlayerDlg::CreateFileFrameCountLabel() const
|
|
{
|
|
FifoDataFile* file = FifoPlayer::GetInstance().GetFile();
|
|
|
|
if (file)
|
|
return wxString::Format(_("%u frames"), file->GetFrameCount());
|
|
|
|
return _("No file loaded");
|
|
}
|
|
|
|
wxString FifoPlayerDlg::CreateCurrentFrameLabel() const
|
|
{
|
|
FifoDataFile* file = FifoPlayer::GetInstance().GetFile();
|
|
|
|
if (file)
|
|
return wxString::Format(_("Frame %u"), FifoPlayer::GetInstance().GetCurrentFrameNum());
|
|
|
|
return wxEmptyString;
|
|
}
|
|
|
|
wxString FifoPlayerDlg::CreateFileObjectCountLabel() const
|
|
{
|
|
FifoDataFile* file = FifoPlayer::GetInstance().GetFile();
|
|
|
|
if (file)
|
|
return wxString::Format(_("%u objects"), FifoPlayer::GetInstance().GetFrameObjectCount());
|
|
|
|
return wxEmptyString;
|
|
}
|
|
|
|
wxString FifoPlayerDlg::CreateRecordingFifoSizeLabel() const
|
|
{
|
|
FifoDataFile* file = FifoRecorder::GetInstance().GetRecordedFile();
|
|
|
|
if (file)
|
|
{
|
|
size_t fifoBytes = 0;
|
|
for (size_t i = 0; i < file->GetFrameCount(); ++i)
|
|
fifoBytes += file->GetFrame(i).fifoData.size();
|
|
|
|
return wxString::Format(_("%zu FIFO bytes"), fifoBytes);
|
|
}
|
|
|
|
return _("No recorded file");
|
|
}
|
|
|
|
wxString FifoPlayerDlg::CreateRecordingMemSizeLabel() const
|
|
{
|
|
FifoDataFile* file = FifoRecorder::GetInstance().GetRecordedFile();
|
|
|
|
if (file)
|
|
{
|
|
size_t memBytes = 0;
|
|
for (size_t frameNum = 0; frameNum < file->GetFrameCount(); ++frameNum)
|
|
{
|
|
const std::vector<MemoryUpdate>& memUpdates = file->GetFrame(frameNum).memoryUpdates;
|
|
for (const auto& memUpdate : memUpdates)
|
|
memBytes += memUpdate.data.size();
|
|
}
|
|
|
|
return wxString::Format(_("%zu memory bytes"), memBytes);
|
|
}
|
|
|
|
return wxEmptyString;
|
|
}
|
|
|
|
wxString FifoPlayerDlg::CreateRecordingFrameCountLabel() const
|
|
{
|
|
FifoDataFile* file = FifoRecorder::GetInstance().GetRecordedFile();
|
|
|
|
if (file)
|
|
return wxString::Format(_("%u frames"), file->GetFrameCount());
|
|
|
|
return wxEmptyString;
|
|
}
|
|
|
|
bool FifoPlayerDlg::GetSaveButtonEnabled() const
|
|
{
|
|
return (FifoRecorder::GetInstance().GetRecordedFile() != nullptr);
|
|
}
|
|
|
|
void FifoPlayerDlg::RecordingFinished()
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock{sMutex};
|
|
|
|
if (m_EvtHandler)
|
|
{
|
|
wxCommandEvent event(RECORDING_FINISHED_EVENT);
|
|
m_EvtHandler->AddPendingEvent(event);
|
|
}
|
|
}
|
|
|
|
void FifoPlayerDlg::FileLoaded()
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock{sMutex};
|
|
|
|
if (m_EvtHandler)
|
|
{
|
|
wxPaintEvent event;
|
|
m_EvtHandler->AddPendingEvent(event);
|
|
}
|
|
}
|
|
|
|
void FifoPlayerDlg::FrameWritten()
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock{sMutex};
|
|
|
|
if (m_EvtHandler)
|
|
{
|
|
wxCommandEvent event(FRAME_WRITTEN_EVENT);
|
|
m_EvtHandler->AddPendingEvent(event);
|
|
}
|
|
}
|