Blame plugins/elements/gstelements_private.c

Packit Service 963350
/* GStreamer
Packit Service 963350
 * Copyright (C) 2011 David Schleef <ds@schleef.org>
Packit Service 963350
 * Copyright (C) 2011 Tim-Philipp Müller <tim.muller@collabora.co.uk>
Packit Service 963350
 * Copyright (C) 2014 Tim-Philipp Müller <tim@centricular.com>
Packit Service 963350
 * Copyright (C) 2014 Vincent Penquerc'h <vincent@collabora.co.uk>
Packit Service 963350
 *
Packit Service 963350
 * gstelements_private.c: Shared code for core elements
Packit Service 963350
 *
Packit Service 963350
 * This library is free software; you can redistribute it and/or
Packit Service 963350
 * modify it under the terms of the GNU Library General Public
Packit Service 963350
 * License as published by the Free Software Foundation; either
Packit Service 963350
 * version 2 of the License, or (at your option) any later version.
Packit Service 963350
 *
Packit Service 963350
 * This library is distributed in the hope that it will be useful,
Packit Service 963350
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 963350
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 963350
 * Library General Public License for more details.
Packit Service 963350
 *
Packit Service 963350
 * You should have received a copy of the GNU Library General Public
Packit Service 963350
 * License along with this library; if not, write to the
Packit Service 963350
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit Service 963350
 * Boston, MA 02110-1301, USA.
Packit Service 963350
 */
