ASOC调用过程
上一篇文章我們將了嵌入式系統(tǒng)注冊聲卡的過程:https://blog.csdn.net/qq_37659294/article/details/104748747
這篇文章我們以打開一個聲卡的播放節(jié)點為例,講解一下在APP調(diào)用open時,最終會如何調(diào)用到硬件相關的函數(shù)。
?
在上一篇文章最后我們說過,當應用程序open設備文件/dev/snd/pcmCxDxp(controlC0和pcmCxDxc也類似,這里我們以pcmCxDxp為例)時,會進入snd_fops的open回調(diào)函數(shù),該open函數(shù)以次設備號為索引,從snd_minors全局數(shù)組中取出當初注冊conrol、pcm設備時填充的snd_minor結(jié)構(gòu)體,然后從snd_minor結(jié)構(gòu)中取出control、pcm設備的f_ops,并且把file->f_op替換為pcm設備的f_ops,緊接著調(diào)用pcm設備的f_ops->open()(也就是snd_pcm_f_ops[0]的snd_pcm_playback_open)。后面我們根據(jù)文件句柄操作這個設備節(jié)點時,就都是調(diào)用這個f_ops里的函數(shù)。
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]; //根據(jù)次設備號找到對應的snd_minor結(jié)構(gòu)體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) { //調(diào)用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; }真正和硬件相關的函數(shù)是我們在platform和codec部分放入對應鏈表且后來又在machine部分根據(jù)dai_link實例化的那四個結(jié)構(gòu)體(s3c24xx_i2s_dai、samsung_asoc_platform、uda134x_dai、soc_codec_dev_uda134x)里的函數(shù),所以snd_pcm_f_ops[0]的snd_pcm_playback_open函數(shù)又會最終調(diào)用到這些硬件相關的操作,下面是調(diào)用的過程
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結(jié)構(gòu)體soc_pcm_open依次調(diào)用cpu_dai, dma, codec_dai, machine的open或startup函數(shù),如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函數(shù)里有一個(snd_pcm)pcm->private_data = rtd;后面我們APP調(diào)用程序的時候會從pcm->private_data取出rtd(rtd里有對應我們開發(fā)板的cpu_dai、codec_dai...)。
現(xiàn)在在snd_pcm_attach_substream函數(shù)里,我們就把這個rtd取出來,放到了substream->private_data。
②在《ASOC注冊過程》這篇文章的第⑥點,內(nèi)核給card->rtd->ops結(jié)構(gòu)體里的函數(shù)指針賦值并把card->rtd->ops的地址賦給substream->ops。
現(xiàn)在我們調(diào)用substream->ops->open(substream),也就是調(diào)用snd_pcm_ops結(jié)構(gòu)體里的soc_pcm_open函數(shù)。
③在前面第①點中,我們已經(jīng)把rtd(rtd里有對應我們開發(fā)板的cpu_dai、codec_dai...)取出來,放到了substream->private_data,在soc_pcm_open函數(shù)里我們通過substream->private_data調(diào)用硬件相關的函數(shù),如cpu_dai, dma, codec_dai的open或startup函數(shù)。
?
總結(jié)
- 上一篇: LOL显示不全怎么办
- 下一篇: 编写声卡驱动(框架)