|
|
|
@ -171,6 +171,64 @@ std::map<IR::Attribute, IR::Attribute> GenerateLegacyToGenericMappings(
|
|
|
|
|
}
|
|
|
|
|
return mapping;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void EmitGeometryPassthrough(IR::IREmitter& ir, const IR::Program& program, const Shader::VaryingState &passthrough_mask, bool passthrough_position, std::optional<IR::Attribute> passthrough_layer_attr) {
|
|
|
|
|
for (u32 i = 0; i < program.output_vertices; i++) {
|
|
|
|
|
// Assign generics from input
|
|
|
|
|
for (u32 j = 0; j < 32; j++) {
|
|
|
|
|
if (!passthrough_mask.Generic(j)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const IR::Attribute attr = IR::Attribute::Generic0X + (j * 4);
|
|
|
|
|
ir.SetAttribute(attr + 0, ir.GetAttribute(attr + 0, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 1, ir.GetAttribute(attr + 1, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 2, ir.GetAttribute(attr + 2, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 3, ir.GetAttribute(attr + 3, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (passthrough_position) {
|
|
|
|
|
// Assign position from input
|
|
|
|
|
const IR::Attribute attr = IR::Attribute::PositionX;
|
|
|
|
|
ir.SetAttribute(attr + 0, ir.GetAttribute(attr + 0, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 1, ir.GetAttribute(attr + 1, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 2, ir.GetAttribute(attr + 2, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 3, ir.GetAttribute(attr + 3, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (passthrough_layer_attr) {
|
|
|
|
|
// Assign layer
|
|
|
|
|
ir.SetAttribute(IR::Attribute::Layer, ir.GetAttribute(*passthrough_layer_attr), ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Emit vertex
|
|
|
|
|
ir.EmitVertex(ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
ir.EndPrimitive(ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u32 GetOutputTopologyVertices(OutputTopology output_topology) {
|
|
|
|
|
switch (output_topology) {
|
|
|
|
|
case OutputTopology::PointList:
|
|
|
|
|
return 1;
|
|
|
|
|
case OutputTopology::LineStrip:
|
|
|
|
|
return 2;
|
|
|
|
|
default:
|
|
|
|
|
return 3;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void LowerGeometryPassthrough(const IR::Program& program, const HostTranslateInfo& host_info) {
|
|
|
|
|
for (IR::Block *const block : program.blocks) {
|
|
|
|
|
for (IR::Inst &inst : block->Instructions()) {
|
|
|
|
|
if (inst.GetOpcode() == IR::Opcode::Epilogue) {
|
|
|
|
|
IR::IREmitter ir{*block, IR::Block::InstructionList::s_iterator_to(inst)};
|
|
|
|
|
EmitGeometryPassthrough(ir, program, program.info.passthrough, program.info.passthrough.AnyComponent(IR::Attribute::PositionX), {});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // Anonymous namespace
|
|
|
|
|
|
|
|
|
|
IR::Program TranslateProgram(ObjectPool<IR::Inst>& inst_pool, ObjectPool<IR::Block>& block_pool,
|
|
|
|
@ -198,6 +256,11 @@ IR::Program TranslateProgram(ObjectPool<IR::Inst>& inst_pool, ObjectPool<IR::Blo
|
|
|
|
|
for (size_t i = 0; i < program.info.passthrough.mask.size(); ++i) {
|
|
|
|
|
program.info.passthrough.mask[i] = ((mask[i / 32] >> (i % 32)) & 1) == 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!host_info.support_geometry_shader_passthrough) {
|
|
|
|
|
program.output_vertices = GetOutputTopologyVertices(program.output_topology);
|
|
|
|
|
LowerGeometryPassthrough(program, host_info);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
@ -342,17 +405,8 @@ IR::Program GenerateGeometryPassthrough(ObjectPool<IR::Inst>& inst_pool,
|
|
|
|
|
IR::Program program;
|
|
|
|
|
program.stage = Stage::Geometry;
|
|
|
|
|
program.output_topology = output_topology;
|
|
|
|
|
switch (output_topology) {
|
|
|
|
|
case OutputTopology::PointList:
|
|
|
|
|
program.output_vertices = 1;
|
|
|
|
|
break;
|
|
|
|
|
case OutputTopology::LineStrip:
|
|
|
|
|
program.output_vertices = 2;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
program.output_vertices = 3;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
program.output_vertices = GetOutputTopologyVertices(output_topology);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
program.is_geometry_passthrough = false;
|
|
|
|
|
program.info.loads.mask = source_program.info.stores.mask;
|
|
|
|
@ -366,35 +420,7 @@ IR::Program GenerateGeometryPassthrough(ObjectPool<IR::Inst>& inst_pool,
|
|
|
|
|
node.data.block = current_block;
|
|
|
|
|
|
|
|
|
|
IR::IREmitter ir{*current_block};
|
|
|
|
|
for (u32 i = 0; i < program.output_vertices; i++) {
|
|
|
|
|
// Assign generics from input
|
|
|
|
|
for (u32 j = 0; j < 32; j++) {
|
|
|
|
|
if (!program.info.stores.Generic(j)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const IR::Attribute attr = IR::Attribute::Generic0X + (j * 4);
|
|
|
|
|
ir.SetAttribute(attr + 0, ir.GetAttribute(attr + 0, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 1, ir.GetAttribute(attr + 1, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 2, ir.GetAttribute(attr + 2, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 3, ir.GetAttribute(attr + 3, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Assign position from input
|
|
|
|
|
const IR::Attribute attr = IR::Attribute::PositionX;
|
|
|
|
|
ir.SetAttribute(attr + 0, ir.GetAttribute(attr + 0, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 1, ir.GetAttribute(attr + 1, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 2, ir.GetAttribute(attr + 2, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
ir.SetAttribute(attr + 3, ir.GetAttribute(attr + 3, ir.Imm32(i)), ir.Imm32(0));
|
|
|
|
|
|
|
|
|
|
// Assign layer
|
|
|
|
|
ir.SetAttribute(IR::Attribute::Layer, ir.GetAttribute(source_program.info.emulated_layer),
|
|
|
|
|
ir.Imm32(0));
|
|
|
|
|
|
|
|
|
|
// Emit vertex
|
|
|
|
|
ir.EmitVertex(ir.Imm32(0));
|
|
|
|
|
}
|
|
|
|
|
ir.EndPrimitive(ir.Imm32(0));
|
|
|
|
|
EmitGeometryPassthrough(ir, program, program.info.stores, true, source_program.info.emulated_layer);
|
|
|
|
|
|
|
|
|
|
IR::Block* return_block{block_pool.Create(inst_pool)};
|
|
|
|
|
IR::IREmitter{*return_block}.Epilogue();
|
|
|
|
|