运行(gstreamer)管道中的动态(非)链接元素?

时间:2010-06-19 03:01:56

标签: audio gstreamer

gstreamer文档中有很多关于构建和运行静态管道的例子。 但是,在 live 管道中更改/重新链接元素并不多 - 而媒体实际上正在流动。 这绝对是可能的,所以问题是:

  1. 在尝试之前我应该​​理解什么gstreamer概念/机制?
  2. 有什么陷阱需要注意吗?
  3. 什么是基本程序,或者是一个很好的例子?
  4. 接受的答案将是勺子喂,全面的,并附带源代码

5 个答案:

答案 0 :(得分:6)

  1. 我最喜欢的用于理解链接(和动态链接)的“概念”,正在考虑将管道作为真正的管道,并通过它流水。一旦你这样做,有些事情会变得非常明显。比如,“你在链接元素之前将源设置为播放吗?”,变成“在连接软管之前你打开水吗?”,它本身就是一种解答。通过动态链接更是如此,你如何确保没有水“泄漏”(这很糟糕,GStreamer中的“泄漏”相当于获得GST_FLOW_NOT_LINKED,并且会停止你的来源和乐趣)或被堵塞(可以导致数据包丢失或拥塞)。

  2. 是。许多。有一点免责声明,我目前仍在使用0.10,其中一些可能已经修复了1.0,遗憾的是非常非常难以动态链接和取消与GStreamer 0.10的链接。让我解释: 假设您正在使用Tee,并且想要取消一个分支的链接。你可以从释放Tees srcpad开始(不要忘记取消链接,这是发布垫的一部分),现在你应该能够安全地拆除该垫下游的元素。 (水当量是你在发球台后关闭一个阀门,现在应该能够在阀门之后拆除管道,除非你想弄湿,否则你不会在没有关闭阀门的情况下开始拆除管道......) 这将在大多数时间工作,但这里有一场比赛。因为在你释放了pad之后,在那个pad上的路上可能仍然有push或pad-alloc,如果你现在在你的代码中开始拆除下游元素,这可能会因为存在的竞争而崩溃在某些元素中,如果他们在拆除时获得推送或填充分配,或者您获得GST_FLOW_WRONG_STATE或GST_FLOW_NOT_LINKED,他们将返回源停止所有人的流...

  3. 我做了很多实验,发现如果你需要稳定性,偶尔崩溃/冻结不是一个选项,你需要一个元素作为你的动态安全网。一个元素,可以保证在释放/取消链接后,在pad上绝对不会发生任何活动。唯一的方法是打破另一个GStreamer范例,即在按住锁定时不要按下:你需要在按下/发送事件/ pad-allocation时按住一个锁。我不久前做了这样的事here。 (测试用例当然是最重要的,因为它允许你测试你自己/其他元素的安全性) 您还可以想象一个无锁的元素会吞下所有不良的FlowReturns,并为其上游绘制漂亮的图片,但是您需要绝对确保所有的下游元素都是“push或pad-alloc received关闭“-safe,因为你的元素无法保证一旦”停止流程“(释放/取消链接)已被执行,一点点不会挤过去。

  4. 当然,你必须将其中的一部分放在一边。我正在谈论的这些可怕的竞争条件的窗口实际上是非常非常小的,并且可能只发生在你运行程序的第1000或第1000次。但对于专业应用来说,这当然是不可接受的。我做了一个演讲,在那里我介绍了一些这些东西here

答案 1 :(得分:4)

我倾向于使用输出选择器输入选择器容器,具体取决于情况而不是阻塞阻塞复杂性(我在另一篇帖子中回答了pad阻止{{3 }})。并在不使用时将选择器连接到fakesrc或fakesink bin。在下面的示例中,如果一个人正在使用GTK,则可以用g_timeout_add (SWITCH_TIMEOUT, switch_cb, osel);替换行gtk_toggle_button,并将当前switch_cb函数中的所有代码放入切换按钮回调函数中。在此代码中,可以在两个imagesinks之间切换。我会用fakesink替换一个图像接收器以保持管道运行,以防我想在将来添加一个tee我想要录制视频的文件链接,然后为玩家提供打开选项(imagesink上的选择器)/关闭(选择器)在fakesink上)显示器。这允许用户在运行时使用选择器添加/删除bin。

