GStreamer 简明教程(六):利用 Tee 复制流数据,巧用 Queue 实现多线程

news2025/1/13 10:16:39

系列文章目录

  • GStreamer 简明教程(一):环境搭建,运行 Basic Tutorial 1 Hello world!
  • GStreamer 简明教程(二):基本概念介绍,Element 和 Pipeline
  • GStreamer 简明教程(三):动态调整 Pipeline
  • GStreamer 简明教程(四):Seek 以及获取文件时长
  • GStreamer 简明教程(五):Pad 相关概念介绍,Pad Capabilities/Templates

文章目录

  • 系列文章目录
  • 前言
  • Pipeline
  • Request Pads
  • Show me the code
  • 参考


前言

本章基于官方教程 Basic tutorial 7: Multithreading and Pad Availability 进行说明和补充。本章内容并不复杂,引入了两个新的 Element:Tee 和 Queue。Tee 用于复制数据, Queue 创建线程让 Pipeline 的一部分在另外线程中运行。

由于官方教程在本人机器上总是无法运行,我对它做了一些修改,具体代码在 basic-tutorial-7.c

Pipeline

在这里插入图片描述

本次教程中使用的 Pipeline 如上图所示,简单做个解释说明:

  1. audiotestsrc: 生成测试音频数据。
  2. tee:接收一个输入流,并可以创建多个完全相同的输出流。这些输出流可以被发送到不同的处理管道中。
  3. queue:创建一个 FIFO(先进先出)缓冲区。它接收输入数据,暂时存储,然后在单独线程中按照接收顺序输出数据。
  4. wavescope: 可视化音频数据的波形
  5. spectrascope:可视化音频数据的频谱
  6. videoconvert: 视频格式转换
  7. autovideosink: 显示视频数据

Request Pads

在 GStreamer 简明教程(三):动态调整 Pipeline 中我们学习了 uridecodebinuridecodebin 是一个典型的例子,展示了如何使用动态创建的 Pad。这种 Pad 类型被称为 Sometime Pad。当 uridecodebin 切换状态后,它会从输入源读取媒体信息,根据识别出的流动态添加相应数量的 Pad。随后,它通过发送 “pad-added” 信号来通知应用程序这些新创建的 Pad。

相比之下,Always Pad 是在元素创建时即存在,并在元素的整个生命周期内保持可用。这类 Pad 通常用于固定格式的数据流处理。

第三种类型是 Request Pad,它们根据需要通过显式请求创建。这类 Pad 被用于动态连接或多路输出的场景。例如,在解码器中,可以根据请求生成相应的 Pad,以处理特定类型的子流。

tee 元素就支持 Request Pad,首先通过 gst-inspect-1.0 来看下 tee 的详细信息,其中有 Pad Templates 的描述

Pad Templates:
  SINK template: 'sink'
    Availability: Always
    Capabilities:
      ANY
  
  SRC template: 'src_%u'
    Availability: On request
    Capabilities:
      ANY

src_%u 表示 Request Pad 的名称格式。其中 %u 是一个占位符,代表一个唯一的无符号整数。这意味着每次请求一个新的源 Pad 时,tee 元素将创建一个新的 Pad,并用依次递增的整数替换 %u。例如,第一次请求可能会创建 src_0,第二次创建 src_1,依此类推。这种命名方式帮助管理和区分多个动态创建的 Pad。

tee 元素创建 pad 的代码示例:

// 请求创建新的 src pad
GstPad *pad1 = gst_element_request_pad_simple(tee, "src_%u");
// 这可能创建名为 "src_0" 的 pad

GstPad *pad2 = gst_element_request_pad_simple(tee, "src_%u");
// 这可能创建名为 "src_1" 的 pad

// 使用这些 pad
// ...

// 在不需要时释放
gst_element_release_request_pad(tee, pad1);
gst_element_release_request_pad(tee, pad2);

Show me the code

接下来对具体代码进行说明,完整代码在 basic-tutorial-7.c

代码会生成两个窗口,一个窗口展示音频的频谱图,另一个显示音频的波形图
在这里插入图片描述

#include <gst/gst.h>

#ifdef __APPLE__
#include <TargetConditionals.h>
#endif

