| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 13:09:52 +10:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							|  |  |  | #include "video_core/dirty_flags.h"
 | 
					
						
							| 
									
										
										
										
											2022-12-06 13:45:26 +08:00
										 |  |  | #include "video_core/engines/draw_manager.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | #include "video_core/engines/maxwell_3d.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-25 13:15:45 -05:00
										 |  |  | #include "video_core/macro/macro.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | #include "video_core/macro/macro_hle.h"
 | 
					
						
							|  |  |  | #include "video_core/rasterizer_interface.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Tegra { | 
					
						
							| 
									
										
										
										
											2020-06-24 12:18:33 +10:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2022-01-25 13:15:45 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | using HLEFunction = void (*)(Engines::Maxwell3D& maxwell3d, const std::vector<u32>& parameters); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | // HLE'd functions
 | 
					
						
							| 
									
										
										
										
											2020-07-16 22:01:45 -04:00
										 |  |  | void HLE_771BB18C62444DA0(Engines::Maxwell3D& maxwell3d, const std::vector<u32>& parameters) { | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  |     const u32 instance_count = parameters[2] & maxwell3d.GetRegisterValue(0xD1B); | 
					
						
							| 
									
										
										
										
											2022-12-06 13:45:26 +08:00
										 |  |  |     maxwell3d.draw_manager->DrawIndex( | 
					
						
							|  |  |  |         static_cast<Tegra::Engines::Maxwell3D::Regs::PrimitiveTopology>(parameters[0] & 0x3ffffff), | 
					
						
							|  |  |  |         parameters[4], parameters[1], parameters[3], parameters[5], instance_count); | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 22:01:45 -04:00
										 |  |  | void HLE_0D61FC9FAAC9FCAD(Engines::Maxwell3D& maxwell3d, const std::vector<u32>& parameters) { | 
					
						
							| 
									
										
										
										
											2022-10-21 15:38:50 +08:00
										 |  |  |     const u32 instance_count = (maxwell3d.GetRegisterValue(0xD1B) & parameters[2]); | 
					
						
							| 
									
										
										
										
											2022-12-06 13:45:26 +08:00
										 |  |  |     maxwell3d.draw_manager->DrawArray( | 
					
						
							|  |  |  |         static_cast<Tegra::Engines::Maxwell3D::Regs::PrimitiveTopology>(parameters[0]), | 
					
						
							|  |  |  |         parameters[3], parameters[1], parameters[4], instance_count); | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 22:01:45 -04:00
										 |  |  | void HLE_0217920100488FF7(Engines::Maxwell3D& maxwell3d, const std::vector<u32>& parameters) { | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  |     const u32 instance_count = (maxwell3d.GetRegisterValue(0xD1B) & parameters[2]); | 
					
						
							|  |  |  |     const u32 element_base = parameters[4]; | 
					
						
							|  |  |  |     const u32 base_instance = parameters[5]; | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |     maxwell3d.regs.vertex_id_base = element_base; | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  |     maxwell3d.dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; | 
					
						
							| 
									
										
										
										
											2022-10-21 15:38:50 +08:00
										 |  |  |     maxwell3d.CallMethod(0x8e3, 0x640, true); | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e4, element_base, true); | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e5, base_instance, true); | 
					
						
							| 
									
										
										
										
											2022-12-06 13:45:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     maxwell3d.draw_manager->DrawIndex( | 
					
						
							|  |  |  |         static_cast<Tegra::Engines::Maxwell3D::Regs::PrimitiveTopology>(parameters[0]), | 
					
						
							|  |  |  |         parameters[3], parameters[1], element_base, base_instance, instance_count); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 10:58:09 +01:00
										 |  |  |     maxwell3d.regs.vertex_id_base = 0x0; | 
					
						
							| 
									
										
										
										
											2022-10-21 15:38:50 +08:00
										 |  |  |     maxwell3d.CallMethod(0x8e3, 0x640, true); | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e4, 0x0, true); | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e5, 0x0, true); | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  | // Multidraw Indirect
 | 
					
						
							|  |  |  | void HLE_3F5E74B9C9A50164(Engines::Maxwell3D& maxwell3d, const std::vector<u32>& parameters) { | 
					
						
							|  |  |  |     const u32 start_indirect = parameters[0]; | 
					
						
							|  |  |  |     const u32 end_indirect = parameters[1]; | 
					
						
							|  |  |  |     if (start_indirect >= end_indirect) { | 
					
						
							|  |  |  |         // Nothing to do.
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-09 15:00:05 +01:00
										 |  |  |     const auto topology = | 
					
						
							|  |  |  |         static_cast<Tegra::Engines::Maxwell3D::Regs::PrimitiveTopology>(parameters[2]); | 
					
						
							|  |  |  |     const u32 padding = parameters[3]; // padding is in words
 | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 15:00:05 +01:00
										 |  |  |     // size of each indirect segment
 | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  |     const u32 indirect_words = 5 + padding; | 
					
						
							| 
									
										
										
										
											2022-02-09 15:00:05 +01:00
										 |  |  |     const u32 stride = indirect_words * sizeof(u32); | 
					
						
							|  |  |  |     const GPUVAddr start_address = maxwell3d.current_dma_segment + 4 * sizeof(u32); | 
					
						
							|  |  |  |     const std::size_t draw_count = end_indirect - start_indirect; | 
					
						
							|  |  |  |     u32 lowest_first = std::numeric_limits<u32>::max(); | 
					
						
							|  |  |  |     u32 highest_limit = std::numeric_limits<u32>::min(); | 
					
						
							|  |  |  |     for (std::size_t index = 0; index < draw_count; index++) { | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  |         const std::size_t base = index * indirect_words + 5; | 
					
						
							| 
									
										
										
										
											2022-02-09 15:00:05 +01:00
										 |  |  |         const u32 count = parameters[base]; | 
					
						
							|  |  |  |         const u32 first_index = parameters[base + 2]; | 
					
						
							|  |  |  |         lowest_first = std::min(lowest_first, first_index); | 
					
						
							|  |  |  |         highest_limit = std::max(highest_limit, first_index + count); | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-09 15:00:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const u32 base_vertex = parameters[8]; | 
					
						
							|  |  |  |     const u32 base_instance = parameters[9]; | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e3, 0x640, true); | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e4, base_vertex, true); | 
					
						
							|  |  |  |     maxwell3d.CallMethod(0x8e5, base_instance, true); | 
					
						
							|  |  |  |     auto& params = maxwell3d.draw_manager->GetIndirectParams(); | 
					
						
							|  |  |  |     params.start_address = start_address; | 
					
						
							|  |  |  |     params.buffer_size = sizeof(u32) + stride * draw_count; | 
					
						
							|  |  |  |     params.max_draw_counts = draw_count; | 
					
						
							|  |  |  |     params.stride = stride; | 
					
						
							|  |  |  |     maxwell3d.dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; | 
					
						
							|  |  |  |     maxwell3d.draw_manager->DrawIndexedIndirect(topology, 0, highest_limit); | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 22:28:58 -05:00
										 |  |  | // Multi-layer Clear
 | 
					
						
							|  |  |  | void HLE_EAD26C3E2109B06B(Engines::Maxwell3D& maxwell3d, const std::vector<u32>& parameters) { | 
					
						
							|  |  |  |     ASSERT(parameters.size() == 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 22:41:40 -05:00
										 |  |  |     const Engines::Maxwell3D::Regs::ClearSurface clear_params{parameters[0]}; | 
					
						
							| 
									
										
										
										
											2022-11-16 22:28:58 -05:00
										 |  |  |     const u32 rt_index = clear_params.RT; | 
					
						
							| 
									
										
										
										
											2022-11-16 22:41:40 -05:00
										 |  |  |     const u32 num_layers = maxwell3d.regs.rt[rt_index].depth; | 
					
						
							|  |  |  |     ASSERT(clear_params.layer == 0); | 
					
						
							| 
									
										
										
										
											2022-11-16 22:28:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 22:41:40 -05:00
										 |  |  |     maxwell3d.regs.clear_surface.raw = clear_params.raw; | 
					
						
							| 
									
										
										
										
											2022-12-06 13:45:26 +08:00
										 |  |  |     maxwell3d.draw_manager->Clear(num_layers); | 
					
						
							| 
									
										
										
										
											2022-11-16 22:28:58 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | constexpr std::array<std::pair<u64, HLEFunction>, 5> hle_funcs{{ | 
					
						
							| 
									
										
										
										
											2020-07-16 21:59:23 -04:00
										 |  |  |     {0x771BB18C62444DA0, &HLE_771BB18C62444DA0}, | 
					
						
							|  |  |  |     {0x0D61FC9FAAC9FCAD, &HLE_0D61FC9FAAC9FCAD}, | 
					
						
							|  |  |  |     {0x0217920100488FF7, &HLE_0217920100488FF7}, | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  |     {0x3F5E74B9C9A50164, &HLE_3F5E74B9C9A50164}, | 
					
						
							| 
									
										
										
										
											2022-11-16 22:28:58 -05:00
										 |  |  |     {0xEAD26C3E2109B06B, &HLE_EAD26C3E2109B06B}, | 
					
						
							| 
									
										
										
										
											2020-06-24 12:18:33 +10:00
										 |  |  | }}; | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-25 13:15:45 -05:00
										 |  |  | class HLEMacroImpl final : public CachedMacro { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit HLEMacroImpl(Engines::Maxwell3D& maxwell3d_, HLEFunction func_) | 
					
						
							|  |  |  |         : maxwell3d{maxwell3d_}, func{func_} {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Execute(const std::vector<u32>& parameters, u32 method) override { | 
					
						
							|  |  |  |         func(maxwell3d, parameters); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     Engines::Maxwell3D& maxwell3d; | 
					
						
							|  |  |  |     HLEFunction func; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-07-10 19:32:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-25 13:15:45 -05:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 14:39:12 -05:00
										 |  |  | HLEMacro::HLEMacro(Engines::Maxwell3D& maxwell3d_) : maxwell3d{maxwell3d_} {} | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | HLEMacro::~HLEMacro() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-25 13:50:10 -05:00
										 |  |  | std::unique_ptr<CachedMacro> HLEMacro::GetHLEProgram(u64 hash) const { | 
					
						
							| 
									
										
										
										
											2020-06-24 12:18:33 +10:00
										 |  |  |     const auto it = std::find_if(hle_funcs.cbegin(), hle_funcs.cend(), | 
					
						
							|  |  |  |                                  [hash](const auto& pair) { return pair.first == hash; }); | 
					
						
							| 
									
										
										
										
											2020-06-05 13:09:52 +10:00
										 |  |  |     if (it == hle_funcs.end()) { | 
					
						
							| 
									
										
										
										
											2022-01-25 13:50:10 -05:00
										 |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-05 13:09:52 +10:00
										 |  |  |     return std::make_unique<HLEMacroImpl>(maxwell3d, it->second); | 
					
						
							| 
									
										
										
										
											2020-06-05 01:42:19 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Tegra
 |