目标

通过手动选择音频和视频的sink,playbin2可以进一步定制。这允许使用playbin2的应用在解码后可以自行做最终的渲染和显示。本教程展示了:

如何替换playbin2选择的sink

如何使用一个复杂的pipeline来作为sink

介绍

playbin2有两个属性:audio-sink和video-sink。应用只需要实例化合适的element然后通过这两个属性传给playbin2就行了。

这个方法只能使用一个简单地element来做sink。如果遇到的情况不是简单地element而是一个复杂的pipeline呢(比如均衡器加一个音频sink),就需要用Bin来包装一下,让playbin2感觉还是一个element。

一个Bin就是一个容器,可以容纳一部分的pipeline,但操作的时候作为一个element。例如,我们在所有教程里面用得GstPipeline就是一个不会和外面的element做交互的GstBin。在Bin里面的element通过虚拟Pad(Ghost Pads)来和外面的element交互。这也就是说,Bin上得一个pad仅仅实现把外面的element上的pad的数据传到内部的element的pad上。

GstBin也是element的一类,所以element可以用的地方,Bin也都能用,特别是,可以作为playbin2的sink。

一个均衡的播放器

[objc] view
plain
 copy

  1. #include <gst/gst.h>
  2. int main(int argc, charchar *argv[]) {
  3. GstElement *pipeline, *bin, *equalizer, *convert, *sink;
  4. GstPad *pad, *ghost_pad;
  5. GstBus *bus;
  6. GstMessage *msg;
  7. /* Initialize GStreamer */
  8. gst_init (&argc, &argv);
  9. /* Build the pipeline */
  10. pipeline = gst_parse_launch ("playbin2 uri=http://docs.gstreamer.com/media/sintel_trailer-480p.webm", NULL);
  11. /* Create the elements inside the sink bin */
  12. equalizer = gst_element_factory_make ("equalizer-3bands", "equalizer");
  13. convert = gst_element_factory_make ("audioconvert", "convert");
  14. sink = gst_element_factory_make ("autoaudiosink", "audio_sink");
  15. if (!equalizer || !convert || !sink) {
  16. g_printerr ("Not all elements could be created.\n");
  17. ;
  18. }
  19. /* Create the sink bin, add the elements and link them */
  20. bin = gst_bin_new ("audio_sink_bin");
  21. gst_bin_add_many (GST_BIN (bin), equalizer, convert, sink, NULL);
  22. gst_element_link_many (equalizer, convert, sink, NULL);
  23. pad = gst_element_get_static_pad (equalizer, "sink");
  24. ghost_pad = gst_ghost_pad_new ("sink", pad);
  25. gst_pad_set_active (ghost_pad, TRUE);
  26. gst_element_add_pad (bin, ghost_pad);
  27. gst_object_unref (pad);
  28. /* Configure the equalizer */
  29. 4.0, NULL);
  30. 4.0, NULL);
  31. /* Set playbin2's audio sink to be our sink bin */
  32. g_object_set (GST_OBJECT (pipeline), "audio-sink", bin, NULL);
  33. /* Start playing */
  34. gst_element_set_state (pipeline, GST_STATE_PLAYING);
  35. /* Wait until error or EOS */
  36. bus = gst_element_get_bus (pipeline);
  37. msg = gst_bus_timed_pop_filtered (bus, GST_CLOCK_TIME_NONE, GST_MESSAGE_ERROR | GST_MESSAGE_EOS);
  38. /* Free resources */
  39. if (msg != NULL)
  40. gst_message_unref (msg);
  41. gst_object_unref (bus);
  42. gst_element_set_state (pipeline, GST_STATE_NULL);
  43. gst_object_unref (pipeline);
  44. ;
  45. }

工作流程

[objc] view
plain
 copy

  1. /* Create the elements inside the sink bin */
  2. equalizer = gst_element_factory_make ("equalizer-3bands", "equalizer");
  3. convert = gst_element_factory_make ("audioconvert", "convert");
  4. sink = gst_element_factory_make ("autoaudiosink", "audio_sink");
  5. if (!equalizer || !convert || !sink) {
  6. g_printerr ("Not all elements could be created.\n");
  7. ;
  8. }

