C语言 在ubuntu中使用gcc生成特定频率的声音



如何在C/c++中生成特定频率的声音?我运行Ubuntu 10.04并使用gcc。在TurboC上有一个void sound(int frequency)函数。gcc有等价的吗?

下面是一个利用PortAudio库生成给定频率的方形声波的代码。在Linux上用gcc buzzer.c -o buzzer -lportaudio编译。应该编译良好的Windows以及。我不知道sound(int frequency)究竟是如何表现的,但下面应该能够模拟任何老式蜂鸣器的使用。您可能需要portaudio-devel(或同等的Ubuntu, PortAudio -dev?)包和Pulse Audio可能是您的repo中的某个较新版本的PortAudio。编译它不是问题。您可以根据WTFPL许可条款使用下面的代码。:-)(源自PortAudio示例)

#include <stdio.h>
#include <math.h>
#include "portaudio.h"
#include <stdint.h>
#include <unistd.h> // for usleep()
#define SAMPLE_RATE   (44100)
#define FRAMES_PER_BUFFER  (64)
typedef struct
{
    uint32_t total_count;
    uint32_t up_count;
    uint32_t counter;
    uint32_t prev_freq;
    uint32_t freq;
} paTestData;
//volatile int freq = 0;
/* This routine will be called by the PortAudio engine when audio is needed.
** It may called at interrupt level on some machines so don't do anything
** that could mess up the system like calling malloc() or free().
*/
static int patestCallback( const void *inputBuffer, void *outputBuffer,
                           unsigned long framesPerBuffer,
                           const PaStreamCallbackTimeInfo* timeInfo,
                           PaStreamCallbackFlags statusFlags,
                           void *userData )
{
    paTestData *data = (paTestData*)userData;
    uint8_t *out = (uint8_t*)outputBuffer;
    unsigned long i;
    uint32_t freq = data->freq;
    (void) timeInfo; /* Prevent unused variable warnings. */
    (void) statusFlags;
    (void) inputBuffer;
    for( i=0; i<framesPerBuffer; i++ )
    {
        if(data->up_count > 0 && data->total_count == data->up_count) {
            *out++ = 0x00;
            continue;
        }
        data->total_count++;
        if(freq != data->prev_freq) {
            data->counter = 0;
        }
        if(freq) {
            int overflow_max = SAMPLE_RATE / freq;
            uint32_t data_cnt = data->counter % overflow_max;
            if(data_cnt > overflow_max/2)
                *out++ = 0xff;
            else {
                *out++ = 0x00;
            }
            data->counter++;
        }
        else {
            data->counter = 0;
            *out++ = 0;
        }
        data->prev_freq = freq;
    }
    return paContinue;
}
static PaStream *stream;
static paTestData data;

void buzzer_set_freq(int frequency)
{
    data.up_count = 0; // do not stop!
    data.freq = frequency;
}
void buzzer_beep(int frequency, int msecs)
{
    data.total_count = 0;
    data.up_count = SAMPLE_RATE * msecs / 1000;
    data.freq = frequency;
}
int buzzer_start(void)
{
    PaStreamParameters outputParameters;
    PaError err;
    int i;
    err = Pa_Initialize();
    if( err != paNoError ) goto error;
    outputParameters.device = Pa_GetDefaultOutputDevice(); /* default output device */
    outputParameters.channelCount = 1;       /* stereo output */
    outputParameters.sampleFormat = paUInt8; /* 32 bit floating point output */
    outputParameters.suggestedLatency = Pa_GetDeviceInfo( outputParameters.device )->defaultLowOutputLatency;
    outputParameters.hostApiSpecificStreamInfo = NULL;
    err = Pa_OpenStream(
        &stream,
        NULL, /* no input */
        &outputParameters,
        SAMPLE_RATE,
        FRAMES_PER_BUFFER,
        paClipOff,      /* we won't output out of range samples so don't bother clipping them */
        patestCallback,
        &data );
    if( err != paNoError ) goto error;
    err = Pa_StartStream( stream );
    if( err != paNoError ) goto error;
    return err;
error:
    Pa_Terminate();
    fprintf( stderr, "An error occured while using the portaudio streamn" );
    fprintf( stderr, "Error number: %dn", err );
    fprintf( stderr, "Error message: %sn", Pa_GetErrorText( err ) );
    return err;
}
int buzzer_stop() 
{
    PaError err = 0;
    err = Pa_StopStream( stream );
    if( err != paNoError ) goto error;
    err = Pa_CloseStream( stream );
    if( err != paNoError ) goto error;
    Pa_Terminate();
    return err;
error:
    Pa_Terminate();
    fprintf( stderr, "An error occured while using the portaudio streamn" );
    fprintf( stderr, "Error number: %dn", err );
    fprintf( stderr, "Error message: %sn", Pa_GetErrorText( err ) );
    return err;
}
void msleep(int d){
    usleep(d*1000);
}
int main(void)
{
    // notes frequency chart: http://www.phy.mtu.edu/~suits/notefreqs.html
    buzzer_start();
    buzzer_set_freq(261);
    msleep(250);
    buzzer_set_freq(293);
    msleep(250);
    buzzer_set_freq(329);
    msleep(250);
    buzzer_set_freq(349);
    msleep(250);
    buzzer_set_freq(392);
    msleep(250);
    buzzer_set_freq(440);
    msleep(250);
    buzzer_set_freq(494);
    msleep(250);
    buzzer_beep(523, 200);
    msleep(250);
    buzzer_stop();
    return 0;
}

最新更新