forked from eden-emu/eden
		
	Merge pull request #2739 from lioncash/cflow
video_core/control_flow: Minor changes/warning cleanup
This commit is contained in:
		
						commit
						b0ff3179ef
					
				
					 4 changed files with 53 additions and 33 deletions
				
			
		|  | @ -15,7 +15,7 @@ | ||||||
| #include "video_core/shader/shader_ir.h" | #include "video_core/shader/shader_ir.h" | ||||||
| 
 | 
 | ||||||
| namespace VideoCommon::Shader { | namespace VideoCommon::Shader { | ||||||
| 
 | namespace { | ||||||
| using Tegra::Shader::Instruction; | using Tegra::Shader::Instruction; | ||||||
| using Tegra::Shader::OpCode; | using Tegra::Shader::OpCode; | ||||||
| 
 | 
 | ||||||
|  | @ -29,8 +29,7 @@ struct Query { | ||||||
| 
 | 
 | ||||||
| struct BlockStack { | struct BlockStack { | ||||||
|     BlockStack() = default; |     BlockStack() = default; | ||||||
|     BlockStack(const BlockStack& b) = default; |     explicit BlockStack(const Query& q) : ssy_stack{q.ssy_stack}, pbk_stack{q.pbk_stack} {} | ||||||
|     BlockStack(const Query& q) : ssy_stack{q.ssy_stack}, pbk_stack{q.pbk_stack} {} |  | ||||||
|     std::stack<u32> ssy_stack{}; |     std::stack<u32> ssy_stack{}; | ||||||
|     std::stack<u32> pbk_stack{}; |     std::stack<u32> pbk_stack{}; | ||||||
| }; | }; | ||||||
|  | @ -58,7 +57,7 @@ struct BlockInfo { | ||||||
| struct CFGRebuildState { | struct CFGRebuildState { | ||||||
|     explicit CFGRebuildState(const ProgramCode& program_code, const std::size_t program_size, |     explicit CFGRebuildState(const ProgramCode& program_code, const std::size_t program_size, | ||||||
|                              const u32 start) |                              const u32 start) | ||||||
|         : program_code{program_code}, program_size{program_size}, start{start} {} |         : start{start}, program_code{program_code}, program_size{program_size} {} | ||||||
| 
 | 
 | ||||||
|     u32 start{}; |     u32 start{}; | ||||||
|     std::vector<BlockInfo> block_info{}; |     std::vector<BlockInfo> block_info{}; | ||||||
|  | @ -85,7 +84,7 @@ std::pair<BlockCollision, u32> TryGetBlock(CFGRebuildState& state, u32 address) | ||||||
|             return {BlockCollision::Inside, index}; |             return {BlockCollision::Inside, index}; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     return {BlockCollision::None, -1}; |     return {BlockCollision::None, 0xFFFFFFFF}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct ParseInfo { | struct ParseInfo { | ||||||
|  | @ -365,27 +364,29 @@ bool TryQuery(CFGRebuildState& state) { | ||||||
|         const auto gather_end = labels.upper_bound(block.end); |         const auto gather_end = labels.upper_bound(block.end); | ||||||
|         while (gather_start != gather_end) { |         while (gather_start != gather_end) { | ||||||
|             cc.push(gather_start->second); |             cc.push(gather_start->second); | ||||||
|             gather_start++; |             ++gather_start; | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|     if (state.queries.empty()) { |     if (state.queries.empty()) { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     Query& q = state.queries.front(); |     Query& q = state.queries.front(); | ||||||
|     const u32 block_index = state.registered[q.address]; |     const u32 block_index = state.registered[q.address]; | ||||||
|     BlockInfo& block = state.block_info[block_index]; |     BlockInfo& block = state.block_info[block_index]; | ||||||
|     // If the block is visted, check if the stacks match, else gather the ssy/pbk
 |     // If the block is visited, 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
 |     // labels into the current stack and look if the branch at the end of the block
 | ||||||
|     // consumes a label. Schedule new queries accordingly
 |     // consumes a label. Schedule new queries accordingly
 | ||||||
|     if (block.visited) { |     if (block.visited) { | ||||||
|         BlockStack& stack = state.stacks[q.address]; |         BlockStack& stack = state.stacks[q.address]; | ||||||
|         const bool all_okay = (stack.ssy_stack.size() == 0 || q.ssy_stack == stack.ssy_stack) && |         const bool all_okay = (stack.ssy_stack.empty() || q.ssy_stack == stack.ssy_stack) && | ||||||
|                               (stack.pbk_stack.size() == 0 || q.pbk_stack == stack.pbk_stack); |                               (stack.pbk_stack.empty() || q.pbk_stack == stack.pbk_stack); | ||||||
|         state.queries.pop_front(); |         state.queries.pop_front(); | ||||||
|         return all_okay; |         return all_okay; | ||||||
|     } |     } | ||||||
|     block.visited = true; |     block.visited = true; | ||||||
|     state.stacks[q.address] = BlockStack{q}; |     state.stacks.insert_or_assign(q.address, BlockStack{q}); | ||||||
|  | 
 | ||||||
|     Query q2(q); |     Query q2(q); | ||||||
|     state.queries.pop_front(); |     state.queries.pop_front(); | ||||||
|     gather_labels(q2.ssy_stack, state.ssy_labels, block); |     gather_labels(q2.ssy_stack, state.ssy_labels, block); | ||||||
|  | @ -394,6 +395,7 @@ bool TryQuery(CFGRebuildState& state) { | ||||||
|         q2.address = block.end + 1; |         q2.address = block.end + 1; | ||||||
|         state.queries.push_back(q2); |         state.queries.push_back(q2); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     Query conditional_query{q2}; |     Query conditional_query{q2}; | ||||||
|     if (block.branch.is_sync) { |     if (block.branch.is_sync) { | ||||||
|         if (block.branch.address == unassigned_branch) { |         if (block.branch.address == unassigned_branch) { | ||||||
|  | @ -408,13 +410,15 @@ bool TryQuery(CFGRebuildState& state) { | ||||||
|         conditional_query.pbk_stack.pop(); |         conditional_query.pbk_stack.pop(); | ||||||
|     } |     } | ||||||
|     conditional_query.address = block.branch.address; |     conditional_query.address = block.branch.address; | ||||||
|     state.queries.push_back(conditional_query); |     state.queries.push_back(std::move(conditional_query)); | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  | } // Anonymous namespace
 | ||||||
| 
 | 
 | ||||||
| std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u32 program_size, | std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, | ||||||
|                                               u32 start_address) { |                                               std::size_t program_size, u32 start_address) { | ||||||
|     CFGRebuildState state{program_code, program_size, start_address}; |     CFGRebuildState state{program_code, program_size, start_address}; | ||||||
|  | 
 | ||||||
|     // Inspect Code and generate blocks
 |     // Inspect Code and generate blocks
 | ||||||
|     state.labels.clear(); |     state.labels.clear(); | ||||||
|     state.labels.emplace(start_address); |     state.labels.emplace(start_address); | ||||||
|  | @ -424,10 +428,9 @@ std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u | ||||||
|             return {}; |             return {}; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     // Decompile Stacks
 |     // Decompile Stacks
 | ||||||
|     Query start_query{}; |     state.queries.push_back(Query{state.start, {}, {}}); | ||||||
|     start_query.address = state.start; |  | ||||||
|     state.queries.push_back(start_query); |  | ||||||
|     bool decompiled = true; |     bool decompiled = true; | ||||||
|     while (!state.queries.empty()) { |     while (!state.queries.empty()) { | ||||||
|         if (!TryQuery(state)) { |         if (!TryQuery(state)) { | ||||||
|  | @ -435,14 +438,15 @@ std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     // Sort and organize results
 |     // Sort and organize results
 | ||||||
|     std::sort(state.block_info.begin(), state.block_info.end(), |     std::sort(state.block_info.begin(), state.block_info.end(), | ||||||
|               [](const BlockInfo& a, const BlockInfo& b) -> bool { return a.start < b.start; }); |               [](const BlockInfo& a, const BlockInfo& b) { return a.start < b.start; }); | ||||||
|     ShaderCharacteristics result_out{}; |     ShaderCharacteristics result_out{}; | ||||||
|     result_out.decompilable = decompiled; |     result_out.decompilable = decompiled; | ||||||
|     result_out.start = start_address; |     result_out.start = start_address; | ||||||
|     result_out.end = start_address; |     result_out.end = start_address; | ||||||
|     for (auto& block : state.block_info) { |     for (const auto& block : state.block_info) { | ||||||
|         ShaderBlock new_block{}; |         ShaderBlock new_block{}; | ||||||
|         new_block.start = block.start; |         new_block.start = block.start; | ||||||
|         new_block.end = block.end; |         new_block.end = block.end; | ||||||
|  | @ -457,8 +461,9 @@ std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u | ||||||
|     } |     } | ||||||
|     if (result_out.decompilable) { |     if (result_out.decompilable) { | ||||||
|         result_out.labels = std::move(state.labels); |         result_out.labels = std::move(state.labels); | ||||||
|         return {result_out}; |         return {std::move(result_out)}; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     // If it's not decompilable, merge the unlabelled blocks together
 |     // If it's not decompilable, merge the unlabelled blocks together
 | ||||||
|     auto back = result_out.blocks.begin(); |     auto back = result_out.blocks.begin(); | ||||||
|     auto next = std::next(back); |     auto next = std::next(back); | ||||||
|  | @ -469,8 +474,8 @@ std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u | ||||||
|             continue; |             continue; | ||||||
|         } |         } | ||||||
|         back = next; |         back = next; | ||||||
|         next++; |         ++next; | ||||||
|     } |     } | ||||||
|     return {result_out}; |     return {std::move(result_out)}; | ||||||
| } | } | ||||||
| } // namespace VideoCommon::Shader
 | } // namespace VideoCommon::Shader
 | ||||||
|  |  | ||||||
|  | @ -4,7 +4,6 @@ | ||||||
| 
 | 
 | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include <cstring> |  | ||||||
| #include <list> | #include <list> | ||||||
| #include <optional> | #include <optional> | ||||||
| #include <unordered_set> | #include <unordered_set> | ||||||
|  | @ -26,27 +25,44 @@ struct Condition { | ||||||
|     bool IsUnconditional() const { |     bool IsUnconditional() const { | ||||||
|         return predicate == Pred::UnusedIndex && cc == ConditionCode::T; |         return predicate == Pred::UnusedIndex && cc == ConditionCode::T; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     bool operator==(const Condition& other) const { |     bool operator==(const Condition& other) const { | ||||||
|         return std::tie(predicate, cc) == std::tie(other.predicate, other.cc); |         return std::tie(predicate, cc) == std::tie(other.predicate, other.cc); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     bool operator!=(const Condition& other) const { | ||||||
|  |         return !operator==(other); | ||||||
|  |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct ShaderBlock { | struct ShaderBlock { | ||||||
|     u32 start{}; |  | ||||||
|     u32 end{}; |  | ||||||
|     bool ignore_branch{}; |  | ||||||
|     struct Branch { |     struct Branch { | ||||||
|         Condition cond{}; |         Condition cond{}; | ||||||
|         bool kills{}; |         bool kills{}; | ||||||
|         s32 address{}; |         s32 address{}; | ||||||
|  | 
 | ||||||
|         bool operator==(const Branch& b) const { |         bool operator==(const Branch& b) const { | ||||||
|             return std::tie(cond, kills, address) == std::tie(b.cond, b.kills, b.address); |             return std::tie(cond, kills, address) == std::tie(b.cond, b.kills, b.address); | ||||||
|         } |         } | ||||||
|     } branch{}; | 
 | ||||||
|  |         bool operator!=(const Branch& b) const { | ||||||
|  |             return !operator==(b); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     u32 start{}; | ||||||
|  |     u32 end{}; | ||||||
|  |     bool ignore_branch{}; | ||||||
|  |     Branch branch{}; | ||||||
|  | 
 | ||||||
|     bool operator==(const ShaderBlock& sb) const { |     bool operator==(const ShaderBlock& sb) const { | ||||||
|         return std::tie(start, end, ignore_branch, branch) == |         return std::tie(start, end, ignore_branch, branch) == | ||||||
|                std::tie(sb.start, sb.end, sb.ignore_branch, sb.branch); |                std::tie(sb.start, sb.end, sb.ignore_branch, sb.branch); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     bool operator!=(const ShaderBlock& sb) const { | ||||||
|  |         return !operator==(sb); | ||||||
|  |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct ShaderCharacteristics { | struct ShaderCharacteristics { | ||||||
|  | @ -57,7 +73,7 @@ struct ShaderCharacteristics { | ||||||
|     std::unordered_set<u32> labels{}; |     std::unordered_set<u32> labels{}; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, u32 program_size, | std::optional<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, | ||||||
|                                               u32 start_address); |                                               std::size_t program_size, u32 start_address); | ||||||
| 
 | 
 | ||||||
| } // namespace VideoCommon::Shader
 | } // namespace VideoCommon::Shader
 | ||||||
|  |  | ||||||
|  | @ -47,14 +47,14 @@ void ShaderIR::Decode() { | ||||||
|         if (shader_info.decompilable) { |         if (shader_info.decompilable) { | ||||||
|             disable_flow_stack = true; |             disable_flow_stack = true; | ||||||
|             const auto insert_block = [this](NodeBlock& nodes, u32 label) { |             const auto insert_block = [this](NodeBlock& nodes, u32 label) { | ||||||
|                 if (label == exit_branch) { |                 if (label == static_cast<u32>(exit_branch)) { | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 basic_blocks.insert({label, nodes}); |                 basic_blocks.insert({label, nodes}); | ||||||
|             }; |             }; | ||||||
|             const auto& blocks = shader_info.blocks; |             const auto& blocks = shader_info.blocks; | ||||||
|             NodeBlock current_block; |             NodeBlock current_block; | ||||||
|             u32 current_label = exit_branch; |             u32 current_label = static_cast<u32>(exit_branch); | ||||||
|             for (auto& block : blocks) { |             for (auto& block : blocks) { | ||||||
|                 if (shader_info.labels.count(block.start) != 0) { |                 if (shader_info.labels.count(block.start) != 0) { | ||||||
|                     insert_block(current_block, current_label); |                     insert_block(current_block, current_label); | ||||||
|  |  | ||||||
|  | @ -24,9 +24,8 @@ StagingCache::StagingCache() = default; | ||||||
| StagingCache::~StagingCache() = default; | StagingCache::~StagingCache() = default; | ||||||
| 
 | 
 | ||||||
| SurfaceBaseImpl::SurfaceBaseImpl(GPUVAddr gpu_addr, const SurfaceParams& params) | SurfaceBaseImpl::SurfaceBaseImpl(GPUVAddr gpu_addr, const SurfaceParams& params) | ||||||
|     : params{params}, mipmap_sizes(params.num_levels), |     : params{params}, host_memory_size{params.GetHostSizeInBytes()}, gpu_addr{gpu_addr}, | ||||||
|       mipmap_offsets(params.num_levels), gpu_addr{gpu_addr}, host_memory_size{ |       mipmap_sizes(params.num_levels), mipmap_offsets(params.num_levels) { | ||||||
|                                                                  params.GetHostSizeInBytes()} { |  | ||||||
|     std::size_t offset = 0; |     std::size_t offset = 0; | ||||||
|     for (u32 level = 0; level < params.num_levels; ++level) { |     for (u32 level = 0; level < params.num_levels; ++level) { | ||||||
|         const std::size_t mipmap_size{params.GetGuestMipmapSize(level)}; |         const std::size_t mipmap_size{params.GetGuestMipmapSize(level)}; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei