aboutsummaryrefslogtreecommitdiff
path: root/src/citra_qt/debugger/graphics/graphics.cpp
blob: 021e1c72067d53f3446313e05bcb7c5b303c9ee1 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
// Copyright 2014 Citra Emulator Project
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.

#include <QListView>
#include "citra_qt/debugger/graphics/graphics.h"
#include "citra_qt/util/util.h"
#include "core/core.h"
#include "video_core/gpu.h"

GPUCommandStreamItemModel::GPUCommandStreamItemModel(QObject* parent)
    : QAbstractListModel(parent), command_count(0) {
    connect(this, &GPUCommandStreamItemModel::GXCommandFinished, this,
            &GPUCommandStreamItemModel::OnGXCommandFinishedInternal);
}

int GPUCommandStreamItemModel::rowCount([[maybe_unused]] const QModelIndex& parent) const {
    return command_count;
}

QVariant GPUCommandStreamItemModel::data(const QModelIndex& index, int role) const {
    if (!index.isValid() || !GetDebugger())
        return QVariant();

    int command_index = index.row();
    const Service::GSP::Command& command = GetDebugger()->ReadGXCommandHistory(command_index);
    if (role == Qt::DisplayRole) {
        std::map<Service::GSP::CommandId, const char*> command_names = {
            {Service::GSP::CommandId::RequestDma, "REQUEST_DMA"},
            {Service::GSP::CommandId::SubmitCmdList, "SUBMIT_GPU_CMDLIST"},
            {Service::GSP::CommandId::MemoryFill, "SET_MEMORY_FILL"},
            {Service::GSP::CommandId::DisplayTransfer, "SET_DISPLAY_TRANSFER"},
            {Service::GSP::CommandId::TextureCopy, "SET_TEXTURE_COPY"},
            {Service::GSP::CommandId::CacheFlush, "CACHE_FLUSH"},
        };
        const u32* command_data = reinterpret_cast<const u32*>(&command);
        QString str = QStringLiteral("%1 %2 %3 %4 %5 %6 %7 %8 %9")
                          .arg(QString::fromUtf8(command_names[command.id]))
                          .arg(command_data[0], 8, 16, QLatin1Char('0'))
                          .arg(command_data[1], 8, 16, QLatin1Char('0'))
                          .arg(command_data[2], 8, 16, QLatin1Char('0'))
                          .arg(command_data[3], 8, 16, QLatin1Char('0'))
                          .arg(command_data[4], 8, 16, QLatin1Char('0'))
                          .arg(command_data[5], 8, 16, QLatin1Char('0'))
                          .arg(command_data[6], 8, 16, QLatin1Char('0'))
                          .arg(command_data[7], 8, 16, QLatin1Char('0'));
        return QVariant(str);
    } else {
        return QVariant();
    }
}

void GPUCommandStreamItemModel::GXCommandProcessed(int total_command_count) {
    emit GXCommandFinished(total_command_count);
}

void GPUCommandStreamItemModel::OnGXCommandFinishedInternal(int total_command_count) {
    if (total_command_count == 0)
        return;

    int prev_command_count = command_count;
    command_count = total_command_count;
    emit dataChanged(index(prev_command_count, 0), index(total_command_count - 1, 0));
}

GPUCommandStreamWidget::GPUCommandStreamWidget(Core::System& system_, QWidget* parent)
    : QDockWidget(tr("Graphics Debugger"), parent), system{system_}, model(this) {
    setObjectName(QStringLiteral("GraphicsDebugger"));

    auto* command_list = new QListView;
    command_list->setModel(&model);
    command_list->setFont(GetMonospaceFont());

    setWidget(command_list);
}

void GPUCommandStreamWidget::Register() {
    auto& debugger = system.GPU().Debugger();
    debugger.RegisterObserver(&model);
}

void GPUCommandStreamWidget::Unregister() {
    auto& debugger = system.GPU().Debugger();
    debugger.UnregisterObserver(&model);
}

void GPUCommandStreamWidget::showEvent(QShowEvent* event) {
    if (system.IsPoweredOn()) {
        Register();
    }
    QDockWidget::showEvent(event);
}

void GPUCommandStreamWidget::hideEvent(QHideEvent* event) {
    if (system.IsPoweredOn()) {
        Unregister();
    }
    QDockWidget::hideEvent(event);
}