ASOC 驱动架构

前言

针对嵌入式设备,linux在ALSA的基础上又封装了一层,名为ASOC(ALSA system on chip)。使用ASOC就不需要自己调用snd_card_create、snd_ctrl_create等函数自己来创建声卡。ASOC将驱动程序分为3部分:

1)machine:单板相关的内容。抽象出一个snd_soc_card结构体 ,重点关注snd_soc_dai_link结构体 ,此结构体表明以下两点:

a. 表明platform是哪一个, CPU DAI是哪一个,DMA是哪一个,即哪个DMA负责数据传输。

b. 表明codec是哪一个,codec DAI是哪一个

2)platform

a. DAI:抽象出snd_soc_dai_driver结构体

b. DMA:抽象出snd_soc_platform_driver结构体 

3)codec

a. DAI:抽象出一个snd_soc_dai_driver结构体 

b. 控制接口:抽象出一个snd_soc_codec_driver结构体。

在Linux驱动中,肯定有多个platform和codec。对于某个单板,使用哪款主芯片(platform),哪款codec芯片在machine中指定。

内核中带有UDA1341的驱动程序,但是没有WM8976的驱动程序。首先分析UDA1341的驱动程序

1. machine

/sound/soc/samsung/S3c24xx_uda134x.c

static struct platform_driver s3c24xx_uda134x_driver = {
    .probe  = s3c24xx_uda134x_probe,
    .remove = s3c24xx_uda134x_remove,
    .driver = {
        .name = "s3c24xx_uda134x",
        .owner = THIS_MODULE,
    },
};

有平台driver,肯定有同名的平台device(这是一种老的方式,现在都使用设备树了),搜索s3c24xx_uda134x

arch/arm/mach-s3c24xx/Mach-mini2440.c

static struct platform_device mini2440_audio = {
    .name        = "s3c24xx_uda134x",
    .id        = 0,
    .dev        = {
        .platform_data    = &mini2440_audio_pins,
    },
};

当内核中有同名的平台driver和平台device时,platform_driver 中的probe函数将会调用。

