2018-12-20 19:09:21 -03:00
|
|
|
// Copyright 2018 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2018-12-27 01:50:22 -03:00
|
|
|
#include <array>
|
2019-10-28 02:31:05 -03:00
|
|
|
#include <list>
|
2018-12-20 19:09:21 -03:00
|
|
|
#include <map>
|
2019-03-29 18:36:07 -03:00
|
|
|
#include <optional>
|
2018-12-20 19:09:21 -03:00
|
|
|
#include <set>
|
|
|
|
#include <tuple>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "common/common_types.h"
|
|
|
|
#include "video_core/engines/maxwell_3d.h"
|
|
|
|
#include "video_core/engines/shader_bytecode.h"
|
|
|
|
#include "video_core/engines/shader_header.h"
|
2019-06-28 22:59:43 -04:00
|
|
|
#include "video_core/shader/ast.h"
|
2019-08-16 16:25:02 -04:00
|
|
|
#include "video_core/shader/compiler_settings.h"
|
2020-04-24 01:44:14 -03:00
|
|
|
#include "video_core/shader/memory_util.h"
|
2019-06-29 01:44:07 -04:00
|
|
|
#include "video_core/shader/node.h"
|
2020-02-28 20:53:10 -03:00
|
|
|
#include "video_core/shader/registry.h"
|
2018-12-20 19:09:21 -03:00
|
|
|
|
|
|
|
namespace VideoCommon::Shader {
|
|
|
|
|
2019-06-25 11:10:45 -04:00
|
|
|
struct ShaderBlock;
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
constexpr u32 MAX_PROGRAM_LENGTH = 0x1000;
|
|
|
|
|
2020-04-16 01:34:45 -03:00
|
|
|
struct ConstBuffer {
|
2020-12-05 11:40:14 -05:00
|
|
|
constexpr explicit ConstBuffer(u32 max_offset_, bool is_indirect_)
|
|
|
|
: max_offset{max_offset_}, is_indirect{is_indirect_} {}
|
2019-01-15 01:07:57 -03:00
|
|
|
|
2020-04-16 01:34:45 -03:00
|
|
|
constexpr ConstBuffer() = default;
|
2019-01-15 01:07:57 -03:00
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
void MarkAsUsed(u64 offset) {
|
|
|
|
max_offset = std::max(max_offset, static_cast<u32>(offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
void MarkAsUsedIndirect() {
|
|
|
|
is_indirect = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsIndirect() const {
|
|
|
|
return is_indirect;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 GetSize() const {
|
2019-11-08 17:08:07 -03:00
|
|
|
return max_offset + static_cast<u32>(sizeof(float));
|
2018-12-20 19:09:21 -03:00
|
|
|
}
|
|
|
|
|
2019-01-15 01:07:57 -03:00
|
|
|
u32 GetMaxOffset() const {
|
|
|
|
return max_offset;
|
|
|
|
}
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
private:
|
2020-04-16 01:34:45 -03:00
|
|
|
u32 max_offset = 0;
|
|
|
|
bool is_indirect = false;
|
2018-12-20 19:09:21 -03:00
|
|
|
};
|
|
|
|
|
2019-02-07 00:05:41 -03:00
|
|
|
struct GlobalMemoryUsage {
|
|
|
|
bool is_read{};
|
|
|
|
bool is_written{};
|
|
|
|
};
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
class ShaderIR final {
|
|
|
|
public:
|
2020-12-05 11:40:14 -05:00
|
|
|
explicit ShaderIR(const ProgramCode& program_code_, u32 main_offset_,
|
|
|
|
CompilerSettings settings_, Registry& registry_);
|
2019-05-19 04:01:59 -04:00
|
|
|
~ShaderIR();
|
2018-12-20 19:09:21 -03:00
|
|
|
|
2019-01-30 02:09:40 -03:00
|
|
|
const std::map<u32, NodeBlock>& GetBasicBlocks() const {
|
2018-12-20 19:09:21 -03:00
|
|
|
return basic_blocks;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::set<u32>& GetRegisters() const {
|
|
|
|
return used_registers;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::set<Tegra::Shader::Pred>& GetPredicates() const {
|
|
|
|
return used_predicates;
|
|
|
|
}
|
|
|
|
|
2019-04-29 23:37:09 -03:00
|
|
|
const std::set<Tegra::Shader::Attribute::Index>& GetInputAttributes() const {
|
2018-12-20 19:09:21 -03:00
|
|
|
return used_input_attributes;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::set<Tegra::Shader::Attribute::Index>& GetOutputAttributes() const {
|
|
|
|
return used_output_attributes;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::map<u32, ConstBuffer>& GetConstantBuffers() const {
|
|
|
|
return used_cbufs;
|
|
|
|
}
|
|
|
|
|
2020-12-30 02:25:23 -03:00
|
|
|
const std::list<SamplerEntry>& GetSamplers() const {
|
2018-12-20 19:09:21 -03:00
|
|
|
return used_samplers;
|
|
|
|
}
|
|
|
|
|
2020-12-30 02:25:23 -03:00
|
|
|
const std::list<ImageEntry>& GetImages() const {
|
2019-04-27 02:07:18 -03:00
|
|
|
return used_images;
|
|
|
|
}
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
const std::array<bool, Tegra::Engines::Maxwell3D::Regs::NumClipDistances>& GetClipDistances()
|
|
|
|
const {
|
|
|
|
return used_clip_distances;
|
|
|
|
}
|
|
|
|
|
2019-02-07 00:05:41 -03:00
|
|
|
const std::map<GlobalMemoryBase, GlobalMemoryUsage>& GetGlobalMemory() const {
|
|
|
|
return used_global_memory;
|
2018-12-29 02:44:54 -03:00
|
|
|
}
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
std::size_t GetLength() const {
|
|
|
|
return static_cast<std::size_t>(coverage_end * sizeof(u64));
|
|
|
|
}
|
|
|
|
|
2019-07-07 20:36:42 -03:00
|
|
|
bool UsesLayer() const {
|
|
|
|
return uses_layer;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool UsesViewportIndex() const {
|
|
|
|
return uses_viewport_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool UsesPointSize() const {
|
|
|
|
return uses_point_size;
|
|
|
|
}
|
|
|
|
|
2019-09-15 14:25:07 -04:00
|
|
|
bool UsesInstanceId() const {
|
|
|
|
return uses_instance_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool UsesVertexId() const {
|
|
|
|
return uses_vertex_id;
|
|
|
|
}
|
|
|
|
|
2020-03-15 21:00:51 -03:00
|
|
|
bool UsesLegacyVaryings() const {
|
|
|
|
return uses_legacy_varyings;
|
|
|
|
}
|
|
|
|
|
2019-12-09 23:40:32 -03:00
|
|
|
bool UsesWarps() const {
|
|
|
|
return uses_warps;
|
|
|
|
}
|
|
|
|
|
2019-04-29 23:28:28 -03:00
|
|
|
bool HasPhysicalAttributes() const {
|
2019-04-30 19:46:49 -03:00
|
|
|
return uses_physical_attributes;
|
2019-04-29 23:28:28 -03:00
|
|
|
}
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
const Tegra::Shader::Header& GetHeader() const {
|
|
|
|
return header;
|
|
|
|
}
|
|
|
|
|
2019-08-16 16:25:02 -04:00
|
|
|
bool IsFlowStackDisabled() const {
|
|
|
|
return disable_flow_stack;
|
|
|
|
}
|
|
|
|
|
2019-06-29 01:44:07 -04:00
|
|
|
bool IsDecompiled() const {
|
|
|
|
return decompiled;
|
|
|
|
}
|
|
|
|
|
2019-08-25 15:32:00 -04:00
|
|
|
const ASTManager& GetASTManager() const {
|
|
|
|
return program_manager;
|
|
|
|
}
|
|
|
|
|
2019-06-29 01:44:07 -04:00
|
|
|
ASTNode GetASTProgram() const {
|
|
|
|
return program_manager.GetProgram();
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 GetASTNumVariables() const {
|
|
|
|
return program_manager.GetVariables();
|
2019-06-25 13:03:51 -04:00
|
|
|
}
|
|
|
|
|
2019-11-08 17:08:07 -03:00
|
|
|
u32 ConvertAddressToNvidiaSpace(u32 address) const {
|
|
|
|
return (address - main_offset) * static_cast<u32>(sizeof(Tegra::Shader::Instruction));
|
2019-07-05 14:13:14 -04:00
|
|
|
}
|
|
|
|
|
2019-06-29 01:44:07 -04:00
|
|
|
/// Returns a condition code evaluated from internal flags
|
|
|
|
Node GetConditionCode(Tegra::Shader::ConditionCode cc) const;
|
|
|
|
|
2020-01-04 14:40:57 -04:00
|
|
|
const Node& GetAmendNode(std::size_t index) const {
|
2019-12-30 13:54:53 -04:00
|
|
|
return amend_code[index];
|
|
|
|
}
|
|
|
|
|
2020-01-24 10:44:34 -04:00
|
|
|
u32 GetNumCustomVariables() const {
|
2020-01-07 14:53:46 -04:00
|
|
|
return num_custom_variables;
|
|
|
|
}
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
private:
|
2019-06-29 01:44:07 -04:00
|
|
|
friend class ASTDecoder;
|
2019-09-25 09:53:18 -04:00
|
|
|
|
|
|
|
struct SamplerInfo {
|
2020-04-16 02:07:42 -03:00
|
|
|
std::optional<Tegra::Shader::TextureType> type;
|
|
|
|
std::optional<bool> is_array;
|
|
|
|
std::optional<bool> is_shadow;
|
|
|
|
std::optional<bool> is_buffer;
|
|
|
|
|
|
|
|
constexpr bool IsComplete() const noexcept {
|
|
|
|
return type && is_array && is_shadow && is_buffer;
|
|
|
|
}
|
2019-09-25 09:53:18 -04:00
|
|
|
};
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
void Decode();
|
2020-01-03 16:16:29 -04:00
|
|
|
void PostDecode();
|
2018-12-20 19:09:21 -03:00
|
|
|
|
2019-01-30 02:09:40 -03:00
|
|
|
NodeBlock DecodeRange(u32 begin, u32 end);
|
2019-06-25 13:03:51 -04:00
|
|
|
void DecodeRangeInner(NodeBlock& bb, u32 begin, u32 end);
|
2019-06-25 11:10:45 -04:00
|
|
|
void InsertControlFlow(NodeBlock& bb, const ShaderBlock& block);
|
2018-12-20 19:09:21 -03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Decodes a single instruction from Tegra to IR.
|
|
|
|
* @param bb Basic block where the nodes will be written to.
|
|
|
|
* @param pc Program counter. Offset to decode.
|
|
|
|
* @return Next address to decode.
|
|
|
|
*/
|
2019-01-30 02:09:40 -03:00
|
|
|
u32 DecodeInstr(NodeBlock& bb, u32 pc);
|
|
|
|
|
|
|
|
u32 DecodeArithmetic(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeArithmeticImmediate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeBfe(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeBfi(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeShift(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeArithmeticInteger(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeArithmeticIntegerImmediate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeArithmeticHalf(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeArithmeticHalfImmediate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeFfma(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeHfma2(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeConversion(NodeBlock& bb, u32 pc);
|
2019-08-09 23:50:21 -03:00
|
|
|
u32 DecodeWarp(NodeBlock& bb, u32 pc);
|
2019-01-30 02:09:40 -03:00
|
|
|
u32 DecodeMemory(NodeBlock& bb, u32 pc);
|
2019-02-22 02:19:45 -03:00
|
|
|
u32 DecodeTexture(NodeBlock& bb, u32 pc);
|
2019-04-27 02:07:18 -03:00
|
|
|
u32 DecodeImage(NodeBlock& bb, u32 pc);
|
2019-01-30 02:09:40 -03:00
|
|
|
u32 DecodeFloatSetPredicate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeIntegerSetPredicate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeHalfSetPredicate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodePredicateSetRegister(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodePredicateSetPredicate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeRegisterSetPredicate(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeFloatSet(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeIntegerSet(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeHalfSet(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeVideo(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeXmad(NodeBlock& bb, u32 pc);
|
|
|
|
u32 DecodeOther(NodeBlock& bb, u32 pc);
|
2018-12-20 19:09:21 -03:00
|
|
|
|
2018-12-20 22:41:31 -03:00
|
|
|
/// Generates a node for a passed register.
|
|
|
|
Node GetRegister(Tegra::Shader::Register reg);
|
2020-01-07 14:53:46 -04:00
|
|
|
/// Generates a node for a custom variable
|
|
|
|
Node GetCustomVariable(u32 id);
|
2018-12-20 22:36:17 -03:00
|
|
|
/// Generates a node representing a 19-bit immediate value
|
|
|
|
Node GetImmediate19(Tegra::Shader::Instruction instr);
|
|
|
|
/// Generates a node representing a 32-bit immediate value
|
|
|
|
Node GetImmediate32(Tegra::Shader::Instruction instr);
|
2018-12-20 22:42:47 -03:00
|
|
|
/// Generates a node representing a constant buffer
|
|
|
|
Node GetConstBuffer(u64 index, u64 offset);
|
|
|
|
/// Generates a node representing a constant buffer with a variadic offset
|
|
|
|
Node GetConstBufferIndirect(u64 index, u64 offset, Node node);
|
2018-12-20 19:09:21 -03:00
|
|
|
/// Generates a node for a passed predicate. It can be optionally negated
|
|
|
|
Node GetPredicate(u64 pred, bool negated = false);
|
2018-12-20 22:36:17 -03:00
|
|
|
/// Generates a predicate node for an immediate true or false value
|
|
|
|
Node GetPredicate(bool immediate);
|
2019-01-28 07:52:02 -05:00
|
|
|
/// Generates a node representing an input attribute. Keeps track of used attributes.
|
2019-04-29 23:37:09 -03:00
|
|
|
Node GetInputAttribute(Tegra::Shader::Attribute::Index index, u64 element, Node buffer = {});
|
2019-04-30 18:12:30 -03:00
|
|
|
/// Generates a node representing a physical input attribute.
|
|
|
|
Node GetPhysicalInputAttribute(Tegra::Shader::Register physical_address, Node buffer = {});
|
2019-01-28 07:52:02 -05:00
|
|
|
/// Generates a node representing an output attribute. Keeps track of used attributes.
|
2018-12-20 22:45:34 -03:00
|
|
|
Node GetOutputAttribute(Tegra::Shader::Attribute::Index index, u64 element, Node buffer);
|
2018-12-20 22:49:59 -03:00
|
|
|
/// Generates a node representing an internal flag
|
2019-06-29 01:44:07 -04:00
|
|
|
Node GetInternalFlag(InternalFlag flag, bool negated = false) const;
|
2018-12-20 22:51:38 -03:00
|
|
|
/// Generates a node representing a local memory address
|
|
|
|
Node GetLocalMemory(Node address);
|
2019-07-30 00:21:46 -03:00
|
|
|
/// Generates a node representing a shared memory address
|
|
|
|
Node GetSharedMemory(Node address);
|
2019-07-16 10:31:17 -04:00
|
|
|
/// Generates a temporary, internally it uses a post-RZ register
|
|
|
|
Node GetTemporary(u32 id);
|
2018-12-20 22:51:38 -03:00
|
|
|
|
2018-12-20 22:53:43 -03:00
|
|
|
/// Sets a register. src value must be a number-evaluated node.
|
2019-01-30 02:09:40 -03:00
|
|
|
void SetRegister(NodeBlock& bb, Tegra::Shader::Register dest, Node src);
|
2018-12-20 22:53:43 -03:00
|
|
|
/// Sets a predicate. src value must be a bool-evaluated node
|
2019-01-30 02:09:40 -03:00
|
|
|
void SetPredicate(NodeBlock& bb, u64 dest, Node src);
|
2018-12-20 22:53:43 -03:00
|
|
|
/// Sets an internal flag. src value must be a bool-evaluated node
|
2019-01-30 02:09:40 -03:00
|
|
|
void SetInternalFlag(NodeBlock& bb, InternalFlag flag, Node value);
|
2019-07-30 00:21:46 -03:00
|
|
|
/// Sets a local memory address with a value.
|
2019-01-30 02:09:40 -03:00
|
|
|
void SetLocalMemory(NodeBlock& bb, Node address, Node value);
|
2019-07-30 00:21:46 -03:00
|
|
|
/// Sets a shared memory address with a value.
|
|
|
|
void SetSharedMemory(NodeBlock& bb, Node address, Node value);
|
2019-07-16 10:31:17 -04:00
|
|
|
/// Sets a temporary. Internally it uses a post-RZ register
|
|
|
|
void SetTemporary(NodeBlock& bb, u32 id, Node value);
|
2018-12-20 19:09:21 -03:00
|
|
|
|
2018-12-27 16:50:36 -03:00
|
|
|
/// Sets internal flags from a float
|
2019-01-30 02:09:40 -03:00
|
|
|
void SetInternalFlagsFromFloat(NodeBlock& bb, Node value, bool sets_cc = true);
|
2018-12-27 16:50:36 -03:00
|
|
|
/// Sets internal flags from an integer
|
2019-01-30 02:09:40 -03:00
|
|
|
void SetInternalFlagsFromInteger(NodeBlock& bb, Node value, bool sets_cc = true);
|
2018-12-27 16:50:36 -03:00
|
|
|
|
2018-12-20 22:56:08 -03:00
|
|
|
/// Conditionally absolute/negated float. Absolute is applied first
|
|
|
|
Node GetOperandAbsNegFloat(Node value, bool absolute, bool negate);
|
|
|
|
/// Conditionally saturates a float
|
|
|
|
Node GetSaturatedFloat(Node value, bool saturate = true);
|
|
|
|
|
2018-12-20 22:57:16 -03:00
|
|
|
/// Converts an integer to different sizes.
|
|
|
|
Node ConvertIntegerSize(Node value, Tegra::Shader::Register::Size size, bool is_signed);
|
|
|
|
/// Conditionally absolute/negated integer. Absolute is applied first
|
|
|
|
Node GetOperandAbsNegInteger(Node value, bool absolute, bool negate, bool is_signed);
|
|
|
|
|
2018-12-20 22:58:33 -03:00
|
|
|
/// Unpacks a half immediate from an instruction
|
|
|
|
Node UnpackHalfImmediate(Tegra::Shader::Instruction instr, bool has_negation);
|
2019-04-15 19:48:11 -03:00
|
|
|
/// Unpacks a binary value into a half float pair with a type format
|
|
|
|
Node UnpackHalfFloat(Node value, Tegra::Shader::HalfType type);
|
2018-12-20 22:58:33 -03:00
|
|
|
/// Merges a half pair into another value
|
|
|
|
Node HalfMerge(Node dest, Node src, Tegra::Shader::HalfMerge merge);
|
|
|
|
/// Conditionally absolute/negated half float pair. Absolute is applied first
|
|
|
|
Node GetOperandAbsNegHalf(Node value, bool absolute, bool negate);
|
2019-04-09 18:41:41 -03:00
|
|
|
/// Conditionally saturates a half float pair
|
|
|
|
Node GetSaturatedHalfFloat(Node value, bool saturate = true);
|
2018-12-20 22:58:33 -03:00
|
|
|
|
2020-04-06 13:09:19 +07:00
|
|
|
/// Get image component value by type and size
|
2020-04-06 13:46:55 +07:00
|
|
|
std::pair<Node, bool> GetComponentValue(Tegra::Texture::ComponentType component_type,
|
|
|
|
u32 component_size, Node original_value);
|
2020-04-06 13:09:19 +07:00
|
|
|
|
2018-12-20 23:01:03 -03:00
|
|
|
/// Returns a predicate comparing two floats
|
|
|
|
Node GetPredicateComparisonFloat(Tegra::Shader::PredCondition condition, Node op_a, Node op_b);
|
|
|
|
/// Returns a predicate comparing two integers
|
|
|
|
Node GetPredicateComparisonInteger(Tegra::Shader::PredCondition condition, bool is_signed,
|
|
|
|
Node op_a, Node op_b);
|
|
|
|
/// Returns a predicate comparing two half floats. meta consumes how both pairs will be compared
|
2019-04-15 19:48:11 -03:00
|
|
|
Node GetPredicateComparisonHalf(Tegra::Shader::PredCondition condition, Node op_a, Node op_b);
|
2018-12-20 23:01:03 -03:00
|
|
|
|
2018-12-20 23:40:54 -03:00
|
|
|
/// Returns a predicate combiner operation
|
|
|
|
OperationCode GetPredicateCombiner(Tegra::Shader::PredOperation operation);
|
|
|
|
|
2019-11-06 04:32:43 -03:00
|
|
|
/// Queries the missing sampler info from the execution context.
|
2020-06-04 23:03:49 -03:00
|
|
|
SamplerInfo GetSamplerInfo(SamplerInfo info,
|
|
|
|
std::optional<Tegra::Engines::SamplerDescriptor> sampler);
|
2019-11-06 04:32:43 -03:00
|
|
|
|
2020-04-16 02:07:42 -03:00
|
|
|
/// Accesses a texture sampler.
|
2020-12-30 02:25:23 -03:00
|
|
|
std::optional<SamplerEntry> GetSampler(Tegra::Shader::Sampler sampler, SamplerInfo info);
|
2018-12-21 01:27:47 -03:00
|
|
|
|
2019-11-06 04:32:43 -03:00
|
|
|
/// Accesses a texture sampler for a bindless texture.
|
2020-12-30 02:25:23 -03:00
|
|
|
std::optional<SamplerEntry> GetBindlessSampler(Tegra::Shader::Register reg, SamplerInfo info,
|
|
|
|
Node& index_var);
|
2019-03-26 17:05:23 -04:00
|
|
|
|
2019-04-27 02:07:18 -03:00
|
|
|
/// Accesses an image.
|
2020-12-30 02:25:23 -03:00
|
|
|
ImageEntry& GetImage(Tegra::Shader::Image image, Tegra::Shader::ImageType type);
|
2019-04-27 02:07:18 -03:00
|
|
|
|
2019-04-27 03:04:13 -03:00
|
|
|
/// Access a bindless image sampler.
|
2020-12-30 02:25:23 -03:00
|
|
|
ImageEntry& GetBindlessImage(Tegra::Shader::Register reg, Tegra::Shader::ImageType type);
|
2019-07-17 21:03:53 -03:00
|
|
|
|
2018-12-26 02:58:47 -03:00
|
|
|
/// Extracts a sequence of bits from a node
|
|
|
|
Node BitfieldExtract(Node value, u32 offset, u32 bits);
|
|
|
|
|
2019-07-11 21:14:44 -03:00
|
|
|
/// Inserts a sequence of bits from a node
|
|
|
|
Node BitfieldInsert(Node base, Node insert, u32 offset, u32 bits);
|
|
|
|
|
2020-03-15 21:00:51 -03:00
|
|
|
/// Marks the usage of a input or output attribute.
|
|
|
|
void MarkAttributeUsage(Tegra::Shader::Attribute::Index index, u64 element);
|
|
|
|
|
2020-03-28 18:49:50 -03:00
|
|
|
/// Decodes VMNMX instruction and inserts its code into the passed basic block.
|
|
|
|
void DecodeVMNMX(NodeBlock& bb, Tegra::Shader::Instruction instr);
|
|
|
|
|
2019-01-30 02:09:40 -03:00
|
|
|
void WriteTexInstructionFloat(NodeBlock& bb, Tegra::Shader::Instruction instr,
|
2019-10-30 21:14:57 -04:00
|
|
|
const Node4& components);
|
2018-12-21 01:27:47 -03:00
|
|
|
|
2019-01-30 02:09:40 -03:00
|
|
|
void WriteTexsInstructionFloat(NodeBlock& bb, Tegra::Shader::Instruction instr,
|
2019-10-22 10:59:07 -04:00
|
|
|
const Node4& components, bool ignore_mask = false);
|
2019-01-30 02:09:40 -03:00
|
|
|
void WriteTexsInstructionHalfFloat(NodeBlock& bb, Tegra::Shader::Instruction instr,
|
2019-12-11 18:12:29 -04:00
|
|
|
const Node4& components, bool ignore_mask = false);
|
2018-12-13 16:59:28 -03:00
|
|
|
|
2018-12-27 01:50:22 -03:00
|
|
|
Node4 GetTexCode(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
2018-12-21 01:27:47 -03:00
|
|
|
Tegra::Shader::TextureProcessMode process_mode, bool depth_compare,
|
2019-03-27 07:11:50 -04:00
|
|
|
bool is_array, bool is_aoffi,
|
|
|
|
std::optional<Tegra::Shader::Register> bindless_reg);
|
2018-12-21 01:27:47 -03:00
|
|
|
|
2018-12-27 01:50:22 -03:00
|
|
|
Node4 GetTexsCode(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
|
|
|
Tegra::Shader::TextureProcessMode process_mode, bool depth_compare,
|
|
|
|
bool is_array);
|
2018-12-23 00:38:01 -03:00
|
|
|
|
2018-12-27 01:50:22 -03:00
|
|
|
Node4 GetTld4Code(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
2019-12-16 04:09:24 -03:00
|
|
|
bool depth_compare, bool is_array, bool is_aoffi, bool is_ptp,
|
|
|
|
bool is_bindless);
|
2018-12-27 01:50:22 -03:00
|
|
|
|
2019-04-27 20:50:35 -03:00
|
|
|
Node4 GetTldCode(Tegra::Shader::Instruction instr);
|
|
|
|
|
2018-12-27 01:50:22 -03:00
|
|
|
Node4 GetTldsCode(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
|
|
|
bool is_array);
|
2018-12-23 01:18:33 -03:00
|
|
|
|
2018-12-21 01:27:47 -03:00
|
|
|
std::tuple<std::size_t, std::size_t> ValidateAndGetCoordinateElement(
|
|
|
|
Tegra::Shader::TextureType texture_type, bool depth_compare, bool is_array,
|
|
|
|
bool lod_bias_enabled, std::size_t max_coords, std::size_t max_inputs);
|
|
|
|
|
2019-03-29 18:36:54 -03:00
|
|
|
std::vector<Node> GetAoffiCoordinates(Node aoffi_reg, std::size_t coord_count, bool is_tld4);
|
|
|
|
|
2019-12-16 04:09:24 -03:00
|
|
|
std::vector<Node> GetPtpCoordinates(std::array<Node, 2> ptp_regs);
|
|
|
|
|
2018-12-27 01:50:22 -03:00
|
|
|
Node4 GetTextureCode(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
2019-01-29 04:31:40 -03:00
|
|
|
Tegra::Shader::TextureProcessMode process_mode, std::vector<Node> coords,
|
2019-03-27 07:11:50 -04:00
|
|
|
Node array, Node depth_compare, u32 bias_offset, std::vector<Node> aoffi,
|
|
|
|
std::optional<Tegra::Shader::Register> bindless_reg);
|
2018-12-21 01:27:47 -03:00
|
|
|
|
2018-12-24 01:23:00 -03:00
|
|
|
Node GetVideoOperand(Node op, bool is_chunk, bool is_signed, Tegra::Shader::VideoType type,
|
|
|
|
u64 byte_height);
|
|
|
|
|
2019-01-30 02:09:40 -03:00
|
|
|
void WriteLogicOperation(NodeBlock& bb, Tegra::Shader::Register dest,
|
2018-12-21 02:02:15 -03:00
|
|
|
Tegra::Shader::LogicOperation logic_op, Node op_a, Node op_b,
|
|
|
|
Tegra::Shader::PredicateResultMode predicate_mode,
|
2018-12-27 16:50:36 -03:00
|
|
|
Tegra::Shader::Pred predicate, bool sets_cc);
|
2019-01-30 02:09:40 -03:00
|
|
|
void WriteLop3Instruction(NodeBlock& bb, Tegra::Shader::Register dest, Node op_a, Node op_b,
|
2018-12-27 16:50:36 -03:00
|
|
|
Node op_c, Node imm_lut, bool sets_cc);
|
2018-12-21 02:02:15 -03:00
|
|
|
|
2019-05-31 19:14:34 -03:00
|
|
|
std::tuple<Node, u32, u32> TrackCbuf(Node tracked, const NodeBlock& code, s64 cursor) const;
|
2018-12-29 02:44:54 -03:00
|
|
|
|
2020-06-04 23:02:55 -03:00
|
|
|
std::pair<Node, TrackSampler> TrackBindlessSampler(Node tracked, const NodeBlock& code,
|
|
|
|
s64 cursor);
|
|
|
|
|
|
|
|
std::pair<Node, TrackSampler> HandleBindlessIndirectRead(const CbufNode& cbuf,
|
|
|
|
const OperationNode& operation,
|
|
|
|
Node gpr, Node base_offset,
|
|
|
|
Node tracked, const NodeBlock& code,
|
|
|
|
s64 cursor);
|
2020-01-05 15:23:24 -04:00
|
|
|
|
2019-05-19 04:10:32 -04:00
|
|
|
std::optional<u32> TrackImmediate(Node tracked, const NodeBlock& code, s64 cursor) const;
|
2019-03-29 18:36:07 -03:00
|
|
|
|
2019-05-19 04:10:32 -04:00
|
|
|
std::pair<Node, s64> TrackRegister(const GprNode* tracked, const NodeBlock& code,
|
|
|
|
s64 cursor) const;
|
2018-12-29 02:44:54 -03:00
|
|
|
|
2019-10-18 01:23:10 -03:00
|
|
|
std::tuple<Node, Node, GlobalMemoryBase> TrackGlobalMemory(NodeBlock& bb,
|
|
|
|
Tegra::Shader::Instruction instr,
|
2020-01-09 01:08:55 -03:00
|
|
|
bool is_read, bool is_write);
|
2019-02-07 00:05:41 -03:00
|
|
|
|
2020-01-04 14:40:57 -04:00
|
|
|
/// Register new amending code and obtain the reference id.
|
|
|
|
std::size_t DeclareAmend(Node new_amend);
|
2019-12-30 13:54:53 -04:00
|
|
|
|
2020-01-07 14:53:46 -04:00
|
|
|
u32 NewCustomVariable();
|
|
|
|
|
2018-12-20 19:09:21 -03:00
|
|
|
const ProgramCode& program_code;
|
|
|
|
const u32 main_offset;
|
2019-09-24 23:34:18 -03:00
|
|
|
const CompilerSettings settings;
|
2020-02-28 20:53:10 -03:00
|
|
|
Registry& registry;
|
2019-09-24 23:34:18 -03:00
|
|
|
|
2019-06-29 01:44:07 -04:00
|
|
|
bool decompiled{};
|
2019-08-16 16:25:02 -04:00
|
|
|
bool disable_flow_stack{};
|
2018-12-20 19:09:21 -03:00
|
|
|
|
|
|
|
u32 coverage_begin{};
|
|
|
|
u32 coverage_end{};
|
|
|
|
|
2019-01-30 02:09:40 -03:00
|
|
|
std::map<u32, NodeBlock> basic_blocks;
|
|
|
|
NodeBlock global_code;
|
2019-09-24 23:34:18 -03:00
|
|
|
ASTManager program_manager{true, true};
|
2020-01-04 14:40:57 -04:00
|
|
|
std::vector<Node> amend_code;
|
2020-01-07 14:53:46 -04:00
|
|
|
u32 num_custom_variables{};
|
2018-12-20 19:09:21 -03:00
|
|
|
|
|
|
|
std::set<u32> used_registers;
|
|
|
|
std::set<Tegra::Shader::Pred> used_predicates;
|
2019-04-29 23:37:09 -03:00
|
|
|
std::set<Tegra::Shader::Attribute::Index> used_input_attributes;
|
2018-12-20 19:09:21 -03:00
|
|
|
std::set<Tegra::Shader::Attribute::Index> used_output_attributes;
|
|
|
|
std::map<u32, ConstBuffer> used_cbufs;
|
2020-12-30 02:25:23 -03:00
|
|
|
std::list<SamplerEntry> used_samplers;
|
|
|
|
std::list<ImageEntry> used_images;
|
2018-12-20 19:09:21 -03:00
|
|
|
std::array<bool, Tegra::Engines::Maxwell3D::Regs::NumClipDistances> used_clip_distances{};
|
2019-02-07 00:05:41 -03:00
|
|
|
std::map<GlobalMemoryBase, GlobalMemoryUsage> used_global_memory;
|
2019-07-07 20:36:42 -03:00
|
|
|
bool uses_layer{};
|
|
|
|
bool uses_viewport_index{};
|
|
|
|
bool uses_point_size{};
|
2019-04-30 19:46:49 -03:00
|
|
|
bool uses_physical_attributes{}; // Shader uses AL2P or physical attribute read/writes
|
2019-09-15 14:25:07 -04:00
|
|
|
bool uses_instance_id{};
|
|
|
|
bool uses_vertex_id{};
|
2020-03-15 21:00:51 -03:00
|
|
|
bool uses_legacy_varyings{};
|
2019-12-09 23:40:32 -03:00
|
|
|
bool uses_warps{};
|
2020-01-05 18:36:21 -04:00
|
|
|
bool uses_indexed_samplers{};
|
2018-12-20 19:09:21 -03:00
|
|
|
|
|
|
|
Tegra::Shader::Header header;
|
|
|
|
};
|
|
|
|
|
2018-12-21 03:22:26 -03:00
|
|
|
} // namespace VideoCommon::Shader
|