| 
									
										
										
										
											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-27 00:39:40 -04:00
										 |  |  | #include <set>
 | 
					
						
							| 
									
										
										
										
											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 <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  | #include "video_core/shader/ast.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | #include "video_core/shader/control_flow.h"
 | 
					
						
							|  |  |  | #include "video_core/shader/shader_ir.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace VideoCommon::Shader { | 
					
						
							| 
									
										
										
										
											2019-07-16 11:32:35 -04:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 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-07-16 11:52:08 -04:00
										 |  |  |     explicit 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-09-23 22:55:25 -04:00
										 |  |  | template <typename T, typename... Args> | 
					
						
							|  |  |  | BlockBranchInfo MakeBranchInfo(Args&&... args) { | 
					
						
							|  |  |  |     static_assert(std::is_convertible_v<T, BranchData>); | 
					
						
							|  |  |  |     return std::make_shared<BranchData>(T(std::forward<Args>(args)...)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool BlockBranchInfoAreEqual(BlockBranchInfo first, BlockBranchInfo second) { | 
					
						
							|  |  |  |     return false; //(*first) == (*second);
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool BlockBranchIsIgnored(BlockBranchInfo first) { | 
					
						
							|  |  |  |     bool ignore = false; | 
					
						
							|  |  |  |     if (std::holds_alternative<SingleBranch>(*first)) { | 
					
						
							|  |  |  |         auto branch = std::get_if<SingleBranch>(first.get()); | 
					
						
							|  |  |  |         ignore = branch->ignore; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 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, | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |                              const u32 start, ConstBufferLocker& locker) | 
					
						
							|  |  |  |         : start{start}, program_code{program_code}, program_size{program_size}, locker{locker} {} | 
					
						
							| 
									
										
										
										
											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-27 00:39:40 -04:00
										 |  |  |     std::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-08-16 16:25:02 -04:00
										 |  |  |     ASTManager* manager; | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |     ConstBufferLocker& locker; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-26 12:19:43 -04:00
										 |  |  | enum class BlockCollision : u32 { None, Found, Inside }; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-27 09:24:40 -04:00
										 |  |  | std::pair<BlockCollision, u32> TryGetBlock(CFGRebuildState& state, u32 address) { | 
					
						
							|  |  |  |     const auto& blocks = state.block_info; | 
					
						
							|  |  |  |     for (u32 index = 0; index < blocks.size(); index++) { | 
					
						
							|  |  |  |         if (blocks[index].start == address) { | 
					
						
							|  |  |  |             return {BlockCollision::Found, index}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-27 09:24:40 -04:00
										 |  |  |         if (blocks[index].IsInside(address)) { | 
					
						
							|  |  |  |             return {BlockCollision::Inside, index}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:56:37 -04:00
										 |  |  |     return {BlockCollision::None, 0xFFFFFFFF}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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-09-23 11:15:09 -04:00
										 |  |  | struct BranchIndirectInfo { | 
					
						
							|  |  |  |     u32 buffer{}; | 
					
						
							|  |  |  |     u32 offset{}; | 
					
						
							|  |  |  |     u32 entries{}; | 
					
						
							|  |  |  |     s32 relative_position{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<BranchIndirectInfo> TrackBranchIndirectInfo(const CFGRebuildState& state, | 
					
						
							|  |  |  |                                                           u32 start_address, u32 current_position) { | 
					
						
							|  |  |  |     const u32 shader_start = state.start; | 
					
						
							|  |  |  |     u32 pos = current_position; | 
					
						
							|  |  |  |     BranchIndirectInfo result{}; | 
					
						
							|  |  |  |     u64 track_register = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Step 0 Get BRX Info
 | 
					
						
							|  |  |  |     const Instruction instr = {state.program_code[pos]}; | 
					
						
							|  |  |  |     const auto opcode = OpCode::Decode(instr); | 
					
						
							|  |  |  |     if (opcode->get().GetId() != OpCode::Id::BRX) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (instr.brx.constant_buffer != 0) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     track_register = instr.gpr8.Value(); | 
					
						
							|  |  |  |     result.relative_position = instr.brx.GetBranchExtend(); | 
					
						
							|  |  |  |     pos--; | 
					
						
							|  |  |  |     bool found_track = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Step 1 Track LDC
 | 
					
						
							|  |  |  |     while (pos >= shader_start) { | 
					
						
							|  |  |  |         if (IsSchedInstruction(pos, shader_start)) { | 
					
						
							|  |  |  |             pos--; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Instruction instr = {state.program_code[pos]}; | 
					
						
							|  |  |  |         const auto opcode = OpCode::Decode(instr); | 
					
						
							|  |  |  |         if (opcode->get().GetId() == OpCode::Id::LD_C) { | 
					
						
							|  |  |  |             if (instr.gpr0.Value() == track_register && | 
					
						
							|  |  |  |                 instr.ld_c.type.Value() == Tegra::Shader::UniformType::Single) { | 
					
						
							|  |  |  |                 result.buffer = instr.cbuf36.index; | 
					
						
							|  |  |  |                 result.offset = instr.cbuf36.GetOffset(); | 
					
						
							|  |  |  |                 track_register = instr.gpr8.Value(); | 
					
						
							|  |  |  |                 pos--; | 
					
						
							|  |  |  |                 found_track = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pos--; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!found_track) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     found_track = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Step 2 Track SHL
 | 
					
						
							|  |  |  |     while (pos >= shader_start) { | 
					
						
							|  |  |  |         if (IsSchedInstruction(pos, shader_start)) { | 
					
						
							|  |  |  |             pos--; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Instruction instr = {state.program_code[pos]}; | 
					
						
							|  |  |  |         const auto opcode = OpCode::Decode(instr); | 
					
						
							|  |  |  |         if (opcode->get().GetId() == OpCode::Id::SHL_IMM) { | 
					
						
							|  |  |  |             if (instr.gpr0.Value() == track_register) { | 
					
						
							|  |  |  |                 track_register = instr.gpr8.Value(); | 
					
						
							|  |  |  |                 pos--; | 
					
						
							|  |  |  |                 found_track = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pos--; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!found_track) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     found_track = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Step 3 Track IMNMX
 | 
					
						
							|  |  |  |     while (pos >= shader_start) { | 
					
						
							|  |  |  |         if (IsSchedInstruction(pos, shader_start)) { | 
					
						
							|  |  |  |             pos--; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Instruction instr = {state.program_code[pos]}; | 
					
						
							|  |  |  |         const auto opcode = OpCode::Decode(instr); | 
					
						
							|  |  |  |         if (opcode->get().GetId() == OpCode::Id::IMNMX_IMM) { | 
					
						
							|  |  |  |             if (instr.gpr0.Value() == track_register) { | 
					
						
							|  |  |  |                 track_register = instr.gpr8.Value(); | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |                 result.entries = instr.alu.GetSignedImm20_20() + 1; | 
					
						
							| 
									
										
										
										
											2019-09-23 11:15:09 -04:00
										 |  |  |                 pos--; | 
					
						
							|  |  |  |                 found_track = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pos--; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!found_track) { | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return {result}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-09-23 22:55:25 -04:00
										 |  |  |     SingleBranch single_branch{}; | 
					
						
							| 
									
										
										
										
											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-09-23 22:55:25 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |             single_branch.address = exit_branch; | 
					
						
							|  |  |  |             single_branch.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-09-23 22:55:25 -04:00
										 |  |  |             single_branch.address = offset; | 
					
						
							|  |  |  |             single_branch.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); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.predicate = GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (single_branch.condition.predicate == Pred::NeverExecute) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.cc = cc; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.address = exit_branch; | 
					
						
							|  |  |  |             single_branch.kill = false; | 
					
						
							|  |  |  |             single_branch.is_sync = false; | 
					
						
							|  |  |  |             single_branch.is_brk = false; | 
					
						
							|  |  |  |             single_branch.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             parse_info.branch_info = MakeBranchInfo<SingleBranch>( | 
					
						
							|  |  |  |                 single_branch.condition, single_branch.address, single_branch.kill, | 
					
						
							|  |  |  |                 single_branch.is_sync, single_branch.is_brk, single_branch.ignore); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.predicate = GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (single_branch.condition.predicate == Pred::NeverExecute) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.cc = cc; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             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) { | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |                 single_branch.address = exit_branch; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |                 single_branch.address = branch_offset; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             insert_label(state, branch_offset); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.kill = false; | 
					
						
							|  |  |  |             single_branch.is_sync = false; | 
					
						
							|  |  |  |             single_branch.is_brk = false; | 
					
						
							|  |  |  |             single_branch.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             parse_info.branch_info = MakeBranchInfo<SingleBranch>( | 
					
						
							|  |  |  |                 single_branch.condition, single_branch.address, single_branch.kill, | 
					
						
							|  |  |  |                 single_branch.is_sync, single_branch.is_brk, single_branch.ignore); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.predicate = GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (single_branch.condition.predicate == Pred::NeverExecute) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.cc = cc; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.address = unassigned_branch; | 
					
						
							|  |  |  |             single_branch.kill = false; | 
					
						
							|  |  |  |             single_branch.is_sync = true; | 
					
						
							|  |  |  |             single_branch.is_brk = false; | 
					
						
							|  |  |  |             single_branch.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             parse_info.branch_info = MakeBranchInfo<SingleBranch>( | 
					
						
							|  |  |  |                 single_branch.condition, single_branch.address, single_branch.kill, | 
					
						
							|  |  |  |                 single_branch.is_sync, single_branch.is_brk, single_branch.ignore); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.predicate = GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (single_branch.condition.predicate == Pred::NeverExecute) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.cc = cc; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.address = unassigned_branch; | 
					
						
							|  |  |  |             single_branch.kill = false; | 
					
						
							|  |  |  |             single_branch.is_sync = false; | 
					
						
							|  |  |  |             single_branch.is_brk = true; | 
					
						
							|  |  |  |             single_branch.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             parse_info.branch_info = MakeBranchInfo<SingleBranch>( | 
					
						
							|  |  |  |                 single_branch.condition, single_branch.address, single_branch.kill, | 
					
						
							|  |  |  |                 single_branch.is_sync, single_branch.is_brk, single_branch.ignore); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.predicate = GetPredicate(pred_index, instr.negate_pred != 0); | 
					
						
							|  |  |  |             if (single_branch.condition.predicate == Pred::NeverExecute) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const ConditionCode cc = instr.flow_condition_code; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.condition.cc = cc; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             if (cc == ConditionCode::F) { | 
					
						
							|  |  |  |                 offset++; | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             single_branch.address = exit_branch; | 
					
						
							|  |  |  |             single_branch.kill = true; | 
					
						
							|  |  |  |             single_branch.is_sync = false; | 
					
						
							|  |  |  |             single_branch.is_brk = false; | 
					
						
							|  |  |  |             single_branch.ignore = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             parse_info.end_address = offset; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             parse_info.branch_info = MakeBranchInfo<SingleBranch>( | 
					
						
							|  |  |  |                 single_branch.condition, single_branch.address, single_branch.kill, | 
					
						
							|  |  |  |                 single_branch.is_sync, single_branch.is_brk, single_branch.ignore); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-09-23 11:15:09 -04:00
										 |  |  |             auto tmp = TrackBranchIndirectInfo(state, address, offset); | 
					
						
							|  |  |  |             if (tmp) { | 
					
						
							|  |  |  |                 auto result = *tmp; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |                 std::vector<CaseBranch> branches{}; | 
					
						
							|  |  |  |                 s32 pc_target = offset + result.relative_position; | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |                 for (u32 i = 0; i < result.entries; i++) { | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |                     auto k = state.locker.ObtainKey(result.buffer, result.offset + i * 4); | 
					
						
							|  |  |  |                     if (!k) { | 
					
						
							|  |  |  |                         return {ParseResult::AbnormalFlow, parse_info}; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     u32 value = *k; | 
					
						
							|  |  |  |                     u32 target = static_cast<u32>((value >> 3) + pc_target); | 
					
						
							|  |  |  |                     insert_label(state, target); | 
					
						
							|  |  |  |                     branches.emplace_back(value, target); | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |                 parse_info.end_address = offset; | 
					
						
							|  |  |  |                 parse_info.branch_info = | 
					
						
							|  |  |  |                     MakeBranchInfo<MultiBranch>(static_cast<u32>(instr.gpr8.Value()), branches); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return {ParseResult::ControlCaught, parse_info}; | 
					
						
							| 
									
										
										
										
											2019-09-23 11:15:09 -04:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |                 LOG_WARNING(HW_GPU, "BRX Track Unsuccesful"); | 
					
						
							| 
									
										
										
										
											2019-09-23 11:15:09 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											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++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     single_branch.kill = false; | 
					
						
							|  |  |  |     single_branch.is_sync = false; | 
					
						
							|  |  |  |     single_branch.is_brk = false; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     parse_info.end_address = offset - 1; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     parse_info.branch_info = MakeBranchInfo<SingleBranch>( | 
					
						
							|  |  |  |         single_branch.condition, single_branch.address, single_branch.kill, single_branch.is_sync, | 
					
						
							|  |  |  |         single_branch.is_brk, single_branch.ignore); | 
					
						
							| 
									
										
										
										
											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-27 09:24:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-27 09:24:40 -04:00
										 |  |  |     const auto [result, block_index] = TryGetBlock(state, address); | 
					
						
							|  |  |  |     switch (result) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     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-27 09:24:40 -04:00
										 |  |  |         const u32 end = state.block_info[block_index].end; | 
					
						
							|  |  |  |         BlockInfo& new_block = CreateBlockInfo(state, address, end); | 
					
						
							|  |  |  |         BlockInfo& current_block = state.block_info[block_index]; | 
					
						
							|  |  |  |         current_block.end = address - 1; | 
					
						
							|  |  |  |         new_block.branch = current_block.branch; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         BlockBranchInfo forward_branch = MakeBranchInfo<SingleBranch>(); | 
					
						
							|  |  |  |         auto branch = std::get_if<SingleBranch>(forward_branch.get()); | 
					
						
							|  |  |  |         branch->address = address; | 
					
						
							|  |  |  |         branch->ignore = true; | 
					
						
							| 
									
										
										
										
											2019-06-27 09:24:40 -04:00
										 |  |  |         current_block.branch = forward_branch; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         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-09-23 22:55:25 -04:00
										 |  |  |     if (std::holds_alternative<SingleBranch>(*block_info.branch)) { | 
					
						
							|  |  |  |         auto branch = std::get_if<SingleBranch>(block_info.branch.get()); | 
					
						
							|  |  |  |         if (branch->condition.IsUnconditional()) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const u32 fallthrough_address = parse_info.end_address + 1; | 
					
						
							|  |  |  |         state.inspect_queries.push_front(fallthrough_address); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     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-07-16 11:42:05 -04:00
										 |  |  |             ++gather_start; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-26 12:56:03 -04:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     if (state.queries.empty()) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:50:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     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-07-16 11:50:36 -04:00
										 |  |  |     // If the block is visited, check if the stacks match, else gather the ssy/pbk
 | 
					
						
							| 
									
										
										
										
											2019-06-25 13:03:51 -04:00
										 |  |  |     // 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-07-16 11:40:58 -04:00
										 |  |  |         const bool all_okay = (stack.ssy_stack.empty() || q.ssy_stack == stack.ssy_stack) && | 
					
						
							|  |  |  |                               (stack.pbk_stack.empty() || 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-07-16 11:50:36 -04:00
										 |  |  |     state.stacks.insert_or_assign(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); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     if (std::holds_alternative<SingleBranch>(*block.branch)) { | 
					
						
							|  |  |  |         auto branch = std::get_if<SingleBranch>(block.branch.get()); | 
					
						
							|  |  |  |         if (!branch->condition.IsUnconditional()) { | 
					
						
							|  |  |  |             q2.address = block.end + 1; | 
					
						
							|  |  |  |             state.queries.push_back(q2); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:50:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         Query conditional_query{q2}; | 
					
						
							|  |  |  |         if (branch->is_sync) { | 
					
						
							|  |  |  |             if (branch->address == unassigned_branch) { | 
					
						
							|  |  |  |                 branch->address = conditional_query.ssy_stack.top(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             conditional_query.ssy_stack.pop(); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         if (branch->is_brk) { | 
					
						
							|  |  |  |             if (branch->address == unassigned_branch) { | 
					
						
							|  |  |  |                 branch->address = conditional_query.pbk_stack.top(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             conditional_query.pbk_stack.pop(); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         conditional_query.address = branch->address; | 
					
						
							|  |  |  |         state.queries.push_back(std::move(conditional_query)); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     auto multi_branch = std::get_if<MultiBranch>(block.branch.get()); | 
					
						
							|  |  |  |     for (auto& branch_case : multi_branch->branches) { | 
					
						
							|  |  |  |         Query conditional_query{q2}; | 
					
						
							|  |  |  |         conditional_query.address = branch_case.address; | 
					
						
							|  |  |  |         state.queries.push_back(std::move(conditional_query)); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:32:35 -04:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  | void InsertBranch(ASTManager& mm, const BlockBranchInfo& branch_info) { | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |     const auto get_expr = ([&](const Condition& cond) -> Expr { | 
					
						
							|  |  |  |         Expr result{}; | 
					
						
							|  |  |  |         if (cond.cc != ConditionCode::T) { | 
					
						
							|  |  |  |             result = MakeExpr<ExprCondCode>(cond.cc); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (cond.predicate != Pred::UnusedIndex) { | 
					
						
							| 
									
										
										
										
											2019-06-28 20:54:21 -04:00
										 |  |  |             u32 pred = static_cast<u32>(cond.predicate); | 
					
						
							| 
									
										
										
										
											2019-06-29 01:44:07 -04:00
										 |  |  |             bool negate = false; | 
					
						
							| 
									
										
										
										
											2019-06-28 20:54:21 -04:00
										 |  |  |             if (pred > 7) { | 
					
						
							|  |  |  |                 negate = true; | 
					
						
							|  |  |  |                 pred -= 8; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Expr extra = MakeExpr<ExprPredicate>(pred); | 
					
						
							|  |  |  |             if (negate) { | 
					
						
							|  |  |  |                 extra = MakeExpr<ExprNot>(extra); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |             if (result) { | 
					
						
							|  |  |  |                 return MakeExpr<ExprAnd>(extra, result); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return extra; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (result) { | 
					
						
							|  |  |  |             return result; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return MakeExpr<ExprBoolean>(true); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     if (std::holds_alternative<SingleBranch>(*branch_info)) { | 
					
						
							|  |  |  |         auto branch = std::get_if<SingleBranch>(branch_info.get()); | 
					
						
							|  |  |  |         if (branch->address < 0) { | 
					
						
							|  |  |  |             if (branch->kill) { | 
					
						
							|  |  |  |                 mm.InsertReturn(get_expr(branch->condition), true); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             mm.InsertReturn(get_expr(branch->condition), false); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         mm.InsertGoto(get_expr(branch->condition), branch->address); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     auto multi_branch = std::get_if<MultiBranch>(branch_info.get()); | 
					
						
							|  |  |  |     for (auto& branch_case : multi_branch->branches) { | 
					
						
							|  |  |  |         mm.InsertGoto(MakeExpr<ExprGprEqual>(multi_branch->gpr, branch_case.cmp_value), | 
					
						
							|  |  |  |                       branch_case.address); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DecompileShader(CFGRebuildState& state) { | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     state.manager->Init(); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |     for (auto label : state.labels) { | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |         state.manager->DeclareLabel(label); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     for (auto& block : state.block_info) { | 
					
						
							|  |  |  |         if (state.labels.count(block.start) != 0) { | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |             state.manager->InsertLabel(block.start); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         const bool ignore = BlockBranchIsIgnored(block.branch); | 
					
						
							|  |  |  |         u32 end = ignore ? block.end + 1 : block.end; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |         state.manager->InsertBlock(block.start, end); | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         if (!ignore) { | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |             InsertBranch(*state.manager, block.branch); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     state.manager->Decompile(); | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-15 18:05:50 -04:00
										 |  |  | std::unique_ptr<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, | 
					
						
							|  |  |  |                                                 std::size_t program_size, u32 start_address, | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |                                                 const CompilerSettings& settings, | 
					
						
							|  |  |  |                                                 ConstBufferLocker& locker) { | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     auto result_out = std::make_unique<ShaderCharacteristics>(); | 
					
						
							|  |  |  |     if (settings.depth == CompileDepth::BruteForce) { | 
					
						
							|  |  |  |         result_out->settings.depth = CompileDepth::BruteForce; | 
					
						
							| 
									
										
										
										
											2019-10-05 08:52:20 -04:00
										 |  |  |         return result_out; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |     CFGRebuildState state{program_code, program_size, start_address, locker}; | 
					
						
							| 
									
										
										
										
											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-08-16 16:25:02 -04:00
										 |  |  |             result_out->settings.depth = CompileDepth::BruteForce; | 
					
						
							| 
									
										
										
										
											2019-10-05 08:52:20 -04:00
										 |  |  |             return result_out; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:50:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     bool use_flow_stack = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool decompiled = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (settings.depth != CompileDepth::FlowStack) { | 
					
						
							|  |  |  |         // Decompile Stacks
 | 
					
						
							|  |  |  |         state.queries.push_back(Query{state.start, {}, {}}); | 
					
						
							|  |  |  |         decompiled = true; | 
					
						
							|  |  |  |         while (!state.queries.empty()) { | 
					
						
							|  |  |  |             if (!TryQuery(state)) { | 
					
						
							|  |  |  |                 decompiled = false; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:50:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     use_flow_stack = !decompiled; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |     // Sort and organize results
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     std::sort(state.block_info.begin(), state.block_info.end(), | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |               [](const BlockInfo& a, const BlockInfo& b) -> bool { return a.start < b.start; }); | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     if (decompiled && settings.depth != CompileDepth::NoFlowStack) { | 
					
						
							| 
									
										
										
										
											2019-09-20 21:12:06 -04:00
										 |  |  |         ASTManager manager{settings.depth != CompileDepth::DecompileBackwards, | 
					
						
							|  |  |  |                            settings.disable_else_derivation}; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |         state.manager = &manager; | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  |         DecompileShader(state); | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |         decompiled = state.manager->IsFullyDecompiled(); | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |         if (!decompiled) { | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |             if (settings.depth == CompileDepth::FullDecompile) { | 
					
						
							|  |  |  |                 LOG_CRITICAL(HW_GPU, "Failed to remove all the gotos!:"); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 LOG_CRITICAL(HW_GPU, "Failed to remove all backward gotos!:"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             state.manager->ShowCurrentState("Of Shader"); | 
					
						
							|  |  |  |             state.manager->Clear(); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2019-10-05 08:55:08 -04:00
										 |  |  |             auto characteristics = std::make_unique<ShaderCharacteristics>(); | 
					
						
							|  |  |  |             characteristics->start = start_address; | 
					
						
							|  |  |  |             characteristics->settings.depth = settings.depth; | 
					
						
							|  |  |  |             characteristics->manager = std::move(manager); | 
					
						
							|  |  |  |             characteristics->end = state.block_info.back().end + 1; | 
					
						
							|  |  |  |             return characteristics; | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-05 08:52:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |     result_out->start = start_address; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     result_out->settings.depth = | 
					
						
							|  |  |  |         use_flow_stack ? CompileDepth::FlowStack : CompileDepth::NoFlowStack; | 
					
						
							|  |  |  |     result_out->blocks.clear(); | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |     for (auto& block : state.block_info) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         ShaderBlock new_block{}; | 
					
						
							|  |  |  |         new_block.start = block.start; | 
					
						
							|  |  |  |         new_block.end = block.end; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         new_block.ignore_branch = BlockBranchIsIgnored(block.branch); | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         if (!new_block.ignore_branch) { | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |             new_block.branch = block.branch; | 
					
						
							| 
									
										
										
										
											2019-06-25 11:10:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |         result_out->end = std::max(result_out->end, block.end); | 
					
						
							|  |  |  |         result_out->blocks.push_back(new_block); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     if (!use_flow_stack) { | 
					
						
							|  |  |  |         result_out->labels = std::move(state.labels); | 
					
						
							| 
									
										
										
										
											2019-10-05 08:52:20 -04:00
										 |  |  |         return result_out; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-05 08:52:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |     auto back = result_out->blocks.begin(); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     auto next = std::next(back); | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |     while (next != result_out->blocks.end()) { | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |         if (state.labels.count(next->start) == 0 && next->start == back->end + 1) { | 
					
						
							|  |  |  |             back->end = next->end; | 
					
						
							| 
									
										
										
										
											2019-06-28 22:59:43 -04:00
										 |  |  |             next = result_out->blocks.erase(next); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         back = next; | 
					
						
							| 
									
										
										
										
											2019-07-16 11:42:05 -04:00
										 |  |  |         ++next; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-05 08:52:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return result_out; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | } | 
					
						
							|  |  |  | } // namespace VideoCommon::Shader
 |