好文档就是一把金锄头!
欢迎来到金锄头文库![会员中心]
电子文档交易市场
安卓APP | ios版本
电子文档交易市场
安卓APP | ios版本

wm8976声卡驱动移植(精品).doc

13页
  • 卖家[上传人]:s9****2
  • 文档编号:390824496
  • 上传时间:2023-06-29
  • 文档格式:DOC
  • 文档大小:646KB
  • / 13 举报 版权申诉 马上下载
  • 文本预览
  • 下载提示
  • 常见问题
    • 2.4的内核的oss声音系统,相对简单,所以我选的是这个声音系统,对于2.6的内核,其声音系统是采用的最新的alsa系统,其结构要复杂很多,2.4内核的声音子系统大致上可以分为如图的两部分,相对简单很多,但是后续的产品测试发现声音的播放效果不是很理想,有很多杂音,芯片本身所能处理的效果应该不会很差,但是我在驱动中所处理的不是很好,导致了后期播放时在高音部分的声音处理出现了很多杂音,处理不是很理想首先对于wm8976硬件原理的分析,对于IIS音频总线接口,linux2.4内核对声音的处理都是用的IIS总线,如上图,对硬件的处理和对声音数据的处理分为了两块,IIS对声音数据的处理对于不同芯片都是相同的,不同的就只有声卡硬件的结构因此移植声卡的要点就是在于提供硬件对驱动的接口,及相关初始化 对于wm8976声卡芯片的移植,2.6内核并没有包含wm8976这块芯片的驱动,2.6内核包含的是UDA1341这块声卡芯片的驱动,由于开发板相近,所以决定在这块声卡芯片驱动的基础上改这个驱动对于芯片手册的分析: Wm8976的控制:根据wm8976声卡芯片的设备介绍上图是截取芯片说明介绍的重点部分,这段话是介绍了wm8976芯片的控制方法:大意是这样的,wm8976提供两种控制方法,一种是两线模式,另外一种是三线模式,并且充分兼容和广泛地支持工业标准微控机和DSP信号处理上图是截取的芯片控制说明简介:这里说明了3线模式和2线模式的选择,当MODE引脚为高电平是选择的是3线模式,当为低电平时选择的是2线模式,根据芯片接线原理图可以看出,MODE接的是3.3V电压,是高电平,因而选择的是3线模式三线模式,如上图,CSB信号是控制数据的有效性的,SCLK提供时钟信号,SDIN提供对芯片的控制信号,这里说明下,SDIN会在CSB为高时发出信号,一共16位的信号,高位先发送,高位的8位为地址信号,通过这个信号提供的地址来访问声卡芯片相关的寄存器,SDIN数据的低8位为数据信号,提供对芯片相关寄存器的控制,也就是往地址中写入这些数据达到控制芯片的效果。

      最后数据传输完把CSB信号拉低表明数据传输结束关于两线模式我大概说明下,因为我改的驱动是采用3线模式的,2线模式采用的是IIC的控制模式,先发wm8976芯片的设备地址,让后是读写信号位,当芯片应答后再发出要访问的芯片地址,然后是数据,最后是IIC的结束信号另外值得一提的是wm8976支持主机模式和从机模式,主机模式就是wm8976做为主控芯片,类似于一块单片机用于处理声音数据的,而做为从机模式就是像这块开发板一样,wm8976由片上系统(SOC)也就是这块开发板的主控芯片S3C2440控制的,wm8976相当于外设芯片,帮助SOC处理声音数据,就像协处理器因此在芯片的初始化中要设置wm8976工作在SLAVE(从机)模式WM8976在linux2.6内核下的移植流程关于wm8976的芯片简介也描述的差不多了,现在来说明下wm8976在linux2.6内核下的移植流程,首先是参考另外一块芯片UDA1341声卡芯片的驱动,因为UDA1341芯片所集成的开发板和这块开发板用的芯片是一样的,其中对于IIS总线的声音处理流程也是相同的,不同点是在与芯片的控制不同,因而只要修改UDA1341芯片驱动对于硬件处理的流程即可如上图,linux采用虚拟总线技术,将声音处理分为几个部分如上图,是我画的一个图,其大致描述了在linux设备驱动中运用很广泛的虚拟平台技术,其中Device设备资源是设备的硬件描述,大多是体系相关的,如果换了一种SOC主控芯片但是声卡还是用的这款的话,就只要改动Device设备资源即可,如果是SOC主控芯片相同而声卡芯片换了的话就只要改Driver设备驱动程序即可,这种技术大大的提高了LINUX的可移植性,这也是linux为什么支持这么多芯片和这么受欢迎的原因之一。

      对于一个设备驱动,将其的设备资源和设备驱动都注册进platform总线后,平台总线会根据设备注册时提供的信息来匹配对应的设备资源和设备驱动,具体的匹配方法是根据注册时提供的设备名称,如果名称相同就匹配这个设备资源和设备驱动.对于UDA1341芯片的驱动,其设备资源在内核的arch\arm\plat-s3c24xx\Devs.c中,其代码如下struct platform_device s3c_device_sdi = { .name = "s3c2410-sdi", .id = -1, .num_resources = ARRAY_SIZE(s3c_sdi_resource), .resource = s3c_sdi_resource,};可以看出来 .name = "s3c2410-sdi", 这个就是匹配的条件,注册如内核后,平台总线会根据这个名字来匹配对应的驱动,其设备驱动在内核的\sound\soc\s3c24xx\s3c2410-uda1341.c中,在里面可以找到如下代码static struct device_driver s3c2410iis_driver = { .name = "s3c2410-iis", .bus = &platform_bus_type, .probe = s3c2410iis_probe, .remove = s3c2410iis_remove,};可以看懂 .name = "s3c2410-iis", 有这条信息,如此平台总线就可以根据这两个名字匹配设备和驱动,对于我要改的地方,是在\sound\soc\s3c24xx\s3c2410-uda1341.c这个文件,因为对于设备资源,可以找到其就是存放了SOC的IIS控制寄存器的地址,如果换了芯片而且其IIS控制寄存器不同的话就只要改arch\arm\plat-s3c24xx\Devs.c中的设备资源即可,对于\sound\soc\s3c24xx\s3c2410-uda1341.c这个UDA1341的设备驱动,我根据自己的想法大概分为了两个部分也就是开始的图其中的IIS声音处理部分两块芯片都是相同的,准确的来说所以声音的处理基本都是用的IIS总线,因此这部分不用改,对于硬件相关部分就是要修改的主要部分,因为这两款声卡的控制区别很大驱动的修改: 接下来是对声卡的修改部分,首先修改的\sound\soc\s3c24xx\s3c2410-uda1341.c中的static void init_uda1341(void)函数,这个是对于UDA1341芯片的初始化函数,对于wm8976的初始化和UDA1341完全不同,要做主要修改修改后代码如下static void init_wm8976(void){ uda1341_volume = 57; //音量控制,基本相同,没改 uda1341_boost = 0; //增益控制,相同,没改下面就是声卡硬件相关的初始化,是修改的主要部分,其控制函数wm8976_write_reg实现了对声卡芯片相应寄存器地址的控制,是根据原来的改的,不过两块芯片的读写流程很不一样 /* software reset */ wm8976_write_reg(0, 0); wm8976_write_reg(0x3, 0x6f); wm8976_write_reg(0x1, 0x1f);//biasen,BUFIOEN.VMIDSEL=11b wm8976_write_reg(0x2, 0x185);//ROUT1EN LOUT1EN, inpu PGA enable ,ADC enable wm8976_write_reg(0x6, 0x0);//SYSCLK=MCLK wm8976_write_reg(0x4, 0x10);//16bit wm8976_write_reg(0x2B,0x10);//BTL OUTPUT wm8976_write_reg(0x9, 0x50);//Jack detect enable wm8976_write_reg(0xD, 0x21);//Jack detect wm8976_write_reg(0x7, 0x01);//Jack detect }对于wm8976_write_reg函数的实现:static void wm8976_write_reg(unsigned char reg, unsigned int data){ int i; unsigned long flags; unsigned short val = (reg << 9) | (data & 0x1ff); s3c2410_gpio_setpin(S3C2410_GPB2,1); s3c2410_gpio_setpin(S3C2410_GPB3,1); s3c2410_gpio_setpin(S3C2410_GPB4,1); local_irq_save(flags); for (i = 0; i < 16; i++){ if (val & (1<<15)) { s3c2410_gpio_setpin(S3C2410_GPB4,0); s3c2410_gpio_setpin(S3C2410_GPB3,1); udelay(1); s3c2410_gpio_setpin(S3C2410_GPB4,1); } else { s3c2410_gpio_setpin(S3C2410_GPB4,0); s3c2410_gpio_setpin(S3C2410_GPB3,0); udelay(1); s3c2410_gpio_setpin(S3C2410_GPB4,1); } val = val << 1; } s3c2410_gpio_setpin(S3C2410_GPB2,0); udelay(1); s3c2410_gpio_setpin(S3C2410_GPB2,1); s3c2410_gpio_setpin(S3C2410_GPB3,1); s3c2410_gpio_setpin(S3C2410_GPB4,1); local_irq_restore(flags); }这个函数实现了对wm8976相应寄存器地址的读写,因为采用的是3线模式因此如上面所说的控制方式为如下的时序其中SDIN信号是由GPB3引脚发出的,具体如下图上图为SOC(S3C2440)接口 下图为wm8976接口。

      点击阅读更多内容
      关于金锄头网 - 版权申诉 - 免责声明 - 诚邀英才 - 联系我们
      手机版 | 川公网安备 51140202000112号 | 经营许可证(蜀ICP备13022795号)
      ©2008-2016 by Sichuan Goldhoe Inc. All Rights Reserved.