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 <initializer_list>
|
|
|
|
#include <map>
|
2021-02-03 00:07:00 +00:00
|
|
|
#include <span>
|
|
|
|
#include <vector>
|
2021-01-09 06:30:07 +00:00
|
|
|
|
|
|
|
#include <boost/intrusive/list.hpp>
|
|
|
|
|
2021-02-16 07:10:22 +00:00
|
|
|
#include "common/bit_cast.h"
|
2021-05-14 03:40:54 +00:00
|
|
|
#include "common/common_types.h"
|
2021-02-11 19:39:06 +00:00
|
|
|
#include "shader_recompiler/frontend/ir/condition.h"
|
|
|
|
#include "shader_recompiler/frontend/ir/value.h"
|
2021-02-06 02:11:23 +00:00
|
|
|
#include "shader_recompiler/object_pool.h"
|
2021-01-09 06:30:07 +00:00
|
|
|
|
|
|
|
namespace Shader::IR {
|
|
|
|
|
|
|
|
class Block {
|
|
|
|
public:
|
|
|
|
using InstructionList = boost::intrusive::list<Inst>;
|
|
|
|
using size_type = InstructionList::size_type;
|
|
|
|
using iterator = InstructionList::iterator;
|
|
|
|
using const_iterator = InstructionList::const_iterator;
|
|
|
|
using reverse_iterator = InstructionList::reverse_iterator;
|
|
|
|
using const_reverse_iterator = InstructionList::const_reverse_iterator;
|
|
|
|
|
2021-02-11 19:39:06 +00:00
|
|
|
explicit Block(ObjectPool<Inst>& inst_pool_);
|
2021-01-09 06:30:07 +00:00
|
|
|
~Block();
|
|
|
|
|
|
|
|
Block(const Block&) = delete;
|
|
|
|
Block& operator=(const Block&) = delete;
|
|
|
|
|
|
|
|
Block(Block&&) = default;
|
|
|
|
Block& operator=(Block&&) = default;
|
|
|
|
|
|
|
|
/// Appends a new instruction to the end of this basic block.
|
|
|
|
void AppendNewInst(Opcode op, std::initializer_list<Value> args);
|
|
|
|
|
|
|
|
/// Prepends a new instruction to this basic block before the insertion point.
|
2021-02-03 00:07:00 +00:00
|
|
|
iterator PrependNewInst(iterator insertion_point, Opcode op,
|
2021-02-15 01:46:40 +00:00
|
|
|
std::initializer_list<Value> args = {}, u32 flags = 0);
|
2021-02-03 00:07:00 +00:00
|
|
|
|
2021-05-14 03:40:54 +00:00
|
|
|
/// Adds a new branch to this basic block.
|
|
|
|
void AddBranch(Block* block);
|
2021-04-21 01:28:06 +00:00
|
|
|
|
2021-01-09 06:30:07 +00:00
|
|
|
/// Gets a mutable reference to the instruction list for this basic block.
|
2021-04-21 01:28:06 +00:00
|
|
|
[[nodiscard]] InstructionList& Instructions() noexcept {
|
|
|
|
return instructions;
|
|
|
|
}
|
2021-01-09 06:30:07 +00:00
|
|
|
/// Gets an immutable reference to the instruction list for this basic block.
|
2021-04-21 01:28:06 +00:00
|
|
|
[[nodiscard]] const InstructionList& Instructions() const noexcept {
|
|
|
|
return instructions;
|
|
|
|
}
|
2021-02-03 00:07:00 +00:00
|
|
|
|
|
|
|
/// Gets an immutable span to the immediate predecessors.
|
2021-05-14 03:40:54 +00:00
|
|
|
[[nodiscard]] std::span<Block* const> ImmPredecessors() const noexcept {
|
2021-04-21 01:28:06 +00:00
|
|
|
return imm_predecessors;
|
|
|
|
}
|
2021-05-14 03:40:54 +00:00
|
|
|
/// Gets an immutable span to the immediate successors.
|
|
|
|
[[nodiscard]] std::span<Block* const> ImmSuccessors() const noexcept {
|
|
|
|
return imm_successors;
|
|
|
|
}
|
2021-02-11 19:39:06 +00:00
|
|
|
|
2021-02-16 07:10:22 +00:00
|
|
|
/// Intrusively store the host definition of this instruction.
|
|
|
|
template <typename DefinitionType>
|
|
|
|
void SetDefinition(DefinitionType def) {
|
|
|
|
definition = Common::BitCast<u32>(def);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the intrusively stored host definition of this instruction.
|
|
|
|
template <typename DefinitionType>
|
|
|
|
[[nodiscard]] DefinitionType Definition() const noexcept {
|
|
|
|
return Common::BitCast<DefinitionType>(definition);
|
|
|
|
}
|
|
|
|
|
2021-04-21 03:27:55 +00:00
|
|
|
void SetSsaRegValue(IR::Reg reg, const Value& value) noexcept {
|
|
|
|
ssa_reg_values[RegIndex(reg)] = value;
|
|
|
|
}
|
|
|
|
const Value& SsaRegValue(IR::Reg reg) const noexcept {
|
|
|
|
return ssa_reg_values[RegIndex(reg)];
|
|
|
|
}
|
|
|
|
|
2021-04-21 07:58:23 +00:00
|
|
|
void SsaSeal() noexcept {
|
|
|
|
is_ssa_sealed = true;
|
|
|
|
}
|
|
|
|
[[nodiscard]] bool IsSsaSealed() const noexcept {
|
|
|
|
return is_ssa_sealed;
|
|
|
|
}
|
|
|
|
|
2021-01-09 06:30:07 +00:00
|
|
|
[[nodiscard]] bool empty() const {
|
|
|
|
return instructions.empty();
|
|
|
|
}
|
|
|
|
[[nodiscard]] size_type size() const {
|
|
|
|
return instructions.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] Inst& front() {
|
|
|
|
return instructions.front();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const Inst& front() const {
|
|
|
|
return instructions.front();
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] Inst& back() {
|
|
|
|
return instructions.back();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const Inst& back() const {
|
|
|
|
return instructions.back();
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] iterator begin() {
|
|
|
|
return instructions.begin();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const_iterator begin() const {
|
|
|
|
return instructions.begin();
|
|
|
|
}
|
|
|
|
[[nodiscard]] iterator end() {
|
|
|
|
return instructions.end();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const_iterator end() const {
|
|
|
|
return instructions.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] reverse_iterator rbegin() {
|
|
|
|
return instructions.rbegin();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const_reverse_iterator rbegin() const {
|
|
|
|
return instructions.rbegin();
|
|
|
|
}
|
|
|
|
[[nodiscard]] reverse_iterator rend() {
|
|
|
|
return instructions.rend();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const_reverse_iterator rend() const {
|
|
|
|
return instructions.rend();
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] const_iterator cbegin() const {
|
|
|
|
return instructions.cbegin();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const_iterator cend() const {
|
|
|
|
return instructions.cend();
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] const_reverse_iterator crbegin() const {
|
|
|
|
return instructions.crbegin();
|
|
|
|
}
|
|
|
|
[[nodiscard]] const_reverse_iterator crend() const {
|
|
|
|
return instructions.crend();
|
|
|
|
}
|
|
|
|
|
2021-10-31 16:46:46 +00:00
|
|
|
// Set the order of the block, it can be set pre order, the user decides
|
|
|
|
void SetOrder(u32 new_order) {
|
|
|
|
order = new_order;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the order of the block.
|
|
|
|
// The higher, the closer is the block to the end.
|
|
|
|
[[nodiscard]] u32 GetOrder() const {
|
|
|
|
return order;
|
|
|
|
}
|
|
|
|
|
2021-01-09 06:30:07 +00:00
|
|
|
private:
|
2021-02-06 02:11:23 +00:00
|
|
|
/// Memory pool for instruction list
|
|
|
|
ObjectPool<Inst>* inst_pool;
|
2021-01-09 06:30:07 +00:00
|
|
|
|
2021-02-03 00:07:00 +00:00
|
|
|
/// List of instructions in this block
|
2021-01-09 06:30:07 +00:00
|
|
|
InstructionList instructions;
|
|
|
|
|
2021-02-03 00:07:00 +00:00
|
|
|
/// Block immediate predecessors
|
2021-02-11 19:39:06 +00:00
|
|
|
std::vector<Block*> imm_predecessors;
|
2021-05-14 03:40:54 +00:00
|
|
|
/// Block immediate successors
|
|
|
|
std::vector<Block*> imm_successors;
|
2021-02-16 07:10:22 +00:00
|
|
|
|
2021-04-21 03:27:55 +00:00
|
|
|
/// Intrusively store the value of a register in the block.
|
|
|
|
std::array<Value, NUM_REGS> ssa_reg_values;
|
2021-04-21 07:58:23 +00:00
|
|
|
/// Intrusively store if the block is sealed in the SSA pass.
|
|
|
|
bool is_ssa_sealed{false};
|
2021-04-21 03:27:55 +00:00
|
|
|
|
2021-02-16 07:10:22 +00:00
|
|
|
/// Intrusively stored host definition of this block.
|
|
|
|
u32 definition{};
|
2021-10-31 16:46:46 +00:00
|
|
|
|
|
|
|
/// Order of the block.
|
|
|
|
u32 order{};
|
2021-01-09 06:30:07 +00:00
|
|
|
};
|
|
|
|
|
2021-02-11 19:39:06 +00:00
|
|
|
using BlockList = std::vector<Block*>;
|
|
|
|
|
2021-01-09 06:30:07 +00:00
|
|
|
[[nodiscard]] std::string DumpBlock(const Block& block);
|
|
|
|
|
|
|
|
[[nodiscard]] std::string DumpBlock(const Block& block,
|
|
|
|
const std::map<const Block*, size_t>& block_to_index,
|
|
|
|
std::map<const Inst*, size_t>& inst_to_index,
|
|
|
|
size_t& inst_index);
|
|
|
|
|
|
|
|
} // namespace Shader::IR
|