2013-06-20 40 views
2

我試圖讓ac程序發送一個正弦波音到我的樹莓pi的音頻輸出使用ALSA Im生成正弦波值的緩衝區,但是當我發送它時它不聽起來像一個正弦 - 但像低中頻率的噪音? 我有一個甩在數組中的值到.csv和Excel中繪製覈實正弦波好 只是想知道有得到的東西在這個方案根本不正確,如果有人能發現一些ID是很感激,感謝從陣列發送正弦波值到音頻輸出

編輯:最終工作代碼在下面!

#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; 

} 

回答

2

已配置樣本格式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); 
} 
+0

謝謝你的幫助 - 是的,我應該已經發現了。但是,我試圖將SND_PCM_FORMAT_U8更改爲SND_PCM_FORMAT_S32 - 因爲我的緩衝區是32位有符號數字?該程序編譯,但它不會運行,只是給出了分段錯誤?我也嘗試了其他語法的SND_PCM_FORMAT,就像你所建議的float選項一樣,但也給出了segmentaiton錯誤 - 任何想法?謝謝! – user2459764

+0

已更新 - 如果要播放整個緩衝區,'&buffer [i]'是開始的錯誤位置。 –

+0

嗨,對不起,在這裏完成c noob ...你是什麼意思混淆了循環初始化緩衝區和循環播放數據?另外如果&緩衝區[我]是錯誤的,我應該如何發送這個數據到聲卡?非常感謝 – user2459764