Blame gst-libs/gst/gl/egl/gstglmemoryegl.c

Packit 971217
/*
Packit 971217
 * GStreamer
Packit 971217
 * Copyright (C) 2012 Collabora Ltd.
Packit 971217
 *   @author: Sebastian Dröge <sebastian.droege@collabora.co.uk>
Packit 971217
 * Copyright (C) 2014 Julien Isorce <julien.isorce@gmail.com>
Packit 971217
 * Copyright (C) 2015 Igalia
Packit 971217
 *    Author: Gwang Yoon Hwang <yoon@igalia.com>
Packit 971217
 *
Packit 971217
 * This library is free software; you can redistribute it and/or
Packit 971217
 * modify it under the terms of the GNU Library General Public
Packit 971217
 * License as published by the Free Software Foundation; either
Packit 971217
 * version 2 of the License, or (at your option) any later version.
Packit 971217
 *
Packit 971217
 * This library is distributed in the hope that it will be useful,
Packit 971217
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 971217
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 971217
 * Library General Public License for more details.
Packit 971217
 *
Packit 971217
 * You should have received a copy of the GNU Library General Public
Packit 971217
 * License along with this library; if not, write to the
Packit 971217
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit 971217
 * Boston, MA 02110-1301, USA.
Packit 971217
 */
Packit 971217
Packit 971217
/**
Packit 971217
 * SECTION:gstglmemoryegl
Packit 971217
 * @short_description: memory subclass for EGLImage's
Packit 971217
 * @see_also: #GstGLMemory, #GstGLAllocator, #GstGLBufferPool
Packit 971217
 *
Packit 971217
 * #GstGLMemoryEGL is created or wrapped through gst_gl_base_memory_alloc()
Packit 971217
 * with #GstGLVideoAllocationParams.
Packit 971217
 */
Packit 971217
Packit 971217
#ifdef HAVE_CONFIG_H
Packit 971217
#include "config.h"
Packit 971217
#endif
Packit 971217
Packit 971217
#include <string.h>
Packit 971217
Packit 971217
#include "gstglmemoryegl.h"
Packit 971217
Packit 971217
#include <gst/gl/gstglfuncs.h>
Packit 971217
Packit 971217
#include "gstegl.h"
Packit 971217
#include "gsteglimage.h"
Packit 971217
#include "gstglcontext_egl.h"
Packit 971217
Packit 971217
static GstAllocator *_gl_memory_egl_allocator;
Packit 971217
Packit 971217
GST_DEBUG_CATEGORY_STATIC (GST_CAT_GL_MEMORY);
Packit 971217
#define GST_CAT_DEFAULT GST_CAT_GL_MEMORY
Packit 971217
Packit 971217
#define parent_class gst_gl_memory_egl_allocator_parent_class
Packit 971217
G_DEFINE_TYPE (GstGLMemoryEGLAllocator, gst_gl_memory_egl_allocator,
Packit 971217
    GST_TYPE_GL_MEMORY_ALLOCATOR);
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_is_gl_memory_egl:
Packit 971217
 * @mem: a #GstMemory to test
Packit 971217
 *
Packit 971217
 * Returns: whether @mem is a #GstGLMemoryEGL
Packit 971217
 *
Packit 971217
 * Since: 1.10
Packit 971217
 */
Packit 971217
gboolean
Packit 971217
gst_is_gl_memory_egl (GstMemory * mem)
Packit 971217
{
Packit 971217
  return mem != NULL && mem->allocator != NULL
Packit 971217
      && g_type_is_a (G_OBJECT_TYPE (mem->allocator),
Packit 971217
      GST_TYPE_GL_MEMORY_EGL_ALLOCATOR);
Packit 971217
}
Packit 971217
Packit 971217
static GstGLMemoryEGL *
Packit 971217
_gl_mem_get_parent (GstGLMemoryEGL * gl_mem)
Packit 971217
{
Packit 971217
  GstGLMemoryEGL *parent = (GstGLMemoryEGL *) gl_mem->mem.mem.mem.parent;
Packit 971217
  return parent ? parent : gl_mem;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_gl_memory_egl_get_image:
Packit 971217
 * @mem: a #GstGLMemoryEGL
Packit 971217
 *
Packit 971217
 * Returns: The EGLImage held by @mem
Packit 971217
 *
Packit 971217
 * Since: 1.10
Packit 971217
 */
Packit 971217
gpointer
Packit 971217
gst_gl_memory_egl_get_image (GstGLMemoryEGL * mem)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (gst_is_gl_memory_egl (GST_MEMORY_CAST (mem)),
Packit 971217
      EGL_NO_IMAGE_KHR);
Packit 971217
  return gst_egl_image_get_image (_gl_mem_get_parent (mem)->image);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_gl_memory_egl_get_display:
Packit 971217
 * @mem: a #GstGLMemoryEGL
Packit 971217
 *
Packit 971217
 * Returns: The EGLDisplay @mem is associated with
Packit 971217
 *
Packit 971217
 * Since: 1.10
Packit 971217
 */
Packit 971217
gpointer
Packit 971217
gst_gl_memory_egl_get_display (GstGLMemoryEGL * mem)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (gst_is_gl_memory_egl (GST_MEMORY_CAST (mem)), NULL);
Packit 971217
  return GST_GL_CONTEXT_EGL (_gl_mem_get_parent (mem)->mem.mem.
Packit 971217
      context)->egl_display;
Packit 971217
}
Packit 971217
Packit 971217
static GstMemory *
Packit 971217
_gl_mem_alloc (GstAllocator * allocator, gsize size,
Packit 971217
    GstAllocationParams * params)
