Created
December 2, 2010 10:47
-
-
Save nzjrs/725122 to your computer and use it in GitHub Desktop.
GStreamer Streaming AppSrc Example
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* gcc gdk-gstappsrc-stream.c -Wall `pkg-config --cflags --libs gstreamer-app-0.10 gdk-pixbuf-2.0` -o gdkstream */ | |
#include <gst/gst.h> | |
#include <gst/app/gstappsrc.h> | |
#include <stdio.h> | |
#include <string.h> | |
#include <stdlib.h> | |
#include <gdk-pixbuf/gdk-pixbuf.h> | |
GST_DEBUG_CATEGORY (appsrc_pipeline_debug); | |
#define GST_CAT_DEFAULT appsrc_pipeline_debug | |
typedef struct _App App; | |
struct _App | |
{ | |
GstElement *pipeline; | |
GstElement *appsrc; | |
GMainLoop *loop; | |
guint sourceid; | |
GTimer *timer; | |
}; | |
App s_app; | |
static gboolean | |
read_data (App * app) | |
{ | |
GstFlowReturn ret; | |
gdouble ms; | |
ms = g_timer_elapsed(app->timer, NULL); | |
if (ms > 1.0) { | |
GstBuffer *buffer; | |
GdkPixbuf *pb; | |
gboolean ok = TRUE; | |
buffer = gst_buffer_new(); | |
pb = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, 640, 480); | |
gdk_pixbuf_fill(pb, 0xfff11fff); | |
GST_BUFFER_DATA (buffer) = gdk_pixbuf_get_pixels(pb); | |
GST_BUFFER_SIZE (buffer) = 640*480*3*sizeof(guchar); | |
GST_DEBUG ("feed buffer"); | |
g_signal_emit_by_name (app->appsrc, "push-buffer", buffer, &ret); | |
gst_buffer_unref (buffer); | |
if (ret != GST_FLOW_OK) { | |
/* some error, stop sending data */ | |
GST_DEBUG ("some error"); | |
ok = FALSE; | |
} | |
g_timer_start(app->timer); | |
return ok; | |
} | |
// g_signal_emit_by_name (app->appsrc, "end-of-stream", &ret); | |
return TRUE; | |
} | |
/* This signal callback is called when appsrc needs data, we add an idle handler | |
* to the mainloop to start pushing data into the appsrc */ | |
static void | |
start_feed (GstElement * pipeline, guint size, App * app) | |
{ | |
if (app->sourceid == 0) { | |
GST_DEBUG ("start feeding"); | |
app->sourceid = g_timeout_add (10, (GSourceFunc) read_data, app); | |
} | |
} | |
/* This callback is called when appsrc has enough data and we can stop sending. | |
* We remove the idle handler from the mainloop */ | |
static void | |
stop_feed (GstElement * pipeline, App * app) | |
{ | |
if (app->sourceid != 0) { | |
GST_DEBUG ("stop feeding"); | |
g_source_remove (app->sourceid); | |
app->sourceid = 0; | |
} | |
} | |
static gboolean | |
bus_message (GstBus * bus, GstMessage * message, App * app) | |
{ | |
GST_DEBUG ("got message %s", | |
gst_message_type_get_name (GST_MESSAGE_TYPE (message))); | |
switch (GST_MESSAGE_TYPE (message)) { | |
case GST_MESSAGE_ERROR: { | |
GError *err = NULL; | |
gchar *dbg_info = NULL; | |
gst_message_parse_error (message, &err, &dbg_info); | |
g_printerr ("ERROR from element %s: %s\n", | |
GST_OBJECT_NAME (message->src), err->message); | |
g_printerr ("Debugging info: %s\n", (dbg_info) ? dbg_info : "none"); | |
g_error_free (err); | |
g_free (dbg_info); | |
g_main_loop_quit (app->loop); | |
break; | |
} | |
case GST_MESSAGE_EOS: | |
g_main_loop_quit (app->loop); | |
break; | |
default: | |
break; | |
} | |
return TRUE; | |
} | |
int | |
main (int argc, char *argv[]) | |
{ | |
App *app = &s_app; | |
GstBus *bus; | |
GstCaps *caps; | |
gst_init (&argc, &argv); | |
GST_DEBUG_CATEGORY_INIT (appsrc_pipeline_debug, "appsrc-pipeline", 0, | |
"appsrc pipeline example"); | |
/* create a mainloop to get messages and to handle the idle handler that will | |
* feed data to appsrc. */ | |
app->loop = g_main_loop_new (NULL, TRUE); | |
app->timer = g_timer_new(); | |
app->pipeline = gst_parse_launch("appsrc name=mysource ! ffmpegcolorspace ! videoscale method=1 ! theoraenc bitrate=150 ! udpsink host=127.0.0.1 port=1234", NULL); | |
g_assert (app->pipeline); | |
bus = gst_pipeline_get_bus (GST_PIPELINE (app->pipeline)); | |
g_assert(bus); | |
/* add watch for messages */ | |
gst_bus_add_watch (bus, (GstBusFunc) bus_message, app); | |
/* get the appsrc */ | |
app->appsrc = gst_bin_get_by_name (GST_BIN(app->pipeline), "mysource"); | |
g_assert(app->appsrc); | |
g_assert(GST_IS_APP_SRC(app->appsrc)); | |
g_signal_connect (app->appsrc, "need-data", G_CALLBACK (start_feed), app); | |
g_signal_connect (app->appsrc, "enough-data", G_CALLBACK (stop_feed), app); | |
/* set the caps on the source */ | |
caps = gst_caps_new_simple ("video/x-raw-rgb", | |
"width", G_TYPE_INT, 640, | |
"height", G_TYPE_INT, 480, | |
"bpp", G_TYPE_INT, 24, | |
"depth", G_TYPE_INT, 24, | |
"red_mask", G_TYPE_INT, 0x00ff0000, | |
"green_mask", G_TYPE_INT, 0x0000ff00, | |
"blue_mask", G_TYPE_INT, 0x000000ff, | |
"framerate", GST_TYPE_FRACTION, 25, 1, | |
"endianness", G_TYPE_INT, G_BIG_ENDIAN, | |
NULL); | |
gst_app_src_set_caps(GST_APP_SRC(app->appsrc), caps); | |
/* go to playing and wait in a mainloop. */ | |
gst_element_set_state (app->pipeline, GST_STATE_PLAYING); | |
/* this mainloop is stopped when we receive an error or EOS */ | |
g_main_loop_run (app->loop); | |
GST_DEBUG ("stopping"); | |
gst_element_set_state (app->pipeline, GST_STATE_NULL); | |
gst_object_unref (bus); | |
g_main_loop_unref (app->loop); | |
return 0; | |
} |
could you please tell why this check is required?
if (ms > 1.0/20.0)
could you please tell why this check is required?
He's targeting 20fps. So if elapsed time is over 1/20 of a second, handle a new buffer.
Any change you would add a BSD or public domain license?
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
I try but this don't work i change the sink element (udpsink) for other (tcpserversink or even glimagesink ) and nothing happen...
It's this right???
Thanks in advance