| 
									
										
										
										
											2021-01-09 03:30:07 -03: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-02 21:07:00 -03:00
										 |  |  | #include <span>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <boost/intrusive/list.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | #include "common/bit_cast.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/condition.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/value.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  | #include "shader_recompiler/object_pool.h"
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03: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 16:39:06 -03:00
										 |  |  |     explicit Block(ObjectPool<Inst>& inst_pool_); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03: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-02 21:07:00 -03:00
										 |  |  |     iterator PrependNewInst(iterator insertion_point, Opcode op, | 
					
						
							| 
									
										
										
										
											2021-02-14 22:46:40 -03:00
										 |  |  |                             std::initializer_list<Value> args = {}, u32 flags = 0); | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     /// Adds a new branch to this basic block.
 | 
					
						
							|  |  |  |     void AddBranch(Block* block); | 
					
						
							| 
									
										
										
										
											2021-04-20 22:28:06 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     /// Gets a mutable reference to the instruction list for this basic block.
 | 
					
						
							| 
									
										
										
										
											2021-04-20 22:28:06 -03:00
										 |  |  |     [[nodiscard]] InstructionList& Instructions() noexcept { | 
					
						
							|  |  |  |         return instructions; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     /// Gets an immutable reference to the instruction list for this basic block.
 | 
					
						
							| 
									
										
										
										
											2021-04-20 22:28:06 -03:00
										 |  |  |     [[nodiscard]] const InstructionList& Instructions() const noexcept { | 
					
						
							|  |  |  |         return instructions; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Gets an immutable span to the immediate predecessors.
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     [[nodiscard]] std::span<Block* const> ImmPredecessors() const noexcept { | 
					
						
							| 
									
										
										
										
											2021-04-20 22:28:06 -03:00
										 |  |  |         return imm_predecessors; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     /// Gets an immutable span to the immediate successors.
 | 
					
						
							|  |  |  |     [[nodiscard]] std::span<Block* const> ImmSuccessors() const noexcept { | 
					
						
							|  |  |  |         return imm_successors; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03: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 00:27:55 -03: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 04:58:23 -03:00
										 |  |  |     void SsaSeal() noexcept { | 
					
						
							|  |  |  |         is_ssa_sealed = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] bool IsSsaSealed() const noexcept { | 
					
						
							|  |  |  |         return is_ssa_sealed; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03: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(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  |     /// Memory pool for instruction list
 | 
					
						
							|  |  |  |     ObjectPool<Inst>* inst_pool; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  |     /// List of instructions in this block
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     InstructionList instructions; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  |     /// Block immediate predecessors
 | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |     std::vector<Block*> imm_predecessors; | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     /// Block immediate successors
 | 
					
						
							|  |  |  |     std::vector<Block*> imm_successors; | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-21 00:27:55 -03:00
										 |  |  |     /// Intrusively store the value of a register in the block.
 | 
					
						
							|  |  |  |     std::array<Value, NUM_REGS> ssa_reg_values; | 
					
						
							| 
									
										
										
										
											2021-04-21 04:58:23 -03:00
										 |  |  |     /// Intrusively store if the block is sealed in the SSA pass.
 | 
					
						
							|  |  |  |     bool is_ssa_sealed{false}; | 
					
						
							| 
									
										
										
										
											2021-04-21 00:27:55 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     /// Intrusively stored host definition of this block.
 | 
					
						
							|  |  |  |     u32 definition{}; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | using BlockList = std::vector<Block*>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03: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
 |