citra_qt: Add enhanced texture debugging widgets.
Double-clicking a texture parameter command in the pica command lists will spawn these as a new tab in the pica command list dock area.
This commit is contained in:
parent
fd194d95b0
commit
2793619dce
5 changed files with 209 additions and 19 deletions
|
@ -4,30 +4,39 @@
|
||||||
|
|
||||||
#include <QLabel>
|
#include <QLabel>
|
||||||
#include <QListView>
|
#include <QListView>
|
||||||
|
#include <QMainWindow>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
#include <QVBoxLayout>
|
#include <QVBoxLayout>
|
||||||
#include <QTreeView>
|
#include <QTreeView>
|
||||||
|
#include <QSpinBox>
|
||||||
#include "graphics_cmdlists.hxx"
|
#include <QComboBox>
|
||||||
|
|
||||||
#include "video_core/pica.h"
|
#include "video_core/pica.h"
|
||||||
#include "video_core/math.h"
|
#include "video_core/math.h"
|
||||||
|
|
||||||
#include "video_core/debug_utils/debug_utils.h"
|
#include "video_core/debug_utils/debug_utils.h"
|
||||||
|
|
||||||
|
#include "graphics_cmdlists.hxx"
|
||||||
|
|
||||||
|
#include "util/spinbox.hxx"
|
||||||
|
|
||||||
|
QImage LoadTexture(u8* src, const Pica::DebugUtils::TextureInfo& info) {
|
||||||
|
QImage decoded_image(info.width, info.height, QImage::Format_ARGB32);
|
||||||
|
for (int y = 0; y < info.height; ++y) {
|
||||||
|
for (int x = 0; x < info.width; ++x) {
|
||||||
|
Math::Vec4<u8> color = Pica::DebugUtils::LookupTexture(src, x, y, info);
|
||||||
|
decoded_image.setPixel(x, y, qRgba(color.r(), color.g(), color.b(), color.a()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return decoded_image;
|
||||||
|
}
|
||||||
|
|
||||||
class TextureInfoWidget : public QWidget {
|
class TextureInfoWidget : public QWidget {
|
||||||
public:
|
public:
|
||||||
TextureInfoWidget(u8* src, const Pica::DebugUtils::TextureInfo& info, QWidget* parent = nullptr) : QWidget(parent) {
|
TextureInfoWidget(u8* src, const Pica::DebugUtils::TextureInfo& info, QWidget* parent = nullptr) : QWidget(parent) {
|
||||||
QImage decoded_image(info.width, info.height, QImage::Format_ARGB32);
|
|
||||||
for (int y = 0; y < info.height; ++y) {
|
|
||||||
for (int x = 0; x < info.width; ++x) {
|
|
||||||
Math::Vec4<u8> color = Pica::DebugUtils::LookupTexture(src, x, y, info);
|
|
||||||
decoded_image.setPixel(x, y, qRgba(color.r(), color.g(), color.b(), color.a()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
QLabel* image_widget = new QLabel;
|
QLabel* image_widget = new QLabel;
|
||||||
QPixmap image_pixmap = QPixmap::fromImage(decoded_image);
|
QPixmap image_pixmap = QPixmap::fromImage(LoadTexture(src, info));
|
||||||
image_pixmap = image_pixmap.scaled(200, 100, Qt::KeepAspectRatio, Qt::SmoothTransformation);
|
image_pixmap = image_pixmap.scaled(200, 100, Qt::KeepAspectRatio, Qt::SmoothTransformation);
|
||||||
image_widget->setPixmap(image_pixmap);
|
image_widget->setPixmap(image_pixmap);
|
||||||
|
|
||||||
|
@ -37,6 +46,120 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
TextureInfoDockWidget::TextureInfoDockWidget(const Pica::DebugUtils::TextureInfo& info, QWidget* parent)
|
||||||
|
: QDockWidget(tr("Texture 0x%1").arg(info.address, 8, 16, QLatin1Char('0'))),
|
||||||
|
info(info) {
|
||||||
|
|
||||||
|
QWidget* main_widget = new QWidget;
|
||||||
|
|
||||||
|
QLabel* image_widget = new QLabel;
|
||||||
|
|
||||||
|
connect(this, SIGNAL(UpdatePixmap(const QPixmap&)), image_widget, SLOT(setPixmap(const QPixmap&)));
|
||||||
|
|
||||||
|
CSpinBox* phys_address_spinbox = new CSpinBox;
|
||||||
|
phys_address_spinbox->SetBase(16);
|
||||||
|
phys_address_spinbox->SetRange(0, 0xFFFFFFFF);
|
||||||
|
phys_address_spinbox->SetPrefix("0x");
|
||||||
|
phys_address_spinbox->SetValue(info.address);
|
||||||
|
connect(phys_address_spinbox, SIGNAL(ValueChanged(qint64)), this, SLOT(OnAddressChanged(qint64)));
|
||||||
|
|
||||||
|
QComboBox* format_choice = new QComboBox;
|
||||||
|
format_choice->addItem(tr("RGBA8"));
|
||||||
|
format_choice->addItem(tr("RGB8"));
|
||||||
|
format_choice->addItem(tr("RGBA5551"));
|
||||||
|
format_choice->addItem(tr("RGB565"));
|
||||||
|
format_choice->addItem(tr("RGBA4"));
|
||||||
|
format_choice->setCurrentIndex(static_cast<int>(info.format));
|
||||||
|
connect(format_choice, SIGNAL(currentIndexChanged(int)), this, SLOT(OnFormatChanged(int)));
|
||||||
|
|
||||||
|
QSpinBox* width_spinbox = new QSpinBox;
|
||||||
|
width_spinbox->setMaximum(65535);
|
||||||
|
width_spinbox->setValue(info.width);
|
||||||
|
connect(width_spinbox, SIGNAL(valueChanged(int)), this, SLOT(OnWidthChanged(int)));
|
||||||
|
|
||||||
|
QSpinBox* height_spinbox = new QSpinBox;
|
||||||
|
height_spinbox->setMaximum(65535);
|
||||||
|
height_spinbox->setValue(info.height);
|
||||||
|
connect(height_spinbox, SIGNAL(valueChanged(int)), this, SLOT(OnHeightChanged(int)));
|
||||||
|
|
||||||
|
QSpinBox* stride_spinbox = new QSpinBox;
|
||||||
|
stride_spinbox->setMaximum(65535 * 4);
|
||||||
|
stride_spinbox->setValue(info.stride);
|
||||||
|
connect(stride_spinbox, SIGNAL(valueChanged(int)), this, SLOT(OnStrideChanged(int)));
|
||||||
|
|
||||||
|
QVBoxLayout* main_layout = new QVBoxLayout;
|
||||||
|
main_layout->addWidget(image_widget);
|
||||||
|
|
||||||
|
{
|
||||||
|
QHBoxLayout* sub_layout = new QHBoxLayout;
|
||||||
|
sub_layout->addWidget(new QLabel(tr("Source Address:")));
|
||||||
|
sub_layout->addWidget(phys_address_spinbox);
|
||||||
|
main_layout->addLayout(sub_layout);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
QHBoxLayout* sub_layout = new QHBoxLayout;
|
||||||
|
sub_layout->addWidget(new QLabel(tr("Format")));
|
||||||
|
sub_layout->addWidget(format_choice);
|
||||||
|
main_layout->addLayout(sub_layout);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
QHBoxLayout* sub_layout = new QHBoxLayout;
|
||||||
|
sub_layout->addWidget(new QLabel(tr("Width:")));
|
||||||
|
sub_layout->addWidget(width_spinbox);
|
||||||
|
sub_layout->addStretch();
|
||||||
|
sub_layout->addWidget(new QLabel(tr("Height:")));
|
||||||
|
sub_layout->addWidget(height_spinbox);
|
||||||
|
sub_layout->addStretch();
|
||||||
|
sub_layout->addWidget(new QLabel(tr("Stride:")));
|
||||||
|
sub_layout->addWidget(stride_spinbox);
|
||||||
|
main_layout->addLayout(sub_layout);
|
||||||
|
}
|
||||||
|
|
||||||
|
main_widget->setLayout(main_layout);
|
||||||
|
|
||||||
|
emit UpdatePixmap(ReloadPixmap());
|
||||||
|
|
||||||
|
setWidget(main_widget);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TextureInfoDockWidget::OnAddressChanged(qint64 value)
|
||||||
|
{
|
||||||
|
info.address = value;
|
||||||
|
emit UpdatePixmap(ReloadPixmap());
|
||||||
|
}
|
||||||
|
|
||||||
|
void TextureInfoDockWidget::OnFormatChanged(int value)
|
||||||
|
{
|
||||||
|
info.format = static_cast<Pica::Regs::TextureFormat>(value);
|
||||||
|
emit UpdatePixmap(ReloadPixmap());
|
||||||
|
}
|
||||||
|
|
||||||
|
void TextureInfoDockWidget::OnWidthChanged(int value)
|
||||||
|
{
|
||||||
|
info.width = value;
|
||||||
|
emit UpdatePixmap(ReloadPixmap());
|
||||||
|
}
|
||||||
|
|
||||||
|
void TextureInfoDockWidget::OnHeightChanged(int value)
|
||||||
|
{
|
||||||
|
info.height = value;
|
||||||
|
emit UpdatePixmap(ReloadPixmap());
|
||||||
|
}
|
||||||
|
|
||||||
|
void TextureInfoDockWidget::OnStrideChanged(int value)
|
||||||
|
{
|
||||||
|
info.stride = value;
|
||||||
|
emit UpdatePixmap(ReloadPixmap());
|
||||||
|
}
|
||||||
|
|
||||||
|
QPixmap TextureInfoDockWidget::ReloadPixmap() const
|
||||||
|
{
|
||||||
|
u8* src = Memory::GetPointer(info.address);
|
||||||
|
return QPixmap::fromImage(LoadTexture(src, info));
|
||||||
|
}
|
||||||
|
|
||||||
GPUCommandListModel::GPUCommandListModel(QObject* parent) : QAbstractListModel(parent)
|
GPUCommandListModel::GPUCommandListModel(QObject* parent) : QAbstractListModel(parent)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -106,30 +229,42 @@ void GPUCommandListModel::OnPicaTraceFinished(const Pica::DebugUtils::PicaTrace&
|
||||||
endResetModel();
|
endResetModel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define COMMAND_IN_RANGE(cmd_id, reg_name) \
|
||||||
|
(cmd_id >= PICA_REG_INDEX(reg_name) && \
|
||||||
|
cmd_id < PICA_REG_INDEX(reg_name) + sizeof(decltype(Pica::registers.reg_name)) / 4)
|
||||||
|
|
||||||
|
void GPUCommandListWidget::OnCommandDoubleClicked(const QModelIndex& index)
|
||||||
|
{
|
||||||
|
|
||||||
|
const int command_id = list_widget->model()->data(index, GPUCommandListModel::CommandIdRole).toInt();
|
||||||
|
if (COMMAND_IN_RANGE(command_id, texture0)) {
|
||||||
|
auto info = Pica::DebugUtils::TextureInfo::FromPicaRegister(Pica::registers.texture0,
|
||||||
|
Pica::registers.texture0_format);
|
||||||
|
QMainWindow* main_window = (QMainWindow*)parent();
|
||||||
|
main_window->tabifyDockWidget(this, new TextureInfoDockWidget(info, main_window));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void GPUCommandListWidget::SetCommandInfo(const QModelIndex& index)
|
void GPUCommandListWidget::SetCommandInfo(const QModelIndex& index)
|
||||||
{
|
{
|
||||||
QWidget* new_info_widget;
|
QWidget* new_info_widget;
|
||||||
|
|
||||||
#define COMMAND_IN_RANGE(cmd_id, reg_name) (cmd_id >= PICA_REG_INDEX(reg_name) && cmd_id < PICA_REG_INDEX(reg_name) + sizeof(decltype(Pica::registers.reg_name)) / 4)
|
|
||||||
const int command_id = list_widget->model()->data(index, GPUCommandListModel::CommandIdRole).toInt();
|
const int command_id = list_widget->model()->data(index, GPUCommandListModel::CommandIdRole).toInt();
|
||||||
if (COMMAND_IN_RANGE(command_id, texture0)) {
|
if (COMMAND_IN_RANGE(command_id, texture0)) {
|
||||||
u8* src = Memory::GetPointer(Pica::registers.texture0.GetPhysicalAddress());
|
u8* src = Memory::GetPointer(Pica::registers.texture0.GetPhysicalAddress());
|
||||||
Pica::DebugUtils::TextureInfo info;
|
auto info = Pica::DebugUtils::TextureInfo::FromPicaRegister(Pica::registers.texture0,
|
||||||
info.width = Pica::registers.texture0.width;
|
Pica::registers.texture0_format);
|
||||||
info.height = Pica::registers.texture0.height;
|
|
||||||
info.stride = 3 * Pica::registers.texture0.width;
|
|
||||||
info.format = Pica::registers.texture0_format;
|
|
||||||
new_info_widget = new TextureInfoWidget(src, info);
|
new_info_widget = new TextureInfoWidget(src, info);
|
||||||
} else {
|
} else {
|
||||||
new_info_widget = new QWidget;
|
new_info_widget = new QWidget;
|
||||||
}
|
}
|
||||||
#undef COMMAND_IN_RANGE
|
|
||||||
|
|
||||||
widget()->layout()->removeWidget(command_info_widget);
|
widget()->layout()->removeWidget(command_info_widget);
|
||||||
delete command_info_widget;
|
delete command_info_widget;
|
||||||
widget()->layout()->addWidget(new_info_widget);
|
widget()->layout()->addWidget(new_info_widget);
|
||||||
command_info_widget = new_info_widget;
|
command_info_widget = new_info_widget;
|
||||||
}
|
}
|
||||||
|
#undef COMMAND_IN_RANGE
|
||||||
|
|
||||||
GPUCommandListWidget::GPUCommandListWidget(QWidget* parent) : QDockWidget(tr("Pica Command List"), parent)
|
GPUCommandListWidget::GPUCommandListWidget(QWidget* parent) : QDockWidget(tr("Pica Command List"), parent)
|
||||||
{
|
{
|
||||||
|
@ -145,6 +280,8 @@ GPUCommandListWidget::GPUCommandListWidget(QWidget* parent) : QDockWidget(tr("Pi
|
||||||
|
|
||||||
connect(list_widget->selectionModel(), SIGNAL(currentChanged(const QModelIndex&,const QModelIndex&)),
|
connect(list_widget->selectionModel(), SIGNAL(currentChanged(const QModelIndex&,const QModelIndex&)),
|
||||||
this, SLOT(SetCommandInfo(const QModelIndex&)));
|
this, SLOT(SetCommandInfo(const QModelIndex&)));
|
||||||
|
connect(list_widget, SIGNAL(doubleClicked(const QModelIndex&)),
|
||||||
|
this, SLOT(OnCommandDoubleClicked(const QModelIndex&)));
|
||||||
|
|
||||||
|
|
||||||
toggle_tracing = new QPushButton(tr("Start Tracing"));
|
toggle_tracing = new QPushButton(tr("Start Tracing"));
|
||||||
|
|
|
@ -45,6 +45,8 @@ public:
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void OnToggleTracing();
|
void OnToggleTracing();
|
||||||
|
void OnCommandDoubleClicked(const QModelIndex&);
|
||||||
|
|
||||||
void SetCommandInfo(const QModelIndex&);
|
void SetCommandInfo(const QModelIndex&);
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
|
@ -57,3 +59,25 @@ private:
|
||||||
QWidget* command_info_widget;
|
QWidget* command_info_widget;
|
||||||
QPushButton* toggle_tracing;
|
QPushButton* toggle_tracing;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class TextureInfoDockWidget : public QDockWidget {
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
TextureInfoDockWidget(const Pica::DebugUtils::TextureInfo& info, QWidget* parent = nullptr);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void UpdatePixmap(const QPixmap& pixmap);
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void OnAddressChanged(qint64 value);
|
||||||
|
void OnFormatChanged(int value);
|
||||||
|
void OnWidthChanged(int value);
|
||||||
|
void OnHeightChanged(int value);
|
||||||
|
void OnStrideChanged(int value);
|
||||||
|
|
||||||
|
private:
|
||||||
|
QPixmap ReloadPixmap() const;
|
||||||
|
|
||||||
|
Pica::DebugUtils::TextureInfo info;
|
||||||
|
};
|
||||||
|
|
|
@ -382,6 +382,18 @@ const Math::Vec4<u8> LookupTexture(const u8* source, int x, int y, const Texture
|
||||||
return { source_ptr[2], source_ptr[1], source_ptr[0], 255 };
|
return { source_ptr[2], source_ptr[1], source_ptr[0], 255 };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TextureInfo TextureInfo::FromPicaRegister(const Regs::TextureConfig& config,
|
||||||
|
const Regs::TextureFormat& format)
|
||||||
|
{
|
||||||
|
TextureInfo info;
|
||||||
|
info.address = config.GetPhysicalAddress();
|
||||||
|
info.width = config.width;
|
||||||
|
info.height = config.height;
|
||||||
|
info.format = format;
|
||||||
|
info.stride = Pica::Regs::BytesPerPixel(info.format) * info.width;
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
|
||||||
void DumpTexture(const Pica::Regs::TextureConfig& texture_config, u8* data) {
|
void DumpTexture(const Pica::Regs::TextureConfig& texture_config, u8* data) {
|
||||||
// NOTE: Permanently enabling this just trashes hard disks for no reason.
|
// NOTE: Permanently enabling this just trashes hard disks for no reason.
|
||||||
// Hence, this is currently disabled.
|
// Hence, this is currently disabled.
|
||||||
|
|
|
@ -192,10 +192,14 @@ void OnPicaRegWrite(u32 id, u32 value);
|
||||||
std::unique_ptr<PicaTrace> FinishPicaTracing();
|
std::unique_ptr<PicaTrace> FinishPicaTracing();
|
||||||
|
|
||||||
struct TextureInfo {
|
struct TextureInfo {
|
||||||
|
unsigned int address;
|
||||||
int width;
|
int width;
|
||||||
int height;
|
int height;
|
||||||
int stride;
|
int stride;
|
||||||
Pica::Regs::TextureFormat format;
|
Pica::Regs::TextureFormat format;
|
||||||
|
|
||||||
|
static TextureInfo FromPicaRegister(const Pica::Regs::TextureConfig& config,
|
||||||
|
const Pica::Regs::TextureFormat& format);
|
||||||
};
|
};
|
||||||
|
|
||||||
const Math::Vec4<u8> LookupTexture(const u8* source, int x, int y, const TextureInfo& info);
|
const Math::Vec4<u8> LookupTexture(const u8* source, int x, int y, const TextureInfo& info);
|
||||||
|
|
|
@ -130,7 +130,20 @@ struct Regs {
|
||||||
// Seems like they are luminance formats and compressed textures.
|
// Seems like they are luminance formats and compressed textures.
|
||||||
};
|
};
|
||||||
|
|
||||||
BitField<0, 1, u32> texturing_enable;
|
static unsigned BytesPerPixel(TextureFormat format) {
|
||||||
|
if (format == TextureFormat::RGBA8)
|
||||||
|
return 4;
|
||||||
|
else if (format == TextureFormat::RGB8)
|
||||||
|
return 3;
|
||||||
|
else if (format == TextureFormat::RGBA5551 ||
|
||||||
|
format == TextureFormat::RGB565 ||
|
||||||
|
format == TextureFormat::RGBA4)
|
||||||
|
return 2;
|
||||||
|
else // placeholder
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
BitField< 0, 1, u32> texturing_enable;
|
||||||
TextureConfig texture0;
|
TextureConfig texture0;
|
||||||
INSERT_PADDING_WORDS(0x8);
|
INSERT_PADDING_WORDS(0x8);
|
||||||
BitField<0, 4, TextureFormat> texture0_format;
|
BitField<0, 4, TextureFormat> texture0_format;
|
||||||
|
|
Loading…
Reference in a new issue