Blame gst-libs/gst/allocators/gstfdmemory.c

Packit 971217
/* GStreamer fd backed memory
Packit 971217
 * Copyright (C) 2013 Linaro SA
Packit 971217
 * Author: Benjamin Gaignard <benjamin.gaignard@linaro.org> for Linaro.
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 mordetails.
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., 59 Temple Place - Suite 330,
Packit 971217
 * Boston, MA 02111-1307, USA.
Packit 971217
 */
Packit 971217
Packit 971217
/**
Packit 971217
 * SECTION:gstfdmemory
Packit 971217
 * @title: GstFdAllocator
Packit 971217
 * @short_description: Memory wrapper for fd backed memory
Packit 971217
 * @see_also: #GstMemory
Packit 971217
 *
Packit 971217
 * Since: 1.4
Packit 971217
 */
Packit 971217
Packit 971217
#ifdef HAVE_CONFIG_H
Packit 971217
#include "config.h"
Packit 971217
#endif
Packit 971217
Packit 971217
#include "gstfdmemory.h"
Packit 971217
Packit 971217
#ifdef HAVE_MMAP
Packit 971217
#include <sys/mman.h>
Packit 971217
#include <unistd.h>
Packit 971217
#endif
Packit 971217
Packit 971217
GST_DEBUG_CATEGORY_STATIC (gst_fdmemory_debug);
Packit 971217
#define GST_CAT_DEFAULT gst_fdmemory_debug
Packit 971217
Packit 971217
typedef struct
Packit 971217
{
Packit 971217
  GstMemory mem;
Packit 971217
Packit 971217
  GstFdMemoryFlags flags;
Packit 971217
  gint fd;
Packit 971217
  gpointer data;
Packit 971217
  gint mmapping_flags;
Packit 971217
  gint mmap_count;
Packit 971217
  GMutex lock;
Packit 971217
} GstFdMemory;
Packit 971217
Packit 971217
static void
Packit 971217
gst_fd_mem_free (GstAllocator * allocator, GstMemory * gmem)
Packit 971217
{
Packit 971217
#ifdef HAVE_MMAP
Packit 971217
  GstFdMemory *mem = (GstFdMemory *) gmem;
Packit 971217
Packit 971217
  if (mem->data) {
Packit 971217
    if (!(mem->flags & GST_FD_MEMORY_FLAG_KEEP_MAPPED))
Packit 971217
      g_warning (G_STRLOC ":%s: Freeing memory %p still mapped", G_STRFUNC,
Packit 971217
          mem);
Packit 971217
Packit 971217
    munmap ((void *) mem->data, gmem->maxsize);
Packit 971217
  }
Packit 971217
  if (mem->fd >= 0 && gmem->parent == NULL
Packit 971217
      && !(mem->flags & GST_FD_MEMORY_FLAG_DONT_CLOSE))
Packit 971217
    close (mem->fd);
Packit 971217
  g_mutex_clear (&mem->lock);
Packit 971217
  g_slice_free (GstFdMemory, mem);
Packit 971217
  GST_DEBUG ("%p: freed", mem);
Packit 971217
#endif
Packit 971217
}
Packit 971217
Packit 971217
static gpointer
Packit 971217
gst_fd_mem_map (GstMemory * gmem, gsize maxsize, GstMapFlags flags)
Packit 971217
{
Packit 971217
#ifdef HAVE_MMAP
Packit 971217
  GstFdMemory *mem = (GstFdMemory *) gmem;
Packit 971217
  gint prot;
Packit 971217
  gpointer ret = NULL;
Packit 971217
Packit 971217
  if (gmem->parent)
Packit 971217
    return gst_fd_mem_map (gmem->parent, maxsize, flags);
Packit 971217
Packit 971217
  prot = flags & GST_MAP_READ ? PROT_READ : 0;
Packit 971217
  prot |= flags & GST_MAP_WRITE ? PROT_WRITE : 0;
Packit 971217
Packit 971217
  g_mutex_lock (&mem->lock);
Packit 971217
  /* do not mmap twice the buffer */
Packit 971217
  if (mem->data) {
Packit 971217
    /* only return address if mapping flags are a subset
Packit 971217
     * of the previous flags */
Packit 971217
    if ((mem->mmapping_flags & prot) == prot) {
Packit 971217
      ret = mem->data;
Packit 971217
      mem->mmap_count++;
Packit 971217
    }
Packit 971217
Packit 971217
    goto out;
Packit 971217
  }
Packit 971217
Packit 971217
  if (mem->fd != -1) {
Packit 971217
    gint flags;
Packit 971217
Packit 971217
    flags =
Packit 971217
        (mem->flags & GST_FD_MEMORY_FLAG_MAP_PRIVATE) ? MAP_PRIVATE :
Packit 971217
        MAP_SHARED;
Packit 971217
Packit 971217
    mem->data = mmap (0, gmem->maxsize, prot, flags, mem->fd, 0);
Packit 971217
    if (mem->data == MAP_FAILED) {
Packit 971217
      GstDebugLevel level;
Packit 971217
      mem->data = NULL;
Packit 971217
Packit 971217
      switch (errno) {
Packit 971217
        case EACCES:
Packit 971217
          level = GST_LEVEL_INFO;
Packit 971217
          break;
Packit 971217
        default:
Packit 971217
          level = GST_LEVEL_ERROR;
Packit 971217
          break;
Packit 971217
      }
Packit 971217
Packit 971217
      GST_CAT_LEVEL_LOG (GST_CAT_DEFAULT, level, NULL,
Packit 971217
          "%p: fd %d: mmap failed: %s", mem, mem->fd, g_strerror (errno));
Packit 971217
      goto out;
Packit 971217
    }
Packit 971217
  }
Packit 971217
Packit 971217
  GST_DEBUG ("%p: fd %d: mapped %p", mem, mem->fd, mem->data);
Packit 971217
Packit 971217
  if (mem->data) {
Packit 971217
    mem->mmapping_flags = prot;
Packit 971217
    mem->mmap_count++;
Packit 971217
    ret = mem->data;
Packit 971217
  }
Packit 971217
Packit 971217
out:
Packit 971217
  g_mutex_unlock (&mem->lock);
Packit 971217
  return ret;
Packit 971217
#else /* !HAVE_MMAP */
Packit 971217
  return FALSE;
Packit 971217
#endif
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_fd_mem_unmap (GstMemory * gmem)
Packit 971217
{
Packit 971217
#ifdef HAVE_MMAP
Packit 971217
  GstFdMemory *mem = (GstFdMemory *) gmem;
Packit 971217
Packit 971217
  if (gmem->parent)
Packit 971217
    return gst_fd_mem_unmap (gmem->parent);
Packit 971217
Packit 971217
  if (mem->flags & GST_FD_MEMORY_FLAG_KEEP_MAPPED)
Packit 971217
    return;
Packit 971217
Packit 971217
  g_mutex_lock (&mem->lock);
Packit 971217
  if (mem->data && !(--mem->mmap_count)) {
Packit 971217
    munmap ((void *) mem->data, gmem->maxsize);
Packit 971217
    mem->data = NULL;
Packit 971217
    mem->mmapping_flags = 0;
Packit 971217
    GST_DEBUG ("%p: fd %d unmapped", mem, mem->fd);
Packit 971217
  }
Packit 971217
  g_mutex_unlock (&mem->lock);
Packit 971217
#endif
Packit 971217
}
Packit 971217
Packit 971217
static GstMemory *
Packit 971217
gst_fd_mem_share (GstMemory * gmem, gssize offset, gssize size)
Packit 971217
{
Packit 971217
#ifdef HAVE_MMAP
Packit 971217
  GstFdMemory *mem = (GstFdMemory *) gmem;
Packit 971217
  GstFdMemory *sub;
Packit 971217
  GstMemory *parent;
Packit 971217
Packit 971217
  GST_DEBUG ("%p: share %" G_GSSIZE_FORMAT " %" G_GSIZE_FORMAT, mem, offset,
Packit 971217
      size);
Packit 971217
Packit 971217
  /* find the real parent */
Packit 971217
  if ((parent = mem->mem.parent) == NULL)
Packit 971217
    parent = (GstMemory *) mem;
Packit 971217
Packit 971217
  if (size == -1)
Packit 971217
    size = gmem->maxsize - offset;
Packit 971217
Packit 971217
  sub = g_slice_new0 (GstFdMemory);
Packit 971217
  /* the shared memory is always readonly */
Packit 971217
  gst_memory_init (GST_MEMORY_CAST (sub), GST_MINI_OBJECT_FLAGS (parent) |
Packit 971217
      GST_MINI_OBJECT_FLAG_LOCK_READONLY, mem->mem.allocator, parent,
Packit 971217
      mem->mem.maxsize, mem->mem.align, mem->mem.offset + offset, size);
Packit 971217
Packit 971217
  sub->fd = mem->fd;
Packit 971217
  g_mutex_init (&sub->lock);
Packit 971217
Packit 971217
  return GST_MEMORY_CAST (sub);
Packit 971217
#else /* !HAVE_MMAP */
Packit 971217
  return NULL;
Packit 971217
#endif
Packit 971217
}
Packit 971217
Packit 971217
G_DEFINE_TYPE (GstFdAllocator, gst_fd_allocator, GST_TYPE_ALLOCATOR);
Packit 971217
Packit 971217
static void
Packit 971217
gst_fd_allocator_class_init (GstFdAllocatorClass * klass)
Packit 971217
{
Packit 971217
  GstAllocatorClass *allocator_class;
Packit 971217
Packit 971217
  allocator_class = (GstAllocatorClass *) klass;
Packit 971217
Packit 971217
  allocator_class->alloc = NULL;
Packit 971217
  allocator_class->free = gst_fd_mem_free;
Packit 971217
Packit 971217
  GST_DEBUG_CATEGORY_INIT (gst_fdmemory_debug, "fdmemory", 0,
Packit 971217
      "GstFdMemory and GstFdAllocator");
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_fd_allocator_init (GstFdAllocator * allocator)
Packit 971217
{
Packit 971217
  GstAllocator *alloc = GST_ALLOCATOR_CAST (allocator);
Packit 971217
Packit 971217
  alloc->mem_type = GST_ALLOCATOR_FD;
Packit 971217
Packit 971217
  alloc->mem_map = gst_fd_mem_map;
Packit 971217
  alloc->mem_unmap = gst_fd_mem_unmap;
Packit 971217
  alloc->mem_share = gst_fd_mem_share;
Packit 971217
Packit 971217
  GST_OBJECT_FLAG_SET (allocator, GST_ALLOCATOR_FLAG_CUSTOM_ALLOC);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_fd_allocator_new:
Packit 971217
 *
Packit 971217
 * Return a new fd allocator.
Packit 971217
 *
Packit 971217
 * Returns: (transfer full): a new fd allocator, or NULL if the allocator
Packit 971217
 *    isn't available. Use gst_object_unref() to release the allocator after
Packit 971217
 *    usage
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
GstAllocator *
Packit 971217
gst_fd_allocator_new (void)
Packit 971217
{
Packit 971217
  GstAllocator *alloc;
Packit 971217
Packit 971217
  alloc = g_object_new (GST_TYPE_FD_ALLOCATOR, NULL);
Packit 971217
  gst_object_ref_sink (alloc);
Packit 971217
Packit 971217
  return alloc;
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_fd_allocator_alloc:
Packit 971217
 * @allocator: allocator to be used for this memory
Packit 971217
 * @fd: file descriptor
Packit 971217
 * @size: memory size
Packit 971217
 * @flags: extra #GstFdMemoryFlags
Packit 971217
 *
Packit 971217
 * Return a %GstMemory that wraps a generic file descriptor.
Packit 971217
 *
Packit 971217
 * Returns: (transfer full): a GstMemory based on @allocator.
Packit 971217
 * When the buffer will be released the allocator will close the @fd unless
Packit 971217
 * the %GST_FD_MEMORY_FLAG_DONT_CLOSE flag is specified.
Packit 971217
 * The memory is only mmapped on gst_buffer_mmap() request.
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
GstMemory *
Packit 971217
gst_fd_allocator_alloc (GstAllocator * allocator, gint fd, gsize size,
Packit 971217
    GstFdMemoryFlags flags)
Packit 971217
{
Packit 971217
#ifdef HAVE_MMAP
Packit 971217
  GstFdMemory *mem;
Packit 971217
Packit 971217
  g_return_val_if_fail (GST_IS_FD_ALLOCATOR (allocator), NULL);
Packit 971217
Packit 971217
  mem = g_slice_new0 (GstFdMemory);
Packit 971217
  gst_memory_init (GST_MEMORY_CAST (mem), 0, GST_ALLOCATOR_CAST (allocator),
Packit 971217
      NULL, size, 0, 0, size);
Packit 971217
Packit 971217
  mem->flags = flags;
Packit 971217
  mem->fd = fd;
Packit 971217
  g_mutex_init (&mem->lock);
Packit 971217
Packit 971217
  GST_DEBUG ("%p: fd: %d size %" G_GSIZE_FORMAT, mem, mem->fd,
Packit 971217
      mem->mem.maxsize);
Packit 971217
Packit 971217
  return (GstMemory *) mem;
Packit 971217
#else /* !HAVE_MMAP */
Packit 971217
  return NULL;
Packit 971217
#endif
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_is_fd_memory:
Packit 971217
 * @mem: #GstMemory
Packit 971217
 *
Packit 971217
 * Check if @mem is memory backed by an fd
Packit 971217
 *
Packit 971217
 * Returns: %TRUE when @mem has an fd that can be retrieved with
Packit 971217
 * gst_fd_memory_get_fd().
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
gboolean
Packit 971217
gst_is_fd_memory (GstMemory * mem)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (mem != NULL, FALSE);
Packit 971217
Packit 971217
  return GST_IS_FD_ALLOCATOR (mem->allocator);
Packit 971217
}
Packit 971217
Packit 971217
/**
Packit 971217
 * gst_fd_memory_get_fd:
Packit 971217
 * @mem: #GstMemory
Packit 971217
 *
Packit 971217
 * Get the fd from @mem. Call gst_is_fd_memory() to check if @mem has
Packit 971217
 * an fd.
Packit 971217
 *
Packit 971217
 * Returns: the fd of @mem or -1 when there is no fd on @mem
Packit 971217
 *
Packit 971217
 * Since: 1.6
Packit 971217
 */
Packit 971217
gint
Packit 971217
gst_fd_memory_get_fd (GstMemory * mem)
Packit 971217
{
Packit 971217
  g_return_val_if_fail (mem != NULL, -1);
Packit 971217
  g_return_val_if_fail (GST_IS_FD_ALLOCATOR (mem->allocator), -1);
Packit 971217
Packit 971217
  return ((GstFdMemory *) mem)->fd;
Packit 971217
}