moved asyncdisksrc to disksrc, no point in having a distinction
authorErik Walthinsen <omega@temple-baptist.org>
Sun, 4 Feb 2001 21:37:14 +0000 (21:37 +0000)
committerErik Walthinsen <omega@temple-baptist.org>
Sun, 4 Feb 2001 21:37:14 +0000 (21:37 +0000)
Original commit message from CVS:
moved asyncdisksrc to disksrc, no point in having a distinction

12 files changed:
gst/elements/Makefile.am
gst/elements/gstasyncdisksrc.c [deleted file]
gst/elements/gstasyncdisksrc.h [deleted file]
gst/elements/gstdisksrc.c
gst/elements/gstdisksrc.h
gst/elements/gstelements.c
plugins/elements/Makefile.am
plugins/elements/gstasyncdisksrc.c [deleted file]
plugins/elements/gstasyncdisksrc.h [deleted file]
plugins/elements/gstdisksrc.c
plugins/elements/gstdisksrc.h
plugins/elements/gstelements.c

index a6292d30440ea003215884b73254261761729de5..0c5b9e6810fad9d6d48cfdd54e9913fc4f277b38 100644 (file)
@@ -15,7 +15,6 @@ libgstelements_la_SOURCES =   \
        gstidentity.c           \
        gstfakesink.c           \
        gstdisksrc.c            \
-       gstasyncdisksrc.c       \
        gstfdsrc.c              \
        gstaudiosink.c          \
        gstaudiosrc.c           \
@@ -30,7 +29,6 @@ noinst_HEADERS =              \
        gstidentity.h           \
        gstfakesink.h           \
        gstdisksrc.h            \
-       gstasyncdisksrc.h       \
        gstfdsrc.h              \
        gsthttpsrc.h            \
        gstaudiosink.h          \
