2021-01-09 06:30:07 +00:00
|
|
|
// Copyright 2021 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "common/common_types.h"
|
|
|
|
#include "shader_recompiler/exception.h"
|
|
|
|
#include "shader_recompiler/frontend/ir/attribute.h"
|
|
|
|
#include "shader_recompiler/frontend/ir/pred.h"
|
|
|
|
#include "shader_recompiler/frontend/ir/reg.h"
|
2021-04-16 02:46:11 +01:00
|
|
|
#include "shader_recompiler/frontend/ir/patch.h"
|
2021-01-09 06:30:07 +00:00
|
|
|
#include "shader_recompiler/frontend/ir/type.h"
|
|
|
|
|
|
|
|
namespace Shader::IR {
|
|
|
|
|
|
|
|
class Block;
|
|
|
|
class Inst;
|
|
|
|
|
|
|
|
class Value {
|
|
|
|
public:
|
|
|
|
Value() noexcept : type{IR::Type::Void}, inst{nullptr} {}
|
|
|
|
explicit Value(IR::Inst* value) noexcept;
|
|
|
|
explicit Value(IR::Block* value) noexcept;
|
|
|
|
explicit Value(IR::Reg value) noexcept;
|
|
|
|
explicit Value(IR::Pred value) noexcept;
|
|
|
|
explicit Value(IR::Attribute value) noexcept;
|
2021-04-16 02:46:11 +01:00
|
|
|
explicit Value(IR::Patch value) noexcept;
|
2021-01-09 06:30:07 +00:00
|
|
|
explicit Value(bool value) noexcept;
|
|
|
|
explicit Value(u8 value) noexcept;
|
|
|
|
explicit Value(u16 value) noexcept;
|
|
|
|
explicit Value(u32 value) noexcept;
|
2021-02-05 22:19:36 +00:00
|
|
|
explicit Value(f32 value) noexcept;
|
2021-01-09 06:30:07 +00:00
|
|
|
explicit Value(u64 value) noexcept;
|
2021-02-05 22:19:36 +00:00
|
|
|
explicit Value(f64 value) noexcept;
|
2021-01-09 06:30:07 +00:00
|
|
|
|
|
|
|
[[nodiscard]] bool IsIdentity() const noexcept;
|
2021-02-24 08:21:30 +00:00
|
|
|
[[nodiscard]] bool IsPhi() const noexcept;
|
2021-01-09 06:30:07 +00:00
|
|
|
[[nodiscard]] bool IsEmpty() const noexcept;
|
|
|
|
[[nodiscard]] bool IsImmediate() const noexcept;
|
|
|
|
[[nodiscard]] bool IsLabel() const noexcept;
|
|
|
|
[[nodiscard]] IR::Type Type() const noexcept;
|
|
|
|
|
|
|
|
[[nodiscard]] IR::Inst* Inst() const;
|
|
|
|
[[nodiscard]] IR::Block* Label() const;
|
|
|
|
[[nodiscard]] IR::Inst* InstRecursive() const;
|
2021-02-16 22:49:24 +00:00
|
|
|
[[nodiscard]] IR::Value Resolve() const;
|
2021-01-09 06:30:07 +00:00
|
|
|
[[nodiscard]] IR::Reg Reg() const;
|
|
|
|
[[nodiscard]] IR::Pred Pred() const;
|
|
|
|
[[nodiscard]] IR::Attribute Attribute() const;
|
2021-04-16 02:46:11 +01:00
|
|
|
[[nodiscard]] IR::Patch Patch() const;
|
2021-01-09 06:30:07 +00:00
|
|
|
[[nodiscard]] bool U1() const;
|
|
|
|
[[nodiscard]] u8 U8() const;
|
|
|
|
[[nodiscard]] u16 U16() const;
|
|
|
|
[[nodiscard]] u32 U32() const;
|
2021-02-05 22:19:36 +00:00
|
|
|
[[nodiscard]] f32 F32() const;
|
2021-01-09 06:30:07 +00:00
|
|
|
[[nodiscard]] u64 U64() const;
|
2021-03-18 06:53:57 +00:00
|
|
|
[[nodiscard]] f64 F64() const;
|
2021-01-09 06:30:07 +00:00
|
|
|
|
2021-02-03 00:07:00 +00:00
|
|
|
[[nodiscard]] bool operator==(const Value& other) const;
|
|
|
|
[[nodiscard]] bool operator!=(const Value& other) const;
|
|
|
|
|
2021-01-09 06:30:07 +00:00
|
|
|
private:
|
|
|
|
void ValidateAccess(IR::Type expected) const;
|
|
|
|
|
|
|
|
IR::Type type;
|
|
|
|
union {
|
|
|
|
IR::Inst* inst;
|
|
|
|
IR::Block* label;
|
|
|
|
IR::Reg reg;
|
|
|
|
IR::Pred pred;
|
|
|
|
IR::Attribute attribute;
|
2021-04-16 02:46:11 +01:00
|
|
|
IR::Patch patch;
|
2021-01-09 06:30:07 +00:00
|
|
|
bool imm_u1;
|
|
|
|
u8 imm_u8;
|
|
|
|
u16 imm_u16;
|
|
|
|
u32 imm_u32;
|
2021-02-05 22:19:36 +00:00
|
|
|
f32 imm_f32;
|
2021-01-09 06:30:07 +00:00
|
|
|
u64 imm_u64;
|
2021-02-05 22:19:36 +00:00
|
|
|
f64 imm_f64;
|
2021-01-09 06:30:07 +00:00
|
|
|
};
|
|
|
|
};
|
2021-03-08 21:31:53 +00:00
|
|
|
static_assert(std::is_trivially_copyable_v<Value>);
|
2021-01-09 06:30:07 +00:00
|
|
|
|
|
|
|
template <IR::Type type_>
|
|
|
|
class TypedValue : public Value {
|
|
|
|
public:
|
|
|
|
TypedValue() = default;
|
|
|
|
|
|
|
|
template <IR::Type other_type>
|
|
|
|
requires((other_type & type_) != IR::Type::Void) explicit(false)
|
|
|
|
TypedValue(const TypedValue<other_type>& value)
|
|
|
|
: Value(value) {}
|
|
|
|
|
|
|
|
explicit TypedValue(const Value& value) : Value(value) {
|
|
|
|
if ((value.Type() & type_) == IR::Type::Void) {
|
|
|
|
throw InvalidArgument("Incompatible types {} and {}", type_, value.Type());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-06 03:25:22 +01:00
|
|
|
explicit TypedValue(IR::Inst* inst_) : TypedValue(Value(inst_)) {}
|
2021-01-09 06:30:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
using U1 = TypedValue<Type::U1>;
|
|
|
|
using U8 = TypedValue<Type::U8>;
|
|
|
|
using U16 = TypedValue<Type::U16>;
|
|
|
|
using U32 = TypedValue<Type::U32>;
|
|
|
|
using U64 = TypedValue<Type::U64>;
|
2021-02-05 22:19:36 +00:00
|
|
|
using F16 = TypedValue<Type::F16>;
|
|
|
|
using F32 = TypedValue<Type::F32>;
|
|
|
|
using F64 = TypedValue<Type::F64>;
|
2021-01-09 06:30:07 +00:00
|
|
|
using U32U64 = TypedValue<Type::U32 | Type::U64>;
|
2021-02-05 22:19:36 +00:00
|
|
|
using F32F64 = TypedValue<Type::F32 | Type::F64>;
|
2021-01-09 06:30:07 +00:00
|
|
|
using U16U32U64 = TypedValue<Type::U16 | Type::U32 | Type::U64>;
|
2021-02-05 22:19:36 +00:00
|
|
|
using F16F32F64 = TypedValue<Type::F16 | Type::F32 | Type::F64>;
|
2021-01-09 06:30:07 +00:00
|
|
|
using UAny = TypedValue<Type::U8 | Type::U16 | Type::U32 | Type::U64>;
|
|
|
|
|
|
|
|
} // namespace Shader::IR
|