| 
									
										
										
										
											2014-12-10 19:24:56 +01: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>
 | 
					
						
							|  |  |  | #include <QTreeView>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-21 19:09:11 -04:00
										 |  |  | #include "video_core/shader/shader_interpreter.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "graphics_vertex_shader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  | using nihstro::OpCode; | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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-27 23:41:29 -07:00
										 |  |  |     return static_cast<int>(info.code.size()); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 13:52:38 -07:00
										 |  |  |             output << std::setw(8) << std::left << instr.opcode.Value().GetInfo().name; | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 13:52:38 -07:00
										 |  |  |             switch (instr.opcode.Value().GetInfo().type) { | 
					
						
							|  |  |  |             case OpCode::Type::Trivial: | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                 // Nothing to do here
 | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |             case OpCode::Type::Arithmetic: | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |             { | 
					
						
							|  |  |  |                 // Use custom code for special instructions
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                 switch (instr.opcode.Value().EffectiveOpCode()) { | 
					
						
							|  |  |  |                 case OpCode::Id::CMP: | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 23:16:43 +02:00
										 |  |  |                     print_input(output, src2, swizzle.negate_src2, swizzle.SelectorToString(true).substr(0,1)); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 23:16:43 +02:00
										 |  |  |                     print_input(output, src2, swizzle.negate_src2, swizzle.SelectorToString(true).substr(1,1)); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                     bool src_is_inverted = 0 != (instr.opcode.Value().GetInfo().subtype & OpCode::Info::SrcInversed); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                     if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::Dest) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                         // e.g. "r12.xy__"
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                         output << std::setw(4) << std::right << instr.common.dest.Value().GetName() + "."; | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                         output << swizzle.DestMaskToString(); | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                     } else if (instr.opcode.Value().GetInfo().subtype == OpCode::Info::MOVA) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                         output << std::setw(4) << std::right << "a0."; | 
					
						
							|  |  |  |                         output << swizzle.DestMaskToString(); | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         output << "        "; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     output << "  "; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                     if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::Src1) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 13:52:38 -07:00
										 |  |  |                     if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::Src2) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                         SourceRegister src2 = instr.common.GetSrc2(src_is_inverted); | 
					
						
							| 
									
										
										
										
											2015-04-21 23:16:43 +02:00
										 |  |  |                         print_input(output, src2, swizzle.negate_src2, swizzle.SelectorToString(true)); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                     } | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |             case OpCode::Type::Conditional: | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                 switch (instr.opcode.Value().EffectiveOpCode()) { | 
					
						
							|  |  |  |                 case OpCode::Id::LOOP: | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                     output << "(unknown instruction format)"; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     output << "if "; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                     if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasCondition) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 13:52:38 -07:00
										 |  |  |                     } else if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasUniformIndex) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 13:52:38 -07:00
										 |  |  |                     if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasAlternative) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |                         output << "else jump to 0x" << std::setw(4) << std::right << std::setfill('0') << 4 * instr.flow_control.dest_offset << " "; | 
					
						
							| 
									
										
										
										
											2015-03-08 13:52:38 -07:00
										 |  |  |                     } else if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasExplicitDest) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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 13:52:38 -07:00
										 |  |  |                     if (instr.opcode.Value().GetInfo().subtype & OpCode::Info::HasFinishPoint) { | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01: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-13 23:29:27 -04:00
										 |  |  |     for (auto instr : Pica::g_state.vs.program_code) | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |         info.code.push_back({instr}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-13 23:29:27 -04:00
										 |  |  |     for (auto pattern : Pica::g_state.vs.swizzle_data) | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |         info.swizzle_info.push_back({pattern}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 00:31:40 +01:00
										 |  |  |     info.labels.insert({ Pica::g_state.regs.vs.main_offset, "main" }); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     endResetModel(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  | } |