diff --git a/gst/elements/gstasyncdisksrc.c b/gst/elements/gstasyncdisksrc.c
deleted file mode 100644 (file)
index f02071d..0000000
+++ /dev/null
@@ -1,386 +0,0 @@
-/* GStreamer
- * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
- *                    2000 Wim Taymans <wtay@chello.be>
- *
- * gstasyncdisksrc.c: 
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Library General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
- * Library General Public License for more details.
- *
- * You should have received a copy of the GNU Library General Public
- * License along with this library; if not, write to the
- * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
- */
-
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <unistd.h>
-#include <sys/mman.h>
-
-//#define GST_DEBUG_ENABLED
-
-#include "gstasyncdisksrc.h"
-
-
-GstElementDetails gst_asyncdisksrc_details = {
-  "Asynchronous Disk Source",
-  "Source/File",
-  "Read from arbitrary point in a file",
-  VERSION,
-  "Erik Walthinsen <omega@cse.ogi.edu>",
-  "(C) 1999",
-};
-
-
-/* AsyncDiskSrc signals and args */
-enum {
-  /* FILL ME */
-  LAST_SIGNAL
-};
-
-enum {
-  ARG_0,
-  ARG_LOCATION,
-  ARG_BYTESPERREAD,
-  ARG_OFFSET,
-  ARG_SIZE,
-};
-
-
-static void                    gst_asyncdisksrc_class_init     (GstAsyncDiskSrcClass *klass);
-static void                    gst_asyncdisksrc_init           (GstAsyncDiskSrc *asyncdisksrc);
-
-static void                    gst_asyncdisksrc_set_arg        (GtkObject *object, GtkArg *arg, guint id);
-static void                    gst_asyncdisksrc_get_arg        (GtkObject *object, GtkArg *arg, guint id);
-
-static GstBuffer *             gst_asyncdisksrc_get            (GstPad *pad);
-static GstBuffer *             gst_asyncdisksrc_get_region     (GstPad *pad, gulong offset, gulong size);
-
-static GstElementStateReturn   gst_asyncdisksrc_change_state   (GstElement *element);
-
-
-static GstElementClass *parent_class = NULL;
-//static guint gst_asyncdisksrc_signals[LAST_SIGNAL] = { 0 };
-
-GtkType
-gst_asyncdisksrc_get_type(void) 
-{
-  static GtkType asyncdisksrc_type = 0;
-
-  if (!asyncdisksrc_type) {
-    static const GtkTypeInfo asyncdisksrc_info = {
-      "GstAsyncDiskSrc",
-      sizeof(GstAsyncDiskSrc),
-      sizeof(GstAsyncDiskSrcClass),
-      (GtkClassInitFunc)gst_asyncdisksrc_class_init,
-      (GtkObjectInitFunc)gst_asyncdisksrc_init,
-      (GtkArgSetFunc)gst_asyncdisksrc_set_arg,
-      (GtkArgGetFunc)gst_asyncdisksrc_get_arg,
-      (GtkClassInitFunc)NULL,
-    };
-    asyncdisksrc_type = gtk_type_unique (GST_TYPE_ELEMENT, &asyncdisksrc_info);
-  }
-  return asyncdisksrc_type;
-}
-
-static void
-gst_asyncdisksrc_class_init (GstAsyncDiskSrcClass *klass) 
-{
-  GtkObjectClass *gtkobject_class;
-  GstElementClass *gstelement_class;
-
-  gtkobject_class = (GtkObjectClass*)klass;
-  gstelement_class = (GstElementClass*)klass;
-
-  parent_class = gtk_type_class (GST_TYPE_ELEMENT);
-
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::location", GST_TYPE_FILENAME,
-                           GTK_ARG_READWRITE, ARG_LOCATION);
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::bytesperread", GTK_TYPE_INT,
-                           GTK_ARG_READWRITE, ARG_BYTESPERREAD);
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::offset", GTK_TYPE_LONG,
-                           GTK_ARG_READWRITE, ARG_OFFSET);
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::size", GTK_TYPE_LONG,
-                           GTK_ARG_READABLE, ARG_SIZE);
-
-  gtkobject_class->set_arg = gst_asyncdisksrc_set_arg;
-  gtkobject_class->get_arg = gst_asyncdisksrc_get_arg;
-
-  gstelement_class->change_state = gst_asyncdisksrc_change_state;
-}
-
-static void 
-gst_asyncdisksrc_init (GstAsyncDiskSrc *asyncdisksrc) 
-{
-//  GST_FLAG_SET (asyncdisksrc, GST_SRC_ASYNC);
-
-  asyncdisksrc->srcpad = gst_pad_new ("src", GST_PAD_SRC);
-  gst_pad_set_get_function (asyncdisksrc->srcpad,gst_asyncdisksrc_get);
-  gst_pad_set_getregion_function (asyncdisksrc->srcpad,gst_asyncdisksrc_get_region);
-  gst_element_add_pad (GST_ELEMENT (asyncdisksrc), asyncdisksrc->srcpad);
-
-  asyncdisksrc->filename = NULL;
-  asyncdisksrc->fd = 0;
-  asyncdisksrc->size = 0;
-  asyncdisksrc->map = NULL;
-  asyncdisksrc->curoffset = 0;
-  asyncdisksrc->bytes_per_read = 4096;
-  asyncdisksrc->seq = 0;
-  asyncdisksrc->new_seek = FALSE;
-}
-
-
-static void 
-gst_asyncdisksrc_set_arg (GtkObject *object, GtkArg *arg, guint id) 
-{
-  GstAsyncDiskSrc *src;
-
-  /* it's not null if we got it, but it might not be ours */
-  g_return_if_fail (GST_IS_ASYNCDISKSRC (object));
-  
-  src = GST_ASYNCDISKSRC (object);
-
-  switch(id) {
-    case ARG_LOCATION:
-      /* the element must be stopped in order to do this */
-      g_return_if_fail (GST_STATE (src) < GST_STATE_PLAYING);
-
-      if (src->filename) g_free (src->filename);
-      /* clear the filename if we get a NULL (is that possible?) */
-      if (GTK_VALUE_STRING (*arg) == NULL) {
-        gst_element_set_state (GST_ELEMENT (object), GST_STATE_NULL);
-        src->filename = NULL;
-      /* otherwise set the new filename */
-      } else {
-        src->filename = g_strdup (GTK_VALUE_STRING (*arg));
-      }
-      break;
-    case ARG_BYTESPERREAD:
-      src->bytes_per_read = GTK_VALUE_INT (*arg);
-      break;
-    case ARG_OFFSET:
-      src->curoffset = GTK_VALUE_LONG (*arg);
-      src->new_seek = TRUE;
-      break;
-    default:
-      break;
-  }
-}
-
-static void 
-gst_asyncdisksrc_get_arg (GtkObject *object, GtkArg *arg, guint id) 
-{
-  GstAsyncDiskSrc *src;
-
-  /* it's not null if we got it, but it might not be ours */
-  g_return_if_fail (GST_IS_ASYNCDISKSRC (object));
-  
-  src = GST_ASYNCDISKSRC (object);
-
-  switch (id) {
-    case ARG_LOCATION:
-      GTK_VALUE_STRING (*arg) = src->filename;
-      break;
-    case ARG_BYTESPERREAD:
-      GTK_VALUE_INT (*arg) = src->bytes_per_read;
-      break;
-    case ARG_OFFSET:
-      GTK_VALUE_LONG (*arg) = src->curoffset;
-      break;
-    case ARG_SIZE:
-      GTK_VALUE_LONG (*arg) = src->size;
-      break;
-    default:
-      arg->type = GTK_TYPE_INVALID;
-      break;
-  }
-}
-
-/**
- * gst_asyncdisksrc_get:
- * @pad: #GstPad to push a buffer from
- *
- * Push a new buffer from the asyncdisksrc at the current offset.
- */
-static GstBuffer *
-gst_asyncdisksrc_get (GstPad *pad) 
-{
-  GstAsyncDiskSrc *src;
-  GstBuffer *buf;
-
-  g_return_val_if_fail (pad != NULL, NULL);
-  src = GST_ASYNCDISKSRC (gst_pad_get_parent (pad));
-  g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_ASYNCDISKSRC_OPEN), NULL);
-
-  /* deal with EOF state */
-  if (src->curoffset >= src->size) {
-    gst_element_signal_eos (GST_ELEMENT (src));
-    return NULL;
-  }
-
-  /* create the buffer */
-  // FIXME: should eventually use a bufferpool for this
-  buf = gst_buffer_new ();
-
-  g_return_val_if_fail (buf != NULL, NULL);
-
-  /* simply set the buffer to point to the correct region of the file */
-  GST_BUFFER_DATA (buf) = src->map + src->curoffset;
-  GST_BUFFER_OFFSET (buf) = src->curoffset;
-  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
-
-  if ((src->curoffset + src->bytes_per_read) > src->size) {
-    GST_BUFFER_SIZE (buf) = src->size - src->curoffset;
-    // FIXME: set the buffer's EOF bit here
-  } else
-    GST_BUFFER_SIZE (buf) = src->bytes_per_read;
-
-  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, src->curoffset, GST_BUFFER_SIZE (buf));
-
-  //gst_util_dump_mem (GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
-
-  src->curoffset += GST_BUFFER_SIZE (buf);
-
-  if (src->new_seek) {
-    GST_BUFFER_FLAG_SET (buf, GST_BUFFER_FLUSH);
-    GST_DEBUG (0,"new seek\n");
-    src->new_seek = FALSE;
-  }
-
-  /* we're done, return the buffer */
-  return buf;
-}
-
-/**
- * gst_asyncdisksrc_get_region:
- * @src: #GstSrc to push a buffer from
- * @offset: offset in file
- * @size: number of bytes
- *
- * Push a new buffer from the asyncdisksrc of given size at given offset.
- */
-static GstBuffer *
-gst_asyncdisksrc_get_region (GstPad *pad, gulong offset, gulong size) 
-{
-  GstAsyncDiskSrc *src;
-  GstBuffer *buf;
-
-  g_return_val_if_fail (pad != NULL, NULL);
-
-  src = GST_ASYNCDISKSRC (gst_pad_get_parent (pad));
-
-  g_return_val_if_fail (GST_IS_ASYNCDISKSRC (src), NULL);
-  g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_ASYNCDISKSRC_OPEN), NULL);
-  
-  /* deal with EOF state */
-  if (offset >= src->size) {
-    gst_element_signal_eos (GST_ELEMENT (src));
-    return NULL;
-  }
-
-  /* create the buffer */
-  // FIXME: should eventually use a bufferpool for this
-  buf = gst_buffer_new ();
-  g_return_val_if_fail (buf != NULL, NULL);
-
-  /* simply set the buffer to point to the correct region of the file */
-  GST_BUFFER_DATA (buf) = src->map + offset;
-  GST_BUFFER_OFFSET (buf) = offset;
-  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
-
-  if ((offset + size) > src->size) {
-    GST_BUFFER_SIZE (buf) = src->size - offset;
-    // FIXME: set the buffer's EOF bit here
-  } else
-    GST_BUFFER_SIZE (buf) = size;
-
-  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, offset, GST_BUFFER_SIZE (buf));
-
-  /* we're done, return the buffer off now */
-  return buf;
-}
-
-
-/* open the file and mmap it, necessary to go to READY state */
-static 
-gboolean gst_asyncdisksrc_open_file (GstAsyncDiskSrc *src) 
-{
-  g_return_val_if_fail (!GST_FLAG_IS_SET (src ,GST_ASYNCDISKSRC_OPEN), FALSE);
-
-  /* open the file */
-  src->fd = open (src->filename, O_RDONLY);
-  if (src->fd < 0) {
-    perror ("open");
-    gst_element_error (GST_ELEMENT (src), g_strconcat("opening file \"", src->filename, "\"", NULL));
-    return FALSE;
-  } else {
-    /* find the file length */
-    src->size = lseek (src->fd, 0, SEEK_END);
-    lseek (src->fd, 0, SEEK_SET);
-    /* map the file into memory */
-    src->map = mmap (NULL, src->size, PROT_READ, MAP_SHARED, src->fd, 0);
-    madvise (src->map,src->size, 2);
-    /* collapse state if that failed */
-    if (src->map == NULL) {
-      close (src->fd);
-      gst_element_error (GST_ELEMENT (src),"mmapping file");
-      return FALSE;
-    }
-    GST_FLAG_SET (src, GST_ASYNCDISKSRC_OPEN);
-    src->new_seek = TRUE;
-  }
-  return TRUE;
-}
-
-/* unmap and close the file */
-static void 
-gst_asyncdisksrc_close_file (GstAsyncDiskSrc *src) 
-{
-  g_return_if_fail (GST_FLAG_IS_SET (src, GST_ASYNCDISKSRC_OPEN));
-
-  /* unmap the file from memory */
-  munmap (src->map, src->size);
-  /* close the file */
-  close (src->fd);
-
-  /* zero out a lot of our state */
-  src->fd = 0;
-  src->size = 0;
-  src->map = NULL;
-  src->curoffset = 0;
-  src->seq = 0;
-  src->new_seek = FALSE;
-
-  GST_FLAG_UNSET (src, GST_ASYNCDISKSRC_OPEN);
-}
-
-
-static GstElementStateReturn 
-gst_asyncdisksrc_change_state (GstElement *element) 
-{
-  g_return_val_if_fail (GST_IS_ASYNCDISKSRC (element), GST_STATE_FAILURE);
-
-  if (GST_STATE_PENDING (element) == GST_STATE_NULL) {
-    if (GST_FLAG_IS_SET (element, GST_ASYNCDISKSRC_OPEN))
-      gst_asyncdisksrc_close_file (GST_ASYNCDISKSRC (element));
-  } else {
-    if (!GST_FLAG_IS_SET (element, GST_ASYNCDISKSRC_OPEN)) {
-      if (!gst_asyncdisksrc_open_file (GST_ASYNCDISKSRC (element))) 
-        return GST_STATE_FAILURE;
-    }
-  }
-
-  if (GST_ELEMENT_CLASS (parent_class)->change_state)
-    return GST_ELEMENT_CLASS (parent_class)->change_state (element);
-
-  return GST_STATE_SUCCESS;
-}
diff --git a/gst/elements/gstasyncdisksrc.h b/gst/elements/gstasyncdisksrc.h
deleted file mode 100644 (file)
index 1d188cb..0000000
+++ /dev/null
@@ -1,91 +0,0 @@
-/* GStreamer
- * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
- *                    2000 Wim Taymans <wtay@chello.be>
- *
- * gstasyncdisksrc.h: 
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Library General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
- * Library General Public License for more details.
- *
- * You should have received a copy of the GNU Library General Public
- * License along with this library; if not, write to the
- * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
- */
-
-
-#ifndef __GST_ASYNCDISKSRC_H__
-#define __GST_ASYNCDISKSRC_H__
-
-
-#include <gst/gst.h>
-
-
-#ifdef __cplusplus
-extern "C" {
-#endif /* __cplusplus */
-
-
-GstElementDetails gst_asyncdisksrc_details;
-
-#define GST_TYPE_ASYNCDISKSRC \
-  (gst_asyncdisksrc_get_type())
-#define GST_ASYNCDISKSRC(obj) \
-  (GTK_CHECK_CAST((obj),GST_TYPE_ASYNCDISKSRC,GstAsyncDiskSrc))
-#define GST_ASYNCDISKSRC_CLASS(klass) \
-  (GTK_CHECK_CLASS_CAST((klass),GST_TYPE_ASYNCDISKSRC,GstAsyncDiskSrcClass))
-#define GST_IS_ASYNCDISKSRC(obj) \
-  (GTK_CHECK_TYPE((obj),GST_TYPE_ASYNCDISKSRC))
-#define GST_IS_ASYNCDISKSRC_CLASS(obj) \
-  (GTK_CHECK_CLASS_TYPE((klass),GST_TYPE_ASYNCDISKSRC))
-
-typedef enum {
-  GST_ASYNCDISKSRC_OPEN                = GST_ELEMENT_FLAG_LAST,
-
-  GST_ASYNCDISKSRC_FLAG_LAST   = GST_ELEMENT_FLAG_LAST + 2,
-} GstAsyncDiskSrcFlags;
-
-typedef struct _GstAsyncDiskSrc GstAsyncDiskSrc;
-typedef struct _GstAsyncDiskSrcClass GstAsyncDiskSrcClass;
-
-struct _GstAsyncDiskSrc {
-  GstElement element;
-  /* pads */
-  GstPad *srcpad;
-
-  /* filename */
-  gchar *filename;
-  /* fd */
-  gint fd;
-
-  /* mapping parameters */
-  gulong size;                         /* how long is the file? */
-  guchar *map;                         /* where the file is mapped to */
-
-  /* details for fallback synchronous read */
-  gulong curoffset;                    /* current offset in file */
-  gulong bytes_per_read;               /* bytes per read */
-  gboolean new_seek;
-
-  gulong seq;                          /* buffer sequence number */
-};
-
-struct _GstAsyncDiskSrcClass {
-  GstElementClass parent_class;
-};
-
-GtkType gst_asyncdisksrc_get_type(void);
-
-#ifdef __cplusplus
-}
-#endif /* __cplusplus */
-
-
-#endif /* __GST_ASYNCDISKSRC_H__ */
index 5333122905f5a57aa0014264b96fc20121ee2b3d..0703076507fab01cc8114281d2119044fe3a5cd0 100644 (file)
 #include <sys/stat.h>
 #include <fcntl.h>
 #include <unistd.h>
+#include <sys/mman.h>
 
