PCM音频设备的操纵函数
副标题#e#
对音频设备的操纵主要是初始化音频设备以及往音频设备发送 PCM(Pulse Code Modulation)数据。为了利便,本文利用 ALSA(Advanced Linux Sound Architecture)提供的库和驱动。在编译和运行本文中的 MP3 流媒体播放器的时候,必需先安装 ALSA 相关的文件。
本文用到的主要对 PCM 设备操纵的函数分为 PCM 设备初始化的函数以及 PCM 接口的一些操纵函数。
PCM 硬件设备参数配置和初始化的函数有:
int snd_pcm_hw_params_malloc (snd_pcm_hw_params_t **ptr) int snd_pcm_hw_params_any (snd_pcm_t *pcm, snd_pcm_hw_params_t *params) void snd_pcm_hw_params_free (snd_pcm_hw_params_t *obj) int snd_pcm_hw_params_set_access ( snd_pcm_t *pcm, snd_pcm_hw_params_t *params, snd_pcm_access_t _access) int snd_pcm_hw_params_set_format ( snd_pcm_t *pcm, snd_pcm_hw_params_t *params, snd_pcm_format_t val) int snd_pcm_hw_params_set_channels(snd_pcm_t *pcm, snd_pcm_hw_params_t *params, unsigned int val) int snd_pcm_hw_params_set_rate_near(snd_pcm_t *pcm, snd_pcm_hw_params_t *params, unsigned int *val, int *dir)
PCM 接口函数有:
int snd_pcm_hw_params (snd_pcm_t *pcm, snd_pcm_hw_params_t *params) int snd_pcm_prepare (snd_pcm_t *pcm) int snd_pcm_open (snd_pcm_t **pcm, const char *name, snd_pcm_stream_t stream, int mode) int snd_pcm_close (snd_pcm_t *pcm) snd_pcm_sframes_t snd_pcm_writei (snd_pcm_t *pcm, const void *buffer, snd_pcm_uframes_t size)
这些函数用到了 snd_pcm_hw_params_t 布局,此布局包括用来播放 PCM 数据流的硬件信息设置。在往音频设备(声卡)写入音频数据之前,必需配置会见范例、采样名目、采样率、声道数等。
首先利用 snd_pcm_open () 打开 PCM 设备,在 ALSA 中,PCM 设备都有名字与之对应。好比我们可以界说 PCM 设备名字为 char *pcm_name = "plughw:0,0"。 最重要的 PCM 设备接口是“plughw”以及“hw”接口。 利用“plughw”接口,措施员不必过多体贴硬件,并且假如配置的设置参数和实际硬件支持的参数纷歧致,ALSA 会自动转换数据。假如利用“hw”接口,我们就必需检测硬件是否支持配置的参数了。Plughw 后头的两个数字别离暗示设备号和次设备(subdevice)号。
snd_pcm_hw_params_malloc( ) 在栈中分派 snd_pcm_hw_params_t 布局的空间,然后利用 snd_pcm_hw_params_any( ) 函数用声卡的全设置空间参数初始化已经分派的 snd_pcm_hw_params_t 布局。snd_pcm_hw_params_set_access ( ) 配置会见范例,常用会见范例的宏界说有:
SND_PCM_ACCESS_RW_INTERLEAVED
交织会见。在缓冲区的每个 PCM 帧都包括所有配置的声道的持续的采样数据。好比声卡要播放采样长度是 16-bit 的 PCM 立体声数据,暗示每个 PCM 帧中有 16-bit 的左声道数据,然后是 16-bit 右声道数据。
SND_PCM_ACCESS_RW_NONINTERLEAVED
非交织会见。每个 PCM 帧只是一个声道需要的数据,假如利用多个声道,那么第一帧是第一个声道的数据,第二帧是第二个声道的数据,依此类推。
函数 snd_pcm_hw_params_set_format() 配置数据名目,主要节制输入的音频数据的范例、无标记照旧有标记、是 little-endian 照旧 bit-endian。好比对付 16-bit 长度的采样数据可以配置为:
SND_PCM_FORMAT_S16_LE 有标记16 bit Little Endian SND_PCM_FORMAT_S16_BE 有标记16 bit Big Endian SND_PCM_FORMAT_U16_LE 无标记16 bit Little Endian SND_PCM_FORMAT_U16_BE 无标记 16 bit Big Endian 好比对付 32-bit 长度的采样数据可以配置为: SND_PCM_FORMAT_S32_LE 有标记32 bit Little Endian SND_PCM_FORMAT_S32_BE 有标记32 bit Big Endian SND_PCM_FORMAT_U32_LE 无标记32 bit Little Endian SND_PCM_FORMAT_U32_BE 无标记 32 bit Big Endian
#p#副标题#e#
函数 snd_pcm_hw_params_set_channels() 配置音频设备的声道,常见的就是单声道和立体声,假如是立体声,配置最后一个参数为2。snd_pcm_hw_params_set_rate_near () 函数配置音频数据的最靠近方针的采样率。snd_pcm_hw_params( ) 从设备设置空间选择一个设置,让函数 snd_pcm_prepare() 筹备好 PCM 设备,以便写入 PCM 数据。snd_pcm_writei() 用来把交织的音频数据写入到音频设备。
初始化 PCM 设备的例程如下:
初始化 PCM 设备的例程
/* open a PCM device */ int open_device(struct mad_header const *header) { int err; snd_pcm_hw_params_t *hw_params; char *pcm_name = "plughw:0,0"; int rate = header->samplerate; int channels = 2; if (header->mode == 0) { channels = 1; } else { channels = 2; } if ((err = snd_pcm_open (&playback_handle, pcm_name, SND_PCM_STREAM_PLAYBACK, 0)) < 0) { printf("cannot open audio device %s (%s)\n", pcm_name, snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params_malloc (&hw_params)) < 0) { printf("cannot allocate hardware parameter structure (%s)\n", snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params_any (playback_handle, hw_params)) < 0) { printf("cannot initialize hardware parameter structure (%s)\n", snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params_set_access (playback_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) { printf("cannot set access type (%s)\n", snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params_set_format (playback_handle, hw_params, SND_PCM_FORMAT_S32_LE)) < 0) { printf("cannot set sample format (%s)\n", snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params_set_rate_near (playback_handle, hw_params, &rate, 0)) < 0) { printf("cannot set sample rate (%s)\n", snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params_set_channels (playback_handle, hw_params, channels)) < 0) { printf("cannot set channel count (%s)\n", snd_strerror (err)); return -1; } if ((err = snd_pcm_hw_params (playback_handle, hw_params)) < 0) { printf("cannot set parameters (%s)\n", snd_strerror (err)); return -1; } snd_pcm_hw_params_free (hw_params); if ((err = snd_pcm_prepare (playback_handle)) < 0) { printf("cannot prepare audio interface for use (%s)\n", snd_strerror (err)); return -1; } return 0; }
#p#分页标题#e#
这里设置的 PCM 名目是 SND_PCM_FORMAT_S32_LE,采样的名目是每个采样有 32-bit 的数据,数据凭据 little-endian 存放。假如通过 mad_frame_decode() 函数获得 PCM 数据后,要求每个采样数据只占 16-bit,需要把数据举办MAD的定点范例到 signed short 范例举办转换。那么,PCM 数据如何写入声卡中呢?函数实现例程如下所示:
PCM 数据写入声卡函数实现例程
while (nsamples--) { /* nsamples 是采样的数目 */ signed int sample; sample = pcm->samples[0][j]; *(OutputPtr++) = sample & 0xff; *(OutputPtr++) = (sample >> 8); *(OutputPtr++) = (sample >> 16); *(OutputPtr++) = (sample >> 24); if (nchannels == 2) { sample = pcm->samples[1][j]; *(OutputPtr++) = sample & 0xff; *(OutputPtr++) = sample >> 8; *(OutputPtr++) = (sample >> 16); *(OutputPtr++) = (sample >> 24); } j++; } if ((err = snd_pcm_writei (playback_handle, buf, samples)) < 0) { err = xrun_recovery(playback_handle, err); if (err < 0) { printf("Write error: %s\n", snd_strerror(err)); return -1; } }
这里用到了 http://www.alsa-project.org/ 关于 ALSA 文档中的例子函数 xrun_recovery( )。具体例子请拜见 http://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_8c-example.html。利用此函数的目标是制止呈现由于网络原因,声卡不能实时获得音频数据而使得 snd_pcm_writei() 不能正常持续事情。实际上在 xrun_recovery( ) 中,又挪用 snd_pcm_prepare() 和 snd_pcm_resume() 以实现能“规复错误”的成果。-EPIPE 错误暗示应用措施没有实时把 PCM 采样数据送入ASLA 库。xrun_recovery() 函数如下所示:
xrun_recovery() 函数
int xrun_recovery(snd_pcm_t *handle, int err) { if (err == -EPIPE) { /* under-run */ err = snd_pcm_prepare(handle); if (err < 0) printf("Can't recovery from underrun, prepare failed: %s\n", snd_strerror(err)); return 0; } else if (err == -ESTRPIPE) { while ((err = snd_pcm_resume(handle)) == -EAGAIN) sleep(1); /* wait until the suspend flag is released */ if (err < 0) { err = snd_pcm_prepare(handle); if (err < 0) printf("Can't recovery from suspend, prepare failed: %s\n", snd_strerror(err)); } return 0; } return err; }
#p#分页标题#e#
知道了详细的音频设备操纵要领,就该利用 MAD 提供的函数详细实现解码了。函数 mp3_decode_buf( ) 提供了利用 libmad 解码的要领。首先挪用 mad_stream_buffer() 函数把 MP3 流数据和 decode_stream 关联,然后开始轮回解码数据。假如在解码数据进程中,有不完整 PCM 数据帧,那么 decode_stream.error 的值就是 MAD_ERROR_BUFLEN,且 decode_stream.next_frame 不为 NULL。这时候,把剩余的未解码的数据再拷贝到数据解码缓冲区里。 mad_frame_decode( ) 函数从 decode_stream 中获得 PCM 数据。
mad_frame_decode( ) 函数从 decode_stream 中获得 PCM 数据
int mp3_decode_buf(char *input_buf, int size) { int decode_over_flag = 0; int remain_bytes = 0; int ret_val = 0; mad_stream_buffer(&decode_stream, input_buf, size); decode_stream.error = MAD_ERROR_NONE; while (1) { if (decode_stream.error == MAD_ERROR_BUFLEN) { if (decode_stream.next_frame != NULL) { remain_bytes = decode_stream.bufend - decode_stream.next_frame; memcpy(input_buf, decode_stream.next_frame, remain_bytes); return remain_bytes; } } ret_val = mad_frame_decode(&decode_frame, &decode_stream); /* 省略部门代码 */ ... if (ret_val == 0) { if (play_frame(&decode_frame) == -1) { return -1; } } /* 后头代码省略 */ ... } return 0; }
recommend from :http://www.ibm.com/developerworks/cn/linux/l-cn-libmadmp3player/index.html
本文出自 “驿落薄暮” 博客,请务必保存此出处http://yiluohuanghun.blog.51cto.com/3407300/868048