首先我們看sound/soc/s3c24xx這個文件,從入口函數(shù)開始分析
s3c2410_uda1341_init
driver_register(&s3c2410iis_driver);//注釋1
注釋1:
static struct device_driver s3c2410iis_driver = {
.name = "s3c2410-iis",
.bus = &platform_bus_type,
.probe = s3c2410iis_probe,
.remove = s3c2410iis_remove,
};
一旦在內(nèi)核中發(fā)現(xiàn)同名的驅(qū)動與設備——"s3c2410-iis",就會調(diào)用probe函數(shù),所以我們重點來分析probe函數(shù)
s3c2410iis_probe(struct device *dev)
/* GPB 4: L3CLOCK, OUTPUT */
s3c2410_gpio_cfgpin(S3C2410_GPB4, S3C2410_GPB4_OUTP);
s3c2410_gpio_pullup(S3C2410_GPB4,1);
/* GPB 3: L3DATA, OUTPUT */
s3c2410_gpio_cfgpin(S3C2410_GPB3,S3C2410_GPB3_OUTP);
/* GPB 2: L3MODE, OUTPUT */
s3c2410_gpio_cfgpin(S3C2410_GPB2,S3C2410_GPB2_OUTP);
s3c2410_gpio_pullup(S3C2410_GPB2,1);//以上五行代碼用于配置L3接口,L3接口接的就是GPB2,GPB3,GPB4
/* GPE 3: I2SSDI */
s3c2410_gpio_cfgpin(S3C2410_GPE3,S3C2410_GPE3_I2SSDI);
s3c2410_gpio_pullup(S3C2410_GPE3,0);
/* GPE 0: I2SLRCK */
s3c2410_gpio_cfgpin(S3C2410_GPE0,S3C2410_GPE0_I2SLRCK);
s3c2410_gpio_pullup(S3C2410_GPE0,0);
/* GPE 1: I2SSCLK */
s3c2410_gpio_cfgpin(S3C2410_GPE1,S3C2410_GPE1_I2SSCLK);
s3c2410_gpio_pullup(S3C2410_GPE1,0);
/* GPE 2: CDCLK */
s3c2410_gpio_cfgpin(S3C2410_GPE2,S3C2410_GPE2_CDCLK);
s3c2410_gpio_pullup(S3C2410_GPE2,0);
/* GPE 4: I2SSDO */
s3c2410_gpio_cfgpin(S3C2410_GPE4,S3C2410_GPE4_I2SSDO);
s3c2410_gpio_pullup(S3C2410_GPE4,0);//以上十行代碼用于將GPIO管腳設置為IIS管腳
init_s3c2410_iis_bus();//初始化IIS相關寄存器,這里將所有寄存器初值都寫為0
init_uda1341();//初始化芯片
/* GPB 4: L3CLOCK */
/* GPB 3: L3DATA */
/* GPB 2: L3MODE */
uda1341_l3_address(UDA1341_REG_STATUS);//傳輸8位STATUS模式地址,詳見注釋2
uda1341_l3_data(0x40 | STAT0_SC_384FS | STAT0_IF_MSB|STAT0_DC_FILTER);//設置STATUS
uda1341_l3_data(STAT1 | STAT1_ADC_ON | STAT1_DAC_ON);//設置STATUS
uda1341_l3_address(UDA1341_REG_DATA0);//傳輸8位DATA0模式地址
uda1341_l3_data(DATA0 |DATA0_VOLUME(0x0));// maximum volume
uda1341_l3_data(DATA1 |DATA1_BASS(uda1341_boost)| DATA1_TREBLE(0));
uda1341_l3_data((DATA2 |DATA2_DEEMP_NONE) &~(DATA2_MUTE));
uda1341_l3_data(EXTADDR(EXT2));
uda1341_l3_data(EXTDATA(EXT2_MIC_GAIN(0x6)) | EXT2_MIXMODE_CH1);//input channel 1 select(input channel 2 off)
/* 設置兩個DMA通道:一個用于播放,另一個用于錄音。代碼省略*/
.........................................................................................
audio_dev_dsp = register_sound_dsp(&smdk2410_audio_fops, -1);
audio_dev_mixer = register_sound_mixer(&smdk2410_mixer_fops, -1);
我們先不分析最后兩行代碼,首先總結(jié)一下在probe函數(shù)里面完成了那些工作:
1、初始化L3接口
2、初始化IIS接口
3、初始化IIS相關寄存器
4、初始化芯片
5、設置兩個DMA通道
如此硬件相關的東西似乎都設置好了,我們再來分析
audio_dev_dsp = register_sound_dsp(&smdk2410_audio_fops, -1);
audio_dev_mixer = register_sound_mixer(&smdk2410_mixer_fops, -1);
這兩行代碼
首先來看:register_sound_dsp(&smdk2410_audio_fops, -1);
register_sound_dsp(&smdk2410_audio_fops, -1);
sound_insert_unit(&chains[3], fops, dev, 3, 131,"dsp", S_IWUSR | S_IRUSR, NULL);
__sound_insert_unit(s, list, fops, index, low, top);//將s加入到鏈表頭
list=&((*list)->next);
s->unit_minor=n;//struct sound_unit * s,用unit_minor來創(chuàng)建,這點要牢記
s->unit_fops=fops;
s->next=*list;
*list=s;
我們注意到register_sound_dsp(&smdk2410_audio_fops, -1);之后的代碼都在文件sound_core.c中,所以這里涉及到分成的概念,我們看看核心層里面都做了哪些工作:
register_chrdev(SOUND_MAJOR, "sound", &soundcore_fops)==-1)
這里的soundcore_fops只包含了一個函數(shù)open函數(shù),詳情如下:
static const struct file_operations soundcore_fops=
{
/* We must have an owner or the module locking fails */
.owner = THIS_MODULE,
.open = soundcore_open,
};
那么這個open函數(shù)肯定任重道遠了,我們進去看看
soundcore_open(struct inode *inode, struct file *file)
unit = iminor(inode);//獲取次設備號
s = __look_for_unit(chain, unit);
if(s->unit_minor==unit)
return s;//我們根據(jù)unit_minor與unit來找到鏈表中的某個對應的struct sound_unit結(jié)構(gòu)體
new_fops = fops_get(s->unit_fops);//獲得file_operation結(jié)構(gòu)體
file->f_op = new_fops;//現(xiàn)在我們用新的file_operation結(jié)構(gòu)體
err = file->f_op->open(inode,file);//來執(zhí)行open函數(shù),我們進去看看
static int smdk2410_audio_open(struct inode *inode, struct file *file)
if ((file->f_mode & FMODE_WRITE))//這是寫的情況,讀的情況就不再分析了
{
init_s3c2410_iis_bus_tx();//設置IIS控制器寄存器
audio_clear_buf(&output_stream);//清空音頻緩沖區(qū)
}
我們來總結(jié)一下關于:register_sound_dsp(&smdk2410_audio_fops, -1);
這里涉及到一個分層的概念,首先通過register_sound_dsp(&smdk2410_audio_fops, -1);這個函數(shù)向核心層注冊了file_operation結(jié)構(gòu)體,在核心層里面通過次設備號找到這個結(jié)構(gòu)體,就可以用這個結(jié)構(gòu)體里面的函數(shù)從而對設備進行相關的操作。
我們再來分析下:register_sound_mixer(&smdk2410_mixer_fops, -1);
register_sound_mixer(&smdk2410_mixer_fops, -1);
sound_insert_unit(&chains[0], fops, dev, 0, 128, "mixer", S_IRUSR | S_IWUSR, NULL);//跟上面是一樣的,就不在分析了
我們看到這兩個函數(shù)有兩點不同,一是添加的鏈表不同,二是操作函數(shù)集不同
這里我們有必要說一下:
audio_dev_dsp = register_sound_dsp(&smdk2410_audio_fops, -1);注冊的函數(shù)集用來播放和錄音
register_sound_mixer(&smdk2410_mixer_fops, -1);注冊的函數(shù)集用來調(diào)整音量
注釋2:
static void uda1341_l3_address(u8 data)
{
int i;
unsigned long flags;
local_irq_save(flags);
//工作于地址模式
s3c2410_gpio_setpin(S3C2410_GPB2,0);
// w