| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | // Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string_view>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/glsl/emit_context.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/glsl/emit_glsl_instructions.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-15 00:30:59 -04:00
										 |  |  | #include "shader_recompiler/frontend/ir/program.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | #include "shader_recompiler/frontend/ir/value.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-15 21:01:44 -04:00
										 |  |  | #include "shader_recompiler/profile.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Backend::GLSL { | 
					
						
							| 
									
										
										
										
											2021-06-15 17:23:57 -04:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2021-06-18 00:54:16 -04:00
										 |  |  | std::string_view OutputVertexIndex(EmitContext& ctx) { | 
					
						
							|  |  |  |     return ctx.stage == Stage::TessellationControl ? "[gl_InvocationID]" : ""; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-16 04:59:30 -03:00
										 |  |  | void InitializeOutputVaryings(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-06-24 20:13:21 -04:00
										 |  |  |     if (ctx.uses_geometry_passthrough) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-16 04:59:30 -03:00
										 |  |  |     if (ctx.stage == Stage::VertexB || ctx.stage == Stage::Geometry) { | 
					
						
							|  |  |  |         ctx.Add("gl_Position=vec4(0,0,0,1);"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     for (size_t index = 0; index < IR::NUM_GENERICS; ++index) { | 
					
						
							|  |  |  |         if (!ctx.info.stores.Generic(index)) { | 
					
						
							| 
									
										
										
										
											2021-06-18 00:54:16 -04:00
										 |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const auto& info_array{ctx.output_generics.at(index)}; | 
					
						
							|  |  |  |         const auto output_decorator{OutputVertexIndex(ctx)}; | 
					
						
							|  |  |  |         size_t element{}; | 
					
						
							|  |  |  |         while (element < info_array.size()) { | 
					
						
							|  |  |  |             const auto& info{info_array.at(element)}; | 
					
						
							|  |  |  |             const auto varying_name{fmt::format("{}{}", info.name, output_decorator)}; | 
					
						
							|  |  |  |             switch (info.num_components) { | 
					
						
							|  |  |  |             case 1: { | 
					
						
							|  |  |  |                 const char value{element == 3 ? '1' : '0'}; | 
					
						
							|  |  |  |                 ctx.Add("{}={}.f;", varying_name, value); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case 2: | 
					
						
							|  |  |  |             case 3: | 
					
						
							|  |  |  |                 if (element + info.num_components < 4) { | 
					
						
							|  |  |  |                     ctx.Add("{}=vec{}(0);", varying_name, info.num_components); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     // last element is the w component, must be initialized to 1
 | 
					
						
							|  |  |  |                     const auto zeros{info.num_components == 3 ? "0,0," : "0,"}; | 
					
						
							|  |  |  |                     ctx.Add("{}=vec{}({}1);", varying_name, info.num_components, zeros); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 4: | 
					
						
							|  |  |  |                 ctx.Add("{}=vec4(0,0,0,1);", varying_name); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             element += info.num_components; | 
					
						
							| 
									
										
										
										
											2021-06-15 17:23:57 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } // Anonymous namespace
 | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | void EmitPhi(EmitContext& ctx, IR::Inst& phi) { | 
					
						
							|  |  |  |     const size_t num_args{phi.NumArgs()}; | 
					
						
							|  |  |  |     for (size_t i = 0; i < num_args; ++i) { | 
					
						
							|  |  |  |         ctx.var_alloc.Consume(phi.Arg(i)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!phi.Definition<Id>().is_valid) { | 
					
						
							|  |  |  |         // The phi node wasn't forward defined
 | 
					
						
							| 
									
										
										
										
											2021-11-01 22:26:17 +01:00
										 |  |  |         ctx.var_alloc.PhiDefine(phi, phi.Type()); | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitVoid(EmitContext&) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitReference(EmitContext& ctx, const IR::Value& value) { | 
					
						
							|  |  |  |     ctx.var_alloc.Consume(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitPhiMove(EmitContext& ctx, const IR::Value& phi_value, const IR::Value& value) { | 
					
						
							|  |  |  |     IR::Inst& phi{*phi_value.InstRecursive()}; | 
					
						
							| 
									
										
										
										
											2021-11-01 22:26:17 +01:00
										 |  |  |     const auto phi_type{phi.Type()}; | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  |     if (!phi.Definition<Id>().is_valid) { | 
					
						
							|  |  |  |         // The phi node wasn't forward defined
 | 
					
						
							|  |  |  |         ctx.var_alloc.PhiDefine(phi, phi_type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const auto phi_reg{ctx.var_alloc.Consume(IR::Value{&phi})}; | 
					
						
							|  |  |  |     const auto val_reg{ctx.var_alloc.Consume(value)}; | 
					
						
							|  |  |  |     if (phi_reg == val_reg) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx.Add("{}={};", phi_reg, val_reg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 00:30:59 -04:00
										 |  |  | void EmitPrologue(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-06-16 04:59:30 -03:00
										 |  |  |     InitializeOutputVaryings(ctx); | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 00:30:59 -04:00
										 |  |  | void EmitEpilogue(EmitContext&) {} | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | void EmitEmitVertex(EmitContext& ctx, const IR::Value& stream) { | 
					
						
							|  |  |  |     ctx.Add("EmitStreamVertex(int({}));", ctx.var_alloc.Consume(stream)); | 
					
						
							| 
									
										
										
										
											2021-06-16 04:59:30 -03:00
										 |  |  |     InitializeOutputVaryings(ctx); | 
					
						
							| 
									
										
										
										
											2021-06-10 00:29:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitEndPrimitive(EmitContext& ctx, const IR::Value& stream) { | 
					
						
							|  |  |  |     ctx.Add("EndStreamPrimitive(int({}));", ctx.var_alloc.Consume(stream)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Backend::GLSL
 |