mirror of
https://git.suyu.dev/suyu/suyu.git
synced 2024-12-01 07:46:27 -05:00
682 lines
16 KiB
C++
682 lines
16 KiB
C++
// Copyright 2021 yuzu Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <string_view>
|
|
|
|
#include "shader_recompiler/backend/glsl/emit_context.h"
|
|
#include "shader_recompiler/backend/glsl/emit_glsl_instructions.h"
|
|
#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::GLSL {
|
|
|
|
static void NotImplemented() {
|
|
throw NotImplementedException("GLSL instruction");
|
|
}
|
|
|
|
void EmitPhi(EmitContext& ctx, IR::Inst& phi) {
|
|
const size_t num_args{phi.NumArgs()};
|
|
for (size_t i = 0; i < num_args; ++i) {
|
|
ctx.reg_alloc.Consume(phi.Arg(i));
|
|
}
|
|
if (!phi.Definition<Id>().is_valid) {
|
|
// The phi node wasn't forward defined
|
|
ctx.Add("{};", ctx.reg_alloc.Define(phi, phi.Arg(0).Type()));
|
|
}
|
|
}
|
|
|
|
void EmitVoid(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitReference(EmitContext&) {
|
|
// NotImplemented();
|
|
}
|
|
|
|
void EmitPhiMove(EmitContext& ctx, const IR::Value& phi_value, const IR::Value& value) {
|
|
IR::Inst& phi{*phi_value.InstRecursive()};
|
|
const auto phi_type{phi.Arg(0).Type()};
|
|
if (!phi.Definition<Id>().is_valid) {
|
|
// The phi node wasn't forward defined
|
|
ctx.Add("{};", ctx.reg_alloc.Define(phi, phi_type));
|
|
}
|
|
const auto phi_reg{ctx.reg_alloc.Consume(IR::Value{&phi})};
|
|
const auto val_reg{ctx.reg_alloc.Consume(value)};
|
|
if (phi_reg == val_reg) {
|
|
return;
|
|
}
|
|
if (phi_type == value.Type()) {
|
|
ctx.Add("{}={}; // PHI MOVE", phi_reg, val_reg);
|
|
} else if (phi_type == IR::Type::U32 && value.Type() == IR::Type::F32) {
|
|
ctx.Add("{}=floatBitsToUint({}); // CAST PHI MOVE", phi_reg, val_reg);
|
|
} else {
|
|
throw NotImplementedException("{} to {} move", phi_type, value.Type());
|
|
const auto cast{ctx.reg_alloc.GetGlslType(phi_type)};
|
|
ctx.Add("{}={}({}); // CAST PHI MOVE", phi_reg, cast, val_reg);
|
|
}
|
|
}
|
|
|
|
void EmitBranch(EmitContext& ctx, std::string_view label) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBranchConditional(EmitContext& ctx, std::string_view condition,
|
|
std::string_view true_label, std::string_view false_label) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoopMerge(EmitContext& ctx, std::string_view merge_label,
|
|
std::string_view continue_label) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSelectionMerge(EmitContext& ctx, std::string_view merge_label) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitReturn(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitJoin(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitUnreachable(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitDemoteToHelperInvocation(EmitContext& ctx, std::string_view continue_label) {
|
|
ctx.Add("discard;");
|
|
}
|
|
|
|
void EmitBarrier(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWorkgroupMemoryBarrier(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitDeviceMemoryBarrier(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitPrologue(EmitContext& ctx) {
|
|
// NotImplemented();
|
|
}
|
|
|
|
void EmitEpilogue(EmitContext& ctx) {
|
|
// NotImplemented();
|
|
}
|
|
|
|
void EmitEmitVertex(EmitContext& ctx, const IR::Value& stream) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitEndPrimitive(EmitContext& ctx, const IR::Value& stream) {
|
|
NotImplemented();
|
|
}
|
|
|
|
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 EmitGetAttributeIndexed(EmitContext& ctx, std::string_view offset, std::string_view vertex) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSetAttributeIndexed(EmitContext& ctx, std::string_view offset, std::string_view value,
|
|
std::string_view vertex) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitGetPatch(EmitContext& ctx, IR::Patch patch) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSetPatch(EmitContext& ctx, IR::Patch patch, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSetSampleMask(EmitContext& ctx, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSetFragDepth(EmitContext& ctx, std::string_view value) {
|
|
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();
|
|
}
|
|
|
|
void EmitWorkgroupId(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLocalInvocationId(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitInvocationId(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSampleId(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitIsHelperInvocation(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitYDirection(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadLocal(EmitContext& ctx, std::string_view word_offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteLocal(EmitContext& ctx, std::string_view word_offset, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitUndefU1(EmitContext& ctx, IR::Inst& inst) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitUndefU8(EmitContext& ctx, IR::Inst& inst) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitUndefU16(EmitContext& ctx, IR::Inst& inst) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitUndefU32(EmitContext& ctx, IR::Inst& inst) {
|
|
ctx.AddU32("{}=0u;", inst);
|
|
}
|
|
|
|
void EmitUndefU64(EmitContext& ctx, IR::Inst& inst) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobalU8(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobalS8(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobalU16(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobalS16(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobal32(EmitContext& ctx, std::string_view address) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobal64(EmitContext& ctx, std::string_view address) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadGlobal128(EmitContext& ctx, std::string_view address) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobalU8(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobalS8(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobalU16(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobalS16(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobal32(EmitContext& ctx, std::string_view address, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobal64(EmitContext& ctx, std::string_view address, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteGlobal128(EmitContext& ctx, std::string_view address, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedU8(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedS8(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedU16(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedS16(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedU32(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedU64(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLoadSharedU128(EmitContext& ctx, std::string_view offset) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteSharedU8(EmitContext& ctx, std::string_view offset, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteSharedU16(EmitContext& ctx, std::string_view offset, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteSharedU32(EmitContext& ctx, std::string_view offset, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteSharedU64(EmitContext& ctx, std::string_view offset, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitWriteSharedU128(EmitContext& ctx, std::string_view offset, std::string_view value) {
|
|
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();
|
|
}
|
|
|
|
void EmitSharedAtomicIAdd32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicSMin32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicUMin32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicSMax32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicUMax32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicInc32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicDec32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicAnd32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicOr32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicXor32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicExchange32(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSharedAtomicExchange64(EmitContext& ctx, std::string_view pointer_offset,
|
|
std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicIAdd32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicSMin32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicUMin32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicSMax32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicUMax32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicInc32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicDec32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicAnd32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicOr32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicXor32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBindlessImageAtomicExchange32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicIAdd32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicSMin32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicUMin32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicSMax32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicUMax32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicInc32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicDec32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicAnd32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicOr32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicXor32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitBoundImageAtomicExchange32(EmitContext&) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicIAdd32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicSMin32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicUMin32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicSMax32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicUMax32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicInc32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicDec32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicAnd32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicOr32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicXor32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitImageAtomicExchange32(EmitContext& ctx, IR::Inst& inst, const IR::Value& index,
|
|
std::string_view coords, std::string_view value) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitLaneId(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitVoteAll(EmitContext& ctx, std::string_view pred) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitVoteAny(EmitContext& ctx, std::string_view pred) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitVoteEqual(EmitContext& ctx, std::string_view pred) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSubgroupBallot(EmitContext& ctx, std::string_view pred) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSubgroupEqMask(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSubgroupLtMask(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSubgroupLeMask(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSubgroupGtMask(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitSubgroupGeMask(EmitContext& ctx) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitShuffleIndex(EmitContext& ctx, IR::Inst& inst, std::string_view value,
|
|
std::string_view index, std::string_view clamp,
|
|
std::string_view segmentation_mask) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitShuffleUp(EmitContext& ctx, IR::Inst& inst, std::string_view value, std::string_view index,
|
|
std::string_view clamp, std::string_view segmentation_mask) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitShuffleDown(EmitContext& ctx, IR::Inst& inst, std::string_view value,
|
|
std::string_view index, std::string_view clamp,
|
|
std::string_view segmentation_mask) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitShuffleButterfly(EmitContext& ctx, IR::Inst& inst, std::string_view value,
|
|
std::string_view index, std::string_view clamp,
|
|
std::string_view segmentation_mask) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitFSwizzleAdd(EmitContext& ctx, std::string_view op_a, std::string_view op_b,
|
|
std::string_view swizzle) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitDPdxFine(EmitContext& ctx, std::string_view op_a) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitDPdyFine(EmitContext& ctx, std::string_view op_a) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitDPdxCoarse(EmitContext& ctx, std::string_view op_a) {
|
|
NotImplemented();
|
|
}
|
|
|
|
void EmitDPdyCoarse(EmitContext& ctx, std::string_view op_a) {
|
|
NotImplemented();
|
|
}
|
|
|
|
} // namespace Shader::Backend::GLSL
|