| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | // Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | #include "common/bit_field.h"
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Tegra { | 
					
						
							|  |  |  | namespace Engines { | 
					
						
							|  |  |  | class Maxwell3D; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MacroInterpreter final { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit MacroInterpreter(Engines::Maxwell3D& maxwell3d); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Executes the macro code with the specified input parameters. | 
					
						
							| 
									
										
										
										
											2018-10-29 23:36:03 -04:00
										 |  |  |      * @param offset Offset to start execution at. | 
					
						
							|  |  |  |      * @param parameters The parameters of the macro. | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-10-29 23:36:03 -04:00
										 |  |  |     void Execute(u32 offset, std::vector<u32> parameters); | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     enum class Operation : u32 { | 
					
						
							|  |  |  |         ALU = 0, | 
					
						
							|  |  |  |         AddImmediate = 1, | 
					
						
							|  |  |  |         ExtractInsert = 2, | 
					
						
							|  |  |  |         ExtractShiftLeftImmediate = 3, | 
					
						
							|  |  |  |         ExtractShiftLeftRegister = 4, | 
					
						
							|  |  |  |         Read = 5, | 
					
						
							|  |  |  |         Unused = 6, // This operation doesn't seem to be a valid encoding.
 | 
					
						
							|  |  |  |         Branch = 7, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum class ALUOperation : u32 { | 
					
						
							|  |  |  |         Add = 0, | 
					
						
							|  |  |  |         AddWithCarry = 1, | 
					
						
							|  |  |  |         Subtract = 2, | 
					
						
							|  |  |  |         SubtractWithBorrow = 3, | 
					
						
							|  |  |  |         // Operations 4-7 don't seem to be valid encodings.
 | 
					
						
							|  |  |  |         Xor = 8, | 
					
						
							|  |  |  |         Or = 9, | 
					
						
							|  |  |  |         And = 10, | 
					
						
							|  |  |  |         AndNot = 11, | 
					
						
							|  |  |  |         Nand = 12 | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum class ResultOperation : u32 { | 
					
						
							|  |  |  |         IgnoreAndFetch = 0, | 
					
						
							|  |  |  |         Move = 1, | 
					
						
							|  |  |  |         MoveAndSetMethod = 2, | 
					
						
							|  |  |  |         FetchAndSend = 3, | 
					
						
							|  |  |  |         MoveAndSend = 4, | 
					
						
							|  |  |  |         FetchAndSetMethod = 5, | 
					
						
							|  |  |  |         MoveAndSetMethodFetchAndSend = 6, | 
					
						
							|  |  |  |         MoveAndSetMethodSend = 7 | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum class BranchCondition : u32 { | 
					
						
							|  |  |  |         Zero = 0, | 
					
						
							|  |  |  |         NotZero = 1, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     union Opcode { | 
					
						
							|  |  |  |         u32 raw; | 
					
						
							|  |  |  |         BitField<0, 3, Operation> operation; | 
					
						
							|  |  |  |         BitField<4, 3, ResultOperation> result_operation; | 
					
						
							|  |  |  |         BitField<4, 1, BranchCondition> branch_condition; | 
					
						
							|  |  |  |         BitField<5, 1, u32> | 
					
						
							|  |  |  |             branch_annul; // If set on a branch, then the branch doesn't have a delay slot.
 | 
					
						
							|  |  |  |         BitField<7, 1, u32> is_exit; | 
					
						
							|  |  |  |         BitField<8, 3, u32> dst; | 
					
						
							|  |  |  |         BitField<11, 3, u32> src_a; | 
					
						
							|  |  |  |         BitField<14, 3, u32> src_b; | 
					
						
							|  |  |  |         // The signed immediate overlaps the second source operand and the alu operation.
 | 
					
						
							|  |  |  |         BitField<14, 18, s32> immediate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         BitField<17, 5, ALUOperation> alu_operation; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Bitfield instructions data
 | 
					
						
							|  |  |  |         BitField<17, 5, u32> bf_src_bit; | 
					
						
							|  |  |  |         BitField<22, 5, u32> bf_size; | 
					
						
							|  |  |  |         BitField<27, 5, u32> bf_dst_bit; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         u32 GetBitfieldMask() const { | 
					
						
							|  |  |  |             return (1 << bf_size) - 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-30 20:09:49 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         s32 GetBranchTarget() const { | 
					
						
							|  |  |  |             return static_cast<s32>(immediate * sizeof(u32)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     union MethodAddress { | 
					
						
							|  |  |  |         u32 raw; | 
					
						
							|  |  |  |         BitField<0, 12, u32> address; | 
					
						
							|  |  |  |         BitField<12, 6, u32> increment; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Resets the execution engine state, zeroing registers, etc.
 | 
					
						
							|  |  |  |     void Reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Executes a single macro instruction located at the current program counter. Returns whether | 
					
						
							|  |  |  |      * the interpreter should keep running. | 
					
						
							| 
									
										
										
										
											2018-10-29 23:36:03 -04:00
										 |  |  |      * @param offset Offset to start execution at. | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  |      * @param is_delay_slot Whether the current step is being executed due to a delay slot in a | 
					
						
							|  |  |  |      * previous instruction. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-10-29 23:36:03 -04:00
										 |  |  |     bool Step(u32 offset, bool is_delay_slot); | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Calculates the result of an ALU operation. src_a OP src_b;
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:32:21 -05:00
										 |  |  |     u32 GetALUResult(ALUOperation operation, u32 src_a, u32 src_b); | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Performs the result operation on the input result and stores it in the specified register
 | 
					
						
							|  |  |  |     /// (if necessary).
 | 
					
						
							|  |  |  |     void ProcessResult(ResultOperation operation, u32 reg, u32 result); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Evaluates the branch condition and returns whether the branch should be taken or not.
 | 
					
						
							|  |  |  |     bool EvaluateBranchCondition(BranchCondition cond, u32 value) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Reads an opcode at the current program counter location.
 | 
					
						
							| 
									
										
										
										
											2018-10-29 23:36:03 -04:00
										 |  |  |     Opcode GetOpcode(u32 offset) const; | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Returns the specified register's value. Register 0 is hardcoded to always return 0.
 | 
					
						
							|  |  |  |     u32 GetRegister(u32 register_id) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Sets the register to the input value.
 | 
					
						
							|  |  |  |     void SetRegister(u32 register_id, u32 value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Sets the method address to use for the next Send instruction.
 | 
					
						
							|  |  |  |     void SetMethodAddress(u32 address); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Calls a GPU Engine method with the input parameter.
 | 
					
						
							|  |  |  |     void Send(u32 value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Reads a GPU register located at the method address.
 | 
					
						
							|  |  |  |     u32 Read(u32 method) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Returns the next parameter in the parameter queue.
 | 
					
						
							|  |  |  |     u32 FetchParameter(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Engines::Maxwell3D& maxwell3d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 pc; ///< Current program counter
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     std::optional<u32> | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  |         delayed_pc; ///< Program counter to execute at after the delay slot is executed.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     static constexpr std::size_t NumMacroRegisters = 8; | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// General purpose macro registers.
 | 
					
						
							|  |  |  |     std::array<u32, NumMacroRegisters> registers = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Method address to use for the next Send instruction.
 | 
					
						
							|  |  |  |     MethodAddress method_address = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Input parameters of the current macro.
 | 
					
						
							|  |  |  |     std::vector<u32> parameters; | 
					
						
							|  |  |  |     /// Index of the next parameter that will be fetched by the 'parm' instruction.
 | 
					
						
							|  |  |  |     u32 next_parameter_index = 0; | 
					
						
							| 
									
										
										
										
											2018-11-21 14:32:21 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool carry_flag{}; | 
					
						
							| 
									
										
										
										
											2018-03-28 15:14:47 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | } // namespace Tegra
 |