int tutorial_main(int argc, char *argv[]) {
  GstElement *pipeline;
  GstElement *audio_source;
  GstElement *queue;
  GstElement *visual;
  GstElement *video_convert;
  GstElement *video_sink;
  GstElement *tee;
  GstElement *queue2;
  GstElement *visual2;
  GstElement *video_convert2;
  GstElement *video_sink2;

  /* Initialize GStreamer */
  gst_init(&argc, &argv);

  // create elements
  audio_source = gst_element_factory_make("audiotestsrc", "audio_source");
  tee = gst_element_factory_make("tee", "tee");

  queue = gst_element_factory_make("queue", "queue");
  visual = gst_element_factory_make("wavescope", "visual");
  video_convert = gst_element_factory_make("videoconvert", "video_convert");
  video_sink = gst_element_factory_make("autovideosink", "video_sink");

  queue2 = gst_element_factory_make("queue", "queue2");
  visual2 = gst_element_factory_make("spectrascope", "visual2");
  video_convert2 = gst_element_factory_make("videoconvert", "video_convert2");
  video_sink2 = gst_element_factory_make("autovideosink", "video_sink2");

  // create pipeline
  pipeline = gst_pipeline_new("test-pipeline");

  if (!pipeline || !audio_source || !tee || !queue || !visual ||
      !video_convert || !video_sink || !queue2 || !visual2 || !video_convert2 ||
      !video_sink2) {
    g_printerr("Not all elements could be created.\n");
    return -1;
  }

  // configure elements
  g_object_set(audio_source, "freq", 215.0f, "wave", 2, NULL);
  g_object_set(visual, "shader", 0, "style", 1, NULL);
//  g_object_set(visual2, "shader", 0, "style", 3, NULL);

  // link elements
  gst_bin_add_many(GST_BIN(pipeline), audio_source, tee, queue, visual,
                   video_convert, video_sink, queue2, visual2, video_convert2,
                   video_sink2, NULL);

  if (gst_element_link_many(audio_source, tee, NULL) != TRUE) {
    g_printerr("Elements could not be linked.\n");
    gst_object_unref(pipeline);
    return -1;
  }

  // link first branch
  if (gst_element_link_many(queue, visual, video_convert, video_sink, NULL) !=
      TRUE) {
    g_printerr("Elements could not be linked.\n");
    gst_object_unref(pipeline);
    return -1;
  }

  // link second branch
  if (gst_element_link_many(queue2, visual2, video_convert2, video_sink2,
                            NULL) != TRUE) {
    g_printerr("Elements could not be linked.\n");
    gst_object_unref(pipeline);
    return -1;
  }

  /* Manually link the Tee, which has "Request" pads */
  GstPad *tee_audio_pad = gst_element_request_pad_simple(tee, "src_%u");
  g_print("Obtained request pad %s for first audio branch.\n",
          gst_pad_get_name(tee_audio_pad));
  GstPad *queue_audio_pad = gst_element_get_static_pad(queue, "sink");

  GstPad *tee_audio_pad2 = gst_element_request_pad_simple(tee, "src_%u");
  g_print("Obtained request pad %s for second audio branch.\n",
          gst_pad_get_name(tee_audio_pad));
  GstPad *queue_audio_pad2 = gst_element_get_static_pad(queue2, "sink");
  if (gst_pad_link(tee_audio_pad, queue_audio_pad) != GST_PAD_LINK_OK ||
      gst_pad_link(tee_audio_pad2, queue_audio_pad2) != GST_PAD_LINK_OK) {
    g_printerr("Tee could not be linked.\n");
    gst_object_unref(pipeline);
    return -1;
  }
  gst_object_unref(queue_audio_pad);
  gst_object_unref(queue_audio_pad2);

  // start playing
  gst_element_set_state(pipeline, GST_STATE_PLAYING);

  // wait until error or EOS
  GstBus *bus = gst_element_get_bus(pipeline);
  GstMessage *msg = gst_bus_timed_pop_filtered(
      bus, GST_CLOCK_TIME_NONE,
      (GstMessageType)(GST_MESSAGE_ERROR | GST_MESSAGE_EOS));

  gst_element_release_request_pad(tee, tee_audio_pad);
  gst_element_release_request_pad(tee, tee_audio_pad2);
  gst_object_unref(tee_audio_pad);
  gst_object_unref(tee_audio_pad2);

  // release resources
  if (msg != NULL) {
    gst_message_unref(msg);
  }
  gst_object_unref(bus);
  gst_element_set_state(pipeline, GST_STATE_NULL);
  gst_object_unref(pipeline);

  return 0;
}

