// Copyright 2018 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #include "DolphinQt/Debugger/MemoryViewWidget.h" #include #include #include #include #include #include #include #include #include "Common/Align.h" #include "Common/FloatUtils.h" #include "Common/StringUtil.h" #include "Core/Core.h" #include "Core/HW/AddressSpace.h" #include "Core/PowerPC/BreakPoints.h" #include "Core/PowerPC/PowerPC.h" #include "DolphinQt/Host.h" #include "DolphinQt/Resources.h" #include "DolphinQt/Settings.h" // "Most mouse types work in steps of 15 degrees, in which case the delta value is a multiple of // 120; i.e., 120 units * 1/8 = 15 degrees." (http://doc.qt.io/qt-5/qwheelevent.html#angleDelta) constexpr double SCROLL_FRACTION_DEGREES = 15.; MemoryViewWidget::MemoryViewWidget(QWidget* parent) : QTableWidget(parent) { horizontalHeader()->hide(); verticalHeader()->hide(); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setShowGrid(false); setContextMenuPolicy(Qt::CustomContextMenu); connect(&Settings::Instance(), &Settings::DebugFontChanged, this, &MemoryViewWidget::UpdateFont); connect(&Settings::Instance(), &Settings::EmulationStateChanged, this, [this] { Update(); }); connect(Host::GetInstance(), &Host::UpdateDisasmDialog, this, &MemoryViewWidget::Update); connect(this, &MemoryViewWidget::customContextMenuRequested, this, &MemoryViewWidget::OnContextMenu); connect(&Settings::Instance(), &Settings::ThemeChanged, this, &MemoryViewWidget::Update); // Also calls update. UpdateFont(); } void MemoryViewWidget::UpdateFont() { const QFontMetrics fm(Settings::Instance().GetDebugFont()); m_font_vspace = fm.lineSpacing(); // BoundingRect is too unpredictable, a custom one would be needed for each view type. Different // fonts have wildly different spacing between two characters and horizontalAdvance includes // spacing. #if QT_VERSION >= QT_VERSION_CHECK(5, 11, 0) m_font_width = fm.horizontalAdvance(QLatin1Char('0')); #else m_font_width = fm.width(QLatin1Char('0')); #endif setFont(Settings::Instance().GetDebugFont()); Update(); } static int GetTypeSize(MemoryViewWidget::Type type) { switch (type) { case MemoryViewWidget::Type::ASCII: case MemoryViewWidget::Type::Hex8: case MemoryViewWidget::Type::Unsigned8: case MemoryViewWidget::Type::Signed8: return 1; case MemoryViewWidget::Type::Unsigned16: case MemoryViewWidget::Type::Signed16: case MemoryViewWidget::Type::Hex16: return 2; case MemoryViewWidget::Type::Hex32: case MemoryViewWidget::Type::Unsigned32: case MemoryViewWidget::Type::Signed32: case MemoryViewWidget::Type::Float32: return 4; case MemoryViewWidget::Type::Double: return 8; default: return 1; } } static int GetCharacterCount(MemoryViewWidget::Type type) { switch (type) { case MemoryViewWidget::Type::ASCII: return 1; case MemoryViewWidget::Type::Hex8: return 2; case MemoryViewWidget::Type::Unsigned8: return 3; case MemoryViewWidget::Type::Hex16: case MemoryViewWidget::Type::Signed8: return 4; case MemoryViewWidget::Type::Unsigned16: return 5; case MemoryViewWidget::Type::Signed16: return 6; case MemoryViewWidget::Type::Hex32: return 8; case MemoryViewWidget::Type::Float32: return 9; case MemoryViewWidget::Type::Double: case MemoryViewWidget::Type::Unsigned32: case MemoryViewWidget::Type::Signed32: return 10; default: return 8; } } void MemoryViewWidget::Update() { clearSelection(); u32 address = m_address; address = Common::AlignDown(address, m_alignment); const int data_columns = m_bytes_per_row / GetTypeSize(m_type); setColumnCount(2 + data_columns); if (rowCount() == 0) setRowCount(1); // This sets all row heights and determines horizontal ascii spacing. verticalHeader()->setDefaultSectionSize(m_font_vspace - 1); verticalHeader()->setMinimumSectionSize(m_font_vspace - 1); horizontalHeader()->setMinimumSectionSize(m_font_width * 2); const AddressSpace::Accessors* accessors = AddressSpace::GetAccessors(m_address_space); // Calculate (roughly) how many rows will fit in our table int rows = std::round((height() / static_cast(rowHeight(0))) - 0.25); setRowCount(rows); for (int i = 0; i < rows; i++) { u32 row_address = address - ((rowCount() / 2) * m_bytes_per_row) + i * m_bytes_per_row; auto* bp_item = new QTableWidgetItem; bp_item->setFlags(Qt::ItemIsEnabled); bp_item->setData(Qt::UserRole, row_address); setItem(i, 0, bp_item); auto* row_item = new QTableWidgetItem(QStringLiteral("%1").arg(row_address, 8, 16, QLatin1Char('0'))); row_item->setData(Qt::UserRole, row_address); row_item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable); setItem(i, 1, row_item); if (row_address == address) row_item->setSelected(true); if (Core::GetState() != Core::State::Paused || !accessors->IsValidAddress(row_address)) { for (int c = 2; c < columnCount(); c++) { auto* item = new QTableWidgetItem(QStringLiteral("-")); item->setFlags(Qt::ItemIsEnabled); item->setData(Qt::UserRole, row_address); setItem(i, c, item); } continue; } bool row_breakpoint = true; auto update_values = [&](auto value_to_string) { for (int c = 0; c < data_columns; c++) { auto* cell_item = new QTableWidgetItem; cell_item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable); if (m_type == Type::Signed32 || m_type == Type::Unsigned32 || m_type == Type::Signed16 || m_type == Type::Unsigned16 || m_type == Type::Signed8 || m_type == Type::Unsigned8) cell_item->setTextAlignment(Qt::AlignRight); const u32 cell_address = row_address + c * GetTypeSize(m_type); // GetMemCheck is more accurate than OverlapsMemcheck, unless standard alginments are // enforced. if (m_address_space == AddressSpace::Type::Effective && PowerPC::memchecks.GetMemCheck(cell_address, GetTypeSize(m_type)) != nullptr) { cell_item->setBackground(Qt::red); } else { row_breakpoint = false; } setItem(i, 2 + c, cell_item); if (accessors->IsValidAddress(cell_address)) { cell_item->setText(value_to_string(cell_address)); cell_item->setData(Qt::UserRole, cell_address); } else { cell_item->setFlags({}); cell_item->setText(QStringLiteral("-")); } } }; switch (m_type) { case Type::Hex8: update_values([&accessors](u32 address) { const u8 value = accessors->ReadU8(address); return QStringLiteral("%1").arg(value, 2, 16, QLatin1Char('0')); }); break; case Type::ASCII: update_values([&accessors](u32 address) { const char value = accessors->ReadU8(address); return IsPrintableCharacter(value) ? QString{QChar::fromLatin1(value)} : QString{QChar::fromLatin1('.')}; }); break; case Type::Hex16: update_values([&accessors](u32 address) { const u16 value = accessors->ReadU16(address); return QStringLiteral("%1").arg(value, 4, 16, QLatin1Char('0')); }); break; case Type::Hex32: update_values([&accessors](u32 address) { const u32 value = accessors->ReadU32(address); return QStringLiteral("%1").arg(value, 8, 16, QLatin1Char('0')); }); break; case Type::Unsigned8: update_values( [&accessors](u32 address) { return QString::number(accessors->ReadU8(address)); }); break; case Type::Unsigned16: update_values( [&accessors](u32 address) { return QString::number(accessors->ReadU16(address)); }); break; case Type::Unsigned32: update_values( [&accessors](u32 address) { return QString::number(accessors->ReadU32(address)); }); break; case Type::Signed8: update_values([&accessors](u32 address) { return QString::number(Common::BitCast(accessors->ReadU8(address))); }); break; case Type::Signed16: update_values([&accessors](u32 address) { return QString::number(Common::BitCast(accessors->ReadU16(address))); }); break; case Type::Signed32: update_values([&accessors](u32 address) { return QString::number(Common::BitCast(accessors->ReadU32(address))); }); break; case Type::Float32: update_values([&accessors](u32 address) { QString string = QString::number(accessors->ReadF32(address), 'g', 4); // Align to first digit. if (!string.startsWith(QLatin1Char('-'))) string.prepend(QLatin1Char(' ')); return string; }); break; case Type::Double: update_values([&accessors](u32 address) { QString string = QString::number(Common::BitCast(accessors->ReadU64(address)), 'g', 4); // Align to first digit. if (!string.startsWith(QLatin1Char('-'))) string.prepend(QLatin1Char(' ')); return string; }); break; } if (row_breakpoint) { bp_item->setData(Qt::DecorationRole, Resources::GetScaledThemeIcon("debugger_breakpoint") .pixmap(QSize(rowHeight(0) - 3, rowHeight(0) - 3))); } } setColumnWidth(0, rowHeight(0)); // Number of characters possible. int max_length = GetCharacterCount(m_type); // Column width is the max number of characters + 1 or 2 for the space between columns. A longer // length means less columns, so a bigger spacing is fine. max_length += max_length < 8 ? 1 : 2; const int width = m_font_width * max_length; for (int i = 2; i < columnCount(); i++) setColumnWidth(i, width); viewport()->update(); update(); } void MemoryViewWidget::SetAddressSpace(AddressSpace::Type address_space) { if (m_address_space == address_space) { return; } m_address_space = address_space; Update(); } AddressSpace::Type MemoryViewWidget::GetAddressSpace() const { return m_address_space; } void MemoryViewWidget::SetDisplay(Type type, int bytes_per_row, int alignment) { m_type = type; m_bytes_per_row = bytes_per_row; if (alignment == 0) m_alignment = GetTypeSize(type); else m_alignment = alignment; Update(); } void MemoryViewWidget::SetBPType(BPType type) { m_bp_type = type; } void MemoryViewWidget::SetAddress(u32 address) { if (m_address == address) return; m_address = address; Update(); } void MemoryViewWidget::SetBPLoggingEnabled(bool enabled) { m_do_log = enabled; } void MemoryViewWidget::resizeEvent(QResizeEvent*) { Update(); } void MemoryViewWidget::keyPressEvent(QKeyEvent* event) { switch (event->key()) { case Qt::Key_Up: m_address -= 16; Update(); return; case Qt::Key_Down: m_address += 16; Update(); return; case Qt::Key_PageUp: m_address -= rowCount() * 16; Update(); return; case Qt::Key_PageDown: m_address += rowCount() * 16; Update(); return; default: QWidget::keyPressEvent(event); break; } } u32 MemoryViewWidget::GetContextAddress() const { return m_context_address; } void MemoryViewWidget::ToggleRowBreakpoint(bool row) { TMemCheck check; const u32 addr = row ? m_base_address : GetContextAddress(); const auto length = row ? m_bytes_per_row : GetTypeSize(m_type); if (m_address_space == AddressSpace::Type::Effective) { if (!PowerPC::memchecks.OverlapsMemcheck(addr, length)) { check.start_address = addr; check.end_address = check.start_address + length - 1; check.is_ranged = length > 0; check.is_break_on_read = (m_bp_type == BPType::ReadOnly || m_bp_type == BPType::ReadWrite); check.is_break_on_write = (m_bp_type == BPType::WriteOnly || m_bp_type == BPType::ReadWrite); check.log_on_hit = m_do_log; check.break_on_hit = true; PowerPC::memchecks.Add(check); } else { PowerPC::memchecks.Remove(addr); } } emit BreakpointsChanged(); Update(); } void MemoryViewWidget::ToggleBreakpoint() { ToggleRowBreakpoint(false); } void MemoryViewWidget::wheelEvent(QWheelEvent* event) { auto delta = -static_cast(std::round((event->angleDelta().y() / (SCROLL_FRACTION_DEGREES * 8)))); if (delta == 0) return; m_address += delta * 16; Update(); } void MemoryViewWidget::mousePressEvent(QMouseEvent* event) { auto* item_selected = itemAt(event->pos()); if (item_selected == nullptr) return; const u32 addr = item_selected->data(Qt::UserRole).toUInt(); m_context_address = addr; m_base_address = item(row(item_selected), 1)->data(Qt::UserRole).toUInt(); switch (event->button()) { case Qt::LeftButton: if (column(item_selected) == 0) ToggleRowBreakpoint(true); else SetAddress(m_base_address); Update(); break; default: break; } } void MemoryViewWidget::OnCopyAddress() { u32 addr = GetContextAddress(); QApplication::clipboard()->setText(QStringLiteral("%1").arg(addr, 8, 16, QLatin1Char('0'))); } void MemoryViewWidget::OnCopyHex() { u32 addr = GetContextAddress(); const auto length = GetTypeSize(m_type); const AddressSpace::Accessors* accessors = AddressSpace::GetAccessors(m_address_space); u64 value = accessors->ReadU64(addr); QApplication::clipboard()->setText( QStringLiteral("%1").arg(value, sizeof(u64) * 2, 16, QLatin1Char('0')).left(length * 2)); } void MemoryViewWidget::OnContextMenu() { auto* menu = new QMenu(this); menu->addAction(tr("Copy Address"), this, &MemoryViewWidget::OnCopyAddress); auto* copy_hex = menu->addAction(tr("Copy Hex"), this, &MemoryViewWidget::OnCopyHex); const AddressSpace::Accessors* accessors = AddressSpace::GetAccessors(m_address_space); copy_hex->setEnabled(Core::GetState() != Core::State::Uninitialized && accessors->IsValidAddress(GetContextAddress())); menu->addSeparator(); menu->addAction(tr("Show in code"), this, [this] { emit ShowCode(GetContextAddress()); }); menu->addSeparator(); menu->addAction(tr("Add to watch"), this, [this] { const u32 address = GetContextAddress(); const QString name = QStringLiteral("mem_%1").arg(address, 8, 16, QLatin1Char('0')); emit RequestWatch(name, address); }); menu->addAction(tr("Toggle Breakpoint"), this, &MemoryViewWidget::ToggleBreakpoint); menu->exec(QCursor::pos()); }