| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  | #include <ranges>
 | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | #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"
 | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/ir_emitter.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | #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 { | 
					
						
							| 
									
										
										
										
											2021-05-10 19:20:44 -03:00
										 |  |  |     Identity(T data_) : data{data_} {} | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 19:20:44 -03:00
										 |  |  |     T Extract() { | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |         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: | 
					
						
							| 
									
										
										
										
											2021-05-15 18:15:13 -03:00
										 |  |  |         case Type::Void: | 
					
						
							| 
									
										
										
										
											2021-05-09 03:11:34 -03:00
										 |  |  |             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-10 19:20:44 -03:00
										 |  |  |     auto Extract() { | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         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)} {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 19:20:44 -03:00
										 |  |  |     ArgType Extract() { | 
					
						
							| 
									
										
										
										
											2021-05-10 03:47:31 -03:00
										 |  |  |         if (!ir_value.IsImmediate()) { | 
					
						
							|  |  |  |             reg_alloc.Unref(*ir_value.InstRecursive()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         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-10 19:20:44 -03:00
										 |  |  |         return Identity<const IR::Value&>{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::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()}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 19:32:10 -03:00
										 |  |  | template <auto func, bool is_first_arg_inst> | 
					
						
							|  |  |  | struct InvokeCall { | 
					
						
							|  |  |  |     template <typename... Args> | 
					
						
							|  |  |  |     InvokeCall(EmitContext& ctx, IR::Inst* inst, Args&&... args) { | 
					
						
							|  |  |  |         if constexpr (is_first_arg_inst) { | 
					
						
							|  |  |  |             func(ctx, *inst, args.Extract()...); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             func(ctx, args.Extract()...); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-10 19:32:10 -03:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											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-10 19:32:10 -03:00
										 |  |  |         InvokeCall<func, is_first_arg_inst>{ | 
					
						
							|  |  |  |             ctx, inst, Arg<typename Traits::template ArgType<I + 2>>(ctx, inst->Arg(I))...}; | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-05-10 19:32:10 -03:00
										 |  |  |         InvokeCall<func, is_first_arg_inst>{ | 
					
						
							|  |  |  |             ctx, inst, Arg<typename Traits::template ArgType<I + 1>>(ctx, inst->Arg(I))...}; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  | void Precolor(EmitContext& ctx, const IR::Program& program) { | 
					
						
							|  |  |  |     for (IR::Block* const block : program.blocks) { | 
					
						
							|  |  |  |         for (IR::Inst& phi : block->Instructions() | std::views::take_while(IR::IsPhi)) { | 
					
						
							|  |  |  |             switch (phi.Arg(0).Type()) { | 
					
						
							|  |  |  |             case IR::Type::U1: | 
					
						
							|  |  |  |             case IR::Type::U32: | 
					
						
							|  |  |  |             case IR::Type::F32: | 
					
						
							|  |  |  |                 ctx.reg_alloc.Define(phi); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case IR::Type::U64: | 
					
						
							|  |  |  |             case IR::Type::F64: | 
					
						
							|  |  |  |                 ctx.reg_alloc.LongDefine(phi); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 throw NotImplementedException("Phi node type {}", phi.Type()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const size_t num_args{phi.NumArgs()}; | 
					
						
							|  |  |  |             for (size_t i = 0; i < num_args; ++i) { | 
					
						
							|  |  |  |                 IR::IREmitter{*phi.PhiBlock(i)}.PhiMove(phi, phi.Arg(i)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             // Add reference to the phi node on the phi predecessor to avoid overwritting it
 | 
					
						
							|  |  |  |             for (size_t i = 0; i < num_args; ++i) { | 
					
						
							| 
									
										
										
										
											2021-05-18 02:04:22 -03:00
										 |  |  |                 IR::IREmitter{*phi.PhiBlock(i)}.Reference(IR::Value{&phi}); | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  | void EmitCode(EmitContext& ctx, const IR::Program& program) { | 
					
						
							|  |  |  |     const auto eval{ | 
					
						
							|  |  |  |         [&](const IR::U1& cond) { return ScalarS32{ctx.reg_alloc.Consume(IR::Value{cond})}; }}; | 
					
						
							|  |  |  |     for (const IR::AbstractSyntaxNode& node : program.syntax_list) { | 
					
						
							|  |  |  |         switch (node.type) { | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::Block: | 
					
						
							| 
									
										
										
										
											2021-05-16 17:06:13 -04:00
										 |  |  |             for (IR::Inst& inst : node.data.block->Instructions()) { | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |                 EmitInst(ctx, &inst); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::If: | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |             ctx.Add("MOV.S.CC RC,{};" | 
					
						
							|  |  |  |                     "IF NE.x;", | 
					
						
							| 
									
										
										
										
											2021-05-16 17:06:13 -04:00
										 |  |  |                     eval(node.data.if_node.cond)); | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::EndIf: | 
					
						
							|  |  |  |             ctx.Add("ENDIF;"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::Loop: | 
					
						
							|  |  |  |             ctx.Add("REP;"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::Repeat: | 
					
						
							| 
									
										
										
										
											2021-05-16 17:06:13 -04:00
										 |  |  |             if (node.data.repeat.cond.IsImmediate()) { | 
					
						
							|  |  |  |                 if (node.data.repeat.cond.U1()) { | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |                     ctx.Add("ENDREP;"); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     ctx.Add("BRK;" | 
					
						
							|  |  |  |                             "ENDREP;"); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 ctx.Add("MOV.S.CC RC,{};" | 
					
						
							|  |  |  |                         "BRK (EQ.x);" | 
					
						
							|  |  |  |                         "ENDREP;", | 
					
						
							| 
									
										
										
										
											2021-05-16 17:06:13 -04:00
										 |  |  |                         eval(node.data.repeat.cond)); | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::Break: | 
					
						
							| 
									
										
										
										
											2021-05-16 17:06:13 -04:00
										 |  |  |             if (node.data.break_node.cond.IsImmediate()) { | 
					
						
							|  |  |  |                 if (node.data.break_node.cond.U1()) { | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |                     ctx.Add("BRK;"); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 ctx.Add("MOV.S.CC RC,{};" | 
					
						
							|  |  |  |                         "BRK (NE.x);", | 
					
						
							| 
									
										
										
										
											2021-05-16 17:06:13 -04:00
										 |  |  |                         eval(node.data.break_node.cond)); | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::Return: | 
					
						
							|  |  |  |         case IR::AbstractSyntaxNode::Type::Unreachable: | 
					
						
							|  |  |  |             ctx.Add("RET;"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 16:32:03 -03:00
										 |  |  | void SetupOptions(const IR::Program& program, const Profile& profile, std::string& header) { | 
					
						
							|  |  |  |     const Info& info{program.info}; | 
					
						
							|  |  |  |     const Stage stage{program.stage}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-15 18:14:29 -03:00
										 |  |  |     // TODO: Track the shared atomic ops
 | 
					
						
							|  |  |  |     header += "OPTION NV_internal;" | 
					
						
							|  |  |  |               "OPTION NV_shader_storage_buffer;" | 
					
						
							|  |  |  |               "OPTION NV_gpu_program_fp64;" | 
					
						
							|  |  |  |               "OPTION NV_bindless_texture;" | 
					
						
							| 
									
										
										
										
											2021-05-20 02:18:52 -03:00
										 |  |  |               "OPTION ARB_derivative_control;" | 
					
						
							|  |  |  |               "OPTION EXT_shader_image_load_formatted;"; | 
					
						
							| 
									
										
										
										
											2021-05-09 22:01:03 -04:00
										 |  |  |     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-14 02:09:33 -03:00
										 |  |  |     if (info.uses_subgroup_invocation_id || info.uses_subgroup_mask || info.uses_subgroup_vote) { | 
					
						
							| 
									
										
										
										
											2021-05-10 18:21:28 -03:00
										 |  |  |         header += "OPTION NV_shader_thread_group;"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_subgroup_shuffles) { | 
					
						
							|  |  |  |         header += "OPTION NV_shader_thread_shuffle;"; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-17 02:52:01 -03:00
										 |  |  |     if (info.uses_sparse_residency) { | 
					
						
							|  |  |  |         header += "OPTION EXT_sparse_texture2;"; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-19 16:32:03 -03:00
										 |  |  |     if ((info.stores_viewport_index || info.stores_layer) && stage != Stage::Geometry) { | 
					
						
							|  |  |  |         if (profile.support_viewport_index_layer_non_geometry) { | 
					
						
							|  |  |  |             header += "OPTION NV_viewport_array2;"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-16 17:52:30 -03:00
										 |  |  |     const auto non_zero_frag_colors{info.stores_frag_color | std::views::drop(1)}; | 
					
						
							|  |  |  |     if (std::ranges::find(non_zero_frag_colors, true) != non_zero_frag_colors.end()) { | 
					
						
							|  |  |  |         header += "OPTION ARB_draw_buffers;"; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-15 18:14:29 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string_view StageHeader(Stage stage) { | 
					
						
							|  |  |  |     switch (stage) { | 
					
						
							|  |  |  |     case Stage::VertexA: | 
					
						
							|  |  |  |     case Stage::VertexB: | 
					
						
							|  |  |  |         return "!!NVvp5.0\n"; | 
					
						
							|  |  |  |     case Stage::TessellationControl: | 
					
						
							| 
									
										
										
										
											2021-05-20 21:39:45 -03:00
										 |  |  |         return "!!NVtcp5.0\n"; | 
					
						
							| 
									
										
										
										
											2021-05-15 18:14:29 -03:00
										 |  |  |     case Stage::TessellationEval: | 
					
						
							| 
									
										
										
										
											2021-05-20 21:39:45 -03:00
										 |  |  |         return "!!NVtep5.0\n"; | 
					
						
							| 
									
										
										
										
											2021-05-15 18:14:29 -03:00
										 |  |  |     case Stage::Geometry: | 
					
						
							|  |  |  |         return "!!NVgp5.0\n"; | 
					
						
							|  |  |  |     case Stage::Fragment: | 
					
						
							|  |  |  |         return "!!NVfp5.0\n"; | 
					
						
							|  |  |  |     case Stage::Compute: | 
					
						
							|  |  |  |         return "!!NVcp5.0\n"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid stage {}", stage); | 
					
						
							| 
									
										
										
										
											2021-05-09 22:01:03 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-20 17:27:39 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | std::string_view InputPrimitive(InputTopology topology) { | 
					
						
							|  |  |  |     switch (topology) { | 
					
						
							|  |  |  |     case InputTopology::Points: | 
					
						
							|  |  |  |         return "POINTS"; | 
					
						
							|  |  |  |     case InputTopology::Lines: | 
					
						
							|  |  |  |         return "LINES"; | 
					
						
							|  |  |  |     case InputTopology::LinesAdjacency: | 
					
						
							|  |  |  |         return "LINESS_ADJACENCY"; | 
					
						
							|  |  |  |     case InputTopology::Triangles: | 
					
						
							|  |  |  |         return "TRIANGLES"; | 
					
						
							|  |  |  |     case InputTopology::TrianglesAdjacency: | 
					
						
							|  |  |  |         return "TRIANGLES_ADJACENCY"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid input topology {}", topology); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string_view OutputPrimitive(OutputTopology topology) { | 
					
						
							|  |  |  |     switch (topology) { | 
					
						
							|  |  |  |     case OutputTopology::PointList: | 
					
						
							|  |  |  |         return "POINTS"; | 
					
						
							|  |  |  |     case OutputTopology::LineStrip: | 
					
						
							|  |  |  |         return "LINE_STRIP"; | 
					
						
							|  |  |  |     case OutputTopology::TriangleStrip: | 
					
						
							|  |  |  |         return "TRIANGLE_STRIP"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid output topology {}", topology); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-20 21:18:39 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | std::string_view GetTessMode(TessPrimitive primitive) { | 
					
						
							|  |  |  |     switch (primitive) { | 
					
						
							|  |  |  |     case TessPrimitive::Triangles: | 
					
						
							|  |  |  |         return "TRIANGLES"; | 
					
						
							|  |  |  |     case TessPrimitive::Quads: | 
					
						
							|  |  |  |         return "QUADS"; | 
					
						
							|  |  |  |     case TessPrimitive::Isolines: | 
					
						
							|  |  |  |         return "ISOLINES"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid tessellation primitive {}", primitive); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string_view GetTessSpacing(TessSpacing spacing) { | 
					
						
							|  |  |  |     switch (spacing) { | 
					
						
							|  |  |  |     case TessSpacing::Equal: | 
					
						
							|  |  |  |         return "EQUAL"; | 
					
						
							|  |  |  |     case TessSpacing::FractionalOdd: | 
					
						
							|  |  |  |         return "FRACTIONAL_ODD"; | 
					
						
							|  |  |  |     case TessSpacing::FractionalEven: | 
					
						
							|  |  |  |         return "FRACTIONAL_EVEN"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid tessellation spacing {}", spacing); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-05-05 02:19:08 -03:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-19 16:32:03 -03:00
										 |  |  | std::string EmitGLASM(const Profile& profile, IR::Program& program, Bindings& bindings) { | 
					
						
							|  |  |  |     EmitContext ctx{program, bindings, profile}; | 
					
						
							| 
									
										
										
										
											2021-05-14 04:48:46 -03:00
										 |  |  |     Precolor(ctx, program); | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     EmitCode(ctx, program); | 
					
						
							| 
									
										
										
										
											2021-05-15 18:14:29 -03:00
										 |  |  |     std::string header{StageHeader(program.stage)}; | 
					
						
							| 
									
										
										
										
											2021-05-19 16:32:03 -03:00
										 |  |  |     SetupOptions(program, profile, header); | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     switch (program.stage) { | 
					
						
							| 
									
										
										
										
											2021-05-20 21:18:39 -03:00
										 |  |  |     case Stage::TessellationControl: | 
					
						
							|  |  |  |         header += fmt::format("VERTICES_OUT {};", program.invocations); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Stage::TessellationEval: | 
					
						
							|  |  |  |         header += | 
					
						
							|  |  |  |             fmt::format("TESS_MODE {};" | 
					
						
							|  |  |  |                         "TESS_SPACING {};" | 
					
						
							|  |  |  |                         "TESS_VERTEX_ORDER {};", | 
					
						
							|  |  |  |                         GetTessMode(profile.tess_primitive), GetTessSpacing(profile.tess_spacing), | 
					
						
							|  |  |  |                         profile.tess_clockwise ? "CW" : "CCW"); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-05-20 17:27:39 -03:00
										 |  |  |     case Stage::Geometry: | 
					
						
							|  |  |  |         header += fmt::format("PRIMITIVE_IN {};" | 
					
						
							|  |  |  |                               "PRIMITIVE_OUT {};" | 
					
						
							|  |  |  |                               "VERTICES_OUT {};", | 
					
						
							|  |  |  |                               InputPrimitive(profile.input_topology), | 
					
						
							|  |  |  |                               OutputPrimitive(program.output_topology), program.output_vertices); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     case Stage::Compute: | 
					
						
							|  |  |  |         header += fmt::format("GROUP_SIZE {} {} {};", program.workgroup_size[0], | 
					
						
							|  |  |  |                               program.workgroup_size[1], program.workgroup_size[2]); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-10 22:35:16 -04:00
										 |  |  |     if (program.shared_memory_size > 0) { | 
					
						
							|  |  |  |         header += fmt::format("SHARED_MEMORY {};", program.shared_memory_size); | 
					
						
							|  |  |  |         header += fmt::format("SHARED shared_mem[]={{program.sharedmem}};"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-08 16:28:52 -03:00
										 |  |  |     header += "TEMP "; | 
					
						
							|  |  |  |     for (size_t index = 0; index < ctx.reg_alloc.NumUsedRegisters(); ++index) { | 
					
						
							|  |  |  |         header += fmt::format("R{},", index); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-14 21:18:53 -04:00
										 |  |  |     if (program.local_memory_size > 0) { | 
					
						
							|  |  |  |         header += fmt::format("lmem[{}],", program.local_memory_size); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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
 |