-#include <gstdisksrc.h>
+//#define GST_DEBUG_ENABLED
+
+#include "gstdisksrc.h"
 
 
 GstElementDetails gst_disksrc_details = {
-  "Disk Source",
+  "hronous Disk Source",
   "Source/File",
-  "Synchronous read from a file",
+  "Read from arbitrary point in a file",
   VERSION,
   "Erik Walthinsen <omega@cse.ogi.edu>",
   "(C) 1999",
@@ -53,15 +56,14 @@ enum {
 };
 
 
-static void                    gst_disksrc_class_init          (GstDiskSrcClass *klass);
+static void                    gst_disksrc_class_init  (GstDiskSrcClass *klass);
 static void                    gst_disksrc_init                (GstDiskSrc *disksrc);
 
-static void                    gst_disksrc_set_arg             (GtkObject *object, GtkArg *arg, guint id);
-static void                    gst_disksrc_get_arg             (GtkObject *object, GtkArg *arg, guint id);
-
-static void                    gst_disksrc_close_file          (GstDiskSrc *src);
+static void                    gst_disksrc_set_arg     (GtkObject *object, GtkArg *arg, guint id);
+static void                    gst_disksrc_get_arg     (GtkObject *object, GtkArg *arg, guint id);
 
-static GstBuffer *             gst_disksrc_get                 (GstPad *pad);
+static GstBuffer *             gst_disksrc_get         (GstPad *pad);
+static GstBuffer *             gst_disksrc_get_region  (GstPad *pad, gulong offset, gulong size);
 
 static GstElementStateReturn   gst_disksrc_change_state        (GstElement *element);
 
@@ -70,7 +72,8 @@ static GstElementClass *parent_class = NULL;
 //static guint gst_disksrc_signals[LAST_SIGNAL] = { 0 };
 
 GtkType
-gst_disksrc_get_type(void) {
+gst_disksrc_get_type(void) 
+{
   static GtkType disksrc_type = 0;
 
   if (!disksrc_type) {
@@ -84,7 +87,7 @@ gst_disksrc_get_type(void) {
       (GtkArgGetFunc)gst_disksrc_get_arg,
       (GtkClassInitFunc)NULL,
     };
-    disksrc_type = gtk_type_unique(GST_TYPE_ELEMENT,&disksrc_info);
+    disksrc_type = gtk_type_unique (GST_TYPE_ELEMENT, &disksrc_info);
   }
   return disksrc_type;
 }
@@ -104,9 +107,9 @@ gst_disksrc_class_init (GstDiskSrcClass *klass)
                            GTK_ARG_READWRITE, ARG_LOCATION);
   gtk_object_add_arg_type ("GstDiskSrc::bytesperread", GTK_TYPE_INT,
                            GTK_ARG_READWRITE, ARG_BYTESPERREAD);
-  gtk_object_add_arg_type ("GstDiskSrc::offset", GTK_TYPE_INT,
-                           GTK_ARG_READABLE, ARG_OFFSET);
-  gtk_object_add_arg_type ("GstDiskSrc::size", GTK_TYPE_INT,
+  gtk_object_add_arg_type ("GstDiskSrc::offset", GTK_TYPE_LONG,
+                           GTK_ARG_READWRITE, ARG_OFFSET);
+  gtk_object_add_arg_type ("GstDiskSrc::size", GTK_TYPE_LONG,
                            GTK_ARG_READABLE, ARG_SIZE);
 
   gtkobject_class->set_arg = gst_disksrc_set_arg;
@@ -118,16 +121,20 @@ gst_disksrc_class_init (GstDiskSrcClass *klass)
 static void 
 gst_disksrc_init (GstDiskSrc *disksrc) 
 {
+//  GST_FLAG_SET (disksrc, GST_SRC_);
+
   disksrc->srcpad = gst_pad_new ("src", GST_PAD_SRC);
-  gst_pad_set_get_function(disksrc->srcpad,gst_disksrc_get);
+  gst_pad_set_get_function (disksrc->srcpad,gst_disksrc_get);
+  gst_pad_set_getregion_function (disksrc->srcpad,gst_disksrc_get_region);
   gst_element_add_pad (GST_ELEMENT (disksrc), disksrc->srcpad);
 
   disksrc->filename = NULL;
   disksrc->fd = 0;
+  disksrc->size = 0;
+  disksrc->map = NULL;
   disksrc->curoffset = 0;
   disksrc->bytes_per_read = 4096;
   disksrc->seq = 0;
-  disksrc->size = 0;
   disksrc->new_seek = FALSE;
 }
 
@@ -144,8 +151,8 @@ gst_disksrc_set_arg (GtkObject *object, GtkArg *arg, guint id)
 
   switch(id) {
     case ARG_LOCATION:
-      /* the element must not be playing in order to do this */
-      g_return_if_fail (GST_STATE(src) < GST_STATE_PLAYING);
+      /* the element must be stopped in order to do this */
+      g_return_if_fail (GST_STATE (src) < GST_STATE_PLAYING);
 
       if (src->filename) g_free (src->filename);
       /* clear the filename if we get a NULL (is that possible?) */
@@ -160,13 +167,10 @@ gst_disksrc_set_arg (GtkObject *object, GtkArg *arg, guint id)
     case ARG_BYTESPERREAD:
       src->bytes_per_read = GTK_VALUE_INT (*arg);
       break;
-      /*
     case ARG_OFFSET:
-      src->curoffset = GTK_VALUE_INT(*arg);
-      lseek(src->fd,src->curoffset, SEEK_SET);
+      src->curoffset = GTK_VALUE_LONG (*arg);
       src->new_seek = TRUE;
       break;
-      */
     default:
       break;
   }
@@ -179,6 +183,7 @@ gst_disksrc_get_arg (GtkObject *object, GtkArg *arg, guint id)
 
   /* it's not null if we got it, but it might not be ours */
   g_return_if_fail (GST_IS_DISKSRC (object));
+  
   src = GST_DISKSRC (object);
 
   switch (id) {
@@ -189,10 +194,10 @@ gst_disksrc_get_arg (GtkObject *object, GtkArg *arg, guint id)
       GTK_VALUE_INT (*arg) = src->bytes_per_read;
       break;
     case ARG_OFFSET:
-      GTK_VALUE_INT (*arg) = src->curoffset;
+      GTK_VALUE_LONG (*arg) = src->curoffset;
       break;
     case ARG_SIZE:
-      GTK_VALUE_INT (*arg) = src->size;
+      GTK_VALUE_LONG (*arg) = src->size;
       break;
     default:
       arg->type = GTK_TYPE_INVALID;
@@ -200,127 +205,180 @@ gst_disksrc_get_arg (GtkObject *object, GtkArg *arg, guint id)
   }
 }
 
+/**
+ * gst_disksrc_get:
+ * @pad: #GstPad to push a buffer from
+ *
+ * Push a new buffer from the disksrc at the current offset.
+ */
 static GstBuffer *
 gst_disksrc_get (GstPad *pad) 
 {
   GstDiskSrc *src;
   GstBuffer *buf;
-  glong readbytes;
 
   g_return_val_if_fail (pad != NULL, NULL);
-  src = GST_DISKSRC(gst_pad_get_parent (pad));
+  src = GST_DISKSRC (gst_pad_get_parent (pad));
   g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN), NULL);
-  g_return_val_if_fail (GST_STATE (src) >= GST_STATE_READY, NULL);
+
+  /* deal with EOF state */
+  if (src->curoffset >= src->size) {
+    gst_element_signal_eos (GST_ELEMENT (src));
+    return NULL;
+  }
 
   /* create the buffer */
   // FIXME: should eventually use a bufferpool for this
   buf = gst_buffer_new ();
-  g_return_val_if_fail (buf, NULL);
 
-  /* allocate the space for the buffer data */
-  GST_BUFFER_DATA (buf) = g_malloc (src->bytes_per_read);
-  g_return_val_if_fail (GST_BUFFER_DATA (buf) != NULL, NULL);
+  g_return_val_if_fail (buf != NULL, NULL);
 
-  /* read it in from the file */
-  readbytes = read (src->fd, GST_BUFFER_DATA (buf), src->bytes_per_read);
-  if (readbytes == -1) {
-    perror ("read()");
-    gst_buffer_unref (buf);
-    return NULL;
-  } else if (readbytes == 0) {
-    gst_element_signal_eos (GST_ELEMENT (src));
-    gst_buffer_unref (buf);
-    return NULL;
-  }
+  /* simply set the buffer to point to the correct region of the file */
+  GST_BUFFER_DATA (buf) = src->map + src->curoffset;
+  GST_BUFFER_OFFSET (buf) = src->curoffset;
+  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
 
-  /* if we didn't get as many bytes as we asked for, we're at EOF */
-  if (readbytes < src->bytes_per_read) {
-    GST_BUFFER_FLAG_SET (buf, GST_BUFFER_EOS);
-    GST_DEBUG (0,"setting GST_BUFFER_EOS\n");
-  }
+  if ((src->curoffset + src->bytes_per_read) > src->size) {
+    GST_BUFFER_SIZE (buf) = src->size - src->curoffset;
+    // FIXME: set the buffer's EOF bit here
+  } else
+    GST_BUFFER_SIZE (buf) = src->bytes_per_read;
+
+  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, src->curoffset, GST_BUFFER_SIZE (buf));
+
+  //gst_util_dump_mem (GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
+
+  src->curoffset += GST_BUFFER_SIZE (buf);
 
-  /* if we have a new buffer from a seek, mark it */
   if (src->new_seek) {
     GST_BUFFER_FLAG_SET (buf, GST_BUFFER_FLUSH);
+    GST_DEBUG (0,"new seek\n");
     src->new_seek = FALSE;
   }
 
-  GST_BUFFER_OFFSET (buf) = src->curoffset;
-  GST_BUFFER_SIZE (buf) = readbytes;
-  src->curoffset += readbytes;
+  /* we're done, return the buffer */
+  return buf;
+}
 
