| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | // Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/spirv/emit_spirv.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Backend::SPIRV { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitIAdd32(EmitContext& ctx, IR::Inst* inst, Id a, Id b) { | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  |     Id result{}; | 
					
						
							|  |  |  |     if (IR::Inst* const carry{inst->GetAssociatedPseudoOperation(IR::Opcode::GetCarryFromOp)}) { | 
					
						
							|  |  |  |         const Id carry_type{ctx.TypeStruct(ctx.U32[1], ctx.U32[1])}; | 
					
						
							|  |  |  |         const Id carry_result{ctx.OpIAddCarry(carry_type, a, b)}; | 
					
						
							|  |  |  |         result = ctx.OpCompositeExtract(ctx.U32[1], carry_result, 0U); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id carry_value{ctx.OpCompositeExtract(ctx.U32[1], carry_result, 1U)}; | 
					
						
							|  |  |  |         carry->SetDefinition(ctx.OpINotEqual(ctx.U1, carry_value, ctx.u32_zero_value)); | 
					
						
							|  |  |  |         carry->Invalidate(); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         result = ctx.OpIAdd(ctx.U32[1], a, b); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  |     if (IR::Inst* const zero{inst->GetAssociatedPseudoOperation(IR::Opcode::GetZeroFromOp)}) { | 
					
						
							|  |  |  |         zero->SetDefinition(ctx.OpIEqual(ctx.U1, result, ctx.u32_zero_value)); | 
					
						
							|  |  |  |         zero->Invalidate(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (IR::Inst* const sign{inst->GetAssociatedPseudoOperation(IR::Opcode::GetSignFromOp)}) { | 
					
						
							|  |  |  |         sign->SetDefinition(ctx.OpSLessThan(ctx.U1, result, ctx.u32_zero_value)); | 
					
						
							|  |  |  |         sign->Invalidate(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (IR::Inst * overflow{inst->GetAssociatedPseudoOperation(IR::Opcode::GetOverflowFromOp)}) { | 
					
						
							|  |  |  |         // https://stackoverflow.com/questions/55468823/how-to-detect-integer-overflow-in-c
 | 
					
						
							|  |  |  |         constexpr u32 s32_max{static_cast<u32>(std::numeric_limits<s32>::max())}; | 
					
						
							|  |  |  |         const Id is_positive{ctx.OpSGreaterThanEqual(ctx.U1, a, ctx.u32_zero_value)}; | 
					
						
							|  |  |  |         const Id sub_a{ctx.OpISub(ctx.U32[1], ctx.Constant(ctx.U32[1], s32_max), a)}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id positive_test{ctx.OpSGreaterThan(ctx.U1, b, sub_a)}; | 
					
						
							|  |  |  |         const Id negative_test{ctx.OpSLessThan(ctx.U1, b, sub_a)}; | 
					
						
							|  |  |  |         const Id carry_flag{ctx.OpSelect(ctx.U1, is_positive, positive_test, negative_test)}; | 
					
						
							|  |  |  |         overflow->SetDefinition(carry_flag); | 
					
						
							|  |  |  |         overflow->Invalidate(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | void EmitIAdd64(EmitContext&) { | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  |     throw NotImplementedException("SPIR-V Instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitISub32(EmitContext& ctx, Id a, Id b) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     return ctx.OpISub(ctx.U32[1], a, b); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | void EmitISub64(EmitContext&) { | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  |     throw NotImplementedException("SPIR-V Instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitIMul32(EmitContext& ctx, Id a, Id b) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     return ctx.OpIMul(ctx.U32[1], a, b); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 02:45:50 -03:00
										 |  |  | Id EmitINeg32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpSNegate(ctx.U32[1], value); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-05 01:15:16 -05:00
										 |  |  | Id EmitINeg64(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpSNegate(ctx.U64, value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 02:45:50 -03:00
										 |  |  | Id EmitIAbs32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpSAbs(ctx.U32[1], value); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 05:04:12 -03:00
										 |  |  | Id EmitIAbs64(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpSAbs(ctx.U64, value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitShiftLeftLogical32(EmitContext& ctx, Id base, Id shift) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     return ctx.OpShiftLeftLogical(ctx.U32[1], base, shift); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 14:48:03 -05:00
										 |  |  | Id EmitShiftLeftLogical64(EmitContext& ctx, Id base, Id shift) { | 
					
						
							|  |  |  |     return ctx.OpShiftLeftLogical(ctx.U64, base, shift); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 14:48:03 -05:00
										 |  |  | Id EmitShiftRightLogical32(EmitContext& ctx, Id base, Id shift) { | 
					
						
							|  |  |  |     return ctx.OpShiftRightLogical(ctx.U32[1], base, shift); | 
					
						
							| 
									
										
										
										
											2021-03-05 01:15:16 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 14:48:03 -05:00
										 |  |  | Id EmitShiftRightLogical64(EmitContext& ctx, Id base, Id shift) { | 
					
						
							|  |  |  |     return ctx.OpShiftRightLogical(ctx.U64, base, shift); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitShiftRightArithmetic32(EmitContext& ctx, Id base, Id shift) { | 
					
						
							|  |  |  |     return ctx.OpShiftRightArithmetic(ctx.U32[1], base, shift); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitShiftRightArithmetic64(EmitContext& ctx, Id base, Id shift) { | 
					
						
							|  |  |  |     return ctx.OpShiftRightArithmetic(ctx.U64, base, shift); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitBitwiseAnd32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpBitwiseAnd(ctx.U32[1], a, b); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitBitwiseOr32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpBitwiseOr(ctx.U32[1], a, b); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitBitwiseXor32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpBitwiseXor(ctx.U32[1], a, b); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-23 04:46:39 -03:00
										 |  |  | Id EmitBitFieldInsert(EmitContext& ctx, Id base, Id insert, Id offset, Id count) { | 
					
						
							|  |  |  |     return ctx.OpBitFieldInsert(ctx.U32[1], base, insert, offset, count); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-23 04:46:39 -03:00
										 |  |  | Id EmitBitFieldSExtract(EmitContext& ctx, Id base, Id offset, Id count) { | 
					
						
							|  |  |  |     return ctx.OpBitFieldSExtract(ctx.U32[1], base, offset, count); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 22:01:22 -05:00
										 |  |  | Id EmitBitFieldUExtract(EmitContext& ctx, IR::Inst* inst, Id base, Id offset, Id count) { | 
					
						
							|  |  |  |     const Id result{ctx.OpBitFieldUExtract(ctx.U32[1], base, offset, count)}; | 
					
						
							|  |  |  |     if (IR::Inst* const zero{inst->GetAssociatedPseudoOperation(IR::Opcode::GetZeroFromOp)}) { | 
					
						
							|  |  |  |         zero->SetDefinition(ctx.OpIEqual(ctx.U1, result, ctx.u32_zero_value)); | 
					
						
							|  |  |  |         zero->Invalidate(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-25 00:46:40 -05:00
										 |  |  | Id EmitBitReverse32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpBitReverse(ctx.U32[1], value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-26 21:41:46 -05:00
										 |  |  | Id EmitBitCount32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpBitCount(ctx.U32[1], value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-01 15:58:16 -05:00
										 |  |  | Id EmitBitwiseNot32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpNot(ctx.U32[1], value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitFindSMsb32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpFindSMsb(ctx.U32[1], value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitFindUMsb32(EmitContext& ctx, Id value) { | 
					
						
							|  |  |  |     return ctx.OpFindUMsb(ctx.U32[1], value); | 
					
						
							| 
									
										
										
										
											2021-02-26 21:41:46 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-28 23:33:53 -05:00
										 |  |  | Id EmitSMin32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpSMin(ctx.U32[1], a, b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitUMin32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpUMin(ctx.U32[1], a, b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitSMax32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpSMax(ctx.U32[1], a, b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitUMax32(EmitContext& ctx, Id a, Id b) { | 
					
						
							|  |  |  |     return ctx.OpUMax(ctx.U32[1], a, b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitSLessThan(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     return ctx.OpSLessThan(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitULessThan(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpULessThan(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitIEqual(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpIEqual(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitSLessThanEqual(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpSLessThanEqual(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitULessThanEqual(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpULessThanEqual(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitSGreaterThan(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     return ctx.OpSGreaterThan(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitUGreaterThan(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpUGreaterThan(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitINotEqual(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpINotEqual(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 17:50:14 -03:00
										 |  |  | Id EmitSGreaterThanEqual(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							|  |  |  |     return ctx.OpSGreaterThanEqual(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  | Id EmitUGreaterThanEqual(EmitContext& ctx, Id lhs, Id rhs) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     return ctx.OpUGreaterThanEqual(ctx.U1, lhs, rhs); | 
					
						
							| 
									
										
										
										
											2021-02-08 02:54:35 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Backend::SPIRV
 |