2018-03-18 15:15:05 -05:00
|
|
|
// Copyright 2018 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-08-10 18:39:37 -04:00
|
|
|
#include "common/assert.h"
|
2018-03-18 15:15:05 -05:00
|
|
|
#include "video_core/engines/fermi_2d.h"
|
2018-09-08 15:58:20 -05:00
|
|
|
#include "video_core/engines/kepler_memory.h"
|
2018-03-18 15:15:05 -05:00
|
|
|
#include "video_core/engines/maxwell_3d.h"
|
|
|
|
#include "video_core/engines/maxwell_compute.h"
|
2018-06-10 17:02:33 -05:00
|
|
|
#include "video_core/engines/maxwell_dma.h"
|
2018-03-18 15:15:05 -05:00
|
|
|
#include "video_core/gpu.h"
|
2018-08-03 12:55:58 -04:00
|
|
|
#include "video_core/rasterizer_interface.h"
|
2018-03-18 15:15:05 -05:00
|
|
|
|
|
|
|
namespace Tegra {
|
|
|
|
|
2018-08-10 18:39:37 -04:00
|
|
|
u32 FramebufferConfig::BytesPerPixel(PixelFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case PixelFormat::ABGR8:
|
|
|
|
return 4;
|
2018-11-20 21:27:34 -04:00
|
|
|
default:
|
|
|
|
return 4;
|
2018-08-10 18:39:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
2018-08-03 12:55:58 -04:00
|
|
|
GPU::GPU(VideoCore::RasterizerInterface& rasterizer) {
|
2018-08-28 10:57:56 -04:00
|
|
|
memory_manager = std::make_unique<Tegra::MemoryManager>();
|
2018-11-23 23:20:56 -05:00
|
|
|
dma_pusher = std::make_unique<Tegra::DmaPusher>(*this);
|
2018-08-03 12:55:58 -04:00
|
|
|
maxwell_3d = std::make_unique<Engines::Maxwell3D>(rasterizer, *memory_manager);
|
2018-10-05 23:46:40 -04:00
|
|
|
fermi_2d = std::make_unique<Engines::Fermi2D>(rasterizer, *memory_manager);
|
2018-03-18 15:15:05 -05:00
|
|
|
maxwell_compute = std::make_unique<Engines::MaxwellCompute>();
|
2018-10-17 21:29:10 -04:00
|
|
|
maxwell_dma = std::make_unique<Engines::MaxwellDMA>(rasterizer, *memory_manager);
|
2018-10-17 20:44:07 -04:00
|
|
|
kepler_memory = std::make_unique<Engines::KeplerMemory>(rasterizer, *memory_manager);
|
2018-03-18 15:15:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
GPU::~GPU() = default;
|
|
|
|
|
2018-08-28 10:57:56 -04:00
|
|
|
Engines::Maxwell3D& GPU::Maxwell3D() {
|
2018-07-20 18:31:36 -04:00
|
|
|
return *maxwell_3d;
|
|
|
|
}
|
|
|
|
|
2018-08-28 10:57:56 -04:00
|
|
|
const Engines::Maxwell3D& GPU::Maxwell3D() const {
|
2018-03-22 15:19:35 -05:00
|
|
|
return *maxwell_3d;
|
|
|
|
}
|
|
|
|
|
2018-08-28 10:57:56 -04:00
|
|
|
MemoryManager& GPU::MemoryManager() {
|
|
|
|
return *memory_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MemoryManager& GPU::MemoryManager() const {
|
|
|
|
return *memory_manager;
|
|
|
|
}
|
|
|
|
|
2018-11-23 23:20:56 -05:00
|
|
|
DmaPusher& GPU::DmaPusher() {
|
|
|
|
return *dma_pusher;
|
|
|
|
}
|
|
|
|
|
|
|
|
const DmaPusher& GPU::DmaPusher() const {
|
|
|
|
return *dma_pusher;
|
|
|
|
}
|
|
|
|
|
2018-04-24 21:57:10 -05:00
|
|
|
u32 RenderTargetBytesPerPixel(RenderTargetFormat format) {
|
|
|
|
ASSERT(format != RenderTargetFormat::NONE);
|
|
|
|
|
|
|
|
switch (format) {
|
2018-06-05 21:07:40 -05:00
|
|
|
case RenderTargetFormat::RGBA32_FLOAT:
|
2018-08-11 20:01:50 +02:00
|
|
|
case RenderTargetFormat::RGBA32_UINT:
|
2018-06-05 21:07:40 -05:00
|
|
|
return 16;
|
2018-08-13 00:04:52 -04:00
|
|
|
case RenderTargetFormat::RGBA16_UINT:
|
2018-08-13 00:34:20 -04:00
|
|
|
case RenderTargetFormat::RGBA16_UNORM:
|
2018-06-05 22:01:20 -05:00
|
|
|
case RenderTargetFormat::RGBA16_FLOAT:
|
2018-07-23 19:10:00 -04:00
|
|
|
case RenderTargetFormat::RG32_FLOAT:
|
2018-08-13 22:55:16 +10:00
|
|
|
case RenderTargetFormat::RG32_UINT:
|
2018-06-05 22:01:20 -05:00
|
|
|
return 8;
|
2018-04-24 21:57:10 -05:00
|
|
|
case RenderTargetFormat::RGBA8_UNORM:
|
2018-08-10 11:44:43 -04:00
|
|
|
case RenderTargetFormat::RGBA8_SNORM:
|
2018-08-11 20:01:50 +02:00
|
|
|
case RenderTargetFormat::RGBA8_SRGB:
|
2018-08-20 22:26:54 +10:00
|
|
|
case RenderTargetFormat::RGBA8_UINT:
|
2018-04-24 21:57:10 -05:00
|
|
|
case RenderTargetFormat::RGB10_A2_UNORM:
|
2018-07-23 16:56:52 -04:00
|
|
|
case RenderTargetFormat::BGRA8_UNORM:
|
2018-09-09 23:41:37 -04:00
|
|
|
case RenderTargetFormat::BGRA8_SRGB:
|
2018-08-11 20:01:50 +02:00
|
|
|
case RenderTargetFormat::RG16_UNORM:
|
|
|
|
case RenderTargetFormat::RG16_SNORM:
|
|
|
|
case RenderTargetFormat::RG16_UINT:
|
|
|
|
case RenderTargetFormat::RG16_SINT:
|
|
|
|
case RenderTargetFormat::RG16_FLOAT:
|
2018-08-01 15:31:42 +02:00
|
|
|
case RenderTargetFormat::R32_FLOAT:
|
2018-08-08 02:40:04 -04:00
|
|
|
case RenderTargetFormat::R11G11B10_FLOAT:
|
2018-08-13 22:55:16 +10:00
|
|
|
case RenderTargetFormat::R32_UINT:
|
2018-04-24 21:57:10 -05:00
|
|
|
return 4;
|
2018-08-11 20:01:50 +02:00
|
|
|
case RenderTargetFormat::R16_UNORM:
|
|
|
|
case RenderTargetFormat::R16_SNORM:
|
|
|
|
case RenderTargetFormat::R16_UINT:
|
|
|
|
case RenderTargetFormat::R16_SINT:
|
|
|
|
case RenderTargetFormat::R16_FLOAT:
|
2018-08-12 23:02:34 -04:00
|
|
|
case RenderTargetFormat::RG8_UNORM:
|
2018-08-10 12:07:37 -04:00
|
|
|
case RenderTargetFormat::RG8_SNORM:
|
2018-08-11 20:01:50 +02:00
|
|
|
return 2;
|
|
|
|
case RenderTargetFormat::R8_UNORM:
|
2018-08-12 03:44:42 +02:00
|
|
|
case RenderTargetFormat::R8_UINT:
|
2018-08-11 20:01:50 +02:00
|
|
|
return 1;
|
2018-04-24 21:57:10 -05:00
|
|
|
default:
|
2018-04-27 07:54:05 -04:00
|
|
|
UNIMPLEMENTED_MSG("Unimplemented render target format {}", static_cast<u32>(format));
|
2018-12-19 12:52:32 +11:00
|
|
|
return 1;
|
2018-04-24 21:57:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-11 20:01:50 +02:00
|
|
|
u32 DepthFormatBytesPerPixel(DepthFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case DepthFormat::Z32_S8_X24_FLOAT:
|
|
|
|
return 8;
|
|
|
|
case DepthFormat::Z32_FLOAT:
|
|
|
|
case DepthFormat::S8_Z24_UNORM:
|
|
|
|
case DepthFormat::Z24_X8_UNORM:
|
|
|
|
case DepthFormat::Z24_S8_UNORM:
|
|
|
|
case DepthFormat::Z24_C8_UNORM:
|
|
|
|
return 4;
|
|
|
|
case DepthFormat::Z16_UNORM:
|
|
|
|
return 2;
|
|
|
|
default:
|
|
|
|
UNIMPLEMENTED_MSG("Unimplemented Depth format {}", static_cast<u32>(format));
|
2018-12-19 12:52:32 +11:00
|
|
|
return 1;
|
2018-08-11 20:01:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-23 23:20:56 -05:00
|
|
|
enum class BufferMethods {
|
|
|
|
BindObject = 0,
|
|
|
|
CountBufferMethods = 0x40,
|
|
|
|
};
|
|
|
|
|
|
|
|
void GPU::CallMethod(const MethodCall& method_call) {
|
2018-12-01 02:05:19 -05:00
|
|
|
LOG_TRACE(HW_GPU, "Processing method {:08X} on subchannel {}", method_call.method,
|
|
|
|
method_call.subchannel);
|
2018-11-23 23:20:56 -05:00
|
|
|
|
|
|
|
ASSERT(method_call.subchannel < bound_engines.size());
|
|
|
|
|
|
|
|
if (method_call.method == static_cast<u32>(BufferMethods::BindObject)) {
|
|
|
|
// Bind the current subchannel to the desired engine id.
|
|
|
|
LOG_DEBUG(HW_GPU, "Binding subchannel {} to engine {}", method_call.subchannel,
|
|
|
|
method_call.argument);
|
|
|
|
bound_engines[method_call.subchannel] = static_cast<EngineID>(method_call.argument);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-12-03 23:52:18 -05:00
|
|
|
if (method_call.method < static_cast<u32>(BufferMethods::CountBufferMethods)) {
|
|
|
|
// TODO(Subv): Research and implement these methods.
|
|
|
|
LOG_ERROR(HW_GPU, "Special buffer methods other than Bind are not implemented");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-11-23 23:20:56 -05:00
|
|
|
const EngineID engine = bound_engines[method_call.subchannel];
|
|
|
|
|
|
|
|
switch (engine) {
|
|
|
|
case EngineID::FERMI_TWOD_A:
|
|
|
|
fermi_2d->CallMethod(method_call);
|
|
|
|
break;
|
|
|
|
case EngineID::MAXWELL_B:
|
|
|
|
maxwell_3d->CallMethod(method_call);
|
|
|
|
break;
|
|
|
|
case EngineID::MAXWELL_COMPUTE_B:
|
|
|
|
maxwell_compute->CallMethod(method_call);
|
|
|
|
break;
|
|
|
|
case EngineID::MAXWELL_DMA_COPY_A:
|
|
|
|
maxwell_dma->CallMethod(method_call);
|
|
|
|
break;
|
|
|
|
case EngineID::KEPLER_INLINE_TO_MEMORY_B:
|
|
|
|
kepler_memory->CallMethod(method_call);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
UNIMPLEMENTED_MSG("Unimplemented engine");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-18 15:15:05 -05:00
|
|
|
} // namespace Tegra
|