ALSA驱动、设备函数调用关系

来源:互联网 发布:java 注解 编辑:程序博客网 时间:2024/06/05 04:36

 soc_probe
==> snd_soc_register_card或者snd_soc_register_dai或者snd_soc_register_platform或者snd_soc_register_codec
==> snd_soc_instantiate_cards
==> snd_soc_instantiate_card
==> codec_dev->probe(uda134x_soc_probe() in uda134x.c)

==> snd_soc_new_pcms
==> soc_new_pcm

      --1>platform->pcm_new(s3c_dma_new() in s3c-dma.c) -> s3c_preallocate_dma_buffer: alloc DMA buffer for playback or capture
      --2>snd_pcm_new

snd_soc_new_pcms
snd_card_create
snd_ctl_create
snd_ctl_dev_register(in sound/core/control.c)创建"controlC%i"声卡对应的控制节点,
他的fops为snd_ctl_f_ops

snd_pcm_dev_register
snd_register_device_for_dev(in sound/core/pcm.c)
创建"pcmC%iD%ip"或者"pcmC%iD%ic"设备节点,它的fops为snd_pcm_f_ops[2]

snd_pcm_f_ops[2]
==> snd_pcm_playback_open或者snd_pcm_capture_open
==> snd_pcm_open或者oss的snd_pcm_oss_f_reg.snd_pcm_oss_open,这就对应/dev/dsp了
==> snd_pcm_open_file或者oss的snd_pcm_oss_open_file
==> snd_pcm_open_substream
==> soc_pcm_open (soc-core.c)

   -->cpu_dai->ops->startup (in s3c24xx-i2s.c, it's not implemented)

   -->platform->pcm_ops->open(s3c_dma_open in s3c-dma.c)

   -->codec_dai->ops->startup(uda134x_startup() in uda1343.c)

   -->machine->ops->startup(s3c24xx_uda134x_startup() in s3c24xx_uda134x.c)

/* ASoC PCM operations */
static struct snd_pcm_ops soc_pcm_ops = {
    .open        = soc_pcm_open,
    .close        = soc_codec_close,
    .hw_params    = soc_pcm_hw_params,
    .hw_free    = soc_pcm_hw_free,
    .prepare    = soc_pcm_prepare,
    .trigger    = soc_pcm_trigger,
};
==> soc_pcm_open
cpu_dai->ops->startup       // 无
platform->pcm_ops->open    

codec_dai->ops->startup     // uda134x_startup 
card_dai_link->ops->startup // s3c24xx_uda134x_startup 获取clock时钟源

// cpu_dai单元
struct snd_soc_dai s3c24xx_i2s_dai = {
    .name = "s3c24xx-i2s",
    .id = 0,
    .probe = s3c24xx_i2s_probe,
    .suspend = s3c24xx_i2s_suspend,
    .resume = s3c24xx_i2s_resume,
    .playback = {
        .channels_min = 2,
        .channels_max = 2,
        .rates = S3C24XX_I2S_RATES,
        .formats = SNDRV_PCM_FMTBIT_S8 | SNDRV_PCM_FMTBIT_S16_LE,},
    .capture = {
        .channels_min = 2,
        .channels_max = 2,
        .rates = S3C24XX_I2S_RATES,
        .formats = SNDRV_PCM_FMTBIT_S8 | SNDRV_PCM_FMTBIT_S16_LE,},
    .ops = &s3c24xx_i2s_dai_ops,
};
//platform单元
struct snd_soc_platform s3c24xx_soc_platform = {
    .name        = "s3c24xx-audio",
    .pcm_ops     = &s3c_dma_ops,
    .pcm_new    = s3c_dma_new,
    .pcm_free    = s3c_dma_free_dma_buffers,
};

static struct snd_pcm_ops s3c_dma_ops = {
    .open        = s3c_dma_open,
    .close        = s3c_dma_close,
    .ioctl        = snd_pcm_lib_ioctl,
    .hw_params    = s3c_dma_hw_params,
    .hw_free    = s3c_dma_hw_free,
    .prepare    = s3c_dma_prepare,
    .trigger    = s3c_dma_trigger,
    .pointer    = s3c_dma_pointer,
    .mmap        = s3c_dma_mmap,
};
// codec_dai单元
struct snd_soc_dai uda134x_dai = {
    .name = "UDA134X",
    /* playback capabilities */
    .playback = {
        .stream_name = "Playback",
        .channels_min = 1,
        .channels_max = 2,
        .rates = UDA134X_RATES,
        .formats = UDA134X_FORMATS,
    },
    /* capture capabilities */
    .capture = {
        .stream_name = "Capture",
        .channels_min = 1,
        .channels_max = 2,
        .rates = UDA134X_RATES,
        .formats = UDA134X_FORMATS,
    },
    /* pcm operations */
    .ops = &uda134x_dai_ops,
};

//card_dai_link单元
static struct snd_soc_dai_link s3c24xx_uda134x_dai_link = {
    .name = "UDA134X",
    .stream_name = "UDA134X",
    .codec_dai = &uda134x_dai,
    .cpu_dai = &s3c24xx_i2s_dai,
    .ops = &s3c24xx_uda134x_ops,

};



Pcm.c

snd_pcm_control_ioctl()

case SNDRV_CTL_IOCTL_PCM_INFO:

*  S1:get params infofrom userspace arg*

*  S2:get info->device

*  S3:get info->stream

*  S4:get info->subdevice

*  S5:get pcm from matching special card and device

 

*  S1:get pstrfrom &pcm->streams[stream]

*  S2:get substreamfrom pstr->substream

*  S3:Now! we got struct snd_pcm_substream*substream

*        which contents the struct snd_pcm_runtime*runtime;

 

snd_pcm_runtime里面有

       /* -- hardware description -- */

       struct snd_pcm_hardware hw;

       struct snd_pcm_hw_constraintshw_constraints;

 

 

xx_pcm.c

xx_pcm_hardware

xx_pcm_open()

xx_pcm_ops

xx_soc_platform

snd_soc_register_platform()

xx_soc_platform_init()

module_init