天天看点

android下调试声卡驱动之Machine部分

      本文介绍的代码是在CPU(imx6)和Codec(wm8960)上做的调试并且Codec做主模式,在“概述篇”中提到Codec部分、

Platform部分代码由设备原厂提供和系统提供,所以我们在声卡调试中需要做的只有Machine部分,其他代码在需要的情况下略做微

调即可,调试过程中还要注意在BSP文件里配置好I2S(SSI和AUD复用)。

1、imx_hifi_hw_params函数介绍

static int imx_hifi_hw_params(struct snd_pcm_substream *substream,
				     struct snd_pcm_hw_params *params)
{

	/* 设置Codec与CPU音频数据通信格式为I2S,BLCK不反转,Codec为主模式*/
	dai_format = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
		SND_SOC_DAIFMT_CBM_CFM;

	/*  set codec DAI configuration */
	ret = snd_soc_dai_set_fmt(codec_dai, dai_format);
    if( ret < 0 ){
        printk( "%s: Codec DAI configuration error, %d\n", __func__, ret );
        return ret;
    }

	/*  set i.MX active slot mask */
	snd_soc_dai_set_tdm_slot(cpu_dai,
			channels == 1 ? 0xfffffffe : 0xfffffffc,
			channels == 1 ? 0xfffffffe : 0xfffffffc,
			2, 32);
	/* 设置CPU与Codec音频数据通信格式为I2S,BLCK不反转,CPU为从模式*/
	dai_format = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
		SND_SOC_DAIFMT_CBM_CFM;

	/*  set cpu DAI configuration */
	ret = snd_soc_dai_set_fmt(cpu_dai, dai_format);
    if( ret < 0 ){
        printk( "%s: AP DAI configuration error, %d\n", __func__, ret );
        return ret;
    }

	sample_rate = params_rate(params);
	switch (sample_rate) {
       case 44100:
        if (channels == 1) { // 根据通道数配置相关时钟频率(1:代表单通道)
            /* Mono mode */
            dacdiv = WM8960_DAC_DIV_2; /* 22.05 KHz */
            bclk = WM8960_BCLK_DIV_16;
        }
        else {
            /* Stereo Mode */
            dacdiv =WM8960_DAC_DIV_1; // 44.1 KHz,为设置采样频率(LRCK),配置Codec寄存器的值
            bclk = WM8960_BCLK_DIV_4; // 为设置BCLK频率,配置Codec寄存器的值
			dclk = WM8960_DCLK_DIV_16;// 为设置D类放大器时钟频率,配置Codec寄存器的值
		}
        sysclk_div = WM8960_SYSCLK_DIV_2; // 为设置Codec的系统时钟频率,配置Codec寄存器的值
        pll_out = 11289600; // 设定PLL的频率值
        break;
      default:
        printk("do not support this sample frequency");
        return -EINVAL;
    }

    /* 设置Codec的PLL频率:11289600*/
	ret=snd_soc_dai_set_pll(codec_dai,1,0,priv->sysclk/2,pll_out);
    if( ret < 0 ){
        printk( "%s: AP codec pll error, %d\n", __func__, ret );
        return ret;
    }
	/* 设置Codec的系统时钟:11289600*/
    ret = snd_soc_dai_set_clkdiv( codec_dai, WM8960_SYSCLKDIV, sysclk_div );
    if( ret < 0 ){
        printk( "%s: Codec SYSCLKDIV setting error, %d\n", __func__, ret );
        return ret;
    }
	/* 设置采样频率(LRCK)的时钟,lrckclk= sysclk/1*256=44100 */
    ret = snd_soc_dai_set_clkdiv( codec_dai, WM8960_DACDIV, dacdiv);
    if( ret < 0 ){
        printk( "%s: Codec DACDIV setting error, %d\n", __func__, ret );
        return ret;
    }
	/* 设置D类放大器时钟频率:dclk= sysclk/16 */
	ret = snd_soc_dai_set_clkdiv( codec_dai, WM8960_DCLKDIV,dclk);
	if( ret < 0 ){
		printk( "%s: Codec DCLKDIV setting error, %d\n", __func__, ret );
		return ret;
	}
	/* 设置BCLK类放大器时钟频率:bitclk=sysclk/4 = 2822400 */
	ret = snd_soc_dai_set_clkdiv( codec_dai, WM8960_BCLKDIV, bclk); 
	if( ret < 0 ){
	printk( "%s: Codec WM8960_BCLKDIV setting error, %d\n", __func__, ret );
	return ret;
    }

    return 0;
}
           

2、snd_soc_dai_link结构体介绍

     platform 连接 codec 与 cpu 的驱动时,需要通过内核函数结构体snd_soc_dai_link来设置两边的接口。

static struct snd_soc_dai_link imx_dai[] = {
	{
		.name = "HiFi",
		.stream_name = "HiFi",
		.codec_dai_name	= "wm8960-hifi",
		.codec_name	= "wm8960.1-001a", 
		/* wm8960.1-001a代表的是用I2C1来控制codec,其I2C地址为0x1a;I2CX, X代表的是0,1,2,开始 */
		.cpu_dai_name	= "imx-ssi.1",  
		/* 
		 * 代表的是同一个声卡中的第一个设备与SSI1相连;I2SX X代表0,1,2;ssi.1代表SSI1。
		 * IMX6中的SSI接口有3对可以接着路I2S;三路SSI可以每路SSI 单独作为一个声卡,
		 * 也可以是一个声卡包含多个设备; 
		 */
		.platform_name	= "imx-pcm-audio.1",
		/* 代表的连接接口与SSI来对应,用哪路SSI,其后面的数字就是几。 */
		.init		= imx_wm8960_init,
		.ops		= &imx_hifi_ops,
	},
};
           

继续阅读