| 
									
										
										
										
											2021-05-05 02:19:08 -03: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/glasm/emit_context.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/glasm/emit_glasm_instructions.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/program.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/value.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef _MSC_VER
 | 
					
						
							|  |  |  | #pragma warning(disable : 4100)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Backend::GLASM { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  | #define NotImplemented() throw NotImplementedException("GLASM instruction {}", __LINE__)
 | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  | void EmitPhi(EmitContext&, IR::Inst&) {} | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 18:21:28 -03:00
										 |  |  | void EmitVoid(EmitContext&) {} | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 02:04:22 -03:00
										 |  |  | void EmitReference(EmitContext&) {} | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | void EmitPhiMove(EmitContext& ctx, const IR::Value& phi, const IR::Value& value) { | 
					
						
							|  |  |  |     if (phi == value) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const Register phi_reg{ctx.reg_alloc.Consume(phi)}; | 
					
						
							|  |  |  |     const Value eval_value{ctx.reg_alloc.Consume(value)}; | 
					
						
							| 
									
										
										
										
											2021-05-19 16:32:38 -03:00
										 |  |  |     switch (phi.Inst()->Flags<IR::Type>()) { | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |     case IR::Type::U1: | 
					
						
							|  |  |  |     case IR::Type::U32: | 
					
						
							|  |  |  |     case IR::Type::F32: | 
					
						
							|  |  |  |         ctx.Add("MOV.S {}.x,{};", phi_reg, ScalarS32{eval_value}); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case IR::Type::U64: | 
					
						
							|  |  |  |     case IR::Type::F64: | 
					
						
							|  |  |  |         ctx.Add("MOV.U64 {}.x,{};", phi_reg, ScalarRegister{eval_value}); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         throw NotImplementedException("Phi node type {}", phi.Type()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | void EmitJoin(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  | void EmitDemoteToHelperInvocation(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-05-15 18:15:50 -03:00
										 |  |  |     ctx.Add("KIL TR.x;"); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitBarrier(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-05-19 16:51:52 -03:00
										 |  |  |     ctx.Add("BAR;"); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitWorkgroupMemoryBarrier(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-05-19 16:51:52 -03:00
										 |  |  |     ctx.Add("MEMBAR.CTA;"); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitDeviceMemoryBarrier(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-05-19 16:51:52 -03:00
										 |  |  |     ctx.Add("MEMBAR;"); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitPrologue(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     // TODO
 | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitEpilogue(EmitContext& ctx) { | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     // TODO
 | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 01:49:26 -03:00
										 |  |  | void EmitEmitVertex(EmitContext& ctx, ScalarS32 stream) { | 
					
						
							| 
									
										
										
										
											2021-05-20 19:22:08 -03:00
										 |  |  |     if (stream.type == Type::U32 && stream.imm_u32) { | 
					
						
							|  |  |  |         ctx.Add("EMIT;"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         ctx.Add("EMITS {};", stream); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitEndPrimitive(EmitContext& ctx, const IR::Value& stream) { | 
					
						
							| 
									
										
										
										
											2021-05-20 01:49:26 -03:00
										 |  |  |     if (!stream.IsImmediate()) { | 
					
						
							|  |  |  |         // LOG_WARNING not immediate
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx.reg_alloc.Consume(stream); | 
					
						
							|  |  |  |     ctx.Add("ENDPRIM;"); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetRegister(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetRegister(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetPred(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetPred(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetGotoVariable(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetGotoVariable(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetIndirectBranchVariable(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetIndirectBranchVariable(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetZFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetSFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetCFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetOFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetZFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetSFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetCFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSetOFlag(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 21:06:19 -03:00
										 |  |  | void EmitWorkgroupId(EmitContext& ctx, IR::Inst& inst) { | 
					
						
							|  |  |  |     ctx.Add("MOV.S {},invocation.groupid;", inst); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  | void EmitLocalInvocationId(EmitContext& ctx, IR::Inst& inst) { | 
					
						
							|  |  |  |     ctx.Add("MOV.S {},invocation.localid;", inst); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 20:36:04 -03:00
										 |  |  | void EmitInvocationId(EmitContext& ctx, IR::Inst& inst) { | 
					
						
							|  |  |  |     ctx.Add("MOV.S {}.x,primitive_invocation.x;", inst); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitSampleId(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitIsHelperInvocation(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitYDirection(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitUndefU1(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitUndefU8(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitUndefU16(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitUndefU32(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitUndefU64(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetZeroFromOp(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetSignFromOp(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetCarryFromOp(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetOverflowFromOp(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetSparseFromOp(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitGetInBoundsFromOp(EmitContext& ctx) { | 
					
						
							|  |  |  |     NotImplemented(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 21:17:48 -03:00
										 |  |  | void EmitLogicalOr(EmitContext& ctx, IR::Inst& inst, ScalarS32 a, ScalarS32 b) { | 
					
						
							|  |  |  |     ctx.Add("OR.S {},{},{};", inst, a, b); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 21:17:48 -03:00
										 |  |  | void EmitLogicalAnd(EmitContext& ctx, IR::Inst& inst, ScalarS32 a, ScalarS32 b) { | 
					
						
							|  |  |  |     ctx.Add("AND.S {},{},{};", inst, a, b); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 21:17:48 -03:00
										 |  |  | void EmitLogicalXor(EmitContext& ctx, IR::Inst& inst, ScalarS32 a, ScalarS32 b) { | 
					
						
							|  |  |  |     ctx.Add("XOR.S {},{},{};", inst, a, b); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 21:17:48 -03:00
										 |  |  | void EmitLogicalNot(EmitContext& ctx, IR::Inst& inst, ScalarS32 value) { | 
					
						
							|  |  |  |     ctx.Add("SEQ.S {},{},0;", inst, value); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Backend::GLASM
 |