2018-02-12 04:44:12 +00:00
|
|
|
// Copyright 2018 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <memory>
|
2021-10-01 04:57:02 +00:00
|
|
|
|
|
|
|
#include "common/bit_field.h"
|
2018-02-12 04:44:12 +00:00
|
|
|
#include "common/common_types.h"
|
2020-10-27 03:07:36 +00:00
|
|
|
#include "video_core/cdma_pusher.h"
|
2020-12-12 06:26:14 +00:00
|
|
|
#include "video_core/framebuffer_config.h"
|
2018-02-12 04:44:12 +00:00
|
|
|
|
2019-02-16 03:05:17 +00:00
|
|
|
namespace Core {
|
2020-03-25 02:58:49 +00:00
|
|
|
namespace Frontend {
|
|
|
|
class EmuWindow;
|
2019-02-16 03:05:17 +00:00
|
|
|
}
|
2020-03-25 02:58:49 +00:00
|
|
|
class System;
|
|
|
|
} // namespace Core
|
2019-02-16 03:05:17 +00:00
|
|
|
|
2018-08-03 16:55:58 +00:00
|
|
|
namespace VideoCore {
|
2019-01-08 04:32:02 +00:00
|
|
|
class RendererBase;
|
2020-07-10 03:36:38 +00:00
|
|
|
class ShaderNotify;
|
2019-01-08 04:32:02 +00:00
|
|
|
} // namespace VideoCore
|
2018-08-03 16:55:58 +00:00
|
|
|
|
2018-02-12 04:44:12 +00:00
|
|
|
namespace Tegra {
|
2021-10-01 04:57:02 +00:00
|
|
|
class DmaPusher;
|
|
|
|
class CDmaPusher;
|
|
|
|
struct CommandList;
|
2018-02-12 04:44:12 +00:00
|
|
|
|
2018-03-24 04:45:24 +00:00
|
|
|
enum class RenderTargetFormat : u32 {
|
2018-03-25 21:57:53 +00:00
|
|
|
NONE = 0x0,
|
2020-07-01 05:28:53 +00:00
|
|
|
R32B32G32A32_FLOAT = 0xC0,
|
|
|
|
R32G32B32A32_SINT = 0xC1,
|
|
|
|
R32G32B32A32_UINT = 0xC2,
|
|
|
|
R16G16B16A16_UNORM = 0xC6,
|
|
|
|
R16G16B16A16_SNORM = 0xC7,
|
|
|
|
R16G16B16A16_SINT = 0xC8,
|
|
|
|
R16G16B16A16_UINT = 0xC9,
|
|
|
|
R16G16B16A16_FLOAT = 0xCA,
|
|
|
|
R32G32_FLOAT = 0xCB,
|
|
|
|
R32G32_SINT = 0xCC,
|
|
|
|
R32G32_UINT = 0xCD,
|
|
|
|
R16G16B16X16_FLOAT = 0xCE,
|
|
|
|
B8G8R8A8_UNORM = 0xCF,
|
|
|
|
B8G8R8A8_SRGB = 0xD0,
|
|
|
|
A2B10G10R10_UNORM = 0xD1,
|
|
|
|
A2B10G10R10_UINT = 0xD2,
|
|
|
|
A8B8G8R8_UNORM = 0xD5,
|
|
|
|
A8B8G8R8_SRGB = 0xD6,
|
|
|
|
A8B8G8R8_SNORM = 0xD7,
|
|
|
|
A8B8G8R8_SINT = 0xD8,
|
|
|
|
A8B8G8R8_UINT = 0xD9,
|
|
|
|
R16G16_UNORM = 0xDA,
|
|
|
|
R16G16_SNORM = 0xDB,
|
|
|
|
R16G16_SINT = 0xDC,
|
|
|
|
R16G16_UINT = 0xDD,
|
|
|
|
R16G16_FLOAT = 0xDE,
|
|
|
|
B10G11R11_FLOAT = 0xE0,
|
2020-02-25 20:19:34 +00:00
|
|
|
R32_SINT = 0xE3,
|
2018-08-13 12:55:16 +00:00
|
|
|
R32_UINT = 0xE4,
|
2018-08-01 13:31:42 +00:00
|
|
|
R32_FLOAT = 0xE5,
|
2020-07-01 05:28:53 +00:00
|
|
|
R5G6B5_UNORM = 0xE8,
|
|
|
|
A1R5G5B5_UNORM = 0xE9,
|
|
|
|
R8G8_UNORM = 0xEA,
|
|
|
|
R8G8_SNORM = 0xEB,
|
|
|
|
R8G8_SINT = 0xEC,
|
|
|
|
R8G8_UINT = 0xED,
|
2018-08-11 18:01:50 +00:00
|
|
|
R16_UNORM = 0xEE,
|
|
|
|
R16_SNORM = 0xEF,
|
|
|
|
R16_SINT = 0xF0,
|
|
|
|
R16_UINT = 0xF1,
|
2018-07-26 04:19:15 +00:00
|
|
|
R16_FLOAT = 0xF2,
|
2018-07-24 21:47:50 +00:00
|
|
|
R8_UNORM = 0xF3,
|
2020-06-30 06:51:42 +00:00
|
|
|
R8_SNORM = 0xF4,
|
2020-06-30 07:00:23 +00:00
|
|
|
R8_SINT = 0xF5,
|
2018-08-12 01:44:42 +00:00
|
|
|
R8_UINT = 0xF6,
|
2018-03-22 21:40:11 +00:00
|
|
|
};
|
|
|
|
|
2018-07-02 17:42:04 +00:00
|
|
|
enum class DepthFormat : u32 {
|
2020-07-01 05:28:53 +00:00
|
|
|
D32_FLOAT = 0xA,
|
|
|
|
D16_UNORM = 0x13,
|
|
|
|
S8_UINT_Z24_UNORM = 0x14,
|
|
|
|
D24X8_UNORM = 0x15,
|
|
|
|
D24S8_UNORM = 0x16,
|
|
|
|
D24C8_UNORM = 0x18,
|
|
|
|
D32_FLOAT_S8X24_UINT = 0x19,
|
2018-07-02 17:42:04 +00:00
|
|
|
};
|
|
|
|
|
2018-09-06 13:48:08 +00:00
|
|
|
struct CommandListHeader;
|
2018-03-22 20:19:35 +00:00
|
|
|
class DebugContext;
|
|
|
|
|
2018-03-18 20:15:05 +00:00
|
|
|
namespace Engines {
|
|
|
|
class Fermi2D;
|
|
|
|
class Maxwell3D;
|
2018-06-10 22:02:33 +00:00
|
|
|
class MaxwellDMA;
|
2019-01-22 23:49:31 +00:00
|
|
|
class KeplerCompute;
|
2018-09-08 20:58:20 +00:00
|
|
|
class KeplerMemory;
|
2018-03-18 20:15:05 +00:00
|
|
|
} // namespace Engines
|
|
|
|
|
2018-02-12 04:44:12 +00:00
|
|
|
enum class EngineID {
|
|
|
|
FERMI_TWOD_A = 0x902D, // 2D Engine
|
|
|
|
MAXWELL_B = 0xB197, // 3D Engine
|
2019-01-22 23:49:31 +00:00
|
|
|
KEPLER_COMPUTE_B = 0xB1C0,
|
2018-02-12 04:44:12 +00:00
|
|
|
KEPLER_INLINE_TO_MEMORY_B = 0xA140,
|
|
|
|
MAXWELL_DMA_COPY_A = 0xB0B5,
|
|
|
|
};
|
|
|
|
|
2019-03-04 04:54:16 +00:00
|
|
|
class MemoryManager;
|
|
|
|
|
2020-12-12 06:26:14 +00:00
|
|
|
class GPU final {
|
2018-02-12 04:44:12 +00:00
|
|
|
public:
|
2018-11-24 04:20:56 +00:00
|
|
|
struct MethodCall {
|
|
|
|
u32 method{};
|
|
|
|
u32 argument{};
|
|
|
|
u32 subchannel{};
|
|
|
|
u32 method_count{};
|
|
|
|
|
2020-12-04 19:39:12 +00:00
|
|
|
explicit MethodCall(u32 method_, u32 argument_, u32 subchannel_ = 0, u32 method_count_ = 0)
|
|
|
|
: method(method_), argument(argument_), subchannel(subchannel_),
|
|
|
|
method_count(method_count_) {}
|
2020-11-17 12:14:44 +00:00
|
|
|
|
|
|
|
[[nodiscard]] bool IsLastCall() const {
|
|
|
|
return method_count <= 1;
|
|
|
|
}
|
2018-11-24 04:20:56 +00:00
|
|
|
};
|
|
|
|
|
2021-10-01 04:57:02 +00:00
|
|
|
enum class FenceOperation : u32 {
|
|
|
|
Acquire = 0,
|
|
|
|
Increment = 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
union FenceAction {
|
|
|
|
u32 raw;
|
|
|
|
BitField<0, 1, FenceOperation> op;
|
|
|
|
BitField<8, 24, u32> syncpoint_id;
|
|
|
|
};
|
|
|
|
|
|
|
|
explicit GPU(Core::System& system, bool is_async, bool use_nvdec);
|
2020-12-12 06:26:14 +00:00
|
|
|
~GPU();
|
2020-06-11 03:58:57 +00:00
|
|
|
|
|
|
|
/// Binds a renderer to the GPU.
|
|
|
|
void BindRenderer(std::unique_ptr<VideoCore::RendererBase> renderer);
|
|
|
|
|
2018-11-24 04:20:56 +00:00
|
|
|
/// Calls a GPU method.
|
|
|
|
void CallMethod(const MethodCall& method_call);
|
2018-02-12 04:44:12 +00:00
|
|
|
|
2020-04-20 06:16:56 +00:00
|
|
|
/// Calls a GPU multivalue method.
|
2020-04-20 17:42:14 +00:00
|
|
|
void CallMultiMethod(u32 method, u32 subchannel, const u32* base_start, u32 amount,
|
|
|
|
u32 methods_pending);
|
2020-04-20 06:16:56 +00:00
|
|
|
|
2020-04-16 16:29:53 +00:00
|
|
|
/// Flush all current written commands into the host GPU for execution.
|
2019-07-26 18:20:43 +00:00
|
|
|
void FlushCommands();
|
2020-04-16 16:29:53 +00:00
|
|
|
/// Synchronizes CPU writes with Host GPU memory.
|
2020-02-16 13:51:37 +00:00
|
|
|
void SyncGuestHost();
|
2020-04-16 16:29:53 +00:00
|
|
|
/// Signal the ending of command list.
|
2020-12-12 06:26:14 +00:00
|
|
|
void OnCommandListEnd();
|
2019-07-26 18:20:43 +00:00
|
|
|
|
2020-04-16 16:29:53 +00:00
|
|
|
/// Request a host GPU memory flush from the CPU.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] u64 RequestFlush(VAddr addr, std::size_t size);
|
2020-02-20 15:55:32 +00:00
|
|
|
|
2020-04-16 16:29:53 +00:00
|
|
|
/// Obtains current flush request fence id.
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] u64 CurrentFlushRequestFence() const;
|
2020-02-20 15:55:32 +00:00
|
|
|
|
2020-04-16 16:29:53 +00:00
|
|
|
/// Tick pending requests within the GPU.
|
2020-02-20 15:55:32 +00:00
|
|
|
void TickWork();
|
|
|
|
|
2018-08-28 14:57:56 +00:00
|
|
|
/// Returns a reference to the Maxwell3D GPU engine.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] Engines::Maxwell3D& Maxwell3D();
|
2018-08-28 14:57:56 +00:00
|
|
|
|
2018-07-20 22:31:36 +00:00
|
|
|
/// Returns a const reference to the Maxwell3D GPU engine.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] const Engines::Maxwell3D& Maxwell3D() const;
|
2018-07-20 22:31:36 +00:00
|
|
|
|
2019-07-15 01:25:13 +00:00
|
|
|
/// Returns a reference to the KeplerCompute GPU engine.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] Engines::KeplerCompute& KeplerCompute();
|
2019-07-15 01:25:13 +00:00
|
|
|
|
|
|
|
/// Returns a reference to the KeplerCompute GPU engine.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] const Engines::KeplerCompute& KeplerCompute() const;
|
2019-07-15 01:25:13 +00:00
|
|
|
|
2018-08-28 14:57:56 +00:00
|
|
|
/// Returns a reference to the GPU memory manager.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] Tegra::MemoryManager& MemoryManager();
|
2018-03-22 20:19:35 +00:00
|
|
|
|
2018-08-28 14:57:56 +00:00
|
|
|
/// Returns a const reference to the GPU memory manager.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] const Tegra::MemoryManager& MemoryManager() const;
|
2018-02-12 04:44:12 +00:00
|
|
|
|
2018-11-24 04:20:56 +00:00
|
|
|
/// Returns a reference to the GPU DMA pusher.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] Tegra::DmaPusher& DmaPusher();
|
2018-11-24 04:20:56 +00:00
|
|
|
|
2020-10-27 03:07:36 +00:00
|
|
|
/// Returns a const reference to the GPU DMA pusher.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] const Tegra::DmaPusher& DmaPusher() const;
|
2020-10-27 03:07:36 +00:00
|
|
|
|
|
|
|
/// Returns a reference to the GPU CDMA pusher.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] Tegra::CDmaPusher& CDmaPusher();
|
2020-10-27 03:07:36 +00:00
|
|
|
|
|
|
|
/// Returns a const reference to the GPU CDMA pusher.
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] const Tegra::CDmaPusher& CDmaPusher() const;
|
2020-10-27 03:07:36 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
/// Returns a reference to the underlying renderer.
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] VideoCore::RendererBase& Renderer();
|
2020-03-25 02:58:49 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
/// Returns a const reference to the underlying renderer.
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] const VideoCore::RendererBase& Renderer() const;
|
2020-03-25 02:58:49 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
/// Returns a reference to the shader notifier.
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] VideoCore::ShaderNotify& ShaderNotify();
|
2020-07-10 03:36:38 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
/// Returns a const reference to the shader notifier.
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] const VideoCore::ShaderNotify& ShaderNotify() const;
|
2020-07-10 03:36:38 +00:00
|
|
|
|
2019-09-25 23:43:23 +00:00
|
|
|
/// Allows the CPU/NvFlinger to wait on the GPU before presenting a frame.
|
2019-12-30 12:03:20 +00:00
|
|
|
void WaitFence(u32 syncpoint_id, u32 value);
|
2019-09-25 23:43:23 +00:00
|
|
|
|
2019-06-19 00:53:21 +00:00
|
|
|
void IncrementSyncPoint(u32 syncpoint_id);
|
2019-06-07 16:56:30 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] u32 GetSyncpointValue(u32 syncpoint_id) const;
|
2019-06-07 16:56:30 +00:00
|
|
|
|
2019-06-19 00:53:21 +00:00
|
|
|
void RegisterSyncptInterrupt(u32 syncpoint_id, u32 value);
|
2019-06-08 01:13:20 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] bool CancelSyncptInterrupt(u32 syncpoint_id, u32 value);
|
2019-06-07 16:56:30 +00:00
|
|
|
|
2020-11-17 12:14:44 +00:00
|
|
|
[[nodiscard]] u64 GetTicks() const;
|
2020-02-10 14:32:51 +00:00
|
|
|
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] bool IsAsync() const;
|
2019-06-10 12:19:27 +00:00
|
|
|
|
2021-10-01 04:57:02 +00:00
|
|
|
[[nodiscard]] bool UseNvdec() const;
|
2018-11-24 04:20:56 +00:00
|
|
|
|
2021-05-16 00:34:20 +00:00
|
|
|
void RendererFrameEndNotify();
|
|
|
|
|
2019-04-09 18:02:00 +00:00
|
|
|
/// Performs any additional setup necessary in order to begin GPU emulation.
|
|
|
|
/// This can be used to launch any necessary threads and register any necessary
|
|
|
|
/// core timing events.
|
2020-12-12 06:26:14 +00:00
|
|
|
void Start();
|
2019-04-09 18:02:00 +00:00
|
|
|
|
2020-04-03 15:58:43 +00:00
|
|
|
/// Obtain the CPU Context
|
2020-12-12 06:26:14 +00:00
|
|
|
void ObtainContext();
|
2020-04-03 15:58:43 +00:00
|
|
|
|
|
|
|
/// Release the CPU Context
|
2020-12-12 06:26:14 +00:00
|
|
|
void ReleaseContext();
|
2020-04-03 15:58:43 +00:00
|
|
|
|
2019-01-21 20:18:09 +00:00
|
|
|
/// Push GPU command entries to be processed
|
2020-12-12 06:26:14 +00:00
|
|
|
void PushGPUEntries(Tegra::CommandList&& entries);
|
2019-01-21 20:18:09 +00:00
|
|
|
|
2020-10-27 03:07:36 +00:00
|
|
|
/// Push GPU command buffer entries to be processed
|
2020-12-12 06:26:14 +00:00
|
|
|
void PushCommandBuffer(Tegra::ChCommandHeaderList& entries);
|
2020-10-27 03:07:36 +00:00
|
|
|
|
2021-04-24 23:22:09 +00:00
|
|
|
/// Frees the CDMAPusher instance to free up resources
|
|
|
|
void ClearCdmaInstance();
|
2021-03-30 09:37:40 +00:00
|
|
|
|
2019-01-21 20:18:09 +00:00
|
|
|
/// Swap buffers (render frame)
|
2020-12-12 06:26:14 +00:00
|
|
|
void SwapBuffers(const Tegra::FramebufferConfig* framebuffer);
|
2019-01-30 02:49:18 +00:00
|
|
|
|
2019-01-24 03:17:55 +00:00
|
|
|
/// Notify rasterizer that any caches of the specified region should be flushed to Switch memory
|
2020-12-12 06:26:14 +00:00
|
|
|
void FlushRegion(VAddr addr, u64 size);
|
2019-01-24 03:17:55 +00:00
|
|
|
|
|
|
|
/// Notify rasterizer that any caches of the specified region should be invalidated
|
2020-12-12 06:26:14 +00:00
|
|
|
void InvalidateRegion(VAddr addr, u64 size);
|
2019-01-24 03:17:55 +00:00
|
|
|
|
|
|
|
/// Notify rasterizer that any caches of the specified region should be flushed and invalidated
|
2020-12-12 06:26:14 +00:00
|
|
|
void FlushAndInvalidateRegion(VAddr addr, u64 size);
|
2019-01-24 03:17:55 +00:00
|
|
|
|
2019-02-09 04:21:53 +00:00
|
|
|
private:
|
2021-10-01 04:57:02 +00:00
|
|
|
struct Impl;
|
|
|
|
std::unique_ptr<Impl> impl;
|
2018-02-12 04:44:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Tegra
|