Files
UltraGrid/src/video_display/drm.cpp

1004 lines
33 KiB
C++

#include "config.h"
#include "config_unix.h"
#include "config_win32.h"
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <xf86drm.h>
#include <xf86drmMode.h>
#include <drm_fourcc.h>
#include <memory>
#include <vector>
#include <condition_variable>
#include <queue>
#include <algorithm>
#include "debug.h"
#include "host.h"
#include "lib_common.h"
#include "pixfmt_conv.h"
#include "utils/color_out.h"
#include "utils/macros.h"
#include "utils/misc.h"
#include "utils/text.h"
#include "utils/string_view_utils.hpp"
#include "video.h"
#include "video_codec.h"
#include "video_display.h"
#include "hwaccel_drm.h"
#define MOD_NAME "[drm] "
namespace{
struct frame_deleter{ void operator()(video_frame *f){ vf_free(f); } };
using frame_uniq = std::unique_ptr<video_frame, frame_deleter>;
struct drm_connector_deleter{ void operator()(drmModeConnectorPtr c) { drmModeFreeConnector(c); } };
using Drm_connector_uniq = std::unique_ptr<drmModeConnector, drm_connector_deleter>;
struct drm_resources_deleter{ void operator()(drmModeResPtr r) { drmModeFreeResources(r); } };
using Drm_res_uniq = std::unique_ptr<drmModeRes, drm_resources_deleter>;
struct drm_encoder_deleter{ void operator()(drmModeEncoderPtr e) { drmModeFreeEncoder(e); } };
using Drm_encoder_uniq = std::unique_ptr<drmModeEncoder, drm_encoder_deleter>;
struct drm_crtc_deleter{ void operator()(drmModeCrtcPtr c) { drmModeFreeCrtc(c); } };
using Drm_crtc_uniq = std::unique_ptr<drmModeCrtc, drm_crtc_deleter>;
struct drm_plane_res_deleter{ void operator()(drmModePlaneResPtr r) { drmModeFreePlaneResources(r); } };
using Drm_plane_res_uniq = std::unique_ptr<drmModePlaneRes, drm_plane_res_deleter>;
struct drm_plane_deleter{ void operator()(drmModePlanePtr p) { drmModeFreePlane(p); } };
using Drm_plane_uniq = std::unique_ptr<drmModePlane, drm_plane_deleter>;
struct drm_obj_props_deleter{ void operator()(drmModeObjectPropertiesPtr p) { drmModeFreeObjectProperties(p); } };
using Drm_object_properties_uniq = std::unique_ptr<drmModeObjectProperties, drm_obj_props_deleter>;
struct drm_prop_deleter{ void operator()(drmModePropertyPtr p) { drmModeFreeProperty(p); } };
using Drm_property_uniq = std::unique_ptr<drmModePropertyRes, drm_prop_deleter>;
class Fd_uniq{
public:
Fd_uniq() = default;
Fd_uniq(int fd) : fd(fd) { }
Fd_uniq(const Fd_uniq&) = delete;
Fd_uniq(Fd_uniq&& o){
std::swap(fd, o.fd);
}
~Fd_uniq(){
destruct();
}
Fd_uniq& operator=(const Fd_uniq&) = delete;
Fd_uniq& operator=(Fd_uniq&& o){
std::swap(fd, o.fd);
return *this;
}
operator bool() const { return fd > 0; }
int get() { return fd; }
void reset(int fd){
destruct();
this->fd = fd;
}
private:
void destruct(){
if(fd > 0)
close(fd);
fd = -1;
}
int fd = -1;
};
class MemoryMapping{
public:
MemoryMapping() = default;
~MemoryMapping() {
if(!valid())
return;
munmap(ptr, mapped_size);
}
MemoryMapping(const MemoryMapping&) = delete;
MemoryMapping& operator=(const MemoryMapping&) = delete;
MemoryMapping(MemoryMapping&& o) noexcept{
std::swap(ptr, o.ptr);
std::swap(mapped_size, o.mapped_size);
}
MemoryMapping& operator=(MemoryMapping&& o) noexcept{
std::swap(ptr, o.ptr);
std::swap(mapped_size, o.mapped_size);
return *this;
}
static MemoryMapping create(void *addr, size_t len, int prot, int flags, int fildes, off_t off){
MemoryMapping res;
res.ptr = mmap(addr, len, prot, flags, fildes, off);
if(res.ptr != MAP_FAILED)
res.mapped_size = len;
return res;
};
bool valid() const { return ptr != MAP_FAILED; }
void *get() const { return ptr; }
size_t size() const { return mapped_size; }
private:
void *ptr = MAP_FAILED;
size_t mapped_size = 0;
};
template<typename T, class Deleter>
class Uniq_wrapper{
public:
Uniq_wrapper() = default;
Uniq_wrapper(T val): val(val) { }
~Uniq_wrapper(){
Deleter()(val);
}
Uniq_wrapper(const Uniq_wrapper&) = delete;
Uniq_wrapper& operator=(const Uniq_wrapper&) = delete;
Uniq_wrapper(Uniq_wrapper&& o){
std::swap(val, o.val);
}
Uniq_wrapper& operator=(Uniq_wrapper&& o){
std::swap(val, o.val);
return *this;
}
T& get() { return val; }
private:
T val = {};
};
struct Fb_handle{
int dri_fd = -1;
uint32_t handle;
};
struct Fb_handle_deleter {
void operator()(Fb_handle h){
if(h.dri_fd < 0)
return;
drm_mode_destroy_dumb destroy_info = {};
destroy_info.handle = h.handle;
drmIoctl(h.dri_fd, DRM_IOCTL_MODE_DESTROY_DUMB, &destroy_info);
}
};
using Fb_handle_uniq = Uniq_wrapper<Fb_handle, Fb_handle_deleter>;
struct Fb_id{
int dri_fd = -1;
uint32_t id;
};
struct Fb_id_deleter {
void operator()(Fb_id i){
if(i.dri_fd < 0)
return;
drmModeRmFB(i.dri_fd, i.id);
}
};
using Fb_id_uniq = Uniq_wrapper<Fb_id, Fb_id_deleter>;
class Gem_handle_manager{
/* The handles returned from drmPrimeFDToHandle() must not be
* double free'd. Two frames can apparently also point to the
* same prime buffer, so we really need to reference count it
* ourselves.
*/
private:
void add_ref(uint32_t handle){
handle_map[handle]++;
};
void unref(uint32_t handle){
int refs = handle_map[handle]--;
assert(refs >= 0 && "Unref called on invalid handle");
if(refs == 0){
drmIoctl(dri_fd, DRM_IOCTL_GEM_CLOSE, &handle);
handle_map.erase(handle);
}
}
int dri_fd = -1;
std::map<uint32_t, int> handle_map;
public:
Gem_handle_manager(int dri_fd): dri_fd(dri_fd) { }
class Handle{
public:
Handle() = default;
~Handle(){
if(ctx)
ctx->unref(handle);
}
Handle(const Handle&) = delete;
Handle& operator=(const Handle&) = delete;
Handle(Handle&& o){
std::swap(handle, o.handle);
std::swap(ctx, o.ctx);
}
Handle& operator=(Handle&& o){
std::swap(handle, o.handle);
std::swap(ctx, o.ctx);
return *this;
}
uint32_t get() const {
return handle;
}
private:
friend Gem_handle_manager;
Handle(uint32_t handle, Gem_handle_manager *ctx): handle(handle), ctx(ctx){
ctx->add_ref(handle);
}
uint32_t handle = 0;
Gem_handle_manager *ctx = nullptr;
};
Handle get_handle(int fd){
uint32_t handle = 0;
int res = drmPrimeFDToHandle(dri_fd, fd, &handle);
if(res < 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get a GEM handle from prime fd %d\n", fd);
return {};
}
return Handle(handle, this);
}
};
} //anon namespace
//
struct Drm_state {
Fd_uniq dri_fd;
std::unique_ptr<Gem_handle_manager> gem_manager;
Drm_res_uniq res;
Drm_connector_uniq connector;
Drm_encoder_uniq encoder;
Drm_crtc_uniq crtc;
int crtc_index = -1;
std::set<uint32_t> supported_drm_formats;
bool prime_support = false;
drmModeModeInfoPtr mode_info;
};
struct Framebuffer{
uint32_t width = 0;
uint32_t height = 0;
uint32_t pitch = 0;
size_t size = 0;
Fb_handle_uniq handle;
uint32_t pix_fmt = 0;
Fb_id_uniq id;
MemoryMapping map;
};
struct Drm_prime_fb{
frame_uniq frame;
Gem_handle_manager::Handle gem_objects[4] = {};
Fb_id_uniq id;
};
struct drm_display_state {
std::string cfg;
std::string device_path;
std::string req_connector;
int req_width = -1;
int req_height = -1;
Drm_state drm;
Framebuffer splashscreen;
Framebuffer back_buffer;
Framebuffer front_buffer;
Drm_prime_fb drm_prime_fb;
video_desc desc;
frame_uniq frame;
std::mutex mut;
std::condition_variable frame_consumed_cv;
std::queue<frame_uniq> queue;
std::vector<frame_uniq> free_frames;
};
static std::string get_connector_str(int type, uint32_t id){
std::string res;
switch(type){
case DRM_MODE_CONNECTOR_VGA: res = "VGA"; break;
case DRM_MODE_CONNECTOR_HDMIA: res = "HDMI-A"; break;
case DRM_MODE_CONNECTOR_HDMIB: res = "HDMI-B"; break;
case DRM_MODE_CONNECTOR_DisplayPort: res = "DP"; break;
case DRM_MODE_CONNECTOR_eDP: res = "eDP"; break;
case DRM_MODE_CONNECTOR_DVII: res = "DVI-I"; break;
case DRM_MODE_CONNECTOR_DVID: res = "DVI-D"; break;
case DRM_MODE_CONNECTOR_DVIA: res = "DVI-A"; break;
#ifdef DRM_MODE_CONNECTOR_USB
case DRM_MODE_CONNECTOR_USB: res = "USB"; break;
#endif
default:
res = std::to_string(type);
}
res += "-";
res += std::to_string(id);
return res;
}
static void print_drm_driver_info(drm_display_state *s){
drmVersionPtr version = drmGetVersion(s->drm.dri_fd.get());
if(!version){
log_msg(LOG_LEVEL_WARNING, MOD_NAME "Failed to get DRM driver version\n");
return;
}
log_msg(LOG_LEVEL_INFO, MOD_NAME "DRM version: %d.%d.%d (%s), Driver: %s\n",
version->version_major,
version->version_minor,
version->version_patchlevel,
version->date,
version->name);
drmFreeVersion(version);
}
static Fd_uniq open_dri(drm_display_state *s){
auto do_open = [](const char *path) -> Fd_uniq {
int fd = open(path, O_RDWR);
if(fd < 0){
if(errno != ENOENT)
log_msg(LOG_LEVEL_INFO, MOD_NAME "Failed to open %s (%s)\n", path, strerror(errno));
return {};
}
Fd_uniq ret(fd);
Drm_res_uniq resources(drmModeGetResources(fd));
if(!resources){
log_msg(LOG_LEVEL_INFO, MOD_NAME "Failed to get resources on %s (%s)\n", path, strerror(errno));
return {};
}
uint64_t dumb_support = false;
int res = 0;
res = drmGetCap(fd, DRM_CAP_DUMB_BUFFER, &dumb_support);
if(res < 0 || !dumb_support){
log_msg(LOG_LEVEL_WARNING, MOD_NAME "%s does not support dumb buffers\n", path);
return {};
}
log_msg(LOG_LEVEL_INFO, MOD_NAME "Opened %s DRI device\n", path);
return ret;
};
if(!s->device_path.empty()){
auto ret = do_open(s->device_path.c_str());
if(!ret)
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to open specified device (%s)\n", s->device_path.c_str());
return ret;
}
char buf[256] = {};
const int max_index = 32;
for(int i = 0; i < max_index; i++){
snprintf(buf, sizeof(buf), DRM_DEV_NAME, DRM_DIR_NAME, i);
auto ret = do_open(buf);
if(ret)
return ret;
}
log_msg(LOG_LEVEL_ERROR, MOD_NAME "No suitable DRI device found\n");
return {};
}
static int64_t get_property(int dri, drmModeObjectPropertiesPtr props, std::string_view name){
for(unsigned i = 0; i < props->count_props; i++){
Drm_property_uniq prop(drmModeGetProperty(dri, props->props[i]));
if(!prop)
continue;
if(prop->name == name)
return props->prop_values[i];
}
return -1;
}
static bool probe_drm_formats(drm_display_state *s){
int dri = s->drm.dri_fd.get();
Drm_plane_res_uniq plane_res(drmModeGetPlaneResources(dri));
if(!plane_res){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get plane resources (%s)\n", strerror(errno));
return false;
}
Drm_plane_uniq primary_plane;
for(unsigned i = 0; i < plane_res->count_planes; i++){
Drm_plane_uniq plane(drmModeGetPlane(dri, plane_res->planes[i]));
if(!(plane->possible_crtcs & (1 << s->drm.crtc_index)))
continue;
Drm_object_properties_uniq props(drmModeObjectGetProperties(dri, plane_res->planes[i], DRM_MODE_OBJECT_PLANE));
if(!props){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get plane props (%s)\n", strerror(errno));
}
if(get_property(dri, props.get(), "type") == DRM_PLANE_TYPE_PRIMARY){
primary_plane = std::move(plane);
break;
}
}
if(!primary_plane){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to find the primary plane\n");
return false;
}
for(unsigned i = 0; i < primary_plane->count_formats; i++){
s->drm.supported_drm_formats.insert(primary_plane->formats[i]);
}
return true;
}
static std::vector<Drm_connector_uniq> get_connectors(drm_display_state *s){
std::vector<Drm_connector_uniq> res;
for(int i = 0; i < s->drm.res->count_connectors; i++){
Drm_connector_uniq conn(drmModeGetConnector(s->drm.dri_fd.get(), s->drm.res->connectors[i]));
if(!conn)
continue;
res.push_back(std::move(conn));
}
return res;
}
static bool init_drm_state(drm_display_state *s){
s->drm.dri_fd = open_dri(s);
int dri = s->drm.dri_fd.get();
if(dri < 0){
return false;
}
print_drm_driver_info(s);
int res = 0;
res = drmSetClientCap(dri, DRM_CLIENT_CAP_UNIVERSAL_PLANES, 1);
if(res != 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to set universal planes capab\n");
return false;
}
s->drm.gem_manager = std::make_unique<Gem_handle_manager>(dri);
s->drm.res.reset(drmModeGetResources(dri));
if(!s->drm.res){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get DRI resources: %s\n", strerror(errno));
return false;
}
uint64_t prime_support = false;
res = drmGetCap(dri, DRM_CAP_PRIME, &prime_support);
if(res < 0 || !prime_support){
log_msg(LOG_LEVEL_WARNING, MOD_NAME "DRM device does not support PRIME buffers\n");
}
s->drm.prime_support = prime_support;
return true;
}
static const char *get_connector_status_str(drmModeConnection c){
switch(c){
case DRM_MODE_CONNECTED:
return " (Connected)";
case DRM_MODE_DISCONNECTED:
return " (Not connected)";
case DRM_MODE_UNKNOWNCONNECTION:
default:
return "";
}
}
static void print_connectors(drm_display_state *s){
color_printf("Connectors:\n");
auto connectors = get_connectors(s);
for(auto& connector : connectors){
color_printf("\t%s%s:\n", get_connector_str(connector->connector_type, connector->connector_type_id).c_str(),
get_connector_status_str(connector->connection));
for(int i = 0; i < connector->count_modes; i++){
auto mode_info = &connector->modes[i];
color_printf("\t\t%dx%d@%d (%s)%s\n", mode_info->hdisplay, mode_info->vdisplay, mode_info->vrefresh, mode_info->name,
mode_info->type & DRM_MODE_TYPE_PREFERRED ? " preferred" : "");
}
}
}
static bool setup_crtc(drm_display_state *s){
auto connectors = get_connectors(s);
for(auto& connector : connectors){
for(int i = 0; i < connector->count_modes; i++){
if(connector->modes[i].type & DRM_MODE_TYPE_PREFERRED){
s->drm.mode_info = &connector->modes[i];
s->drm.connector = std::move(connector);
break;
}
}
if(s->drm.mode_info){
break;
}
}
int dri = s->drm.dri_fd.get();
s->drm.encoder.reset(drmModeGetEncoder(dri, s->drm.connector->encoder_id));
if(!s->drm.encoder){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get encoder\n");
return false;
}
s->drm.crtc.reset(drmModeGetCrtc(dri, s->drm.encoder->crtc_id));
if(!s->drm.crtc){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get crtc\n");
return false;
}
for(int i = 0; i < s->drm.res->count_crtcs; i++){
if(s->drm.res->crtcs[i] == s->drm.crtc->crtc_id){
s->drm.crtc_index = i;
break;
}
}
if(!probe_drm_formats(s)){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to probe DRM supported formats\n");
return false;
}
int res = 0;
res = drmSetMaster(dri);
if(res != 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Unable to get DRM master. Is X11 or Wayland running?\n");
return false;
}
return true;
}
static Framebuffer create_dumb_fb(int dri, int width, int height, uint32_t pix_fmt){
Framebuffer buf;
//TODO: Currently pix_fmt is assumed to be single plane and 32 bpp
int res = 0;
{
Fb_handle handle;
drm_mode_create_dumb create_info = {};
create_info.width = width;
create_info.height = height;
create_info.bpp = 32;
res = drmIoctl(dri, DRM_IOCTL_MODE_CREATE_DUMB, &create_info);
if(res != 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to create a dumb framebuffer (%d)\n", res);
return {};
}
handle.handle = create_info.handle;
buf.width = width;
buf.height = height;
buf.pitch = create_info.pitch;
buf.size = create_info.size;
handle.dri_fd = dri;
buf.handle = Fb_handle_uniq(handle);
}
{
uint32_t handles[] = {buf.handle.get().handle, 0, 0, 0};
uint32_t pitches[] = {buf.pitch, 0, 0, 0};
uint32_t offsets[] = {0, 0, 0, 0};
Fb_id fb_id;
res = drmModeAddFB2(dri, width, height,
pix_fmt, handles, pitches, offsets, &fb_id.id, 0);
if(res != 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to add framebuffer (%d)\n", res);
return {};
}
fb_id.dri_fd = dri;
buf.id = Fb_id_uniq(fb_id);
}
log_msg(LOG_LEVEL_INFO, MOD_NAME "Created dumb buffer: pitch %u, handle: %u\n", buf.pitch, buf.handle.get().handle);
struct drm_mode_map_dumb map_info = {};
map_info.handle = buf.handle.get().handle;
res = drmIoctl(dri, DRM_IOCTL_MODE_MAP_DUMB, &map_info);
if(res != 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to get map info (%d)\n", res);
return {};
}
buf.map = MemoryMapping::create(0, buf.size, PROT_READ | PROT_WRITE, MAP_SHARED, dri, map_info.offset);
if(!buf.map.valid()){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to map buffer\n");
return {};
}
return buf;
}
static bool set_framebuffer(drm_display_state *s, uint32_t fb_id){
int res = 0;
res = drmModeSetCrtc(s->drm.dri_fd.get(), s->drm.crtc->crtc_id,
fb_id, 0, 0, &s->drm.connector->connector_id, 1, s->drm.mode_info);
if(res < 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to set crtc (%d)\n", res);
return false;
}
return true;
}
static void draw_splash(drm_display_state *s){
int res = 0;
res = drmModeSetCrtc(s->drm.dri_fd.get(), s->drm.crtc->crtc_id,
s->splashscreen.id.get().id, 0, 0, &s->drm.connector->connector_id, 1, s->drm.mode_info);
if(res < 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to set crtc (%d)\n", res);
}
}
static void draw_frame(Framebuffer *dst, video_frame *src, bool center = true){
auto dst_p = static_cast<char *>(dst->map.get());
auto src_p = static_cast<char *>(src->tiles[0].data);
auto src_w = src->tiles[0].width;
auto src_h = src->tiles[0].height;
unsigned x = 0;
unsigned y = 0;
if(center){
if(src_w < dst->width)
x = (dst->width - src_w) / 2;
if(src_h < dst->height)
y = (dst->height - src_h) / 2;
}
int width = std::min(src_w, dst->width - x);
int height = std::min(src_h, dst->height - y);
auto src_pitch = vc_get_linesize(src_w, src->color_spec);
auto linesize = vc_get_size(width, src->color_spec);
dst_p += dst->pitch * y;
dst_p += vc_get_size(x, src->color_spec);
for(int y = 0; y < height; y++){
memcpy(dst_p, src_p, linesize);
dst_p += dst->pitch;
src_p += src_pitch;
}
}
static bool drm_format_supported(drm_display_state *s, uint32_t fmt){
return s->drm.supported_drm_formats.find(fmt) != s->drm.supported_drm_formats.end();
}
static Framebuffer get_splash_fb(drm_display_state *s, int width, int height){
frame_uniq splash_frame(get_splashscreen());
uint32_t pix_fmt;
if(drm_format_supported(s, DRM_FORMAT_XBGR8888)){
pix_fmt = DRM_FORMAT_XBGR8888;
} else {
//XRGB_8888 should be always present
//red and blue will be swapped, but for splash it doesn't matter much
pix_fmt = DRM_FORMAT_XRGB8888;
}
auto fb = create_dumb_fb(s->drm.dri_fd.get(), width, height, pix_fmt);
draw_frame(&fb, splash_frame.get(), true);
return fb;
}
static void *display_drm_init(struct module *parent, const char *cfg, unsigned int flags)
{
UNUSED(parent), UNUSED(flags);
auto s = std::make_unique<drm_display_state>();
if(cfg)
s->cfg = cfg;
bool help_requested = false;
std::string_view sv_cfg(s->cfg);
while(!sv_cfg.empty()){
auto token = tokenize(sv_cfg, ':');
auto key = tokenize(token, '=');
auto val = tokenize(token, '=');
if(key == "help"){
help_requested = true;
} else if(key == "dev"){
s->device_path = val;
}
}
if(!init_drm_state(s.get())){
return nullptr;
}
if(help_requested){
color_printf("DRM display\n");
color_printf("Usage: drm[:dev=<path>]\n");
color_printf("\n");
print_connectors(s.get());
return INIT_NOERR;
}
if(!setup_crtc(s.get())){
return nullptr;
}
s->splashscreen = get_splash_fb(s.get(), s->drm.mode_info->hdisplay, s->drm.mode_info->vdisplay);
draw_splash(s.get());
return s.release();
}
static void display_drm_done(void *state)
{
auto s = std::unique_ptr<drm_display_state>(static_cast<drm_display_state *>(state));
int res = 0;
res = drmModeSetCrtc(s->drm.dri_fd.get(), s->drm.crtc->crtc_id, s->drm.crtc->buffer_id,
s->drm.crtc->x , s->drm.crtc->y, &s->drm.connector->connector_id, 1, &s->drm.crtc->mode);
if(res < 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to restore original crtc (%d)\n", res);
}
}
static struct video_frame *display_drm_getf(void *state)
{
auto s = static_cast<drm_display_state *>(state);
while(!s->free_frames.empty()){
auto frame = std::move(s->free_frames.back());
s->free_frames.pop_back();
if(video_desc_eq(video_desc_from_frame(frame.get()), s->desc)){
return frame.release();
}
}
return vf_alloc_desc_data(s->desc);
}
static bool swap_buffers(drm_display_state *s){
std::swap(s->front_buffer, s->back_buffer);
return set_framebuffer(s, s->front_buffer.id.get().id);
}
static Drm_prime_fb drm_fb_from_frame(drm_display_state *s, frame_uniq frame){
assert(frame->color_spec == DRM_PRIME);
Drm_prime_fb fb;
fb.frame = std::move(frame);
auto drm_frame = (drm_prime_frame *) fb.frame->tiles[0].data;
for(int i = 0; i < drm_frame->fd_count; i++){
fb.gem_objects[i] = s->drm.gem_manager->get_handle(drm_frame->dmabuf_fds[i]);
}
uint32_t handles[4] = {};
for(int i = 0; i < drm_frame->planes; i++){
handles[i] = fb.gem_objects[drm_frame->fd_indices[i]].get();
}
int res = 0;
Fb_id fb_id;
res = drmModeAddFB2WithModifiers(s->drm.dri_fd.get(), fb.frame->tiles[0].width, fb.frame->tiles[0].height, drm_frame->drm_format,
handles, drm_frame->pitches, drm_frame->offsets, drm_frame->modifiers, &fb_id.id, DRM_MODE_FB_MODIFIERS);
if(res != 0){
log_msg(LOG_LEVEL_ERROR, MOD_NAME "Failed to add FB\n");
}
fb_id.dri_fd = s->drm.dri_fd.get();
fb.id = Fb_id_uniq(fb_id);
return fb;
}
static void recycle_frame(drm_display_state *s, frame_uniq& frame){
vf_recycle(frame.get());
s->free_frames.push_back(std::move(frame));
}
static void recycle_prime_frame(drm_display_state *s, Drm_prime_fb& fb){
frame_uniq frame = std::move(fb.frame);
fb = {};
if(frame){
vf_recycle(frame.get());
s->free_frames.push_back(std::move(frame));
}
}
static bool display_drm_putf(void *state, struct video_frame *f, long long flags)
{
frame_uniq frame(f);
if (!frame) {
return true;
}
auto s = static_cast<drm_display_state *>(state);
if(flags == PUTF_DISCARD){
recycle_frame(s, frame);
return true;
}
if(frame->color_spec == DRM_PRIME){
Drm_prime_fb fb = drm_fb_from_frame(s, std::move(frame));
if(!set_framebuffer(s, fb.id.get().id)){
return false;
}
recycle_prime_frame(s, s->drm_prime_fb);
s->drm_prime_fb = std::move(fb);
return true;
}
draw_frame(&s->back_buffer, frame.get());
recycle_frame(s, frame);
swap_buffers(s);
return true;
}
static bool get_codecs(drm_display_state *s, void *val, size_t *len){
std::vector<codec_t> out;
if(s->drm.prime_support)
out.push_back(DRM_PRIME);
if(s->drm.supported_drm_formats.find(DRM_FORMAT_UYVY) != s->drm.supported_drm_formats.end())
out.push_back(UYVY);
out.push_back(RGBA);
size_t length = sizeof(codec_t) * out.size();
if(*len < length) {
return false;
}
*len = length;
memcpy(val, out.data(), *len);
return true;
}
static bool display_drm_get_property(void *state, int property, void *val, size_t *len)
{
auto s = static_cast<drm_display_state *>(state);
int rgb_shift[] = {0, 8, 16};
if(!drm_format_supported(s, DRM_FORMAT_XBGR8888)){
std::swap(rgb_shift[0], rgb_shift[2]);
}
switch (property) {
case DISPLAY_PROPERTY_CODECS:
return get_codecs(s, val, len);
case DISPLAY_PROPERTY_RGB_SHIFT:
if(sizeof(rgb_shift) > *len) {
return false;
}
memcpy(val, rgb_shift, sizeof(rgb_shift));
*len = sizeof(rgb_shift);
break;
default:
return false;
}
return true;
}
static bool display_drm_reconfigure(void *state, struct video_desc desc)
{
auto s = static_cast<drm_display_state *>(state);
s->desc = desc;
s->frame.reset(vf_alloc_desc_data(desc));
uint32_t pix_fmt;
switch(desc.color_spec){
case RGBA:
pix_fmt = drm_format_supported(s, DRM_FORMAT_XBGR8888) ? DRM_FORMAT_XBGR8888 : DRM_FORMAT_XRGB8888;
break;
case UYVY:
pix_fmt = DRM_FORMAT_UYVY;
break;
case DRM_PRIME:
pix_fmt = 0; //UNUSED
/* We don't create dumb buffers in this case, we import framebuffers from video frames instead
*/
return true;
default:
return false;
}
if(desc.width > s->drm.mode_info->hdisplay || desc.height > s->drm.mode_info->vdisplay){
log_msg(LOG_LEVEL_WARNING, MOD_NAME "Video resolution is larger than framebuffer. Only part of video frames will be visible\n");
}
s->front_buffer = create_dumb_fb(s->drm.dri_fd.get(), s->drm.mode_info->hdisplay, s->drm.mode_info->vdisplay, pix_fmt);
s->back_buffer = create_dumb_fb(s->drm.dri_fd.get(), s->drm.mode_info->hdisplay, s->drm.mode_info->vdisplay, pix_fmt);
return true;
}
static void display_drm_probe(struct device_info **available_cards, int *count, void (**deleter)(void *)) {
UNUSED(deleter);
*available_cards = NULL;
*count = 0;
}
static const struct video_display_info display_drm_info = {
display_drm_probe,
display_drm_init,
NULL, // _run
display_drm_done,
display_drm_getf,
display_drm_putf,
display_drm_reconfigure,
display_drm_get_property,
NULL, // _put_audio_frame
NULL, // _reconfigure_audio
MOD_NAME,
};
REGISTER_MODULE(drm, &display_drm_info, LIBRARY_CLASS_VIDEO_DISPLAY, VIDEO_DISPLAY_ABI_VERSION);