mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-09 08:55:33 +00:00
9920b88c3f
Original commit message from CVS: This is the audio/video sync release. Changed the mpegvideoparser to parse complete pictures. Added the PTS timestamps to the pictures. Added PTS timestamps to the MPEG audio frames. Made the clock a littlebit better. Gstplay now uses two more threads one for video, one for audio playback. Added the first QoS callbacks for the pads. hopefully fix an mmx compilation problem.
440 lines
11 KiB
C
440 lines
11 KiB
C
/* Gnome-Streamer
|
|
* Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
|
|
//#define DEBUG_ENABLED
|
|
#include <gst/gst.h>
|
|
#include <gst/gstpad.h>
|
|
#include <gst/gstelement.h>
|
|
#include <gst/gsttype.h>
|
|
|
|
|
|
/* Pad signals and args */
|
|
enum {
|
|
/* FILL ME */
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
enum {
|
|
ARG_0,
|
|
/* FILL ME */
|
|
};
|
|
|
|
|
|
static void gst_pad_class_init(GstPadClass *klass);
|
|
static void gst_pad_init(GstPad *pad);
|
|
static void gst_pad_real_destroy(GtkObject *object);
|
|
|
|
|
|
static GstObject *parent_class = NULL;
|
|
//static guint gst_pad_signals[LAST_SIGNAL] = { 0 };
|
|
|
|
GtkType
|
|
gst_pad_get_type(void) {
|
|
static GtkType pad_type = 0;
|
|
|
|
if (!pad_type) {
|
|
static const GtkTypeInfo pad_info = {
|
|
"GstPad",
|
|
sizeof(GstPad),
|
|
sizeof(GstPadClass),
|
|
(GtkClassInitFunc)gst_pad_class_init,
|
|
(GtkObjectInitFunc)gst_pad_init,
|
|
(GtkArgSetFunc)NULL,
|
|
(GtkArgGetFunc)NULL,
|
|
(GtkClassInitFunc)NULL,
|
|
};
|
|
pad_type = gtk_type_unique(GST_TYPE_OBJECT,&pad_info);
|
|
}
|
|
return pad_type;
|
|
}
|
|
|
|
static void
|
|
gst_pad_class_init(GstPadClass *klass) {
|
|
GtkObjectClass *gtkobject_class;
|
|
|
|
gtkobject_class = (GtkObjectClass*)klass;
|
|
|
|
parent_class = gtk_type_class(GST_TYPE_OBJECT);
|
|
|
|
gtkobject_class->destroy = gst_pad_real_destroy;
|
|
}
|
|
|
|
static void gst_pad_init(GstPad *pad) {
|
|
pad->type = 0;
|
|
pad->direction = GST_PAD_UNKNOWN;
|
|
pad->peer = NULL;
|
|
pad->chain = NULL;
|
|
pad->pull = NULL;
|
|
pad->qos = NULL;
|
|
pad->parent = NULL;
|
|
pad->ghostparents = NULL;
|
|
}
|
|
|
|
/**
|
|
* gst_pad_new:
|
|
* @name: name of new pad
|
|
* @direction: either GST_PAD_SRC or GST_PAD_SINK
|
|
*
|
|
* Create a new pad with given name.
|
|
*
|
|
* Returns: new pad
|
|
*/
|
|
GstPad *gst_pad_new(gchar *name,GstPadDirection direction) {
|
|
GstPad *pad;
|
|
|
|
g_return_val_if_fail(name != NULL, NULL);
|
|
g_return_val_if_fail(direction != GST_PAD_UNKNOWN, NULL);
|
|
|
|
pad = GST_PAD(gtk_type_new(gst_pad_get_type()));
|
|
pad->name = g_strdup(name);
|
|
pad->direction = direction;
|
|
return pad;
|
|
}
|
|
|
|
/**
|
|
* gst_pad_get_direction:
|
|
* @pad: the Pad to get the direction from
|
|
*
|
|
* get the direction of the pad
|
|
*
|
|
* Returns: the direction of the pad
|
|
*/
|
|
GstPadDirection gst_pad_get_direction(GstPad *pad) {
|
|
g_return_val_if_fail(pad != NULL, GST_PAD_UNKNOWN);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), GST_PAD_UNKNOWN);
|
|
|
|
return pad->direction;
|
|
}
|
|
|
|
/**
|
|
* gst_pad_set_name:
|
|
* @pad: the pad to set the name of
|
|
* @name: the name of the pad
|
|
*
|
|
* set the name of a pad
|
|
*/
|
|
void gst_pad_set_name(GstPad *pad,gchar *name) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
|
|
if (pad->name != NULL)
|
|
g_free(pad->name);
|
|
|
|
pad->name = g_strdup(name);
|
|
}
|
|
|
|
/**
|
|
* gst_pad_get_name:
|
|
* @pad: the pad to get the name of
|
|
*
|
|
* get the name of a pad
|
|
*
|
|
* Returns: the name of the pad
|
|
*/
|
|
gchar *gst_pad_get_name(GstPad *pad) {
|
|
g_return_val_if_fail(pad != NULL, NULL);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), NULL);
|
|
|
|
return pad->name;
|
|
}
|
|
|
|
void gst_pad_set_pull_function(GstPad *pad,GstPadPullFunction pull) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
|
|
pad->pull = pull;
|
|
}
|
|
|
|
void gst_pad_set_chain_function(GstPad *pad,GstPadChainFunction chain) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
|
|
pad->chain = chain;
|
|
}
|
|
|
|
void gst_pad_set_qos_function(GstPad *pad,GstPadQoSFunction qos) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
|
|
pad->qos = qos;
|
|
}
|
|
|
|
void gst_pad_push(GstPad *pad,GstBuffer *buffer) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
g_return_if_fail(GST_PAD_CONNECTED(pad));
|
|
g_return_if_fail(buffer != NULL);
|
|
|
|
gst_trace_add_entry(NULL,0,buffer,"push buffer");
|
|
// if the chain function exists for the pad, call it directly
|
|
if (pad->chain)
|
|
(pad->chain)(pad->peer,buffer);
|
|
// else we're likely going to have to coroutine it
|
|
else {
|
|
pad->peer->bufpen = buffer;
|
|
g_print("GstPad: would switch to a coroutine here...\n");
|
|
if (!GST_IS_ELEMENT(pad->peer->parent))
|
|
g_print("GstPad: eek, this isn't an element!\n");
|
|
if (GST_ELEMENT(pad->peer->parent)->threadstate != NULL)
|
|
cothread_switch(GST_ELEMENT(pad->peer->parent)->threadstate);
|
|
}
|
|
}
|
|
|
|
GstBuffer *gst_pad_pull(GstPad *pad) {
|
|
GstBuffer *buf;
|
|
GstElement *peerparent;
|
|
cothread_state *state;
|
|
|
|
g_return_val_if_fail(pad != NULL, NULL);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), NULL);
|
|
|
|
// if the pull function exists for the pad, call it directly
|
|
if (pad->pull) {
|
|
return (pad->pull)(pad->peer);
|
|
// else we're likely going to have to coroutine it
|
|
} else if (pad->bufpen == NULL) {
|
|
g_print("GstPad: no buffer available, will have to do something about it\n");
|
|
peerparent = GST_ELEMENT(pad->peer->parent);
|
|
// if they're a cothread too, we can just switch to them
|
|
if (peerparent->threadstate != NULL) {
|
|
cothread_switch(peerparent->threadstate);
|
|
// otherwise we have to switch to the main thread
|
|
} else {
|
|
state = cothread_main(GST_ELEMENT(pad->parent)->threadstate->ctx);
|
|
g_print("GstPad: switching to supposed 0th thread at %p\n",state);
|
|
cothread_switch(state);
|
|
}
|
|
} else {
|
|
g_print("GstPad: buffer available, pulling\n");
|
|
buf = pad->bufpen;
|
|
pad->bufpen = NULL;
|
|
return buf;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void gst_pad_chain(GstPad *pad) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
g_return_if_fail(pad->peer != NULL);
|
|
g_return_if_fail(pad->chain != NULL);
|
|
|
|
if (pad->bufpen)
|
|
(pad->chain)(pad,pad->bufpen);
|
|
}
|
|
|
|
/**
|
|
* gst_pad_handle_qos:
|
|
* @element: element to change state of
|
|
* @state: new element state
|
|
*
|
|
*/
|
|
void gst_pad_handle_qos(GstPad *pad,
|
|
glong qos_message)
|
|
{
|
|
GstElement *element;
|
|
GList *pads;
|
|
GstPad *target_pad;
|
|
|
|
DEBUG("gst_pad_handle_qos(\"%s\",%08ld)\n", GST_ELEMENT(pad->parent)->name,qos_message);
|
|
|
|
if (pad->qos) {
|
|
(pad->qos)(pad,qos_message);
|
|
}
|
|
else {
|
|
element = GST_ELEMENT(pad->peer->parent);
|
|
|
|
pads = element->pads;
|
|
DEBUG("gst_pad_handle_qos recurse(\"%s\",%08ld)\n", element->name,qos_message);
|
|
while (pads) {
|
|
target_pad = GST_PAD(pads->data);
|
|
if (target_pad->direction == GST_PAD_SINK) {
|
|
gst_pad_handle_qos(target_pad, qos_message);
|
|
}
|
|
pads = g_list_next(pads);
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
void gst_pad_disconnect(GstPad *srcpad,GstPad *sinkpad) {
|
|
|
|
/* generic checks */
|
|
g_return_if_fail(srcpad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(srcpad));
|
|
g_return_if_fail(srcpad->peer != NULL);
|
|
g_return_if_fail(sinkpad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(sinkpad));
|
|
g_return_if_fail(sinkpad->peer != NULL);
|
|
|
|
g_return_if_fail((srcpad->direction == GST_PAD_SRC) &&
|
|
(sinkpad->direction == GST_PAD_SINK));
|
|
|
|
/* first clear peers */
|
|
srcpad->peer = NULL;
|
|
sinkpad->peer = NULL;
|
|
|
|
srcpad->chain = NULL;
|
|
srcpad->pull = NULL;
|
|
|
|
}
|
|
|
|
void gst_pad_connect(GstPad *srcpad,GstPad *sinkpad) {
|
|
GstPad *temppad;
|
|
|
|
/* generic checks */
|
|
g_return_if_fail(srcpad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(srcpad));
|
|
g_return_if_fail(srcpad->peer == NULL);
|
|
g_return_if_fail(sinkpad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(sinkpad));
|
|
g_return_if_fail(sinkpad->peer == NULL);
|
|
// g_return_if_fail(sinkpad->chain != NULL);
|
|
|
|
/* check for reversed directions and swap if necessary */
|
|
if ((srcpad->direction == GST_PAD_SINK) &&
|
|
(sinkpad->direction == GST_PAD_SRC)) {
|
|
temppad = srcpad;
|
|
srcpad = sinkpad;
|
|
sinkpad = temppad;
|
|
}
|
|
g_return_if_fail((srcpad->direction == GST_PAD_SRC) &&
|
|
(sinkpad->direction == GST_PAD_SINK));
|
|
|
|
/* first set peers */
|
|
srcpad->peer = sinkpad;
|
|
sinkpad->peer = srcpad;
|
|
|
|
/* now copy the chain pointer from sink to src */
|
|
srcpad->chain = sinkpad->chain;
|
|
/* and the pull function */
|
|
srcpad->pull = sinkpad->pull;
|
|
|
|
/* set the connected flag */
|
|
/* FIXME: set connected flag */
|
|
}
|
|
|
|
void gst_pad_set_parent(GstPad *pad,GstObject *parent) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
g_return_if_fail(pad->parent == NULL);
|
|
g_return_if_fail(parent != NULL);
|
|
g_return_if_fail(GTK_IS_OBJECT(parent));
|
|
g_return_if_fail((gpointer)pad != (gpointer)parent);
|
|
|
|
//g_print("set parent %s\n", gst_element_get_name(parent));
|
|
|
|
pad->parent = parent;
|
|
}
|
|
|
|
void gst_pad_add_ghost_parent(GstPad *pad,GstObject *parent) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
g_return_if_fail(parent != NULL);
|
|
g_return_if_fail(GTK_IS_OBJECT(parent));
|
|
|
|
pad->ghostparents = g_list_prepend(pad->ghostparents,parent);
|
|
}
|
|
|
|
|
|
void gst_pad_remove_ghost_parent(GstPad *pad,GstObject *parent) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
g_return_if_fail(parent != NULL);
|
|
g_return_if_fail(GTK_IS_OBJECT(parent));
|
|
|
|
pad->ghostparents = g_list_remove(pad->ghostparents,parent);
|
|
}
|
|
|
|
GstObject *gst_pad_get_parent(GstPad *pad) {
|
|
g_return_val_if_fail(pad != NULL, NULL);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), NULL);
|
|
|
|
return pad->parent;
|
|
}
|
|
|
|
GList *gst_pad_get_ghost_parents(GstPad *pad) {
|
|
g_return_val_if_fail(pad != NULL, NULL);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), NULL);
|
|
|
|
return pad->ghostparents;
|
|
}
|
|
|
|
guint16 gst_pad_get_type_id(GstPad *pad) {
|
|
g_return_val_if_fail(pad != NULL, 0);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), 0);
|
|
|
|
return pad->type;
|
|
}
|
|
|
|
void gst_pad_set_type_id(GstPad *pad,guint16 id) {
|
|
g_return_if_fail(pad != NULL);
|
|
g_return_if_fail(GST_IS_PAD(pad));
|
|
g_return_if_fail(gst_type_find_by_id(id) != NULL);
|
|
|
|
pad->type = id;
|
|
}
|
|
|
|
GstPad *gst_pad_get_peer(GstPad *pad) {
|
|
g_return_val_if_fail(pad != NULL, NULL);
|
|
g_return_val_if_fail(GST_IS_PAD(pad), NULL);
|
|
|
|
return pad->peer;
|
|
}
|
|
|
|
static void gst_pad_real_destroy(GtkObject *object) {
|
|
GstPad *pad = GST_PAD(object);
|
|
|
|
// g_print("in gst_pad_real_destroy()\n");
|
|
|
|
if (pad->name)
|
|
g_free(pad->name);
|
|
g_list_free(pad->ghostparents);
|
|
}
|
|
|
|
xmlNodePtr gst_pad_save_thyself(GstPad *pad,xmlNodePtr parent) {
|
|
xmlNodePtr self;
|
|
GstPad *peer;
|
|
|
|
self = xmlNewChild(parent,NULL,"pad",NULL);
|
|
xmlNewChild(self,NULL,"name",pad->name);
|
|
if (pad->peer != NULL) {
|
|
peer = pad->peer;
|
|
// first check to see if the peer's parent's parent is the same
|
|
if (pad->parent->parent == peer->parent->parent)
|
|
// we just save it off
|
|
xmlNewChild(self,NULL,"peer",g_strdup_printf("%s.%s",
|
|
GST_ELEMENT(peer->parent)->name,peer->name));
|
|
} else
|
|
xmlNewChild(self,NULL,"peer","");
|
|
|
|
return self;
|
|
}
|
|
|
|
xmlNodePtr gst_pad_ghost_save_thyself(GstPad *pad,GstElement *bin,xmlNodePtr parent) {
|
|
xmlNodePtr self;
|
|
|
|
self = xmlNewChild(parent,NULL,"ghostpad",NULL);
|
|
xmlNewChild(self,NULL,"name",pad->name);
|
|
xmlNewChild(self,NULL,"parent",GST_ELEMENT(pad->parent)->name);
|
|
|
|
return self;
|
|
}
|