生成所有的sink bin所需要的element。我们使用了一个equalizer-3bands和一个autoaudiosink,中间还用audioconvert连接起来。

[objc] view
plain
 copy

  1. /* Create the sink bin, add the elements and link them */
  2. bin = gst_bin_new ("audio_sink_bin");
  3. gst_bin_add_many (GST_BIN (bin), equalizer, convert, sink, NULL);
  4. gst_element_link_many (equalizer, convert, sink, NULL);

这几句把新的element加到Bin里面然后连接起来,就和在pipeline中一样。

[objc] view
plain
 copy

  1. pad = gst_element_get_static_pad (equalizer, "sink");
  2. ghost_pad = gst_ghost_pad_new ("sink", pad);
  3. gst_pad_set_active (ghost_pad, TRUE);
  4. gst_element_add_pad (bin, ghost_pad);
  5. gst_object_unref (pad);

现在我们需要生成虚拟Pad,这也一部分在Bin里面的pipeline可以连接到外面。这个虚拟Pad会和内部的一个element的Pad连接起来,我们会用gst_element_get_static_pad()来获得这个Pad。如果是一个RequestPad而不是Always
Pad的话,那么我们就用get_element_request_pad()方法,具体请参考《GStreamer基础教程07——多线程和Pad的有效性

虚拟Pad用gst_ghost_pad_new()来创建,用gst_pad_set_active()来激活。然后用gst_element_add_pad()来加入Bin。

最后,我们用gst_object_unref()来释放获得的均衡器的sink pad。

在这点上,我们有一个有sink功能的Bin,可以在playbin2里面用作音频sink,我们只需要设置一下playbin2的相关属性就行了。

[objc] view
plain
 copy

  1. /* Set playbin2's audio sink to be our sink bin */
  2. g_object_set (GST_OBJECT (pipeline), "audio-sink", bin, NULL);

这和用一个sink element设置playbin2的audio-sink属性是完全一样的。

[objc] view
plain
 copy

  1. /* Configure the equalizer */
  2. 4.0, NULL);
  3. 4.0, NULL);

剩下的就是均衡器的配置了。在这个例子中,两个高频波段设置了最大的衰减,这样就增强了低音修改不同的值来实际看看效果。