-  GST_DEBUG (0,"pushing %d bytes with offset %d\n", GST_BUFFER_SIZE(buf), GST_BUFFER_OFFSET (buf));
-  /* we're done, push the buffer off now */
-  GST_DEBUG (0,"returning %d bytes with offset %d done\n", GST_BUFFER_SIZE(buf), GST_BUFFER_OFFSET (buf));
+/**
+ * gst_disksrc_get_region:
+ * @src: #GstSrc to push a buffer from
+ * @offset: offset in file
+ * @size: number of bytes
+ *
+ * Push a new buffer from the disksrc of given size at given offset.
+ */
+static GstBuffer *
+gst_disksrc_get_region (GstPad *pad, gulong offset, gulong size) 
+{
+  GstDiskSrc *src;
+  GstBuffer *buf;
+
+  g_return_val_if_fail (pad != NULL, NULL);
+
+  src = GST_DISKSRC (gst_pad_get_parent (pad));
+
+  g_return_val_if_fail (GST_IS_DISKSRC (src), NULL);
+  g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN), NULL);
+  
+  /* deal with EOF state */
+  if (offset >= src->size) {
+    gst_element_signal_eos (GST_ELEMENT (src));
+    return NULL;
+  }
+
+  /* create the buffer */
+  // FIXME: should eventually use a bufferpool for this
+  buf = gst_buffer_new ();
+  g_return_val_if_fail (buf != NULL, NULL);
+
+  /* simply set the buffer to point to the correct region of the file */
+  GST_BUFFER_DATA (buf) = src->map + offset;
+  GST_BUFFER_OFFSET (buf) = offset;
+  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
+
+  if ((offset + size) > src->size) {
+    GST_BUFFER_SIZE (buf) = src->size - offset;
+    // FIXME: set the buffer's EOF bit here
+  } else
+    GST_BUFFER_SIZE (buf) = size;
+
+  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, offset, GST_BUFFER_SIZE (buf));
+
+  /* we're done, return the buffer off now */
   return buf;
 }
 
 
-/* open the file, necessary to go to RUNNING state */
-static gboolean 
-gst_disksrc_open_file (GstDiskSrc *src) 
+/* open the file and mmap it, necessary to go to READY state */
+static 
+gboolean gst_disksrc_open_file (GstDiskSrc *src) 
 {
-  struct stat f_stat;
-
-  g_return_val_if_fail (!GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN), FALSE);
-  g_return_val_if_fail (src->filename != NULL, FALSE);
+  g_return_val_if_fail (!GST_FLAG_IS_SET (src ,GST_DISKSRC_OPEN), FALSE);
 
   /* open the file */
   src->fd = open (src->filename, O_RDONLY);
   if (src->fd < 0) {
-    perror ("open()");
-    gst_element_error (GST_ELEMENT (src), "opening file");
+    perror ("open");
+    gst_element_error (GST_ELEMENT (src), g_strconcat("opening file \"", src->filename, "\"", NULL));
     return FALSE;
+  } else {
+    /* find the file length */
+    src->size = lseek (src->fd, 0, SEEK_END);
+    lseek (src->fd, 0, SEEK_SET);
+    /* map the file into memory */
+    src->map = mmap (NULL, src->size, PROT_READ, MAP_SHARED, src->fd, 0);
+    madvise (src->map,src->size, 2);
+    /* collapse state if that failed */
+    if (src->map == NULL) {
+      close (src->fd);
+      gst_element_error (GST_ELEMENT (src),"mmapping file");
+      return FALSE;
+    }
+    GST_FLAG_SET (src, GST_DISKSRC_OPEN);
+    src->new_seek = TRUE;
   }
-  if (fstat (src->fd, &f_stat) < 0) {
-    perror("fstat()");
-  }
-  else {
-    src->size = f_stat.st_size;
-    GST_DEBUG (0,"gstdisksrc: file size %ld\n", src->size);
-  }
-  GST_FLAG_SET (src, GST_DISKSRC_OPEN);
   return TRUE;
 }
 
-/* close the file */
+/* unmap and close the file */
 static void 
 gst_disksrc_close_file (GstDiskSrc *src) 
 {
   g_return_if_fail (GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN));
 
+  /* unmap the file from memory */
+  munmap (src->map, src->size);
   /* close the file */
   close (src->fd);
 
   /* zero out a lot of our state */
   src->fd = 0;
+  src->size = 0;
+  src->map = NULL;
   src->curoffset = 0;
   src->seq = 0;
-  src->size = 0;
+  src->new_seek = FALSE;
 
   GST_FLAG_UNSET (src, GST_DISKSRC_OPEN);
 }
 
