Gstreamer-GstStructure

相关连接:https://blog.csdn.net/knowledgebao/article/details/84621238


GstStructure — Generic structure containing fields of names and values。类似于map的一个结构。key的类型是:GQuary,value的类型是GValue,详见参考资料6.

关于GQuery的详细描述,详见参考资料的1,2,3.

关于GValue的详细描述,详见参考资料4。

关于GstValue的详细描述,详见参考资料5.

//功能:tries to transform the given value into a string representation that allows getting back this string later on using gst_value_deserialize().
//value:a GValue to serialize
//Returns:the serialization for value or NULL if none exists.
//必须调用g_free释放。
gchar *gst_value_serialize (
const GValue *value
):

 


gst_structure_foreach:
功能:遍历structure,通过回调的方式列举,举例如下: 

gboolean gst_structure_foreach (const GstStructure *structure,
                       GstStructureForeachFunc func,
                       gpointer user_data);
static gboolean print_field(GQuark field, const GValue * value, gpointer pfx) {
    gchar *str = gst_value_serialize(value);
    g_print("%s  %15s: %s\n", (gchar *)pfx, g_quark_to_string(field), str);
    g_free(str);
    return TRUE;
}
void print_caps(const GstCaps * caps, const gchar * pfx) {
    guint i;
    for (i = 0; i < gst_caps_get_size(caps); i++) {
        GstStructure *structure = gst_caps_get_structure(caps, i);
        gst_structure_foreach(structure, print_field, (gpointer)pfx); 
    }
}

gst_structure_id_get_value ():

功能:通过GQark获取GValue。

const GValue * gst_structure_id_get_value (const GstStructure *structure,
                            GQuark field);

 

参考资料:

  1. Glib中Gquark浅析
  2. Glib中Gquark浅析
  3. glib库中的GQuark介绍
  4. GObject笔记-通用类型
  5. GstValue
  6. GstStructure

有任何问题,请联系:knowledgebao@163.com

