alsa声卡dev_snd_pcmC0D0p的open打开流程 下载本文

alsa声卡/dev/snd/pcmC0D0p的open打开流程

原文地址:http://blog.chinaunix.net/space.php?uid=20564848&do=blog&cuid=2116725 aplay.c ==> main

==> snd_pcm_open(&handle, pcm_name, stream, open_mode); // 打开一路pcm,刷新config配置 如果是\同时type等于SND_CONFIG_TYPE_COMPOUND那么这里对应\static const char *const build_in_pcms[] = {

\

\ \ NULL };

_snd_pcm_empty_open和snd_pcm_open_named_slave

==> snd_pcm_open_conf(pcmp, name, root, conf, stream, mode);

==> open_func = snd_dlobj_cache_lookup(open_name);将获得lib库中_snd_pcm_empty_open函数 所以open_func将等于_snd_pcm_empty_open

_snd_pcm_empty_open _snd_pcm_asym_open

_snd_pcm_plug_open _snd_pcm_softvol_open _snd_pcm_dmix_open _snd_pcm_hw_open

==> snd_pcm_hw_open(pcmp, name, card, device, subdevice, stream, mode | (nonblock ? SND_PCM_NONBLOCK : 0), 0, sync_ptr_ioctl);

==> snd_ctl_hw_open

filename等于\==> snd_open_device(filename, fmode); ctl->ops = &snd_ctl_hw_ops; ctl->private_data = hw; ctl->poll_fd = fd; *handle = ctl;

filename等于\

==> fd = snd_open_device(filename, fmode);

==> return snd_pcm_hw_open_fd(pcmp, name, fd, 0, sync_ptr_ioctl);

==> snd_pcm_new(&pcm, SND_PCM_TYPE_HW, name, info.stream, mode); pcm->ops = &snd_pcm_hw_ops;

pcm->fast_ops = &snd_pcm_hw_fast_ops;

static int snd_pcm_hw_mmap_control(snd_pcm_t *pcm) {

snd_pcm_hw_t *hw = pcm->private_data; void *ptr; int err;

if (hw->sync_ptr == NULL) { // 如果还没有mmap,那么执行mmap映射内核空间驱动使用的声音缓冲区 ptr = mmap(NULL, page_align(sizeof(struct sndrv_pcm_mmap_control)), PROT_READ|PROT_WRITE, MAP_FILE|MAP_SHARED, hw->fd, SNDRV_PCM_MMAP_OFFSET_CONTROL); if (ptr == MAP_FAILED || ptr == NULL) { err = -errno;

SYSMSG(\

return err; }

hw->mmap_control = ptr; // 声卡驱动头部填充了一个结构体sndrv_pcm_mmap_control,类似qvfb显示原理.

// struct sndrv_pcm_mmap_control {

// sndrv_pcm_uframes_t appl_ptr; /* RW: appl ptr (0...boundary-1) */

// sndrv_pcm_uframes_t avail_min; /* RW: min available frames for wakeup */ // };

} else {

hw->mmap_control->avail_min = 1; }

snd_pcm_set_appl_ptr(pcm, &hw->mmap_control->appl_ptr, hw->fd, SNDRV_PCM_MMAP_OFFSET_CONTROL); return 0; }

snd_pcm_mmap

switch (i->type) {

case SND_PCM_AREA_MMAP: // 表示为数据区分配驱动内存,在snd_pcm_hw_channel_info中设置了type

ptr = mmap(NULL, size, PROT_READ|PROT_WRITE, MAP_FILE|MAP_SHARED, i->u.mmap.fd, i->u.mmap.offset); /*

mmap

==> snd_pcm_mmap_data ==> snd_pcm_default_mmap // mmap the DMA buffer on RAM

static int snd_pcm_default_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *area)

{

area->vm_ops = &snd_pcm_vm_ops_data; // vma操作函数,当应用程序向该area读写不存在的内存数据时,

area->vm_private_data = substream; // 将执行snd_pcm_vm_ops_data中的fault

// 函数snd_pcm_mmap_data_fault进一步以页为单位申请内存空间,所以如果用户程序需要64k,那么将执行16次,每次申请4k空间[luther.gliethttp]. area->vm_flags |= VM_RESERVED; atomic_inc(&substream->mmap_count); return 0; } */

if (ptr == MAP_FAILED) { SYSERR(\

return -errno; }

i->addr = ptr;

==> snd_pcm_mmap_control

static int snd_pcm_mmap_control(struct snd_pcm_substream *substream, struct file *file, struct vm_area_struct *area) {

struct snd_pcm_runtime *runtime; long size;

if (!(area->vm_flags & VM_READ)) return -EINVAL;

runtime = substream->runtime;

size = area->vm_end - area->vm_start;

if (size != PAGE_ALIGN(sizeof(struct snd_pcm_mmap_control))) return -EINVAL;

area->vm_ops = &snd_pcm_vm_ops_control; // 当对( area->vm_start,area->vm_end)之间空间操作,发生

area->vm_private_data = substream; // 缺页时,内核将调用该vm_ops方法来处理fault异常, area->vm_flags |= VM_RESERVED; // 进而执行snd_pcm_mmap_control_fault申请1个page空间

return 0; }

==> writei_func = snd_pcm_writei; ==> playback(argv[optind++]);

==> playback_go(fd, dtawave, pbrec_count, FORMAT_WAVE, name);