// Copyright 2014 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include #include #include #include #include #include "citra_qt/debugger/graphics/graphics_breakpoints.h" #include "citra_qt/debugger/graphics/graphics_breakpoints_p.h" BreakPointModel::BreakPointModel(std::shared_ptr debug_context, QObject* parent) : QAbstractListModel(parent), context_weak(debug_context), at_breakpoint(debug_context->at_breakpoint), active_breakpoint(debug_context->active_breakpoint) {} int BreakPointModel::columnCount([[maybe_unused]] const QModelIndex& parent) const { return 1; } int BreakPointModel::rowCount([[maybe_unused]] const QModelIndex& parent) const { return static_cast(Pica::DebugContext::Event::NumEvents); } QVariant BreakPointModel::data(const QModelIndex& index, int role) const { const auto event = static_cast(index.row()); switch (role) { case Qt::DisplayRole: { if (index.column() == 0) { return DebugContextEventToString(event); } break; } case Qt::CheckStateRole: { if (index.column() == 0) return data(index, Role_IsEnabled).toBool() ? Qt::Checked : Qt::Unchecked; break; } case Qt::BackgroundRole: { if (at_breakpoint && index.row() == static_cast(active_breakpoint)) { return QBrush(QColor(0xE0, 0xE0, 0x10)); } break; } case Role_IsEnabled: { auto context = context_weak.lock(); return context && context->breakpoints[(int)event].enabled; } default: break; } return QVariant(); } Qt::ItemFlags BreakPointModel::flags(const QModelIndex& index) const { if (!index.isValid()) { return {}; } Qt::ItemFlags flags = Qt::ItemIsEnabled; if (index.column() == 0) { flags |= Qt::ItemIsUserCheckable; } return flags; } bool BreakPointModel::setData(const QModelIndex& index, const QVariant& value, int role) { const auto event = static_cast(index.row()); switch (role) { case Qt::CheckStateRole: { if (index.column() != 0) return false; auto context = context_weak.lock(); if (!context) return false; context->breakpoints[(int)event].enabled = value == Qt::Checked; QModelIndex changed_index = createIndex(index.row(), 0); emit dataChanged(changed_index, changed_index); return true; } } return false; } void BreakPointModel::OnBreakPointHit(Pica::DebugContext::Event event) { auto context = context_weak.lock(); if (!context) return; active_breakpoint = context->active_breakpoint; at_breakpoint = context->at_breakpoint; emit dataChanged(createIndex(static_cast(event), 0), createIndex(static_cast(event), 0)); } void BreakPointModel::OnResumed() { auto context = context_weak.lock(); if (!context) return; at_breakpoint = context->at_breakpoint; emit dataChanged(createIndex(static_cast(active_breakpoint), 0), createIndex(static_cast(active_breakpoint), 0)); active_breakpoint = context->active_breakpoint; } QString BreakPointModel::DebugContextEventToString(Pica::DebugContext::Event event) { switch (event) { case Pica::DebugContext::Event::PicaCommandLoaded: return tr("Pica command loaded"); case Pica::DebugContext::Event::PicaCommandProcessed: return tr("Pica command processed"); case Pica::DebugContext::Event::IncomingPrimitiveBatch: return tr("Incoming primitive batch"); case Pica::DebugContext::Event::FinishedPrimitiveBatch: return tr("Finished primitive batch"); case Pica::DebugContext::Event::VertexShaderInvocation: return tr("Vertex shader invocation"); case Pica::DebugContext::Event::IncomingDisplayTransfer: return tr("Incoming display transfer"); case Pica::DebugContext::Event::GSPCommandProcessed: return tr("GSP command processed"); case Pica::DebugContext::Event::BufferSwapped: return tr("Buffers swapped"); case Pica::DebugContext::Event::NumEvents: break; } return tr("Unknown debug context event"); } GraphicsBreakPointsWidget::GraphicsBreakPointsWidget( std::shared_ptr debug_context, QWidget* parent) : QDockWidget(tr("Pica Breakpoints"), parent), Pica::DebugContext::BreakPointObserver( debug_context) { setObjectName(QStringLiteral("PicaBreakPointsWidget")); status_text = new QLabel(tr("Emulation running")); resume_button = new QPushButton(tr("Resume")); resume_button->setEnabled(false); breakpoint_model = new BreakPointModel(debug_context, this); breakpoint_list = new QTreeView; breakpoint_list->setRootIsDecorated(false); breakpoint_list->setHeaderHidden(true); breakpoint_list->setModel(breakpoint_model); qRegisterMetaType("Pica::DebugContext::Event"); connect(breakpoint_list, &QTreeView::doubleClicked, this, &GraphicsBreakPointsWidget::OnItemDoubleClicked); connect(resume_button, &QPushButton::clicked, this, &GraphicsBreakPointsWidget::OnResumeRequested); connect(this, &GraphicsBreakPointsWidget::BreakPointHit, this, &GraphicsBreakPointsWidget::OnBreakPointHit, Qt::BlockingQueuedConnection); connect(this, &GraphicsBreakPointsWidget::Resumed, this, &GraphicsBreakPointsWidget::OnResumed); connect(this, &GraphicsBreakPointsWidget::BreakPointHit, breakpoint_model, &BreakPointModel::OnBreakPointHit, Qt::BlockingQueuedConnection); connect(this, &GraphicsBreakPointsWidget::Resumed, breakpoint_model, &BreakPointModel::OnResumed); connect(this, &GraphicsBreakPointsWidget::BreakPointsChanged, [this](const QModelIndex& top_left, const QModelIndex& bottom_right) { breakpoint_model->dataChanged(top_left, bottom_right); }); QWidget* main_widget = new QWidget; auto main_layout = new QVBoxLayout; { auto sub_layout = new QHBoxLayout; sub_layout->addWidget(status_text); sub_layout->addWidget(resume_button); main_layout->addLayout(sub_layout); } main_layout->addWidget(breakpoint_list); main_widget->setLayout(main_layout); setWidget(main_widget); } void GraphicsBreakPointsWidget::OnPicaBreakPointHit(Event event, const void* data) { // Process in GUI thread emit BreakPointHit(event, data); } void GraphicsBreakPointsWidget::OnBreakPointHit(Pica::DebugContext::Event event, const void* data) { status_text->setText(tr("Emulation halted at breakpoint")); resume_button->setEnabled(true); } void GraphicsBreakPointsWidget::OnPicaResume() { // Process in GUI thread emit Resumed(); } void GraphicsBreakPointsWidget::OnResumed() { status_text->setText(tr("Emulation running")); resume_button->setEnabled(false); } void GraphicsBreakPointsWidget::OnResumeRequested() { if (auto context = context_weak.lock()) context->Resume(); } void GraphicsBreakPointsWidget::OnItemDoubleClicked(const QModelIndex& index) { if (!index.isValid()) return; QModelIndex check_index = breakpoint_list->model()->index(index.row(), 0); QVariant enabled = breakpoint_list->model()->data(check_index, Qt::CheckStateRole); QVariant new_state = Qt::Unchecked; if (enabled == Qt::Unchecked) new_state = Qt::Checked; breakpoint_list->model()->setData(check_index, new_state, Qt::CheckStateRole); }