diff options
| author | 2018-03-18 03:13:22 -0500 | |
|---|---|---|
| committer | 2018-03-18 11:51:45 -0500 | |
| commit | 7ac8657432f2dad14c985ef3df3972cd126fc9d8 (patch) | |
| tree | bba6f790c07cfa8702ff69bf79b8431a0f63527d /src | |
| parent | Merge pull request #245 from Subv/set_shader2 (diff) | |
| download | yuzu-7ac8657432f2dad14c985ef3df3972cd126fc9d8.tar.gz yuzu-7ac8657432f2dad14c985ef3df3972cd126fc9d8.tar.xz yuzu-7ac8657432f2dad14c985ef3df3972cd126fc9d8.zip | |
GPU: Macros are specific to the Maxwell3D engine, so handle them internally.
Diffstat (limited to 'src')
| -rw-r--r-- | src/video_core/command_processor.cpp | 29 | ||||
| -rw-r--r-- | src/video_core/engines/fermi_2d.cpp | 1 | ||||
| -rw-r--r-- | src/video_core/engines/fermi_2d.h | 8 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_3d.cpp | 41 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_3d.h | 27 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_compute.cpp | 1 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_compute.h | 8 | ||||
| -rw-r--r-- | src/video_core/gpu.h | 3 |
8 files changed, 55 insertions, 63 deletions
diff --git a/src/video_core/command_processor.cpp b/src/video_core/command_processor.cpp index 26ba8c40b..1d578582e 100644 --- a/src/video_core/command_processor.cpp +++ b/src/video_core/command_processor.cpp | |||
| @@ -64,35 +64,6 @@ void GPU::WriteReg(u32 method, u32 subchannel, u32 value) { | |||
| 64 | } | 64 | } |
| 65 | } | 65 | } |
| 66 | 66 | ||
| 67 | void GPU::CallMethod(u32 method, u32 subchannel, const std::vector<u32>& parameters) { | ||
| 68 | LOG_WARNING(HW_GPU, "Processing method %08X on subchannel %u num params %zu", method, | ||
| 69 | subchannel, parameters.size()); | ||
| 70 | |||
| 71 | if (method < static_cast<u32>(BufferMethods::CountBufferMethods)) { | ||
| 72 | // TODO(Subv): Research and implement these methods. | ||
| 73 | LOG_ERROR(HW_GPU, "Special buffer methods other than Bind are not implemented"); | ||
| 74 | return; | ||
| 75 | } | ||
| 76 | |||
| 77 | ASSERT(bound_engines.find(subchannel) != bound_engines.end()); | ||
| 78 | |||
| 79 | const EngineID engine = bound_engines[subchannel]; | ||
| 80 | |||
| 81 | switch (engine) { | ||
| 82 | case EngineID::FERMI_TWOD_A: | ||
| 83 | fermi_2d->CallMethod(method, parameters); | ||
| 84 | break; | ||
| 85 | case EngineID::MAXWELL_B: | ||
| 86 | maxwell_3d->CallMethod(method, parameters); | ||
| 87 | break; | ||
| 88 | case EngineID::MAXWELL_COMPUTE_B: | ||
| 89 | maxwell_compute->CallMethod(method, parameters); | ||
| 90 | break; | ||
| 91 | default: | ||
| 92 | UNIMPLEMENTED(); | ||
| 93 | } | ||
| 94 | } | ||
| 95 | |||
| 96 | void GPU::ProcessCommandList(GPUVAddr address, u32 size) { | 67 | void GPU::ProcessCommandList(GPUVAddr address, u32 size) { |
| 97 | // TODO(Subv): PhysicalToVirtualAddress is a misnomer, it converts a GPU VAddr into an | 68 | // TODO(Subv): PhysicalToVirtualAddress is a misnomer, it converts a GPU VAddr into an |
| 98 | // application VAddr. | 69 | // application VAddr. |
diff --git a/src/video_core/engines/fermi_2d.cpp b/src/video_core/engines/fermi_2d.cpp index 6c6162cf3..7aab163dc 100644 --- a/src/video_core/engines/fermi_2d.cpp +++ b/src/video_core/engines/fermi_2d.cpp | |||
| @@ -8,7 +8,6 @@ namespace Tegra { | |||
| 8 | namespace Engines { | 8 | namespace Engines { |
| 9 | 9 | ||
| 10 | void Fermi2D::WriteReg(u32 method, u32 value) {} | 10 | void Fermi2D::WriteReg(u32 method, u32 value) {} |
| 11 | void Fermi2D::CallMethod(u32 method, const std::vector<u32>& parameters) {} | ||
| 12 | 11 | ||
| 13 | } // namespace Engines | 12 | } // namespace Engines |
| 14 | } // namespace Tegra | 13 | } // namespace Tegra |
diff --git a/src/video_core/engines/fermi_2d.h b/src/video_core/engines/fermi_2d.h index ce8920cee..8967ddede 100644 --- a/src/video_core/engines/fermi_2d.h +++ b/src/video_core/engines/fermi_2d.h | |||
| @@ -4,7 +4,6 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <vector> | ||
| 8 | #include "common/common_types.h" | 7 | #include "common/common_types.h" |
| 9 | 8 | ||
| 10 | namespace Tegra { | 9 | namespace Tegra { |
| @@ -17,13 +16,6 @@ public: | |||
| 17 | 16 | ||
| 18 | /// Write the value to the register identified by method. | 17 | /// Write the value to the register identified by method. |
| 19 | void WriteReg(u32 method, u32 value); | 18 | void WriteReg(u32 method, u32 value); |
| 20 | |||
| 21 | /** | ||
| 22 | * Handles a method call to this engine. | ||
| 23 | * @param method Method to call | ||
| 24 | * @param parameters Arguments to the method call | ||
| 25 | */ | ||
| 26 | void CallMethod(u32 method, const std::vector<u32>& parameters); | ||
| 27 | }; | 19 | }; |
| 28 | 20 | ||
| 29 | } // namespace Engines | 21 | } // namespace Engines |
diff --git a/src/video_core/engines/maxwell_3d.cpp b/src/video_core/engines/maxwell_3d.cpp index db12fc702..67b1b4e7f 100644 --- a/src/video_core/engines/maxwell_3d.cpp +++ b/src/video_core/engines/maxwell_3d.cpp | |||
| @@ -8,28 +8,59 @@ | |||
| 8 | namespace Tegra { | 8 | namespace Tegra { |
| 9 | namespace Engines { | 9 | namespace Engines { |
| 10 | 10 | ||
| 11 | /// First register id that is actually a Macro call. | ||
| 12 | constexpr u32 MacroRegistersStart = 0xE00; | ||
| 13 | |||
| 11 | const std::unordered_map<u32, Maxwell3D::MethodInfo> Maxwell3D::method_handlers = { | 14 | const std::unordered_map<u32, Maxwell3D::MethodInfo> Maxwell3D::method_handlers = { |
| 12 | {0xE24, {"SetShader", 5, &Maxwell3D::SetShader}}, | 15 | {0xE24, {"SetShader", 5, &Maxwell3D::SetShader}}, |
| 13 | }; | 16 | }; |
| 14 | 17 | ||
| 15 | Maxwell3D::Maxwell3D(MemoryManager& memory_manager) : memory_manager(memory_manager) {} | 18 | Maxwell3D::Maxwell3D(MemoryManager& memory_manager) : memory_manager(memory_manager) {} |
| 16 | 19 | ||
| 17 | void Maxwell3D::CallMethod(u32 method, const std::vector<u32>& parameters) { | 20 | void Maxwell3D::AttemptMethodCall(u32 method, const std::vector<u32>& parameters) { |
| 18 | // TODO(Subv): Write an interpreter for the macros uploaded via registers 0x45 and 0x47 | 21 | // TODO(Subv): Write an interpreter for the macros uploaded via registers 0x45 and 0x47 |
| 19 | auto itr = method_handlers.find(method); | 22 | auto itr = method_handlers.find(method); |
| 20 | if (itr == method_handlers.end()) { | 23 | ASSERT_MSG(itr != method_handlers.end(), "Unhandled method call %08X", method); |
| 21 | LOG_ERROR(HW_GPU, "Unhandled method call %08X", method); | 24 | |
| 25 | // Only execute the macro handler once we've been fed the expected number of parameters. | ||
| 26 | if (itr->second.arguments != parameters.size()) | ||
| 22 | return; | 27 | return; |
| 23 | } | ||
| 24 | 28 | ||
| 25 | ASSERT(itr->second.arguments == parameters.size()); | ||
| 26 | (this->*itr->second.handler)(parameters); | 29 | (this->*itr->second.handler)(parameters); |
| 30 | |||
| 31 | // Reset the current macro and its parameters. | ||
| 32 | executing_macro = 0; | ||
| 33 | macro_params.clear(); | ||
| 27 | } | 34 | } |
| 28 | 35 | ||
| 29 | void Maxwell3D::WriteReg(u32 method, u32 value) { | 36 | void Maxwell3D::WriteReg(u32 method, u32 value) { |
| 30 | ASSERT_MSG(method < Regs::NUM_REGS, | 37 | ASSERT_MSG(method < Regs::NUM_REGS, |
| 31 | "Invalid Maxwell3D register, increase the size of the Regs structure"); | 38 | "Invalid Maxwell3D register, increase the size of the Regs structure"); |
| 32 | 39 | ||
| 40 | // It is an error to write to a register other than the current macro's ARG register before it | ||
| 41 | // has finished execution. | ||
| 42 | if (executing_macro != 0) { | ||
| 43 | ASSERT(method == executing_macro + 1); | ||
| 44 | } | ||
| 45 | |||
| 46 | // Methods after 0xE00 are special, they're actually triggers for some microcode that was | ||
| 47 | // uploaded to the GPU during initialization. | ||
| 48 | if (method >= MacroRegistersStart) { | ||
| 49 | // We're trying to execute a macro | ||
| 50 | if (executing_macro == 0) { | ||
| 51 | // A macro call must begin by writing the macro method's register, not its argument. | ||
| 52 | ASSERT_MSG((method % 2) == 0, | ||
| 53 | "Can't start macro execution by writing to the ARGS register"); | ||
| 54 | executing_macro = method; | ||
| 55 | } | ||
| 56 | |||
| 57 | macro_params.push_back(value); | ||
| 58 | |||
| 59 | // Try to call the macro with the current number of parameters. | ||
| 60 | AttemptMethodCall(executing_macro, macro_params); | ||
| 61 | return; | ||
| 62 | } | ||
| 63 | |||
| 33 | regs.reg_array[method] = value; | 64 | regs.reg_array[method] = value; |
| 34 | 65 | ||
| 35 | #define MAXWELL3D_REG_INDEX(field_name) (offsetof(Regs, field_name) / sizeof(u32)) | 66 | #define MAXWELL3D_REG_INDEX(field_name) (offsetof(Regs, field_name) / sizeof(u32)) |
diff --git a/src/video_core/engines/maxwell_3d.h b/src/video_core/engines/maxwell_3d.h index 98137f94b..75a1c05bc 100644 --- a/src/video_core/engines/maxwell_3d.h +++ b/src/video_core/engines/maxwell_3d.h | |||
| @@ -23,13 +23,6 @@ public: | |||
| 23 | /// Write the value to the register identified by method. | 23 | /// Write the value to the register identified by method. |
| 24 | void WriteReg(u32 method, u32 value); | 24 | void WriteReg(u32 method, u32 value); |
| 25 | 25 | ||
| 26 | /** | ||
| 27 | * Handles a method call to this engine. | ||
| 28 | * @param method Method to call | ||
| 29 | * @param parameters Arguments to the method call | ||
| 30 | */ | ||
| 31 | void CallMethod(u32 method, const std::vector<u32>& parameters); | ||
| 32 | |||
| 33 | /// Register structure of the Maxwell3D engine. | 26 | /// Register structure of the Maxwell3D engine. |
| 34 | /// TODO(Subv): This structure will need to be made bigger as more registers are discovered. | 27 | /// TODO(Subv): This structure will need to be made bigger as more registers are discovered. |
| 35 | struct Regs { | 28 | struct Regs { |
| @@ -166,7 +159,11 @@ public: | |||
| 166 | INSERT_PADDING_WORDS(7); | 159 | INSERT_PADDING_WORDS(7); |
| 167 | } cb_bind[MaxShaderStage]; | 160 | } cb_bind[MaxShaderStage]; |
| 168 | 161 | ||
| 169 | INSERT_PADDING_WORDS(0x50A); | 162 | INSERT_PADDING_WORDS(0x56); |
| 163 | |||
| 164 | u32 tex_cb_index; | ||
| 165 | |||
| 166 | INSERT_PADDING_WORDS(0x4B3); | ||
| 170 | }; | 167 | }; |
| 171 | std::array<u32, NUM_REGS> reg_array; | 168 | std::array<u32, NUM_REGS> reg_array; |
| 172 | }; | 169 | }; |
| @@ -201,6 +198,19 @@ public: | |||
| 201 | private: | 198 | private: |
| 202 | MemoryManager& memory_manager; | 199 | MemoryManager& memory_manager; |
| 203 | 200 | ||
| 201 | /// Macro method that is currently being executed / being fed parameters. | ||
| 202 | u32 executing_macro = 0; | ||
| 203 | /// Parameters that have been submitted to the macro call so far. | ||
| 204 | std::vector<u32> macro_params; | ||
| 205 | |||
| 206 | /** | ||
| 207 | * Attempts a method call to this engine. Will return without doing anything if the number of | ||
| 208 | * parameters doesn't match what is expected for the method. | ||
| 209 | * @param method Method to call | ||
| 210 | * @param parameters Arguments to the method call | ||
| 211 | */ | ||
| 212 | void AttemptMethodCall(u32 method, const std::vector<u32>& parameters); | ||
| 213 | |||
| 204 | /// Handles a write to the QUERY_GET register. | 214 | /// Handles a write to the QUERY_GET register. |
| 205 | void ProcessQueryGet(); | 215 | void ProcessQueryGet(); |
| 206 | 216 | ||
| @@ -234,6 +244,7 @@ ASSERT_REG_POSITION(vertex_array_limit[0], 0x7C0); | |||
| 234 | ASSERT_REG_POSITION(shader_config[0], 0x800); | 244 | ASSERT_REG_POSITION(shader_config[0], 0x800); |
| 235 | ASSERT_REG_POSITION(const_buffer, 0x8E0); | 245 | ASSERT_REG_POSITION(const_buffer, 0x8E0); |
| 236 | ASSERT_REG_POSITION(cb_bind[0], 0x904); | 246 | ASSERT_REG_POSITION(cb_bind[0], 0x904); |
| 247 | ASSERT_REG_POSITION(tex_cb_index, 0x982); | ||
| 237 | 248 | ||
| 238 | #undef ASSERT_REG_POSITION | 249 | #undef ASSERT_REG_POSITION |
| 239 | 250 | ||
diff --git a/src/video_core/engines/maxwell_compute.cpp b/src/video_core/engines/maxwell_compute.cpp index 3bef7fe86..e4e5f9e5e 100644 --- a/src/video_core/engines/maxwell_compute.cpp +++ b/src/video_core/engines/maxwell_compute.cpp | |||
| @@ -8,7 +8,6 @@ namespace Tegra { | |||
| 8 | namespace Engines { | 8 | namespace Engines { |
| 9 | 9 | ||
| 10 | void MaxwellCompute::WriteReg(u32 method, u32 value) {} | 10 | void MaxwellCompute::WriteReg(u32 method, u32 value) {} |
| 11 | void MaxwellCompute::CallMethod(u32 method, const std::vector<u32>& parameters) {} | ||
| 12 | 11 | ||
| 13 | } // namespace Engines | 12 | } // namespace Engines |
| 14 | } // namespace Tegra | 13 | } // namespace Tegra |
diff --git a/src/video_core/engines/maxwell_compute.h b/src/video_core/engines/maxwell_compute.h index 5fc7ed635..7262e1bcb 100644 --- a/src/video_core/engines/maxwell_compute.h +++ b/src/video_core/engines/maxwell_compute.h | |||
| @@ -4,7 +4,6 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <vector> | ||
| 8 | #include "common/common_types.h" | 7 | #include "common/common_types.h" |
| 9 | 8 | ||
| 10 | namespace Tegra { | 9 | namespace Tegra { |
| @@ -17,13 +16,6 @@ public: | |||
| 17 | 16 | ||
| 18 | /// Write the value to the register identified by method. | 17 | /// Write the value to the register identified by method. |
| 19 | void WriteReg(u32 method, u32 value); | 18 | void WriteReg(u32 method, u32 value); |
| 20 | |||
| 21 | /** | ||
| 22 | * Handles a method call to this engine. | ||
| 23 | * @param method Method to call | ||
| 24 | * @param parameters Arguments to the method call | ||
| 25 | */ | ||
| 26 | void CallMethod(u32 method, const std::vector<u32>& parameters); | ||
| 27 | }; | 19 | }; |
| 28 | 20 | ||
| 29 | } // namespace Engines | 21 | } // namespace Engines |
diff --git a/src/video_core/gpu.h b/src/video_core/gpu.h index c5ec6fdef..ba7781756 100644 --- a/src/video_core/gpu.h +++ b/src/video_core/gpu.h | |||
| @@ -41,9 +41,6 @@ private: | |||
| 41 | /// Writes a single register in the engine bound to the specified subchannel | 41 | /// Writes a single register in the engine bound to the specified subchannel |
| 42 | void WriteReg(u32 method, u32 subchannel, u32 value); | 42 | void WriteReg(u32 method, u32 subchannel, u32 value); |
| 43 | 43 | ||
| 44 | /// Calls a method in the engine bound to the specified subchannel with the input parameters. | ||
| 45 | void CallMethod(u32 method, u32 subchannel, const std::vector<u32>& parameters); | ||
| 46 | |||
| 47 | /// Mapping of command subchannels to their bound engine ids. | 44 | /// Mapping of command subchannels to their bound engine ids. |
| 48 | std::unordered_map<u32, EngineID> bound_engines; | 45 | std::unordered_map<u32, EngineID> bound_engines; |
| 49 | 46 | ||