diff --git a/src/beep.c b/src/beep.c new file mode 100644 index 0000000..408acb5 --- /dev/null +++ b/src/beep.c @@ -0,0 +1,181 @@ +#include +#include +#include +#include // for usleep() +#include +#include "beep.h" + + +// http://stackoverflow.com/questions/7678470/generating-sound-of-a-particular-frequency-using-gcc-in-ubuntu + +static PaStream *stream; +static paTestData data; + + +/* 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; iup_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; +} + +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; + + 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 stream\n" ); + fprintf( stderr, "Error number: %d\n", err ); + fprintf( stderr, "Error message: %s\n", 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 stream\n" ); + fprintf( stderr, "Error number: %d\n", err ); + fprintf( stderr, "Error message: %s\n", Pa_GetErrorText( err ) ); + return err; +} +void msleep(int d){ + usleep(d*1000); +} + + +int beep(double freq_hz, double duration_sec) +{ + buzzer_set_freq(freq_hz); + msleep(duration_sec*1000.); + return 0; +} +int beep_init() +{ + buzzer_start(); + return 0; +} + + +int beep_test(void) +{ + + // notes frequency chart: http://www.phy.mtu.edu/~suits/notefreqs.html + + buzzer_start(); + buzzer_set_freq(261); + msleep(250); + buzzer_set_freq(0); + 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; +}