#include <gst/gst.h>

#define SWITCH_TIMEOUT 1000
#define NUM_VIDEO_BUFFERS 500

static GMainLoop *loop;

/* Output selector src pads */
static GstPad *osel_src1 = NULL;
static GstPad *osel_src2 = NULL;

static gboolean
my_bus_callback (GstBus * bus, GstMessage * message, gpointer data)
{
  g_print ("Got %s message\n", GST_MESSAGE_TYPE_NAME (message));

  switch (GST_MESSAGE_TYPE (message)) {
    case GST_MESSAGE_ERROR:{
      GError *err;
      gchar *debug;

      gst_message_parse_error (message, &err, &debug);
      g_print ("Error: %s\n", err->message);
      g_error_free (err);
      g_free (debug);

      g_main_loop_quit (loop);
      break;
    }
    case GST_MESSAGE_EOS:
      /* end-of-stream */
      g_main_loop_quit (loop);
      break;
    default:
      /* unhandled message */
      break;
  }
  /* we want to be notified again the next time there is a message
   * on the bus, so returning TRUE (FALSE means we want to stop watching
   * for messages on the bus and our callback should not be called again)
   */
  return TRUE;
}

static gboolean
switch_cb (gpointer user_data)
{
  GstElement *sel = GST_ELEMENT (user_data);
  GstPad *old_pad, *new_pad = NULL;

  g_object_get (G_OBJECT (sel), "active-pad", &old_pad, NULL);

  if (old_pad == osel_src1)
    new_pad = osel_src2;
  else
    new_pad = osel_src1;

  g_object_set (G_OBJECT (sel), "active-pad", new_pad, NULL);

  g_print ("switched from %s:%s to %s:%s\n", GST_DEBUG_PAD_NAME (old_pad),
      GST_DEBUG_PAD_NAME (new_pad));

  gst_object_unref (old_pad);

  return TRUE;

}

gint
main (gint argc, gchar * argv[])
{
  GstElement *pipeline, *src, *toverlay, *osel, *sink1, *sink2, *convert;
  GstPad *sinkpad1;
  GstPad *sinkpad2;
  GstBus *bus;

  /* init GStreamer */
  gst_init (&argc, &argv);
  loop = g_main_loop_new (NULL, FALSE);

  /* create elements */
  pipeline = gst_element_factory_make ("pipeline", "pipeline");
  src = gst_element_factory_make ("videotestsrc", "src");
  toverlay = gst_element_factory_make ("timeoverlay", "timeoverlay");
  osel = gst_element_factory_make ("output-selector", "osel");
  convert = gst_element_factory_make ("ffmpegcolorspace", "convert");
  sink1 = gst_element_factory_make ("xvimagesink", "sink1");
  sink2 = gst_element_factory_make ("ximagesink", "sink2");

  if (!pipeline || !src || !toverlay || !osel || !convert || !sink1 || !sink2) {
    g_print ("missing element\n");
    return -1;
  }

  /* add them to bin */
  gst_bin_add_many (GST_BIN (pipeline), src, toverlay, osel, convert, sink1,
      sink2, NULL);

  /* set properties */
  g_object_set (G_OBJECT (src), "is-live", TRUE, NULL);
  g_object_set (G_OBJECT (src), "do-timestamp", TRUE, NULL);
  g_object_set (G_OBJECT (src), "num-buffers", NUM_VIDEO_BUFFERS, NULL);
  g_object_set (G_OBJECT (sink1), "sync", FALSE, "async", FALSE, NULL);
  g_object_set (G_OBJECT (sink2), "sync", FALSE, "async", FALSE, NULL);
  g_object_set (G_OBJECT (osel), "resend-latest", TRUE, NULL);

  /* link src ! timeoverlay ! osel */
  if (!gst_element_link_many (src, toverlay, osel, NULL)) {
    g_print ("linking failed\n");
    return -1;
  }

  /* link output 1 */
  sinkpad1 = gst_element_get_static_pad (sink1, "sink");
  osel_src1 = gst_element_get_request_pad (osel, "src%d");
  if (gst_pad_link (osel_src1, sinkpad1) != GST_PAD_LINK_OK) {
    g_print ("linking output 1 failed\n");
    return -1;
  }
  gst_object_unref (sinkpad1);

  /* link output 2 */
  sinkpad2 = gst_element_get_static_pad (convert, "sink");
  osel_src2 = gst_element_get_request_pad (osel, "src%d");
  if (gst_pad_link (osel_src2, sinkpad2) != GST_PAD_LINK_OK) {
    g_print ("linking output 2 failed\n");
    return -1;
  }
  gst_object_unref (sinkpad2);

  if (!gst_element_link (convert, sink2)) {
    g_print ("linking output 2 failed\n");
    return -1;
  }

  /* add switch callback */
  g_timeout_add (SWITCH_TIMEOUT, switch_cb, osel);

  /* change to playing */
  bus = gst_pipeline_get_bus (GST_PIPELINE (pipeline));
  gst_bus_add_watch (bus, my_bus_callback, loop);
  gst_object_unref (bus);

  gst_element_set_state (pipeline, GST_STATE_PLAYING);

  /* now run */
  g_main_loop_run (loop);

  /* also clean up */
  gst_element_set_state (pipeline, GST_STATE_NULL);
  gst_element_release_request_pad (osel, osel_src1);
  gst_element_release_request_pad (osel, osel_src2);
  gst_object_unref (GST_OBJECT (pipeline));

  return 0;
}

