| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <initializer_list>
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/bit_cast.h"
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/basic_block.h"
 | 
					
						
							|  |  |  | #include "shader_recompiler/frontend/ir/value.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::IR { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  | Block::Block(ObjectPool<Inst>& inst_pool_) : inst_pool{&inst_pool_} {} | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | Block::~Block() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Block::AppendNewInst(Opcode op, std::initializer_list<Value> args) { | 
					
						
							|  |  |  |     PrependNewInst(end(), op, args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-25 21:34:17 -03:00
										 |  |  | Block::iterator Block::PrependNewInst(iterator insertion_point, const Inst& base_inst) { | 
					
						
							|  |  |  |     Inst* const inst{inst_pool->Create(base_inst)}; | 
					
						
							|  |  |  |     return instructions.insert(insertion_point, *inst); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  | Block::iterator Block::PrependNewInst(iterator insertion_point, Opcode op, | 
					
						
							| 
									
										
										
										
											2021-02-14 22:46:40 -03:00
										 |  |  |                                       std::initializer_list<Value> args, u32 flags) { | 
					
						
							| 
									
										
										
										
											2021-02-05 23:11:23 -03:00
										 |  |  |     Inst* const inst{inst_pool->Create(op, flags)}; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     const auto result_it{instructions.insert(insertion_point, *inst)}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (inst->NumArgs() != args.size()) { | 
					
						
							|  |  |  |         throw InvalidArgument("Invalid number of arguments {} in {}", args.size(), op); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     std::ranges::for_each(args, [inst, index = size_t{0}](const Value& arg) mutable { | 
					
						
							|  |  |  |         inst->SetArg(index, arg); | 
					
						
							|  |  |  |         ++index; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     return result_it; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  | void Block::AddBranch(Block* block) { | 
					
						
							|  |  |  |     if (std::ranges::find(imm_successors, block) != imm_successors.end()) { | 
					
						
							|  |  |  |         throw LogicError("Successor already inserted"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (std::ranges::find(block->imm_predecessors, this) != block->imm_predecessors.end()) { | 
					
						
							|  |  |  |         throw LogicError("Predecessor already inserted"); | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     imm_successors.push_back(block); | 
					
						
							|  |  |  |     block->imm_predecessors.push_back(this); | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  | static std::string BlockToIndex(const std::map<const Block*, size_t>& block_to_index, | 
					
						
							|  |  |  |                                 Block* block) { | 
					
						
							|  |  |  |     if (const auto it{block_to_index.find(block)}; it != block_to_index.end()) { | 
					
						
							|  |  |  |         return fmt::format("{{Block ${}}}", it->second); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return fmt::format("$<unknown block {:016x}>", reinterpret_cast<u64>(block)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  | static size_t InstIndex(std::map<const Inst*, size_t>& inst_to_index, size_t& inst_index, | 
					
						
							|  |  |  |                         const Inst* inst) { | 
					
						
							|  |  |  |     const auto [it, is_inserted]{inst_to_index.emplace(inst, inst_index + 1)}; | 
					
						
							|  |  |  |     if (is_inserted) { | 
					
						
							|  |  |  |         ++inst_index; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return it->second; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  | static std::string ArgToIndex(std::map<const Inst*, size_t>& inst_to_index, size_t& inst_index, | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |                               const Value& arg) { | 
					
						
							|  |  |  |     if (arg.IsEmpty()) { | 
					
						
							|  |  |  |         return "<null>"; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-14 01:24:32 -03:00
										 |  |  |     if (!arg.IsImmediate() || arg.IsIdentity()) { | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |         return fmt::format("%{}", InstIndex(inst_to_index, inst_index, arg.Inst())); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     switch (arg.Type()) { | 
					
						
							|  |  |  |     case Type::U1: | 
					
						
							| 
									
										
										
										
											2021-02-06 04:47:53 -03:00
										 |  |  |         return fmt::format("#{}", arg.U1() ? "true" : "false"); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     case Type::U8: | 
					
						
							|  |  |  |         return fmt::format("#{}", arg.U8()); | 
					
						
							|  |  |  |     case Type::U16: | 
					
						
							|  |  |  |         return fmt::format("#{}", arg.U16()); | 
					
						
							|  |  |  |     case Type::U32: | 
					
						
							|  |  |  |         return fmt::format("#{}", arg.U32()); | 
					
						
							|  |  |  |     case Type::U64: | 
					
						
							|  |  |  |         return fmt::format("#{}", arg.U64()); | 
					
						
							| 
									
										
										
										
											2021-02-17 00:59:28 -03:00
										 |  |  |     case Type::F32: | 
					
						
							|  |  |  |         return fmt::format("#{}", arg.F32()); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     case Type::Reg: | 
					
						
							|  |  |  |         return fmt::format("{}", arg.Reg()); | 
					
						
							|  |  |  |     case Type::Pred: | 
					
						
							|  |  |  |         return fmt::format("{}", arg.Pred()); | 
					
						
							|  |  |  |     case Type::Attribute: | 
					
						
							|  |  |  |         return fmt::format("{}", arg.Attribute()); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return "<unknown immediate type>"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string DumpBlock(const Block& block) { | 
					
						
							|  |  |  |     size_t inst_index{0}; | 
					
						
							|  |  |  |     std::map<const Inst*, size_t> inst_to_index; | 
					
						
							|  |  |  |     return DumpBlock(block, {}, inst_to_index, inst_index); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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) { | 
					
						
							|  |  |  |     std::string ret{"Block"}; | 
					
						
							|  |  |  |     if (const auto it{block_to_index.find(&block)}; it != block_to_index.end()) { | 
					
						
							|  |  |  |         ret += fmt::format(" ${}", it->second); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |     ret += '\n'; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |     for (const Inst& inst : block) { | 
					
						
							| 
									
										
										
										
											2021-04-05 22:25:22 -04:00
										 |  |  |         const Opcode op{inst.GetOpcode()}; | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |         ret += fmt::format("[{:016x}] ", reinterpret_cast<u64>(&inst)); | 
					
						
							|  |  |  |         if (TypeOf(op) != Type::Void) { | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |             ret += fmt::format("%{:<5} = {}", InstIndex(inst_to_index, inst_index, &inst), op); | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             ret += fmt::format("         {}", op); // '%00000 = ' -> 1 + 5 + 3 = 9 spaces
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |         const size_t arg_count{inst.NumArgs()}; | 
					
						
							| 
									
										
										
										
											2021-02-06 02:38:22 -03:00
										 |  |  |         for (size_t arg_index = 0; arg_index < arg_count; ++arg_index) { | 
					
						
							|  |  |  |             const Value arg{inst.Arg(arg_index)}; | 
					
						
							| 
									
										
										
										
											2021-05-14 00:40:54 -03:00
										 |  |  |             const std::string arg_str{ArgToIndex(inst_to_index, inst_index, arg)}; | 
					
						
							| 
									
										
										
										
											2021-02-06 02:38:22 -03:00
										 |  |  |             ret += arg_index != 0 ? ", " : " "; | 
					
						
							|  |  |  |             if (op == Opcode::Phi) { | 
					
						
							| 
									
										
										
										
											2021-02-14 01:24:32 -03:00
										 |  |  |                 ret += fmt::format("[ {}, {} ]", arg_str, | 
					
						
							| 
									
										
										
										
											2021-02-06 02:38:22 -03:00
										 |  |  |                                    BlockToIndex(block_to_index, inst.PhiBlock(arg_index))); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 ret += arg_str; | 
					
						
							| 
									
										
										
										
											2021-02-02 21:07:00 -03:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-02-11 16:39:06 -03:00
										 |  |  |             if (op != Opcode::Phi) { | 
					
						
							|  |  |  |                 const Type actual_type{arg.Type()}; | 
					
						
							|  |  |  |                 const Type expected_type{ArgTypeOf(op, arg_index)}; | 
					
						
							|  |  |  |                 if (!AreTypesCompatible(actual_type, expected_type)) { | 
					
						
							|  |  |  |                     ret += fmt::format("<type error: {} != {}>", actual_type, expected_type); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-01-09 03:30:07 -03:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (TypeOf(op) != Type::Void) { | 
					
						
							|  |  |  |             ret += fmt::format(" (uses: {})\n", inst.UseCount()); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             ret += '\n'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Shader::IR
 |