Chinaunix首页 | 论坛 | 博客
  • 博客访问: 4024624
  • 博文数量: 366
  • 博客积分: 9916
  • 博客等级: 中将
  • 技术积分: 7195
  • 用 户 组: 普通用户
  • 注册时间: 2011-05-29 23:27
个人简介

简单!

文章分类

全部博文(366)

文章存档

2013年(51)

2012年(269)

2011年(46)

分类: LINUX

2012-12-25 23:41:50

一、安装插件


首先需要使用mad解码插件,因此先安装gstreamer0.10-plugins-ugly



、编写mp3播放器


下面来看看如何利用GStreamer框架提供的组件,来实现一个简单的MP3播放器。数据源元件负责从磁盘上读取数据,过滤器元件负责对数据进行解码,而接受器元件则负责将解码后的数据写入声卡。

      想要在程序中应用GStreamer提供的各种功能,首先必须在主函数中调用gst_init()来完成初始化工作,以便将用户从命令行输入的参数传递给GStreamer函数库。一个典型的GStreamer应用程序的初始化如下所示:

  1. #include <gst/gst.h>

  2. int main (int argc, char *argv[])
  3. {
  4.     gst_init (&argc, &argv);
  5.     /* */
  6. }

      接下去需要创建三个元件并连接成管道,由于所有GStreamer元件都具有相同的基类GstElement,因此能够采用如下方式进行定义:

  1. GstElement *pipeline, *filesrc, *decoder, *audiosink;

      管道在GStreamer框架中是用来容纳和管理元件的,下面的代码将创建一条名为pipeline的新管道:

  1. /* 创建用来容纳元件的新管道 */
  2.   pipeline = gst_pipeline_new ("pipeline");

      数据源元件负责从磁盘文件中读取数据,它具有名为location的属性,用来指明文件在磁盘上的位置。使用标准的GObject属性机制可以为元件设置相应的属性:

  1. /* 创建数据源元件 */
  2. filesrc = gst_element_factory_make ("filesrc", "disk_source");
  3. g_object_set (G_OBJECT (filesrc), "location", argv[1], NULL);

      过滤器元件负责完成对MP3格式的数据进行解码,最简单的办法是安装mad这一插件,借助它来完成相应的解码工作:

  1. /* 创建过滤器元件 */
  2. decoder = gst_element_factory_make ("mad", "decoder");

      接收器元件负责将解码后的数据利用声卡播放出来:

  1. /* 创建接收器元件 */
  2. audiosink = gst_element_factory_make ("audiosink", "play_audio");

      已经创建好的三个元件需要全部添加到管道中,并按顺序连接起来:

  1. /* 添加元件到管道中 */
  2. gst_bin_add_many (GST_BIN (pipeline), filesrc, decoder, audiosink, NULL);
  3. /* 通过衬垫连接元件 */
  4. gst_element_link_many (filesrc, decoder, audiosink, NULL);

所有准备工作都做好之后,就可以通过将管道的状态切换到PLAYING状态,来启动整个管道的数据处理流程:

  1. /* 启动管道 */
  2. gst_element_set_state (pipeline, GST_STATE_PLAYING);

      这里加入一个消息处理函数bus_call来监视产生的消息

  1. /* 终止管道 */
  2. gst_element_set_state (pipeline, GST_STATE_NULL);
  3. /* 释放资源 */
  4. gst_object_unref (GST_OBJECT (pipeline));


三、完整的源码


代码如下所示:

  1. #include <gst/gst.h>
  2. #include <glib.h>

  3. //定义消息处理函数
  4. static gboolean bus_call(GstBus * bus, GstMessage * msg, gpointer data)
  5. {
  6.     GMainLoop *loop = (GMainLoop *) data;

  7.     switch (GST_MESSAGE_TYPE(msg)) {
  8.     case GST_MESSAGE_EOS:
  9.         g_print("End of stream\n");
  10.         g_main_loop_quit(loop);
  11.         break;
  12.     case GST_MESSAGE_ERROR:
  13.         {
  14.             gchar *debug;
  15.             GError *error;

  16.             gst_message_parse_error(msg, &error, &debug);
  17.             g_free(debug);
  18.             g_printerr("ERROR:%s\n", error->message);
  19.             g_error_free(error);
  20.             g_main_loop_quit(loop);
  21.             break;
  22.         }
  23.     default:
  24.         break;
  25.     }

  26.     return TRUE;
  27. }

  28. int main(int argc, char *argv[])
  29. {
  30.     GMainLoop *loop;
  31.     GstElement *pipeline, *source, *decoder, *sink;    //定义组件
  32.     GstBus *bus;

  33.     gst_init(&argc, &argv);
  34.     loop = g_main_loop_new(NULL, FALSE);    //创建主循环,在执行 g_main_loop_run后正式开始循环

  35.     if (argc != 2) {
  36.         g_printerr("Usage:%s \n", argv[0]);
  37.         return -1;
  38.     }

  39.     //创建管道和组件
  40.     pipeline = gst_pipeline_new("audio-player");
  41.     source = gst_element_factory_make("filesrc", "file-source");
  42.     decoder = gst_element_factory_make("mad", "mad-decoder");
  43.     sink = gst_element_factory_make("autoaudiosink", "audio-output");

  44.     if (!pipeline || !source || !decoder || !sink) {
  45.         g_printerr("One element could not be created.Exiting.\n");
  46.         return -1;
  47.     }

  48.     //设置source的location参数,即文件地址
  49.     g_object_set(G_OBJECT(source), "location", argv[1], NULL);
  50.     //得到管道的消息总线
  51.     bus = gst_pipeline_get_bus(GST_PIPELINE(pipeline));
  52.     //添加消息监视器
  53.     gst_bus_add_watch(bus, bus_call, loop);
  54.     gst_object_unref(bus);
  55.     //把组件添加到管道中.管道是一个特殊的组件,可以更好的让数据流动
  56.     gst_bin_add_many(GST_BIN(pipeline), source, decoder, sink, NULL);
  57.     //依次连接组件
  58.     gst_element_link_many(source, decoder, sink, NULL);

  59.     //开始播放
  60.     gst_element_set_state(pipeline, GST_STATE_PLAYING);
  61.     g_print("Running\n");

  62.     //开始循环
  63.     g_main_loop_run(loop);
  64.     g_print("Returned,stopping playback\n");
  65.     gst_element_set_state(pipeline, GST_STATE_NULL);
  66.     gst_object_unref(GST_OBJECT(pipeline));

  67.     return 0;
  68. }


、编译运行


$ gcc -Wall $(pkg-config --cflags --libs gstreamer-0.10)  test.c -o test  `pkg-config --libs glib-2.0` `pkg-config --libs gstreamer-0.10`
$  ./test test.mp3
阅读(5538) | 评论(0) | 转发(0) |
0

上一篇:Kibo

下一篇:利用SDL创建一个窗口

给主人留下些什么吧!~~