[GStreamer] gst_element_link 及 pad 在继承体系中如何存在

前言:

pad 是每个 element实例 都有的,是 elemenet 之间沟通的代理人,没有 pad 的 element 没法于其他element交流。

考虑到gstreamer存在继承体系,那么如果继承类element不在init函数里创建pad,那么在gst_element_link的时候是否可以使用父类element实例的pad呢? 下面从源码来找答案。




gst_element_link

在 gstutils.c 找到 gst_element_link(...) 的实现。

/*** gst_element_link:* @src: (transfer none): a #GstElement containing the source pad.* @dest: (transfer none): the #GstElement containing the destination pad.** Links @src to @dest. The link must be from source to* destination; the other direction will not be tried. The function looks for* existing pads that aren't linked yet. It will request new pads if necessary.* Such pads need to be released manually when unlinking.* If multiple links are possible, only one is established.** Make sure you have added your elements to a bin or pipeline with* gst_bin_add() before trying to link them.** Returns: %TRUE if the elements could be linked, %FALSE otherwise.*/
gboolean
gst_element_link (GstElement * src, GstElement * dest)
{return gst_element_link_pads (src, NULL, dest, NULL);
}

直接调用 gst_element_link_pads(...) , 同样在 gstutils.c ,入参为上有element和下游element 。




gst_element_link_pads

实现同样在 gstutils.c  里。

/*** gst_element_link_pads:* @src: a #GstElement containing the source pad.* @srcpadname: (nullable): the name of the #GstPad in source element*     or %NULL for any pad.* @dest: (transfer none): the #GstElement containing the destination pad.* @destpadname: (nullable): the name of the #GstPad in destination element,* or %NULL for any pad.** Links the two named pads of the source and destination elements.* Side effect is that if one of the pads has no parent, it becomes a* child of the parent of the other element.  If they have different* parents, the link fails.** Returns: %TRUE if the pads could be linked, %FALSE otherwise.*/
gboolean
gst_element_link_pads (GstElement * src, const gchar * srcpadname,GstElement * dest, const gchar * destpadname)
{return gst_element_link_pads_full (src, srcpadname, dest, destpadname,GST_PAD_LINK_CHECK_DEFAULT);
}

回到gst_element_link,第二个和第四个参数为NULL,结合gst_element_link_pads,第二个和第四个参数为padname,因此gst_element_link的行为就是“链接两个element,不在两个element中指定使用哪些pad实例进行互相链接”




gst_element_link_pads_full

这个函数有点长,我们节选,全文同样在 gstutils.c  里。

定义及注释说明,暂且略过。

/*** gst_element_link_pads_full:* @src: a #GstElement containing the source pad.* @srcpadname: (nullable): the name of the #GstPad in source element*     or %NULL for any pad.* @dest: (transfer none): the #GstElement containing the destination pad.* @destpadname: (nullable): the name of the #GstPad in destination element,* or %NULL for any pad.* @flags: the #GstPadLinkCheck to be performed when linking pads.** Links the two named pads of the source and destination elements.* Side effect is that if one of the pads has no parent, it becomes a* child of the parent of the other element.  If they have different* parents, the link fails.** Calling gst_element_link_pads_full() with @flags == %GST_PAD_LINK_CHECK_DEFAULT* is the same as calling gst_element_link_pads() and the recommended way of* linking pads with safety checks applied.** This is a convenience function for gst_pad_link_full().** Returns: %TRUE if the pads could be linked, %FALSE otherwise.*/
gboolean
gst_element_link_pads_full (GstElement * src, const gchar * srcpadname,GstElement * dest, const gchar * destpadname, GstPadLinkCheck flags)

如果指定了上游element的padname

则在上游element里面查找pad实例,如果查找过程出现问题,则return false(对应name的pad没找到 / pad不是src pad / pad已经有peer pad了,即pad已经被链接过了)。

  /* get a src pad */if (srcpadname) {/* name specified, look it up */if (!(srcpad = gst_element_get_static_pad (src, srcpadname))) {if ((srcpad = gst_element_request_pad_simple (src, srcpadname)))srcrequest = TRUE;}if (!srcpad) {GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS, "no pad %s:%s",GST_ELEMENT_NAME (src), srcpadname);return FALSE;} else {if (!(GST_PAD_DIRECTION (srcpad) == GST_PAD_SRC)) {GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS, "pad %s:%s is no src pad",GST_DEBUG_PAD_NAME (srcpad));release_and_unref_pad (src, srcpad, srcrequest);return FALSE;}if (GST_PAD_PEER (srcpad) != NULL) {GST_CAT_DEBUG (GST_CAT_ELEMENT_PADS,"pad %s:%s is already linked to %s:%s", GST_DEBUG_PAD_NAME (srcpad),GST_DEBUG_PAD_NAME (GST_PAD_PEER (srcpad)));/* already linked request pads look like static pads, so the request pad* was never requested a second time above, so no need to release it */gst_object_unref (srcpad);return FALSE;}}srcpads = NULL;}