static int s3c24xx_uda134x_probe(struct platform_device *pdev)
{
    int ret;

    printk(KERN_INFO "S3C24XX_UDA134X SoC Audio driver
");
    
    ......
    s3c24xx_uda134x_snd_device = platform_device_alloc("soc-audio", -1);
    
    platform_set_drvdata(s3c24xx_uda134x_snd_device,
                 &snd_soc_s3c24xx_uda134x);
    platform_device_add_data(s3c24xx_uda134x_snd_device, &s3c24xx_uda134x, sizeof(s3c24xx_uda134x));
    ret = platform_device_add(s3c24xx_uda134x_snd_device);
    
    return ret;
}

又出现了平台device soc-audio,在内核中搜索soc-audio

/sound/soc/Soc-core.c

/* ASoC platform driver */
static struct platform_driver soc_driver = {
    .driver        = {
        .name        = "soc-audio",
        .owner        = THIS_MODULE,
        .pm        = &snd_soc_pm_ops,
    },
    .probe        = soc_probe,
    .remove        = soc_remove,
};

soc_probe函数将被调用

/* probes a new socdev */
static int soc_probe(struct platform_device *pdev)
{
    struct snd_soc_card *card = platform_get_drvdata(pdev);
    int ret = 0;
    ........
    card->dev = &pdev->dev;

    ret = snd_soc_register_card(card);

    return 0;
}

注册一个snd_soc_card的结构体。该结构体具体指哪一个?就是snd_soc_s3c24xx_uda134x

/sound/soc/samsung/S3c24xx_uda134x.c

static struct snd_soc_card snd_soc_s3c24xx_uda134x = {
    .name = "S3C24XX_UDA134X",
    .owner = THIS_MODULE,
    .dai_link = &s3c24xx_uda134x_dai_link,
    .num_links = 1,
};
static struct snd_soc_dai_link s3c24xx_uda134x_dai_link = {
    .name = "UDA134X",
    .stream_name = "UDA134X",
    .codec_name = "uda134x-codec",//用哪一个codec
    .codec_dai_name = "uda134x-hifi",//codec芯片中的哪一个DAI,因为有些codec中有多个数字接口
    .cpu_dai_name = "s3c24xx-iis",//2440的DAI接口
    .ops = &s3c24xx_uda134x_ops,
    .platform_name    = "samsung-audio",//DMA
};
static struct snd_soc_ops s3c24xx_uda134x_ops = {
    .startup = s3c24xx_uda134x_startup,
    .shutdown = s3c24xx_uda134x_shutdown,
    .hw_params = s3c24xx_uda134x_hw_params,
};

machine部分到此结束,它主要就是构造了一个snd_soc_card结构体 。在该结构体中有一个dai_link,在dai_link中指定了codec_name,code_dai_name,cpu_dai_name以及platform_name.

2. platform

2.1 cpu dai

搜索s3c24xx-iis

/sound/soc/samsung/S3c24xx-i2s.c

static struct platform_driver s3c24xx_iis_driver = {
    .probe  = s3c24xx_iis_dev_probe,
    .remove = __devexit_p(s3c24xx_iis_dev_remove),
    .driver = {
        .name = "s3c24xx-iis",
        .owner = THIS_MODULE,
    },
};

有平台driver,必然有同名的平台device

/arch/arm/plat-samsung/Devs.c

struct platform_device s3c_device_iis = {
    .name        = "s3c24xx-iis",
    .id        = -1,
    .num_resources    = ARRAY_SIZE(s3c_iis_resource),
    .resource    = s3c_iis_resource,
    .dev        = {
        .dma_mask        = &samsung_device_dma_mask,
        .coherent_dma_mask    = DMA_BIT_MASK(32),
    }
};

因此函数s3c24xx_iis_dev_probe将被调用,

static __devinit int s3c24xx_iis_dev_probe(struct platform_device *pdev)
{
    return snd_soc_register_dai(&pdev->dev, &s3c24xx_i2s_dai);
}
static struct snd_soc_dai_driver s3c24xx_i2s_dai = {
    .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中最重要的cpu DAI接口——> snd_soc_dai_driver 就出现了

static const struct snd_soc_dai_ops s3c24xx_i2s_dai_ops = {
    .trigger    = s3c24xx_i2s_trigger,
    .hw_params    = s3c24xx_i2s_hw_params,
    .set_fmt    = s3c24xx_i2s_set_fmt,
    .set_clkdiv    = s3c24xx_i2s_set_clkdiv,
    .set_sysclk    = s3c24xx_i2s_set_sysclk,
};

2.2 DMA

根据结构体snd_soc_dai_link中的  platform_name = "samsung-audio",在代码中搜索samsung-audio

/sound/soc/samsung/Dma.c

static struct platform_driver asoc_dma_driver = {
    .driver = {
        .name = "samsung-audio",
        .owner = THIS_MODULE,
    },

    .probe = samsung_asoc_platform_probe,
    .remove = __devexit_p(samsung_asoc_platform_remove),
};

又出现了一个平台driver,要想这个平台driver起作用,必定会有一个平台device.

/arch/arm/plat-samsung/Devs.c

/* ASOC DMA */

struct platform_device samsung_asoc_dma = {
    .name        = "samsung-audio",
    .id        = -1,
    .dev        = {
        .dma_mask        = &samsung_device_dma_mask,
        .coherent_dma_mask    = DMA_BIT_MASK(32),
    }
};

/sound/soc/samsung/Dma.c

static int __devinit samsung_asoc_platform_probe(struct platform_device *pdev)
{
    return snd_soc_register_platform(&pdev->dev, &samsung_asoc_platform);
}
static struct snd_soc_platform_driver samsung_asoc_platform = {
    .ops        = &dma_ops,
    .pcm_new    = dma_new,
    .pcm_free    = dma_free_dma_buffers,
};
static struct snd_pcm_ops dma_ops = {
    .open        = dma_open,
    .close        = dma_close,
    .ioctl        = snd_pcm_lib_ioctl,
    .hw_params    = dma_hw_params,
    .hw_free    = dma_hw_free,
    .prepare    = dma_prepare,
    .trigger    = dma_trigger,
    .pointer    = dma_pointer,
    .mmap        = dma_mmap,
};

至此,platform的DMA这部分分析完毕,这部分主要就是注册了一个snd_soc_platform_driver的结构体。

3. codec

 搜索 uda134x-codec

/sound/soc/codecs/Uda134x.c

static struct platform_driver uda134x_codec_driver = {
    .driver = {
        .name = "uda134x-codec",
        .owner = THIS_MODULE,
    },
    .probe = uda134x_codec_probe,
    .remove = __devexit_p(uda134x_codec_remove),
};

内核中肯定存在名为uda134x-codec的platform device

/arch/arm/mach-s3c24xx/Mach-mini2440.c

static struct platform_device uda1340_codec = {
        .name = "uda134x-codec",
        .id = -1,
};

函数uda134x_codec_probe将被调用

static int __devinit uda134x_codec_probe(struct platform_device *pdev)
{
    return snd_soc_register_codec(&pdev->dev,
            &soc_codec_dev_uda134x, &uda134x_dai, 1);
}

3.1 soc_codec_dev_uda134x

这个结构体其实就是codec中的控制接口。

static struct snd_soc_codec_driver soc_codec_dev_uda134x = {
    .probe =        uda134x_soc_probe,
    .remove =       uda134x_soc_remove,
    .suspend =      uda134x_soc_suspend,
    .resume =       uda134x_soc_resume,
    .reg_cache_size = sizeof(uda134x_reg),
    .reg_word_size = sizeof(u8),
    .reg_cache_default = uda134x_reg,
    .reg_cache_step = 1,
    .read = uda134x_read_reg_cache,
    .write = uda134x_write,
    .set_bias_level = uda134x_set_bias_level,
};

3.2 uda134x_dai

static struct snd_soc_dai_driver uda134x_dai = {
    .name = "uda134x-hifi",//这个地方正好对应s3c24xx_uda134x_dai_link结构体中的codec_dai_name
/* 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, };

3.2.1 uda134x_dai_ops

static const struct snd_soc_dai_ops uda134x_dai_ops = {
    .startup    = uda134x_startup,
    .shutdown    = uda134x_shutdown,
    .hw_params    = uda134x_hw_params,
    .digital_mute    = uda134x_mute,
    .set_sysclk    = uda134x_set_dai_sysclk,
    .set_fmt    = uda134x_set_dai_fmt,
};

 4. ASOC如何与ALSA建立联系

4.1 platform

4.1.1 s3c244-i2s.c

把s3c24xx_i2s_dai放入了链表dai_list中。.name = "s3c24xx-iis"

s3c24xx_iis_dev_probe
    snd_soc_register_dai(&pdev->dev, &s3c24xx_i2s_dai)
        list_add(&dai->list, &dai_list);

4.1.2 /sound/soc/samsung/dma.c

把结构体samsung_asoc_platform放入了链表platform_list中。name = "samsung-audio",

samsung_asoc_platform_probe
    snd_soc_register_platform(&pdev->dev, &samsung_asoc_platform);
        list_add(&platform->list, &platform_list);

4.2 uda134x.c

.name = "uda134x-codec",

uda134x_codec_probe
    snd_soc_register_codec(&pdev->dev,&soc_codec_dev_uda134x, &uda134x_dai, 1);
        struct snd_soc_codec *codec;
        //分配一个snd_soc_codec结构体
        codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL);
        codec->driver = codec_drv; //codec_drv即soc_codec_dev_uda134x
        snd_soc_register_dais(dev, dai_drv, num_dai); //dai_drv即uda134x_dai
            //把结构体uda134x_dai放入了链表dai_list中。在链表dai_list中既有s3c24xx_i2s_dai,也有uda134x_dai
            list_add(&dai->list, &dai_list); 
        
        list_add(&codec->list, &codec_list); //把上面分配的codec放入链表codec_list中。

 它们之间是如何联系起来的,就是通过machine中的snd_soc_dai_link中的名字把它们联系起来。下面具体来分析

4.3 s3c24xx_uda134x_probe

s3c24xx_uda134x_probe
    //分配一个名为soc-audio的平台设备
    s3c24xx_uda134x_snd_device = platform_device_alloc("soc-audio", -1);
    //设置该设备的私有数据
    platform_set_drvdata(s3c24xx_uda134x_snd_device, &snd_soc_s3c24xx_uda134x);    
    //添加该设备,此时将导致soc-core.c中的soc_driver中的soc_probe函数将被调用。因为在该结构体中有同名soc-audio platform driver.
    platform_device_add(s3c24xx_uda134x_snd_device)
    
    .....
    soc_probe
        //card其实就是snd_soc_s3c24xx_uda134x
        struct snd_soc_card *card = platform_get_drvdata(pdev);
        snd_soc_register_card(card);
            snd_soc_instantiate_cards();
                struct snd_soc_card *card;
                list_for_each_entry(card, &card_list, list)//对card_list中的每一个card,执行snd_soc_instantiate_card(card)
                        snd_soc_instantiate_card(card);
                            4.3.1/* bind DAIs */
                            for (i = 0; i < card->num_links; i++)
                                soc_bind_dai_link(card, i);
                                    /*find CPU DAI from registered DAIs*/
                                        list_for_each_entry(cpu_dai, &dai_list, list) {
                                            if (strcmp(cpu_dai->name, dai_link->cpu_dai_name))
                                        rtd->cpu_dai = cpu_dai //即s3c24xx_i2s_dai
                                        ....
                                    /*find CODEC from registered CODECs*/
                                        list_for_each_entry(codec, &codec_list, list) {
                                            if (dai_link->codec_of_node) {
                                            if (strcmp(codec->name, dai_link->codec_name))
                                        rtd->codec = codec //即code->codec_drv = soc_codec_dev_uda134x
                                        
                                    .....
                                        /*
                                     * CODEC found, so find CODEC DAI from registered DAIs from
                                     * this CODEC
                                     */
                                    list_for_each_entry(codec_dai, &dai_list, list) {
                                        if (codec->dev == codec_dai->dev &&
                                            !strcmp(codec_dai->name,
                                                dai_link->codec_dai_name)) {

                                        rtd->codec_dai = codec_dai;//即s3c24xx_i2s_dai
                                    .......
                                    /* no, then find one from the set of registered platforms */
                                    list_for_each_entry(platform, &platform_list, list) {
                                            if (strcmp(platform->name, platform_name))
                                        
                                        rtd->platform = platform;//即samsung_asoc_platform
                                        
                            ........
                            
                            4.3.2 /* initialize the register cache for each available codec */
                            list_for_each_entry(codec, &codec_list, list) {
                                //初始化codec_list链表中的每个codec中的寄存器,这不是我们所关心的点
                                ret = snd_soc_init_codec_cache(codec, compress_type);
                            }
                            
                            4.3.3/* card bind complete so register a sound card */
                            ret = snd_card_create(SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1,
                                    card->owner, 0, &card->snd_card);
                            .....
                            4.3.4/* early DAI link probe */
                            soc_probe_dai_link(card, i, order)
                                /* probe the cpu_dai,调用cpu_dai中的probe函数 */
                                /* probe the CODEC,调用codec中的probe函数*/
                                /* probe the platform,调用platform中的probe函数 */
                                /* probe the CODEC DAI */
                                * create the pcm */
                                ret = soc_new_pcm(rtd, num);
                                        soc_pcm_ops->open    = soc_pcm_open;
                                        soc_pcm_ops->close    = soc_pcm_close;
                                        soc_pcm_ops->hw_params    = soc_pcm_hw_params;
                                        soc_pcm_ops->hw_free    = soc_pcm_hw_free;
                                        soc_pcm_ops->prepare    = soc_pcm_prepare;
                                        soc_pcm_ops->trigger    = soc_pcm_trigger;
                                        soc_pcm_ops->pointer    = soc_pcm_pointer;
                                        ......
                                        snd_pcm_new
                            4.3.5 snd_card_register(card->snd_card);

在文章的开始,曾经说过当使用ASOC驱动时,它将自动调用snd_card_create、snd_pcm_new、以及snd_card_register。从这个地方也可以看出ASOC是在ALSA的基础上,进行了封装。

附:

static struct snd_soc_card snd_soc_s3c24xx_uda134x = {
    .name = "S3C24XX_UDA134X",
    .owner = THIS_MODULE,
    .dai_link = &s3c24xx_uda134x_dai_link,
    .num_links = 1,
};
static struct snd_soc_dai_link s3c24xx_uda134x_dai_link = {
    .name = "UDA134X",
    .stream_name = "UDA134X",
    .codec_name = "uda134x-codec",
    .codec_dai_name = "uda134x-hifi",
    .cpu_dai_name = "s3c24xx-iis",
    .ops = &s3c24xx_uda134x_ops,
    .platform_name    = "samsung-audio",
};    
原文地址:https://www.cnblogs.com/-glb/p/14359899.html