3

当我尝试使用 ALSA 库时,我的 C 代码出现了一个奇怪的行为。我使用此代码生成设备的 sid

snd_mixer_selem_id_t*
getSid(){
  snd_mixer_selem_id_t *sid;
  snd_mixer_selem_id_alloca(&sid);
  snd_mixer_selem_id_set_index(sid,0);
  snd_mixer_selem_id_set_name(sid, selem_name);

  return sid;
}

然后我尝试通过

snd_mixer_t *handle = getHandle();
snd_mixer_selem_id_t *sid = getSid();

snd_mixer_elem_t *elem = snd_mixer_find_selem(handle,sid);

getHandle()是获取卡处理程序的等效函数,但这个函数正在工作。奇怪的是,如果我在使用函数之前直接使用函数的代码,snd_mixer_find_selem它就可以工作。我不得不说我是 C 新手,这是我的第一个项目之一,所以这可能是初学者对指针的错误。

错误信息是simple.c:282: snd_mixer_selem_get_playback_volume_range: Assertion 'elem' failed.

那么有人知道问题是什么吗?

这是我使用的代码的最小示例:

#include <alsa/asoundlib.h>

static const char *selem_name = "Master";
static const char *card = "default";

snd_mixer_t*
getHandle(){
  snd_mixer_t *handle;

  snd_mixer_open(&handle, 0);
  snd_mixer_attach(handle, card);
  snd_mixer_selem_register(handle, NULL, NULL);
  snd_mixer_load(handle);

  return handle;
}

snd_mixer_selem_id_t*
getSid(){
  snd_mixer_selem_id_t *sid;
  snd_mixer_selem_id_alloca(&sid);
  snd_mixer_selem_id_set_index(sid,0);
  snd_mixer_selem_id_set_name(sid, selem_name);

  return sid;
}

void
incMasterVol(long step){ // Step is a value in [-100,100]
  long min,max,curr;

  snd_mixer_t *handle = getHandle();

/*snd_mixer_open(&handle, 0);
  snd_mixer_attach(handle, card);
  snd_mixer_selem_register(handle, NULL, NULL);
  snd_mixer_load(handle);*/

  snd_mixer_selem_id_t *sid;
  snd_mixer_selem_id_alloca(&sid);
  snd_mixer_selem_id_set_index(sid,0);
  snd_mixer_selem_id_set_name(sid, selem_name);

  snd_mixer_elem_t *elem = snd_mixer_find_selem(handle,sid);

  snd_mixer_selem_get_playback_volume_range(elem, &min, &max);
  snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_SIDE_LEFT, &curr);
  curr = curr * 100/max;
  if(curr + step > 100)
    curr = 100;
  else if(curr + step < 0)
    curr = 0;
  else
    curr += step;
  snd_mixer_selem_set_playback_volume_all(elem, curr * max / 100);

  snd_mixer_close(handle);
}

void
toggleMasterVol(){
 int swiVal;
 snd_mixer_t *handle = getHandle();
 snd_mixer_elem_t *elem = snd_mixer_find_selem(handle,getSid());

 snd_mixer_selem_get_playback_switch(elem, SND_MIXER_SCHN_SIDE_LEFT, &swiVal);
 snd_mixer_selem_set_playback_switch_all(elem, !swiVal);

 snd_mixer_close(handle);
}

int
main(int argc, char *argv[]){
  incMasterVol(5);
}
4

2 回答 2

6

我认为这源于alloca()系统调用

alloca在堆栈帧中分配,因此当getSid()返回时释放分配的变量并最终得到一个摇晃指针......(当你将它放在主函数中时它会起作用,因为分配alloca主激活记录上的空间)

这只是一个建议,但我认为您应该使用malloc(在堆上分配空间,因此在您调用free()它之前分配动态内存)

试试吧snd_mixer_selem_id_malloc():)

于 2013-03-01T13:16:40.083 回答
1

在遇到同样的问题并进行了一些搜索之后,我给你一个提示:更改 的值selem_name,如有必要,更改card! 通过发出以下系统命令找出正确的卡名:

cat /proc/asound/cards

然后,使用alsamixer确定所需音量控制的名称。就我而言,这是Digital,但请自己尝试。

于 2019-12-06T23:29:25.273 回答