ASOC调用过程

上一篇文章我们将了嵌入式系统注册声卡的过程:https://blog.csdn.net/qq_37659294/article/details/104748747

这篇文章我们以打开一个声卡的播放节点为例,讲解一下在APP调用open时,最终会如何调用到硬件相关的函数。

 

在上一篇文章最后我们说过,当应用程序open设备文件/dev/snd/pcmCxDxp(controlC0和pcmCxDxc也类似,这里我们以pcmCxDxp为例)时,会进入snd_fops的open回调函数,该open函数以次设备号为索引,从snd_minors全局数组中取出当初注册conrol、pcm设备时填充的snd_minor结构体,然后从snd_minor结构中取出control、pcm设备的f_ops,并且把file->f_op替换为pcm设备的f_ops,紧接着调用pcm设备的f_ops->open()(也就是snd_pcm_f_ops[0]的snd_pcm_playback_open)。后面我们根据文件句柄操作这个设备节点时,就都是调用这个f_ops里的函数。

static const struct file_operations snd_fops =
{.owner =	THIS_MODULE,.open =		snd_open,.llseek =	noop_llseek,
};static int snd_open(struct inode *inode, struct file *file)
{unsigned int minor = iminor(inode);struct snd_minor *mptr = NULL;const struct file_operations *old_fops;int err = 0;if (minor >= ARRAY_SIZE(snd_minors))return -ENODEV;mutex_lock(&sound_mutex);mptr = snd_minors[minor];    //根据次设备号找到对应的snd_minor结构体if (mptr == NULL) {mptr = autoload_device(minor);if (!mptr) {mutex_unlock(&sound_mutex);return -ENODEV;}}old_fops = file->f_op;file->f_op = fops_get(mptr->f_ops);   //把file->f_op替换为pcm设备的f_ops if (file->f_op == NULL) {file->f_op = old_fops;err = -ENODEV;}mutex_unlock(&sound_mutex);if (err < 0)return err;if (file->f_op->open) {            //调用pcm设备的f_ops->open()err = file->f_op->open(inode, file);if (err) {fops_put(file->f_op);file->f_op = fops_get(old_fops);}}fops_put(old_fops);return err;
}

真正和硬件相关的函数是我们在platform和codec部分放入对应链表且后来又在machine部分根据dai_link实例化的那四个结构体(s3c24xx_i2s_dai、samsung_asoc_platform、uda134x_dai、soc_codec_dev_uda134x)里的函数,所以snd_pcm_f_ops[0]的snd_pcm_playback_open函数又会最终调用到这些硬件相关的操作,下面是调用的过程

2. /dev/snd/pcmC0D0p 对应的file_operations是snd_pcm_f_ops[0]
open :  snd_pcm_playback_opensnd_pcm_opensnd_pcm_open_filestruct snd_pcm_substream *substream;snd_pcm_open_substreamsnd_pcm_attach_substreamsubstream->private_data = pcm->private_data;err = snd_pcm_hw_constraints_init(substream);snd_mask_anysnd_interval_any......err = substream->ops->open(substream) // substream->ops : snd_pcm_ops结构体soc_pcm_open依次调用cpu_dai, dma, codec_dai, machine的open或startup函数,如uda134x_startup、dma_openstruct snd_soc_pcm_runtime *rtd = substream->private_data;struct snd_soc_dai *cpu_dai = rtd->cpu_dai;ret = cpu_dai->driver->ops->startup(substream, cpu_dai);...

①在《ASOC注册过程》这篇文章的第⑦点中我们提到一点:soc_new_pcm函数里有一个(snd_pcm)pcm->private_data = rtd;后面我们APP调用程序的时候会从pcm->private_data取出rtd(rtd里有对应我们开发板的cpu_dai、codec_dai...)。

现在在snd_pcm_attach_substream函数里,我们就把这个rtd取出来,放到了substream->private_data。

②在《ASOC注册过程》这篇文章的第⑥点,内核给card->rtd->ops结构体里的函数指针赋值并把card->rtd->ops的地址赋给substream->ops。

现在我们调用substream->ops->open(substream),也就是调用snd_pcm_ops结构体里的soc_pcm_open函数。

③在前面第①点中,我们已经把rtd(rtd里有对应我们开发板的cpu_dai、codec_dai...)取出来,放到了substream->private_data,在soc_pcm_open函数里我们通过substream->private_data调用硬件相关的函数,如cpu_dai, dma, codec_dai的open或startup函数。

 

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

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

相关文章

编写声卡驱动(框架)

在前面两篇文章中&#xff0c;我们分别讲了嵌入式Linux系统声卡注册的过程和调用的过程&#xff1a; https://blog.csdn.net/qq_37659294/article/details/104748747 https://blog.csdn.net/qq_37659294/article/details/104802868 讲了那么多&#xff0c;我们最终的目的无非…

声卡学习笔记

分享几篇关于韦东山声卡驱动的学习笔记&#xff0c;作者写得非常详细。 ALSA驱动框架&#xff1a;https://blog.csdn.net/qingkongyeyue/article/details/52328991 ASoC驱动框架&#xff1a;https://blog.csdn.net/qingkongyeyue/article/details/52349120 ASoC驱动重要结构…

