当前位置: 首页 > news >正文

Gstreamer之”pad-added“事件

"pad-added" 事件详解

"pad-added" 是 GStreamer 中一个非常重要的信号,当元素动态创建一个新的 pad(连接点)时会发出此信号。

基本概念:

  • Pad:GStreamer 中元素之间的连接点,数据通过 pad 流动

  • 动态 Pad:某些元素(如 rtspsrc、decodebin)在运行时才会创建 pad,而不是在初始化时就存在

  • 典型应用场景:RTSP 源、解码器等元素经常使用动态 pad

为什么需要处理 "pad-added":

  1. 对于像 rtspsrc 这样的源元素,你无法预先知道它会提供多少个流(视频、音频等)

  2. 对于像 decodebin 这样的解码元素,你无法预先知道输入媒体的类型

  3. 必须在 pad 可用时才能建立元素间的连接

典型回调函数签名:

void callback(GstElement *src, GstPad *new_pad, gpointer user_data);

rtspsrc 的常见信号

除了 "pad-added",rtspsrc 还提供以下重要信号:

1. 连接相关信号

  • "error":发生错误时触发

    void callback(GstElement *src, guint error_code, gchar *error_msg, gpointer user_data);
  • "warning":发生警告时触发

    void callback(GstElement *src, guint warning_code, gchar *warning_msg, gpointer user_data);
  • "state-changed":元素状态改变时触发

    void callback(GstElement *src, GstState old_state, GstState new_state, GstState pending_state, gpointer user_data);

2. 流管理信号

  • "new-stream":检测到新流时触发

    void callback(GstElement *src, GstPad *pad, gpointer user_data);
  • "select-stream":允许应用程序选择特定流

    gboolean callback(GstElement *src, guint stream_id, GstCaps *caps, gpointer user_data);
  • "pad-removed":当 pad 被移除时触发

    void callback(GstElement *src, GstPad *pad, gpointer user_data);

3. 媒体处理信号

  • "eos":流结束时触发(End Of Stream)

    void callback(GstElement *src, gpointer user_data);
  • "enough-streams":当收集到足够多流时触发

    void callback(GstElement *src, gpointer user_data);

其他常用 GStreamer 元素的信号

decodebin 信号

  • "pad-added":当解码器创建新 pad 时

  • "autoplug-continue":决定是否继续自动插件

  • "autoplug-select":选择自动插件类型

  • "drained":元素已排空

appsrc 信号

  • "need-data":需要新数据时触发

  • "enough-data":有足够数据时触发

  • "seek-data":请求定位时触发

videosink 信号

  • "preroll-handoff":第一帧显示前触发

  • "handoff":每帧显示时触发

完整示例:处理多个 rtspsrc 信号

#include <gst/gst.h>// pad-added 回调
static void on_pad_added(GstElement *src, GstPad *new_pad, gpointer data) {GstElement *pipeline = (GstElement *)data;GstElement *queue = gst_bin_get_by_name(GST_BIN(pipeline), "queue");GstPad *sink_pad = gst_element_get_static_pad(queue, "sink");if (gst_pad_is_linked(sink_pad)) {gst_object_unref(sink_pad);return;}GstPadLinkReturn ret = gst_pad_link(new_pad, sink_pad);if (GST_PAD_LINK_FAILED(ret)) {g_printerr("Pad link failed.\n");}gst_object_unref(sink_pad);
}// 错误处理回调
static void on_error(GstElement *src, guint error_code, gchar *error_msg, gpointer user_data) {g_printerr("RTSP Error %d: %s\n", error_code, error_msg);// 可以在这里实现重连逻辑
}// 新流回调
static void on_new_stream(GstElement *src, GstPad *pad, gpointer user_data) {g_print("New stream detected on pad %s\n", GST_PAD_NAME(pad));
}// EOS回调
static void on_eos(GstElement *src, gpointer user_data) {g_print("End of stream reached\n");// 可以在这里实现流重启逻辑
}int main(int argc, char *argv[]) {gst_init(&argc, &argv);GstElement *pipeline = gst_pipeline_new("rtsp-pipeline");GstElement *rtspsrc = gst_element_factory_make("rtspsrc", "source");GstElement *queue = gst_element_factory_make("queue", "queue");// ... 其他元素创建// 设置RTSP URLg_object_set(rtspsrc, "location", "rtsp://example.com/stream", NULL);// 添加元素到管道gst_bin_add_many(GST_BIN(pipeline), rtspsrc, queue, /* 其他元素 */, NULL);// 连接信号g_signal_connect(rtspsrc, "pad-added", G_CALLBACK(on_pad_added), pipeline);g_signal_connect(rtspsrc, "error", G_CALLBACK(on_error), NULL);g_signal_connect(rtspsrc, "new-stream", G_CALLBACK(on_new_stream), NULL);g_signal_connect(rtspsrc, "eos", G_CALLBACK(on_eos), NULL);// 启动管道并运行主循环...return 0;
}

信号处理的最佳实践

  1. 保持回调简洁:不要在回调中执行耗时操作

  2. 线程安全:GStreamer 信号可能来自不同的线程

  3. 错误处理:始终处理错误信号

  4. 资源管理:正确管理 pad 和元素的引用计数

  5. 状态检查:在回调中检查管道/元素状态

理解这些信号及其用途对于构建健壮的 GStreamer 应用程序至关重要,特别是在处理网络流媒体时。

http://www.dtcms.com/a/279872.html

相关文章:

  • cron监控进程逻辑
  • C#中发布订阅的阻塞非阻塞
  • 微美全息借区块链与DRL算法打造资源管理协同架构,达成边缘计算与区块链动态适配
  • Function-——函数中文翻译渊源及历史背景
  • 学习笔记(35):了解原理:从密度到了解概率密度
  • iperf3 网络带宽测试工具学习
  • 国内隧道IP代理技术解析:原理、优势与实战应用
  • 网络地址转换(NAT)与单臂路由实验
  • 2.逻辑回归、Softmax回归
  • 智能节气装置
  • 记录网络切换时同步操作
  • TypeScript 配置全解析:tsconfig.json、tsconfig.app.json 与 tsconfig.node.json 的深度指南
  • JJ20 Final Lap演唱会纪念票根生成工具
  • 信息收集的基本流程
  • 大模型呼叫系统选型指南
  • 【Linux】Linux 操作系统 - 28 , 进程间通信(四) -- IPC 资源的管理方式_信号量_临界区等基本概念介绍
  • 递推预处理floor(log_2{n})
  • Class9简洁实现
  • HashMap的put过程以及hashMap的简单介绍
  • kt 中商品的金额字段使用double 还是 bigdecimal
  • 动态规划题解——最长递增子序列【LeetCode】记忆化搜索方法
  • 【每日刷题】杨辉三角
  • Git根据标签Tag强制回滚版本
  • 面试常问:如何在一个长度为n的无序数据快速获取前k个数值
  • 网络传输过程
  • GaussDB between的用法
  • 光伏板如何最大化铺设?
  • 【PostgreSQL异常解决】`PostgreSQL`异常之类型转换错误
  • 记录自己在将python文件变成可访问库文件是碰到的问题
  • vert.x 官网docs, vert.x中文文档地址 vertx文档