【问题标题】:Sending sine wave values from array to audio output将正弦波值从数组发送到音频输出
【发布时间】:2013-06-17 13:22:16
【问题描述】:

我试图让一个 c 程序使用 ALSA 向我的树莓派的音频输出发送一个正弦波音调 我正在生成正弦波值的缓冲区,但是当我发送它们时,它听起来不像正弦波——而是像中低频噪声? 我将数组中的值转储到 .csv 中并在 excel 中绘制以验证正弦波是否良好 只是想知道这个程序中有一个根本不正确的东西,如果有人能发现一些东西,非常感谢,谢谢

编辑:下面的最终工作代码!!!

#include <alsa/asoundlib.h>
#include <alsa/pcm.h>
#include <math.h>
#define BUFFER_LEN 48000

static char *device = "default";                       //soundcard
snd_output_t *output = NULL;
float buffer [BUFFER_LEN];


int main(void)
{
    int err;
    int j,k;

    int f = 440;                //frequency
    int fs = 48000;             //sampling frequency

    snd_pcm_t *handle;
    snd_pcm_sframes_t frames;


    // ERROR HANDLING

    if ((err = snd_pcm_open(&handle, device, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {
            printf("Playback open error: %s\n", snd_strerror(err));
            exit(EXIT_FAILURE);
    }

    if ((err = snd_pcm_set_params(handle,
                                  SND_PCM_FORMAT_FLOAT,
                                  SND_PCM_ACCESS_RW_INTERLEAVED,
                                  1,
                                  48000,
                                  1,
                                  500000)) < 0) {   
            printf("Playback open error: %s\n", snd_strerror(err));
            exit(EXIT_FAILURE);


    }

    // SINE WAVE
    printf("Sine tone at %dHz ",f);

        for (k=0; k<BUFFER_LEN; k++){

            buffer[k] = (sin(2*M_PI*f/fs*k));                 //sine wave value generation                        
            }       

        for (j=0; j<5; j++){
            frames = snd_pcm_writei(handle, buffer, BUFFER_LEN);    //sending values to sound driver
            }

    snd_pcm_close(handle);
    return 0;

}

【问题讨论】:

    标签: c alsa trigonometry


    【解决方案1】:

    您已配置样本格式SND_PCM_FORMAT_U8,但实际缓冲区包含 32 位浮点样本。

    使用SND_PCM_FORMAT_FLOAT,或将缓冲区定义为unsigned char 的数组。

    此外,您混淆了初始化buffer的循环和播放数据的循环以及许多字节/帧数,而fs是错误的;你需要使用这样的东西:

    for (i = 0; i < BUFFER_LEN; i++)
        buffer [i] = sin(2*M_PI*f/48000*i);  // sine wave value generation
    
    for (i = 0; i < 10 * 48000 / BUFFER_LEN; i++) { // 10 seconds
        frames = snd_pcm_writei(handle, buffer, BUFFER_LEN);
    
        if (frames < 0)
            frames = snd_pcm_recover(handle, frames, 0);
        if (frames < 0) {
            printf("snd_pcm_writei failed: %s\n", snd_strerror(frames));
            break;
        }
        if (frames > 0 && frames < BUFFER_LEN)
            printf("Short write (expected %li, wrote %li)\n", BUFFER_LEN, frames);
    }
    

    【讨论】:

    • 感谢您的帮助 - 是的,我应该发现这一点。但是,我尝试将 SND_PCM_FORMAT_U8 更改为 SND_PCM_FORMAT_S32 - 因为我的缓冲区是 32 位有符号数字?程序编译但它不会运行,只是给出一个分段错误?我还尝试了 SND_PCM_FORMAT 的其他语法,例如您建议的 float 选项,但也给出了分段错误 - 有什么想法吗?谢谢!
    • 更新——如果你想播放整个缓冲区,&amp;buffer[i] 是错误的起点。
    • 嗨,对不起,在这里完成 c noob ... 你是什么意思混淆了初始化缓冲区的循环和播放数据的循环?此外,如果 &buffer[i] 错误,我应该如何将此数据发送到声卡?非常感谢
    • 嗨,我尝试了您建议的上述代码 - 但是当我 printf 缓冲区时它一直为零?知道为什么会这样吗?谢谢
    • 这假定一个float 缓冲区(sin 返回值在-1..1 范围内)。如果使用整数,则必须对结果进行缩放。
    猜你喜欢
    • 1970-01-01
    • 2011-07-03
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-05-21
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多