以文本方式查看主题 - 曙海教育集团论坛 (http://sun4.cn/bbs/index.asp) -- Linux驱动开发 (http://sun4.cn/bbs/list.asp?boardid=33) ---- SEP4020的Linux音频驱动设计开发 (http://sun4.cn/bbs/dispbbs.asp?boardid=33&id=1724) |
-- 作者:wangxinxin -- 发布时间:2010-11-24 10:50:52 -- SEP4020的Linux音频驱动设计开发 为了实现mp3播放,我们最近在sep4020上完成了i2s的驱动,主要经验总结如下:
1. 首先是要在probe函数里进行一系列的初始化,这些初始化对于i2s是很重要的,而且很多 ● 配置操作codec的L3的gpio口线; L3接口相对于一个混音器控制接口,也就是对应在驱动中的mixer结构体,在这里我们需要利用3根gpio口线实现对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; //高电平为一个计数时钟 ● 初始化codec(UDA1341),实际这一步是和第一步配置控制L3口线一起的,配置好口线后,通过这些口线将codec的参数配置好,当然具体codec的参数要看uda1341的手册,其中的uda1341_l3_address,uda1341_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接受数据,并传递给上层的功能(录音)。这部分中又以write,read函数最为重要,ioctl可以沿用别人的,因此我们的主要工作也是集中在write,read函数上。
而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) |