| 
									
										
										
										
											2018-02-11 21:34:20 -05:00
										 |  |  | // Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-12 12:34:41 -05:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-11 21:34:20 -05:00
										 |  |  | #include "video_core/engines/maxwell_3d.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Tegra { | 
					
						
							|  |  |  | namespace Engines { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 20:32:44 -05:00
										 |  |  | const std::unordered_map<u32, Maxwell3D::MethodInfo> Maxwell3D::method_handlers = { | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  |     {0xE24, {"SetShader", 5, &Maxwell3D::SetShader}}, | 
					
						
							| 
									
										
										
										
											2018-03-16 20:32:44 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-12 12:34:41 -05:00
										 |  |  | Maxwell3D::Maxwell3D(MemoryManager& memory_manager) : memory_manager(memory_manager) {} | 
					
						
							| 
									
										
										
										
											2018-02-11 21:34:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 20:32:44 -05:00
										 |  |  | void Maxwell3D::CallMethod(u32 method, const std::vector<u32>& parameters) { | 
					
						
							| 
									
										
										
										
											2018-03-17 13:55:42 -05:00
										 |  |  |     // TODO(Subv): Write an interpreter for the macros uploaded via registers 0x45 and 0x47
 | 
					
						
							| 
									
										
										
										
											2018-03-16 20:32:44 -05:00
										 |  |  |     auto itr = method_handlers.find(method); | 
					
						
							|  |  |  |     if (itr == method_handlers.end()) { | 
					
						
							|  |  |  |         LOG_ERROR(HW_GPU, "Unhandled method call %08X", method); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASSERT(itr->second.arguments == parameters.size()); | 
					
						
							|  |  |  |     (this->*itr->second.handler)(parameters); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-12 12:34:41 -05:00
										 |  |  | void Maxwell3D::WriteReg(u32 method, u32 value) { | 
					
						
							|  |  |  |     ASSERT_MSG(method < Regs::NUM_REGS, | 
					
						
							|  |  |  |                "Invalid Maxwell3D register, increase the size of the Regs structure"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     regs.reg_array[method] = value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MAXWELL3D_REG_INDEX(field_name) (offsetof(Regs, field_name) / sizeof(u32))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (method) { | 
					
						
							| 
									
										
										
										
											2018-03-16 19:24:41 -05:00
										 |  |  |     case MAXWELL3D_REG_INDEX(code_address.code_address_high): | 
					
						
							|  |  |  |     case MAXWELL3D_REG_INDEX(code_address.code_address_low): { | 
					
						
							|  |  |  |         // Note: For some reason games (like Puyo Puyo Tetris) seem to write 0 to the CODE_ADDRESS
 | 
					
						
							|  |  |  |         // register, we do not currently know if that's intended or a bug, so we assert it lest
 | 
					
						
							|  |  |  |         // stuff breaks in other places (like the shader address calculation).
 | 
					
						
							|  |  |  |         ASSERT_MSG(regs.code_address.CodeAddress() == 0, "Unexpected CODE_ADDRESS register value."); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-03-04 19:13:15 -05:00
										 |  |  |     case MAXWELL3D_REG_INDEX(draw.vertex_end_gl): { | 
					
						
							|  |  |  |         DrawArrays(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-12 12:34:41 -05:00
										 |  |  |     case MAXWELL3D_REG_INDEX(query.query_get): { | 
					
						
							|  |  |  |         ProcessQueryGet(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef MAXWELL3D_REG_INDEX
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Maxwell3D::ProcessQueryGet() { | 
					
						
							|  |  |  |     GPUVAddr sequence_address = regs.query.QueryAddress(); | 
					
						
							|  |  |  |     // Since the sequence address is given as a GPU VAddr, we have to convert it to an application
 | 
					
						
							|  |  |  |     // VAddr before writing.
 | 
					
						
							|  |  |  |     VAddr address = memory_manager.PhysicalToVirtualAddress(sequence_address); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (regs.query.query_get.mode) { | 
					
						
							|  |  |  |     case Regs::QueryMode::Write: { | 
					
						
							|  |  |  |         // Write the current query sequence to the sequence address.
 | 
					
						
							|  |  |  |         u32 sequence = regs.query.query_sequence; | 
					
						
							|  |  |  |         Memory::Write32(address, sequence); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         UNIMPLEMENTED_MSG("Query mode %u not implemented", regs.query.query_get.mode.Value()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-04 19:13:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | void Maxwell3D::DrawArrays() { | 
					
						
							|  |  |  |     LOG_WARNING(HW_GPU, "Game requested a DrawArrays, ignoring"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  | void Maxwell3D::SetShader(const std::vector<u32>& parameters) { | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Parameters description: | 
					
						
							|  |  |  |      * [0] = Shader Program. | 
					
						
							| 
									
										
										
										
											2018-03-17 16:17:45 -05:00
										 |  |  |      * [1] = Unknown, presumably the shader id. | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  |      * [2] = Offset to the start of the shader, after the 0x30 bytes header. | 
					
						
							|  |  |  |      * [3] = Shader Type. | 
					
						
							| 
									
										
										
										
											2018-03-17 13:55:42 -05:00
										 |  |  |      * [4] = Const Buffer Address >> 8. | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  |      */ | 
					
						
							|  |  |  |     auto shader_program = static_cast<Regs::ShaderProgram>(parameters[0]); | 
					
						
							|  |  |  |     // TODO(Subv): This address is probably an offset from the CODE_ADDRESS register.
 | 
					
						
							| 
									
										
										
										
											2018-03-17 13:55:42 -05:00
										 |  |  |     GPUVAddr address = parameters[2]; | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  |     auto shader_type = static_cast<Regs::ShaderType>(parameters[3]); | 
					
						
							| 
									
										
										
										
											2018-03-17 13:55:42 -05:00
										 |  |  |     GPUVAddr cb_address = parameters[4] << 8; | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto& shader = state.shaders[static_cast<size_t>(shader_program)]; | 
					
						
							|  |  |  |     shader.program = shader_program; | 
					
						
							|  |  |  |     shader.type = shader_type; | 
					
						
							| 
									
										
										
										
											2018-03-17 13:55:42 -05:00
										 |  |  |     shader.address = address; | 
					
						
							|  |  |  |     shader.cb_address = cb_address; | 
					
						
							| 
									
										
										
										
											2018-03-17 16:17:45 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Perform the same operations as the real macro code.
 | 
					
						
							|  |  |  |     // TODO(Subv): Early exit if register 0xD1C + shader_program contains the same as params[1].
 | 
					
						
							|  |  |  |     auto& shader_regs = regs.shader_config[static_cast<size_t>(shader_program)]; | 
					
						
							|  |  |  |     shader_regs.start_id = address; | 
					
						
							|  |  |  |     // TODO(Subv): Write params[1] to register 0xD1C + shader_program.
 | 
					
						
							|  |  |  |     // TODO(Subv): Write params[2] to register 0xD22 + shader_program.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Note: This value is hardcoded in the macro's code.
 | 
					
						
							|  |  |  |     static constexpr u32 DefaultCBSize = 0x10000; | 
					
						
							|  |  |  |     regs.const_buffer.cb_size = DefaultCBSize; | 
					
						
							|  |  |  |     regs.const_buffer.cb_address_high = cb_address >> 32; | 
					
						
							|  |  |  |     regs.const_buffer.cb_address_low = cb_address & 0xFFFFFFFF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Write a hardcoded 0x11 to CB_BIND, this binds the current const buffer to buffer c1[] in the
 | 
					
						
							|  |  |  |     // shader. It's likely that these are the constants for the shader.
 | 
					
						
							|  |  |  |     regs.cb_bind[static_cast<size_t>(shader_type)].valid.Assign(1); | 
					
						
							|  |  |  |     regs.cb_bind[static_cast<size_t>(shader_type)].index.Assign(1); | 
					
						
							| 
									
										
										
										
											2018-03-16 22:06:24 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-16 20:32:44 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-11 21:34:20 -05:00
										 |  |  | } // namespace Engines
 | 
					
						
							|  |  |  | } // namespace Tegra
 |