介绍一种很棒的wince 如何替换系统声音的方法

时间:2022-05-01 20:08:34

Topic:介绍一种很棒的wince 如何替换系统声音的方法(作者:Baiduluckyboy)
//------------------------------------------------------------------------------------------------------------------------------

// Topic:介绍一种很棒的wince 如何替换系统声音的方法(作者:Baiduluckyboy)

// 作者:Baiduluckyboy(wangyuluyulu@126.com

// 论坛账号:Baiduluckyboy (经常在CSDN出没)

// 版权:解放军电子工程学院Baiduluckyboy(wangyuluyulu@126.com

// 平台:wince6.0.0 S5PC100 BSP

// 发布日期:2010-09-08

// 最后修改:

// 注意事项:未经作者同意,商业网站不能转载,并且不得在转载的时候擅自修改、删除文章的任何部分

//------------------------------------------------------------------------------------------------------------------------------

 

 

 

 

前面有做一个操作系统为WinCE5.0的嵌入式设备,临到发布阶段版本了,突然来了个新需求(一直痛恨随意变更和拍脑袋式样的软件需求,但是顾客是上帝,没有办法),要求更改点击触摸屏的声音(还要怪微软默认的声音太难听)为自定义的声音。

根据以往经验,以为只要替换一下Image里面包含的声音文件,顶多再修改注册表里面某个键值就可以了;却发现这次不管用了,在注册表里根本找不到相关的项。在设备的Windows目录下虽然找到很多.wav的声音文件,却都是一些比如菜单选择,连接断开等事件对应的声音,唯独没有键盘音和触摸屏的声音文件。

于是发动Google大法,遍寻天下WinCE高手留下的语录和足迹,终于有所领悟,原来这两种声音不是直接以.wav形式存在在设备当中,而是被当作资源文件编译到动态链接库当中。不过还是搜索到这两种声音.wav的名称,键盘音两个为keysoft.wav和keyloud.wav,触摸屏的声音为tchsoft.wav和tchloud.wav. 再用Windows 搜索工具在WinCE500目录下定位到这个目录:../WINCE500/PUBLIC/COMMON/OAK/DRIVERS/WAVEUI. 这个目录下除了有上述四个文件,waveui.c里面没有实际内容,主要是waveui.rc这个资源文件和 它使用的头文件resource.h.

接下来看看这个目录下的sources文件,目标文件是生成waveui.lib,连接的文件是waveui.c和waveui.rc。在../WINCE500目录下的build.log(sysgen时生成的)里面找了下,这个库最终是被waveapi.dll链接的。使用sysgen_capture工具获取到生成waveapi.dll的sources文件,主要内容如下: 

TARGETNAME=waveapi

TARGETTYPE=DYNLINK

RELEASETYPE=PLATFORM

SYNCHRONIZE_DRAIN=1

DLLENTRY=WaveApiDllEntry



