ALSA:snd_pcm_writei返回EAGAIN

时间:2018-03-31 21:47:10

标签: linux audio alsa

我很难理解我在音频播放程序中做错了什么。 我有一个线程从其他线程获取缓冲区,并以与alsa示例程序相同的方式播放它们:https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_8c-example.html 我指的是write_loop()函数。这是pcm.c示例程序的设备配置设置(snd_pcm_dump()的输出):

ALSA <-> PulseAudio PCM I/O Plugin
Its setup is:
  stream       : PLAYBACK
  access       : RW_INTERLEAVED
  format       : S16_LE
  subformat    : STD
  channels     : 1
  rate         : 44100
  exact rate   : 44100 (44100/1)
  msbits       : 16
  buffer_size  : 22050
  period_size  : 4410
  period_time  : 100000
  tstamp_mode  : NONE
  tstamp_type  : GETTIMEOFDAY
  period_step  : 1
  avail_min    : 4410
  period_event : 0
  start_threshold  : 22050
  stop_threshold   : 22050
  silence_threshold: 0
  silence_size : 0
  boundary     : 6206523236469964800

我看到在snd_pcm_writei()周围放置一些printf()是它连续执行5次,每次下一个循环snd_pcm_writei()需要100ms才能完成。这正是我期待看到的。

这是我的程序的设备设置:

ALSA <-> PulseAudio PCM I/O Plugin
Its setup is:
  stream       : PLAYBACK
  access       : RW_INTERLEAVED
  format       : FLOAT_LE
  subformat    : STD
  channels     : 1
  rate         : 44100
  exact rate   : 44100 (44100/1)
  msbits       : 32
  buffer_size  : 13230
  period_size  : 4410
  period_time  : 100000
  tstamp_mode  : NONE
  tstamp_type  : GETTIMEOFDAY
  period_step  : 1
  avail_min    : 4410
  period_event : 0
  start_threshold  : 4410
  stop_threshold   : 13230
  silence_threshold: 0
  silence_size : 0
  boundary     : 7447827883763957760

snd_pcm_writei()会运行5次(这没关系)但在此之后每个新循环都会立即返回-EAGAIN。连续重试100ms(100%cpu使用率)播放相同的缓冲区最终播放,snd_pcm_writei()返回一个正数,对于下一个音频缓冲区,我立即得到-EAGAIN,持续100ms;等等。然而,音频播放很好。

我不明白为什么它不会等待100毫秒来播放新缓冲区而不是立即返回-EAGAIN(在ALSA文档中找不到任何关于snd_pcm_writei()返回-EAGAIN的文档)。

提前感谢您的帮助!

1 个答案:

答案 0 :(得分:2)

PCM设备可以处于阻塞模式(等待)或非阻塞模式(返回-EAGAIN)。

调用snd_pcm_open()snd_pcm_nonblock()时,可以使用标记设置此模式。