mirror of
				https://github.com/PabloMK7/citra.git
				synced 2025-10-31 13:50:03 +00:00 
			
		
		
		
	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…
	
	Add table
		Add a link
		
	
		Reference in a new issue