2024-05-01 03:51:01 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2020-11-21 03:32:58 +00:00
|
|
|
#pragma once
|
|
|
|
#include "common/heap_array.h"
|
2022-07-11 13:03:29 +00:00
|
|
|
#include "common/threading.h"
|
2020-11-21 03:32:58 +00:00
|
|
|
#include "gpu_types.h"
|
|
|
|
#include <atomic>
|
|
|
|
#include <condition_variable>
|
|
|
|
#include <memory>
|
|
|
|
#include <mutex>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(push)
|
|
|
|
#pragma warning(disable : 4324) // warning C4324: 'GPUBackend': structure was padded due to alignment specifier
|
|
|
|
#endif
|
|
|
|
|
|
|
|
class GPUBackend
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
GPUBackend();
|
|
|
|
virtual ~GPUBackend();
|
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
ALWAYS_INLINE const Threading::Thread* GetThread() const { return m_use_gpu_thread ? &m_gpu_thread : nullptr; }
|
2020-11-21 03:32:58 +00:00
|
|
|
|
2021-05-19 03:43:49 +00:00
|
|
|
virtual bool Initialize(bool force_thread);
|
2020-11-21 03:32:58 +00:00
|
|
|
virtual void UpdateSettings();
|
2023-12-23 06:53:15 +00:00
|
|
|
virtual void Reset();
|
2020-11-21 03:32:58 +00:00
|
|
|
virtual void Shutdown();
|
|
|
|
|
|
|
|
GPUBackendFillVRAMCommand* NewFillVRAMCommand();
|
|
|
|
GPUBackendUpdateVRAMCommand* NewUpdateVRAMCommand(u32 num_words);
|
|
|
|
GPUBackendCopyVRAMCommand* NewCopyVRAMCommand();
|
|
|
|
GPUBackendSetDrawingAreaCommand* NewSetDrawingAreaCommand();
|
2024-05-01 03:51:01 +00:00
|
|
|
GPUBackendUpdateCLUTCommand* NewUpdateCLUTCommand();
|
2020-11-21 03:32:58 +00:00
|
|
|
GPUBackendDrawPolygonCommand* NewDrawPolygonCommand(u32 num_vertices);
|
|
|
|
GPUBackendDrawRectangleCommand* NewDrawRectangleCommand();
|
|
|
|
GPUBackendDrawLineCommand* NewDrawLineCommand(u32 num_vertices);
|
|
|
|
|
|
|
|
void PushCommand(GPUBackendCommand* cmd);
|
2021-04-17 12:16:59 +00:00
|
|
|
void Sync(bool allow_sleep);
|
2020-11-21 03:32:58 +00:00
|
|
|
|
|
|
|
/// Processes all pending GPU commands.
|
|
|
|
void RunGPULoop();
|
|
|
|
|
|
|
|
protected:
|
2020-12-08 13:35:25 +00:00
|
|
|
void* AllocateCommand(GPUBackendCommandType command, u32 size);
|
2020-11-21 03:32:58 +00:00
|
|
|
u32 GetPendingCommandSize() const;
|
|
|
|
void WakeGPUThread();
|
|
|
|
void StartGPUThread();
|
|
|
|
void StopGPUThread();
|
|
|
|
|
|
|
|
virtual void FillVRAM(u32 x, u32 y, u32 width, u32 height, u32 color, GPUBackendCommandParameters params) = 0;
|
|
|
|
virtual void UpdateVRAM(u32 x, u32 y, u32 width, u32 height, const void* data,
|
|
|
|
GPUBackendCommandParameters params) = 0;
|
|
|
|
virtual void CopyVRAM(u32 src_x, u32 src_y, u32 dst_x, u32 dst_y, u32 width, u32 height,
|
|
|
|
GPUBackendCommandParameters params) = 0;
|
|
|
|
virtual void DrawPolygon(const GPUBackendDrawPolygonCommand* cmd) = 0;
|
|
|
|
virtual void DrawRectangle(const GPUBackendDrawRectangleCommand* cmd) = 0;
|
|
|
|
virtual void DrawLine(const GPUBackendDrawLineCommand* cmd) = 0;
|
|
|
|
virtual void FlushRender() = 0;
|
|
|
|
virtual void DrawingAreaChanged() = 0;
|
2024-05-01 03:51:01 +00:00
|
|
|
virtual void UpdateCLUT(GPUTexturePaletteReg reg, bool clut_is_8bit) = 0;
|
2020-11-21 03:32:58 +00:00
|
|
|
|
|
|
|
void HandleCommand(const GPUBackendCommand* cmd);
|
|
|
|
|
2024-05-01 04:11:20 +00:00
|
|
|
GPUDrawingArea m_drawing_area = {};
|
2020-11-21 03:32:58 +00:00
|
|
|
|
2022-07-30 11:49:56 +00:00
|
|
|
Threading::KernelSemaphore m_sync_semaphore;
|
2020-11-21 03:32:58 +00:00
|
|
|
std::atomic_bool m_gpu_thread_sleeping{false};
|
|
|
|
std::atomic_bool m_gpu_loop_done{false};
|
2022-07-11 13:03:29 +00:00
|
|
|
Threading::Thread m_gpu_thread;
|
2020-11-21 03:32:58 +00:00
|
|
|
bool m_use_gpu_thread = false;
|
|
|
|
|
|
|
|
std::mutex m_sync_mutex;
|
|
|
|
std::condition_variable m_sync_cpu_thread_cv;
|
|
|
|
std::condition_variable m_wake_gpu_thread_cv;
|
|
|
|
bool m_sync_done = false;
|
|
|
|
|
|
|
|
enum : u32
|
|
|
|
{
|
|
|
|
COMMAND_QUEUE_SIZE = 4 * 1024 * 1024,
|
|
|
|
THRESHOLD_TO_WAKE_GPU = 256
|
|
|
|
};
|
|
|
|
|
2023-08-19 03:52:51 +00:00
|
|
|
FixedHeapArray<u8, COMMAND_QUEUE_SIZE> m_command_fifo_data;
|
2024-04-18 10:38:01 +00:00
|
|
|
alignas(HOST_CACHE_LINE_SIZE) std::atomic<u32> m_command_fifo_read_ptr{0};
|
|
|
|
alignas(HOST_CACHE_LINE_SIZE) std::atomic<u32> m_command_fifo_write_ptr{0};
|
2020-11-21 03:32:58 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(pop)
|
|
|
|
#endif
|