+
 static GstElementStateReturn 
 gst_disksrc_change_state (GstElement *element) 
 {
   g_return_val_if_fail (GST_IS_DISKSRC (element), GST_STATE_FAILURE);
 
-  GST_DEBUG (0,"gstdisksrc: state pending %d\n", GST_STATE_PENDING (element));
-
-  /* if going down into NULL state, close the file if it's open */
   if (GST_STATE_PENDING (element) == GST_STATE_NULL) {
     if (GST_FLAG_IS_SET (element, GST_DISKSRC_OPEN))
       gst_disksrc_close_file (GST_DISKSRC (element));
-  /* otherwise (READY or higher) we need to open the file */
   } else {
     if (!GST_FLAG_IS_SET (element, GST_DISKSRC_OPEN)) {
-      if (!gst_disksrc_open_file (GST_DISKSRC (element)))
+      if (!gst_disksrc_open_file (GST_DISKSRC (element))) 
         return GST_STATE_FAILURE;
     }
   }
 
-  /* if we haven't failed already, give the parent class a chance to ;-) */
   if (GST_ELEMENT_CLASS (parent_class)->change_state)
     return GST_ELEMENT_CLASS (parent_class)->change_state (element);
 
index bf0cde111ab5767cc68b4b40a89ae2ac9c67cc5d..e049a10d7b6b3d68db6bdff3ac40a4900742a4b4 100644 (file)
@@ -25,7 +25,6 @@
 #define __GST_DISKSRC_H__
 
 
-#include <config.h>
 #include <gst/gst.h>
 
 
@@ -34,8 +33,7 @@ extern "C" {
 #endif /* __cplusplus */
 
 
-extern GstElementDetails gst_disksrc_details;
-
+GstElementDetails gst_disksrc_details;
 
 #define GST_TYPE_DISKSRC \
   (gst_disksrc_get_type())
@@ -51,7 +49,7 @@ extern GstElementDetails gst_disksrc_details;
 typedef enum {
   GST_DISKSRC_OPEN             = GST_ELEMENT_FLAG_LAST,
 
-  GST_DISKSRC_FLAG_LAST                = GST_ELEMENT_FLAG_LAST+2,
+  GST_DISKSRC_FLAG_LAST        = GST_ELEMENT_FLAG_LAST + 2,
 } GstDiskSrcFlags;
 
 typedef struct _GstDiskSrc GstDiskSrc;
@@ -62,16 +60,21 @@ struct _GstDiskSrc {
   /* pads */
   GstPad *srcpad;
 
-  /* file state */
+  /* filename */
   gchar *filename;
+  /* fd */
   gint fd;
 
+  /* mapping parameters */
+  gulong size;                         /* how long is the file? */
+  guchar *map;                         /* where the file is mapped to */
+
+  /* details for fallback synchronous read */
   gulong curoffset;                    /* current offset in file */
   gulong bytes_per_read;               /* bytes per read */
   gboolean new_seek;
 
   gulong seq;                          /* buffer sequence number */
-  gulong size;                         
 };
 
 struct _GstDiskSrcClass {
index c91a010845e6a8bd88bd96bf377a77442dc7be4f..294793df448884a9113266bb9dc6ee7b3fbff83f 100644 (file)
@@ -23,7 +23,6 @@
 
 #include <gst/gst.h>
 
-#include "gstasyncdisksrc.h"
 #include "gstaudiosink.h"
 #include "gstaudiosrc.h"
 #include "gstdisksrc.h"
@@ -49,18 +48,18 @@ struct _elements_entry {
 };
 
 static struct _elements_entry _elements[] = {
-  { "fakesrc",             gst_fakesrc_get_type,       &gst_fakesrc_details,      NULL },
-  { "fakesink",     gst_fakesink_get_type,     &gst_fakesink_details,     NULL },
-  { "asyncdisksrc", gst_asyncdisksrc_get_type,         &gst_asyncdisksrc_details, NULL },
-  { "audiosink",    gst_audiosink_get_type,    &gst_audiosink_details,    gst_audiosink_factory_init },
-  { "audiosrc",     gst_audiosrc_get_type,     &gst_audiosrc_details,     NULL },
-  { "disksrc",             gst_disksrc_get_type,       &gst_disksrc_details,      NULL },
-  { "identity",     gst_identity_get_type,     &gst_identity_details,     NULL },
-  { "fdsink",       gst_fdsink_get_type,       &gst_fdsink_details,       NULL },
-  { "fdsrc",       gst_fdsrc_get_type,         &gst_fdsrc_details,        NULL },
-  { "pipefilter",   gst_pipefilter_get_type,   &gst_pipefilter_details,   NULL },
-  { "sinesrc",             gst_sinesrc_get_type,       &gst_sinesrc_details,      NULL },
-  { "tee",                 gst_tee_get_type,           &gst_tee_details,          gst_tee_factory_init },
+  { "fakesrc",             gst_fakesrc_get_type,       &gst_fakesrc_details,           NULL },
+  { "fakesink",     gst_fakesink_get_type,     &gst_fakesink_details,          NULL },
+  { "asyncdisksrc", gst_disksrc_get_type,      &gst_disksrc_details,           NULL },
+  { "audiosink",    gst_audiosink_get_type,    &gst_audiosink_details,         gst_audiosink_factory_init },
+  { "audiosrc",     gst_audiosrc_get_type,     &gst_audiosrc_details,          NULL },
+  { "disksrc",             gst_disksrc_get_type,       &gst_disksrc_details,           NULL },
+  { "identity",     gst_identity_get_type,     &gst_identity_details,          NULL },
+  { "fdsink",       gst_fdsink_get_type,       &gst_fdsink_details,            NULL },
+  { "fdsrc",       gst_fdsrc_get_type,         &gst_fdsrc_details,             NULL },
+  { "pipefilter",   gst_pipefilter_get_type,   &gst_pipefilter_details,        NULL },
+  { "sinesrc",             gst_sinesrc_get_type,       &gst_sinesrc_details,           NULL },
+  { "tee",                 gst_tee_get_type,           &gst_tee_details,               gst_tee_factory_init },
 
 #if HAVE_LIBGHTTP
   { "httpsrc",             gst_httpsrc_get_type,       &gst_httpsrc_details,      NULL },
index a6292d30440ea003215884b73254261761729de5..0c5b9e6810fad9d6d48cfdd54e9913fc4f277b38 100644 (file)
@@ -15,7 +15,6 @@ libgstelements_la_SOURCES =   \
        gstidentity.c           \
        gstfakesink.c           \
        gstdisksrc.c            \
-       gstasyncdisksrc.c       \
        gstfdsrc.c              \
        gstaudiosink.c          \
        gstaudiosrc.c           \
@@ -30,7 +29,6 @@ noinst_HEADERS =              \
        gstidentity.h           \
        gstfakesink.h           \
        gstdisksrc.h            \
-       gstasyncdisksrc.h       \
        gstfdsrc.h              \
        gsthttpsrc.h            \
        gstaudiosink.h          \
diff --git a/plugins/elements/gstasyncdisksrc.c b/plugins/elements/gstasyncdisksrc.c
deleted file mode 100644 (file)
index f02071d..0000000
+++ /dev/null
@@ -1,386 +0,0 @@
-/* GStreamer
- * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
- *                    2000 Wim Taymans <wtay@chello.be>
- *
- * gstasyncdisksrc.c: 
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Library General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
- * Library General Public License for more details.
- *
- * You should have received a copy of the GNU Library General Public
- * License along with this library; if not, write to the
- * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
- */
-
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <unistd.h>
-#include <sys/mman.h>
-
-//#define GST_DEBUG_ENABLED
-
-#include "gstasyncdisksrc.h"
-
-
-GstElementDetails gst_asyncdisksrc_details = {
-  "Asynchronous Disk Source",
-  "Source/File",
-  "Read from arbitrary point in a file",
-  VERSION,
-  "Erik Walthinsen <omega@cse.ogi.edu>",
-  "(C) 1999",
-};
-
-
-/* AsyncDiskSrc signals and args */
-enum {
-  /* FILL ME */
-  LAST_SIGNAL
-};
-
-enum {
-  ARG_0,
-  ARG_LOCATION,
-  ARG_BYTESPERREAD,
-  ARG_OFFSET,
-  ARG_SIZE,
-};
-
-
-static void                    gst_asyncdisksrc_class_init     (GstAsyncDiskSrcClass *klass);
-static void                    gst_asyncdisksrc_init           (GstAsyncDiskSrc *asyncdisksrc);
-
-static void                    gst_asyncdisksrc_set_arg        (GtkObject *object, GtkArg *arg, guint id);
-static void                    gst_asyncdisksrc_get_arg        (GtkObject *object, GtkArg *arg, guint id);
-
-static GstBuffer *             gst_asyncdisksrc_get            (GstPad *pad);
-static GstBuffer *             gst_asyncdisksrc_get_region     (GstPad *pad, gulong offset, gulong size);
-
-static GstElementStateReturn   gst_asyncdisksrc_change_state   (GstElement *element);
-
-
-static GstElementClass *parent_class = NULL;
-//static guint gst_asyncdisksrc_signals[LAST_SIGNAL] = { 0 };
-
-GtkType
-gst_asyncdisksrc_get_type(void) 
-{
-  static GtkType asyncdisksrc_type = 0;
-
-  if (!asyncdisksrc_type) {
-    static const GtkTypeInfo asyncdisksrc_info = {
-      "GstAsyncDiskSrc",
-      sizeof(GstAsyncDiskSrc),
-      sizeof(GstAsyncDiskSrcClass),
-      (GtkClassInitFunc)gst_asyncdisksrc_class_init,
-      (GtkObjectInitFunc)gst_asyncdisksrc_init,
-      (GtkArgSetFunc)gst_asyncdisksrc_set_arg,
-      (GtkArgGetFunc)gst_asyncdisksrc_get_arg,
-      (GtkClassInitFunc)NULL,
-    };
-    asyncdisksrc_type = gtk_type_unique (GST_TYPE_ELEMENT, &asyncdisksrc_info);
-  }
-  return asyncdisksrc_type;
-}
-
-static void
-gst_asyncdisksrc_class_init (GstAsyncDiskSrcClass *klass) 
-{
-  GtkObjectClass *gtkobject_class;
-  GstElementClass *gstelement_class;
-
-  gtkobject_class = (GtkObjectClass*)klass;
-  gstelement_class = (GstElementClass*)klass;
-
-  parent_class = gtk_type_class (GST_TYPE_ELEMENT);
-
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::location", GST_TYPE_FILENAME,
-                           GTK_ARG_READWRITE, ARG_LOCATION);
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::bytesperread", GTK_TYPE_INT,
-                           GTK_ARG_READWRITE, ARG_BYTESPERREAD);
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::offset", GTK_TYPE_LONG,
-                           GTK_ARG_READWRITE, ARG_OFFSET);
-  gtk_object_add_arg_type ("GstAsyncDiskSrc::size", GTK_TYPE_LONG,
-                           GTK_ARG_READABLE, ARG_SIZE);
-
-  gtkobject_class->set_arg = gst_asyncdisksrc_set_arg;
-  gtkobject_class->get_arg = gst_asyncdisksrc_get_arg;
-
-  gstelement_class->change_state = gst_asyncdisksrc_change_state;
-}
-
-static void 
-gst_asyncdisksrc_init (GstAsyncDiskSrc *asyncdisksrc) 
-{
-//  GST_FLAG_SET (asyncdisksrc, GST_SRC_ASYNC);
-
-  asyncdisksrc->srcpad = gst_pad_new ("src", GST_PAD_SRC);
-  gst_pad_set_get_function (asyncdisksrc->srcpad,gst_asyncdisksrc_get);
-  gst_pad_set_getregion_function (asyncdisksrc->srcpad,gst_asyncdisksrc_get_region);
-  gst_element_add_pad (GST_ELEMENT (asyncdisksrc), asyncdisksrc->srcpad);
-
-  asyncdisksrc->filename = NULL;
-  asyncdisksrc->fd = 0;
-  asyncdisksrc->size = 0;
-  asyncdisksrc->map = NULL;
-  asyncdisksrc->curoffset = 0;
-  asyncdisksrc->bytes_per_read = 4096;
-  asyncdisksrc->seq = 0;
-  asyncdisksrc->new_seek = FALSE;
-}
-
-
-static void 
-gst_asyncdisksrc_set_arg (GtkObject *object, GtkArg *arg, guint id) 
-{
-  GstAsyncDiskSrc *src;
-
-  /* it's not null if we got it, but it might not be ours */
-  g_return_if_fail (GST_IS_ASYNCDISKSRC (object));
-  
-  src = GST_ASYNCDISKSRC (object);
-
-  switch(id) {
-    case ARG_LOCATION:
-      /* the element must be stopped in order to do this */
-      g_return_if_fail (GST_STATE (src) < GST_STATE_PLAYING);
-
-      if (src->filename) g_free (src->filename);
-      /* clear the filename if we get a NULL (is that possible?) */
-      if (GTK_VALUE_STRING (*arg) == NULL) {
-        gst_element_set_state (GST_ELEMENT (object), GST_STATE_NULL);
-        src->filename = NULL;
-      /* otherwise set the new filename */
-      } else {
-        src->filename = g_strdup (GTK_VALUE_STRING (*arg));
-      }
-      break;
-    case ARG_BYTESPERREAD:
-      src->bytes_per_read = GTK_VALUE_INT (*arg);
-      break;
-    case ARG_OFFSET:
-      src->curoffset = GTK_VALUE_LONG (*arg);
-      src->new_seek = TRUE;
-      break;
-    default:
-      break;
-  }
-}
-
-static void 
-gst_asyncdisksrc_get_arg (GtkObject *object, GtkArg *arg, guint id) 
-{
-  GstAsyncDiskSrc *src;
-
-  /* it's not null if we got it, but it might not be ours */
-  g_return_if_fail (GST_IS_ASYNCDISKSRC (object));
-  
-  src = GST_ASYNCDISKSRC (object);
-
-  switch (id) {
-    case ARG_LOCATION:
-      GTK_VALUE_STRING (*arg) = src->filename;
-      break;
-    case ARG_BYTESPERREAD:
-      GTK_VALUE_INT (*arg) = src->bytes_per_read;
-      break;
-    case ARG_OFFSET:
-      GTK_VALUE_LONG (*arg) = src->curoffset;
-      break;
-    case ARG_SIZE:
-      GTK_VALUE_LONG (*arg) = src->size;
-      break;
-    default:
-      arg->type = GTK_TYPE_INVALID;
-      break;
-  }
-}
-
-/**
- * gst_asyncdisksrc_get:
- * @pad: #GstPad to push a buffer from
- *
- * Push a new buffer from the asyncdisksrc at the current offset.
- */
-static GstBuffer *
-gst_asyncdisksrc_get (GstPad *pad) 
-{
-  GstAsyncDiskSrc *src;
-  GstBuffer *buf;
-
-  g_return_val_if_fail (pad != NULL, NULL);
-  src = GST_ASYNCDISKSRC (gst_pad_get_parent (pad));
-  g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_ASYNCDISKSRC_OPEN), NULL);
-
-  /* deal with EOF state */
-  if (src->curoffset >= src->size) {
-    gst_element_signal_eos (GST_ELEMENT (src));
-    return NULL;
-  }
-
-  /* create the buffer */
-  // FIXME: should eventually use a bufferpool for this
-  buf = gst_buffer_new ();
-
-  g_return_val_if_fail (buf != NULL, NULL);
-
-  /* simply set the buffer to point to the correct region of the file */
-  GST_BUFFER_DATA (buf) = src->map + src->curoffset;
-  GST_BUFFER_OFFSET (buf) = src->curoffset;
-  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
-
-  if ((src->curoffset + src->bytes_per_read) > src->size) {
-    GST_BUFFER_SIZE (buf) = src->size - src->curoffset;
-    // FIXME: set the buffer's EOF bit here
-  } else
-    GST_BUFFER_SIZE (buf) = src->bytes_per_read;
-
-  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, src->curoffset, GST_BUFFER_SIZE (buf));
-
-  //gst_util_dump_mem (GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
-
-  src->curoffset += GST_BUFFER_SIZE (buf);
-
-  if (src->new_seek) {
-    GST_BUFFER_FLAG_SET (buf, GST_BUFFER_FLUSH);
-    GST_DEBUG (0,"new seek\n");
-    src->new_seek = FALSE;
-  }
-
-  /* we're done, return the buffer */
-  return buf;
-}
-
-/**
- * gst_asyncdisksrc_get_region:
- * @src: #GstSrc to push a buffer from
- * @offset: offset in file
- * @size: number of bytes
- *
- * Push a new buffer from the asyncdisksrc of given size at given offset.
- */
-static GstBuffer *
-gst_asyncdisksrc_get_region (GstPad *pad, gulong offset, gulong size) 
-{
-  GstAsyncDiskSrc *src;
-  GstBuffer *buf;
-
-  g_return_val_if_fail (pad != NULL, NULL);
-
-  src = GST_ASYNCDISKSRC (gst_pad_get_parent (pad));
-
-  g_return_val_if_fail (GST_IS_ASYNCDISKSRC (src), NULL);
-  g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_ASYNCDISKSRC_OPEN), NULL);
-  
-  /* deal with EOF state */
-  if (offset >= src->size) {
-    gst_element_signal_eos (GST_ELEMENT (src));
-    return NULL;
-  }
-
-  /* create the buffer */
-  // FIXME: should eventually use a bufferpool for this
-  buf = gst_buffer_new ();
-  g_return_val_if_fail (buf != NULL, NULL);
-
-  /* simply set the buffer to point to the correct region of the file */
-  GST_BUFFER_DATA (buf) = src->map + offset;
-  GST_BUFFER_OFFSET (buf) = offset;
-  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
-
-  if ((offset + size) > src->size) {
-    GST_BUFFER_SIZE (buf) = src->size - offset;
-    // FIXME: set the buffer's EOF bit here
-  } else
-    GST_BUFFER_SIZE (buf) = size;
-
-  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, offset, GST_BUFFER_SIZE (buf));
-
-  /* we're done, return the buffer off now */
-  return buf;
-}
-
-
-/* open the file and mmap it, necessary to go to READY state */
-static 
-gboolean gst_asyncdisksrc_open_file (GstAsyncDiskSrc *src) 
-{
-  g_return_val_if_fail (!GST_FLAG_IS_SET (src ,GST_ASYNCDISKSRC_OPEN), FALSE);
-
-  /* open the file */
-  src->fd = open (src->filename, O_RDONLY);
-  if (src->fd < 0) {
-    perror ("open");
-    gst_element_error (GST_ELEMENT (src), g_strconcat("opening file \"", src->filename, "\"", NULL));
-    return FALSE;
-  } else {
-    /* find the file length */
-    src->size = lseek (src->fd, 0, SEEK_END);
-    lseek (src->fd, 0, SEEK_SET);
-    /* map the file into memory */
-    src->map = mmap (NULL, src->size, PROT_READ, MAP_SHARED, src->fd, 0);
-    madvise (src->map,src->size, 2);
-    /* collapse state if that failed */
-    if (src->map == NULL) {
-      close (src->fd);
-      gst_element_error (GST_ELEMENT (src),"mmapping file");
-      return FALSE;
-    }
-    GST_FLAG_SET (src, GST_ASYNCDISKSRC_OPEN);
-    src->new_seek = TRUE;
-  }
-  return TRUE;
-}
-
-/* unmap and close the file */
-static void 
-gst_asyncdisksrc_close_file (GstAsyncDiskSrc *src) 
-{
-  g_return_if_fail (GST_FLAG_IS_SET (src, GST_ASYNCDISKSRC_OPEN));
-
-  /* unmap the file from memory */
-  munmap (src->map, src->size);
-  /* close the file */
-  close (src->fd);
-
-  /* zero out a lot of our state */
-  src->fd = 0;
-  src->size = 0;
-  src->map = NULL;
-  src->curoffset = 0;
-  src->seq = 0;
-  src->new_seek = FALSE;
-
-  GST_FLAG_UNSET (src, GST_ASYNCDISKSRC_OPEN);
-}
-
-
-static GstElementStateReturn 
-gst_asyncdisksrc_change_state (GstElement *element) 
-{
-  g_return_val_if_fail (GST_IS_ASYNCDISKSRC (element), GST_STATE_FAILURE);
-
-  if (GST_STATE_PENDING (element) == GST_STATE_NULL) {
-    if (GST_FLAG_IS_SET (element, GST_ASYNCDISKSRC_OPEN))
-      gst_asyncdisksrc_close_file (GST_ASYNCDISKSRC (element));
-  } else {
-    if (!GST_FLAG_IS_SET (element, GST_ASYNCDISKSRC_OPEN)) {
-      if (!gst_asyncdisksrc_open_file (GST_ASYNCDISKSRC (element))) 
-        return GST_STATE_FAILURE;
-    }
-  }
-
-  if (GST_ELEMENT_CLASS (parent_class)->change_state)
-    return GST_ELEMENT_CLASS (parent_class)->change_state (element);
-
-  return GST_STATE_SUCCESS;
-}
diff --git a/plugins/elements/gstasyncdisksrc.h b/plugins/elements/gstasyncdisksrc.h
deleted file mode 100644 (file)
index 1d188cb..0000000
+++ /dev/null
@@ -1,91 +0,0 @@
-/* GStreamer
- * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu>
- *                    2000 Wim Taymans <wtay@chello.be>
- *
- * gstasyncdisksrc.h: 
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Library General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
- * Library General Public License for more details.
- *
- * You should have received a copy of the GNU Library General Public
- * License along with this library; if not, write to the
- * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
- */
-
-
-#ifndef __GST_ASYNCDISKSRC_H__
-#define __GST_ASYNCDISKSRC_H__
-
-
-#include <gst/gst.h>
-
-
-#ifdef __cplusplus
-extern "C" {
-#endif /* __cplusplus */
-
-
-GstElementDetails gst_asyncdisksrc_details;
-
-#define GST_TYPE_ASYNCDISKSRC \
-  (gst_asyncdisksrc_get_type())
-#define GST_ASYNCDISKSRC(obj) \
-  (GTK_CHECK_CAST((obj),GST_TYPE_ASYNCDISKSRC,GstAsyncDiskSrc))
-#define GST_ASYNCDISKSRC_CLASS(klass) \
-  (GTK_CHECK_CLASS_CAST((klass),GST_TYPE_ASYNCDISKSRC,GstAsyncDiskSrcClass))
-#define GST_IS_ASYNCDISKSRC(obj) \
-  (GTK_CHECK_TYPE((obj),GST_TYPE_ASYNCDISKSRC))
-#define GST_IS_ASYNCDISKSRC_CLASS(obj) \
-  (GTK_CHECK_CLASS_TYPE((klass),GST_TYPE_ASYNCDISKSRC))
-
-typedef enum {
-  GST_ASYNCDISKSRC_OPEN                = GST_ELEMENT_FLAG_LAST,
-
-  GST_ASYNCDISKSRC_FLAG_LAST   = GST_ELEMENT_FLAG_LAST + 2,
-} GstAsyncDiskSrcFlags;
-
-typedef struct _GstAsyncDiskSrc GstAsyncDiskSrc;
-typedef struct _GstAsyncDiskSrcClass GstAsyncDiskSrcClass;
-
-struct _GstAsyncDiskSrc {
-  GstElement element;
-  /* pads */
-  GstPad *srcpad;
-
-  /* filename */
-  gchar *filename;
-  /* fd */
-  gint fd;
-
-  /* mapping parameters */
-  gulong size;                         /* how long is the file? */
-  guchar *map;                         /* where the file is mapped to */
-
-  /* details for fallback synchronous read */
-  gulong curoffset;                    /* current offset in file */
-  gulong bytes_per_read;               /* bytes per read */
-  gboolean new_seek;
-
-  gulong seq;                          /* buffer sequence number */
-};
-
-struct _GstAsyncDiskSrcClass {
-  GstElementClass parent_class;
-};
-
-GtkType gst_asyncdisksrc_get_type(void);
-
-#ifdef __cplusplus
-}
-#endif /* __cplusplus */
-
-
-#endif /* __GST_ASYNCDISKSRC_H__ */
index 5333122905f5a57aa0014264b96fc20121ee2b3d..0703076507fab01cc8114281d2119044fe3a5cd0 100644 (file)
 #include <sys/stat.h>
 #include <fcntl.h>
 #include <unistd.h>
