AM335x(TQ335x)學(xué)習(xí)筆記——WM8960聲卡驅(qū)動移植
ASoC是對ALSA驅(qū)動架構(gòu)的進(jìn)一步封裝。ASoC將ALSA驅(qū)動中的各模塊抽象為三部分:Platform、Codec和Machine。Platform主要是平臺硬件驅(qū)動,包括SoC的IIS模塊、DMA等,在本文中就是指AM335x的McASP模塊及AM335x用于音頻讀寫操作的EDMA。Codec是編解碼芯片驅(qū)動,在本文中就是WM8960。Machine是用來描述單板音頻系統(tǒng)連接關(guān)系的驅(qū)動,在本文中其作用是將WM8960與McASP綁定起來,注冊聲卡設(shè)備節(jié)點(diǎn)。由于3.17版本的內(nèi)核已經(jīng)帶有TI維護(hù)的McASP驅(qū)動和Wolf公司維護(hù)的WM8960驅(qū)動,因此,原理上講,我們只需要編寫Machine部分,建立WM8960與McASP的連接關(guān)系即可。不幸的是Wolf對WM8960的維護(hù)不是太完善,還需要我們進(jìn)一步修改。下面我們來看下WM8960在TQ335x上的移植方法。
本文引用地址:http://m.butianyuan.cn/article/201611/322823.htm1. 在DTS中添加聲卡信息
Step1. 完善sound信息
在DTS有一個(gè)節(jié)點(diǎn)名為sound,該節(jié)點(diǎn)用來描述單板上聲卡設(shè)備信息,修改后的內(nèi)容如下:
- sound{
- compatible="ti,tq-evm-audio";
- ti,model="AM335x-EVM";
- ti,audio-codec=<&wm8960>;
- ti,mcasp-controller=<&mcasp1>;
- ti,codec-clock-rate=<24576000>;
- ti,audio-routing=
- "HeadphoneJack","HP_L",
- "HeadphoneJack","HP_R",
- "LINPUT1","LineIn";
- };
(1)compatible = "ti,tq-evm-audio" --> 指定聲卡兼容的設(shè)備,與Machine驅(qū)動中的compatible匹配。
(2)ti,model = "AM335x-EVM" --> 聲卡的名稱,原則上講可以隨意指定,但最好具有一定的可讀性,這里沒有修改。
(3)ti,audio-codec = <&wm8960> --> 指定單板使用的Codec,具體的Codec信息由其指向的節(jié)點(diǎn)wm8960描述。
(4)ti,mcasp-controller = <&mcasp1> --> 指定單板使用的Codec連接到AM335x的McASP1上,McASP1的具體信息由其指向的節(jié)點(diǎn)mcasp1描述。
(5)ti,codec-clock-rate = <24576000> --> 指定Codec的MCLK時(shí)鐘頻率,單位是HZ。TQ335x的Codec使用24.576MHZ的有源晶振提供MCLK,故設(shè)置為24576000。
(6)ti,audio-routing --> DAPM信息描述,用來指定Codec與McASP的連接關(guān)系。此處若不設(shè)置,則需要在Machine驅(qū)動中進(jìn)行設(shè)置。本文在這里做了修改。
Step2. 完善Codec信息
通過閱讀TQ335x的原理圖可知,WM8960的控制端口連接到了AM335x的I2C0端口上,因此,可以i2c0節(jié)點(diǎn)內(nèi)添加如下信息(類似上篇文章中觸摸設(shè)備驅(qū)動節(jié)點(diǎn)):
- wm8960:wm8960@1a{
- compatible="wlf,wm8960";
- reg=<0x1a>;
- };
(1)compatible = "wlf,wm8960" --> 指定Codec兼容設(shè)備,與Codec驅(qū)動中的compatible匹配。
(2) reg = <0x1a> --> WM8960的I2C地址是1A,故設(shè)置為0x1a。
Step3. 完善Platform信息
AM335x的Platform信息主要指McASP和EMDA設(shè)置信息。由于默認(rèn)的DTS已經(jīng)配置好了McASP及EDMA的大部分信息,需要我們配置的是McASP的pinmux和i2s信息。
(1) 修改pinmux信息需要具體參考TQ335x的原理圖,下面是根據(jù)原理圖中的引腳連接方式修改的pinmux信息,如果有啥不懂的可以留言討論:
- am335x_evm_audio_pins:am335x_evm_audio_pins{
- pinctrl-single,pins=<
- 0x1A0(PIN_INPUT_PULLDOWN|MUX_MODE3)/*mcasp0_aclkr.mcasp1_aclkx*/
- 0x1A4(PIN_INPUT_PULLDOWN|MUX_MODE3)/*mcasp0_fsr.mcasp1_fsx*/
- 0x1A8(PIN_OUTPUT_PULLDOWN|MUX_MODE3)/*mcasp0_axr1.mcasp1_axr0*/
- 0x1AC(PIN_INPUT_PULLDOWN|MUX_MODE3)/*mcasp0_ahclkx.mcasp1_axr1*/
- >;
- };
- &mcasp1{
- pinctrl-names="default";
- pinctrl-0=<&am335x_evm_audio_pins>;
- status="okay";
- op-mode=<0>;/*MCASP_IIS_MODE*/
- tdm-slots=<2>;
- /*4serializers*/
- serial-dir=*0:INACTIVE,1:TX,2:RX*/
- 1200
- >;
- tx-num-evt=<1>;
- rx-num-evt=<1>;
- };
(1)pinctrl-0 = <&am335x_evm_audio_pins> --> 指定mcasp1的pinmux信息。
(2)op-mode = <0> --> 指定McASP為I2S工作模式。
(3)tdm-slots = <2> --> 指定通道數(shù)。AM335x的手冊以更廣泛意義的單詞slot命名,具體到I2S接口,其含義就是Channel。
(4)serial-dir --> 指定serializer的方向。AM335x的手冊中提到每個(gè)McASP有16個(gè)serializer,但AM335x這款芯片的McAPS只有4個(gè)serializer,分別用于AXR0、AXR1、AXR2和ARX3。由于TQ335x中將AXR0作為發(fā)送(輸出)、ARX1作為接收(輸入)且沒有ARX2和ARX3,故設(shè)置4個(gè)serial-dir為1、2、0、0(0表示沒有使用,1表示發(fā)送,2表示接收)。
(5)tx-num-evt = <1> --> 指定發(fā)送FIFO大小,本文設(shè)置為1。
(6)rx-num-evt = <1> --> 指定接收FIFO大小,本文設(shè)置為1。
至此,就完成了DTS的全部配置,后面我會將完整的DTS文件上傳到我的資源。
2. Codec驅(qū)動完善
Step1. 修改Codec驅(qū)動,使其支持DTS
由于我們在DTS中指定了Codec的compatible為"wlf,wm8960",而Linux內(nèi)核自帶的WM8960驅(qū)動并沒有支持新式的DTS模式關(guān)聯(lián)。修改方法很簡單,添加i2c_driver的.driver中指定of_match_table即可,修改后的代碼片段如下:
- staticconststructof_device_idwm8960_of_match[]={
- {.compatible="wlf,wm8960",},
- {}
- };
- MODULE_DEVICE_TABLE(of,wm8960_of_match);
- staticstructi2c_driverwm8960_i2c_driver={
- .driver={
- .name="wm8960",
- .owner=THIS_MODULE,
- .of_match_table=wm8960_of_match,
- },
- .probe=wm8960_i2c_probe,
- .remove=wm8960_i2c_remove,
- .id_table=wm8960_i2c_id,
- };
默認(rèn)的WM8960驅(qū)動初始化信息不夠完整,還需要對WM8960進(jìn)行額外的初始化,修改后的代碼片段如下:
- staticintwm8960_probe(structsnd_soc_codec*codec)
- {
- structwm8960_priv*wm8960=snd_soc_codec_get_drvdata(codec);
- structwm8960_data*pdata=dev_get_platdata(codec->dev);
- intret;
- wm8960->set_bias_level=wm8960_set_bias_level_out3;
- if(!pdata){
- dev_warn(codec->dev,"Noplatformdatasupplied");
- }else{
- if(pdata->capless)
- wm8960->set_bias_level=wm8960_set_bias_level_capless;
- }
- ret=wm8960_reset(codec);
- if(ret<0){
- dev_err(codec->dev,"Failedtoissuereset");
- returnret;
- }
- wm8960->set_bias_level(codec,SND_SOC_BIAS_STANDBY);
- /*Latchtheupdatebits*/
- snd_soc_update_bits(codec,WM8960_LINVOL,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_RINVOL,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_LADC,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_RADC,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_LDAC,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_RDAC,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_LOUT1,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_ROUT1,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_LOUT2,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_ROUT2,0x100,0x100);
- /*otherconfiguration*/
- snd_soc_update_bits(codec,WM8960_POWER1,0x1ea,0x1ea);
- snd_soc_update_bits(codec,WM8960_POWER2,0x1f8,0x1f8);
- snd_soc_update_bits(codec,WM8960_POWER3,0xcc,0xcc);
- snd_soc_update_bits(codec,WM8960_LOUTMIX,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_ROUTMIX,0x100,0x100);
- snd_soc_update_bits(codec,WM8960_POWER3,0xc,0xc);
- snd_soc_update_bits(codec,WM8960_LOUT1,0x7f,0x7f);
- snd_soc_update_bits(codec,WM8960_ROUT1,0x7f,0x7f);
- snd_soc_update_bits(codec,WM8960_IFACE2,0x40,0x40);
- snd_soc_update_bits(codec,WM8960_MONOMIX2,0x120,0x120);
- snd_soc_update_bits(codec,WM8960_LINPATH,0x1f8,0x138);
- snd_soc_update_bits(codec,WM8960_LINVOL,0x19f,0x11f);
- snd_soc_update_bits(codec,WM8960_RINVOL,0x19f,0x11f);
- snd_soc_update_bits(codec,WM8960_LOUT2,0x1ff,0x1ff);
- snd_soc_update_bits(codec,WM8960_ROUT2,0x1ff,0x1ff);
- snd_soc_update_bits(codec,WM8960_CLASSD3,0x1a,0x12);
- snd_soc_update_bits(codec,WM8960_CLASSD1,0xc0,0xc0);
- snd_soc_add_codec_controls(codec,wm8960_snd_controls,
- ARRAY_SIZE(wm8960_snd_controls));
- wm8960_add_widgets(codec);
- return0;
- }
評論