int main(int argc, char *argv[]) {
#if defined(__APPLE__) && TARGET_OS_MAC && !TARGET_OS_IPHONE
  return gst_macos_main((GstMainFunc)tutorial_main, argc, argv, NULL);
#else
  return tutorial_main(argc, argv);
#endif
}
  1. 首先,创建各个元素,并将可以连接的部分先进行连接:
    • audio_source -> tee
    • queue -> visual -> video_convert -> video_sink
    • queue2 -> visual2 -> video_convert2 -> video_sink2
  2. 向 tee 元素请求两个 src pad,连接 tee 与 queue、queue2
  3. 其他就是启动 pipeline,监听 bus 数据等常规操作,不再赘述
  4. 最后就是清理工作

总之非常简单,非常清晰。


参考

  • Basic tutorial 7: Multithreading and Pad Availability
  • basic-tutorial-7.c

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2208792.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

超声波清洗机靠谱吗?适合学生党入手的四款眼镜清洗机品牌推荐!

有没有学生党还不知道双十一买什么&#xff1f;其实可以去看看超声波清洗机&#xff0c;说实话它的实用性真的很高&#xff0c;对于日常用于清洗眼镜真的非常合适&#xff0c;不仅可以帮助大家节约时间而且还能把眼镜清洗的干净透亮&#xff0c;接下来我就来为大家带来四款好用…

浅谈钓鱼攻防之道-制作免杀word文件钓鱼

梦里明明有六趣&#xff0c;觉后空空无大千 1、制作基本的word宏文件 Cobalt Strike生成宏代码 选择监听器 成功生成宏文件 新建word文档&#xff0c;点击视图——宏——查看宏 选择编辑 点击视图中的工程资源管理器 选择本文件中ThisDocument&#xff0c;将cs生成的文件复制…

MySql表结构设计

创建 create table 表名(字段1 字段类型 [约束] [comment 字段1注释],...) [comment 表注释];约束是作用于表中字段上的规则&#xff0c;用于限制存储在表中的数据。它的目的是保证数据库中数据的正确性、有效性和完整性。 约束描述关键字非空约束限制该字段不能为nullnot nu…

某宝228滑块和普通228滑块之间的区别

声明&#xff1a; 该文章为学习使用&#xff0c;严禁用于商业用途和非法用途&#xff0c;违者后果自负&#xff0c;由此产生的一切后果均与作者无关。 本文章未经许可禁止转载&#xff0c;禁止任何修改后二次传播&#xff0c;擅自使用本文讲解的技术而导致的任何意外&#xff…

C++11 新特性 学习笔记

C11 新特性 | 侯捷C11学习笔记 笔者作为侯捷C11新特性课程的笔记进行记录&#xff0c;供自己查阅方便 文章目录 C11 新特性 | 侯捷C11学习笔记1.Variadic TemplatesC11支持函数模板的默认模板参数C11在函数模板和类模板中使用可变参数 可变参数模板1) 可变参数函数模板2) 可变…

Spring Boot 项目中 Redis 与数据库性能对比实战:从缓存配置到时间分析,详解最佳实践

一、前言&#xff1a; 在现代应用中&#xff0c;随着数据量的增大和访问频率的提高&#xff0c;如何提高数据存取的性能变得尤为重要。缓存技术作为一种常见的优化手段&#xff0c;被广泛应用于减少数据库访问压力、提升系统响应速度。Redis 作为一种高效的内存缓存数据库&…

DBeaver连接mysql 9报错:Public Key Retrieval is not allowed

DBeaver连接mysql 9报错&#xff1a;Public Key Retrieval is not allowed 如图&#xff1a; 解决方案 编辑连接属性&#xff1a; 修改 allowPublicKeyRetrieval 的值为 true DBeaver连接mysql数据库执行.sql脚本&#xff0c;Windows_dbeaver执行sql脚本.sql文件-CSDN博客文章…

【Java】 —— 数据结构与集合源码:Vector、LinkedList在JDK8中的源码剖析

目录 7.2.4 Vector部分源码分析 7.3 链表LinkedList 7.3.1 链表与动态数组的区别 7.3.2 LinkedList源码分析 启示与开发建议 7.2.4 Vector部分源码分析 jdk1.8.0_271中&#xff1a; //属性 protected Object[] elementData; protected int elementCount;//构造器 public …

2024ccna考试时间?新手小白看这些就够了

