mirror of
https://github.com/moonlight-stream/moonlight-qt
synced 2024-12-16 14:22:28 +00:00
442 lines
13 KiB
C++
442 lines
13 KiB
C++
#include "drm.h"
|
|
|
|
extern "C" {
|
|
#include <libavutil/hwcontext_drm.h>
|
|
}
|
|
|
|
#include <libdrm/drm_fourcc.h>
|
|
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
|
|
#include "streaming/streamutils.h"
|
|
#include "streaming/session.h"
|
|
|
|
#include <Limelight.h>
|
|
|
|
#ifdef HAVE_EGL
|
|
#include <SDL_egl.h>
|
|
#endif
|
|
|
|
#include <SDL_syswm.h>
|
|
|
|
DrmRenderer::DrmRenderer()
|
|
: m_HwContext(nullptr),
|
|
m_DrmFd(-1),
|
|
m_SdlOwnsDrmFd(false),
|
|
m_SupportsDirectRendering(false),
|
|
m_CrtcId(0),
|
|
m_PlaneId(0),
|
|
m_CurrentFbId(0)
|
|
{
|
|
}
|
|
|
|
DrmRenderer::~DrmRenderer()
|
|
{
|
|
if (m_CurrentFbId != 0) {
|
|
drmModeRmFB(m_DrmFd, m_CurrentFbId);
|
|
}
|
|
|
|
if (m_HwContext != nullptr) {
|
|
av_buffer_unref(&m_HwContext);
|
|
}
|
|
|
|
if (!m_SdlOwnsDrmFd && m_DrmFd != -1) {
|
|
close(m_DrmFd);
|
|
}
|
|
}
|
|
|
|
bool DrmRenderer::prepareDecoderContext(AVCodecContext* context, AVDictionary**)
|
|
{
|
|
context->hw_device_ctx = av_buffer_ref(m_HwContext);
|
|
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Using DRM renderer");
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DrmRenderer::initialize(PDECODER_PARAMETERS params)
|
|
{
|
|
int i;
|
|
|
|
#if SDL_VERSION_ATLEAST(2, 0, 15)
|
|
SDL_SysWMinfo info;
|
|
|
|
SDL_VERSION(&info.version);
|
|
|
|
if (!SDL_GetWindowWMInfo(params->window, &info)) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"SDL_GetWindowWMInfo() failed: %s",
|
|
SDL_GetError());
|
|
return false;
|
|
}
|
|
|
|
if (info.subsystem == SDL_SYSWM_KMSDRM) {
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Sharing DRM FD with SDL");
|
|
|
|
SDL_assert(info.info.kmsdrm.drm_fd >= 0);
|
|
m_DrmFd = info.info.kmsdrm.drm_fd;
|
|
m_SdlOwnsDrmFd = true;
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
const char* device = SDL_getenv("DRM_DEV");
|
|
|
|
if (device == nullptr) {
|
|
device = "/dev/dri/card0";
|
|
}
|
|
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Opening DRM device: %s",
|
|
device);
|
|
|
|
m_DrmFd = open(device, O_RDWR | O_CLOEXEC);
|
|
if (m_DrmFd < 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Failed to open DRM device: %d",
|
|
errno);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Create the device context first because it is needed whether we can
|
|
// actually use direct rendering or not.
|
|
m_HwContext = av_hwdevice_ctx_alloc(AV_HWDEVICE_TYPE_DRM);
|
|
if (m_HwContext == nullptr) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"av_hwdevice_ctx_alloc(DRM) failed");
|
|
return false;
|
|
}
|
|
|
|
AVHWDeviceContext* deviceContext = (AVHWDeviceContext*)m_HwContext->data;
|
|
AVDRMDeviceContext* drmDeviceContext = (AVDRMDeviceContext*)deviceContext->hwctx;
|
|
|
|
drmDeviceContext->fd = m_DrmFd;
|
|
|
|
int err = av_hwdevice_ctx_init(m_HwContext);
|
|
if (err < 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"av_hwdevice_ctx_init(DRM) failed: %d",
|
|
err);
|
|
return false;
|
|
}
|
|
|
|
#ifdef HAVE_EGL
|
|
// Still return true if we fail to initialize DRM direct rendering
|
|
// stuff, since we have EGL that we can use for indirect rendering.
|
|
const bool DIRECT_RENDERING_INIT_FAILED = true;
|
|
#else
|
|
// Fail if we can't initialize direct rendering and we don't have EGL.
|
|
const bool DIRECT_RENDERING_INIT_FAILED = false;
|
|
#endif
|
|
|
|
drmModeRes* resources = drmModeGetResources(m_DrmFd);
|
|
if (resources == nullptr) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"drmModeGetResources() failed: %d",
|
|
errno);
|
|
return DIRECT_RENDERING_INIT_FAILED;
|
|
}
|
|
|
|
// Look for a connected connector and get the associated encoder
|
|
uint32_t encoderId = 0;
|
|
for (i = 0; i < resources->count_connectors && encoderId == 0; i++) {
|
|
drmModeConnector* connector = drmModeGetConnector(m_DrmFd, resources->connectors[i]);
|
|
if (connector != nullptr) {
|
|
if (connector->connection == DRM_MODE_CONNECTED && connector->count_modes > 0) {
|
|
encoderId = connector->encoder_id;
|
|
}
|
|
|
|
drmModeFreeConnector(connector);
|
|
}
|
|
}
|
|
|
|
if (encoderId == 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"No connected displays found!");
|
|
drmModeFreeResources(resources);
|
|
return DIRECT_RENDERING_INIT_FAILED;
|
|
}
|
|
|
|
// Now find the CRTC from the encoder
|
|
m_CrtcId = 0;
|
|
for (i = 0; i < resources->count_encoders && m_CrtcId == 0; i++) {
|
|
drmModeEncoder* encoder = drmModeGetEncoder(m_DrmFd, resources->encoders[i]);
|
|
if (encoder != nullptr) {
|
|
if (encoder->encoder_id == encoderId) {
|
|
m_CrtcId = encoder->crtc_id;
|
|
}
|
|
|
|
drmModeFreeEncoder(encoder);
|
|
}
|
|
}
|
|
|
|
if (m_CrtcId == 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"DRM encoder not found!");
|
|
drmModeFreeResources(resources);
|
|
return DIRECT_RENDERING_INIT_FAILED;
|
|
}
|
|
|
|
int crtcIndex = -1;
|
|
for (int i = 0; i < resources->count_crtcs; i++) {
|
|
if (resources->crtcs[i] == m_CrtcId) {
|
|
drmModeCrtc* crtc = drmModeGetCrtc(m_DrmFd, resources->crtcs[i]);
|
|
crtcIndex = i;
|
|
m_OutputRect.x = m_OutputRect.y = 0;
|
|
m_OutputRect.w = crtc->width;
|
|
m_OutputRect.h = crtc->height;
|
|
drmModeFreeCrtc(crtc);
|
|
break;
|
|
}
|
|
}
|
|
|
|
drmModeFreeResources(resources);
|
|
|
|
if (crtcIndex == -1) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Failed to get CRTC!");
|
|
return DIRECT_RENDERING_INIT_FAILED;
|
|
}
|
|
|
|
drmSetClientCap(m_DrmFd, DRM_CLIENT_CAP_UNIVERSAL_PLANES, 1);
|
|
|
|
drmModePlaneRes* planeRes = drmModeGetPlaneResources(m_DrmFd);
|
|
if (planeRes == nullptr) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"drmGetPlaneResources() failed: %d",
|
|
errno);
|
|
return DIRECT_RENDERING_INIT_FAILED;
|
|
}
|
|
|
|
// Find an NV12 overlay plane to render on
|
|
m_PlaneId = 0;
|
|
for (uint32_t i = 0; i < planeRes->count_planes && m_PlaneId == 0; i++) {
|
|
drmModePlane* plane = drmModeGetPlane(m_DrmFd, planeRes->planes[i]);
|
|
if (plane != nullptr) {
|
|
bool matchingFormat = false;
|
|
for (uint32_t j = 0; j < plane->count_formats; j++) {
|
|
if (plane->formats[j] == DRM_FORMAT_NV12) {
|
|
matchingFormat = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (matchingFormat == false) {
|
|
drmModeFreePlane(plane);
|
|
continue;
|
|
}
|
|
|
|
if ((plane->possible_crtcs & (1 << crtcIndex)) && plane->crtc_id == 0) {
|
|
drmModeObjectPropertiesPtr props = drmModeObjectGetProperties(m_DrmFd, planeRes->planes[i], DRM_MODE_OBJECT_PLANE);
|
|
if (props != nullptr) {
|
|
for (uint32_t j = 0; j < props->count_props && m_PlaneId == 0; j++) {
|
|
drmModePropertyPtr prop = drmModeGetProperty(m_DrmFd, props->props[j]);
|
|
if (prop != nullptr) {
|
|
if (!strcmp(prop->name, "type") && props->prop_values[j] == DRM_PLANE_TYPE_OVERLAY) {
|
|
m_PlaneId = plane->plane_id;
|
|
}
|
|
|
|
drmModeFreeProperty(prop);
|
|
}
|
|
}
|
|
|
|
drmModeFreeObjectProperties(props);
|
|
}
|
|
}
|
|
|
|
drmModeFreePlane(plane);
|
|
}
|
|
}
|
|
|
|
drmModeFreePlaneResources(planeRes);
|
|
|
|
if (m_PlaneId == 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Failed to find suitable NV12 overlay plane!");
|
|
return DIRECT_RENDERING_INIT_FAILED;
|
|
}
|
|
|
|
// If we got this far, we can do direct rendering via the DRM FD.
|
|
m_SupportsDirectRendering = true;
|
|
|
|
return true;
|
|
}
|
|
|
|
enum AVPixelFormat DrmRenderer::getPreferredPixelFormat(int)
|
|
{
|
|
// DRM PRIME buffers
|
|
return AV_PIX_FMT_DRM_PRIME;
|
|
}
|
|
|
|
int DrmRenderer::getRendererAttributes()
|
|
{
|
|
// This renderer can only draw in full-screen
|
|
return RENDERER_ATTRIBUTE_FULLSCREEN_ONLY;
|
|
}
|
|
|
|
void DrmRenderer::renderFrame(AVFrame* frame)
|
|
{
|
|
if (frame == nullptr) {
|
|
// End of stream - nothing to do for us
|
|
return;
|
|
}
|
|
|
|
AVDRMFrameDescriptor* drmFrame = (AVDRMFrameDescriptor*)frame->data[0];
|
|
int err;
|
|
uint32_t primeHandle;
|
|
uint32_t handles[4] = {};
|
|
uint32_t pitches[4] = {};
|
|
uint32_t offsets[4] = {};
|
|
|
|
SDL_Rect src, dst;
|
|
|
|
src.x = src.y = 0;
|
|
src.w = frame->width;
|
|
src.h = frame->height;
|
|
dst = m_OutputRect;
|
|
|
|
StreamUtils::scaleSourceToDestinationSurface(&src, &dst);
|
|
|
|
// Convert the FD in the AVDRMFrameDescriptor to a PRIME handle
|
|
// that can be used in drmModeAddFB2()
|
|
SDL_assert(drmFrame->nb_objects == 1);
|
|
err = drmPrimeFDToHandle(m_DrmFd, drmFrame->objects[0].fd, &primeHandle);
|
|
if (err < 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"drmPrimeFDToHandle() failed: %d",
|
|
errno);
|
|
return;
|
|
}
|
|
|
|
SDL_assert(drmFrame->nb_layers == 1);
|
|
SDL_assert(drmFrame->layers[0].nb_planes == 2);
|
|
for (int i = 0; i < drmFrame->layers[0].nb_planes; i++) {
|
|
handles[i] = primeHandle;
|
|
pitches[i] = drmFrame->layers[0].planes[i].pitch;
|
|
offsets[i] = drmFrame->layers[0].planes[i].offset;
|
|
}
|
|
|
|
// Remember the last FB object we created so we can free it
|
|
// when we are finished rendering this one (if successful).
|
|
uint32_t lastFbId = m_CurrentFbId;
|
|
|
|
// Create a frame buffer object from the PRIME buffer
|
|
err = drmModeAddFB2(m_DrmFd, frame->width, frame->height,
|
|
drmFrame->layers[0].format,
|
|
handles, pitches, offsets, &m_CurrentFbId, 0);
|
|
if (err < 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"drmModeAddFB2() failed: %d",
|
|
errno);
|
|
m_CurrentFbId = lastFbId;
|
|
return;
|
|
}
|
|
|
|
// Update the overlay
|
|
err = drmModeSetPlane(m_DrmFd, m_PlaneId, m_CrtcId, m_CurrentFbId, 0,
|
|
dst.x, dst.y,
|
|
dst.w, dst.h,
|
|
0, 0,
|
|
frame->width << 16,
|
|
frame->height << 16);
|
|
if (err < 0) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"drmModeSetPlane() failed: %d",
|
|
errno);
|
|
drmModeRmFB(m_DrmFd, m_CurrentFbId);
|
|
m_CurrentFbId = lastFbId;
|
|
return;
|
|
}
|
|
|
|
// Free the previous FB object which has now been superseded
|
|
drmModeRmFB(m_DrmFd, lastFbId);
|
|
}
|
|
|
|
bool DrmRenderer::needsTestFrame()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool DrmRenderer::isDirectRenderingSupported()
|
|
{
|
|
return m_SupportsDirectRendering;
|
|
}
|
|
|
|
#ifdef HAVE_EGL
|
|
|
|
bool DrmRenderer::canExportEGL() {
|
|
if (qgetenv("DRM_FORCE_DIRECT") == "1") {
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Using direct rendering due to environment variable");
|
|
return false;
|
|
}
|
|
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"DRM backend supports exporting EGLImage");
|
|
return true;
|
|
}
|
|
|
|
bool DrmRenderer::initializeEGL(EGLDisplay,
|
|
const EGLExtensions &ext) {
|
|
if (!ext.isSupported("EGL_EXT_image_dma_buf_import")) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"DRM-EGL: DMABUF unsupported");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
ssize_t DrmRenderer::exportEGLImages(AVFrame *frame, EGLDisplay dpy,
|
|
EGLImage images[EGL_MAX_PLANES]) {
|
|
ssize_t count = 0;
|
|
AVDRMFrameDescriptor* drmFrame = (AVDRMFrameDescriptor*)frame->data[0];
|
|
|
|
memset(images, 0, sizeof(EGLImage) * EGL_MAX_PLANES);
|
|
|
|
SDL_assert(drmFrame->nb_objects == 1);
|
|
SDL_assert(drmFrame->nb_layers == 1);
|
|
SDL_assert(drmFrame->layers[0].nb_planes == 2);
|
|
|
|
for (int i = 0; i < drmFrame->layers[0].nb_planes; ++i) {
|
|
const auto &plane = drmFrame->layers[0].planes[i];
|
|
const auto &object = drmFrame->objects[plane.object_index];
|
|
|
|
EGLAttrib attribs[17] = {
|
|
EGL_LINUX_DRM_FOURCC_EXT, i == 0 ? DRM_FORMAT_R8 : DRM_FORMAT_GR88,
|
|
EGL_WIDTH, i == 0 ? frame->width : frame->width / 2,
|
|
EGL_HEIGHT, i == 0 ? frame->height : frame->height / 2,
|
|
EGL_DMA_BUF_PLANE0_FD_EXT, object.fd,
|
|
EGL_DMA_BUF_PLANE0_OFFSET_EXT, (EGLint)plane.offset,
|
|
EGL_DMA_BUF_PLANE0_PITCH_EXT, (EGLint)plane.pitch,
|
|
EGL_NONE,
|
|
};
|
|
images[i] = eglCreateImage(dpy, EGL_NO_CONTEXT,
|
|
EGL_LINUX_DMA_BUF_EXT,
|
|
nullptr, attribs);
|
|
if (!images[i]) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"eglCreateImage() Failed: %d", eglGetError());
|
|
goto fail;
|
|
}
|
|
++count;
|
|
}
|
|
|
|
return count;
|
|
|
|
fail:
|
|
freeEGLImages(dpy, images);
|
|
return -1;
|
|
}
|
|
|
|
void DrmRenderer::freeEGLImages(EGLDisplay dpy, EGLImage images[EGL_MAX_PLANES]) {
|
|
for (size_t i = 0; i < EGL_MAX_PLANES; ++i) {
|
|
eglDestroyImage(dpy, images[i]);
|
|
}
|
|
}
|
|
|
|
#endif
|