Воспроизведение Gstreamer с использованием typefind

Для начала я просто начинающий в gstreamer, и я занимаюсь разработкой приложения, которое отправляет видеоданные слушателю, а слушатель декодирует его для воспроизведения. Для этого я использую gstreamer. Но до этого я пытаюсь понять gstreamer.

У меня есть автономный код, который генерирует сигнал с добавлением пэдов и обратный вызов. Iam с использованием платы I.MX6 для воспроизведения.

Я использовал gst-launch для воспроизведения "ts файла, а также файла mp4" с помощью команды ниже, и он работает.

gst-launch-0.10 filesrc location=/opt/zu/test/stream2.ts typefind=true ! aiurdemux name=demux demux. ! queue ! beepdec ! audioconvert ! autoaudiosink demux. ! queue ! vpudec! mfw_v4lsink

который затем я привел его в код с отличным использованием demuxer и необходимых элементов.

Здесь я нахожу "typefind = true", который я не уверен, как привести его в код. Насколько я понял, typefind устанавливает наборы src после нахождения CAP. Это то, что я использовал автономно для typefind.

<!--?xml version="1.0"?-->
<capabilities>
 <caps1>video/mpegts, systemstream=(boolean)true, packetsize=(int)188</caps1>
</capabilities>

Cap = видео /mpegts

При запуске автономного gstreamer для воспроизведения ts файла, я получаю следующее сообщение

"Состояние трубопровода изменилось с NULL на READY:" и оно не воспроизводится. Он просто продолжает работать. Ниже мой отдельный код

data.source = gst_element_factory_make ("filesrc", "source");
g_object_set (data.source, "location", argv[1], NULL);
data.typefind = gst_element_factory_make ("typefind", "typefinder");
data.demuxer = gst_element_factory_make ("aiurdemux", "demuxer");
data.audioqueue = gst_element_factory_make("queue","audioqueue");
data.videoqueue = gst_element_factory_make("queue","videoqueue");
data.audio_decoder = gst_element_factory_make ("beepdec", "audio_decoder");
data.audio_convert = gst_element_factory_make ("audioconvert", "audio_convert");
data.audio_sink = gst_element_factory_make ("autoaudiosink", "audio_sink");
data.video_decoder = gst_element_factory_make("vpudec","video_decoder");
data.video_sink = gst_element_factory_make("mfw_v4lsink","video_sink");

 if (!gst_element_link(data.source,data.demuxer)) {
 g_printerr ("Elements could not be linked.\n");
 gst_object_unref (data.pipeline);
 return -1;
 }

 if (!gst_element_link_many (data.audioqueue,data.audio_decoder,data.audio_convert, data.audio_sink,NULL)) {
 g_printerr (" audio Elements could not be linked.\n");
 gst_object_unref (data.pipeline);
 return -1;
 }
 if (!gst_element_link_many(data.videoqueue,data.video_decoder, data.video_sink,NULL)) {
 g_printerr("video Elements could not be linked.\n");
 gst_object_unref(data.pipeline);
 return -1;
 }

 //g_object_set (data.source, "location", argv[1], NULL);
 g_signal_connect (data.demuxer, "pad-added", G_CALLBACK (pad_added_handler), &data);
 /* Start playing */


// CALLBACK FUNCTION

static void pad_added_handler (GstElement *src, GstPad *new_pad, CustomData *data) {
 g_print("Inside the pad_added_handler method \n");
 printf("Inside the pad_added_handler method \n");
 GstPad *sink_pad_audio = gst_element_get_static_pad (data->audioqueue, "sink");
 GstPad *sink_pad_video = gst_element_get_static_pad (data->videoqueue, "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));
 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-vorbis"*/ "audio/mpeg"))
 {
 ret = gst_pad_link (new_pad, sink_pad_audio);
 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);
 }
 }

 //else if (g_str_has_prefix (new_pad_type, /*"video/x-theora"*/"video/x-h264"))
 else if (g_str_has_prefix (new_pad_type, /*"video/x-theora"*/"video/mpegts"))
 {
 ret = gst_pad_link (new_pad, sink_pad_video);

 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);
 }
 }


 else {
 g_print (" It has type '%s' which is not raw audio. Ignoring.\n", new_pad_type);
 goto exit;
 }
exit:
 if (new_pad_caps != NULL)
 gst_caps_unref (new_pad_caps);
 gst_object_unref (sink_pad_audio);
 gst_object_unref (sink_pad_video);

}

Я сократил часть кода. Не могли бы вы посоветовать, что здесь отсутствует Iam?.. Я не уверен, как привести тип поиска в код.

благодаря

1 ответ

Я решил это, наконец, без использования typefind.

Если кто-нибудь встретит этот пост, надеюсь, что это будет полезно для них.

Похоже, что typefind не требуется, я использовал следующий код.

GstCaps *caps;
/* for mp4 playback use "video/quicktime", if it is ts file use "video/mpegts" , likewise corresponding to other files. You can find what to use as caps using gst-discoverer command for the file*/
caps = gst_caps_new_simple("video/quicktime",NULL); 
//After creating caps link using link_filtered to the demuxer
if (!gst_element_link_filtered(data.source,data.demuxer,caps)) 
 g_printerr ("Elements could not be linked.\n");
 gst_object_unref (data.pipeline);
 return -1;
 }
gst_caps_unref(caps); //unref after use, then the normal code continues with generating //pad-added signal.

Вместо того, чтобы находить кепки с помощью truefind, приведенный выше код устанавливает колпачки вручную. Надеюсь, это поможет кому-то.

licensed under cc by-sa 3.0 with attribution.