2014-10-25 20:28:24 +02:00
|
|
|
// Copyright 2014 Citra Emulator Project
|
2014-12-17 06:38:14 +01:00
|
|
|
// Licensed under GPLv2 or any later version
|
2014-10-25 20:28:24 +02:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2015-09-11 06:23:00 +02:00
|
|
|
#include <QLabel>
|
2014-10-25 20:28:24 +02:00
|
|
|
#include <QMetaType>
|
|
|
|
#include <QPushButton>
|
2015-07-23 03:42:59 +02:00
|
|
|
#include <QTreeView>
|
2014-10-25 20:28:24 +02:00
|
|
|
#include <QVBoxLayout>
|
2016-12-21 23:19:12 +01:00
|
|
|
#include "citra_qt/debugger/graphics/graphics_breakpoints.h"
|
|
|
|
#include "citra_qt/debugger/graphics/graphics_breakpoints_p.h"
|
2014-10-25 20:28:24 +02:00
|
|
|
|
|
|
|
BreakPointModel::BreakPointModel(std::shared_ptr<Pica::DebugContext> debug_context, QObject* parent)
|
|
|
|
: QAbstractListModel(parent), context_weak(debug_context),
|
|
|
|
at_breakpoint(debug_context->at_breakpoint),
|
2016-09-19 03:01:46 +02:00
|
|
|
active_breakpoint(debug_context->active_breakpoint) {}
|
2014-10-25 20:28:24 +02:00
|
|
|
|
2020-08-15 15:55:54 +02:00
|
|
|
int BreakPointModel::columnCount([[maybe_unused]] const QModelIndex& parent) const {
|
2015-07-23 03:42:59 +02:00
|
|
|
return 1;
|
2014-10-25 20:28:24 +02:00
|
|
|
}
|
|
|
|
|
2020-08-15 15:55:54 +02:00
|
|
|
int BreakPointModel::rowCount([[maybe_unused]] const QModelIndex& parent) const {
|
2014-10-25 20:28:24 +02:00
|
|
|
return static_cast<int>(Pica::DebugContext::Event::NumEvents);
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
QVariant BreakPointModel::data(const QModelIndex& index, int role) const {
|
2014-10-25 20:28:24 +02:00
|
|
|
const auto event = static_cast<Pica::DebugContext::Event>(index.row());
|
|
|
|
|
|
|
|
switch (role) {
|
2016-09-18 02:38:01 +02:00
|
|
|
case Qt::DisplayRole: {
|
2015-07-23 03:42:59 +02:00
|
|
|
if (index.column() == 0) {
|
2018-10-27 00:33:31 +02:00
|
|
|
return DebugContextEventToString(event);
|
2014-12-04 21:00:06 +01:00
|
|
|
}
|
2015-07-23 03:42:59 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-10-25 20:28:24 +02:00
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
case Qt::CheckStateRole: {
|
2015-07-23 03:42:59 +02:00
|
|
|
if (index.column() == 0)
|
|
|
|
return data(index, Role_IsEnabled).toBool() ? Qt::Checked : Qt::Unchecked;
|
2014-10-25 20:28:24 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
case Qt::BackgroundRole: {
|
2014-10-25 20:28:24 +02:00
|
|
|
if (at_breakpoint && index.row() == static_cast<int>(active_breakpoint)) {
|
|
|
|
return QBrush(QColor(0xE0, 0xE0, 0x10));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
case Role_IsEnabled: {
|
2014-10-25 20:28:24 +02:00
|
|
|
auto context = context_weak.lock();
|
2016-04-24 14:19:49 +02:00
|
|
|
return context && context->breakpoints[(int)event].enabled;
|
2014-10-25 20:28:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
Qt::ItemFlags BreakPointModel::flags(const QModelIndex& index) const {
|
2022-11-04 23:32:57 +01:00
|
|
|
if (!index.isValid()) {
|
|
|
|
return {};
|
|
|
|
}
|
2014-10-25 20:28:24 +02:00
|
|
|
|
2015-07-23 03:42:59 +02:00
|
|
|
Qt::ItemFlags flags = Qt::ItemIsEnabled;
|
2022-11-04 23:32:57 +01:00
|
|
|
if (index.column() == 0) {
|
2015-07-23 03:42:59 +02:00
|
|
|
flags |= Qt::ItemIsUserCheckable;
|
2022-11-04 23:32:57 +01:00
|
|
|
}
|
|
|
|
|
2015-07-23 03:42:59 +02:00
|
|
|
return flags;
|
2014-10-25 20:28:24 +02:00
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
bool BreakPointModel::setData(const QModelIndex& index, const QVariant& value, int role) {
|
2014-10-25 20:28:24 +02:00
|
|
|
const auto event = static_cast<Pica::DebugContext::Event>(index.row());
|
|
|
|
|
|
|
|
switch (role) {
|
2016-09-18 02:38:01 +02:00
|
|
|
case Qt::CheckStateRole: {
|
2015-07-23 03:42:59 +02:00
|
|
|
if (index.column() != 0)
|
|
|
|
return false;
|
|
|
|
|
2014-10-25 20:28:24 +02:00
|
|
|
auto context = context_weak.lock();
|
|
|
|
if (!context)
|
|
|
|
return false;
|
|
|
|
|
2016-04-24 14:19:49 +02:00
|
|
|
context->breakpoints[(int)event].enabled = value == Qt::Checked;
|
2015-07-23 03:42:59 +02:00
|
|
|
QModelIndex changed_index = createIndex(index.row(), 0);
|
2014-10-25 20:28:24 +02:00
|
|
|
emit dataChanged(changed_index, changed_index);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void BreakPointModel::OnBreakPointHit(Pica::DebugContext::Event event) {
|
2014-10-25 20:28:24 +02:00
|
|
|
auto context = context_weak.lock();
|
|
|
|
if (!context)
|
|
|
|
return;
|
|
|
|
|
|
|
|
active_breakpoint = context->active_breakpoint;
|
|
|
|
at_breakpoint = context->at_breakpoint;
|
|
|
|
emit dataChanged(createIndex(static_cast<int>(event), 0),
|
2015-07-23 03:42:59 +02:00
|
|
|
createIndex(static_cast<int>(event), 0));
|
2014-10-25 20:28:24 +02:00
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void BreakPointModel::OnResumed() {
|
2014-10-25 20:28:24 +02:00
|
|
|
auto context = context_weak.lock();
|
|
|
|
if (!context)
|
|
|
|
return;
|
|
|
|
|
|
|
|
at_breakpoint = context->at_breakpoint;
|
|
|
|
emit dataChanged(createIndex(static_cast<int>(active_breakpoint), 0),
|
2015-07-23 03:42:59 +02:00
|
|
|
createIndex(static_cast<int>(active_breakpoint), 0));
|
2014-10-25 20:28:24 +02:00
|
|
|
active_breakpoint = context->active_breakpoint;
|
|
|
|
}
|
|
|
|
|
2018-10-27 00:33:31 +02:00
|
|
|
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");
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
GraphicsBreakPointsWidget::GraphicsBreakPointsWidget(
|
|
|
|
std::shared_ptr<Pica::DebugContext> debug_context, QWidget* parent)
|
2018-03-09 18:54:43 +01:00
|
|
|
: QDockWidget(tr("Pica Breakpoints"), parent), Pica::DebugContext::BreakPointObserver(
|
|
|
|
debug_context) {
|
2019-07-25 21:50:36 +02:00
|
|
|
setObjectName(QStringLiteral("PicaBreakPointsWidget"));
|
2014-10-25 20:28:24 +02:00
|
|
|
|
|
|
|
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;
|
2015-07-23 03:42:59 +02:00
|
|
|
breakpoint_list->setRootIsDecorated(false);
|
|
|
|
breakpoint_list->setHeaderHidden(true);
|
2014-10-25 20:28:24 +02:00
|
|
|
breakpoint_list->setModel(breakpoint_model);
|
|
|
|
|
|
|
|
qRegisterMetaType<Pica::DebugContext::Event>("Pica::DebugContext::Event");
|
|
|
|
|
2017-12-17 22:24:19 +01:00
|
|
|
connect(breakpoint_list, &QTreeView::doubleClicked, this,
|
|
|
|
&GraphicsBreakPointsWidget::OnItemDoubleClicked);
|
2015-07-23 03:42:59 +02:00
|
|
|
|
2017-12-17 22:24:19 +01:00
|
|
|
connect(resume_button, &QPushButton::clicked, this,
|
|
|
|
&GraphicsBreakPointsWidget::OnResumeRequested);
|
2014-10-25 20:28:24 +02:00
|
|
|
|
2017-12-17 22:24:19 +01:00
|
|
|
connect(this, &GraphicsBreakPointsWidget::BreakPointHit, this,
|
|
|
|
&GraphicsBreakPointsWidget::OnBreakPointHit, Qt::BlockingQueuedConnection);
|
|
|
|
connect(this, &GraphicsBreakPointsWidget::Resumed, this, &GraphicsBreakPointsWidget::OnResumed);
|
2014-10-25 20:28:24 +02:00
|
|
|
|
2017-12-17 22:24:19 +01:00
|
|
|
connect(this, &GraphicsBreakPointsWidget::BreakPointHit, breakpoint_model,
|
|
|
|
&BreakPointModel::OnBreakPointHit, Qt::BlockingQueuedConnection);
|
|
|
|
connect(this, &GraphicsBreakPointsWidget::Resumed, breakpoint_model,
|
|
|
|
&BreakPointModel::OnResumed);
|
2014-10-25 20:28:24 +02:00
|
|
|
|
2017-12-17 22:24:19 +01:00
|
|
|
connect(this, &GraphicsBreakPointsWidget::BreakPointsChanged,
|
|
|
|
[this](const QModelIndex& top_left, const QModelIndex& bottom_right) {
|
|
|
|
breakpoint_model->dataChanged(top_left, bottom_right);
|
|
|
|
});
|
2014-10-25 20:28:24 +02:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void GraphicsBreakPointsWidget::OnPicaBreakPointHit(Event event, void* data) {
|
2014-10-25 20:28:24 +02:00
|
|
|
// Process in GUI thread
|
|
|
|
emit BreakPointHit(event, data);
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void GraphicsBreakPointsWidget::OnBreakPointHit(Pica::DebugContext::Event event, void* data) {
|
2014-10-25 20:28:24 +02:00
|
|
|
status_text->setText(tr("Emulation halted at breakpoint"));
|
|
|
|
resume_button->setEnabled(true);
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void GraphicsBreakPointsWidget::OnPicaResume() {
|
2014-10-25 20:28:24 +02:00
|
|
|
// Process in GUI thread
|
|
|
|
emit Resumed();
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void GraphicsBreakPointsWidget::OnResumed() {
|
2014-10-25 20:28:24 +02:00
|
|
|
status_text->setText(tr("Emulation running"));
|
|
|
|
resume_button->setEnabled(false);
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void GraphicsBreakPointsWidget::OnResumeRequested() {
|
2014-10-25 20:28:24 +02:00
|
|
|
if (auto context = context_weak.lock())
|
|
|
|
context->Resume();
|
|
|
|
}
|
|
|
|
|
2016-09-18 02:38:01 +02:00
|
|
|
void GraphicsBreakPointsWidget::OnItemDoubleClicked(const QModelIndex& index) {
|
2015-07-23 03:42:59 +02:00
|
|
|
if (!index.isValid())
|
2014-10-25 20:28:24 +02:00
|
|
|
return;
|
|
|
|
|
2015-07-23 03:42:59 +02:00
|
|
|
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);
|
2014-10-25 20:28:24 +02:00
|
|
|
}
|