简介
FreeBSD声音子系统清晰地将通用声音处理问题与设备特定的问题分离开来。这使得更容易加入对新设备的支持。
一个到数字化声音和混音器函数的系统调用接口(read, write, ioctls)。ioctl命令集合兼容老的OSS 或Voxware接口,允许一般
多媒体应用程序 不加修改地移植。处理声音数据的公共代码(格式转换,虚拟通道)。一个统一的
软件接口,与硬件特定的音频接口模块接口对某些通用硬件接口(ac97)或共享的硬件特定代码(例如:ISA DMA例程)的额外支持。对特定
声卡的支持是通过硬件特定的驱动程序来实现的,这些驱动程序提供通道和混音器接口,插入到通用pcm代码中。
预期的驱动程序编写者当然希望从现有模块开始,并使用那些代码作为最终参考。但是,由于声音代码十分简洁漂亮,这也基本上免除了注释。本文档试图给出框架接口的一个概览,并回答改写现有代码时可能出现的 一些问题.
他们将自己声明为pcm类设备,带有一个 设备私有结构struct snddev_info: static driver_t xxx_driver = { "pcm", xxx_methods, sizeof(struct snddev_info) }; DRIVER_MODULE(snd_xxxpci, pci, xxx_driver, pcm_devclass, 0, 0); MODULE_DEPEND(snd_xxxpci, snd_pcm, PCM_MINVER, PCM_PREFVER,PCM_MAXVER);大多数声音驱动程序需要存储关于其设备的附加私有信息。私有
数据结构通常在连接例程中分配。其地址通过调用 pcm_register()和 mixer_init()传递给 pcm。后面pcm 将此地址作为调用声音驱动程序接口时的参数传递回来。声音驱动程序的连接例程应当通过调用mixer_init() 向pcm声明它的MIXER或AC97 接口。对于MIXER接口,这会接着引起调用 xxxmixer_init()。声音驱动程序的连接例程通过调用 pcm_register(dev, sc, nplay, nrec) 向pcm声明其通用CHANNEL配置,其中 sc是设备
数据结构的地址,在pcm以后的调用
中将会用到它, nplay和nrec是播放和录音通道的数目。声音驱动程序的连接例程通过调用 pcm_addchan()声明它的每个通道对象。这会在 pcm中建立起通道合成,并接着会引起调用 xxxchannel_init() (译注:请参考原文)。声音驱动程序的分离例程在释放其资源之前应当调用 pcm_unregister()。有两种可能的方法来处理非
PnP设备:
使用device_identify()方法(范例:sound/isa/es1888.c)。 device_identify()方法在已知地址探测硬件,如果发现支持的设备就会创建一个新的pcm设备,这个pcm设备接着 会被传递到probe/attach。使用定制内核配置的方法,为pcm设备设置适当的hints(范例: sound/isa/mss.c)。pcm驱动程序应当实现 device_suspend, device_resume和 device_
shutdown例程,这样电源管理和模块卸载就能正确地发挥作用。