drm: Implement hardware plane support
authorDerek Foreman <derekf@osg.samsung.com>
Thu, 29 Jun 2017 22:15:36 +0000 (17:15 -0500)
committerDerek Foreman <derekf@osg.samsung.com>
Thu, 29 Jun 2017 22:20:23 +0000 (17:20 -0500)
Adds the ability for drm to place native surfaces on harware planes.

ref 33b50238d2d54e77c23f42c00c6bfafc9a9a1080

src/modules/evas/engines/drm/evas_engine.c

index c665782..f5474ab 100644 (file)
@@ -1,10 +1,19 @@
 #include "evas_engine.h"
+#include "../software_generic/evas_native_common.h"
 
 typedef struct _Render_Engine
 {
    Render_Engine_Software_Generic generic;
+
+   int fd;
 } Render_Engine;
 
+struct scanout_handle
+{
+   Evas_Native_Scanout_Handler handler;
+   void *data;
+};
+
 static Evas_Func func, pfunc;
 
 int _evas_engine_drm_log_dom;
@@ -21,6 +30,8 @@ _render_engine_setup(Evas_Engine_Info_Drm *info, int w, int h)
    ob = _outbuf_setup(info, w, h);
    if (!ob) goto err;
 
+   re->fd = info->info.fd;
+
    if (!evas_render_engine_software_generic_init(&re->generic, ob,
                                                  _outbuf_state_get,
                                                  _outbuf_rotation_get,
@@ -108,6 +119,113 @@ eng_output_free(void *engine EINA_UNUSED, void *data)
      }
 }
 
+static Ecore_Drm2_Fb *
+drm_import_simple_dmabuf(int fd, struct dmabuf_attributes *attributes)
+{
+   unsigned int stride[4] = { 0 };
+   int dmabuf_fd[4] = { 0 };
+   int i;
+
+   for (i = 0; i < attributes->n_planes; i++)
+     {
+        stride[i] = attributes->stride[i];
+        dmabuf_fd[i] = attributes->fd[i];
+     }
+
+   return ecore_drm2_fb_dmabuf_import(fd, attributes->width,
+                                      attributes->height, 32, 32,
+                                      attributes->format, stride,
+                                      dmabuf_fd, attributes->n_planes);
+}
+
+static void
+_eng_fb_release(Ecore_Drm2_Fb *fb EINA_UNUSED, Ecore_Drm2_Fb_Status status, void *data)
+{
+   struct scanout_handle *sh;
+
+   sh = data;
+   if (status == ECORE_DRM2_FB_STATUS_DELETED)
+     {
+        free(sh);
+        return;
+     }
+
+   if (!sh->handler) return;
+
+   switch (status)
+     {
+      case ECORE_DRM2_FB_STATUS_SCANOUT_ON:
+        sh->handler(sh->data, EVAS_NATIVE_SURFACE_STATUS_SCANOUT_ON);
+        break;
+      case ECORE_DRM2_FB_STATUS_SCANOUT_OFF:
+        sh->handler(sh->data, EVAS_NATIVE_SURFACE_STATUS_SCANOUT_OFF);
+        break;
+      case ECORE_DRM2_FB_STATUS_PLANE_ASSIGN:
+        sh->handler(sh->data, EVAS_NATIVE_SURFACE_STATUS_PLANE_ASSIGN);
+        break;
+      case ECORE_DRM2_FB_STATUS_PLANE_RELEASE:
+        sh->handler(sh->data, EVAS_NATIVE_SURFACE_STATUS_PLANE_RELEASE);
+        break;
+      default:
+        ERR("Unhandled framebuffer status");
+     }
+}
+
+static void *
+eng_image_plane_assign(void *data, void *image, int x, int y)
+{
+   Render_Engine *re;
+   Outbuf *ob;
+   RGBA_Image *img;
+   Native *n;
+   Ecore_Drm2_Fb *fb = NULL;
+   Ecore_Drm2_Plane *plane;
+   struct scanout_handle *g;
+
+   EINA_SAFETY_ON_NULL_RETURN_VAL(image, EINA_FALSE);
+
+   re = (Render_Engine *)data;
+   EINA_SAFETY_ON_NULL_RETURN_VAL(re, EINA_FALSE);
+
+   ob = re->generic.ob;
+   EINA_SAFETY_ON_NULL_RETURN_VAL(ob, EINA_FALSE);
+
+   img = image;
+   n = img->native.data;
+
+   /* Perhaps implementable on other surface types, but we're
+    * sticking to this one for now */
+   if (n->ns.type != EVAS_NATIVE_SURFACE_WL_DMABUF) return NULL;
+
+   fb = drm_import_simple_dmabuf(re->fd, &n->ns_data.wl_surface_dmabuf.attr);
+
+   if (!fb) return NULL;
+
+   g = calloc(1, sizeof(struct scanout_handle));
+   if (!g) goto out;
+
+   g->handler = n->ns.data.wl_dmabuf.scanout.handler;
+   g->data = n->ns.data.wl_dmabuf.scanout.data;
+   ecore_drm2_fb_status_handler_set(fb, _eng_fb_release, g);
+
+   /* Fail or not, we're going to drop that fb and let refcounting get rid of
+    * it later
+    */
+   plane = ecore_drm2_plane_assign(ob->priv.output, fb, x, y);
+
+out:
+   ecore_drm2_fb_discard(fb);
+   return plane;
+}
+
+static void
+eng_image_plane_release(void *data EINA_UNUSED, void *image EINA_UNUSED, void *plin)
+{
+   Ecore_Drm2_Plane *plane = plin;
+
+   ecore_drm2_plane_release(plane);
+}
+
 static int
 module_open(Evas_Module *em)
 {
@@ -139,6 +257,8 @@ module_open(Evas_Module *em)
    EVAS_API_OVERRIDE(setup, &func, eng_);
    EVAS_API_OVERRIDE(update, &func, eng_);
    EVAS_API_OVERRIDE(output_free, &func, eng_);
+   EVAS_API_OVERRIDE(image_plane_assign, &func, eng_);
+   EVAS_API_OVERRIDE(image_plane_release, &func, eng_);
 
    /* advertise our engine functions */
    em->functions = (void *)(&func);