原创:ALSA音频编程自我分析

最近搞了一阵子的ALSA音频编程的东西,其实单是说其编程结构的确是比较简单的,相信很多朋友和我一样都在网络上搜索到很多相关资料,从ALSA的驱动-----》ALSA的lib库(提供了编程的API)-----》ALSA的utils,这三部分的确组成了ALSA开发中不可或缺的主要部分,同时ALSA项目的开放源码特性,使得开发者有更多的源码可以参考,其中ALSA的utils中提供的源码就是很好的材料。不过有了这些只是提供了能够参阅的资料,编程开发的事情,真的没有一定就能成,有些问题出来还是需要解决时间的。有些问题涉及到开发者自身的具体操作环境,所以很多问题出了还是不能在网上找到合适的答案,只有自己慢慢摸索。我从做些录放音小程序到时先语音的传输过程中,也曾面临着许多问题,但是要相信自己一定能解决并为之去查阅资料、去调试程序。所以出现问题是正常的,问题也是我们能解决问题的。总结一下这么久的工作,细节很多,但是有个问题在ALSA中很重要,很多问题就是由于这个问题导致的,那就是对音频文件的认识以及对该设备文件的参数设置。这个设置又涉及到硬件参数设置和软件参数设置。这个参数设置没对的话,很多莫名其妙的问题都会出来,让我们真的是摸不着头脑。同时也要说明的是同样的代码在不同的操作系统上会有不同的反应,这也是正常的,这就需要我们实时的调整了。这里就最普遍的参数设置问题给大家提出警示。不管是编写复杂程序还是编写测试小程序,只要参数设置正确无误了,问题就会少很多。因此,将aply.c文件中的设置参数的函数摘出来给分析一下,希望给继续做这方面的朋友做个分享,少走点我走过的弯路,同时也希望大家能够将自己的所学贡献出来,相互交流,共同进步!

下面首先大体介绍一下,接着会以注解的形式来重点分析我认为比较容易忽视,但是有很重要的参数设置。同时申明这些知识自己的浅薄一点认识,如果有不对的地方还请读者指正,不胜感激!

首先是硬件参数设置,如果设置好了硬件参数,做单方面的录音或放音是不会有大问题的,但是如果要涉及到同时的录放音交替,就必须好好的设置下软参数了,这样才能使得听起来的语音流畅。

