|
|
@ -2,53 +2,21 @@ |
|
|
// Licensed under GPLv2
|
|
|
// Licensed under GPLv2
|
|
|
// Refer to the license.txt file included.
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
|
#include "graphics_cmdlists.hxx"
|
|
|
|
|
|
|
|
|
#include <QListView>
|
|
|
|
|
|
#include <QPushButton>
|
|
|
|
|
|
#include <QVBoxLayout>
|
|
|
#include <QTreeView>
|
|
|
#include <QTreeView>
|
|
|
|
|
|
|
|
|
extern GraphicsDebugger g_debugger; |
|
|
|
|
|
|
|
|
|
|
|
GPUCommandListModel::GPUCommandListModel(QObject* parent) : QAbstractItemModel(parent) |
|
|
|
|
|
{ |
|
|
|
|
|
root_item = new TreeItem(TreeItem::ROOT, 0, NULL, this); |
|
|
|
|
|
|
|
|
|
|
|
connect(this, SIGNAL(CommandListCalled()), this, SLOT(OnCommandListCalledInternal()), Qt::UniqueConnection); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
QModelIndex GPUCommandListModel::index(int row, int column, const QModelIndex& parent) const |
|
|
|
|
|
{ |
|
|
|
|
|
TreeItem* item; |
|
|
|
|
|
|
|
|
|
|
|
if (!parent.isValid()) { |
|
|
|
|
|
item = root_item; |
|
|
|
|
|
} else { |
|
|
|
|
|
item = (TreeItem*)parent.internalPointer(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return createIndex(row, column, item->children[row]); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
#include "graphics_cmdlists.hxx"
|
|
|
|
|
|
|
|
|
QModelIndex GPUCommandListModel::parent(const QModelIndex& child) const |
|
|
|
|
|
|
|
|
GPUCommandListModel::GPUCommandListModel(QObject* parent) : QAbstractListModel(parent) |
|
|
{ |
|
|
{ |
|
|
if (!child.isValid()) |
|
|
|
|
|
return QModelIndex(); |
|
|
|
|
|
|
|
|
|
|
|
TreeItem* item = (TreeItem*)child.internalPointer(); |
|
|
|
|
|
|
|
|
|
|
|
if (item->parent == NULL) |
|
|
|
|
|
return QModelIndex(); |
|
|
|
|
|
|
|
|
|
|
|
return createIndex(item->parent->index, 0, item->parent); |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
int GPUCommandListModel::rowCount(const QModelIndex& parent) const |
|
|
int GPUCommandListModel::rowCount(const QModelIndex& parent) const |
|
|
{ |
|
|
{ |
|
|
TreeItem* item; |
|
|
|
|
|
if (!parent.isValid()) { |
|
|
|
|
|
item = root_item; |
|
|
|
|
|
} else { |
|
|
|
|
|
item = (TreeItem*)parent.internalPointer(); |
|
|
|
|
|
} |
|
|
|
|
|
return item->children.size(); |
|
|
|
|
|
|
|
|
return pica_trace.writes.size(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
int GPUCommandListModel::columnCount(const QModelIndex& parent) const |
|
|
int GPUCommandListModel::columnCount(const QModelIndex& parent) const |
|
|
@ -61,79 +29,67 @@ QVariant GPUCommandListModel::data(const QModelIndex& index, int role) const |
|
|
if (!index.isValid()) |
|
|
if (!index.isValid()) |
|
|
return QVariant(); |
|
|
return QVariant(); |
|
|
|
|
|
|
|
|
const TreeItem* item = (const TreeItem*)index.internalPointer(); |
|
|
|
|
|
|
|
|
|
|
|
if (item->type == TreeItem::COMMAND_LIST) |
|
|
|
|
|
{ |
|
|
|
|
|
const GraphicsDebugger::PicaCommandList& cmdlist = command_lists[item->index].second; |
|
|
|
|
|
u32 address = command_lists[item->index].first; |
|
|
|
|
|
|
|
|
|
|
|
if (role == Qt::DisplayRole && index.column() == 0) |
|
|
|
|
|
{ |
|
|
|
|
|
return QVariant(QString("0x%1 bytes at 0x%2").arg(cmdlist.size(), 0, 16).arg(address, 8, 16, QLatin1Char('0'))); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
else |
|
|
|
|
|
{ |
|
|
|
|
|
// index refers to a specific command
|
|
|
|
|
|
const GraphicsDebugger::PicaCommandList& cmdlist = command_lists[item->parent->index].second; |
|
|
|
|
|
const GraphicsDebugger::PicaCommand& cmd = cmdlist[item->index]; |
|
|
|
|
|
const Pica::CommandProcessor::CommandHeader& header = cmd.GetHeader(); |
|
|
|
|
|
|
|
|
const auto& writes = pica_trace.writes; |
|
|
|
|
|
const Pica::CommandProcessor::CommandHeader cmd{writes[index.row()].Id()}; |
|
|
|
|
|
const u32 val{writes[index.row()].Value()}; |
|
|
|
|
|
|
|
|
if (role == Qt::DisplayRole) { |
|
|
if (role == Qt::DisplayRole) { |
|
|
QString content; |
|
|
QString content; |
|
|
if (index.column() == 0) { |
|
|
if (index.column() == 0) { |
|
|
content = QString::fromLatin1(Pica::Regs::GetCommandName(header.cmd_id).c_str()); |
|
|
|
|
|
|
|
|
content = QString::fromLatin1(Pica::Regs::GetCommandName(cmd.cmd_id).c_str()); |
|
|
content.append(" "); |
|
|
content.append(" "); |
|
|
} else if (index.column() == 1) { |
|
|
} else if (index.column() == 1) { |
|
|
for (int j = 0; j < cmd.size(); ++j) |
|
|
|
|
|
content.append(QString("%1 ").arg(cmd[j], 8, 16, QLatin1Char('0'))); |
|
|
|
|
|
|
|
|
content.append(QString("%1 ").arg(cmd.hex, 8, 16, QLatin1Char('0'))); |
|
|
|
|
|
content.append(QString("%1 ").arg(val, 8, 16, QLatin1Char('0'))); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return QVariant(content); |
|
|
return QVariant(content); |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return QVariant(); |
|
|
return QVariant(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void GPUCommandListModel::OnCommandListCalled(const GraphicsDebugger::PicaCommandList& lst, bool is_new) |
|
|
|
|
|
|
|
|
void GPUCommandListModel::OnPicaTraceFinished(const Pica::DebugUtils::PicaTrace& trace) |
|
|
{ |
|
|
{ |
|
|
emit CommandListCalled(); |
|
|
|
|
|
|
|
|
beginResetModel(); |
|
|
|
|
|
|
|
|
|
|
|
pica_trace = trace; |
|
|
|
|
|
|
|
|
|
|
|
endResetModel(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void GPUCommandListModel::OnCommandListCalledInternal() |
|
|
|
|
|
|
|
|
GPUCommandListWidget::GPUCommandListWidget(QWidget* parent) : QDockWidget(tr("Pica Command List"), parent) |
|
|
{ |
|
|
{ |
|
|
beginResetModel(); |
|
|
|
|
|
|
|
|
GPUCommandListModel* model = new GPUCommandListModel(this); |
|
|
|
|
|
|
|
|
command_lists = GetDebugger()->GetCommandLists(); |
|
|
|
|
|
|
|
|
QWidget* main_widget = new QWidget; |
|
|
|
|
|
|
|
|
// delete root item and rebuild tree
|
|
|
|
|
|
delete root_item; |
|
|
|
|
|
root_item = new TreeItem(TreeItem::ROOT, 0, NULL, this); |
|
|
|
|
|
|
|
|
QTreeView* list_widget = new QTreeView; |
|
|
|
|
|
list_widget->setModel(model); |
|
|
|
|
|
list_widget->setFont(QFont("monospace")); |
|
|
|
|
|
list_widget->setRootIsDecorated(false); |
|
|
|
|
|
|
|
|
for (int command_list_idx = 0; command_list_idx < command_lists.size(); ++command_list_idx) { |
|
|
|
|
|
TreeItem* command_list_item = new TreeItem(TreeItem::COMMAND_LIST, command_list_idx, root_item, root_item); |
|
|
|
|
|
root_item->children.push_back(command_list_item); |
|
|
|
|
|
|
|
|
QPushButton* toggle_tracing = new QPushButton(tr("Start Tracing")); |
|
|
|
|
|
|
|
|
const GraphicsDebugger::PicaCommandList& command_list = command_lists[command_list_idx].second; |
|
|
|
|
|
for (int command_idx = 0; command_idx < command_list.size(); ++command_idx) { |
|
|
|
|
|
TreeItem* command_item = new TreeItem(TreeItem::COMMAND, command_idx, command_list_item, command_list_item); |
|
|
|
|
|
command_list_item->children.push_back(command_item); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
connect(toggle_tracing, SIGNAL(clicked()), this, SLOT(OnToggleTracing())); |
|
|
|
|
|
connect(this, SIGNAL(TracingFinished(const Pica::DebugUtils::PicaTrace&)), |
|
|
|
|
|
model, SLOT(OnPicaTraceFinished(const Pica::DebugUtils::PicaTrace&))); |
|
|
|
|
|
|
|
|
endResetModel(); |
|
|
|
|
|
|
|
|
QVBoxLayout* main_layout = new QVBoxLayout; |
|
|
|
|
|
main_layout->addWidget(list_widget); |
|
|
|
|
|
main_layout->addWidget(toggle_tracing); |
|
|
|
|
|
main_widget->setLayout(main_layout); |
|
|
|
|
|
|
|
|
|
|
|
setWidget(main_widget); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
GPUCommandListWidget::GPUCommandListWidget(QWidget* parent) : QDockWidget(tr("Pica Command List"), parent) |
|
|
|
|
|
|
|
|
void GPUCommandListWidget::OnToggleTracing() |
|
|
{ |
|
|
{ |
|
|
GPUCommandListModel* model = new GPUCommandListModel(this); |
|
|
|
|
|
g_debugger.RegisterObserver(model); |
|
|
|
|
|
|
|
|
|
|
|
QTreeView* tree_widget = new QTreeView; |
|
|
|
|
|
tree_widget->setModel(model); |
|
|
|
|
|
tree_widget->setFont(QFont("monospace")); |
|
|
|
|
|
setWidget(tree_widget); |
|
|
|
|
|
|
|
|
if (!Pica::DebugUtils::IsPicaTracing()) { |
|
|
|
|
|
Pica::DebugUtils::StartPicaTracing(); |
|
|
|
|
|
} else { |
|
|
|
|
|
pica_trace = Pica::DebugUtils::FinishPicaTracing(); |
|
|
|
|
|
emit TracingFinished(*pica_trace); |
|
|
|
|
|
} |
|
|
} |
|
|
} |