如果没有指定上有element的padname

从上游element中找到所有pad,即padlist。

else {/* no name given, get the first available pad */GST_OBJECT_LOCK (src);srcpads = GST_ELEMENT_PADS (src);srcpad = srcpads ? GST_PAD_CAST (srcpads->data) : NULL;if (srcpad)gst_object_ref (srcpad);GST_OBJECT_UNLOCK (src);}

注意,这里用的是 GST_ELEMENT_PADS (src) ,是直接吧当前 element 转换为 GstElement,然后获取GstElement的 pads 成员。记住 这里是 “获取 GstElement 的pads 成员”。到这里我想大家应该已经猜出来pad的继承关系了,如果还没懂,我们回忆一下自定义Element如何给自己添加pad。

在instance结构体里定义两个pad成员。

typedef struct _GstMyFilter {GstElement element;//padsGstPad* srcpad;		//src padGstPad* sinkpad;		//sink pad
} GstMyFilter;

在init函数里实例化pad,并且append 到 其 parent element GstElement 的 pads 列表中。

static void gst_my_filter_init(GstMyFilter *filter)
{...//instantiates and assigns padsfilter->srcpad = gst_pad_new_from_static_template(&src_factory,"src");filter->sinkpad = gst_pad_new_from_static_template(&sink_factory, "sink");//add pads to elementgst_element_add_pad(GST_ELEMENT(filter),filter->srcpad);gst_element_add_pad(GST_ELEMENT(filter), filter->sinkpad);...
}



结论

至此,结合上下文,我们已经可以知道 pad 是如何在继承体系中存在的了。

pad仅仅由GstElement使用,所有继承自GstElement 的 element 仅仅是为 GstElement 提供了自定义的 pad实例,同时参与 pad的生命周期管理(fixme if wrong)。

所以,即便 自定义 element 不创建自己的 pad,那么只要其 继承树里某个父类 element 在 init 函数里通过 add pad 函数向 根节点 GstElement里添加过 pad ,那么这个pad就会作为总代理。

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

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

相关文章

51打开工程,发现cannot read project file问题

解决办法: 1、路径太长,可以在桌面重新创建一个文件夹,把所有的文件都移动到新的文件夹中 2、创建工程的时候,都勾选上,就没问题

OracleDG原理

一、DataGuard架构介绍 1、基本介绍 在DG环境中,至少会有两个数据库,一个数据库处于Open状态,对外提供服务,这个数据库叫做primary Database。第二个数据库处于恢复状态,叫做Standby Database。运行时Primay Databas…

vue从入门到精通(四):MVVM模型

一,MVVM MVVM(Model–view–viewmodel)是一种软件架构模式。MVVM有助于将图形用户界面的开发与业务逻辑或后端逻辑(数据模型)的开发分离开来。详见MVVM 二,Vue中的MVVM Vue虽然没有完全遵循 MVVM 模型,但是 Vue 的设…

手机相册怎么恢复?如何挽救误删的照片?

手机相册成为我们存储和分享照片的主要途径,这些照片可能代表着我们的美好回忆、重要时刻或珍贵的瞬间。然而,有时候我们会因为误操作或一时冲动,将一些珍贵的照片误删,并且很难找回来。手机相册怎么恢复呢?本文将为你…

基于Tensorflow+Keras的卷积神经网络(CNN)人脸识别

欢迎大家点赞、收藏、关注、评论啦 ,由于篇幅有限,只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景与意义 人脸识别是计算机视觉领域的一个重要研究方向,广泛应用于安全监控、身份验证、人机…

Android 性能为王时代SparseArray和HashMap一争高下

文章目录 一、SparseArray 源码分析1. **类定义和构造函数**2. **基本方法**2.1 put(int key, E value)2.2 get(int key)2.3 delete(int key)2.4 removeAt(int index)2.5 gc()2.6 size()2.7 keyAt(int index) 和 valueAt(int index) 3. **辅助方法**3.1 binarySearch() 二、使用…

最新AI智能问答创作系统ChatGPT网站源码V4.20版本,GPTs、AI绘画、AI换脸、垫图混图+(GoMaxAI系统搭建部署教程文档)

Midjourney(人工智能图像生成器),仅用了一年时间就已经火爆全球。它就像是一个想象力的助力器,总能带给人们一些不可思议的奇妙的作品。Midjourney虽然是一款最强的AI绘画工具,但对于很多人来说Midjourney怎么用&#…

使用JavaScript日历小部件和DHTMLX Gantt的应用场景(三)

