以文本方式查看主題

-  曙海教育集團論壇  (http://www.scb-ycwb.com/bbs/index.asp)
--  Linux驅動開發  (http://www.scb-ycwb.com/bbs/list.asp?boardid=33)
----  SEP4020的Linux音頻驅動設計開發  (http://www.scb-ycwb.com/bbs/dispbbs.asp?boardid=33&id=1724)

--  作者:wangxinxin
--  發布時間:2010-11-24 10:50:52
--  SEP4020的Linux音頻驅動設計開發

為了實現mp3播放,我們最近在sep4020上完成了i2s的驅動,主要經驗總結如下:

 

1.       首先是要在probe函數里進行一系列的初始化,這些初始化對于i2s是很重要的,而且很多

     配置操作codecL3gpio口線;

L3接口相對于一個混音器控制接口,也就是對應在驅動中的mixer結構體,在這里我們需要利用3gpio口線實現對L3的控制,以下是初始化代碼:

*(volatile unsigned long*)(GPIO_PORTD_DIR_V) &= ~(0xd<<1);       //GPB[4:1]=00_0 Output(L3CLOCK):Output(L3DATA):Output(L3MODE)  

 

*(volatile unsigned long*)(GPIO_PORTD_SEL_V) |= (0xd<<1);      

 //GPD[4:1] 1 1010

     配置端口為放音功能,因為sep4020只支持單獨放音和錄音,不能全雙工,因此我們在這里配置為放音,是通過一個口線置高置低實現的,具體代碼:

       *(volatile unsigned long*)(GPIO_PORTG_DIR_V) &= ~(0x1<<11);

*(volatile unsigned long*)(GPIO_PORTG_SEL_V) |= 0x1<<11;

       *(volatile unsigned long*)(GPIO_PORTG_DATA_V) |= 0x1<<11;

     配置pwm,實現對codec時鐘的供給:

*(volatile unsigned long*)PWM4_CTRL_V =0x00;

*(volatile unsigned long*)PWM4_DIV_V =0x4; //88MHz/(4*2)=11Mhz 11M/256fs=42.96k

 *(volatile unsigned long*)PWM4_PERIOD_V =0x2;   //計數時鐘為總線的DIV分頻

 *(volatile unsigned long*)PWM4_DATA_V =0x1;     //周期為兩個計數時鐘

 *(volatile unsigned long*)PWM_ENABLE_V =0x1<<3;     //高電平為一個計數時鐘

     初始化codecUDA1341,實際這一步是和第一步配置控制L3口線一起的,配置好口線后,通過這些口線將codec的參數配置好,當然具體codec的參數要看uda1341的手冊,其中的uda1341_l3_addressuda1341_l3_data是單獨為其編寫的函數:

*(volatile unsigned long*)(GPIO_PORTD_DATA_V) &= ~(L3M|L3C|L3D);

*(volatile unsigned long*)(GPIO_PORTD_DATA_V) |= (L3M|L3C); //Start condition : L3M=H, L3C=H

//以下配置可能需要修改 marked at 11-08

uda1341_l3_address(0x14 + 2);

uda1341_l3_data(0x61);               //1110 dc-filtering開不開無所謂 不能像三星的選成MSB

uda1341_l3_address(0x14 + 2);

uda1341_l3_data(0x21);

uda1341_l3_address(0x14 + 2);

uda1341_l3_data(0xc1);        //Status 1,Gain of DAC 6 dB,Gain of ADC 0dB,ADC non-inverting,DAC non-inverting,Single speed playback,ADC-Off DAC-On

              uda1341_l3_address(0x14 + 0);

uda1341_l3_data(0x0f);        //00,00 ffff  : Volume control (6 bits)  -14dB

              uda1341_l3_address(0x14 + 0);

uda1341_l3_data(0x7b);        //01,11 10,11 : Data0, Bass Boost 18~24dB, Treble 6dB

              uda1341_l3_address(0x14 + 0);

uda1341_l3_data(0x83);

     配置dma,主要實現了對dma通道的使能,清除中斷標志位,具體對dma的緩沖區分配等會在使用dma之前的一個dmasetup函數中實現,并且有對應的dmaclear清除緩沖區。

 

2.       音頻驅動的audio結構體,和mixer結構體

在音頻驅動中主要就是實現這兩個結構體的operation函數:

static struct file_operations sep4020_audio_fops = {

llseek: sep4020_audio_llseek,

write: sep4020_audio_write,

read: sep4020_audio_read,

poll: sep4020_audio_poll,

ioctl: sep4020_audio_ioctl,

open: sep4020_audio_open,

release: sep4020_audio_release

};

 

static struct file_operations sep4020_mixer_fops = {

ioctl: sep4020_mixer_ioctl,

open: sep4020_mixer_open,

release: sep4020_mixer_release

};

sep4020_audio_fops這個結構體主要實現了i2s控制器的操作,包括讀寫,控制,查詢(poll),打開,釋放等等。Audio主要實現了接受上層應用數據,并將數據傳遞給codec進行播放(放音);從codec接受數據,并傳遞給上層的功能(錄音)。這部分中又以writeread函數最為重要,ioctl可以沿用別人的,因此我們的主要工作也是集中在writeread函數上。

 

sep4020_mixer_fops則主要實現了對codec參數的配置,我們也可以很清晰的看到它的operation結構體中只有控制函數,沒有讀寫。并且由于codec的通用性,這部分的代碼基本上可以沿用別人的,如2410

 

3.       關于sep4020_audio_write函數:

這個是整個驅動的核心,也是難點,牽涉了dma操作,buffer ring的思想,linux中信號量的思想。一下內容讀起來會有點吃力,請好好理解代碼

●關于dma

dma的操作,在這里使用了一個buffer ring的思想,這里我們來看一下建立dma緩沖環的代碼來理解這種buffer ring

static int audio_setup_buf(audio_stream_t * s)

{

int frag;

int dmasize = 0;

char *dmabuf = 0;

dma_addr_t dmaphys = 0;

 

if (s->buffers)


主站蜘蛛池模板: 人人狠狠综合久久亚洲| 成人性开放大片| 国产精品亚洲欧美大片在线看| 人妻少妇一区二区三区| 一级做a爱视频| 精品久久久无码中文字幕边打电话| 日日操天天操夜夜操| 国产亚洲精品美女2020久久| 久久狠狠高潮亚洲精品| 好吊妞视频这里只有精品| 校园放荡三个女同学| 国产成人综合精品| 久久精品国产亚洲av麻| 韩国一级做a爱性色毛片| 日韩中文有码高清| 国产又大又粗又猛又爽的视频| 久久久精品电影| 色综合天天综合网国产成人网| 日本免费人成视频播放| 国产偷久久久精品专区| 久久久久无码国产精品一区| 高级秘密俱乐部的娇妻| 日本19禁啪啪无遮挡免费动图| 啊用力太猛了啊好深视频免费| 久久久久久AV无码免费看大片 | 九九综合九九综合| 黄网站在线观看| 无毒不卡在线观看| 免费黄色毛片视频| 97久久精品人妻人人搡人人玩| 欧美顶级aaaaaaaaaaa片| 国产精品另类激情久久久免费| 亚洲Av人人澡人人爽人人夜夜| 韩国免费乱理论片在线观看2018| 日韩国产在线观看| 啦啦啦中文中国免费高清| japanese日本护士xxxx10一16 | 欧美视频免费在线| 国产日韩精品欧美一区喷水| 久久久久久久久久国产精品免费| 精品乱码一区二区三区在线|