| 
									
										
										
										
											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-01-09 03:30:07 -03:00
										 |  |  | #include "shader_recompiler/frontend/maxwell/program.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  | #include "shader_recompiler/frontend/maxwell/structured_control_flow.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | #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-04-04 05:18:09 -03:00
										 |  |  | namespace { | 
					
						
							|  |  |  | void RemoveUnreachableBlocks(IR::Program& program) { | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  |     // Some blocks might be unreachable if a function call exists unconditionally
 | 
					
						
							|  |  |  |     // If this happens the number of blocks and post order blocks will mismatch
 | 
					
						
							|  |  |  |     if (program.blocks.size() == program.post_order_blocks.size()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-04 20:00:34 -03:00
										 |  |  |     const auto begin{program.blocks.begin() + 1}; | 
					
						
							| 
									
										
										
										
											2021-04-04 05:18:09 -03:00
										 |  |  |     const auto end{program.blocks.end()}; | 
					
						
							|  |  |  |     const auto pred{[](IR::Block* block) { return block->ImmediatePredecessors().empty(); }}; | 
					
						
							|  |  |  |     program.blocks.erase(std::remove_if(begin, end, pred), end); | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-04 05:18:09 -03:00
										 |  |  | void CollectInterpolationInfo(Environment& env, IR::Program& program) { | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |     if (program.stage != Stage::Fragment) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const ProgramHeader& sph{env.SPH()}; | 
					
						
							|  |  |  |     for (size_t index = 0; index < program.info.input_generics.size(); ++index) { | 
					
						
							|  |  |  |         std::optional<PixelImap> imap; | 
					
						
							|  |  |  |         for (const PixelImap value : sph.ps.GenericInputMap(static_cast<u32>(index))) { | 
					
						
							|  |  |  |             if (value == PixelImap::Unused) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (imap && imap != value) { | 
					
						
							|  |  |  |                 throw NotImplementedException("Per component interpolation"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             imap = value; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!imap) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         program.info.input_generics[index].interpolation = [&] { | 
					
						
							|  |  |  |             switch (*imap) { | 
					
						
							|  |  |  |             case PixelImap::Unused: | 
					
						
							|  |  |  |             case PixelImap::Perspective: | 
					
						
							|  |  |  |                 return Interpolation::Smooth; | 
					
						
							|  |  |  |             case PixelImap::Constant: | 
					
						
							|  |  |  |                 return Interpolation::Flat; | 
					
						
							|  |  |  |             case PixelImap::ScreenLinear: | 
					
						
							|  |  |  |                 return Interpolation::NoPerspective; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             throw NotImplementedException("Unknown interpolation {}", *imap); | 
					
						
							|  |  |  |         }(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-04 05:18:09 -03:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -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; | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  |     program.blocks = VisitAST(inst_pool, block_pool, env, cfg); | 
					
						
							|  |  |  |     program.post_order_blocks = PostOrder(program.blocks); | 
					
						
							| 
									
										
										
										
											2021-03-19 19:28:31 -03:00
										 |  |  |     program.stage = env.ShaderStage(); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     program.local_memory_size = env.LocalMemorySize(); | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |     switch (program.stage) { | 
					
						
							|  |  |  |     case Stage::Geometry: { | 
					
						
							|  |  |  |         const ProgramHeader& sph{env.SPH()}; | 
					
						
							|  |  |  |         program.output_topology = sph.common3.output_topology; | 
					
						
							|  |  |  |         program.output_vertices = sph.common4.max_output_vertices; | 
					
						
							|  |  |  |         program.invocations = sph.common2.threads_per_input_primitive; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case Stage::Compute: | 
					
						
							| 
									
										
										
										
											2021-03-27 03:08:31 -03:00
										 |  |  |         program.workgroup_size = env.WorkgroupSize(); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         program.shared_memory_size = env.SharedMemorySize(); | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-03-27 03:08:31 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  |     RemoveUnreachableBlocks(program); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Replace instructions before the SSA rewrite
 | 
					
						
							| 
									
										
										
										
											2021-02-19 18:10:18 -03:00
										 |  |  |     Optimization::LowerFp16ToFp32(program); | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Optimization::SsaRewritePass(program); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     Optimization::GlobalMemoryToStorageBufferPass(program); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     Optimization::TexturePass(env, program); | 
					
						
							| 
									
										
										
										
											2021-03-14 03:41:05 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Optimization::ConstantPropagationPass(program); | 
					
						
							|  |  |  |     Optimization::DeadCodeEliminationPass(program); | 
					
						
							|  |  |  |     Optimization::IdentityRemovalPass(program); | 
					
						
							|  |  |  |     Optimization::VerificationPass(program); | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     Optimization::CollectShaderInfoPass(env, program); | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |     CollectInterpolationInfo(env, program); | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  |     return program; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Maxwell
 |