mirror of
https://github.com/moonlight-stream/moonlight-qt
synced 2025-01-10 18:28:47 +00:00
218 lines
5.6 KiB
C++
218 lines
5.6 KiB
C++
#include "cuda.h"
|
|
|
|
#include <SDL_opengl.h>
|
|
|
|
CUDARenderer::CUDARenderer()
|
|
: m_HwContext(nullptr)
|
|
{
|
|
|
|
}
|
|
|
|
CUDARenderer::~CUDARenderer()
|
|
{
|
|
if (m_HwContext != nullptr) {
|
|
av_buffer_unref(&m_HwContext);
|
|
}
|
|
}
|
|
|
|
bool CUDARenderer::initialize(PDECODER_PARAMETERS)
|
|
{
|
|
int err;
|
|
|
|
err = av_hwdevice_ctx_create(&m_HwContext, AV_HWDEVICE_TYPE_CUDA, nullptr, nullptr, 0);
|
|
if (err != 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"av_hwdevice_ctx_create(CUDA) failed: %d",
|
|
err);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CUDARenderer::prepareDecoderContext(AVCodecContext* context, AVDictionary**)
|
|
{
|
|
context->hw_device_ctx = av_buffer_ref(m_HwContext);
|
|
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Using CUDA accelerated decoder");
|
|
|
|
return true;
|
|
}
|
|
|
|
void CUDARenderer::renderFrame(AVFrame*)
|
|
{
|
|
// We only support indirect rendering
|
|
SDL_assert(false);
|
|
}
|
|
|
|
bool CUDARenderer::needsTestFrame()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool CUDARenderer::isDirectRenderingSupported()
|
|
{
|
|
// We only support rendering via SDL read-back
|
|
return false;
|
|
}
|
|
|
|
int CUDARenderer::getDecoderCapabilities()
|
|
{
|
|
return CAPABILITY_REFERENCE_FRAME_INVALIDATION_HEVC;
|
|
}
|
|
|
|
CUDAGLInteropHelper::CUDAGLInteropHelper(AVHWDeviceContext* context)
|
|
: m_Funcs(nullptr),
|
|
m_Context((AVCUDADeviceContext*)context->hwctx)
|
|
{
|
|
memset(m_Resources, 0, sizeof(m_Resources));
|
|
|
|
// One-time init of CUDA library
|
|
cuda_load_functions(&m_Funcs, nullptr);
|
|
if (m_Funcs == nullptr) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Failed to initialize CUDA library");
|
|
return;
|
|
}
|
|
}
|
|
|
|
CUDAGLInteropHelper::~CUDAGLInteropHelper()
|
|
{
|
|
unregisterTextures();
|
|
|
|
if (m_Funcs != nullptr) {
|
|
cuda_free_functions(&m_Funcs);
|
|
}
|
|
}
|
|
|
|
bool CUDAGLInteropHelper::registerBoundTextures()
|
|
{
|
|
int err;
|
|
|
|
if (m_Funcs == nullptr) {
|
|
// Already logged in constructor
|
|
return false;
|
|
}
|
|
|
|
// Push FFmpeg's CUDA context to use for our CUDA operations
|
|
err = m_Funcs->cuCtxPushCurrent(m_Context->cuda_ctx);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuCtxPushCurrent() failed: %d", err);
|
|
return false;
|
|
}
|
|
|
|
// Register each plane as a separate resource
|
|
for (int i = 0; i < NV12_PLANES; i++) {
|
|
GLint tex;
|
|
|
|
// Get the ID of this plane's texture
|
|
glActiveTexture(GL_TEXTURE0 + i);
|
|
glGetIntegerv(GL_TEXTURE_BINDING_2D, &tex);
|
|
|
|
// Register it with CUDA
|
|
err = m_Funcs->cuGraphicsGLRegisterImage(&m_Resources[i], tex, GL_TEXTURE_2D, CU_GRAPHICS_REGISTER_FLAGS_WRITE_DISCARD);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuGraphicsGLRegisterImage() failed: %d", err);
|
|
m_Resources[i] = 0;
|
|
unregisterTextures();
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
{
|
|
CUcontext dummy;
|
|
m_Funcs->cuCtxPopCurrent(&dummy);
|
|
}
|
|
return err == CUDA_SUCCESS;
|
|
}
|
|
|
|
void CUDAGLInteropHelper::unregisterTextures()
|
|
{
|
|
int err;
|
|
|
|
if (m_Funcs == nullptr) {
|
|
// Already logged in constructor
|
|
return;
|
|
}
|
|
|
|
// Push FFmpeg's CUDA context to use for our CUDA operations
|
|
err = m_Funcs->cuCtxPushCurrent(m_Context->cuda_ctx);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuCtxPushCurrent() failed: %d", err);
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < NV12_PLANES; i++) {
|
|
if (m_Resources[i] != 0) {
|
|
m_Funcs->cuGraphicsUnregisterResource(m_Resources[i]);
|
|
m_Resources[i] = 0;
|
|
}
|
|
}
|
|
|
|
{
|
|
CUcontext dummy;
|
|
m_Funcs->cuCtxPopCurrent(&dummy);
|
|
}
|
|
}
|
|
|
|
bool CUDAGLInteropHelper::copyCudaFrameToTextures(AVFrame* frame)
|
|
{
|
|
int err;
|
|
|
|
if (m_Funcs == nullptr) {
|
|
// Already logged in constructor
|
|
return false;
|
|
}
|
|
|
|
// Push FFmpeg's CUDA context to use for our CUDA operations
|
|
err = m_Funcs->cuCtxPushCurrent(m_Context->cuda_ctx);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuCtxPushCurrent() failed: %d", err);
|
|
return false;
|
|
}
|
|
|
|
// Map our resources
|
|
err = m_Funcs->cuGraphicsMapResources(NV12_PLANES, m_Resources, m_Context->stream);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuGraphicsMapResources() failed: %d", err);
|
|
goto PopCtxExit;
|
|
}
|
|
|
|
for (int i = 0; i < NV12_PLANES; i++) {
|
|
CUarray cudaArray;
|
|
|
|
// Get a pointer to the mapped array for this plane
|
|
err = m_Funcs->cuGraphicsSubResourceGetMappedArray(&cudaArray, m_Resources[i], 0, 0);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuGraphicsSubResourceGetMappedArray() failed: %d", err);
|
|
goto UnmapExit;
|
|
}
|
|
|
|
// Do the copy
|
|
CUDA_MEMCPY2D cu2d = {
|
|
.srcMemoryType = CU_MEMORYTYPE_DEVICE,
|
|
.srcDevice = (CUdeviceptr)frame->data[i],
|
|
.srcPitch = (size_t)frame->linesize[i],
|
|
.dstMemoryType = CU_MEMORYTYPE_ARRAY,
|
|
.dstArray = cudaArray,
|
|
.dstPitch = (size_t)frame->width >> i,
|
|
.WidthInBytes = (size_t)frame->width,
|
|
.Height = (size_t)frame->height >> i
|
|
};
|
|
err = m_Funcs->cuMemcpy2D(&cu2d);
|
|
if (err != CUDA_SUCCESS) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "cuMemcpy2D() failed: %d", err);
|
|
goto UnmapExit;
|
|
}
|
|
}
|
|
|
|
UnmapExit:
|
|
m_Funcs->cuGraphicsUnmapResources(NV12_PLANES, m_Resources, m_Context->stream);
|
|
PopCtxExit:
|
|
{
|
|
CUcontext dummy;
|
|
m_Funcs->cuCtxPopCurrent(&dummy);
|
|
}
|
|
return err == CUDA_SUCCESS;
|
|
}
|