目标

通过手动选择音频和视频的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. Java为什么会流行

    为什么Java语言能长期占据编程语言排行榜首位? 面向对象设计,使用简单,可以很快速的入门: 开源生态做得好,很多可重用的组件,拿来即用: 跨平台,高性能,是做后台开发的首选 另一方面,Java命好, ...

  2. Angular惰性加载的特性模块

    一:Angular-CLI建立应用 cmd命令:ng new lazy-app --routing    (创建一个名叫 lazy-app 的应用,而 --routing 标识生成了一个名叫 app- ...

  3. Matlab中矩阵的数据结构

    在Matlab中,矩阵默认的数据类型是double, 并不是integer. 而且奇怪的是,矩阵乘法默认按照浮点数类型进行, 整数矩阵相乘会报错.另外,可以用a= int16(A)这种形式实现数据类型 ...

  4. Kafka 幂等生产者和事务生产者特性(讨论基于 kafka-python | confluent-kafka 客户端)

    Kafka 提供了一个消息交付可靠性保障以及精确处理一次语义的实现.通常来说消息队列都提供多种消息语义保证 最多一次 (at most once): 消息可能会丢失,但绝不会被重复发送. 至少一次 ( ...

  5. 《挑战30天C++入门极限》C++运算符重载函数基础及其值返回状态

        C++运算符重载函数基础及其值返回状态 运算符重载是C++的重要组成部分,它可以让程序更加的简单易懂,简单的运算符使用可以使复杂函数的理解更直观. 对于普通对象来说我们很自然的会频繁使用算数运 ...

  6. java如何实现多线程?线程的状态有哪些?

    java实现多线程有两种方法    1.继承Thread类    2.实现Runnable接口    这两种方法的共同点:    不论用哪种方法,都必须用Thread(如果是Thead子类就用它本身) ...

  7. GO语言网络编程

    socket编程 Socket是BSD UNIX的进程通信机制,通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄.Socket可以理解为TCP/IP网络的API,它 ...

  8. JVM命令行参数

    root@ubuntu-blade2:/sdf/jdk# javaUsage: java [-options] class [args...] (to execute a class) or java ...

  9. ciscn2019华北赛区半决赛day1_web1题解

    感谢buuoj的大佬们搭建的复现环境.作为一位CTF的初学者,我会把每个题目的writeup都写的尽量详细,希望能帮到后面的初学者. http://web42.buuoj.cn 文章会不定时继续完善, ...

  10. NumPyArray

    import arcpy import numpy # Create a simple array from scratch using random values myArray = numpy.r ...