DEFFILE=$(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/waveapi.def



SOURCELIBS=/

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/wapiwave.lib /

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/wapimap.lib /

   $(_PUBLICROOT)/common/oak/lib /$(_CPUINDPATH)/waveui.lib



TARGETLIBS=/

   $(_SYSGENSDKROOT)/lib/$(_CPUINDPATH)/coredll.lib /

   $(_SYSGENOAKROOT)/lib/$(_CPUINDPATH)/audevman.lib /

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/wapistub.lib /

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/0409/waveapi.res



SOURCES=



从上面可以看到,和waveui这个目录有关的一个是waveui.lib被链接,一个是waveapi.res这个文件的输入文件之一应该就是waveui.rc. 

到这一步,基本上可以确定触摸屏的声音通过waveui.rc文件编译为资源文件waveapi.res,然后成为waveapi.dll的一部分供系统调用。现在的问题是应该如何通过更新waveapi.dll来达到更新触摸屏的声音?

Waveapi.dll在系统Image文件生成前存在于release目录下,所以我们只需要保证release目录下的waveapi.dll包含的声音更新即可。

一种办法是直接用自定义文件重命名后替换../WINCE500/PUBLIC/COMMON/OAK/DRIVERS/WAVEUI下的.wav文件,然后在WinCE Platform Builder的Build OS菜单里面打开的命令行里面重新编译这个目录(Build -c),最后用Build OS菜单里面的sysgen命令重新生成整个工程文件和镜像文件。这种办法在一些论坛上也有找到一些前辈的说法为依据,不过却违反了BSP开发的独立性原则,即BSP的实现应该仅在BSP的代码中修改实现,而不应该修改公共目录代码。另外这种办法也显得有些笨拙。

[注意,即使是这种笨拙的办法也会遇到一些意想不到的问题,比如自定义的.wav文件大小和内容要适合,我开始用一个音乐文件做了一个较大(两秒,约8K)的.wav,结果重复试了好几次,就是没有声音;后面换了客户提供的声音文件就OK了。Faint中]。



借鉴以前移植public下的驱动代码的经验,于是有了第二种想法,将Waveapi的相关实现都移植到BSP中。在BSP下新建waveapi目录,将/WINCE500/PUBLIC/COMMON/OAK/DRIVERS/WAVEUI拷贝到下面作为一个子目录,新建另外一个子目录Dll,里面存放以下文件

(1)    makefile:从其他任意目录拷贝

(2)    sources:使用上述用sysgen_capture获取的sources文件,作适当修改(红色部分),如下

TARGETNAME=waveapi

TARGETTYPE=DYNLINK

RELEASETYPE=PLATFORM

SYNCHRONIZE_DRAIN=1

TARGETDEFNAME=waveapi

DEFFILE=$(TARGETDEFNAME).def

DLLENTRY=WaveApiDllEntry



SOURCELIBS=/

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/wapiwave.lib /

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/wapimap.lib /

   $(_TARGETPLATROOT)/lib/$(_CPUINDPATH)/waveui.lib



TARGETLIBS=/

   $(_SYSGENSDKROOT)/lib/$(_CPUINDPATH)/coredll.lib /

   $(_SYSGENOAKROOT)/lib/$(_CPUINDPATH)/audevman.lib /

   $(_PUBLICROOT)/common/oak/lib/$(_CPUINDPATH)/wapistub.lib /

   waveapi.res



SOURCES=



(3)    waveapi.def,新建如下

LIBRARY WAVEAPI

EXPORTS WAM_Init

      WAM_Deinit

      WAM_Open

      WAM_Close

      WAM_Read

      WAM_Write

      WAM_Seek

      WAM_IOControl

      WAM_PowerDown

      WAM_PowerUp



(4)    waveapi.res:这个是关键,如何从waveui.rc和替换的.wav文件生成一个新的waveapi.res文件,然后放在这个目录下。

一种做法是通过VS2005建立一个Win32 Smart Device应用程序的工程,将waveui.rc文件、头文件resource.h以及新的声音文件加入进去编译,生成的waveui.res,然后重命名为waveapi.res拷贝过来。不过这种方法的缺点是要依赖PB编译器以外的工具。     

    完成以上文件操作之后,即可在PB当中单独编译这个目录,或者将这个目录加入到上一级目录的dirs文件当中编译,生成新的waveapi.dll到release目录,然后重新makeimg即可。    

关于waveapi.res生成方法的另外一种想法是使用WinCE自带的资源编译器Resource Compiler—rc.exe在命令行里面编译这个rc文件,碰到一些编译错误,在PB命令行环境里面(工程已经打开)windows.h这个头文件没有办法找到,造成编译错误,一时还没有办法继续下去。 

不知道是否还有其他更好的办法,希望各位踏足此地的WinCE前辈们指点,十分感谢!





一位WINCE同好给我的回复:

我也发现可以不需要使用VS2005,单独用PlatformBuilder即可以编译BSP底下的WaveApi目录,

1.      无法编译WAVEUI是因为sources File缺少WINCEOEM=1

2.      修改sources File WINCETARGETFILES=$(_COMMONOAKROOT)/lib/$(_CPUINDPATH)/waveapi.res 为WINCETARGETFILES=$(_TARGETPLATROOT)/lib/$(_CPUINDPATH)/waveapi.res

3.      修改makefile.inc $(_COMMONOAKROOT)/lib/$(_CPUINDPATH)/waveapi.res:$(_OBJDIR)/$(TARGETNAME).res 为$(_TARGETPLATROOT)/lib/$(_CPUINDPATH)/waveapi.res:$(_OBJDIR)/$(TARGETNAME).res

4.      移除Dll目录下waveapi.res

5.      修改Dll目录下的sources File将 TARGETLIBS=/ waveapi.res修改为 $(_TARGETPLATROOT)/lib/$(_CPUINDPATH)/waveapi.res

6.      在该项目的Release目录下的bib文件的waveapi.dll.0804.mui必须Mask起来,否则没有作用,上述档案的0804为其项目的default language.

7.      主要目的是将独立的waveapi.res导到参考TARGETPLATROOT下的waveapi.res

8.      当然也可以加上DIR档案,由PlatformBuilder经由项目编译。