| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  | // Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <list>
 | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  | #include <map>
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  | #include <unordered_map>
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | #include <unordered_set>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "video_core/shader/control_flow.h"
 | 
					
						
							|  |  |  | #include "video_core/shader/shader_ir.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace VideoCommon::Shader { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using Tegra::Shader::Instruction; | 
					
						
							|  |  |  | using Tegra::Shader::OpCode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | constexpr s32 unassigned_branch = -2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  | /***
 | 
					
						
							|  |  |  |  * 'ControlStack' represents a static stack of control jumps such as SSY and PBK | 
					
						
							|  |  |  |  * stacks in Maxwell. | 
					
						
							|  |  |  |  ***/ | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  | struct ControlStack { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     static constexpr std::size_t stack_fixed_size = 20; | 
					
						
							|  |  |  |     std::array<u32, stack_fixed_size> stack{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     u32 index{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool Compare(const ControlStack& cs) const { | 
					
						
							|  |  |  |         if (index != cs.index) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return std::memcmp(stack.data(), cs.stack.data(), index * sizeof(u32)) == 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     /// This compare just compares the top of the stack against one another
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     bool SoftCompare(const ControlStack& cs) const { | 
					
						
							|  |  |  |         if (index == 0 || cs.index == 0) { | 
					
						
							|  |  |  |             return index == cs.index; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return Top() == cs.Top(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 Size() const { | 
					
						
							|  |  |  |         return index; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 Top() const { | 
					
						
							|  |  |  |         return stack[index - 1]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool Push(u32 address) { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |         if (index >= stack.size()) { | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         stack[index] = address; | 
					
						
							|  |  |  |         index++; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool Pop() { | 
					
						
							|  |  |  |         if (index == 0) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         index--; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct Query { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     u32 address{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     ControlStack ssy_stack{}; | 
					
						
							|  |  |  |     ControlStack pbk_stack{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BlockStack { | 
					
						
							|  |  |  |     BlockStack() = default; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     BlockStack(const BlockStack& b) = default; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     BlockStack(const Query& q) : ssy_stack{q.ssy_stack}, pbk_stack{q.pbk_stack} {} | 
					
						
							|  |  |  |     ControlStack ssy_stack{}; | 
					
						
							|  |  |  |     ControlStack pbk_stack{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | struct BlockBranchInfo { | 
					
						
							|  |  |  |     Condition condition{}; | 
					
						
							|  |  |  |     s32 address{exit_branch}; | 
					
						
							|  |  |  |     bool kill{}; | 
					
						
							|  |  |  |     bool is_sync{}; | 
					
						
							|  |  |  |     bool is_brk{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     bool ignore{}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BlockInfo { | 
					
						
							|  |  |  |     u32 start{}; | 
					
						
							|  |  |  |     u32 end{}; | 
					
						
							|  |  |  |     bool visited{}; | 
					
						
							|  |  |  |     BlockBranchInfo branch{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool IsInside(const u32 address) const { | 
					
						
							|  |  |  |         return start <= address && address <= end; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct CFGRebuildState { | 
					
						
							|  |  |  |     explicit CFGRebuildState(const ProgramCode& program_code, const std::size_t program_size) | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |         : program_code{program_code}, program_size{program_size} {} | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::vector<BlockInfo> block_info{}; | 
					
						
							|  |  |  |     std::list<u32> inspect_queries{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     std::list<Query> queries{}; | 
					
						
							|  |  |  |     std::unordered_map<u32, u32> registered{}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     std::unordered_set<u32> labels{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     std::map<u32, u32> ssy_labels{}; | 
					
						
							|  |  |  |     std::map<u32, u32> pbk_labels{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     std::unordered_map<u32, BlockStack> stacks{}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     const ProgramCode& program_code; | 
					
						
							|  |  |  |     const std::size_t program_size; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class BlockCollision : u32 { None = 0, Found = 1, Inside = 2 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::pair<BlockCollision, std::vector<BlockInfo>::iterator> TryGetBlock(CFGRebuildState& state, | 
					
						
							|  |  |  |                                                                         u32 address) { | 
					
						
							|  |  |  |     auto it = state.block_info.begin(); | 
					
						
							|  |  |  |     while (it != state.block_info.end()) { | 
					
						
							|  |  |  |         if (it->start == address) { | 
					
						
							|  |  |  |             return {BlockCollision::Found, it}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (it->IsInside(address)) { | 
					
						
							|  |  |  |             return {BlockCollision::Inside, it}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         it++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return {BlockCollision::None, it}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ParseInfo { | 
					
						
							|  |  |  |     BlockBranchInfo branch_info{}; | 
					
						
							|  |  |  |     u32 end_address{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BlockInfo* CreateBlockInfo(CFGRebuildState& state, u32 start, u32 end) { | 
					
						
							|  |  |  |     auto& it = state.block_info.emplace_back(); | 
					
						
							|  |  |  |     it.start = start; | 
					
						
							|  |  |  |     it.end = end; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const u32 index = static_cast<u32>(state.block_info.size() - 1); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     state.registered.insert({start, index}); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     return ⁢ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Pred GetPredicate(u32 index, bool negated) { | 
					
						
							|  |  |  |     return static_cast<Pred>(index + (negated ? 8 : 0)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class ParseResult : u32 { | 
					
						
							|  |  |  |     ControlCaught = 0, | 
					
						
							|  |  |  |     BlockEnd = 1, | 
					
						
							|  |  |  |     AbnormalFlow = 2, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ParseResult ParseCode(CFGRebuildState& state, u32 address, ParseInfo& parse_info) { | 
					
						
							|  |  |  |     u32 offset = static_cast<u32>(address); | 
					
						
							| 
									
										
										
										
											2019-06-25 20:56:04 -04:00
										 |  |  |     const u32 end_address = static_cast<u32>(state.program_size / 8U); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const auto insert_label = ([](CFGRebuildState& state, u32 address) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         auto pair = state.labels.emplace(address); | 
					
						
							|  |  |  |         if (pair.second) { | 
					
						
							|  |  |  |             state.inspect_queries.push_back(address); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (true) { | 
					
						
							|  |  |  |         if (offset >= end_address) { | 
					
						
							|  |  |  |             parse_info.branch_info.address = exit_branch; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         if (state.registered.count(offset) != 0) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.branch_info.address = offset; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = true; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Instruction instr = {state.program_code[offset]}; | 
					
						
							|  |  |  |         const auto opcode = OpCode::Decode(instr); | 
					
						
							|  |  |  |         if (!opcode || opcode->get().GetType() != OpCode::Type::Flow) { | 
					
						
							|  |  |  |             offset++; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch (opcode->get().GetId()) { | 
					
						
							|  |  |  |         case OpCode::Id::EXIT: { | 
					
						
							|  |  |  |             const auto pred_index = static_cast<u32>(instr.pred.pred_index); | 
					
						
							|  |  |  |             parse_info.branch_info.condition.predicate = | 
					
						
							|  |  |  |                 GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (parse_info.branch_info.condition.predicate == Pred::NeverExecute) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							|  |  |  |             parse_info.branch_info.condition.cc = cc; | 
					
						
							|  |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             parse_info.branch_info.address = exit_branch; | 
					
						
							|  |  |  |             parse_info.branch_info.kill = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_sync = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_brk = false; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return ParseResult::ControlCaught; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::BRA: { | 
					
						
							|  |  |  |             if (instr.bra.constant_buffer != 0) { | 
					
						
							|  |  |  |                 return ParseResult::AbnormalFlow; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const auto pred_index = static_cast<u32>(instr.pred.pred_index); | 
					
						
							|  |  |  |             parse_info.branch_info.condition.predicate = | 
					
						
							|  |  |  |                 GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (parse_info.branch_info.condition.predicate == Pred::NeverExecute) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							|  |  |  |             parse_info.branch_info.condition.cc = cc; | 
					
						
							|  |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             u32 branch_offset = offset + instr.bra.GetBranchTarget(); | 
					
						
							|  |  |  |             if (branch_offset == 0) { | 
					
						
							|  |  |  |                 parse_info.branch_info.address = exit_branch; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 parse_info.branch_info.address = branch_offset; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             insert_label(state, branch_offset); | 
					
						
							|  |  |  |             parse_info.branch_info.kill = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_sync = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_brk = false; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return ParseResult::ControlCaught; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::SYNC: { | 
					
						
							|  |  |  |             parse_info.branch_info.condition; | 
					
						
							|  |  |  |             const auto pred_index = static_cast<u32>(instr.pred.pred_index); | 
					
						
							|  |  |  |             parse_info.branch_info.condition.predicate = | 
					
						
							|  |  |  |                 GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (parse_info.branch_info.condition.predicate == Pred::NeverExecute) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							|  |  |  |             parse_info.branch_info.condition.cc = cc; | 
					
						
							|  |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             parse_info.branch_info.address = unassigned_branch; | 
					
						
							|  |  |  |             parse_info.branch_info.kill = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_sync = true; | 
					
						
							|  |  |  |             parse_info.branch_info.is_brk = false; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return ParseResult::ControlCaught; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::BRK: { | 
					
						
							|  |  |  |             parse_info.branch_info.condition; | 
					
						
							|  |  |  |             const auto pred_index = static_cast<u32>(instr.pred.pred_index); | 
					
						
							|  |  |  |             parse_info.branch_info.condition.predicate = | 
					
						
							|  |  |  |                 GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (parse_info.branch_info.condition.predicate == Pred::NeverExecute) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							|  |  |  |             parse_info.branch_info.condition.cc = cc; | 
					
						
							|  |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             parse_info.branch_info.address = unassigned_branch; | 
					
						
							|  |  |  |             parse_info.branch_info.kill = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_sync = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_brk = true; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return ParseResult::ControlCaught; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::KIL: { | 
					
						
							|  |  |  |             parse_info.branch_info.condition; | 
					
						
							|  |  |  |             const auto pred_index = static_cast<u32>(instr.pred.pred_index); | 
					
						
							|  |  |  |             parse_info.branch_info.condition.predicate = | 
					
						
							|  |  |  |                 GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (parse_info.branch_info.condition.predicate == Pred::NeverExecute) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							|  |  |  |             parse_info.branch_info.condition.cc = cc; | 
					
						
							|  |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             parse_info.branch_info.address = exit_branch; | 
					
						
							|  |  |  |             parse_info.branch_info.kill = true; | 
					
						
							|  |  |  |             parse_info.branch_info.is_sync = false; | 
					
						
							|  |  |  |             parse_info.branch_info.is_brk = false; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             parse_info.branch_info.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return ParseResult::ControlCaught; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::SSY: { | 
					
						
							|  |  |  |             const u32 target = offset + instr.bra.GetBranchTarget(); | 
					
						
							|  |  |  |             insert_label(state, target); | 
					
						
							|  |  |  |             state.ssy_labels.emplace(offset, target); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::PBK: { | 
					
						
							|  |  |  |             const u32 target = offset + instr.bra.GetBranchTarget(); | 
					
						
							|  |  |  |             insert_label(state, target); | 
					
						
							|  |  |  |             state.pbk_labels.emplace(offset, target); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-24 21:25:38 -04:00
										 |  |  |         case OpCode::Id::BRX: { | 
					
						
							|  |  |  |             return ParseResult::AbnormalFlow; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         default: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         offset++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     parse_info.branch_info.kill = false; | 
					
						
							|  |  |  |     parse_info.branch_info.is_sync = false; | 
					
						
							|  |  |  |     parse_info.branch_info.is_brk = false; | 
					
						
							|  |  |  |     parse_info.end_address = offset - 1; | 
					
						
							|  |  |  |     return ParseResult::BlockEnd; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool TryInspectAddress(CFGRebuildState& state) { | 
					
						
							|  |  |  |     if (state.inspect_queries.empty()) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const u32 address = state.inspect_queries.front(); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     state.inspect_queries.pop_front(); | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const auto search_result = TryGetBlock(state, address); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     switch (search_result.first) { | 
					
						
							|  |  |  |     case BlockCollision::Found: { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case BlockCollision::Inside: { | 
					
						
							|  |  |  |         // This case is the tricky one:
 | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |         // We need to Split the block in 2 sepparate blocks
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         auto it = search_result.second; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |         BlockInfo* block_info = CreateBlockInfo(state, address, it->end); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         it->end = address - 1; | 
					
						
							|  |  |  |         block_info->branch = it->branch; | 
					
						
							|  |  |  |         BlockBranchInfo forward_branch{}; | 
					
						
							|  |  |  |         forward_branch.address = address; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         forward_branch.ignore = true; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         it->branch = forward_branch; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ParseInfo parse_info; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const ParseResult parse_result = ParseCode(state, address, parse_info); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     if (parse_result == ParseResult::AbnormalFlow) { | 
					
						
							|  |  |  |         // if it's AbnormalFlow, we end it as false, ending the CFG reconstruction
 | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     BlockInfo* block_info = CreateBlockInfo(state, address, parse_info.end_address); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     block_info->branch = parse_info.branch_info; | 
					
						
							|  |  |  |     if (parse_info.branch_info.condition.IsUnconditional()) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const u32 fallthrough_address = parse_info.end_address + 1; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     state.inspect_queries.push_front(fallthrough_address); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  | bool TryQuery(CFGRebuildState& state) { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const auto gather_labels = ([](ControlStack& cc, std::map<u32, u32>& labels, BlockInfo& block) { | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |         auto gather_start = labels.lower_bound(block.start); | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |         const auto gather_end = labels.upper_bound(block.end); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         while (gather_start != gather_end) { | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |             cc.Push(gather_start->second); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             gather_start++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     if (state.queries.empty()) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Query& q = state.queries.front(); | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     const u32 block_index = state.registered[q.address]; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     BlockInfo& block = state.block_info[block_index]; | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |     // If the block is visted, check if the stacks match, else gather the ssy/pbk
 | 
					
						
							|  |  |  |     // labels into the current stack and look if the branch at the end of the block
 | 
					
						
							|  |  |  |     // consumes a label. Schedule new queries accordingly
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     if (block.visited) { | 
					
						
							|  |  |  |         BlockStack& stack = state.stacks[q.address]; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |         const bool all_okay = | 
					
						
							|  |  |  |             (stack.ssy_stack.Size() == 0 || q.ssy_stack.Compare(stack.ssy_stack)) && | 
					
						
							|  |  |  |             (stack.pbk_stack.Size() == 0 || q.pbk_stack.Compare(stack.pbk_stack)); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         state.queries.pop_front(); | 
					
						
							|  |  |  |         return all_okay; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     block.visited = true; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     state.stacks[q.address] = BlockStack{q}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     Query q2(q); | 
					
						
							|  |  |  |     state.queries.pop_front(); | 
					
						
							|  |  |  |     gather_labels(q2.ssy_stack, state.ssy_labels, block); | 
					
						
							|  |  |  |     gather_labels(q2.pbk_stack, state.pbk_labels, block); | 
					
						
							|  |  |  |     if (!block.branch.condition.IsUnconditional()) { | 
					
						
							|  |  |  |         q2.address = block.end + 1; | 
					
						
							|  |  |  |         state.queries.push_back(q2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Query conditional_query{q2}; | 
					
						
							|  |  |  |     if (block.branch.is_sync) { | 
					
						
							|  |  |  |         if (block.branch.address == unassigned_branch) { | 
					
						
							|  |  |  |             block.branch.address = conditional_query.ssy_stack.Top(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         conditional_query.ssy_stack.Pop(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (block.branch.is_brk) { | 
					
						
							|  |  |  |         if (block.branch.address == unassigned_branch) { | 
					
						
							|  |  |  |             block.branch.address = conditional_query.pbk_stack.Top(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         conditional_query.pbk_stack.Pop(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     conditional_query.address = block.branch.address; | 
					
						
							|  |  |  |     state.queries.push_back(conditional_query); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  | std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u32 program_size, | 
					
						
							|  |  |  |                                               u32 start_address) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     CFGRebuildState state{program_code, program_size}; | 
					
						
							|  |  |  |     // Inspect Code and generate blocks
 | 
					
						
							|  |  |  |     state.labels.clear(); | 
					
						
							|  |  |  |     state.labels.emplace(start_address); | 
					
						
							|  |  |  |     state.inspect_queries.push_back(start_address); | 
					
						
							|  |  |  |     while (!state.inspect_queries.empty()) { | 
					
						
							|  |  |  |         if (!TryInspectAddress(state)) { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     // Decompile Stacks
 | 
					
						
							|  |  |  |     Query start_query{}; | 
					
						
							|  |  |  |     start_query.address = start_address; | 
					
						
							|  |  |  |     state.queries.push_back(start_query); | 
					
						
							|  |  |  |     bool decompiled = true; | 
					
						
							|  |  |  |     while (!state.queries.empty()) { | 
					
						
							|  |  |  |         if (!TryQuery(state)) { | 
					
						
							|  |  |  |             decompiled = false; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // Sort and organize results
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     std::sort(state.block_info.begin(), state.block_info.end(), | 
					
						
							|  |  |  |               [](const BlockInfo& a, const BlockInfo& b) -> bool { return a.start < b.start; }); | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  |     ShaderCharacteristics result_out{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     result_out.decompilable = decompiled; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     result_out.start = start_address; | 
					
						
							|  |  |  |     result_out.end = start_address; | 
					
						
							|  |  |  |     for (auto& block : state.block_info) { | 
					
						
							|  |  |  |         ShaderBlock new_block{}; | 
					
						
							|  |  |  |         new_block.start = block.start; | 
					
						
							|  |  |  |         new_block.end = block.end; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         new_block.ignore_branch = block.branch.ignore; | 
					
						
							|  |  |  |         if (!new_block.ignore_branch) { | 
					
						
							|  |  |  |             new_block.branch.cond = block.branch.condition; | 
					
						
							|  |  |  |             new_block.branch.kills = block.branch.kill; | 
					
						
							|  |  |  |             new_block.branch.address = block.branch.address; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         result_out.end = std::max(result_out.end, block.end); | 
					
						
							|  |  |  |         result_out.blocks.push_back(new_block); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (result_out.decompilable) { | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |         result_out.labels = std::move(state.labels); | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  |         return {result_out}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |     // If it's not decompilable, merge the unlabelled blocks together
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     auto back = result_out.blocks.begin(); | 
					
						
							|  |  |  |     auto next = std::next(back); | 
					
						
							|  |  |  |     while (next != result_out.blocks.end()) { | 
					
						
							|  |  |  |         if (state.labels.count(next->start) == 0 && next->start == back->end + 1) { | 
					
						
							|  |  |  |             back->end = next->end; | 
					
						
							|  |  |  |             next = result_out.blocks.erase(next); | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         back = next; | 
					
						
							|  |  |  |         next++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  |     return {result_out}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | } | 
					
						
							|  |  |  | } // namespace VideoCommon::Shader
 |