| 
									
										
										
										
											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>
 | 
					
						
							|  |  |  | #include <tuple>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/bindings.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/glasm/emit_context.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/glasm/emit_glasm.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/glasm/emit_glasm_instructions.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/program.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/profile.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Backend::GLASM { | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | template <class Func> | 
					
						
							|  |  |  | struct FuncTraits {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <class ReturnType_, class... Args> | 
					
						
							|  |  |  | struct FuncTraits<ReturnType_ (*)(Args...)> { | 
					
						
							|  |  |  |     using ReturnType = ReturnType_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static constexpr size_t NUM_ARGS = sizeof...(Args); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template <size_t I> | 
					
						
							|  |  |  |     using ArgType = std::tuple_element_t<I, std::tuple<Args...>>; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  | template <typename T> | 
					
						
							|  |  |  | struct Identity { | 
					
						
							|  |  |  |     Identity(const T& data_) : data{data_} {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const T& Extract() { | 
					
						
							|  |  |  |         return data; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     T data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <bool scalar> | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  | class RegWrapper { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     RegWrapper(EmitContext& ctx, const IR::Value& ir_value) : reg_alloc{ctx.reg_alloc} { | 
					
						
							|  |  |  |         const Value value{reg_alloc.Peek(ir_value)}; | 
					
						
							|  |  |  |         if (value.type == Type::Register) { | 
					
						
							|  |  |  |             inst = ir_value.InstRecursive(); | 
					
						
							|  |  |  |             reg = Register{value}; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2021-05-10 01:33:24 -03:00
										 |  |  |             const bool is_long{value.type == Type::F64 || value.type == Type::U64}; | 
					
						
							|  |  |  |             reg = is_long ? reg_alloc.AllocLongReg() : reg_alloc.AllocReg(); | 
					
						
							| 
									
										
										
										
											2021-05-09 18:03:01 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         switch (value.type) { | 
					
						
							|  |  |  |         case Type::Register: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Type::U32: | 
					
						
							|  |  |  |             ctx.Add("MOV.U {}.x,{};", reg, value.imm_u32); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Type::S32: | 
					
						
							|  |  |  |             ctx.Add("MOV.S {}.x,{};", reg, value.imm_s32); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Type::F32: | 
					
						
							|  |  |  |             ctx.Add("MOV.F {}.x,{};", reg, value.imm_f32); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-05-09 22:43:29 -03:00
										 |  |  |         case Type::U64: | 
					
						
							|  |  |  |             ctx.Add("MOV.U64 {}.x,{};", reg, value.imm_u64); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-05-09 18:03:01 -03:00
										 |  |  |         case Type::F64: | 
					
						
							|  |  |  |             ctx.Add("MOV.F64 {}.x,{};", reg, value.imm_f64); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |     ~RegWrapper() { | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  |         if (inst) { | 
					
						
							|  |  |  |             reg_alloc.Unref(*inst); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |             reg_alloc.FreeReg(reg); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto Extract() { | 
					
						
							|  |  |  |         return std::conditional_t<scalar, ScalarRegister, Register>{Value{reg}}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |     RegAlloc& reg_alloc; | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  |     IR::Inst* inst{}; | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |     Register reg{}; | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename ArgType> | 
					
						
							|  |  |  | class ValueWrapper { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ValueWrapper(EmitContext& ctx, const IR::Value& ir_value_) | 
					
						
							|  |  |  |         : reg_alloc{ctx.reg_alloc}, ir_value{ir_value_}, value{reg_alloc.Peek(ir_value)} {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~ValueWrapper() { | 
					
						
							|  |  |  |         if (!ir_value.IsImmediate()) { | 
					
						
							|  |  |  |             reg_alloc.Unref(*ir_value.InstRecursive()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ArgType Extract() { | 
					
						
							|  |  |  |         return value; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     RegAlloc& reg_alloc; | 
					
						
							|  |  |  |     const IR::Value& ir_value; | 
					
						
							|  |  |  |     ArgType value; | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | template <typename ArgType> | 
					
						
							|  |  |  | auto Arg(EmitContext& ctx, const IR::Value& arg) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |     if constexpr (std::is_same_v<ArgType, Register>) { | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  |         return RegWrapper<false>{ctx, arg}; | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, ScalarRegister>) { | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  |         return RegWrapper<true>{ctx, arg}; | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |     } else if constexpr (std::is_base_of_v<Value, ArgType>) { | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  |         return ValueWrapper<ArgType>{ctx, arg}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, const IR::Value&>) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         return Identity{arg}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, u32>) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         return Identity{arg.U32()}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, IR::Block*>) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         return Identity{arg.Label()}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, IR::Attribute>) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         return Identity{arg.Attribute()}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, IR::Patch>) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         return Identity{arg.Patch()}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else if constexpr (std::is_same_v<ArgType, IR::Reg>) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         return Identity{arg.Reg()}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <auto func, bool is_first_arg_inst, typename... Args> | 
					
						
							|  |  |  | void InvokeCall(EmitContext& ctx, IR::Inst* inst, Args&&... args) { | 
					
						
							|  |  |  |     if constexpr (is_first_arg_inst) { | 
					
						
							|  |  |  |         func(ctx, *inst, std::forward<Args>(args.Extract())...); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         func(ctx, std::forward<Args>(args.Extract())...); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <auto func, bool is_first_arg_inst, size_t... I> | 
					
						
							|  |  |  | void Invoke(EmitContext& ctx, IR::Inst* inst, std::index_sequence<I...>) { | 
					
						
							|  |  |  |     using Traits = FuncTraits<decltype(func)>; | 
					
						
							|  |  |  |     if constexpr (is_first_arg_inst) { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         func(ctx, *inst, | 
					
						
							|  |  |  |              Arg<typename Traits::template ArgType<I + 2>>(ctx, inst->Arg(I)).Extract()...); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         func(ctx, Arg<typename Traits::template ArgType<I + 1>>(ctx, inst->Arg(I)).Extract()...); | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <auto func> | 
					
						
							|  |  |  | void Invoke(EmitContext& ctx, IR::Inst* inst) { | 
					
						
							|  |  |  |     using Traits = FuncTraits<decltype(func)>; | 
					
						
							|  |  |  |     static_assert(Traits::NUM_ARGS >= 1, "Insufficient arguments"); | 
					
						
							|  |  |  |     if constexpr (Traits::NUM_ARGS == 1) { | 
					
						
							|  |  |  |         Invoke<func, false>(ctx, inst, std::make_index_sequence<0>{}); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         using FirstArgType = typename Traits::template ArgType<1>; | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |         static constexpr bool is_first_arg_inst = std::is_same_v<FirstArgType, IR::Inst&>; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |         using Indices = std::make_index_sequence<Traits::NUM_ARGS - (is_first_arg_inst ? 2 : 1)>; | 
					
						
							|  |  |  |         Invoke<func, is_first_arg_inst>(ctx, inst, Indices{}); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitInst(EmitContext& ctx, IR::Inst* inst) { | 
					
						
							|  |  |  |     switch (inst->GetOpcode()) { | 
					
						
							|  |  |  | #define OPCODE(name, result_type, ...)                                                             \
 | 
					
						
							|  |  |  |     case IR::Opcode::name:                                                                         \ | 
					
						
							|  |  |  |         return Invoke<&Emit##name>(ctx, inst); | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/opcodes.inc"
 | 
					
						
							|  |  |  | #undef OPCODE
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw LogicError("Invalid opcode {}", inst->GetOpcode()); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-09 22:01:03 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | void SetupOptions(std::string& header, Info info) { | 
					
						
							|  |  |  |     if (info.uses_int64_bit_atomics) { | 
					
						
							|  |  |  |         header += "OPTION NV_shader_atomic_int64;"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f32_add) { | 
					
						
							|  |  |  |         header += "OPTION NV_shader_atomic_float;"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f16x2_add || info.uses_atomic_f16x2_min || info.uses_atomic_f16x2_max) { | 
					
						
							|  |  |  |         header += "OPTION NV_shader_atomic_fp16_vector;"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string EmitGLASM(const Profile&, IR::Program& program, Bindings&) { | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     EmitContext ctx{program}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     for (IR::Block* const block : program.blocks) { | 
					
						
							|  |  |  |         for (IR::Inst& inst : block->Instructions()) { | 
					
						
							|  |  |  |             EmitInst(ctx, &inst); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     std::string header = "!!NVcp5.0\n" | 
					
						
							|  |  |  |                          "OPTION NV_internal;"; | 
					
						
							| 
									
										
										
										
											2021-05-09 22:01:03 -04:00
										 |  |  |     SetupOptions(header, program.info); | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     switch (program.stage) { | 
					
						
							|  |  |  |     case Stage::Compute: | 
					
						
							|  |  |  |         header += fmt::format("GROUP_SIZE {} {} {};", program.workgroup_size[0], | 
					
						
							|  |  |  |                               program.workgroup_size[1], program.workgroup_size[2]); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     header += "TEMP "; | 
					
						
							|  |  |  |     for (size_t index = 0; index < ctx.reg_alloc.NumUsedRegisters(); ++index) { | 
					
						
							|  |  |  |         header += fmt::format("R{},", index); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-09 18:03:01 -03:00
										 |  |  |     header += "RC;" | 
					
						
							|  |  |  |               "LONG TEMP "; | 
					
						
							|  |  |  |     for (size_t index = 0; index < ctx.reg_alloc.NumUsedLongRegisters(); ++index) { | 
					
						
							|  |  |  |         header += fmt::format("D{},", index); | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-09 18:03:01 -03:00
										 |  |  |     header += "DC;"; | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     ctx.code.insert(0, header); | 
					
						
							|  |  |  |     ctx.code += "END"; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     return ctx.code; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::Backend::GLASM
 |