+#include <sys/mman.h>
 
-#include <gstdisksrc.h>
+//#define GST_DEBUG_ENABLED
+
+#include "gstdisksrc.h"
 
 
 GstElementDetails gst_disksrc_details = {
-  "Disk Source",
+  "hronous Disk Source",
   "Source/File",
-  "Synchronous read from a file",
+  "Read from arbitrary point in a file",
   VERSION,
   "Erik Walthinsen <omega@cse.ogi.edu>",
   "(C) 1999",
@@ -53,15 +56,14 @@ enum {
 };
 
 
-static void                    gst_disksrc_class_init          (GstDiskSrcClass *klass);
+static void                    gst_disksrc_class_init  (GstDiskSrcClass *klass);
 static void                    gst_disksrc_init                (GstDiskSrc *disksrc);
 
-static void                    gst_disksrc_set_arg             (GtkObject *object, GtkArg *arg, guint id);
-static void                    gst_disksrc_get_arg             (GtkObject *object, GtkArg *arg, guint id);
-
-static void                    gst_disksrc_close_file          (GstDiskSrc *src);
+static void                    gst_disksrc_set_arg     (GtkObject *object, GtkArg *arg, guint id);
+static void                    gst_disksrc_get_arg     (GtkObject *object, GtkArg *arg, guint id);
 
-static GstBuffer *             gst_disksrc_get                 (GstPad *pad);
+static GstBuffer *             gst_disksrc_get         (GstPad *pad);
+static GstBuffer *             gst_disksrc_get_region  (GstPad *pad, gulong offset, gulong size);
 
 static GstElementStateReturn   gst_disksrc_change_state        (GstElement *element);
 
@@ -70,7 +72,8 @@ static GstElementClass *parent_class = NULL;
 //static guint gst_disksrc_signals[LAST_SIGNAL] = { 0 };
 
 GtkType
-gst_disksrc_get_type(void) {
+gst_disksrc_get_type(void) 
+{
   static GtkType disksrc_type = 0;
 
   if (!disksrc_type) {
@@ -84,7 +87,7 @@ gst_disksrc_get_type(void) {
       (GtkArgGetFunc)gst_disksrc_get_arg,
       (GtkClassInitFunc)NULL,
     };
-    disksrc_type = gtk_type_unique(GST_TYPE_ELEMENT,&disksrc_info);
+    disksrc_type = gtk_type_unique (GST_TYPE_ELEMENT, &disksrc_info);
   }
   return disksrc_type;
 }
@@ -104,9 +107,9 @@ gst_disksrc_class_init (GstDiskSrcClass *klass)
                            GTK_ARG_READWRITE, ARG_LOCATION);
   gtk_object_add_arg_type ("GstDiskSrc::bytesperread", GTK_TYPE_INT,
                            GTK_ARG_READWRITE, ARG_BYTESPERREAD);
-  gtk_object_add_arg_type ("GstDiskSrc::offset", GTK_TYPE_INT,
-                           GTK_ARG_READABLE, ARG_OFFSET);
-  gtk_object_add_arg_type ("GstDiskSrc::size", GTK_TYPE_INT,
+  gtk_object_add_arg_type ("GstDiskSrc::offset", GTK_TYPE_LONG,
+                           GTK_ARG_READWRITE, ARG_OFFSET);
+  gtk_object_add_arg_type ("GstDiskSrc::size", GTK_TYPE_LONG,
                            GTK_ARG_READABLE, ARG_SIZE);
 
   gtkobject_class->set_arg = gst_disksrc_set_arg;
@@ -118,16 +121,20 @@ gst_disksrc_class_init (GstDiskSrcClass *klass)
 static void 
 gst_disksrc_init (GstDiskSrc *disksrc) 
 {
+//  GST_FLAG_SET (disksrc, GST_SRC_);
+
   disksrc->srcpad = gst_pad_new ("src", GST_PAD_SRC);
-  gst_pad_set_get_function(disksrc->srcpad,gst_disksrc_get);
+  gst_pad_set_get_function (disksrc->srcpad,gst_disksrc_get);
+  gst_pad_set_getregion_function (disksrc->srcpad,gst_disksrc_get_region);
   gst_element_add_pad (GST_ELEMENT (disksrc), disksrc->srcpad);
 
   disksrc->filename = NULL;
   disksrc->fd = 0;
+  disksrc->size = 0;
+  disksrc->map = NULL;
   disksrc->curoffset = 0;
   disksrc->bytes_per_read = 4096;
   disksrc->seq = 0;
-  disksrc->size = 0;
   disksrc->new_seek = FALSE;
 }
 
@@ -144,8 +151,8 @@ gst_disksrc_set_arg (GtkObject *object, GtkArg *arg, guint id)
 
   switch(id) {
     case ARG_LOCATION:
-      /* the element must not be playing in order to do this */
-      g_return_if_fail (GST_STATE(src) < GST_STATE_PLAYING);
+      /* the element must be stopped in order to do this */
+      g_return_if_fail (GST_STATE (src) < GST_STATE_PLAYING);
 
       if (src->filename) g_free (src->filename);
       /* clear the filename if we get a NULL (is that possible?) */
@@ -160,13 +167,10 @@ gst_disksrc_set_arg (GtkObject *object, GtkArg *arg, guint id)
     case ARG_BYTESPERREAD:
       src->bytes_per_read = GTK_VALUE_INT (*arg);
       break;
-      /*
     case ARG_OFFSET:
-      src->curoffset = GTK_VALUE_INT(*arg);
-      lseek(src->fd,src->curoffset, SEEK_SET);
+      src->curoffset = GTK_VALUE_LONG (*arg);
       src->new_seek = TRUE;
       break;
-      */
     default:
       break;
   }
@@ -179,6 +183,7 @@ gst_disksrc_get_arg (GtkObject *object, GtkArg *arg, guint id)
 
   /* it's not null if we got it, but it might not be ours */
   g_return_if_fail (GST_IS_DISKSRC (object));
+  
   src = GST_DISKSRC (object);
 
   switch (id) {
@@ -189,10 +194,10 @@ gst_disksrc_get_arg (GtkObject *object, GtkArg *arg, guint id)
       GTK_VALUE_INT (*arg) = src->bytes_per_read;
       break;
     case ARG_OFFSET:
-      GTK_VALUE_INT (*arg) = src->curoffset;
+      GTK_VALUE_LONG (*arg) = src->curoffset;
       break;
     case ARG_SIZE:
-      GTK_VALUE_INT (*arg) = src->size;
+      GTK_VALUE_LONG (*arg) = src->size;
       break;
     default:
       arg->type = GTK_TYPE_INVALID;
@@ -200,127 +205,180 @@ gst_disksrc_get_arg (GtkObject *object, GtkArg *arg, guint id)
   }
 }
 
+/**
+ * gst_disksrc_get:
+ * @pad: #GstPad to push a buffer from
+ *
+ * Push a new buffer from the disksrc at the current offset.
+ */
 static GstBuffer *
 gst_disksrc_get (GstPad *pad) 
 {
   GstDiskSrc *src;
   GstBuffer *buf;
-  glong readbytes;
 
   g_return_val_if_fail (pad != NULL, NULL);
-  src = GST_DISKSRC(gst_pad_get_parent (pad));
+  src = GST_DISKSRC (gst_pad_get_parent (pad));
   g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN), NULL);
