| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | // Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/basic_block.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-14 20:15:42 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/post_order.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/structured_control_flow.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | #include "shader_recompiler/frontend/maxwell/program.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/maxwell/translate/translate.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | #include "shader_recompiler/ir_opt/passes.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Maxwell { | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | IR::BlockList TranslateCode(ObjectPool<IR::Inst>& inst_pool, ObjectPool<IR::Block>& block_pool, | 
					
						
							|  |  |  |                             Environment& env, Flow::Function& cfg_function) { | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  |     const size_t num_blocks{cfg_function.blocks.size()}; | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |     std::vector<IR::Block*> blocks(cfg_function.blocks.size()); | 
					
						
							|  |  |  |     std::ranges::for_each(cfg_function.blocks, [&, i = size_t{0}](auto& cfg_block) mutable { | 
					
						
							|  |  |  |         const u32 begin{cfg_block.begin.Offset()}; | 
					
						
							|  |  |  |         const u32 end{cfg_block.end.Offset()}; | 
					
						
							|  |  |  |         blocks[i] = block_pool.Create(inst_pool, begin, end); | 
					
						
							|  |  |  |         cfg_block.ir = blocks[i]; | 
					
						
							|  |  |  |         ++i; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     std::ranges::for_each(cfg_function.blocks, [&, i = size_t{0}](auto& cfg_block) mutable { | 
					
						
							|  |  |  |         IR::Block* const block{blocks[i]}; | 
					
						
							|  |  |  |         ++i; | 
					
						
							|  |  |  |         if (cfg_block.end_class != Flow::EndClass::Branch) { | 
					
						
							|  |  |  |             block->SetReturn(); | 
					
						
							|  |  |  |         } else if (cfg_block.cond == IR::Condition{true}) { | 
					
						
							|  |  |  |             block->SetBranch(cfg_block.branch_true->ir); | 
					
						
							|  |  |  |         } else if (cfg_block.cond == IR::Condition{false}) { | 
					
						
							|  |  |  |             block->SetBranch(cfg_block.branch_false->ir); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             block->SetBranches(cfg_block.cond, cfg_block.branch_true->ir, | 
					
						
							|  |  |  |                                cfg_block.branch_false->ir); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     return IR::VisitAST(inst_pool, block_pool, blocks, | 
					
						
							|  |  |  |                         [&](IR::Block* block) { Translate(env, block); }); | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | } | 
					
						
							|  |  |  | } // Anonymous namespace
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  | IR::Program TranslateProgram(ObjectPool<IR::Inst>& inst_pool, ObjectPool<IR::Block>& block_pool, | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |                              Environment& env, Flow::CFG& cfg) { | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  |     IR::Program program; | 
					
						
							|  |  |  |     auto& functions{program.functions}; | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  |     functions.reserve(cfg.Functions().size()); | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |     for (Flow::Function& cfg_function : cfg.Functions()) { | 
					
						
							|  |  |  |         functions.push_back(IR::Function{ | 
					
						
							|  |  |  |             .blocks{TranslateCode(inst_pool, block_pool, env, cfg_function)}, | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |             .post_order_blocks{}, | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     for (IR::Function& function : functions) { | 
					
						
							| 
									
										
										
										
											2021-02-14 20:15:42 -03:00
										 |  |  |         function.post_order_blocks = PostOrder(function.blocks); | 
					
						
							|  |  |  |         Optimization::SsaRewritePass(function.post_order_blocks); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     fmt::print(stdout, "{}\n", IR::DumpProgram(program)); | 
					
						
							|  |  |  |     Optimization::GlobalMemoryToStorageBufferPass(program); | 
					
						
							| 
									
										
										
										
											2021-02-14 20:15:42 -03:00
										 |  |  |     for (IR::Function& function : functions) { | 
					
						
							|  |  |  |         Optimization::PostOrderInvoke(Optimization::ConstantPropagationPass, function); | 
					
						
							|  |  |  |         Optimization::PostOrderInvoke(Optimization::DeadCodeEliminationPass, function); | 
					
						
							| 
									
										
										
										
											2021-02-05 05:58:02 -03:00
										 |  |  |         Optimization::IdentityRemovalPass(function); | 
					
						
							|  |  |  |         Optimization::VerificationPass(function); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     Optimization::CollectShaderInfoPass(program); | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  |     fmt::print(stdout, "{}\n", IR::DumpProgram(program)); | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  |     return program; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Maxwell
 |