| /*************************************************************************** |
| * __________ __ ___. |
| * Open \______ \ ____ ____ | | _\_ |__ _______ ___ |
| * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / |
| * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < |
| * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ |
| * \/ \/ \/ \/ \/ |
| * $Id$ |
| * |
| * Copyright (c) 2011 Michael Sevakis |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License |
| * as published by the Free Software Foundation; either version 2 |
| * of the License, or (at your option) any later version. |
| * |
| * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY |
| * KIND, either express or implied. |
| * |
| ****************************************************************************/ |
| #include "config.h" |
| #include "system.h" |
| #include "settings.h" |
| #include "pcm.h" |
| #include "pcm_mixer.h" |
| #include "misc.h" |
| #include "fixedpoint.h" |
| |
| /** Beep generation, CPU optimized **/ |
| #include "asm/beep.c" |
| |
| static uint32_t beep_phase; /* Phase of square wave generator */ |
| static uint32_t beep_step; /* Step of square wave generator on each sample */ |
| #ifdef BEEP_GENERIC |
| static int16_t beep_amplitude; /* Amplitude of square wave generator */ |
| #else |
| /* Optimized routines do XOR with phase sign bit in both channels at once */ |
| static uint32_t beep_amplitude; /* Amplitude of square wave generator */ |
| #endif |
| static int beep_count; /* Number of samples remaining to generate */ |
| |
| #define BEEP_COUNT(fs, duration) ((fs) / 1000 * (duration)) |
| |
| /* Reserve enough static space for keyclick to fit in worst case */ |
| #define BEEP_BUF_COUNT BEEP_COUNT(PLAY_SAMPR_MAX, KEYCLICK_DURATION) |
| static int16_t beep_buf[BEEP_BUF_COUNT*2] IBSS_ATTR __attribute__((aligned(4))); |
| |
| /* Callback to generate the beep frames - also don't want inlining of |
| call below in beep_play */ |
| static void __attribute__((noinline)) |
| beep_get_more(const void **start, size_t *size) |
| { |
| int count = beep_count; |
| |
| if (count > 0) |
| { |
| count = MIN(count, BEEP_BUF_COUNT); |
| beep_count -= count; |
| *start = beep_buf; |
| *size = count * 2 * sizeof (int16_t); |
| beep_generate((void *)beep_buf, count, &beep_phase, |
| beep_step, beep_amplitude); |
| } |
| } |
| |
| /* Generates a constant square wave sound with a given frequency in Hertz for |
| a duration in milliseconds */ |
| void beep_play(unsigned int frequency, unsigned int duration, |
| unsigned int amplitude) |
| { |
| mixer_channel_stop(PCM_MIXER_CHAN_BEEP); |
| |
| if (frequency == 0 || duration == 0 || amplitude == 0) |
| return; |
| |
| if (amplitude > INT16_MAX) |
| amplitude = INT16_MAX; |
| |
| /* Setup the parameters for the square wave generator */ |
| uint32_t fout = mixer_get_frequency(); |
| beep_phase = 0; |
| beep_step = fp_div(frequency, fout, 32); |
| beep_count = BEEP_COUNT(fout, duration); |
| |
| #ifdef BEEP_GENERIC |
| beep_amplitude = amplitude; |
| #else |
| /* Optimized routines do XOR with phase sign bit in both channels at once */ |
| beep_amplitude = amplitude | (amplitude << 16); /* Word:|AMP16|AMP16| */ |
| #endif |
| |
| /* If it fits - avoid cb overhead */ |
| const void *start; |
| size_t size; |
| |
| /* Generate first frame here */ |
| beep_get_more(&start, &size); |
| |
| mixer_channel_set_amplitude(PCM_MIXER_CHAN_BEEP, MIX_AMP_UNITY); |
| mixer_channel_play_data(PCM_MIXER_CHAN_BEEP, |
| beep_count ? beep_get_more : NULL, |
| start, size); |
| } |