2024年想要考取ccna证书的新手小白们&#xff0c;是不是正在为考试时间而烦恼呀&#xff0c;其实ccna的考试时间其实非常灵活&#xff0c;并不需要像其他考试那样死记硬背固定的日期。那么小编马上就给大家说说2024ccna考试时间&#xff0c;并且附带一些考试内容&#xff0c;让…

LeetCode[简单] 70. 爬楼梯

假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢&#xff1f; 思路 利用滚动数组 public class Solution {public int ClimbStairs(int n) { //滚动数组int f0 0, f1 0, f2 1;for(int i 1; i < n; i){…

拱坝与重力坝:结构特性与应用差异的深度解析

在水利工程领域&#xff0c;坝体结构的选择对于工程的稳定性、安全性以及经济效益具有至关重要的影响。拱坝与重力坝作为两种主要的坝型&#xff0c;各自具有独特的结构特性和应用场景。本文旨在深入探讨拱坝与重力坝的区别&#xff0c;从工程特点、坝体结构型式、材料选用、水…

国内如何下载谷歌浏览器(chrome浏览器)历史版本和chromedriver驱动,长期更新,建议收藏

众所周知&#xff0c;google是一直被国内屏蔽的&#xff0c;有时候想要下载个chrome浏览器都要去外网&#xff0c;或者到处去搜索才能下载到。因为下载chrome浏览器的这个网址&#xff1a;google.com/chrome/ 在国内是一直被屏蔽掉的。 今天主要讲解的是国内ChromeDriver 的下…

ET实现游戏中的红点提示系统(服务端)

目录 ☝&#x1f913;前言 ☝&#x1f913;一、实现思路 ☝&#x1f913;二、实现 &#x1f920;2.1 定义红点组件 &#x1f920;2.2 定义Proto消息体 &#x1f920;2.3 RedPointComponentSystem ☝&#x1f913;难点 ☝&#x1f913;前言 当我们闲来无事时打开农药想消…

数据结构-5.3.二叉树的定义和基本术语

一.二叉树的基本概念&#xff1a; 树是一种递归定义的数据结构&#xff0c;因此二叉树是递归定义的数据结构。 二.二叉树的五种状态&#xff1a; 三.几个特殊的二叉树&#xff1a; 1.满二叉树&#xff1a;结点总数就是通过等比数列公式求出来的&#xff0c;首项为1即根节点&a…

【网络协议】TCP协议常用机制——延迟应答、捎带应答、面向字节流、异常处理,保姆级详解,建议收藏

&#x1f490;个人主页&#xff1a;初晴~ &#x1f4da;相关专栏&#xff1a;计算机网络那些事 前几篇文章&#xff0c;博主带大家梳理了一下TCP协议的几个核心机制&#xff0c;比如保证可靠性的 确认应答、超时重传 机制&#xff0c;和提高传输效率的 滑动窗口及其相关优化机…

C++ Builder XE12关于KonopkaControls与TMS VCL UI Pack组件的安装

1、先打开open project&#xff0c;选中安装的组件工程&#xff0c;并打开。 2、在option中设置 3、点击编译并进行安装install

洞察AI趋势:智享AI直播,打造专属你的数字化直播AIGC系统!

洞察AI趋势&#xff1a;智享AI直播&#xff0c;打造专属你的数字化直播AIGC系统&#xff01; 在当今这个日新月异的数字时代&#xff0c;人工智能&#xff08;AI&#xff09;已不再是遥不可及的未来科技&#xff0c;而是正深刻改变着我们生活、工作的每一个角落。其中&#xf…

幽默视频下载网站推荐

在快节奏的生活中&#xff0c;搞笑视频无疑是缓解压力的良药&#xff0c;不论是自制搞笑视频还是寻找素材来增添创作的趣味性&#xff0c;找到合适的视频素材至关重要。幸运的是&#xff0c;网络上有许多优秀的网站能够满足这一需求。以下是8个适合下载幽默搞笑视频素材的网站&…

华为 静态路由和bfd 侦测的实验

实验要求 sw1 上业务地址192.168.1.1/24 SW3 业务地址192.168.2.1/24 正常情况下走主链路&#xff0c;不正常的情况下走备份链路 2 配置 这是基本地址配置 开启了bfd 本端地址为 10.1.1.1 对端地址是10.1.1.2 关键是discrimination 分辨参数 …

静态路由和nqa 联动实验

nqa 配置 1 test 断端口 很明显是切换到备机上了