|
|
本文试阐述在alsa环境下多个进程共享声卡的问题以及具体如何通过alsa让skype在arts的帮助下和其他程序同时使用声卡。本文可以看作是笔者的个人经验介绍。同时有些信息翻译自一些英文资料。笔者的系统是gentoo,
内核2.6.11-nitro0, KDE3.4(split ebuild,arts-3.4.0),alsa-driver, alsa-utils, alsa-oss均为1.0.8,skype是www.skype.com 上下载的static binary with Qt 3.2 compiled in(因为非static的版本在笔者的系统中总是segfault。这个稍微大一点,只有Qt的共享库是静态编译的),mplayer是1.0pre6-3.4.3-20050110,realplayer是10.0.2.608(Gold),xmms是1.2.10,stardict 2.4.4(安装了WyabdcRealPeopleTTS),flash plugin 版本7.0.25.0。读者如果用的是KDE,只要软件版本差不多,本文介绍的方法值得一试。注意本文不涉及如何安装alsa驱动的问题,请参阅其他资料。实现多音频还可以使用www.opensound.com的商业驱动。不过是close-sourced,以前还是要收钱的。现在虽然个人使用已经免费(free)了,但总归是不自由(free)的。好了,闲话少叙,我们来切入正题。
先来看一看一些背景知识
首先要了解Linux有两种不同的声音系统
* OSS (Open Sound System) 旧的.对于44100 Hz, 立体声(stereo), 16-bit 音频与一般的声卡及要求它可以工作的很好. 但不足以胜任专业级的音频处理工作。 很多旧的软件都只能使用OSS。 OSS系统中通常有下列设备文件/dev/audio /dev/dsp /dev/midi /dev/mixer /dev/music /dev/sequencer等等
* ALSA (Advanced Linux Sound Architecture)新的。对于很多旧的声卡支持的不是很好,但是支持很多新的声卡以及很多高级的专业的功能,而且具备很强扩展性。可参见http://www.alsa-project.org/alsa-doc/alsa-lib/pcm_plugins.html。现在已经是2.6内核的一部分。ALSA音频系统的设备文件在/dev/snd下。如果装了ALSA oss模拟层,也会有/dev/dsp等文件。
人们往往希望多个进程能同时使用声卡。这就需要将多个进程的声音输入混入一个音频流的能力,即multiplexing。如果希望使用alsa来达到这个要求,针对几种不同情况需要使用不同的技巧。区分的标准主要在于声卡/芯片是否支持硬件混音,程序通过alsa库直接访问声卡,还是通过声音服务器(即artsd之类), 还是通过OSS模拟。
如果硬件支持,alsa驱动支持共享声卡。在硬件不支持的情况下,alsa库也支持共享,只不过需要一些配置。对于使用OSS的程序,aoss能够让它们使用alsa。最后使用声音服务器(arts, esound)的程序,大多数声音服务器能作软件混音并支持alsa输出。所有的情况总结如下:
* 声卡支持硬件混音
* 声卡不支持硬件但是程序使用alsa库来访问声卡
* 程序使用声音服务器访问声卡
* 程序使用OSS API访问声卡
如果声卡支持硬件混音,那么声卡的共享应该不是个问题。
如果声卡不支持硬件混音但程序使用alsa库来访问声卡,那么可以创建一个允许软件混音的.asoundrc。通过使用dmix(允许多个进程使用一个声卡输出),dsnoop(允许多个进程从一个设备录音),asym(将前两者合并成)来实现。这三者都是alsa的插件。下面有具体的例子。这样的程序有aplay, arecord,alsaplayer。很多程序通过配置都可以直接使用alsa库来访问声卡,比如mplayer, xmms。这是最理想的情况。开发比较活跃的自由软件往往都可以做到。
如果程序使用声音服务器,那么可以将声音服务器的输出选择为alsa,然后就可以了。对于本来不使用声音服务器的软件(这样的软件实际上只剩下了使用OSS来访问声卡的,直接使用alsa的不必多此一举)也可以令它们使用声音器。以arts为例,执行时前面加artsdsp就可以做到这一点。Skype在笔者这里正是这样处理的。
如果程序使用OSS API来访问声卡,可以在执行时前面加上aoss来令它们使用alsa。
小结:上面Linux的程序不能截然分成上面三类,因为很多程序都可以选择输出插件,比如xmms。还有,程序能否共享声卡,跟程序本身有很大关系,要看它采用什么样的手段输出。最好的程序就是可以直接使用alsa的了。
下面来看具体该怎么样做
首先要有一个合理配置的.asoundrc,这个是笔者的
- pcm.!dmix {
- type dmix
- ipc_key 5678293
- ipc_key_add_uid yes
- slave {
- pcm "hw:0,0"
- period_time 0
- period_size 2048
- buffer_size 16384
- format S16_LE
- rate 48000
- }
- }
- pcm.!dsnoop {
- type dsnoop
- ipc_key 5778293
- ipc_key_add_uid yes
- slave {
- pcm "hw:0,0"
- period_time 0
- period_size 2048
- buffer_size 16384
- format S16_LE
- rate 48000
- }
- }
- pcm.asymed {
- type asym
- playback.pcm "dmix"
- capture.pcm "dsnoop"
- }
- pcm.!default {
- type plug
- slave.pcm "asymed"
- }
- pcm.dsp0 {
- type plug
- slave.pcm "asymed"
- }
- ctl.mixer0 {
- type hw
- card 0
- }
复制代码
[未完待续....] |
|