- UID
- 1029342
- 性别
- 男
|
------------------------------------------------------------------------
紧接着就来看一下init_uda1341 这个初始化UDA1341 芯片的函数:
static void init_uda1341(void)
uda1341_volume = 62 -((DEF_VOLUME * 61) / 100);
uda1341_boost = 0;
uda_sampling =DATA2_DEEMP_NONE;
uda_sampling &=~(DATA2_MUTE);
首先上来就是设定几个待会儿配置要用的参数。参考UDA1341 芯片datasheet 后,可以知道uda1341_volume参数的含义,62 表示音量设置表中有效音量的总档数,61 表示音量总共有61档,DEF_VOLUME%表示所要调的音量的百分比大小,这样61*DEF_VOLUME%所得出的就是所要调的音量是音量总档数的第几档,由于音量设置表中列出值的是按衰减量递增的,所以刚才得到的音量档数需要在总档数下衰减多少才能得到呢?显然只要将音量总档数减去所要调到的音量档数即可,即62-61*DEF_VOLUME%。
local_irq_save(flags);
同先前一样,调用该宏函数来保存IRQ 中断使能状态,并禁止IRQ 中断。
write_gpio_bit(GPIO_L3MODE, 1);
write_gpio_bit(GPIO_L3CLOCK, 1);
调用write_gpio_bit 宏函数,将GPIO 相应的引脚设为高电平或低电平。这里是把GPIO_L3MODE和GPIO_L3CLOCK 这两个引脚设为高电平。
local_irq_restore(flags);
同先前一样,调用该宏函数来恢复IRQ 和FIQ 的中断使能状态。
/
uda1341_l3_data(EXTADDR(EXT2));
uda1341_l3_data(EXTDATA(EXT2_MIC_GAIN(0x6)) |EXT2_MIXMODE_CH1);
再次调用uda1341_l3_address函数,用00010100(二进制)参数设置为直接地址寄存器模式。接着分5次调用uda1341_l3_data函数来进行配置,第一次用uda1341_volume参数的值23(十进制)将音量大小设置为总音量的65%;第二次用01000000(二进制)参数将低音推进设置为0,高音设置为0;第三次用00000000(二进制)参数又将音量调到衰减0dB,即调到最大(不理解为什么);最后两次要一起看,先用11000010(二进制)参数将EA2~EA0设为010(二进制)进入设置特定功能的外部地址,然后用11111001(二进制)参数将ED4~ED0 设为11001(二进制)将MIC的灵敏度设为+27dB,将混频器模式设为选择通道1输入(这时通道2输入关闭)。
【其实这里的“uda1341_l3_data(uda_sampling);”,这句话应该是不正确的,不是准备再将音量调到最大。应该改为:
uda_l3_data(DATA2 | uda_sampling);
即用10000000(二进制)参数设置静音关闭和高低音模式为flat 模式(高低音增益都为0dB)等。】
------------------------------------------------------------------------
马上来看一下uda1341_l3_address 和uda1341_l3_data 这两个具体控制GPIO口时序来传输数据的函数。首先看uda1341_l3_address 函数:
static void uda1341_l3_address(u8 data)
local_irq_save(flags);
在对GPIO口设置或操作前总要先调用该宏函数来保存IRQ 中断使能状态,并禁止IRQ 中断。
write_gpio_bit(GPIO_L3MODE, 0);
write_gpio_bit(GPIO_L3DATA, 0);
write_gpio_bit(GPIO_L3CLOCK, 1);
分别将GPIO_L3MODE 引脚设为低电平,将GPIO_L3DATA 引脚设为低电平,将GPIO_L3CLOCK引脚设为高电平。根据UDA1341 芯片datasheet 里的时序图,把GPIO_L3MODE引脚设为低电平,就是地址模式。
udelay(1);
调用udelay函数来短暂延时1us。在驱动程序中用udelay 函数来延时微秒级时间,mdelay函数来延时毫秒级时间,而在应用程序中用usleep 函数来延时微秒级时间,sleep 函数来延时毫秒级时间。
在/kernel/include/asm-arm/delay.h 文件中:
extern void udelay(unsigned long usecs);
在/kernel/include/linux/delay.h 文件中:
#ifdef notdef
#define mdelay(n) (\
{unsigned long msec=(n); while (msec--)udelay(1000);})
#else
#define mdelay(n) (\
(__builtin_constant_p(n)&&(n)<=MAX_UDELAY_MS) ? udelay((n)*1000) : \
({unsigned long msec=(n); while (msec--)udelay(1000);}))
#endif
在/kernel/arch/arm/lib/delay.S 文件中:
ENTRY(udelay)
mov r2, #0x6800
orr r2, r2,#0x00db
mul r1, r0,r2
ldr r2,LC0
ldr r2,[r2]
mov r1, r1,lsr #11
mov r2, r2,lsr #11
mul r0, r1,r2
movs r0, r0,lsr #6
RETINSTR(moveq,pc,lr)
最后用ARM 汇编指令实现了微秒级的短暂延时。
for (i = 0; i < 8; i++){
if (data & 0x1){
write_gpio_bit(GPIO_L3CLOCK,0);
udelay(1);
write_gpio_bit(GPIO_L3DATA,1);
udelay(1);
write_gpio_bit(GPIO_L3CLOCK,1);
udelay(1);
} else {
write_gpio_bit(GPIO_L3CLOCK,0);
udelay(1);
write_gpio_bit(GPIO_L3DATA,0);
udelay(1);
write_gpio_bit(GPIO_L3CLOCK,1);
udelay(1);
}
data>>= 1;
}
接下来就是将一个字节一位一位通过GPIO 口发送出去的循环结构,从该字节的最低位(D0)开始发送。若D0为1,则设置GPIO_L3DATA 引脚为高电平,否则为低电平。同时需要控制GPIO_L3CLOCK引脚的时钟信号,数据会在时钟的上升沿写入UDA1341芯片,所以需要在时钟引脚为低电平时准备好要传送的数据,然后再将时钟设为高电平。在设置时钟和数据引脚之间用udelay函数进行短暂延时1us。
write_gpio_bit(GPIO_L3MODE, 1);
udelay(1);
在地址模式下数据传送完成后,则设置GPIO_L3MODE 引脚为高电平,准备进入数据传输模式,并短暂延时1us。
local_irq_restore(flags);
最后调用该宏函数来恢复IRQ 和FIQ 的中断使能状态。
------------------------------------------------------------------------
接着来看uda1341_l3_data 函数:
static void uda1341_l3_data(u8 data)
local_irq_save(flags);
同样首先要调用该宏函数来保存IRQ 中断使能状态,并禁止IRQ 中断。
write_gpio_bit(GPIO_L3MODE, 1);
udelay(1);
write_gpio_bit(GPIO_L3MODE, 0);
udelay(1);
write_gpio_bit(GPIO_L3MODE, 1);
在进入数据传输模式前,先要将GPIO_L3MODE信号需要有一个低电平的脉冲,所以依次设置该引脚为高电平,低电平,高电平,这样就进入了数据传输模式。
for (i = 0; i < 8; i++){
if (data & 0x1){
write_gpio_bit(GPIO_L3CLOCK,0);
udelay(1);
write_gpio_bit(GPIO_L3DATA,1);
udelay(1);
write_gpio_bit(GPIO_L3CLOCK,1);
udelay(1);
} else {
write_gpio_bit(GPIO_L3CLOCK,0);
udelay(1);
write_gpio_bit(GPIO_L3DATA,0);
udelay(1);
write_gpio_bit(GPIO_L3CLOCK,1);
udelay(1);
}
data>>= 1;
}
接下来的这个步骤和uda1341_l3_address 函数一样,一位一位将数据发送出去。
write_gpio_bit(GPIO_L3MODE, 1);
write_gpio_bit(GPIO_L3MODE, 0);
udelay(1);
write_gpio_bit(GPIO_L3MODE, 1);
最后,GPIO_L3MODE信号同样需要有一个低电平的脉冲才表示数据传输模式结束,所以再次设置该引脚为高电平,低电平,高电平。
local_irq_restore(flags);
完成后,调用该宏函数来恢复IRQ 和FIQ 的中断使能状态。
------------------------------------------------------------------------
看一下卸载驱动模块时调用的函数:
void __exit s3c2410_uda1341_exit(void)
这个函数就比较简单了。
unregister_sound_dsp(audio_dev_dsp);
unregister_sound_mixer(audio_dev_mixer);
首先调用unregister_sound_dsp 和unregister_sound_mixer 这两个函数来分别注销原先注册的DSP设备和混频器设备。
在/kernel/drivers/sound/sound_core.c 文件中:
void unregister_sound_dsp(int unit)
void unregister_sound_mixer(int unit)
这两个函数的参数一样,为刚才调用注册函数时返回的内核所分配的设备序号。
audio_clear_dma(&output_stream);
audio_clear_dma(&input_stream);
分两次调用audio_clear_dma 函数来分别释放已申请的音频输入和音频输出的DMA 通道。
------------------------------------------------------------------------
继续来看一下打开设备文件的接口函数,这里先看针对DSP 设备文件的函数:
static int smdk2410_audio_open(struct inode *inode, struct file*file)
if ((file->f_flags& O_ACCMODE) == O_RDONLY) {
if (audio_rd_refcount ||audio_wr_refcount)
return-EBUSY;
audio_rd_refcount++;
} else if ((file->f_flags& O_ACCMODE) == O_WRONLY) {
if(audio_wr_refcount)
return-EBUSY;
audio_wr_refcount++;
} else if ((file->f_flags& O_ACCMODE) == O_RDWR) {
if (audio_rd_refcount ||audio_wr_refcount)
return-EBUSY;
audio_rd_refcount++;
audio_wr_refcount++;
} else
return -EINVAL;
首先上来就是一大段条件判断,主要就是判断file->f_flags这个表示设备文件的打开方式是读取,写入,还是可读写。用audio_rd_refcount 和audio_wr_refcount这两个变量来设置类似于信号量一样的读写占位标志(要写设备的话,只要没有用写方式打开过设备即可;要读的话,则需要该设备同时没有用读或写方式打开过),只要打开过设备文件,相应的方式标志就会加一。
if (cold) {
audio_rate =AUDIO_RATE_DEFAULT;
audio_channels =AUDIO_CHANNELS_DEFAULT;
audio_fragsize =AUDIO_FRAGSIZE_DEFAULT;
audio_nbfrags =AUDIO_NBFRAGS_DEFAULT;
在audio_rd_refcount 和audio_wr_refcount 这两个变量都为0的时候才进入这一步,即对已经打开过的设备文件不进行下面的操作。
这里先设置一下待会儿要配置到IIS 相关寄存器中的变量。
if((file->f_mode &FMODE_WRITE)){
init_s3c2410_iis_bus_tx();
audio_clear_buf(&output_stream);
}
if((file->f_mode &FMODE_READ)){
init_s3c2410_iis_bus_rx();
audio_clear_buf(&input_stream);
}
}
从file->f_mode中判断文件是否可读可写,根据设备文件的打开模式,分别调用了init_s3c2410_iis_bus_tx和init_s3c2410_iis_bus_rx 函数来进行对IIS 总线读写的初始化配置,在这两个函数中对S3C2410芯片的IIS 相关寄存器进行了相应的配置,会在后面说明。然后又调用了audio_clear_buf函数来分别对音频输入和输出两个DMA 缓冲区进行了清空,该函数也会在后面说明。
因为读写操作控制必须用f_mode 来进行判断,所以这里要根据f_mode为可读或可写的标识来进行读写模式的硬件设置。而read,write 函数不需要检查f_mode因为读写权限的检查是由内核在调用他们之前进行的。
MOD_INC_USE_COUNT;
最后调用MOD_INC_USE_COUNT; 来对设备文件计数器加一计数,并返回。
------------------------------------------------------------------------
下面马上来看一下init_s3c2410_iis_bus_tx 和init_s3c2410_iis_bus_rx这两个函数,首先是init_s3c2410_iis_bus_tx 函数:
static void init_s3c2410_iis_bus_tx(void)
IISCON = 0;
IISMOD = 0;
IISFIFOC = 0;
首先初始化IIS控制寄存器,IIS 模式寄存器和IIS FIFO 控制寄存器都为0。
IISPSR = (IISPSR_A(iispsr_value(S_CLOCK_FREQ,44100))
|IISPSR_B(iispsr_value(S_CLOCK_FREQ, 44100)));
设置IIS预分频寄存器,其中调用了iispsr_value 函数来计算预分频值,该函数会在后面说明。
参考S3C2410芯片datasheet 中关于IIS 总线接口的章节,具体设置参数如下:
IISPSR_A(iispsr_value(S_CLOCK_FREQ, 44100)) =IISPSR_A(iispsr_value(384, 44100)) = (一个0~31之间的值)<<5 预分频控制器A,用于内部时钟块
IISPSR_B(iispsr_value(S_CLOCK_FREQ, 44100))) = (一个0~31之间的值)<<0 预分频控制器B,用于外部时钟块 |
|