DHTMLX Suite UI 组件库允许您更快地构建跨平台、跨浏览器 Web 和移动应用程序。它包括一组丰富的即用式 HTML5 组件,这些组件可以轻松组合到单个应用程序界面中。 DHTMLX Gantt是用于跨浏览器和跨平台应用程序的功能齐全的Gantt图表,可满足项目管理应用…

匝间冲击耐压试验仪产品介绍及工作原理

产品简介 武汉凯迪正大KD2684S匝间冲击耐压试验仪适用于电机、变压器、电器线圈等这些由漆包线绕制的产品。因漆包线的绝缘涂敷层本身存在着质量问题,以及在绕线、嵌线、刮线、接头端部整形、绝缘浸漆、装配等工序工艺中不慎而引起绝缘层的损伤等,都会造…

在线投票系统源码 网上投票平台创建 安全稳定 支持自定义投票规则+礼物道具功能

分享一款在线投票系统源码,是一款功能丰富、安全稳定的网络投票平台解决方案。通过本源码,用户可以轻松创建并管理各种在线投票活动,支持自定义投票规则,同时集成礼物道具功能,增强用户参与度和投票活动的趣味性&#…

2024 年 电工杯(B题)大学生数学建模挑战赛 | 平衡膳食食谱 | 数学建模完整代码+建模过程全解全析

当大家面临着复杂的数学建模问题时,你是否曾经感到茫然无措?作为2022年美国大学生数学建模比赛的O奖得主,我为大家提供了一套优秀的解题思路,让你轻松应对各种难题。 CS团队倾注了大量时间和心血,深入挖掘解决方案。通…

常用的框架——— Android UtilCode

AndroidUtilCode是一个功能强大且易于使用的Android库。该库封装了Android开发中经常使用的具备完整演示和单元测试的功能。经过使用其封装的API,能够大大提升开发效率。该程序主要由两个模块组成,utilcode(一般在开发中使用)和su…

【真人Q版手办风】线稿手绘+ AI绘图 Stable Diffusion 完整制作过程分享

大家好,我是设计师阿威。 今天给大家分享一篇【真人Q版卡通手办】风格的制作过程,话不多说,进入正题。 成品预览 手绘线稿 首先,我使用的是老款手绘软件【SAI】,用[钢笔工具]进行了人物的线稿Q版描绘。&#x1f447…

ArcEngine/ArcObject转换影像解决偏色的问题IRasterLayer

IRasterLayer里面的Renderer使用RasterRGBRendererClass,StretchType设置为NONE。以下是我将图像转换为img格式,采用RLE无损压缩。 IRasterLayer rasterLayer new RasterLayerClass(); rasterLayer.CreateFromFilePath(strFromName); IRaster pRaster …

巨控无线通讯模块在煤化工皮带保护系统中的应用

一、项目介绍 近年来由于煤矿化工行业在实际生产过程中事故频发,国家安监部门自2022年起开展了为其三年的专项整治行动,皮带运输系统作为煤矿化工行业自动化系统的重要组成部分成为此次专项整治重点。 兖矿新疆能化有限公司作为山能集团在新疆分部的龙…

AWS计算之Lambda

AWS Lambda是亚马逊提供的一项无服务器(Serveless)计算服务,它允许开发人员在无需管理服务器的情况下运行代码。Lambda可以自动扩展以处理任何规模的请求,并且只会收取实际执行代码的费用,而不会收取任何预定或固定费用。Lambda 函数在托管的…

如何为海量计数场景设计缓存体系?no.34

计数常规方案 计数服务在互联网系统中非常常见,用户的关注粉丝数、帖子数、评论数等都需要进行计数存储。计数的存储格式也很简单,key 一般是用户 uid 或者帖子 id 加上后缀,value 一般是 8 字节的 long 型整数。 最常见的计数方案是采用缓存…

常见的100个Shell命令,超级实用!

在大多数的Linux和Unix系统、及其他类Unix系统中,Shell是用户与操作系统内核交互的主要方式。作为一种强大的命令行解释器,它也支持编程功能,用户可以写脚本来处理各种任务。 熟悉shell脚本,首先要对shell指令熟悉,今…

Vue3学习-用 vite@latest 初始化项目后,遇到无法识别 .vue 文件

引入app界面遇到 我的解决方案 1.根目录创建 env.d.ts&#xff0c;添加 declare module "*.vue" {import type { DefineComponent } from "vue"const vueComponent: DefineComponent<{}, {}, any>export default vueComponent }2.在 tsconfig.json…

基于ARM|DSP+FPGA+NVIDIA AI平台的摄像头ISP图像画质调试定制服务

基本框架及算法介绍 ISP(Image Signal Processor)&#xff0c;即图像处理&#xff0c;主要作用是对前端图像传感器输出的信号做后期处理&#xff0c;主要功能有线性纠正、噪声去除、坏点去除、内插、白平衡、自动曝光控制等&#xff0c;依赖于ISP才能在不同的光学条件下都…