#include "stdafx_gui.h" #include "Utilities/Log.h" #include "Emu/Memory/Memory.h" #include "Emu/System.h" #include "Emu/IdManager.h" #include "Emu/Cell/PPUThread.h" #include "Emu/Cell/SPUThread.h" #include "Emu/Cell/RawSPUThread.h" #include "Emu/SysCalls/lv2/sleep_queue.h" #include "Emu/SysCalls/lv2/sys_lwmutex.h" #include "Emu/SysCalls/lv2/sys_lwcond.h" #include "Emu/SysCalls/lv2/sys_mutex.h" #include "Emu/SysCalls/lv2/sys_cond.h" #include "Emu/SysCalls/lv2/sys_semaphore.h" #include "Emu/SysCalls/lv2/sys_event.h" #include "Emu/SysCalls/lv2/sys_event_flag.h" #include "Emu/SysCalls/lv2/sys_prx.h" #include "Emu/SysCalls/lv2/sys_memory.h" #include "KernelExplorer.h" KernelExplorer::KernelExplorer(wxWindow* parent) : wxFrame(parent, wxID_ANY, "Kernel Explorer", wxDefaultPosition, wxSize(700, 450)) { this->SetBackgroundColour(wxColour(240,240,240)); //This fix the ugly background color under Windows wxBoxSizer* s_panel = new wxBoxSizer(wxVERTICAL); // Buttons wxBoxSizer* box_buttons = new wxBoxSizer(wxHORIZONTAL); wxButton* b_refresh = new wxButton(this, wxID_ANY, "Refresh"); box_buttons->AddSpacer(10); box_buttons->Add(b_refresh); box_buttons->AddSpacer(10); wxStaticBoxSizer* box_tree = new wxStaticBoxSizer(wxHORIZONTAL, this, "Kernel"); m_tree = new wxTreeCtrl(this, wxID_ANY, wxDefaultPosition, wxSize(600,300)); box_tree->Add(m_tree); // Merge and display everything s_panel->AddSpacer(10); s_panel->Add(box_buttons); s_panel->AddSpacer(10); s_panel->Add(box_tree, 0, 0, 100); s_panel->AddSpacer(10); SetSizerAndFit(s_panel); // Events b_refresh->Bind(wxEVT_BUTTON, &KernelExplorer::OnRefresh, this); // Fill the wxTreeCtrl Update(); }; void KernelExplorer::Update() { char name[4096]; m_tree->DeleteAllItems(); const u32 total_memory_usage = vm::get(vm::user_space)->used.load(); const auto& root = m_tree->AddRoot(fmt::Format("Process, ID = 0x00000001, Total Memory Usage = 0x%x (%0.2f MB)", total_memory_usage, (float)total_memory_usage / (1024 * 1024))); union name64 { u64 u64_data; char string[8]; name64(u64 data) : u64_data(data & 0x00ffffffffffffffull) { } const char* operator &() const { return string; } }; // TODO: FileSystem // Semaphores if (u32 count = idm::get_count()) { sprintf(name, "Semaphores (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { const auto& sema = *data.second; sprintf(name, "Semaphore: ID = 0x%x '%s', Count = %d, Max Count = %d, Waiters = %#llx", data.first, &name64(sema.name), sema.value.load(), sema.max, sema.sq.size()); m_tree->AppendItem(node, name); } } // Mutexes if (u32 count = idm::get_count()) { sprintf(name, "Mutexes (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Mutex: ID = 0x%x '%s'", data.first, &name64(data.second->name)); m_tree->AppendItem(node, name); } } // Light Weight Mutexes if (u32 count = idm::get_count()) { sprintf(name, "Lightweight Mutexes (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Lightweight Mutex: ID = 0x%x '%s'", data.first, &name64(data.second->name)); m_tree->AppendItem(node, name); } } // Condition Variables if (u32 count = idm::get_count()) { sprintf(name, "Condition Variables (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Condition Variable: ID = 0x%x '%s'", data.first, &name64(data.second->name)); m_tree->AppendItem(node, name); } } // Light Weight Condition Variables if (u32 count = idm::get_count()) { sprintf(name, "Lightweight Condition Variables (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Lightweight Condition Variable: ID = 0x%x '%s'", data.first, &name64(data.second->name)); m_tree->AppendItem(node, name); } } // Event Queues if (u32 count = idm::get_count()) { sprintf(name, "Event Queues (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Event Queue: ID = 0x%x '%s', Key = %#llx", data.first, &name64(data.second->name), data.second->key); m_tree->AppendItem(node, name); } } // Event Ports if (u32 count = idm::get_count()) { sprintf(name, "Event Ports (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Event Port: ID = 0x%x, Name = %#llx", data.first, data.second->name); m_tree->AppendItem(node, name); } } // Modules if (u32 count = idm::get_count()) { sprintf(name, "Modules (%d)", count); const auto& node = m_tree->AppendItem(root, name); //sprintf(name, "Segment List (%l)", 2 * objects.size()); // TODO: Assuming 2 segments per PRX file is not good //m_tree->AppendItem(node, name); for (const auto id : idm::get_set()) { sprintf(name, "PRX: ID = 0x%x", id); m_tree->AppendItem(node, name); } } // Memory Containers if (u32 count = idm::get_count()) { sprintf(name, "Memory Containers (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto id : idm::get_set()) { sprintf(name, "Memory Container: ID = 0x%x", id); m_tree->AppendItem(node, name); } } // Event Flags if (u32 count = idm::get_count()) { sprintf(name, "Event Flags (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Event Flag: ID = 0x%x", data.first); m_tree->AppendItem(node, name); } } // PPU Threads if (u32 count = idm::get_count()) { sprintf(name, "PPU Threads (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Thread: ID = 0x%08x '%s', - %s", data.first, data.second->get_name().c_str(), data.second->ThreadStatusToString()); m_tree->AppendItem(node, name); } } if (u32 count = idm::get_count()) { sprintf(name, "SPU Threads (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Thread: ID = 0x%08x '%s', - %s", data.first, data.second->get_name().c_str(), data.second->ThreadStatusToString()); m_tree->AppendItem(node, name); } } if (u32 count = idm::get_count()) { sprintf(name, "RawSPU Threads (%d)", count); const auto& node = m_tree->AppendItem(root, name); for (const auto& data : idm::get_map()) { sprintf(name, "Thread: ID = 0x%08x '%s', - %s", data.first, data.second->get_name().c_str(), data.second->ThreadStatusToString()); m_tree->AppendItem(node, name); } } m_tree->Expand(root); } void KernelExplorer::OnRefresh(wxCommandEvent& WXUNUSED(event)) { Update(); }