我正在尝试使用PulseAudio实时捕获麦克风信号。该程序用C语言编写,并为此使用PulseAudio Simple API。不幸的是,我请求的音频缓冲区不包含任何信号。我的代码有问题或者设备源无法识别。我使用gstreamer的pulsesrc和pulsesink运行了一些测试,这些程序正常工作。我还测试了以下命令,该命令也有效:
parec -d alsa_input.usb-041e_30d3_121023000184-00-U0x41e0x30d3.analog-mono | sox -t raw -r 44100 -sLb 16 -c 2 - /home/roos/Arbeitsfläche/pulsetest.wav
在我的第二张卡片上,它也起作用:
parec -d alsa_input.usb-Creative_Technology_Ltd_Sound_Blaster_X-Fi_Go__Pro_00173634-00-Pro_1.analog-stereo | sox -t raw -r 44100 -sLb 16 -c 2 - /home/roos/Arbeitsfläche/pulsetest.wav
这是应该起作用的代码:
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <math.h>
#include <pulse/simple.h>
#include <pulse/error.h>
#define BUFSIZE 1024
#define SAMPLE_BITS 16
/* A simple routine calling UNIX write() in a loop */
static ssize_t loop_write(int fd, const uint8_t *data, size_t size)
{
int i = 0;
for (i = 0; i < size; i += 2)
{
// put two bytes into one __signed__ integer
int16_t val = data[i] + ((uint32_t)data[i+1] << 8);
printf("%d", val);
}
return size;
}
int main(int argc, char*argv[])
{
char *device = "alsa_output.usb-Creative_Technology_Ltd_Sound_Blaster_X-Fi_Go__Pro_00173634-00-Pro_1.analog-stereo";
// The sample type to use
static const pa_sample_spec ss = {
.format = PA_SAMPLE_S16LE,
.rate = 44100,
.channels = 2
};
pa_simple *s = NULL;
int ret = 1;
int error;
// Create the recording stream
if (!(s = pa_simple_new(NULL, argv[0], PA_STREAM_RECORD, NULL, "record", &ss, NULL, NULL, &error))) {
fprintf(stderr, __FILE__": pa_simple_new() failed: %s\n", pa_strerror(error));
goto finish;
}
for (;;) {
uint8_t buf[BUFSIZE];
// Record some data ...
if (pa_simple_read(s, buf, sizeof(buf), &error) < 0) {
fprintf(stderr, __FILE__": pa_simple_read() failed: %s\n", pa_strerror(error));
goto finish;
}
// And write it to STDOUT
if (loop_write(STDOUT_FILENO, buf, sizeof(buf)) != sizeof(buf)) {
fprintf(stderr, __FILE__": write() failed: %s\n", strerror(errno));
goto finish;
}
}
ret = 0;
finish:
if (s)
pa_simple_free(s);
return 0;
}
方法“ssize_t loop_write”接收缓冲区,由于它是一个16位的小端字节数组,因此我将两个字节合并为一个16位整数。这意味着振幅(由变量“val”表示)应该在0到32768之间。但是到目前为止,它全为0。所以,我主要关心的两个问题是设备源(imo似乎更可能)和转换为一个整数值。
您对此有何建议?先感谢您!
编辑/更新:好的,我不知道我做了什么-但是如果我通过特定设备,我现在会收到以下消息:
pa_simple_new() failed: Connection refused
当我传递NULL时,它适用于默认声卡。仍然可以使用我先前描述的命令行命令。任何线索,这可能是什么?
最佳答案
好的,刚弄清楚-我犯了一个很愚蠢的错误。
我将设备传递给第一个功能参数,而不是第四个。
这样吧
if (!(s = pa_simple_new(NULL, argv[0], PA_STREAM_RECORD, device, "record", &ss, NULL, NULL, &error)))
代替这个
if (!(s = pa_simple_new(device, argv[0], PA_STREAM_RECORD, NULL, "record", &ss, NULL, NULL, &error)))