為了實(shí)現(xiàn)mp3播放,我們最近在sep4020上完成了i2s的驅(qū)動(dòng),主要經(jīng)驗(yàn)總結(jié)如下:
1. 首先是要在probe函數(shù)里進(jìn)行一系列的初始化,這些初始化對(duì)于i2s是很重要的,而且很多
● 配置操作codec的L3的gpio口線;
L3接口相對(duì)于一個(gè)混音器控制接口,也就是對(duì)應(yīng)在驅(qū)動(dòng)中的mixer結(jié)構(gòu)體,在這里我們需要利用3根gpio口線實(shí)現(xiàn)對(duì)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
● 配置端口為放音功能,因?yàn)閟ep4020只支持單獨(dú)放音和錄音,不能全雙工,因此我們?cè)谶@里配置為放音,是通過(guò)一個(gè)口線置高置低實(shí)現(xiàn)的,具體代碼:
*(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,實(shí)現(xiàn)對(duì)codec時(shí)鐘的供給:
*(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; //計(jì)數(shù)時(shí)鐘為總線的DIV分頻
*(volatile unsigned long*)PWM4_DATA_V =0x1; //周期為兩個(gè)計(jì)數(shù)時(shí)鐘
*(volatile unsigned long*)PWM_ENABLE_V =0x1<<3; //高電平為一個(gè)計(jì)數(shù)時(shí)鐘
● 初始化codec(UDA1341),實(shí)際這一步是和第一步配置控制L3口線一起的,配置好口線后,通過(guò)這些口線將codec的參數(shù)配置好,當(dāng)然具體codec的參數(shù)要看uda1341的手冊(cè),其中的uda1341_l3_address,uda1341_l3_data是單獨(dú)為其編寫(xiě)的函數(shù):
*(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開(kāi)不開(kāi)無(wú)所謂 不能像三星的選成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,主要實(shí)現(xiàn)了對(duì)dma通道的使能,清除中斷標(biāo)志位,具體對(duì)dma的緩沖區(qū)分配等會(huì)在使用dma之前的一個(gè)dmasetup函數(shù)中實(shí)現(xiàn),并且有對(duì)應(yīng)的dmaclear清除緩沖區(qū)。
2. 音頻驅(qū)動(dòng)的audio結(jié)構(gòu)體,和mixer結(jié)構(gòu)體
在音頻驅(qū)動(dòng)中主要就是實(shí)現(xiàn)這兩個(gè)結(jié)構(gòu)體的operation函數(shù):
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這個(gè)結(jié)構(gòu)體主要實(shí)現(xiàn)了i2s控制器的操作,包括讀寫(xiě),控制,查詢(poll),打開(kāi),釋放等等。Audio主要實(shí)現(xiàn)了接受上層應(yīng)用數(shù)據(jù),并將數(shù)據(jù)傳遞給codec進(jìn)行播放(放音);從codec接受數(shù)據(jù),并傳遞給上層的功能(錄音)。這部分中又以write,read函數(shù)最為重要,ioctl可以沿用別人的,因此我們的主要工作也是集中在write,read函數(shù)上。
而sep4020_mixer_fops則主要實(shí)現(xiàn)了對(duì)codec參數(shù)的配置,我們也可以很清晰的看到它的operation結(jié)構(gòu)體中只有控制函數(shù),沒(méi)有讀寫(xiě)。并且由于codec的通用性,這部分的代碼基本上可以沿用別人的,如2410。
3. 關(guān)于sep4020_audio_write函數(shù):
這個(gè)是整個(gè)驅(qū)動(dòng)的核心,也是難點(diǎn),牽涉了dma操作,buffer ring的思想,linux中信號(hào)量的思想。一下內(nèi)容讀起來(lái)會(huì)有點(diǎn)吃力,請(qǐng)好好理解代碼
●關(guān)于dma:
對(duì)dma的操作,在這里使用了一個(gè)buffer ring的思想,這里我們來(lái)看一下建立dma緩沖環(huán)的代碼來(lái)理解這種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)