【GStreamer开发】GStreamer播放教程07——自定义playbin2的sink的更多相关文章

  1. 新手编译开发OpenWrt入门教程(自定义固件、ubuntu学习)

    转自:   http://www.znck007.com/forum.php?mod=viewthread&tid=21571 由于openwrt编译教程资料很多,不同的cpu芯片只需要选择对 ...

  2. 【GStreamer开发】GStreamer基础教程13——播放速度

    目标 快进,倒放和慢放是trick模式的共同技巧,它们有一个共同点就是它们都修改了播放的速度.本教程会展示如何来获得这些效果和如何进行逐帧的跳跃.主要内容是: 如何来变换播放的速度,变快或者变慢,前进 ...

  3. 【GStreamer开发】GStreamer基础教程14——常用的element

    目标 本教程给出了一系列开发中常用的element.它们包括大杂烩般的eleemnt(比如playbin2)以及一些调试时很有用的element. 简单来说,下面用gst-launch这个工具给出一个 ...

  4. 【GStreamer开发】GStreamer基础教程10——GStreamer工具

    目标 GStreamer提供了一系列方便使用的工具.这篇教程里不牵涉任何代码,但还是会讲一些有用的内容: 如何在命令行下建立一个pipeline--完全不使用C 如何找出一个element的Capab ...

  5. 【GStreamer开发】GStreamer基础教程15——继承Clutter

    目标 Clutter是一个开源的库,用来创建快速.可移植和动态的GUI.GStreamer可以通过cluttersink这个element把clutter集成进来,允许视频像纹理一样使用.本教程会展示 ...

  6. 【GStreamer开发】GStreamer基础教程12——流

    目标 直接播放Internet上的文件而不在本地保存就被称为流播放.我们在前面教程里已经这样做过了,使用了http://的URL.本教程展示的是在播放流的时候需要记住的几个点,特别是: 如何设置缓冲 ...

  7. 【GStreamer开发】GStreamer基础教程05——集成GUI工具

    目标 本教程展示了如何在GStreamer集成一个GUI(比如:GTK+).最基本的原则是GStreamer处理多媒体的播放而GUI处理和用户的交互. 在这个教程里面,我们可以学到: 如何告诉GStr ...

  8. 【GStreamer开发】GStreamer基础教程08——pipeline的快捷访问

    目标 GStreamer建立的pipeline不需要完全关闭.有多种方法可以让数据在任何时候送到pipeline中或者从pipeline中取出.本教程会展示: 如何把外部数据送到pipeline中 如 ...

  9. gstreamer应用开发(播放器)之旅

    GStreamer开发,主要分为两块:应用开发.插件开发. 插件开发人员,通常是编解码库的作者(做出了编解码库后,希望gstreamer能用起来这个库,因此增加这个适配层).芯片原厂人员(将自家的hw ...

随机推荐

  1. 4-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案升级篇(远程升级WIFI内部程序)

    https://www.cnblogs.com/yangfengwu/p/10360618.html 演示视频: https://www.bilibili.com/video/av54894356/ ...

  2. noi.ac #30 思维

    \(des\) 给定升序数组 \(A, B\) 对于任意两个集合 \(a, b\) 分别是 \(A, B\) 的子集,总价值为较小的集合的和, 总代价为 \((|a| + |b|) \times w\ ...

  3. UVA 1613 K度图染色

    题目 \(dfs+\)证明. 对于题目描述,可以发现\(K\)其实就是大于等于原图中最大度数的最小奇数,因为如果原图度数最大为奇数,则最多颜色肯定为K,而如果原图最大度数为偶数,则\(K\)又是奇数, ...

  4. 解决Spring Boot 拦截器注入service为空的问题

    问题:在自定义拦截器中,使用了@Autowaire注解注入了封装JPA方法的Service,结果发现无法注入,注入的service为空 0.原因分析 拦截器加载的时间点在springcontext之前 ...

  5. MyBatis项目配置案例详解与Web下的增删改查实现[附项目源码]

    MyBatis项目案例 项目图示: 项目源码地址:https://github.com/JluTiger/mybatispro 1.项目功能 项目案例:后台管理系统用户数据维护平台 所有用户数据查询 ...

  6. OpenFOAM显示残差

    本文主要讲解两种方法用来显示OpenFOAM的计算残差,一种是采用OpenFOAM自带的foamMonitor来输出残差,另一种就是大家经常看见的采用pyFoam来输出残差.不管采用哪一种方法都必须安 ...

  7. 小福bbs-冲刺日志(第七天)

    [小福bbs-冲刺日志(第七天)] 这个作业属于哪个课程 班级链接 这个作业要求在哪里 作业要求的链接 团队名称 小福bbs 这个作业的目标 目前完成静态 作业的正文 小福bbs-冲刺日志(第七天) ...

  8. Linux中的定时自动执行功能(at,crontab)

    Linux中的定时自动执行功能(at,crontab) 概念 在Linux系统中,提供了两种提前对工作进行安排的方式 at 只执行一次 crontab 周期性重复执行 通过对这两个工具的应用可以让我们 ...

  9. 根据udev的信息判断设备物理路径

    udev会生成by-path路径,根据这个就可以判断 dev目录下 [toybrick@localhost dev]$ find | grep platform-fe3c0000 ./disk/by- ...

  10. Apache 使用ssl模块配置HTTPS(Centos7 httpd2.4.6)

    根据原文:http://blog.csdn.net/ithomer/article/details/50433363改编 Web服务器在默认情况下使用HTTP,这是一个纯文本的协议.正如其名称所暗示的 ...