| 
									
										
										
										
											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-26 12:56:03 -04:00
										 |  |  | #include <stack>
 | 
					
						
							| 
									
										
										
										
											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 11:10:45 -04:00
										 |  |  | struct Query { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     u32 address{}; | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     std::stack<u32> ssy_stack{}; | 
					
						
							|  |  |  |     std::stack<u32> pbk_stack{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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} {} | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     std::stack<u32> ssy_stack{}; | 
					
						
							|  |  |  |     std::stack<u32> pbk_stack{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |     explicit CFGRebuildState(const ProgramCode& program_code, const std::size_t program_size, | 
					
						
							|  |  |  |                              const u32 start) | 
					
						
							|  |  |  |         : program_code{program_code}, program_size{program_size}, start{start} {} | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |     u32 start{}; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  | enum class BlockCollision : u32 { None, Found, Inside }; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  | BlockInfo& CreateBlockInfo(CFGRebuildState& state, u32 start, u32 end) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     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-26 12:56:03 -04:00
										 |  |  |     return it; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Pred GetPredicate(u32 index, bool negated) { | 
					
						
							|  |  |  |     return static_cast<Pred>(index + (negated ? 8 : 0)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Returns whether the instruction at the specified offset is a 'sched' instruction. | 
					
						
							|  |  |  |  * Sched instructions always appear before a sequence of 3 instructions. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | constexpr bool IsSchedInstruction(u32 offset, u32 main_offset) { | 
					
						
							|  |  |  |     constexpr u32 SchedPeriod = 4; | 
					
						
							|  |  |  |     u32 absolute_offset = offset - main_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return (absolute_offset % SchedPeriod) == 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | enum class ParseResult : u32 { | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |     ControlCaught, | 
					
						
							|  |  |  |     BlockEnd, | 
					
						
							|  |  |  |     AbnormalFlow, | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  | std::pair<ParseResult, ParseInfo> ParseCode(CFGRebuildState& state, u32 address) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     u32 offset = static_cast<u32>(address); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |     const u32 end_address = static_cast<u32>(state.program_size / sizeof(Instruction)); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     ParseInfo parse_info{}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     const auto insert_label = [](CFGRebuildState& state, u32 address) { | 
					
						
							|  |  |  |         const auto pair = state.labels.emplace(address); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         if (pair.second) { | 
					
						
							|  |  |  |             state.inspect_queries.push_back(address); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     while (true) { | 
					
						
							|  |  |  |         if (offset >= end_address) { | 
					
						
							| 
									
										
										
										
											2019-06-26 13:16:13 -04:00
										 |  |  |             // ASSERT_OR_EXECUTE can't be used, as it ignores the break
 | 
					
						
							|  |  |  |             ASSERT_MSG(false, "Shader passed the current limit!"); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             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; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |         if (IsSchedInstruction(offset, state.start)) { | 
					
						
							|  |  |  |             offset++; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             return {ParseResult::ControlCaught, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::BRA: { | 
					
						
							|  |  |  |             if (instr.bra.constant_buffer != 0) { | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |                 return {ParseResult::AbnormalFlow, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             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; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             const u32 branch_offset = offset + instr.bra.GetBranchTarget(); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             return {ParseResult::ControlCaught, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::SYNC: { | 
					
						
							|  |  |  |             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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             return {ParseResult::ControlCaught, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::BRK: { | 
					
						
							|  |  |  |             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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             return {ParseResult::ControlCaught, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         case OpCode::Id::KIL: { | 
					
						
							|  |  |  |             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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             return {ParseResult::ControlCaught, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         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: { | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             return {ParseResult::AbnormalFlow, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 21:25:38 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     return {ParseResult::BlockEnd, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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-26 12:56:03 -04:00
										 |  |  |         const auto it = search_result.second; | 
					
						
							|  |  |  |         BlockInfo& block_info = CreateBlockInfo(state, address, it->end); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         it->end = address - 1; | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |         block_info.branch = it->branch; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         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; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     const auto [parse_result, parse_info] = ParseCode(state, address); | 
					
						
							| 
									
										
										
										
											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-26 12:56:03 -04:00
										 |  |  |     BlockInfo& block_info = CreateBlockInfo(state, address, parse_info.end_address); | 
					
						
							|  |  |  |     block_info.branch = parse_info.branch_info; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     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-26 12:56:03 -04:00
										 |  |  |     const auto gather_labels = [](std::stack<u32>& 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-26 12:56:03 -04:00
										 |  |  |             cc.push(gather_start->second); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |             gather_start++; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     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-26 12:56:03 -04:00
										 |  |  |         const bool all_okay = (stack.ssy_stack.size() == 0 || q.ssy_stack == stack.ssy_stack) && | 
					
						
							|  |  |  |                               (stack.pbk_stack.size() == 0 || q.pbk_stack == 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) { | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             block.branch.address = conditional_query.ssy_stack.top(); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |         conditional_query.ssy_stack.pop(); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (block.branch.is_brk) { | 
					
						
							|  |  |  |         if (block.branch.address == unassigned_branch) { | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |             block.branch.address = conditional_query.pbk_stack.top(); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |         conditional_query.pbk_stack.pop(); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     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-26 12:19:43 -04:00
										 |  |  |     CFGRebuildState state{program_code, program_size, start_address}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     // Inspect Code and generate blocks
 | 
					
						
							|  |  |  |     state.labels.clear(); | 
					
						
							|  |  |  |     state.labels.emplace(start_address); | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |     state.inspect_queries.push_back(state.start); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     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{}; | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  |     start_query.address = state.start; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     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
 |