-  g_return_val_if_fail (GST_STATE (src) >= GST_STATE_READY, NULL);
+
+  /* deal with EOF state */
+  if (src->curoffset >= src->size) {
+    gst_element_signal_eos (GST_ELEMENT (src));
+    return NULL;
+  }
 
   /* create the buffer */
   // FIXME: should eventually use a bufferpool for this
   buf = gst_buffer_new ();
-  g_return_val_if_fail (buf, NULL);
 
-  /* allocate the space for the buffer data */
-  GST_BUFFER_DATA (buf) = g_malloc (src->bytes_per_read);
-  g_return_val_if_fail (GST_BUFFER_DATA (buf) != NULL, NULL);
+  g_return_val_if_fail (buf != NULL, NULL);
 
-  /* read it in from the file */
-  readbytes = read (src->fd, GST_BUFFER_DATA (buf), src->bytes_per_read);
-  if (readbytes == -1) {
-    perror ("read()");
-    gst_buffer_unref (buf);
-    return NULL;
-  } else if (readbytes == 0) {
-    gst_element_signal_eos (GST_ELEMENT (src));
-    gst_buffer_unref (buf);
-    return NULL;
-  }
+  /* simply set the buffer to point to the correct region of the file */
+  GST_BUFFER_DATA (buf) = src->map + src->curoffset;
+  GST_BUFFER_OFFSET (buf) = src->curoffset;
+  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
 
-  /* if we didn't get as many bytes as we asked for, we're at EOF */
-  if (readbytes < src->bytes_per_read) {
-    GST_BUFFER_FLAG_SET (buf, GST_BUFFER_EOS);
-    GST_DEBUG (0,"setting GST_BUFFER_EOS\n");
-  }
+  if ((src->curoffset + src->bytes_per_read) > src->size) {
+    GST_BUFFER_SIZE (buf) = src->size - src->curoffset;
+    // FIXME: set the buffer's EOF bit here
+  } else
+    GST_BUFFER_SIZE (buf) = src->bytes_per_read;
+
+  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, src->curoffset, GST_BUFFER_SIZE (buf));
+
+  //gst_util_dump_mem (GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
+
+  src->curoffset += GST_BUFFER_SIZE (buf);
 
-  /* if we have a new buffer from a seek, mark it */
   if (src->new_seek) {
     GST_BUFFER_FLAG_SET (buf, GST_BUFFER_FLUSH);
+    GST_DEBUG (0,"new seek\n");
     src->new_seek = FALSE;
   }
 
-  GST_BUFFER_OFFSET (buf) = src->curoffset;
-  GST_BUFFER_SIZE (buf) = readbytes;
-  src->curoffset += readbytes;
+  /* we're done, return the buffer */
+  return buf;
+}
 