答案 2 :(得分:2)

其实我也是想做同样的事情。运气不太好:(

我通过询问#gstreamer IRC频道获得以下链接: http://cgit.freedesktop.org/gstreamer/gstreamer/tree/docs/design/part-dynamic.txt

可能暗示正确的方向。

如果您找到其他文件,请与我们联系......

答案 3 :(得分:2)

当我寻找动态修改任何gstreamer管道时,这篇文章首先出现了。找到了一些链接,但现在手册中已有详细记录:http://gstreamer.freedesktop.org/data/doc/gstreamer/head/manual/html/section-dynamic-pipelines.html

答案 4 :(得分:0)

我没有在 multifilesink 或<上面 Gstreamer 0.10 创建完整清晰的多路复用文件 EM>输出选择器

在分析了许多替代方案之后,我的解决方案将代码基于以下示例中描述的示例: http://gstreamer.freedesktop.org/data/doc/gstreamer/head/manual/html/section-dynamic-pipelines.html

探测器功能API已从0.10更改为1.0,但下面的解决方案可以创建每N秒不同的MP4文件:

static GstElement *pipeline = NULL;

// Pipeline -> src                    -> dynamic pipeline
// Pipeline -> capsfilter(f264file)   -> mp4mux(mux0)                                 -> filesink(fsink0)
// Pipeline -> elem_before||blockpad| -> |elem_cur_sinkpad||elem_cur||elem_cur_srcpad -> |elem_after_sinkpad||elem_after
static gulong probe_id;             // probe ID
static GstElement *elem_before;     // SRC of dynamic pipeline
static GstElement *elem_after;      // SINK of dynamic pipeline
static GstElement *elem_cur;        // Main element of dynamic pipeline
static GstPad *blockpad;            // SRC pad to be blocked
static GstPad *elem_cur_srcpad;     // SRC pad where check EOS
static GstPad *elem_cur_sinkpad;    // SINK of dynamic pipeline
static GstPad *elem_after_sinkpad;  // SINK of SINK element

// Last Buffer Timestamp
static GstClockTime last_ts = 0;

typedef enum {
  NO_NEW_FILE,  // Keep current file destination
  NEW_FILE,     // Switch file destination
} NewFileStatus;
static NewFileStatus newfile = NO_NEW_FILE; // Switch File Flag

static int counter = 1; // Index filename

// EOS listener to switch to other file destination
static gboolean
event_probe_cb (GstPad * pad, GstEvent * event, gpointer user_data)
{
  g_print ("INSIDE event_probe_cb:%d type:%s\n",probe_id,
      GST_EVENT_TYPE (event)==GST_EVENT_EOS?"EOS":GST_EVENT_TYPE (event)==GST_EVENT_NEWSEGMENT?"NEWSEGMENT":"OTHER");

  if (GST_EVENT_TYPE (event) != GST_EVENT_EOS)
  {
    // Push the event in the pipe flow (false DROP)
    return TRUE;
  }

  // remove the probe first
  gst_pad_remove_event_probe (pad, probe_id);

  gst_object_unref (elem_cur_srcpad);
  gst_object_unref (elem_after_sinkpad);
  gst_element_release_request_pad(elem_cur, elem_cur_sinkpad);

  gst_element_set_state (elem_cur, GST_STATE_NULL);
  gst_element_set_state (elem_after, GST_STATE_NULL);

  // remove unlinks automatically
  GST_DEBUG_OBJECT (pipeline, "removing %" GST_PTR_FORMAT, elem_cur);
  gst_bin_remove (GST_BIN (pipeline), elem_cur);
  GST_DEBUG_OBJECT (pipeline, "removing %" GST_PTR_FORMAT, elem_after);
  gst_bin_remove (GST_BIN (pipeline), elem_after);

  GstElement * mux0 = gst_element_factory_make("mp4mux", "mux0");
  GstElement * fsink0 = gst_element_factory_make("filesink", "fsink0");
  elem_cur = mux0;
  elem_after = fsink0;

  if(!mux0 || !fsink0)
  {
    printf("mising elements\n");
  }

  GST_DEBUG_OBJECT (pipeline, "adding   %" GST_PTR_FORMAT, elem_cur);
  gst_bin_add (GST_BIN (pipeline), elem_cur);
  GST_DEBUG_OBJECT (pipeline, "adding   %" GST_PTR_FORMAT, elem_after);
  gst_bin_add (GST_BIN (pipeline), elem_after);

  char buffer[128];
  sprintf(buffer, "test_%d.mp4", counter++);
  g_print ("File Switching %s\n", buffer);
  g_object_set(G_OBJECT(elem_after), "location", buffer, NULL);

  GST_DEBUG_OBJECT (pipeline, "linking..");
  elem_cur_srcpad = gst_element_get_static_pad (elem_cur, "src");
  elem_cur_sinkpad = gst_element_get_request_pad (elem_cur, "video_%d");
  elem_after_sinkpad = gst_element_get_static_pad (elem_after, "sink");

  if(gst_pad_link(blockpad, elem_cur_sinkpad) != GST_PAD_LINK_OK)
  {
    printf("linking output 0 failed\n");
    return -1;
  }
  if(gst_pad_link(elem_cur_srcpad, elem_after_sinkpad) != GST_PAD_LINK_OK)
  {
    printf("linking output 1 failed\n");
    return -1;
  }

  g_print ("Moving to PLAYING\n");
  gst_element_set_state (elem_cur, GST_STATE_PLAYING);
  gst_element_set_state (elem_after, GST_STATE_PLAYING);

  GST_DEBUG_OBJECT (pipeline, "done");

  newfile = NO_NEW_FILE;
  // Push the event in the pipe flow (false DROP)
  return TRUE;
}

// Check if Buffer contains a KEY FRAME
static gboolean
is_sync_frame (GstBuffer * buffer)
{
  if (GST_BUFFER_FLAG_IS_SET(buffer, GST_BUFFER_FLAG_DELTA_UNIT)) 
  {
    return FALSE;
  }
  else if (!GST_BUFFER_FLAG_IS_SET(buffer, GST_BUFFER_FLAG_IN_CAPS)) 
  {
    return TRUE;
  }
}

// Block source and launch EOS to MUXER to achieve a full muxed file
static gboolean
pad_probe_cb (GstPad * pad, GstBuffer * buffer, gpointer user_data)
{
  g_print ("\n\tINSIDE pad_probe_cb:%d %s %s\n",probe_id, (newfile?"newfile":"thesame"), 
      (is_sync_frame (buffer)?"KEYframe":"frame"));
  GST_DEBUG_OBJECT (pad, "pad is blocked now");

  last_ts = GST_BUFFER_TIMESTAMP(buffer);
  if(!GST_CLOCK_TIME_IS_VALID(last_ts))
      last_ts=0;

  if((newfile==NO_NEW_FILE) || !is_sync_frame (buffer))
    return TRUE;

  /* remove the probe first */
  gst_pad_remove_buffer_probe (pad, probe_id);

  /* install new probe for EOS */
  probe_id = gst_pad_add_event_probe (elem_after_sinkpad, G_CALLBACK(event_probe_cb), user_data);

  /* push EOS into the element, the probe will be fired when the
   * EOS leaves the effect and it has thus drained all of its data */
  gst_pad_send_event (elem_cur_sinkpad, gst_event_new_eos ());

  // Wait til the EOS have been processed the Buffer with the Key frame will be the FIRST
  while(newfile != NO_NEW_FILE)
      Sleep(1);

  // Push the buffer in the pipe flow (false DROP)
  return TRUE;
}

// this timeout is periodically run as part of the mainloop
static gboolean timeout (gpointer user_data)
{
  g_print ("TIMEOUT\n");
  if(!playing)
      return false;
  newfile = NEW_FILE;
  /* install new probe for Keyframe and New File */
  probe_id = gst_pad_add_buffer_probe (blockpad, G_CALLBACK(pad_probe_cb), pipeline);
  return true;
}

static GstElement *pipeline = NULL; // Pipeline -> src -> dynamic pipeline // Pipeline -> capsfilter(f264file) -> mp4mux(mux0) -> filesink(fsink0) // Pipeline -> elem_before||blockpad| -> |elem_cur_sinkpad||elem_cur||elem_cur_srcpad -> |elem_after_sinkpad||elem_after static gulong probe_id; // probe ID static GstElement *elem_before; // SRC of dynamic pipeline static GstElement *elem_after; // SINK of dynamic pipeline static GstElement *elem_cur; // Main element of dynamic pipeline static GstPad *blockpad; // SRC pad to be blocked static GstPad *elem_cur_srcpad; // SRC pad where check EOS static GstPad *elem_cur_sinkpad; // SINK of dynamic pipeline static GstPad *elem_after_sinkpad; // SINK of SINK element // Last Buffer Timestamp static GstClockTime last_ts = 0; typedef enum { NO_NEW_FILE, // Keep current file destination NEW_FILE, // Switch file destination } NewFileStatus; static NewFileStatus newfile = NO_NEW_FILE; // Switch File Flag static int counter = 1; // Index filename // EOS listener to switch to other file destination static gboolean event_probe_cb (GstPad * pad, GstEvent * event, gpointer user_data) { g_print ("INSIDE event_probe_cb:%d type:%s\n",probe_id, GST_EVENT_TYPE (event)==GST_EVENT_EOS?"EOS":GST_EVENT_TYPE (event)==GST_EVENT_NEWSEGMENT?"NEWSEGMENT":"OTHER"); if (GST_EVENT_TYPE (event) != GST_EVENT_EOS) { // Push the event in the pipe flow (false DROP) return TRUE; } // remove the probe first gst_pad_remove_event_probe (pad, probe_id); gst_object_unref (elem_cur_srcpad); gst_object_unref (elem_after_sinkpad); gst_element_release_request_pad(elem_cur, elem_cur_sinkpad); gst_element_set_state (elem_cur, GST_STATE_NULL); gst_element_set_state (elem_after, GST_STATE_NULL); // remove unlinks automatically GST_DEBUG_OBJECT (pipeline, "removing %" GST_PTR_FORMAT, elem_cur); gst_bin_remove (GST_BIN (pipeline), elem_cur); GST_DEBUG_OBJECT (pipeline, "removing %" GST_PTR_FORMAT, elem_after); gst_bin_remove (GST_BIN (pipeline), elem_after); GstElement * mux0 = gst_element_factory_make("mp4mux", "mux0"); GstElement * fsink0 = gst_element_factory_make("filesink", "fsink0"); elem_cur = mux0; elem_after = fsink0; if(!mux0 || !fsink0) { printf("mising elements\n"); } GST_DEBUG_OBJECT (pipeline, "adding %" GST_PTR_FORMAT, elem_cur); gst_bin_add (GST_BIN (pipeline), elem_cur); GST_DEBUG_OBJECT (pipeline, "adding %" GST_PTR_FORMAT, elem_after); gst_bin_add (GST_BIN (pipeline), elem_after); char buffer[128]; sprintf(buffer, "test_%d.mp4", counter++); g_print ("File Switching %s\n", buffer); g_object_set(G_OBJECT(elem_after), "location", buffer, NULL); GST_DEBUG_OBJECT (pipeline, "linking.."); elem_cur_srcpad = gst_element_get_static_pad (elem_cur, "src"); elem_cur_sinkpad = gst_element_get_request_pad (elem_cur, "video_%d"); elem_after_sinkpad = gst_element_get_static_pad (elem_after, "sink"); if(gst_pad_link(blockpad, elem_cur_sinkpad) != GST_PAD_LINK_OK) { printf("linking output 0 failed\n"); return -1; } if(gst_pad_link(elem_cur_srcpad, elem_after_sinkpad) != GST_PAD_LINK_OK) { printf("linking output 1 failed\n"); return -1; } g_print ("Moving to PLAYING\n"); gst_element_set_state (elem_cur, GST_STATE_PLAYING); gst_element_set_state (elem_after, GST_STATE_PLAYING); GST_DEBUG_OBJECT (pipeline, "done"); newfile = NO_NEW_FILE; // Push the event in the pipe flow (false DROP) return TRUE; } // Check if Buffer contains a KEY FRAME static gboolean is_sync_frame (GstBuffer * buffer) { if (GST_BUFFER_FLAG_IS_SET(buffer, GST_BUFFER_FLAG_DELTA_UNIT)) { return FALSE; } else if (!GST_BUFFER_FLAG_IS_SET(buffer, GST_BUFFER_FLAG_IN_CAPS)) { return TRUE; } } // Block source and launch EOS to MUXER to achieve a full muxed file static gboolean pad_probe_cb (GstPad * pad, GstBuffer * buffer, gpointer user_data) { g_print ("\n\tINSIDE pad_probe_cb:%d %s %s\n",probe_id, (newfile?"newfile":"thesame"), (is_sync_frame (buffer)?"KEYframe":"frame")); GST_DEBUG_OBJECT (pad, "pad is blocked now"); last_ts = GST_BUFFER_TIMESTAMP(buffer); if(!GST_CLOCK_TIME_IS_VALID(last_ts)) last_ts=0; if((newfile==NO_NEW_FILE) || !is_sync_frame (buffer)) return TRUE; /* remove the probe first */ gst_pad_remove_buffer_probe (pad, probe_id); /* install new probe for EOS */ probe_id = gst_pad_add_event_probe (elem_after_sinkpad, G_CALLBACK(event_probe_cb), user_data); /* push EOS into the element, the probe will be fired when the * EOS leaves the effect and it has thus drained all of its data */ gst_pad_send_event (elem_cur_sinkpad, gst_event_new_eos ()); // Wait til the EOS have been processed the Buffer with the Key frame will be the FIRST while(newfile != NO_NEW_FILE) Sleep(1); // Push the buffer in the pipe flow (false DROP) return TRUE; } // this timeout is periodically run as part of the mainloop static gboolean timeout (gpointer user_data) { g_print ("TIMEOUT\n"); if(!playing) return false; newfile = NEW_FILE; /* install new probe for Keyframe and New File */ probe_id = gst_pad_add_buffer_probe (blockpad, G_CALLBACK(pad_probe_cb), pipeline); return true; }