以下是Linux平台C语言利用GStreamer实现mp4格式视频播放的代码,包括管道配置,连接,以及动态连接element的pad。其中难点在于demux控件通过回调方式进行音视频的分流,连接解码器、sink输出。 ```c #include <gst/gst.h> typedef struct _CustomData { GstElement *pipeline; GstElement *source; GstElement *demuxer; GstElement *audio_decoder; GstElement *video_decoder; GstElement *audio_sink; GstElement *video_sink; GstElement *convert; GstElement *resample; GstElement *audio_queue; GstElement *video_queue; GstElement *video_convert; GstElement *video_scale; GstElement *video_filter; GstElement *audio_filter; GstElement *audio_convert; GstElement *audio_resample; GstElement *audio_sink2; GstElement *video_sink2; GstBus *bus; guint bus_watch_id; gboolean is_live; } CustomData; static void pad_added_handler (GstElement *src, GstPad *new_pad, CustomData *data); int main(int argc, char *argv[]) { CustomData data; GstStateChangeReturn ret; GstPad *audio_sink_pad = NULL; GstPad *video_sink_pad = NULL; GstPad *audio_src_pad = NULL; GstPad *video_src_pad = NULL; GstCaps *audio_caps = NULL; GstCaps *video_caps = NULL; GstCaps *filter_caps = NULL; GstCaps *convert_caps = NULL; GstCaps *scale_caps = NULL; GstCaps *resample_caps = NULL; GstCaps *audio_convert_caps = NULL; GstCaps *audio_resample_caps = NULL; GstCaps *audio_filter_caps = NULL; GstCaps *video_filter_caps = NULL; GstCaps *video_convert_caps = NULL; GstCaps *video_scale_caps = NULL; /* Initialize GStreamer */ gst_init (&argc, &argv); /* Initialize our data structure */ memset (&data, 0, sizeof (data)); data.is_live = FALSE; /* Create the elements */ data.source = gst_element_factory_make ("filesrc", "source"); data.demuxer = gst_element_factory_make ("qtdemux", "demuxer"); data.audio_decoder = gst_element_factory_make ("decodebin", "audio_decoder"); data.video_decoder = gst_element_factory_make ("decodebin", "video_decoder"); data.convert = gst_element_factory_make ("videoconvert", "convert"); data.resample = gst_element_factory_make ("audioresample", "resample"); data.audio_queue = gst_element_factory_make ("queue", "audio_queue"); data.video_queue = gst_element_factory_make ("queue", "video_queue"); data.audio_sink = gst_element_factory_make ("autoaudiosink", "audio_sink"); data.video_sink = gst_element_factory_make ("autovideosink", "video_sink"); data.video_convert = gst_element_factory_make ("videoconvert", "video_convert"); data.video_scale = gst_element_factory_make ("videoscale", "video_scale"); data.video_filter = gst_element_factory_make ("capsfilter", "video_filter"); data.audio_filter = gst_element_factory_make ("capsfilter", "audio_filter"); data.audio_convert = gst_element_factory_make ("audioconvert", "audio_convert"); data.audio_resample = gst_element_factory_make ("audioresample", "audio_resample"); data.audio_sink2 = gst_element_factory_make ("alsasink", "audio_sink2"); data.video_sink2 = gst_element_factory_make ("xvimagesink", "video_sink2"); /* Create the empty pipeline */ data.pipeline = gst_pipeline_new ("test-pipeline"); if (!data.pipeline || !data.source || !data.demuxer || !data.audio_decoder || !data.video_decoder || !data.convert || !data.resample || !data.audio_queue || !data.video_queue || !data.audio_sink || !data.video_sink || !data.video_convert || !data.video_scale || !data.video_filter || !data.audio_filter || !data.audio_convert || !data.audio_resample || !data.audio_sink2 || !data.video_sink2) { g_printerr ("Not all elements could be created.\n"); return -1; } /* Build the pipeline */ gst_bin_add_many (GST_BIN (data.pipeline), data.source, data.demuxer, data.audio_decoder, data.video_decoder, data.convert, data.resample, data.audio_queue, data.video_queue, data.audio_sink, data.video_sink, data.video_convert, data.video_scale, data.video_filter, data.audio_filter, data.audio_convert, data.audio_resample, data.audio_sink2, data.video_sink2, NULL); if (gst_element_link (data.source, data.demuxer) != TRUE || gst_element_link_many (data.audio_queue, data.audio_convert, data.audio_resample, data.audio_filter, data.audio_sink2, NULL) != TRUE || gst_element_link_many (data.video_queue, data.video_convert, data.video_scale, data.video_filter, data.video_sink2, NULL) != TRUE) { g_printerr ("Elements could not be linked.\n"); gst_object_unref (data.pipeline); return -1; } /* Set the URI to play */ g_object_set (data.source, "location", argv[1], NULL); /* Connect to the pad-added signal */ g_signal_connect (data.demuxer, "pad-added", G_CALLBACK (pad_added_handler), &data); /* Start playing */ ret = gst_element_set_state (data.pipeline, GST_STATE_PLAYING); if (ret == GST_STATE_CHANGE_FAILURE) { g_printerr ("Unable to set the pipeline to the playing state.\n"); gst_object_unref (data.pipeline); return -1; } /* Wait until error or EOS */ data.bus = gst_element_get_bus (data.pipeline); data.bus_watch_id = gst_bus_add_watch (data.bus, bus_call, loop); gst_object_unref (data.bus); g_main_loop_run (loop); /* Free resources */ gst_element_set_state (data.pipeline, GST_STATE_NULL); gst_object_unref (data.pipeline); return 0; } static void pad_added_handler (GstElement *src, GstPad *new_pad, CustomData *data) { GstPad *sink_pad = NULL; GstPadLinkReturn ret; GstCaps *new_pad_caps = NULL; GstStructure *new_pad_struct = NULL; const gchar *new_pad_type = NULL; g_print ("Received new pad '%s' from '%s':\n", GST_PAD_NAME (new_pad), GST_ELEMENT_NAME (src)); /* Check the new pad's type */ new_pad_caps = gst_pad_get_current_caps (new_pad); new_pad_struct = gst_caps_get_structure (new_pad_caps, 0); new_pad_type = gst_structure_get_name (new_pad_struct); if (g_str_has_prefix (new_pad_type, "audio/x-raw")) { sink_pad = gst_element_get_static_pad (data->audio_queue, "sink"); g_print ("Linking audio demuxer to audio queue.\n"); } else if (g_str_has_prefix (new_pad_type, "video/x-raw")) { sink_pad = gst_element_get_static_pad (data->video_queue, "sink"); g_print ("Linking video demuxer to video queue.\n"); } else { g_print ("It has type '%s' which is not raw audio. Ignoring.\n", new_pad_type); goto exit; } /* If we have an unlinked sink pad, link it now */ if (gst_pad_is_linked (sink_pad)) { g_print ("We are already linked. Ignoring.\n"); goto exit; } /* Attempt the link */ ret = gst_pad_link (new_pad, sink_pad); if (GST_PAD_LINK_FAILED (ret)) { g_print ("Type is '%s' but link failed.\n", new_pad_type); } else { g_print ("Link succeeded (type '%s').\n", new_pad_type); } exit: /* Unreference the new pad's caps, if we got them */ if (new_pad_caps != NULL) gst_caps_unref (new_pad_caps); /* Unreference the sink pad */ if (sink_pad != NULL) gst_object_unref (sink_pad); } ```
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值