-  GST_DEBUG (0,"pushing %d bytes with offset %d\n", GST_BUFFER_SIZE(buf), GST_BUFFER_OFFSET (buf));
-  /* we're done, push the buffer off now */
-  GST_DEBUG (0,"returning %d bytes with offset %d done\n", GST_BUFFER_SIZE(buf), GST_BUFFER_OFFSET (buf));
+/**
+ * gst_disksrc_get_region:
+ * @src: #GstSrc to push a buffer from
+ * @offset: offset in file
+ * @size: number of bytes
+ *
+ * Push a new buffer from the disksrc of given size at given offset.
+ */
+static GstBuffer *
+gst_disksrc_get_region (GstPad *pad, gulong offset, gulong size) 
+{
+  GstDiskSrc *src;
+  GstBuffer *buf;
+
+  g_return_val_if_fail (pad != NULL, NULL);
+
+  src = GST_DISKSRC (gst_pad_get_parent (pad));
+
+  g_return_val_if_fail (GST_IS_DISKSRC (src), NULL);
+  g_return_val_if_fail (GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN), NULL);
+  
+  /* deal with EOF state */
+  if (offset >= src->size) {
+    gst_element_signal_eos (GST_ELEMENT (src));
+    return NULL;
+  }
+
+  /* create the buffer */
+  // FIXME: should eventually use a bufferpool for this
+  buf = gst_buffer_new ();
+  g_return_val_if_fail (buf != NULL, NULL);
+
+  /* simply set the buffer to point to the correct region of the file */
+  GST_BUFFER_DATA (buf) = src->map + offset;
+  GST_BUFFER_OFFSET (buf) = offset;
+  GST_BUFFER_FLAG_SET (buf, GST_BUFFER_DONTFREE);
+
+  if ((offset + size) > src->size) {
+    GST_BUFFER_SIZE (buf) = src->size - offset;
+    // FIXME: set the buffer's EOF bit here
+  } else
+    GST_BUFFER_SIZE (buf) = size;
+
+  GST_DEBUG (0,"map %p, offset %ld, size %d\n", src->map, offset, GST_BUFFER_SIZE (buf));
+
+  /* we're done, return the buffer off now */
   return buf;
 }
 
 
-/* open the file, necessary to go to RUNNING state */
-static gboolean 
-gst_disksrc_open_file (GstDiskSrc *src) 
+/* open the file and mmap it, necessary to go to READY state */
+static 
+gboolean gst_disksrc_open_file (GstDiskSrc *src) 
 {
-  struct stat f_stat;
-
-  g_return_val_if_fail (!GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN), FALSE);
-  g_return_val_if_fail (src->filename != NULL, FALSE);
+  g_return_val_if_fail (!GST_FLAG_IS_SET (src ,GST_DISKSRC_OPEN), FALSE);
 
   /* open the file */
   src->fd = open (src->filename, O_RDONLY);
   if (src->fd < 0) {
-    perror ("open()");
-    gst_element_error (GST_ELEMENT (src), "opening file");
+    perror ("open");
+    gst_element_error (GST_ELEMENT (src), g_strconcat("opening file \"", src->filename, "\"", NULL));
     return FALSE;
+  } else {
+    /* find the file length */
+    src->size = lseek (src->fd, 0, SEEK_END);
+    lseek (src->fd, 0, SEEK_SET);
+    /* map the file into memory */
+    src->map = mmap (NULL, src->size, PROT_READ, MAP_SHARED, src->fd, 0);
+    madvise (src->map,src->size, 2);
+    /* collapse state if that failed */
+    if (src->map == NULL) {
+      close (src->fd);
+      gst_element_error (GST_ELEMENT (src),"mmapping file");
+      return FALSE;
+    }
+    GST_FLAG_SET (src, GST_DISKSRC_OPEN);
+    src->new_seek = TRUE;
   }
-  if (fstat (src->fd, &f_stat) < 0) {
-    perror("fstat()");
-  }
-  else {
-    src->size = f_stat.st_size;
-    GST_DEBUG (0,"gstdisksrc: file size %ld\n", src->size);
-  }
-  GST_FLAG_SET (src, GST_DISKSRC_OPEN);
   return TRUE;
 }
 
-/* close the file */
+/* unmap and close the file */
 static void 
 gst_disksrc_close_file (GstDiskSrc *src) 
 {
   g_return_if_fail (GST_FLAG_IS_SET (src, GST_DISKSRC_OPEN));
 
+  /* unmap the file from memory */
+  munmap (src->map, src->size);
   /* close the file */
   close (src->fd);
 
   /* zero out a lot of our state */
   src->fd = 0;
+  src->size = 0;
+  src->map = NULL;
   src->curoffset = 0;
   src->seq = 0;
-  src->size = 0;
+  src->new_seek = FALSE;
 
   GST_FLAG_UNSET (src, GST_DISKSRC_OPEN);
 }
 
+
 static GstElementStateReturn 
 gst_disksrc_change_state (GstElement *element) 
 {
   g_return_val_if_fail (GST_IS_DISKSRC (element), GST_STATE_FAILURE);
 
-  GST_DEBUG (0,"gstdisksrc: state pending %d\n", GST_STATE_PENDING (element));
-
-  /* if going down into NULL state, close the file if it's open */
   if (GST_STATE_PENDING (element) == GST_STATE_NULL) {
     if (GST_FLAG_IS_SET (element, GST_DISKSRC_OPEN))
       gst_disksrc_close_file (GST_DISKSRC (element));
-  /* otherwise (READY or higher) we need to open the file */
   } else {
     if (!GST_FLAG_IS_SET (element, GST_DISKSRC_OPEN)) {
-      if (!gst_disksrc_open_file (GST_DISKSRC (element)))
+      if (!gst_disksrc_open_file (GST_DISKSRC (element))) 
         return GST_STATE_FAILURE;
     }
   }
 
-  /* if we haven't failed already, give the parent class a chance to ;-) */
   if (GST_ELEMENT_CLASS (parent_class)->change_state)
     return GST_ELEMENT_CLASS (parent_class)->change_state (element);
 
index bf0cde111ab5767cc68b4b40a89ae2ac9c67cc5d..e049a10d7b6b3d68db6bdff3ac40a4900742a4b4 100644 (file)
@@ -25,7 +25,6 @@
 #define __GST_DISKSRC_H__
 
 
-#include <config.h>
 #include <gst/gst.h>
 
 
@@ -34,8 +33,7 @@ extern "C" {
 #endif /* __cplusplus */
 
 
-extern GstElementDetails gst_disksrc_details;
-
+GstElementDetails gst_disksrc_details;
 
 #define GST_TYPE_DISKSRC \
   (gst_disksrc_get_type())
@@ -51,7 +49,7 @@ extern GstElementDetails gst_disksrc_details;
 typedef enum {
   GST_DISKSRC_OPEN             = GST_ELEMENT_FLAG_LAST,
 
-  GST_DISKSRC_FLAG_LAST                = GST_ELEMENT_FLAG_LAST+2,
+  GST_DISKSRC_FLAG_LAST        = GST_ELEMENT_FLAG_LAST + 2,
 } GstDiskSrcFlags;
 
 typedef struct _GstDiskSrc GstDiskSrc;
@@ -62,16 +60,21 @@ struct _GstDiskSrc {
   /* pads */
   GstPad *srcpad;
 
-  /* file state */
+  /* filename */
   gchar *filename;
+  /* fd */
   gint fd;
 
+  /* mapping parameters */
+  gulong size;                         /* how long is the file? */
+  guchar *map;                         /* where the file is mapped to */
+
+  /* details for fallback synchronous read */
   gulong curoffset;                    /* current offset in file */
   gulong bytes_per_read;               /* bytes per read */
   gboolean new_seek;
 
   gulong seq;                          /* buffer sequence number */
-  gulong size;                         
 };
 
 struct _GstDiskSrcClass {
index c91a010845e6a8bd88bd96bf377a77442dc7be4f..294793df448884a9113266bb9dc6ee7b3fbff83f 100644 (file)
@@ -23,7 +23,6 @@
 
 #include <gst/gst.h>
 
-#include "gstasyncdisksrc.h"
 #include "gstaudiosink.h"
 #include "gstaudiosrc.h"
 #include "gstdisksrc.h"
@@ -49,18 +48,18 @@ struct _elements_entry {
 };
 
 static struct _elements_entry _elements[] = {
-  { "fakesrc",             gst_fakesrc_get_type,       &gst_fakesrc_details,      NULL },
-  { "fakesink",     gst_fakesink_get_type,     &gst_fakesink_details,     NULL },
-  { "asyncdisksrc", gst_asyncdisksrc_get_type,         &gst_asyncdisksrc_details, NULL },
-  { "audiosink",    gst_audiosink_get_type,    &gst_audiosink_details,    gst_audiosink_factory_init },
-  { "audiosrc",     gst_audiosrc_get_type,     &gst_audiosrc_details,     NULL },
-  { "disksrc",             gst_disksrc_get_type,       &gst_disksrc_details,      NULL },
-  { "identity",     gst_identity_get_type,     &gst_identity_details,     NULL },
-  { "fdsink",       gst_fdsink_get_type,       &gst_fdsink_details,       NULL },
-  { "fdsrc",       gst_fdsrc_get_type,         &gst_fdsrc_details,        NULL },
-  { "pipefilter",   gst_pipefilter_get_type,   &gst_pipefilter_details,   NULL },
-  { "sinesrc",             gst_sinesrc_get_type,       &gst_sinesrc_details,      NULL },
-  { "tee",                 gst_tee_get_type,           &gst_tee_details,          gst_tee_factory_init },
+  { "fakesrc",             gst_fakesrc_get_type,       &gst_fakesrc_details,           NULL },
+  { "fakesink",     gst_fakesink_get_type,     &gst_fakesink_details,          NULL },
+  { "asyncdisksrc", gst_disksrc_get_type,      &gst_disksrc_details,           NULL },
+  { "audiosink",    gst_audiosink_get_type,    &gst_audiosink_details,         gst_audiosink_factory_init },
+  { "audiosrc",     gst_audiosrc_get_type,     &gst_audiosrc_details,          NULL },
+  { "disksrc",             gst_disksrc_get_type,       &gst_disksrc_details,           NULL },
+  { "identity",     gst_identity_get_type,     &gst_identity_details,          NULL },
+  { "fdsink",       gst_fdsink_get_type,       &gst_fdsink_details,            NULL },
+  { "fdsrc",       gst_fdsrc_get_type,         &gst_fdsrc_details,             NULL },
+  { "pipefilter",   gst_pipefilter_get_type,   &gst_pipefilter_details,        NULL },
+  { "sinesrc",             gst_sinesrc_get_type,       &gst_sinesrc_details,           NULL },
+  { "tee",                 gst_tee_get_type,           &gst_tee_details,               gst_tee_factory_init },
 
 #if HAVE_LIBGHTTP
   { "httpsrc",             gst_httpsrc_get_type,       &gst_httpsrc_details,      NULL },