Packit 971217
{
Packit 971217
  g_warning ("Use gst_gl_base_memory_allocator_alloc() to allocate from this "
Packit 971217
      "GstGLMemoryEGL allocator");
Packit 971217
Packit 971217
  return NULL;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
_gl_mem_destroy (GstGLMemoryEGL * mem)
Packit 971217
{
Packit 971217
  if (mem->image)
Packit 971217
    gst_egl_image_unref (mem->image);
Packit 971217
Packit 971217
  GST_GL_BASE_MEMORY_ALLOCATOR_CLASS (parent_class)->destroy ((GstGLBaseMemory
Packit 971217
          *) mem);
Packit 971217
}
Packit 971217
Packit 971217
static GstGLMemoryEGL *
Packit 971217
_gl_mem_egl_alloc (GstGLBaseMemoryAllocator * allocator,
Packit 971217
    GstGLVideoAllocationParams * params)
Packit 971217
{
Packit 971217
  guint alloc_flags = params->parent.alloc_flags;
Packit 971217
  GstGLMemoryEGL *mem;
Packit 971217
Packit 971217
  g_return_val_if_fail (alloc_flags & GST_GL_ALLOCATION_PARAMS_ALLOC_FLAG_VIDEO,
Packit 971217
      NULL);
Packit 971217
  g_return_val_if_fail ((alloc_flags &
Packit 971217
          GST_GL_ALLOCATION_PARAMS_ALLOC_FLAG_WRAP_SYSMEM) == 0, NULL);
Packit 971217
  if (alloc_flags & GST_GL_ALLOCATION_PARAMS_ALLOC_FLAG_WRAP_GPU_HANDLE) {
Packit 971217
    g_return_val_if_fail (GST_IS_EGL_IMAGE (params->parent.gl_handle), NULL);
Packit 971217
  }
Packit 971217
Packit 971217
  mem = g_new0 (GstGLMemoryEGL, 1);
Packit 971217
  if (alloc_flags & GST_GL_ALLOCATION_PARAMS_ALLOC_FLAG_WRAP_GPU_HANDLE) {
Packit 971217
    if (params->target != GST_GL_TEXTURE_TARGET_2D) {
Packit 971217
      g_free (mem);
Packit 971217
      GST_CAT_ERROR (GST_CAT_GL_MEMORY, "GstGLMemoryEGL only supports wrapping "
Packit 971217
          "2D textures");
Packit 971217
      return NULL;
Packit 971217
    }
Packit 971217
    mem->image = gst_egl_image_ref (params->parent.gl_handle);
Packit 971217
  }
Packit 971217
Packit 971217
  gst_gl_memory_init (GST_GL_MEMORY_CAST (mem), GST_ALLOCATOR_CAST (allocator),
Packit 971217
      NULL, params->parent.context, params->target, params->tex_format,
Packit 971217
      params->parent.alloc_params, params->v_info, params->plane,
Packit 971217
      params->valign, params->parent.user_data, params->parent.notify);
Packit 971217
Packit 971217
  if (!mem->image) {
Packit 971217
    gst_allocator_free (GST_ALLOCATOR_CAST (allocator), GST_MEMORY_CAST (mem));
Packit 971217
    return NULL;
Packit 971217
  }
Packit 971217
Packit 971217
  return mem;
Packit 971217
}
Packit 971217
Packit 971217
static gboolean
Packit 971217
_gl_mem_create (GstGLMemoryEGL * gl_mem, GError ** error)
Packit 971217
{
Packit 971217
  GstGLContext *context = gl_mem->mem.mem.context;
Packit 971217
  const GstGLFuncs *gl = context->gl_vtable;
Packit 971217
  GstGLBaseMemoryAllocatorClass *alloc_class;
Packit 971217
Packit 971217
  if (!gst_gl_context_check_feature (GST_GL_CONTEXT (context),
Packit 971217
          "EGL_KHR_image_base")) {
Packit 971217
    g_set_error (error, GST_GL_CONTEXT_ERROR, GST_GL_CONTEXT_ERROR_WRONG_API,
Packit 971217
        "EGL_KHR_image_base is not supported");
Packit 971217
    return FALSE;
Packit 971217
  }
Packit 971217
Packit 971217
  alloc_class = GST_GL_BASE_MEMORY_ALLOCATOR_CLASS (parent_class);
Packit 971217
  if (!alloc_class->create ((GstGLBaseMemory *) gl_mem, error))
Packit 971217
    return FALSE;
Packit 971217
Packit 971217
  if (gl_mem->image == NULL) {
Packit 971217
    gl_mem->image = gst_egl_image_from_texture (context,
Packit 971217
        (GstGLMemory *) gl_mem, NULL);
Packit 971217
Packit 971217
    if (!gl_mem->image) {
Packit 971217
      g_set_error (error, GST_GL_CONTEXT_ERROR, GST_GL_CONTEXT_ERROR_FAILED,
Packit 971217
          "Failed to create EGLImage");
Packit 971217
      return FALSE;
Packit 971217
    }
Packit 971217
  } else {
Packit 971217
    gl->ActiveTexture (GL_TEXTURE0 + gl_mem->mem.plane);
Packit 971217
    gl->BindTexture (GL_TEXTURE_2D, gl_mem->mem.tex_id);
Packit 971217
    gl->EGLImageTargetTexture2D (GL_TEXTURE_2D,
Packit 971217
        gst_egl_image_get_image (GST_EGL_IMAGE (gl_mem->image)));
Packit 971217
  }
Packit 971217
Packit 971217
  return TRUE;
Packit 971217
}
Packit 971217
Packit 971217
static GstMemory *
Packit 971217
_gl_mem_copy (GstGLMemoryEGL * src, gssize offset, gssize size)
Packit 971217
{
Packit 971217
  GST_CAT_ERROR (GST_CAT_GL_MEMORY, "GstGLMemoryEGL does not support copy");
Packit 971217
  return NULL;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_memory_egl_allocator_class_init (GstGLMemoryEGLAllocatorClass * klass)
Packit 971217
{
Packit 971217
  GstGLBaseMemoryAllocatorClass *gl_base;
Packit 971217
  GstGLMemoryAllocatorClass *gl_tex;
Packit 971217
  GstAllocatorClass *allocator_class;
Packit 971217
Packit 971217
  gl_tex = (GstGLMemoryAllocatorClass *) klass;
Packit 971217
  gl_base = (GstGLBaseMemoryAllocatorClass *) klass;
Packit 971217
  allocator_class = (GstAllocatorClass *) klass;
Packit 971217
Packit 971217
  gl_base->alloc = (GstGLBaseMemoryAllocatorAllocFunction) _gl_mem_egl_alloc;
Packit 971217
  gl_base->create = (GstGLBaseMemoryAllocatorCreateFunction) _gl_mem_create;
Packit 971217
  gl_base->destroy = (GstGLBaseMemoryAllocatorDestroyFunction) _gl_mem_destroy;
Packit 971217
  gl_tex->copy = (GstGLBaseMemoryAllocatorCopyFunction) _gl_mem_copy;
Packit 971217
Packit 971217
  allocator_class->alloc = _gl_mem_alloc;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_memory_egl_allocator_init (GstGLMemoryEGLAllocator * allocator)
Packit 971217
{
Packit 971217
  GstAllocator *alloc = GST_ALLOCATOR_CAST (allocator);
Packit 971217
Packit 971217
  alloc->mem_type = GST_GL_MEMORY_EGL_ALLOCATOR_NAME;
Packit 971217
Packit 971217
  GST_OBJECT_FLAG_SET (allocator, GST_ALLOCATOR_FLAG_CUSTOM_ALLOC);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_gl_memory_egl_init_once:
Packit 971217
 *
Packit 971217
 * Initializes the GL Memory allocator. It is safe to call this function
Packit 971217
 * multiple times.  This must be called before any other GstGLMemoryEGL operation.
Packit 971217
 *
Packit 971217
 * Since: 1.10
Packit 971217
 */
Packit 971217
void
Packit 971217
gst_gl_memory_egl_init_once (void)
Packit 971217
{
Packit 971217
  static volatile gsize _init = 0;
Packit 971217
Packit 971217
  if (g_once_init_enter (&_init)) {
Packit 971217
    gst_gl_memory_init_once ();
Packit 971217
Packit 971217
    GST_DEBUG_CATEGORY_INIT (GST_CAT_GL_MEMORY, "glmemory", 0,
Packit 971217
        "OpenGL Texture with EGLImage memory");
Packit 971217
Packit 971217
    _gl_memory_egl_allocator =
Packit 971217
        g_object_new (GST_TYPE_GL_MEMORY_EGL_ALLOCATOR, NULL);
Packit 971217
    gst_object_ref_sink (_gl_memory_egl_allocator);
Packit 971217
Packit 971217
    /* The allocator is never unreffed */
Packit 971217
    GST_OBJECT_FLAG_SET (_gl_memory_egl_allocator,
Packit 971217
        GST_OBJECT_FLAG_MAY_BE_LEAKED);
Packit 971217
Packit 971217
    gst_allocator_register (GST_GL_MEMORY_EGL_ALLOCATOR_NAME,
Packit 971217
        gst_object_ref (_gl_memory_egl_allocator));
Packit 971217
    g_once_init_leave (&_init, 1);
Packit 971217
  }
Packit 971217
}