路由器、交换机、集线器的区别

https://blog.csdn.net/weibo1230123/article/details/82779040

$PATH环境变量的作用

echo $PATH 显示当前PATH环境变量&#xff0c;该变量的值由一系列以冒号分隔的目录名组成&#xff0c;如&#xff1a;/usr/local/bin:/bin:/usr/bin。(冒号:是路径分隔符) 在执行一个程序的时候如果没有PATH的话&#xff0c;就需要写出路径名&#xff08;绝对或者相对&#xf…

dmesg

https://blog.csdn.net/zm_21/article/details/31760569

进程上下文与中断上下文的理解

一.什么是内核态和用户态 内核态&#xff1a;在内核空间执行&#xff0c;通常是驱动程序&#xff0c;中断相关程序&#xff0c;内核调度程序&#xff0c;内存管理及其操作程序。 用户态&#xff1a;用户程序运行空间。 二.什么是进程上下文与中断上下文 1.进程上下文&#xf…

GDB调试教程:1小时玩转Linux gdb命令

原文链接&#xff1a;http://c.biancheng.net/gdb/ GDB 入门教程 本教程以下面的代码为例&#xff0c;在 Linux 系统下来讲解 GBD 的调试流程&#xff1a; int main (void) {unsigned long long int n, sum;n 1;sum 0;while (n < 100){sum sum n;n n 1;}return 0; …

shell将命令执行的结果赋值给 变量

https://blog.csdn.net/lemontree1945/article/details/79126819/

Linux下shell脚本指定程序运行时长

https://www.cnblogs.com/yychuyu/p/12626798.html

vim编辑器如何删除一行或者多行内容

http://blog.itpub.net/69955379/viewspace-2681334/

C++经典问题:如果对象A中有对象成员B,对象B没有默认构造函数,那么对象A必须在初始化列表中初始化对象B?

对象成员特点总结&#xff1a; &#xff08;1&#xff09;实例化对象A时&#xff0c;如果对象A有对象成员B,那么先执行对象B的构造函数&#xff0c;再执行A的构造函数。 &#xff08;2&#xff09;如果对象A中有对象成员B,那么销毁对象A时&#xff0c;先执行对象A的析构函数&…

JZ2440用U-Boot给Nand-Flash烧写程序时报错:NAND write: incorrect device type in bootloader ‘bootloader‘ is not

JZ2440开发板使用问题&#xff0c;U-Boot烧写程序到Nand Flash时报错&#xff1a;NAND write: incorrect device type in bootloader bootloader is not a number 这是因为分区名中u-boot&#xff0c;不是bootloader&#xff0c;而cmd_menu.c里用的是bootloader 可以执行&#…

韦东山衔接班——4.4_构建根文件系统之构建根文件系统

文章地址&#xff1a; https://blog.csdn.net/gongweidi/article/details/100086289?biz_id102&utm_term%E9%9F%A6%E4%B8%9C%E5%B1%B1%E8%A1%94%E6%8E%A5%E7%8F%AD&utm_mediumdistribute.pc_search_result.none-task-blog-2~blog~sobaiduweb~default-5-100086289&…

C++中const char *p和char const *p

const char *p;他的意思是p指向的目标空间的内容不可变化 例如定义char cA; p&c;则c的内容不可以变化.如cB;等一些企图改变变量c的值的做法都不行. 然而p仍然是动态的,就是它还可以指向别的空间,被赋予新的地址值,只是被他指向的目标空间的内容不可变化,如上面的c值始终为A…

qt 分割字符串的两种方法

https://blog.csdn.net/a724699769/article/details/62216435

【YOLO系列】YOLOv3代码详解(五):utils.py脚本

前言 以下内容仅为个人在学习人工智能中所记录的笔记&#xff0c;先将目标识别算法yolo系列的整理出来分享给大家&#xff0c;供大家学习参考。 本文仅对YOLOV3代码中关键部分进行了注释&#xff0c;未掌握基础代码的铁汁可以自己百度一下。 若文中内容有误&#xff0c;希望大家…

内核的Makefile与Kconfig关系解析

在子目录下的Kconfig里添加make menuconfig的选项&#xff08;如图一&#xff09;&#xff0c;并默认设置为y&#xff0c;make menuconfig的菜单里就会有该项并默认为选上状态&#xff0c;make menuconfig配置完之后在.config文件里就有该选项&#xff0c;并等于y&#xff08;如…

C语言extern的用法

在x.c文件里定义如&#xff1a;int x 100; 在x.h文件里声明如&#xff1a;extern int x; 然后在main.c里 #include "x.h"即可 或者直接在main.c里 extern int x; 而不使用#include "x.h"&#xff08;此时x.h里当然也不用extern int x;&#xff09;也…

C语言中.和->区别

结构体变量用 . 运算符来访问结构体的成员 struct A { int a; int b; };A object; object.a 1;指向结构体的指针用->来访问其指向的结构体的成员 A *point malloc(sizeof(struct A)); point->a 1;

Qt中定时器使用的两种方法

https://blog.csdn.net/qq_28877125/article/details/88389559