static void set_params(void)
{
 snd_pcm_hw_params_t *params;
 snd_pcm_sw_params_t *swparams;
 snd_pcm_uframes_t buffer_size;
 int err;
 size_t n;
 snd_pcm_uframes_t xfer_align;
 unsigned int rate;
 snd_pcm_uframes_t start_threshold, stop_threshold;//设置软参数主要这两个参数要设置
 snd_pcm_hw_params_alloca(&params);
 snd_pcm_sw_params_alloca(&swparams);
 err = snd_pcm_hw_params_any(handle, params);
 if (err < 0) {
  error(_("Broken configuration for this PCM: no configurations available"));
  exit(EXIT_FAILURE);
 }
 if (mmap_flag) {
  snd_pcm_access_mask_t *mask = alloca(snd_pcm_access_mask_sizeof());
  snd_pcm_access_mask_none(mask);
  snd_pcm_access_mask_set(mask, SND_PCM_ACCESS_MMAP_INTERLEAVED);
  snd_pcm_access_mask_set(mask, SND_PCM_ACCESS_MMAP_NONINTERLEAVED);
  snd_pcm_access_mask_set(mask, SND_PCM_ACCESS_MMAP_COMPLEX);
  err = snd_pcm_hw_params_set_access_mask(handle, params, mask);
 } else if (interleaved)
  err = snd_pcm_hw_params_set_access(handle, params,
         SND_PCM_ACCESS_RW_INTERLEAVED);
 else
  err = snd_pcm_hw_params_set_access(handle, params,
         SND_PCM_ACCESS_RW_NONINTERLEAVED);
 if (err < 0) {
  error(_("Access type not available"));
  exit(EXIT_FAILURE);
 }
 err = snd_pcm_hw_params_set_format(handle, params, hwparams.format);
 if (err < 0) {
  error(_("Sample format non available"));
  exit(EXIT_FAILURE);
 }
 err = snd_pcm_hw_params_set_channels(handle, params, hwparams.channels);
 if (err < 0) {
  error(_("Channels count non available"));
  exit(EXIT_FAILURE);
 }

#if 0
 err = snd_pcm_hw_params_set_periods_min(handle, params, 2);
 assert(err >= 0);
#endif
 rate = hwparams.rate;
 err = snd_pcm_hw_params_set_rate_near(handle, params, &hwparams.rate, 0);
 assert(err >= 0);
 if ((float)rate * 1.05 < hwparams.rate || (float)rate * 0.95 > hwparams.rate) {
  if (!quiet_mode) {
   char plugex[64];
   const char *pcmname = snd_pcm_name(handle);
   fprintf(stderr, _("Warning: rate is not accurate (requested = %iHz, got = %iHz)\n"), rate, hwparams.rate);
   if (! pcmname || strchr(snd_pcm_name(handle), ':'))
    *plugex = 0;
   else
    snprintf(plugex, sizeof(plugex), "(-Dplug:%s)",
      snd_pcm_name(handle));
   fprintf(stderr, _("         please, try the plug plugin %s\n"),
    plugex);
  }
 }
 rate = hwparams.rate;
 if (buffer_time == 0 && buffer_frames == 0) {
  err = snd_pcm_hw_params_get_buffer_time_max(params,
           &buffer_time, 0);
  assert(err >= 0);
  if (buffer_time > 500000)//这是指具体的环形缓从区大小,自己认为是以字节为单位的
   buffer_time = 500000;
 }
 if (period_time == 0 && period_frames == 0) {
  if (buffer_time > 0)
   period_time = buffer_time / 4;
  else
   period_frames = buffer_frames / 4;
 }
 if (period_time > 0)
  err = snd_pcm_hw_params_set_period_time_near(handle, params,
            &period_time, 0);
 else
  err = snd_pcm_hw_params_set_period_size_near(handle, params,
            &period_frames, 0);
 assert(err >= 0);
 if (buffer_time > 0) {
  err = snd_pcm_hw_params_set_buffer_time_near(handle, params,
            &buffer_time, 0);
 } else {
  err = snd_pcm_hw_params_set_buffer_size_near(handle, params,
            &buffer_frames);
 }
 assert(err >= 0);
 err = snd_pcm_hw_params(handle, params);
 if (err < 0) {
  error(_("Unable to install hw params:"));
  snd_pcm_hw_params_dump(params, log);
  exit(EXIT_FAILURE);
 }
 snd_pcm_hw_params_get_period_size(params, &chunk_size, 0);
 snd_pcm_hw_params_get_buffer_size(params, &buffer_size);
 if (chunk_size == buffer_size) {
  error(_("Can't use period equal to buffer size (%lu == %lu)"),
        chunk_size, buffer_size);
  exit(EXIT_FAILURE);
 }
 snd_pcm_sw_params_current(handle, swparams);
 err = snd_pcm_sw_params_get_xfer_align(swparams, &xfer_align);
 if (err < 0) {
  error(_("Unable to obtain xfer align\n"));
  exit(EXIT_FAILURE);
 }
 if (sleep_min)
  xfer_align = 1;
 err = snd_pcm_sw_params_set_sleep_min(handle, swparams,
           sleep_min);
 assert(err >= 0);
 if (avail_min < 0)    
  n = chunk_size;//如果语音数据的长度小于这个值得话会播放不出来的
 else
  n = (double) rate * avail_min / 1000000;//注意,将后面的数字强制转换为n的数据类型会减少出错
 err = snd_pcm_sw_params_set_avail_min(handle, swparams, n);

 /* round up to closest transfer boundary */
 n = (buffer_size / xfer_align) * xfer_align;//这部分就是软参数的设置了,主要设置了两个阈值,主要问题是注意数据类型的转换,设置合适的阈值,合适的阈值不是固定的,是一句实际情况来定的。
 if (start_delay <= 0) {
  start_threshold = n + (double) rate * start_delay / 1000000;
 } else
  start_threshold = (double) rate * start_delay / 1000000;
 if (start_threshold < 1)
  start_threshold = 1;
 if (start_threshold > n)
  start_threshold = n;
 err = snd_pcm_sw_params_set_start_threshold(handle, swparams, start_threshold);
 assert(err >= 0);
 if (stop_delay <= 0) 
  stop_threshold = buffer_size + (double) rate * stop_delay / 1000000;
 else
  stop_threshold = (double) rate * stop_delay / 1000000;
 err = snd_pcm_sw_params_set_stop_threshold(handle, swparams, stop_threshold);
 assert(err >= 0);

 err = snd_pcm_sw_params_set_xfer_align(handle, swparams, xfer_align);
 assert(err >= 0);

 if (snd_pcm_sw_params(handle, swparams) < 0) {
  error(_("unable to install sw params:"));
  snd_pcm_sw_params_dump(swparams, log);
  exit(EXIT_FAILURE);
 }

 if (verbose)
  snd_pcm_dump(handle, log);

 bits_per_sample = snd_pcm_format_physical_width(hwparams.format);
 bits_per_frame = bits_per_sample * hwparams.channels;
 chunk_bytes = chunk_size * bits_per_frame / 8;
 audiobuf = realloc(audiobuf, chunk_bytes);
 if (audiobuf == NULL) {
  error(_("not enough memory"));
  exit(EXIT_FAILURE);
 }
 // fprintf(stderr, "real chunk_size = %i, frags = %i, total = %i\n", chunk_size, setup.buf.block.frags, setup.buf.block.frags * chunk_size);
}

说起来就这么简单的一点,但是要做好还是需要细致和不断尝试。很多实例程序源码是很值得参考的,至少在流程和结构上还比较规范。在懂得流程的基础上,我们再根据条件来调整细节,这样我们就有希望作出符合要求的程序来,希望做这一块的朋友,大家都合理作出更好的东西,期待ing!

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值