Packit Service 963350
Packit Service 963350
#ifdef HAVE_CONFIG_H
Packit Service 963350
# include "config.h"
Packit Service 963350
#endif
Packit Service 963350
#include <stdio.h>
Packit Service 963350
#ifdef HAVE_UNISTD_H
Packit Service 963350
#include <unistd.h>
Packit Service 963350
#endif
Packit Service 963350
#ifdef HAVE_SYS_UIO_H
Packit Service 963350
#include <sys/uio.h>
Packit Service 963350
#endif
Packit Service 963350
#include <errno.h>
Packit Service 963350
#include <string.h>
Packit Service 963350
#include <string.h>
Packit Service 963350
#include "gst/gst.h"
Packit Service 963350
#include "gstelements_private.h"
Packit Service 963350
Packit Service 963350
#ifdef G_OS_WIN32
Packit Service 963350
#  define WIN32_LEAN_AND_MEAN   /* prevents from including too many things */
Packit Service 963350
#  include <windows.h>
Packit Service 963350
#  undef WIN32_LEAN_AND_MEAN
Packit Service 963350
#  ifndef EWOULDBLOCK
Packit Service 963350
#  define EWOULDBLOCK EAGAIN
Packit Service 963350
#  endif
Packit Service 963350
#endif /* G_OS_WIN32 */
Packit Service 963350
Packit Service 963350
#define BUFFER_FLAG_SHIFT 4
Packit Service 963350
Packit Service 963350
G_STATIC_ASSERT ((1 << BUFFER_FLAG_SHIFT) == GST_MINI_OBJECT_FLAG_LAST);
Packit Service 963350
Packit Service 963350
/* Returns a newly allocated string describing the flags on this buffer */
Packit Service 963350
gchar *
Packit Service 963350
gst_buffer_get_flags_string (GstBuffer * buffer)
Packit Service 963350
{
Packit Service 963350
  static const char flag_strings[] =
Packit Service 963350
      "\000\000\000\000live\000decode-only\000discont\000resync\000corrupted\000"
Packit Service 963350
      "marker\000header\000gap\000droppable\000delta-unit\000tag-memory\000"
Packit Service 963350
      "FIXME";
Packit Service 963350
  static const guint8 flag_idx[] = { 0, 1, 2, 3, 4, 9, 21, 29, 36, 46, 53,
Packit Service 963350
    60, 64, 74, 85, 96
Packit Service 963350
  };
Packit Service 963350
  int i, max_bytes;
Packit Service 963350
  char *flag_str, *end;
Packit Service 963350
Packit Service 963350
  /* max size is all flag strings plus a space or terminator after each one */
Packit Service 963350
  max_bytes = sizeof (flag_strings);
Packit Service 963350
  flag_str = g_malloc (max_bytes);
Packit Service 963350
Packit Service 963350
  end = flag_str;
Packit Service 963350
  end[0] = '\0';
Packit Service 963350
  for (i = BUFFER_FLAG_SHIFT; i < G_N_ELEMENTS (flag_idx); i++) {
Packit Service 963350
    if (GST_MINI_OBJECT_CAST (buffer)->flags & (1 << i)) {
Packit Service 963350
      strcpy (end, flag_strings + flag_idx[i]);
Packit Service 963350
      end += strlen (end);
Packit Service 963350
      end[0] = ' ';
Packit Service 963350
      end[1] = '\0';
Packit Service 963350
      end++;
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return flag_str;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* Returns a newly-allocated string describing the metas on this buffer, or NULL */
Packit Service 963350
gchar *
Packit Service 963350
gst_buffer_get_meta_string (GstBuffer * buffer)
Packit Service 963350
{
Packit Service 963350
  gpointer state = NULL;
Packit Service 963350
  GstMeta *meta;
Packit Service 963350
  GString *s = NULL;
Packit Service 963350
Packit Service 963350
  while ((meta = gst_buffer_iterate_meta (buffer, &state))) {
Packit Service 963350
    const gchar *desc = g_type_name (meta->info->type);
Packit Service 963350
Packit Service 963350
    if (s == NULL)
Packit Service 963350
      s = g_string_new (NULL);
Packit Service 963350
    else
Packit Service 963350
      g_string_append (s, ", ");
Packit Service 963350
Packit Service 963350
    g_string_append (s, desc);
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return (s != NULL) ? g_string_free (s, FALSE) : NULL;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
/* Define our own iovec structure here, so that we can use it unconditionally
Packit Service 963350
 * in the code below and use almost the same code path for systems where
Packit Service 963350
 * writev() is supported and those were it's not supported */
Packit Service 963350
#ifndef HAVE_SYS_UIO_H
Packit Service 963350
struct iovec
Packit Service 963350
{
Packit Service 963350
  gpointer iov_base;
Packit Service 963350
  gsize iov_len;
Packit Service 963350
};
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
/* completely arbitrary thresholds */
Packit Service 963350
#define FDSINK_MAX_ALLOCA_SIZE (64 * 1024)      /* 64k */
Packit Service 963350
#define FDSINK_MAX_MALLOC_SIZE ( 8 * 1024 * 1024)       /*  8M */
Packit Service 963350
Packit Service 963350
/* UIO_MAXIOV is documented in writev(2), but <sys/uio.h> only
Packit Service 963350
 * declares it on osx/ios if defined(KERNEL) */
Packit Service 963350
#ifndef UIO_MAXIOV
Packit Service 963350
#define UIO_MAXIOV 512
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
static gssize
Packit Service 963350
gst_writev (gint fd, const struct iovec *iov, gint iovcnt, gsize total_bytes)
Packit Service 963350
{
Packit Service 963350
  gssize written;
Packit Service 963350
Packit Service 963350
#ifdef HAVE_SYS_UIO_H
Packit Service 963350
  if (iovcnt <= UIO_MAXIOV) {
Packit Service 963350
    do {
Packit Service 963350
      written = writev (fd, iov, iovcnt);
Packit Service 963350
    } while (written < 0 && errno == EINTR);
Packit Service 963350
  } else
Packit Service 963350
#endif
Packit Service 963350
  {
Packit Service 963350
    gint i;
Packit Service 963350
Packit Service 963350
    /* We merge the memories here because technically write()/writev() is
Packit Service 963350
     * supposed to be atomic, which it's not if we do multiple separate
Packit Service 963350
     * write() calls. It's very doubtful anyone cares though in our use
Packit Service 963350
     * cases, and it's not clear how that can be reconciled with the
Packit Service 963350
     * possibility of short writes, so in any case we might want to
Packit Service 963350
     * simplify this later or just remove it. */
Packit Service 963350
    if (total_bytes <= FDSINK_MAX_MALLOC_SIZE) {
Packit Service 963350
      gchar *mem, *p;
Packit Service 963350
Packit Service 963350
      if (total_bytes <= FDSINK_MAX_ALLOCA_SIZE)
Packit Service 963350
        mem = g_alloca (total_bytes);
Packit Service 963350
      else
Packit Service 963350
        mem = g_malloc (total_bytes);
Packit Service 963350
Packit Service 963350
      p = mem;
Packit Service 963350
      for (i = 0; i < iovcnt; ++i) {
Packit Service 963350
        memcpy (p, iov[i].iov_base, iov[i].iov_len);
Packit Service 963350
        p += iov[i].iov_len;
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
      do {
Packit Service 963350
        written = write (fd, mem, total_bytes);
Packit Service 963350
      } while (written < 0 && errno == EINTR);
Packit Service 963350
Packit Service 963350
      if (total_bytes > FDSINK_MAX_ALLOCA_SIZE)
Packit Service 963350
        g_free (mem);
Packit Service 963350
    } else {
Packit Service 963350
      gssize ret;
Packit Service 963350
Packit Service 963350
      written = 0;
Packit Service 963350
      for (i = 0; i < iovcnt; ++i) {
Packit Service 963350
        do {
Packit Service 963350
          ret = write (fd, iov[i].iov_base, iov[i].iov_len);
Packit Service 963350
        } while (ret < 0 && errno == EINTR);
Packit Service 963350
        if (ret > 0)
Packit Service 963350
          written += ret;
Packit Service 963350
        if (ret != iov[i].iov_len)
Packit Service 963350
          break;
Packit Service 963350
      }
Packit Service 963350
    }
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return written;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
static gsize
Packit Service 963350
fill_vectors (struct iovec *vecs, GstMapInfo * maps, guint n, GstBuffer * buf)
Packit Service 963350
{
Packit Service 963350
  GstMemory *mem;
Packit Service 963350
  gsize size = 0;
Packit Service 963350
  guint i;
Packit Service 963350
Packit Service 963350
  g_assert (gst_buffer_n_memory (buf) == n);
Packit Service 963350
Packit Service 963350
  for (i = 0; i < n; ++i) {
Packit Service 963350
    mem = gst_buffer_peek_memory (buf, i);
Packit Service 963350
    if (gst_memory_map (mem, &maps[i], GST_MAP_READ)) {
Packit Service 963350
      vecs[i].iov_base = maps[i].data;
Packit Service 963350
      vecs[i].iov_len = maps[i].size;
Packit Service 963350
    } else {
Packit Service 963350
      GST_WARNING ("Failed to map memory %p for reading", mem);
Packit Service 963350
      vecs[i].iov_base = (void *) "";
Packit Service 963350
      vecs[i].iov_len = 0;
Packit Service 963350
    }
Packit Service 963350
    size += vecs[i].iov_len;
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  return size;
Packit Service 963350
}
Packit Service 963350
Packit Service 963350
GstFlowReturn
Packit Service 963350
gst_writev_buffers (GstObject * sink, gint fd, GstPoll * fdset,
Packit Service 963350
    GstBuffer ** buffers, guint num_buffers, guint8 * mem_nums,
Packit Service 963350
    guint total_mem_num, guint64 * bytes_written, guint64 skip)
Packit Service 963350
{
Packit Service 963350
  struct iovec *vecs;
Packit Service 963350
  GstMapInfo *map_infos;
Packit Service 963350
  GstFlowReturn flow_ret;
Packit Service 963350
  gsize size = 0;
Packit Service 963350
  guint i, j;
Packit Service 963350
Packit Service 963350
  GST_LOG_OBJECT (sink, "%u buffers, %u memories", num_buffers, total_mem_num);
Packit Service 963350
Packit Service 963350
  vecs = g_newa (struct iovec, total_mem_num);
Packit Service 963350
  map_infos = g_newa (GstMapInfo, total_mem_num);
Packit Service 963350
Packit Service 963350
  /* populate output vectors */
Packit Service 963350
  for (i = 0, j = 0; i < num_buffers; ++i) {
Packit Service 963350
    size += fill_vectors (&vecs[j], &map_infos[j], mem_nums[i], buffers[i]);
Packit Service 963350
    j += mem_nums[i];
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  /* now write it all out! */
Packit Service 963350
  {
Packit Service 963350
    gssize ret, left;
Packit Service 963350
    guint n_vecs = total_mem_num;
Packit Service 963350
Packit Service 963350
    left = size;
Packit Service 963350
Packit Service 963350
    if (skip) {
Packit Service 963350
      ret = skip;
Packit Service 963350
      errno = 0;
Packit Service 963350
      goto skip_first;
Packit Service 963350
    }
Packit Service 963350
Packit Service 963350
    do {
Packit Service 963350
#ifndef HAVE_WIN32
Packit Service 963350
      if (fdset != NULL) {
Packit Service 963350
        do {
Packit Service 963350
          GST_DEBUG_OBJECT (sink, "going into select, have %" G_GSSIZE_FORMAT
Packit Service 963350
              " bytes to write", left);
Packit Service 963350
          ret = gst_poll_wait (fdset, GST_CLOCK_TIME_NONE);
Packit Service 963350
        } while (ret == -1 && (errno == EINTR || errno == EAGAIN));
Packit Service 963350
Packit Service 963350
        if (ret == -1) {
Packit Service 963350
          if (errno == EBUSY)
Packit Service 963350
            goto stopped;
Packit Service 963350
          else
Packit Service 963350
            goto select_error;
Packit Service 963350
        }
Packit Service 963350
      }
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
      ret = gst_writev (fd, vecs, n_vecs, left);
Packit Service 963350
Packit Service 963350
      if (ret > 0) {
Packit Service 963350
        if (bytes_written)
Packit Service 963350
          *bytes_written += ret;
Packit Service 963350
      }
Packit Service 963350
Packit Service 963350
    skip_first:
Packit Service 963350
Packit Service 963350
      if (ret == left)
Packit Service 963350
        break;
Packit Service 963350
Packit Service 963350
      if (ret < 0 && (errno == EAGAIN || errno == EWOULDBLOCK)) {
Packit Service 963350
        /* do nothing, try again */
Packit Service 963350
      } else if (ret < 0) {
Packit Service 963350
        goto write_error;
Packit Service 963350
      } else if (ret < left) {
Packit Service 963350
        /* skip vectors that have been written in full */
Packit Service 963350
        while (ret >= vecs[0].iov_len) {
Packit Service 963350
          ret -= vecs[0].iov_len;
Packit Service 963350
          left -= vecs[0].iov_len;
Packit Service 963350
          ++vecs;
Packit Service 963350
          --n_vecs;
Packit Service 963350
        }
Packit Service 963350
        g_assert (n_vecs > 0);
Packit Service 963350
        /* skip partially written vector data */
Packit Service 963350
        if (ret > 0) {
Packit Service 963350
          vecs[0].iov_len -= ret;
Packit Service 963350
          vecs[0].iov_base = ((guint8 *) vecs[0].iov_base) + ret;
Packit Service 963350
          left -= ret;
Packit Service 963350
        }
Packit Service 963350
      }
Packit Service 963350
#ifdef HAVE_WIN32
Packit Service 963350
      /* do short sleep on windows where we don't use gst_poll(),
Packit Service 963350
       * to avoid excessive busy looping */
Packit Service 963350
      if (fdset != NULL)
Packit Service 963350
        g_usleep (1000);
Packit Service 963350
#endif
Packit Service 963350
Packit Service 963350
    }
Packit Service 963350
    while (left > 0);
Packit Service 963350
  }
Packit Service 963350
Packit Service 963350
  flow_ret = GST_FLOW_OK;
Packit Service 963350
Packit Service 963350
out:
Packit Service 963350
Packit Service 963350
  for (i = 0; i < total_mem_num; ++i)
Packit Service 963350
    gst_memory_unmap (map_infos[i].memory, &map_infos[i]);
Packit Service 963350
Packit Service 963350
  return flow_ret;
Packit Service 963350
Packit Service 963350
/* ERRORS */
Packit Service 963350
#ifndef HAVE_WIN32
Packit Service 963350
select_error:
Packit Service 963350
  {
Packit Service 963350
    GST_ELEMENT_ERROR (sink, RESOURCE, READ, (NULL),
Packit Service 963350
        ("select on file descriptor: %s", g_strerror (errno)));
Packit Service 963350
    GST_DEBUG_OBJECT (sink, "Error during select: %s", g_strerror (errno));
Packit Service 963350
    flow_ret = GST_FLOW_ERROR;
Packit Service 963350
    goto out;
Packit Service 963350
  }
Packit Service 963350
stopped:
Packit Service 963350
  {
Packit Service 963350
    GST_DEBUG_OBJECT (sink, "Select stopped");
Packit Service 963350
    flow_ret = GST_FLOW_FLUSHING;
Packit Service 963350
    goto out;
Packit Service 963350
  }
Packit Service 963350
#endif
Packit Service 963350
write_error:
Packit Service 963350
  {
Packit Service 963350
    switch (errno) {
Packit Service 963350
      case ENOSPC:
Packit Service 963350
        GST_ELEMENT_ERROR (sink, RESOURCE, NO_SPACE_LEFT, (NULL), (NULL));
Packit Service 963350
        break;
Packit Service 963350
      default:{
Packit Service 963350
        GST_ELEMENT_ERROR (sink, RESOURCE, WRITE, (NULL),
Packit Service 963350
            ("Error while writing to file descriptor %d: %s",
Packit Service 963350
                fd, g_strerror (errno)));
Packit Service 963350
      }
Packit Service 963350
    }
Packit Service 963350
    flow_ret = GST_FLOW_ERROR;
Packit Service 963350
    goto out;
Packit Service 963350
  }
Packit Service 963350
}