2019-08-09 22:50:21 -04:00
|
|
|
// Copyright 2019 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/common_types.h"
|
|
|
|
#include "video_core/engines/shader_bytecode.h"
|
|
|
|
#include "video_core/shader/node_helper.h"
|
|
|
|
#include "video_core/shader/shader_ir.h"
|
|
|
|
|
|
|
|
namespace VideoCommon::Shader {
|
|
|
|
|
|
|
|
using Tegra::Shader::Instruction;
|
|
|
|
using Tegra::Shader::OpCode;
|
|
|
|
using Tegra::Shader::Pred;
|
2019-08-26 21:09:12 -04:00
|
|
|
using Tegra::Shader::ShuffleOperation;
|
2019-08-09 22:50:21 -04:00
|
|
|
using Tegra::Shader::VoteOperation;
|
|
|
|
|
|
|
|
namespace {
|
2019-11-02 22:44:13 -04:00
|
|
|
|
2019-08-09 22:50:21 -04:00
|
|
|
OperationCode GetOperationCode(VoteOperation vote_op) {
|
|
|
|
switch (vote_op) {
|
|
|
|
case VoteOperation::All:
|
|
|
|
return OperationCode::VoteAll;
|
|
|
|
case VoteOperation::Any:
|
|
|
|
return OperationCode::VoteAny;
|
|
|
|
case VoteOperation::Eq:
|
|
|
|
return OperationCode::VoteEqual;
|
|
|
|
default:
|
2020-12-07 00:41:47 -05:00
|
|
|
UNREACHABLE_MSG("Invalid vote operation={}", vote_op);
|
2019-08-09 22:50:21 -04:00
|
|
|
return OperationCode::VoteAll;
|
|
|
|
}
|
|
|
|
}
|
2019-11-02 22:44:13 -04:00
|
|
|
|
2019-08-09 22:50:21 -04:00
|
|
|
} // Anonymous namespace
|
|
|
|
|
|
|
|
u32 ShaderIR::DecodeWarp(NodeBlock& bb, u32 pc) {
|
|
|
|
const Instruction instr = {program_code[pc]};
|
|
|
|
const auto opcode = OpCode::Decode(instr);
|
|
|
|
|
2019-12-09 21:40:32 -05:00
|
|
|
// Signal the backend that this shader uses warp instructions.
|
|
|
|
uses_warps = true;
|
|
|
|
|
2019-08-09 22:50:21 -04:00
|
|
|
switch (opcode->get().GetId()) {
|
|
|
|
case OpCode::Id::VOTE: {
|
|
|
|
const Node value = GetPredicate(instr.vote.value, instr.vote.negate_value != 0);
|
|
|
|
const Node active = Operation(OperationCode::BallotThread, value);
|
|
|
|
const Node vote = Operation(GetOperationCode(instr.vote.operation), value);
|
|
|
|
SetRegister(bb, instr.gpr0, active);
|
|
|
|
SetPredicate(bb, instr.vote.dest_pred, vote);
|
|
|
|
break;
|
|
|
|
}
|
2019-08-26 21:09:12 -04:00
|
|
|
case OpCode::Id::SHFL: {
|
2019-11-02 22:44:13 -04:00
|
|
|
Node mask = instr.shfl.is_mask_imm ? Immediate(static_cast<u32>(instr.shfl.mask_imm))
|
|
|
|
: GetRegister(instr.gpr39);
|
|
|
|
Node index = instr.shfl.is_index_imm ? Immediate(static_cast<u32>(instr.shfl.index_imm))
|
|
|
|
: GetRegister(instr.gpr20);
|
2019-10-23 21:26:07 -04:00
|
|
|
|
2019-11-02 22:44:13 -04:00
|
|
|
Node thread_id = Operation(OperationCode::ThreadId);
|
|
|
|
Node clamp = Operation(OperationCode::IBitwiseAnd, mask, Immediate(0x1FU));
|
|
|
|
Node seg_mask = BitfieldExtract(mask, 8, 16);
|
|
|
|
|
|
|
|
Node neg_seg_mask = Operation(OperationCode::IBitwiseNot, seg_mask);
|
|
|
|
Node min_thread_id = Operation(OperationCode::IBitwiseAnd, thread_id, seg_mask);
|
|
|
|
Node max_thread_id = Operation(OperationCode::IBitwiseOr, min_thread_id,
|
|
|
|
Operation(OperationCode::IBitwiseAnd, clamp, neg_seg_mask));
|
2019-08-26 21:09:12 -04:00
|
|
|
|
2019-11-02 22:44:13 -04:00
|
|
|
Node src_thread_id = [instr, index, neg_seg_mask, min_thread_id, thread_id] {
|
2019-08-26 21:09:12 -04:00
|
|
|
switch (instr.shfl.operation) {
|
|
|
|
case ShuffleOperation::Idx:
|
2019-11-02 22:44:13 -04:00
|
|
|
return Operation(OperationCode::IBitwiseOr,
|
|
|
|
Operation(OperationCode::IBitwiseAnd, index, neg_seg_mask),
|
|
|
|
min_thread_id);
|
2019-08-26 21:09:12 -04:00
|
|
|
case ShuffleOperation::Down:
|
2019-11-02 22:44:13 -04:00
|
|
|
return Operation(OperationCode::IAdd, thread_id, index);
|
|
|
|
case ShuffleOperation::Up:
|
|
|
|
return Operation(OperationCode::IAdd, thread_id,
|
|
|
|
Operation(OperationCode::INegate, index));
|
2019-08-26 21:09:12 -04:00
|
|
|
case ShuffleOperation::Bfly:
|
2019-11-02 22:44:13 -04:00
|
|
|
return Operation(OperationCode::IBitwiseXor, thread_id, index);
|
2019-08-26 21:09:12 -04:00
|
|
|
}
|
2019-11-02 22:44:13 -04:00
|
|
|
UNREACHABLE();
|
|
|
|
return Immediate(0U);
|
2019-08-26 21:09:12 -04:00
|
|
|
}();
|
|
|
|
|
2019-11-02 22:44:13 -04:00
|
|
|
Node in_bounds = [instr, src_thread_id, min_thread_id, max_thread_id] {
|
|
|
|
if (instr.shfl.operation == ShuffleOperation::Up) {
|
|
|
|
return Operation(OperationCode::LogicalIGreaterEqual, src_thread_id, min_thread_id);
|
|
|
|
} else {
|
|
|
|
return Operation(OperationCode::LogicalILessEqual, src_thread_id, max_thread_id);
|
|
|
|
}
|
|
|
|
}();
|
|
|
|
|
|
|
|
SetPredicate(bb, instr.shfl.pred48, in_bounds);
|
2019-08-26 21:09:12 -04:00
|
|
|
SetRegister(
|
|
|
|
bb, instr.gpr0,
|
2019-11-02 22:44:13 -04:00
|
|
|
Operation(OperationCode::ShuffleIndexed, GetRegister(instr.gpr8), src_thread_id));
|
2019-08-26 21:09:12 -04:00
|
|
|
break;
|
|
|
|
}
|
2019-11-02 22:44:46 -04:00
|
|
|
case OpCode::Id::FSWZADD: {
|
|
|
|
UNIMPLEMENTED_IF(instr.fswzadd.ndv);
|
|
|
|
|
|
|
|
Node op_a = GetRegister(instr.gpr8);
|
|
|
|
Node op_b = GetRegister(instr.gpr20);
|
|
|
|
Node mask = Immediate(static_cast<u32>(instr.fswzadd.swizzle));
|
|
|
|
SetRegister(bb, instr.gpr0, Operation(OperationCode::FSwizzleAdd, op_a, op_b, mask));
|
|
|
|
break;
|
|
|
|
}
|
2019-08-09 22:50:21 -04:00
|
|
|
default:
|
|
|
|
UNIMPLEMENTED_MSG("Unhandled warp instruction: {}", opcode->get().GetName());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pc;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace VideoCommon::Shader
|