| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 18:34:31 -04:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/container/small_vector.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/basic_block.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-21 00:35:47 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/value.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | #include "shader_recompiler/ir_opt/passes.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Optimization { | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  | namespace { | 
					
						
							|  |  |  | template <bool TEST_USES> | 
					
						
							|  |  |  | void DeadInstElimination(IR::Block* const block) { | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     // We iterate over the instructions in reverse order.
 | 
					
						
							|  |  |  |     // This is because removing an instruction reduces the number of uses for earlier instructions.
 | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  |     auto it{block->end()}; | 
					
						
							|  |  |  |     while (it != block->begin()) { | 
					
						
							|  |  |  |         --it; | 
					
						
							|  |  |  |         if constexpr (TEST_USES) { | 
					
						
							|  |  |  |             if (it->HasUses() || it->MayHaveSideEffects()) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  |         it->Invalidate(); | 
					
						
							|  |  |  |         it = block->Instructions().erase(it); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 18:34:31 -04:00
										 |  |  | void DeletedPhiArgElimination(IR::Program& program, std::span<const IR::Block*> dead_blocks) { | 
					
						
							|  |  |  |     for (IR::Block* const block : program.blocks) { | 
					
						
							|  |  |  |         for (IR::Inst& phi : *block) { | 
					
						
							|  |  |  |             if (!IR::IsPhi(phi)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (size_t i = 0; i < phi.NumArgs(); ++i) { | 
					
						
							|  |  |  |                 if (std::ranges::find(dead_blocks, phi.PhiBlock(i)) == dead_blocks.end()) { | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 // Phi operand at this index is an unreachable block
 | 
					
						
							|  |  |  |                 phi.ErasePhiOperand(i); | 
					
						
							|  |  |  |                 --i; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  | void DeadBranchElimination(IR::Program& program) { | 
					
						
							| 
									
										
										
										
											2022-03-22 18:34:31 -04:00
										 |  |  |     boost::container::small_vector<const IR::Block*, 3> dead_blocks; | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  |     const auto begin_it{program.syntax_list.begin()}; | 
					
						
							|  |  |  |     for (auto node_it = begin_it; node_it != program.syntax_list.end(); ++node_it) { | 
					
						
							|  |  |  |         if (node_it->type != IR::AbstractSyntaxNode::Type::If) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         IR::Inst* const cond_ref{node_it->data.if_node.cond.Inst()}; | 
					
						
							|  |  |  |         const IR::U1 cond{cond_ref->Arg(0)}; | 
					
						
							|  |  |  |         if (!cond.IsImmediate()) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (cond.U1()) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // False immediate condition. Remove condition ref, erase the entire branch.
 | 
					
						
							|  |  |  |         cond_ref->Invalidate(); | 
					
						
							|  |  |  |         // Account for nested if-statements within the if(false) branch
 | 
					
						
							|  |  |  |         u32 nested_ifs{1u}; | 
					
						
							|  |  |  |         while (node_it->type != IR::AbstractSyntaxNode::Type::EndIf || nested_ifs > 0) { | 
					
						
							|  |  |  |             node_it = program.syntax_list.erase(node_it); | 
					
						
							|  |  |  |             switch (node_it->type) { | 
					
						
							|  |  |  |             case IR::AbstractSyntaxNode::Type::If: | 
					
						
							|  |  |  |                 ++nested_ifs; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case IR::AbstractSyntaxNode::Type::EndIf: | 
					
						
							|  |  |  |                 --nested_ifs; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case IR::AbstractSyntaxNode::Type::Block: { | 
					
						
							|  |  |  |                 IR::Block* const block{node_it->data.block}; | 
					
						
							|  |  |  |                 DeadInstElimination<false>(block); | 
					
						
							| 
									
										
										
										
											2022-03-22 18:34:31 -04:00
										 |  |  |                 dead_blocks.push_back(block); | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Erase EndIf node of the if(false) branch
 | 
					
						
							|  |  |  |         node_it = program.syntax_list.erase(node_it); | 
					
						
							|  |  |  |         // Account for loop increment
 | 
					
						
							|  |  |  |         --node_it; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-22 18:34:31 -04:00
										 |  |  |     if (!dead_blocks.empty()) { | 
					
						
							|  |  |  |         DeletedPhiArgElimination(program, std::span(dead_blocks.data(), dead_blocks.size())); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-22 01:22:21 -04:00
										 |  |  | } | 
					
						
							|  |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DeadCodeEliminationPass(IR::Program& program) { | 
					
						
							|  |  |  |     DeadBranchElimination(program); | 
					
						
							|  |  |  |     for (IR::Block* const block : program.post_order_blocks) { | 
					
						
							|  |  |  |         DeadInstElimination<true>(block); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Optimization
 |