[英]Strange sine wave on output of an alsa linux c program
我正在 ubuntu 上学习 alsa 编程。
我正在尝试将正弦波输出到我的笔记本电脑声卡的线路输出,然后通过音频电缆重定向到线路输入(麦克风)。
___LINE_IN(microphone)\
/ \
| \ _____________
| |soundcard-pc |
cable /
| /
\ ___LINE_OUT(speakers)/
我正在使用此代码
#include<stdio.h>
#include<stdlib.h>
#include<alsa/asoundlib.h>
#include<math.h>
#define SIZEBUF 2048
int main(void)
{
int i;
int err;
double x;
double cost;
double frequency=500;
unsigned int rate=44100;
short buf[SIZEBUF];
snd_pcm_t *phandle;
snd_pcm_hw_params_t *hw_params;
snd_pcm_open(&phandle, "default", SND_PCM_STREAM_PLAYBACK, 0);
snd_pcm_hw_params_malloc(&hw_params);
snd_pcm_hw_params_any(phandle, hw_params);
if ((err = snd_pcm_hw_params_set_access(phandle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {
printf("Cannot set access type.\n");
exit(1);
}
snd_pcm_hw_params_set_format(phandle, hw_params, SND_PCM_FORMAT_S16_LE);
snd_pcm_hw_params_set_rate_near(phandle, hw_params, &rate, 0);
snd_pcm_hw_params_set_channels(phandle, hw_params, 1);
snd_pcm_hw_params(phandle, hw_params);
snd_pcm_hw_params_free(hw_params);
snd_pcm_prepare(phandle);
cost = 2.0 * M_PI * frequency / (double)rate;
printf("cost=%f.\n", cost);
for (i = 1 ; i < SIZEBUF ; i++) {
x = sin(i * cost);
buf[i] = (short)(32767 * x + 32768);
}
for (i = 0 ; i < 50 ; i++) {
snd_pcm_writei(phandle, buf, SIZEBUF);
}
snd_pcm_close(phandle);
exit(0);
}
我正在使用大胆来查看波浪,但它看起来很奇怪,就像在这张图片中一样
它似乎没有正弦波的行为。 为什么?
您正在使用类型为short
样本,并且已将它们正确声明为S16_LE
。 但是,您的代码随后会像未签名一样计算这些值:
buf[i] = (short)(32767 * x + 32768);
您不需要偏移量; 对于带符号的样本,零实际上是零。 用这个:
buf[i] = (short)(32767 * x);
要将浮点样本转换为 SND_PCM_FORMAT_S16_LE,请执行以下操作:
#include <endian.h>
#include <math.h>
#include <stdint.h>
// assume -1.0 <= x <= 1.0
int16_t float_to_s16le (float x)
{
int16_t s16 = trunc (x * 32767); // rounds toward zero
return htole16 (s16);
}
一些备注:
htole16
或等效项)。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.