2021-02-08 06:54:35 +01:00
|
|
|
// Copyright 2021 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2021-03-24 05:33:45 +01:00
|
|
|
#include <tuple>
|
|
|
|
|
2021-02-08 06:54:35 +01:00
|
|
|
#include "shader_recompiler/backend/spirv/emit_spirv.h"
|
|
|
|
|
|
|
|
namespace Shader::Backend::SPIRV {
|
2021-03-19 23:28:31 +01:00
|
|
|
namespace {
|
2021-03-27 08:59:58 +01:00
|
|
|
struct AttrInfo {
|
|
|
|
Id pointer;
|
|
|
|
Id id;
|
|
|
|
bool needs_cast;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::optional<AttrInfo> AttrTypes(EmitContext& ctx, u32 index) {
|
2021-03-24 05:33:45 +01:00
|
|
|
const AttributeType type{ctx.profile.generic_input_types.at(index)};
|
|
|
|
switch (type) {
|
|
|
|
case AttributeType::Float:
|
2021-03-27 08:59:58 +01:00
|
|
|
return AttrInfo{ctx.input_f32, ctx.F32[1], false};
|
2021-03-24 05:33:45 +01:00
|
|
|
case AttributeType::UnsignedInt:
|
2021-03-27 08:59:58 +01:00
|
|
|
return AttrInfo{ctx.input_u32, ctx.U32[1], true};
|
2021-03-24 05:33:45 +01:00
|
|
|
case AttributeType::SignedInt:
|
2021-03-27 08:59:58 +01:00
|
|
|
return AttrInfo{ctx.input_s32, ctx.TypeInt(32, true), true};
|
|
|
|
case AttributeType::Disabled:
|
|
|
|
return std::nullopt;
|
2021-03-19 23:28:31 +01:00
|
|
|
}
|
2021-03-24 05:33:45 +01:00
|
|
|
throw InvalidArgument("Invalid attribute type {}", type);
|
2021-03-19 23:28:31 +01:00
|
|
|
}
|
|
|
|
|
2021-04-01 08:34:45 +02:00
|
|
|
std::optional<Id> OutputAttrPointer(EmitContext& ctx, IR::Attribute attr) {
|
2021-03-19 23:28:31 +01:00
|
|
|
const u32 element{static_cast<u32>(attr) % 4};
|
|
|
|
const auto element_id{[&] { return ctx.Constant(ctx.U32[1], element); }};
|
|
|
|
if (IR::IsGeneric(attr)) {
|
|
|
|
const u32 index{IR::GenericAttributeIndex(attr)};
|
|
|
|
return ctx.OpAccessChain(ctx.output_f32, ctx.output_generics.at(index), element_id());
|
|
|
|
}
|
|
|
|
switch (attr) {
|
2021-03-26 23:52:06 +01:00
|
|
|
case IR::Attribute::PointSize:
|
|
|
|
return ctx.output_point_size;
|
2021-03-19 23:28:31 +01:00
|
|
|
case IR::Attribute::PositionX:
|
|
|
|
case IR::Attribute::PositionY:
|
|
|
|
case IR::Attribute::PositionZ:
|
|
|
|
case IR::Attribute::PositionW:
|
|
|
|
return ctx.OpAccessChain(ctx.output_f32, ctx.output_position, element_id());
|
2021-03-30 21:52:06 +02:00
|
|
|
case IR::Attribute::ClipDistance0:
|
|
|
|
case IR::Attribute::ClipDistance1:
|
|
|
|
case IR::Attribute::ClipDistance2:
|
|
|
|
case IR::Attribute::ClipDistance3:
|
|
|
|
case IR::Attribute::ClipDistance4:
|
|
|
|
case IR::Attribute::ClipDistance5:
|
|
|
|
case IR::Attribute::ClipDistance6:
|
|
|
|
case IR::Attribute::ClipDistance7: {
|
|
|
|
const u32 base{static_cast<u32>(IR::Attribute::ClipDistance0)};
|
|
|
|
const u32 index{static_cast<u32>(attr) - base};
|
|
|
|
const Id clip_num{ctx.Constant(ctx.U32[1], index)};
|
|
|
|
return ctx.OpAccessChain(ctx.output_f32, ctx.clip_distances, clip_num);
|
|
|
|
}
|
2021-04-01 08:34:45 +02:00
|
|
|
case IR::Attribute::ViewportIndex:
|
2021-04-03 01:48:39 +02:00
|
|
|
return (ctx.profile.support_viewport_index_layer_non_geometry ||
|
|
|
|
ctx.stage == Shader::Stage::Geometry)
|
|
|
|
? std::optional<Id>{ctx.viewport_index}
|
|
|
|
: std::nullopt;
|
2021-03-19 23:28:31 +01:00
|
|
|
default:
|
|
|
|
throw NotImplementedException("Read attribute {}", attr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} // Anonymous namespace
|
2021-02-08 06:54:35 +01:00
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetRegister(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetRegister(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetPred(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetPred(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetGotoVariable(EmitContext&) {
|
2021-02-11 20:39:06 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetGotoVariable(EmitContext&) {
|
2021-02-11 20:39:06 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-03-27 22:30:24 +01:00
|
|
|
void EmitSetIndirectBranchVariable(EmitContext&) {
|
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
|
|
|
void EmitGetIndirectBranchVariable(EmitContext&) {
|
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-03-09 21:14:57 +01:00
|
|
|
static Id GetCbuf(EmitContext& ctx, Id result_type, Id UniformDefinitions::*member_ptr,
|
|
|
|
u32 element_size, const IR::Value& binding, const IR::Value& offset) {
|
2021-02-08 06:54:35 +01:00
|
|
|
if (!binding.IsImmediate()) {
|
|
|
|
throw NotImplementedException("Constant buffer indexing");
|
|
|
|
}
|
2021-03-09 21:14:57 +01:00
|
|
|
const Id cbuf{ctx.cbufs[binding.U32()].*member_ptr};
|
|
|
|
const Id uniform_type{ctx.uniform_types.*member_ptr};
|
2021-02-08 06:54:35 +01:00
|
|
|
if (!offset.IsImmediate()) {
|
2021-03-09 21:14:57 +01:00
|
|
|
Id index{ctx.Def(offset)};
|
|
|
|
if (element_size > 1) {
|
|
|
|
const u32 log2_element_size{static_cast<u32>(std::countr_zero(element_size))};
|
|
|
|
const Id shift{ctx.Constant(ctx.U32[1], log2_element_size)};
|
|
|
|
index = ctx.OpShiftRightArithmetic(ctx.U32[1], ctx.Def(offset), shift);
|
|
|
|
}
|
|
|
|
const Id access_chain{ctx.OpAccessChain(uniform_type, cbuf, ctx.u32_zero_value, index)};
|
|
|
|
return ctx.OpLoad(result_type, access_chain);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
2021-03-09 21:14:57 +01:00
|
|
|
if (offset.U32() % element_size != 0) {
|
|
|
|
throw NotImplementedException("Unaligned immediate constant buffer load");
|
|
|
|
}
|
|
|
|
const Id imm_offset{ctx.Constant(ctx.U32[1], offset.U32() / element_size)};
|
|
|
|
const Id access_chain{ctx.OpAccessChain(uniform_type, cbuf, ctx.u32_zero_value, imm_offset)};
|
|
|
|
return ctx.OpLoad(result_type, access_chain);
|
|
|
|
}
|
|
|
|
|
|
|
|
Id EmitGetCbufU8(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
const Id load{GetCbuf(ctx, ctx.U8, &UniformDefinitions::U8, sizeof(u8), binding, offset)};
|
|
|
|
return ctx.OpUConvert(ctx.U32[1], load);
|
|
|
|
}
|
|
|
|
|
|
|
|
Id EmitGetCbufS8(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
const Id load{GetCbuf(ctx, ctx.S8, &UniformDefinitions::S8, sizeof(s8), binding, offset)};
|
|
|
|
return ctx.OpSConvert(ctx.U32[1], load);
|
|
|
|
}
|
|
|
|
|
|
|
|
Id EmitGetCbufU16(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
const Id load{GetCbuf(ctx, ctx.U16, &UniformDefinitions::U16, sizeof(u16), binding, offset)};
|
|
|
|
return ctx.OpUConvert(ctx.U32[1], load);
|
|
|
|
}
|
|
|
|
|
|
|
|
Id EmitGetCbufS16(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
const Id load{GetCbuf(ctx, ctx.S16, &UniformDefinitions::S16, sizeof(s16), binding, offset)};
|
|
|
|
return ctx.OpSConvert(ctx.U32[1], load);
|
|
|
|
}
|
|
|
|
|
|
|
|
Id EmitGetCbufU32(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
return GetCbuf(ctx, ctx.U32[1], &UniformDefinitions::U32, sizeof(u32), binding, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
Id EmitGetCbufF32(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
return GetCbuf(ctx, ctx.F32[1], &UniformDefinitions::F32, sizeof(f32), binding, offset);
|
|
|
|
}
|
|
|
|
|
2021-04-04 07:31:09 +02:00
|
|
|
Id EmitGetCbufU32x2(EmitContext& ctx, const IR::Value& binding, const IR::Value& offset) {
|
|
|
|
return GetCbuf(ctx, ctx.U32[2], &UniformDefinitions::U32x2, sizeof(u32[2]), binding, offset);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-03-19 23:28:31 +01:00
|
|
|
Id EmitGetAttribute(EmitContext& ctx, IR::Attribute attr) {
|
2021-03-24 05:33:45 +01:00
|
|
|
const u32 element{static_cast<u32>(attr) % 4};
|
|
|
|
const auto element_id{[&] { return ctx.Constant(ctx.U32[1], element); }};
|
|
|
|
if (IR::IsGeneric(attr)) {
|
|
|
|
const u32 index{IR::GenericAttributeIndex(attr)};
|
2021-03-27 08:59:58 +01:00
|
|
|
const std::optional<AttrInfo> type{AttrTypes(ctx, index)};
|
|
|
|
if (!type) {
|
|
|
|
// Attribute is disabled
|
|
|
|
return ctx.Constant(ctx.F32[1], 0.0f);
|
|
|
|
}
|
2021-03-24 05:33:45 +01:00
|
|
|
const Id generic_id{ctx.input_generics.at(index)};
|
2021-03-27 08:59:58 +01:00
|
|
|
const Id pointer{ctx.OpAccessChain(type->pointer, generic_id, element_id())};
|
|
|
|
const Id value{ctx.OpLoad(type->id, pointer)};
|
|
|
|
return type->needs_cast ? ctx.OpBitcast(ctx.F32[1], value) : value;
|
2021-03-24 05:33:45 +01:00
|
|
|
}
|
|
|
|
switch (attr) {
|
|
|
|
case IR::Attribute::PositionX:
|
|
|
|
case IR::Attribute::PositionY:
|
|
|
|
case IR::Attribute::PositionZ:
|
|
|
|
case IR::Attribute::PositionW:
|
|
|
|
return ctx.OpLoad(ctx.F32[1],
|
|
|
|
ctx.OpAccessChain(ctx.input_f32, ctx.input_position, element_id()));
|
|
|
|
case IR::Attribute::InstanceId:
|
|
|
|
if (ctx.profile.support_vertex_instance_id) {
|
|
|
|
return ctx.OpLoad(ctx.U32[1], ctx.instance_id);
|
|
|
|
} else {
|
2021-03-20 23:11:56 +01:00
|
|
|
return ctx.OpISub(ctx.U32[1], ctx.OpLoad(ctx.U32[1], ctx.instance_index),
|
|
|
|
ctx.OpLoad(ctx.U32[1], ctx.base_instance));
|
2021-03-24 05:33:45 +01:00
|
|
|
}
|
|
|
|
case IR::Attribute::VertexId:
|
|
|
|
if (ctx.profile.support_vertex_instance_id) {
|
|
|
|
return ctx.OpLoad(ctx.U32[1], ctx.vertex_id);
|
|
|
|
} else {
|
2021-03-20 23:11:56 +01:00
|
|
|
return ctx.OpISub(ctx.U32[1], ctx.OpLoad(ctx.U32[1], ctx.vertex_index),
|
|
|
|
ctx.OpLoad(ctx.U32[1], ctx.base_vertex));
|
|
|
|
}
|
2021-03-27 06:55:37 +01:00
|
|
|
case IR::Attribute::FrontFace:
|
|
|
|
return ctx.OpSelect(ctx.U32[1], ctx.OpLoad(ctx.U1, ctx.front_face),
|
|
|
|
ctx.Constant(ctx.U32[1], std::numeric_limits<u32>::max()),
|
|
|
|
ctx.u32_zero_value);
|
2021-03-29 20:05:38 +02:00
|
|
|
case IR::Attribute::PointSpriteS:
|
|
|
|
return ctx.OpLoad(ctx.F32[1], ctx.OpAccessChain(ctx.input_f32, ctx.point_coord,
|
|
|
|
ctx.Constant(ctx.U32[1], 0U)));
|
|
|
|
case IR::Attribute::PointSpriteT:
|
|
|
|
return ctx.OpLoad(ctx.F32[1], ctx.OpAccessChain(ctx.input_f32, ctx.point_coord,
|
|
|
|
ctx.Constant(ctx.U32[1], 1U)));
|
2021-03-24 05:33:45 +01:00
|
|
|
default:
|
|
|
|
throw NotImplementedException("Read attribute {}", attr);
|
2021-03-20 23:11:56 +01:00
|
|
|
}
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-03-19 23:28:31 +01:00
|
|
|
void EmitSetAttribute(EmitContext& ctx, IR::Attribute attr, Id value) {
|
2021-04-03 01:48:39 +02:00
|
|
|
const std::optional<Id> output{OutputAttrPointer(ctx, attr)};
|
2021-04-01 08:34:45 +02:00
|
|
|
if (!output) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ctx.OpStore(*output, value);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-04-04 06:47:14 +02:00
|
|
|
Id EmitGetAttributeIndexed(EmitContext& ctx, Id offset) {
|
|
|
|
return ctx.OpFunctionCall(ctx.F32[1], ctx.indexed_load_func, offset);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-04-04 06:47:14 +02:00
|
|
|
void EmitSetAttributeIndexed(EmitContext& ctx, Id offset, Id value) {
|
|
|
|
ctx.OpFunctionCall(ctx.void_id, ctx.indexed_store_func, offset, value);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-03-19 23:28:31 +01:00
|
|
|
void EmitSetFragColor(EmitContext& ctx, u32 index, u32 component, Id value) {
|
|
|
|
const Id component_id{ctx.Constant(ctx.U32[1], component)};
|
|
|
|
const Id pointer{ctx.OpAccessChain(ctx.output_f32, ctx.frag_color.at(index), component_id)};
|
|
|
|
ctx.OpStore(pointer, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EmitSetFragDepth(EmitContext& ctx, Id value) {
|
|
|
|
ctx.OpStore(ctx.frag_depth, value);
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetZFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetSFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetCFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitGetOFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetZFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetSFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetCFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
void EmitSetOFlag(EmitContext&) {
|
2021-02-08 06:54:35 +01:00
|
|
|
throw NotImplementedException("SPIR-V Instruction");
|
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
Id EmitWorkgroupId(EmitContext& ctx) {
|
2021-02-16 08:10:22 +01:00
|
|
|
return ctx.OpLoad(ctx.U32[3], ctx.workgroup_id);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-02-17 04:59:28 +01:00
|
|
|
Id EmitLocalInvocationId(EmitContext& ctx) {
|
2021-02-16 08:10:22 +01:00
|
|
|
return ctx.OpLoad(ctx.U32[3], ctx.local_invocation_id);
|
2021-02-08 06:54:35 +01:00
|
|
|
}
|
|
|
|
|
2021-04-02 23:05:47 +02:00
|
|
|
Id EmitLaneId(EmitContext& ctx) {
|
|
|
|
return ctx.OpLoad(ctx.U32[1], ctx.subgroup_local_invocation_id);
|
|
|
|
}
|
|
|
|
|
2021-03-29 00:53:34 +02:00
|
|
|
Id EmitLoadLocal(EmitContext& ctx, Id word_offset) {
|
|
|
|
const Id pointer{ctx.OpAccessChain(ctx.private_u32, ctx.local_memory, word_offset)};
|
|
|
|
return ctx.OpLoad(ctx.U32[1], pointer);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EmitWriteLocal(EmitContext& ctx, Id word_offset, Id value) {
|
|
|
|
const Id pointer{ctx.OpAccessChain(ctx.private_u32, ctx.local_memory, word_offset)};
|
|
|
|
ctx.OpStore(pointer, value);
|
|
|
|
}
|
|
|
|
|
2021-02-08 06:54:35 +01:00
|
|
|
} // namespace Shader::Backend::SPIRV
|