2014-12-10 18:24:56 +00:00
|
|
|
// Copyright 2014 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <iomanip>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include <QBoxLayout>
|
2015-05-27 14:20:46 +00:00
|
|
|
#include <QLabel>
|
|
|
|
#include <QPushButton>
|
2014-12-10 18:24:56 +00:00
|
|
|
#include <QTreeView>
|
|
|
|
|
2015-07-21 23:09:11 +00:00
|
|
|
#include "video_core/shader/shader_interpreter.h"
|
2014-12-10 18:24:56 +00:00
|
|
|
|
|
|
|
#include "graphics_vertex_shader.h"
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
using nihstro::OpCode;
|
2014-12-10 18:24:56 +00:00
|
|
|
using nihstro::Instruction;
|
|
|
|
using nihstro::SourceRegister;
|
|
|
|
using nihstro::SwizzlePattern;
|
|
|
|
|
|
|
|
GraphicsVertexShaderModel::GraphicsVertexShaderModel(QObject* parent): QAbstractItemModel(parent) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex GraphicsVertexShaderModel::index(int row, int column, const QModelIndex& parent) const {
|
|
|
|
return createIndex(row, column);
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex GraphicsVertexShaderModel::parent(const QModelIndex& child) const {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
int GraphicsVertexShaderModel::columnCount(const QModelIndex& parent) const {
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GraphicsVertexShaderModel::rowCount(const QModelIndex& parent) const {
|
2015-06-28 06:41:29 +00:00
|
|
|
return static_cast<int>(info.code.size());
|
2014-12-10 18:24:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant GraphicsVertexShaderModel::headerData(int section, Qt::Orientation orientation, int role) const {
|
|
|
|
switch(role) {
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
{
|
|
|
|
if (section == 0) {
|
|
|
|
return tr("Offset");
|
|
|
|
} else if (section == 1) {
|
|
|
|
return tr("Raw");
|
|
|
|
} else if (section == 2) {
|
|
|
|
return tr("Disassembly");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant GraphicsVertexShaderModel::data(const QModelIndex& index, int role) const {
|
|
|
|
switch (role) {
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
{
|
|
|
|
switch (index.column()) {
|
|
|
|
case 0:
|
|
|
|
if (info.HasLabel(index.row()))
|
|
|
|
return QString::fromStdString(info.GetLabel(index.row()));
|
|
|
|
|
|
|
|
return QString("%1").arg(4*index.row(), 4, 16, QLatin1Char('0'));
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
return QString("%1").arg(info.code[index.row()].hex, 8, 16, QLatin1Char('0'));
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
std::stringstream output;
|
|
|
|
output.flags(std::ios::hex);
|
|
|
|
|
|
|
|
Instruction instr = info.code[index.row()];
|
|
|
|
const SwizzlePattern& swizzle = info.swizzle_info[instr.common.operand_desc_id].pattern;
|
|
|
|
|
|
|
|
// longest known instruction name: "setemit "
|
2015-03-08 20:52:38 +00:00
|
|
|
output << std::setw(8) << std::left << instr.opcode.Value().GetInfo().name;
|
2014-12-10 18:24:56 +00:00
|
|
|
|
|
|
|
// e.g. "-c92.xyzw"
|
|
|
|
static auto print_input = [](std::stringstream& output, const SourceRegister& input,
|
|
|
|
bool negate, const std::string& swizzle_mask) {
|
|
|
|
output << std::setw(4) << std::right << (negate ? "-" : "") + input.GetName();
|
|
|
|
output << "." << swizzle_mask;
|
|
|
|
};
|
|
|
|
|
|
|
|
// e.g. "-c92[a0.x].xyzw"
|
|
|
|
static auto print_input_indexed = [](std::stringstream& output, const SourceRegister& input,
|
|
|
|
bool negate, const std::string& swizzle_mask,
|
|
|
|
const std::string& address_register_name) {
|
|
|
|
std::string relative_address;
|
|
|
|
if (!address_register_name.empty())
|
|
|
|
relative_address = "[" + address_register_name + "]";
|
|
|
|
|
|
|
|
output << std::setw(10) << std::right << (negate ? "-" : "") + input.GetName() + relative_address;
|
|
|
|
output << "." << swizzle_mask;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Use print_input or print_input_indexed depending on whether relative addressing is used or not.
|
|
|
|
static auto print_input_indexed_compact = [](std::stringstream& output, const SourceRegister& input,
|
|
|
|
bool negate, const std::string& swizzle_mask,
|
|
|
|
const std::string& address_register_name) {
|
|
|
|
if (address_register_name.empty())
|
|
|
|
print_input(output, input, negate, swizzle_mask);
|
|
|
|
else
|
|
|
|
print_input_indexed(output, input, negate, swizzle_mask, address_register_name);
|
|
|
|
};
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
switch (instr.opcode.Value().GetInfo().type) {
|
|
|
|
case OpCode::Type::Trivial:
|
2014-12-10 18:24:56 +00:00
|
|
|
// Nothing to do here
|
|
|
|
break;
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
case OpCode::Type::Arithmetic:
|
2014-12-10 18:24:56 +00:00
|
|
|
{
|
|
|
|
// Use custom code for special instructions
|
2015-03-08 20:52:38 +00:00
|
|
|
switch (instr.opcode.Value().EffectiveOpCode()) {
|
|
|
|
case OpCode::Id::CMP:
|
2014-12-10 18:24:56 +00:00
|
|
|
{
|
|
|
|
// NOTE: CMP always writes both cc components, so we do not consider the dest mask here.
|
|
|
|
output << std::setw(4) << std::right << "cc.";
|
|
|
|
output << "xy ";
|
|
|
|
|
|
|
|
SourceRegister src1 = instr.common.GetSrc1(false);
|
|
|
|
SourceRegister src2 = instr.common.GetSrc2(false);
|
|
|
|
|
|
|
|
print_input_indexed_compact(output, src1, swizzle.negate_src1, swizzle.SelectorToString(false).substr(0,1), instr.common.AddressRegisterName());
|
|
|
|
output << " " << instr.common.compare_op.ToString(instr.common.compare_op.x) << " ";
|
2015-04-21 21:16:43 +00:00
|
|
|
print_input(output, src2, swizzle.negate_src2, swizzle.SelectorToString(true).substr(0,1));
|
2014-12-10 18:24:56 +00:00
|
|
|
|
|
|
|
output << ", ";
|
|
|
|
|
|
|
|
print_input_indexed_compact(output, src1, swizzle.negate_src1, swizzle.SelectorToString(false).substr(1,1), instr.common.AddressRegisterName());
|
|
|
|
output << " " << instr.common.compare_op.ToString(instr.common.compare_op.y) << " ";
|
2015-04-21 21:16:43 +00:00
|
|
|
print_input(output, src2, swizzle.negate_src2, swizzle.SelectorToString(true).substr(1,1));
|
2014-12-10 18:24:56 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
2015-03-08 20:52:38 +00:00
|
|
|
bool src_is_inverted = 0 != (instr.opcode.Value().GetInfo().subtype & OpCode::Info::SrcInversed);
|
2014-12-10 18:24:56 +00:00
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::Dest) {
|
2014-12-10 18:24:56 +00:00
|
|
|
// e.g. "r12.xy__"
|
2015-03-08 20:52:38 +00:00
|
|
|
output << std::setw(4) << std::right << instr.common.dest.Value().GetName() + ".";
|
2014-12-10 18:24:56 +00:00
|
|
|
output << swizzle.DestMaskToString();
|
2015-03-08 20:52:38 +00:00
|
|
|
} else if (instr.opcode.Value().GetInfo().subtype == OpCode::Info::MOVA) {
|
2014-12-10 18:24:56 +00:00
|
|
|
output << std::setw(4) << std::right << "a0.";
|
|
|
|
output << swizzle.DestMaskToString();
|
|
|
|
} else {
|
|
|
|
output << " ";
|
|
|
|
}
|
|
|
|
output << " ";
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::Src1) {
|
2014-12-10 18:24:56 +00:00
|
|
|
SourceRegister src1 = instr.common.GetSrc1(src_is_inverted);
|
|
|
|
print_input_indexed(output, src1, swizzle.negate_src1, swizzle.SelectorToString(false), instr.common.AddressRegisterName());
|
|
|
|
} else {
|
|
|
|
output << " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: In some cases, the Address Register is used as an index for SRC2 instead of SRC1
|
2015-03-08 20:52:38 +00:00
|
|
|
if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::Src2) {
|
2014-12-10 18:24:56 +00:00
|
|
|
SourceRegister src2 = instr.common.GetSrc2(src_is_inverted);
|
2015-04-21 21:16:43 +00:00
|
|
|
print_input(output, src2, swizzle.negate_src2, swizzle.SelectorToString(true));
|
2014-12-10 18:24:56 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
case OpCode::Type::Conditional:
|
2014-12-10 18:24:56 +00:00
|
|
|
{
|
2015-03-08 20:52:38 +00:00
|
|
|
switch (instr.opcode.Value().EffectiveOpCode()) {
|
|
|
|
case OpCode::Id::LOOP:
|
2014-12-10 18:24:56 +00:00
|
|
|
output << "(unknown instruction format)";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
output << "if ";
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasCondition) {
|
2014-12-10 18:24:56 +00:00
|
|
|
const char* ops[] = {
|
|
|
|
" || ", " && ", "", ""
|
|
|
|
};
|
|
|
|
if (instr.flow_control.op != instr.flow_control.JustY)
|
|
|
|
output << ((!instr.flow_control.refx) ? "!" : " ") << "cc.x";
|
|
|
|
|
|
|
|
output << ops[instr.flow_control.op];
|
|
|
|
|
|
|
|
if (instr.flow_control.op != instr.flow_control.JustX)
|
|
|
|
output << ((!instr.flow_control.refy) ? "!" : " ") << "cc.y";
|
|
|
|
|
|
|
|
output << " ";
|
2015-03-08 20:52:38 +00:00
|
|
|
} else if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasUniformIndex) {
|
2014-12-10 18:24:56 +00:00
|
|
|
output << "b" << instr.flow_control.bool_uniform_id << " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 target_addr = instr.flow_control.dest_offset;
|
|
|
|
u32 target_addr_else = instr.flow_control.dest_offset;
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasAlternative) {
|
2014-12-10 18:24:56 +00:00
|
|
|
output << "else jump to 0x" << std::setw(4) << std::right << std::setfill('0') << 4 * instr.flow_control.dest_offset << " ";
|
2015-03-08 20:52:38 +00:00
|
|
|
} else if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasExplicitDest) {
|
2014-12-10 18:24:56 +00:00
|
|
|
output << "jump to 0x" << std::setw(4) << std::right << std::setfill('0') << 4 * instr.flow_control.dest_offset << " ";
|
|
|
|
} else {
|
|
|
|
// TODO: Handle other cases
|
|
|
|
}
|
|
|
|
|
2015-03-08 20:52:38 +00:00
|
|
|
if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasFinishPoint) {
|
2014-12-10 18:24:56 +00:00
|
|
|
output << "(return on " << std::setw(4) << std::right << std::setfill('0')
|
|
|
|
<< 4 * instr.flow_control.dest_offset + 4 * instr.flow_control.num_instructions << ")";
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
output << "(unknown instruction format)";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return QString::fromLatin1(output.str().c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
case Qt::FontRole:
|
|
|
|
return QFont("monospace");
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsVertexShaderModel::OnUpdate()
|
|
|
|
{
|
|
|
|
beginResetModel();
|
|
|
|
|
|
|
|
info.Clear();
|
|
|
|
|
2015-05-27 14:20:46 +00:00
|
|
|
auto& shader_setup = Pica::g_state.vs;
|
|
|
|
for (auto instr : shader_setup.program_code)
|
2014-12-10 18:24:56 +00:00
|
|
|
info.code.push_back({instr});
|
|
|
|
|
2015-05-27 14:20:46 +00:00
|
|
|
for (auto pattern : shader_setup.swizzle_data)
|
2014-12-10 18:24:56 +00:00
|
|
|
info.swizzle_info.push_back({pattern});
|
|
|
|
|
2015-05-27 14:20:46 +00:00
|
|
|
u32 entry_point = Pica::g_state.regs.vs.main_offset;
|
|
|
|
info.labels.insert({ entry_point, "main" });
|
2014-12-10 18:24:56 +00:00
|
|
|
|
|
|
|
endResetModel();
|
|
|
|
}
|
|
|
|
|
2015-05-27 14:20:46 +00:00
|
|
|
void GraphicsVertexShaderModel::DumpShader() {
|
|
|
|
auto& setup = Pica::g_state.vs;
|
|
|
|
auto& config = Pica::g_state.regs.vs;
|
2014-12-10 18:24:56 +00:00
|
|
|
|
2015-05-27 14:20:46 +00:00
|
|
|
Pica::DebugUtils::DumpShader(setup.program_code.data(), setup.program_code.size(),
|
|
|
|
setup.swizzle_data.data(), setup.swizzle_data.size(),
|
|
|
|
config.main_offset, Pica::g_state.regs.vs_output_attributes);
|
|
|
|
}
|
2014-12-10 18:24:56 +00:00
|
|
|
GraphicsVertexShaderWidget::GraphicsVertexShaderWidget(std::shared_ptr< Pica::DebugContext > debug_context,
|
|
|
|
QWidget* parent)
|
|
|
|
: BreakPointObserverDock(debug_context, "Pica Vertex Shader", parent) {
|
|
|
|
setObjectName("PicaVertexShader");
|
|
|
|
|
|
|
|
auto binary_model = new GraphicsVertexShaderModel(this);
|
|
|
|
auto binary_list = new QTreeView;
|
|
|
|
binary_list->setModel(binary_model);
|
|
|
|
binary_list->setRootIsDecorated(false);
|
|
|
|
binary_list->setAlternatingRowColors(true);
|
|
|
|
|
2015-05-27 14:20:46 +00:00
|
|
|
auto dump_shader = new QPushButton(tr("Dump"));
|
|
|
|
|
|
|
|
connect(dump_shader, SIGNAL(clicked()), binary_model, SLOT(DumpShader()));
|
2014-12-10 18:24:56 +00:00
|
|
|
connect(this, SIGNAL(Update()), binary_model, SLOT(OnUpdate()));
|
|
|
|
|
|
|
|
auto main_widget = new QWidget;
|
|
|
|
auto main_layout = new QVBoxLayout;
|
|
|
|
{
|
|
|
|
auto sub_layout = new QHBoxLayout;
|
|
|
|
sub_layout->addWidget(binary_list);
|
|
|
|
main_layout->addLayout(sub_layout);
|
|
|
|
}
|
2015-05-27 14:20:46 +00:00
|
|
|
main_layout->addWidget(dump_shader);
|
2014-12-10 18:24:56 +00:00
|
|
|
main_widget->setLayout(main_layout);
|
|
|
|
setWidget(main_widget);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsVertexShaderWidget::OnBreakPointHit(Pica::DebugContext::Event event, void* data) {
|
|
|
|
emit Update();
|
|
|
|
widget()->setEnabled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsVertexShaderWidget::OnResumed() {
|
|
|
|
widget()->setEnabled(false);
|
|
|
|
}
|