| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  | // Copyright 2019 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <list>
 | 
					
						
							|  |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2019-06-27 00:39:40 -04:00
										 |  |  | #include <set>
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  | #include <variant>
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "video_core/engines/shader_bytecode.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"
 | 
					
						
							|  |  |  | #include "video_core/shader/shader_ir.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace VideoCommon::Shader { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using Tegra::Shader::ConditionCode; | 
					
						
							|  |  |  | using Tegra::Shader::Pred; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | constexpr s32 exit_branch = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct Condition { | 
					
						
							|  |  |  |     Pred predicate{Pred::UnusedIndex}; | 
					
						
							|  |  |  |     ConditionCode cc{ConditionCode::T}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool IsUnconditional() const { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |         return predicate == Pred::UnusedIndex && cc == ConditionCode::T; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 20:15:40 -04:00
										 |  |  |     bool operator==(const Condition& other) const { | 
					
						
							|  |  |  |         return std::tie(predicate, cc) == std::tie(other.predicate, other.cc); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool operator!=(const Condition& other) const { | 
					
						
							|  |  |  |         return !operator==(other); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  | class SingleBranch { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     SingleBranch() = default; | 
					
						
							|  |  |  |     SingleBranch(Condition condition, s32 address, bool kill, bool is_sync, bool is_brk, | 
					
						
							|  |  |  |                  bool ignore) | 
					
						
							|  |  |  |         : condition{condition}, address{address}, kill{kill}, is_sync{is_sync}, is_brk{is_brk}, | 
					
						
							|  |  |  |           ignore{ignore} {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool operator==(const SingleBranch& b) const { | 
					
						
							|  |  |  |         return std::tie(condition, address, kill, is_sync, is_brk, ignore) == | 
					
						
							|  |  |  |                std::tie(b.condition, b.address, b.kill, b.is_sync, b.is_brk, b.ignore); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Condition condition{}; | 
					
						
							|  |  |  |     s32 address{exit_branch}; | 
					
						
							|  |  |  |     bool kill{}; | 
					
						
							|  |  |  |     bool is_sync{}; | 
					
						
							|  |  |  |     bool is_brk{}; | 
					
						
							|  |  |  |     bool ignore{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct CaseBranch { | 
					
						
							|  |  |  |     CaseBranch(u32 cmp_value, u32 address) : cmp_value{cmp_value}, address{address} {} | 
					
						
							|  |  |  |     u32 cmp_value; | 
					
						
							|  |  |  |     u32 address; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MultiBranch { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     MultiBranch(u32 gpr, std::vector<CaseBranch>& branches) | 
					
						
							|  |  |  |         : gpr{gpr}, branches{std::move(branches)} {} | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     u32 gpr{}; | 
					
						
							|  |  |  |     std::vector<CaseBranch> branches{}; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  | using BranchData = std::variant<SingleBranch, MultiBranch>; | 
					
						
							|  |  |  | using BlockBranchInfo = std::shared_ptr<BranchData>; | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  | bool BlockBranchInfoAreEqual(BlockBranchInfo first, BlockBranchInfo second); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ShaderBlock { | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  |     u32 start{}; | 
					
						
							|  |  |  |     u32 end{}; | 
					
						
							|  |  |  |     bool ignore_branch{}; | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |     BlockBranchInfo branch{}; | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     bool operator==(const ShaderBlock& sb) const { | 
					
						
							| 
									
										
										
										
											2019-09-23 22:55:25 -04:00
										 |  |  |         return std::tie(start, end, ignore_branch) == | 
					
						
							|  |  |  |                    std::tie(sb.start, sb.end, sb.ignore_branch) && | 
					
						
							|  |  |  |                BlockBranchInfoAreEqual(branch, sb.branch); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-07-16 11:59:57 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool operator!=(const ShaderBlock& sb) const { | 
					
						
							|  |  |  |         return !operator==(sb); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ShaderCharacteristics { | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  |     std::list<ShaderBlock> blocks{}; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     std::set<u32> labels{}; | 
					
						
							| 
									
										
										
										
											2019-06-25 20:40:38 -04:00
										 |  |  |     u32 start{}; | 
					
						
							|  |  |  |     u32 end{}; | 
					
						
							| 
									
										
										
										
											2019-09-20 21:12:06 -04:00
										 |  |  |     ASTManager manager{true, true}; | 
					
						
							| 
									
										
										
										
											2019-08-16 16:25:02 -04:00
										 |  |  |     CompilerSettings settings{}; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-15 18:05:50 -04:00
										 |  |  | std::unique_ptr<ShaderCharacteristics> ScanFlow(const ProgramCode& program_code, | 
					
						
							|  |  |  |                                                 std::size_t program_size, u32 start_address, | 
					
						
							| 
									
										
										
										
											2019-09-23 15:40:58 -04:00
										 |  |  |                                                 const CompilerSettings& settings, | 
					
						
							|  |  |  |                                                 ConstBufferLocker& locker); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:46:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | } // namespace VideoCommon::Shader
 |