简体   繁体   中英

Creating a gstreamer pipeline with the intent of modifying parts

I'm working on an audio streamer, and I want to be able to modify the file I'm streaming, and also the target I'm streaming to. To do this I would modify the location for my filesrc, or I would modify the host/port of my udpsink.

I am having trouble understanding everything I need to know to get this pipeline linked together and playing. Previously I hard coded everything and used the gst pipeline parsing tool with this pipeline:

filesrc location=/storage/sdcard0/Music/RunToTheHills.ogg ! oggdemux ! vorbisdec ! audioresample ! audioconvert ! audio/x-raw-int,channels=2,depth=16,width=16,rate=44100 ! rtpL16pay  ! udpsink host=192.168.100.126 port=9001

Now I want to change the filesrc location, and udp host/port as mentioned above.

My application is an Android app using the NDK. However, this should not effect the code needed to set up a pipeline.

Here's what I've got so far, which results in a segfault.

My data structure:

/**
 * Structure to hold all the various variables we need.
 * This is handed to callbacks
 */
typedef struct _CustomData {
    jobject app; /* The Java app */
    GstElement *pipeline; /* gStreamer pipeline */
    GstElement *filesrc; /* Input file */
    GstPad *fileblock; /* Used to block filesrc */
    GstElement *ogg; /* Ogg demultiplexer */
    GstElement *vorbis; /* Vorbis decoder */
    GstElement *resample;
    GstElement *convert;
    GstCaps *caps;
    GstElement *rtp; /* RTP packer */
    GstElement *udp; /* UDP sender */
    GMainContext *context; /* GLib Context */
    GMainLoop *main_loop; /* GLib main loop */
    gboolean initialised; /* True after initialisation */
    GstState state; /* Pipeline state */
    GstState target_state; /* What state we want to put the pipeline into */
    gint64 duration; /* Clip length */
    gint64 desired_position; /* Where we want to track to within the clip */
    GstClockTime last_seek_time; /* Used to throttle seeking */
    gboolean is_live; /* Live streams don't need buffering */
} CustomData;

And here's my creation of the pipeline:

data->pipeline = gst_pipeline_new("pipeline");

data->filesrc = gst_element_factory_make("filesrc", NULL);
if (!data->filesrc) {
    GST_ERROR("Failed to create filesrc.");
    return NULL;
}
g_object_set(G_OBJECT(data->filesrc), "location", "/storage/sdcard0/Music/RunToTheHills.ogg", NULL);

data->fileblock = gst_element_get_static_pad(data->filesrc, "src");

data->ogg = gst_element_factory_make("oggdemux", NULL);
if (!data->ogg) {
    GST_ERROR("Failed to create oggdemux.");
    return NULL;
}

data->vorbis = gst_element_factory_make("vorbisdec", NULL);
if (!data->vorbis) {
    GST_ERROR("Failed to create vorbisdec.");
    return NULL;
}

data->resample = gst_element_factory_make("audioresample", NULL);
if (!data->resample) {
    GST_ERROR("Failed to create audioresample.");
    return NULL;
}

data->convert = gst_element_factory_make("audioconvert", NULL);
if (!data->convert) {
    GST_ERROR("Failed to create audioconvert.");
    return NULL;
}

data->caps = gst_caps_new_simple("audio/x-raw-int",
        "channels", G_TYPE_INT, 2,
        "depth", G_TYPE_INT, 16,
        "width", G_TYPE_INT, 16,
        "rate", G_TYPE_INT, 44100);

if (!data->caps) {
    GST_ERROR("Failed to create caps");
    return NULL;
}

data->rtp = gst_element_factory_make("rtpL16pay", NULL);
if (!data->rtp) {
    GST_ERROR("Failed to create rtpL16pay.");
    return NULL;
}

data->udp = gst_element_factory_make("udpsink", NULL);
if (!data->udp) {
    GST_ERROR("Failed to create udpsink.");
    return NULL;
}
g_object_set(G_OBJECT(data->udp), "host", "192.168.100.126", NULL);
g_object_set(G_OBJECT(data->udp), "port", 9001, NULL);


if (!data->ogg || !data->vorbis || !data->resample || !data->convert || !data->caps || !data->rtp || !data->udp) {
    GST_ERROR("Unable to create all elements!");
    return NULL;
}

gst_bin_add_many(GST_BIN(data->pipeline), data->filesrc, data->ogg, data->vorbis,
        data->resample, data->convert, data->caps, data->rtp, data->udp);

/* Link all the elements together */
gst_element_link(data->filesrc, data->ogg);
gst_element_link(data->ogg, data->vorbis);
gst_element_link(data->vorbis, data->resample);
gst_element_link(data->resample, data->convert);
gst_element_link_filtered(data->convert, data->rtp, data->caps);
gst_element_link(data->rtp, data->udp);

Can someone give me some hints as to where I went wrong?

For interest, here's my previously working pipeline:

data->pipeline = gst_parse_launch("filesrc location=/storage/sdcard0/Music/RunToTheHills.ogg ! oggdemux ! vorbisdec ! audioresample ! audioconvert ! audio/x-raw-int,channels=2,depth=16,width=16,rate=44100 ! rtpL16pay  ! udpsink host=192.168.100.126 port=9001", &error);
if (error) {
    gchar *message = g_strdup_printf("Unable to build pipeline: %s", error->message);
    g_clear_error (&error);
    set_ui_message(message, data);
    g_free (message);
    return NULL;
}

You cannot simply link the oggdemux to the vorbisdec, because the demux has sometimes pads .

You need to add a handler function for the 'pad-added' signal of the demux and then perform the link there.

/* Connect to the pad-added signal */
g_signal_connect (data->ogg, "pad-added", G_CALLBACK (pad_added_handler), data);

And the handler:

void on_pad_added (GstElement *src, GstPad *new_pad, CustomData *data)
{
GstPad *sink_pad = gst_element_get_static_pad (data->vorbis, "sink");
GstPadLinkReturn ret;
GstCaps *new_pad_caps = NULL;
GstStructure *new_pad_struct = NULL;
const gchar *new_pad_type = NULL;

g_print ("Received new pad '%s' from '%s':\n", GST_PAD_NAME (new_pad), GST_ELEMENT_NAME (src));

/* If our converter is already linked, we have nothing to do here */
if (gst_pad_is_linked (sink_pad)) {
g_print ("  We are already linked. Ignoring.\n");
goto exit;
}

/* Check the new pad's type */
new_pad_caps = gst_pad_get_caps (new_pad);
new_pad_struct = gst_caps_get_structure (new_pad_caps, 0);
new_pad_type = gst_structure_get_name (new_pad_struct);
if (!g_str_has_prefix (new_pad_type, "audio/x-raw")) {
g_print ("  It has type '%s' which is not raw audio. Ignoring.\n", new_pad_type);
goto exit;
}

/* Attempt the link */
ret = gst_pad_link (new_pad, sink_pad);
if (GST_PAD_LINK_FAILED (ret)) {
g_print ("  Type is '%s' but link failed.\n", new_pad_type);
} else {
g_print ("  Link succeeded (type '%s').\n", new_pad_type);
}

exit:
/* Unreference the new pad's caps, if we got them */
if (new_pad_caps != NULL)
gst_caps_unref (new_pad_caps);

/* Unreference the sink pad */
gst_object_unref (sink_pad);
}

Also, since you're getting a segmentation fault, i believe there is a memory issue. Are you sure you're using the CustomData structure right? I notice you're using data->element instead of data.element.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM